2017-02-22 13:18:06 +00:00
|
|
|
# About chromedp [![Build Status](https://travis-ci.org/knq/chromedp.svg)](https://travis-ci.org/knq/chromedp) [![Coverage Status](https://coveralls.io/repos/knq/chromedp/badge.svg?branch=master&service=github)](https://coveralls.io/github/knq/chromedp?branch=master) #
|
2017-01-24 15:09:23 +00:00
|
|
|
|
|
|
|
Package chromedp is a faster, simpler way to drive browsers in Go using the
|
|
|
|
[Chrome Debugging Protocol](https://developer.chrome.com/devtools/docs/debugger-protocol)
|
|
|
|
(for Chrome, Edge, Safari, etc) without external dependencies (ie, Selenium, PhantomJS, etc).
|
|
|
|
|
|
|
|
**NOTE:** chromedp's API is currently unstable, and may change at a moments
|
|
|
|
notice. There are likely extremely bad bugs lurking in this code. **CAVEAT USER**.
|
|
|
|
|
|
|
|
## Installation
|
|
|
|
|
|
|
|
Install in the usual way:
|
|
|
|
|
|
|
|
```sh
|
|
|
|
go get -u github.com/knq/chromedp
|
|
|
|
```
|
|
|
|
|
|
|
|
## Usage
|
|
|
|
|
2017-01-24 15:38:56 +00:00
|
|
|
Below is a simple Google search performed using chromedp (taken from
|
|
|
|
[examples/simple](examples/simple/main.go)):
|
|
|
|
|
|
|
|
This example shows logic for a simple search for a known website, clicking on
|
|
|
|
the right link, and then taking a screenshot of a specific element on the
|
|
|
|
loaded page and saving that to a local file on disk.
|
|
|
|
|
|
|
|
```go
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
cdp "github.com/knq/chromedp"
|
|
|
|
cdptypes "github.com/knq/chromedp/cdp"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
// create context
|
|
|
|
ctxt, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
// create chrome instance
|
2017-06-18 01:12:33 +00:00
|
|
|
c, err := cdp.New(ctxt, cdp.WithLog(log.Printf))
|
2017-01-24 15:38:56 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// run task list
|
|
|
|
var site, res string
|
|
|
|
err = c.Run(ctxt, googleSearch("site:brank.as", "Easy Money Management", &site, &res))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// shutdown chrome
|
|
|
|
err = c.Shutdown(ctxt)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait for chrome to finish
|
|
|
|
err = c.Wait()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("saved screenshot of #testimonials from search result listing `%s` (%s)", res, site)
|
|
|
|
}
|
|
|
|
|
|
|
|
func googleSearch(q, text string, site, res *string) cdp.Tasks {
|
|
|
|
var buf []byte
|
|
|
|
sel := fmt.Sprintf(`//a[text()[contains(., '%s')]]`, text)
|
|
|
|
return cdp.Tasks{
|
|
|
|
cdp.Navigate(`https://www.google.com`),
|
|
|
|
cdp.Sleep(2 * time.Second),
|
|
|
|
cdp.WaitVisible(`#hplogo`, cdp.ByID),
|
2017-02-08 01:38:55 +00:00
|
|
|
cdp.SendKeys(`#lst-ib`, q+"\n", cdp.ByID),
|
|
|
|
cdp.WaitVisible(`#res`, cdp.ByID),
|
2017-01-24 15:38:56 +00:00
|
|
|
cdp.Text(sel, res),
|
|
|
|
cdp.Click(sel),
|
|
|
|
cdp.Sleep(2 * time.Second),
|
|
|
|
cdp.WaitVisible(`#footer`, cdp.ByQuery),
|
|
|
|
cdp.WaitNotVisible(`div.v-middle > div.la-ball-clip-rotate`, cdp.ByQuery),
|
|
|
|
cdp.Location(site),
|
|
|
|
cdp.Screenshot(`#testimonials`, &buf, cdp.ByID),
|
2017-06-18 01:12:33 +00:00
|
|
|
cdp.ActionFunc(func(context.Context, cdptypes.Handler) error {
|
2017-01-24 15:38:56 +00:00
|
|
|
return ioutil.WriteFile("testimonials.png", buf, 0644)
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2017-03-13 02:24:25 +00:00
|
|
|
Please see the [examples](examples/) directory for some more examples, and
|
|
|
|
please refer to the [GoDoc API listing](https://godoc.org/github.com/knq/chromedp)
|
|
|
|
for a summary of the API and Actions.
|
2017-01-24 15:09:23 +00:00
|
|
|
|
|
|
|
## TODO
|
|
|
|
* Move timeouts to context (defaults)
|
|
|
|
* Implement more query selector options (allow over riding context timeouts)
|
|
|
|
* Contextual actions for "dry run" (or via an accumulator?)
|
|
|
|
* Network loader / manager
|
|
|
|
* Profiler
|