2018-11-08 12:02:28 +02:00
|
|
|
browserless
|
|
|
|
===========
|
|
|
|
|
2018-11-13 11:38:17 +02:00
|
|
|
[Browserless][1] makes it easy to run your puppeteer scripts in an optimized
|
|
|
|
way. It takes care of all the binaries and managing of Chrome so you don't have
|
|
|
|
to.
|
|
|
|
|
2018-11-08 12:02:28 +02:00
|
|
|
## screenshot.js
|
|
|
|
|
|
|
|
```javascript
|
|
|
|
'use strict';
|
|
|
|
|
2023-02-22 12:38:45 +02:00
|
|
|
const puppeteer = require('puppeteer-core');
|
2018-11-08 12:02:28 +02:00
|
|
|
|
|
|
|
(async() => {
|
|
|
|
const browser = await puppeteer.connect({browserWSEndpoint: 'ws://localhost:3000'});
|
|
|
|
const page = await browser.newPage();
|
2023-02-22 12:38:45 +02:00
|
|
|
await page.setViewport({width: 1280, height: 720});
|
2018-11-08 12:02:28 +02:00
|
|
|
await page.goto('https://www.google.com/', {waitUntil: 'networkidle2'});
|
2023-02-22 12:38:45 +02:00
|
|
|
await page.waitForSelector('img[alt="Google"]');
|
2018-11-08 12:02:28 +02:00
|
|
|
await page.screenshot({path: 'google.png', fullPage: true});
|
|
|
|
await browser.close();
|
|
|
|
})();
|
|
|
|
```
|
|
|
|
|
2023-02-22 12:38:45 +02:00
|
|
|
## screenshot.py
|
|
|
|
|
|
|
|
```python
|
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
from selenium import webdriver
|
|
|
|
from selenium.webdriver.common.by import By
|
|
|
|
from selenium.webdriver.support.ui import WebDriverWait
|
|
|
|
from selenium.webdriver.support import expected_conditions as EC
|
|
|
|
|
|
|
|
options = webdriver.ChromeOptions()
|
|
|
|
options.add_argument("--window-size=1024,768")
|
|
|
|
options.add_argument('--headless')
|
|
|
|
options.add_argument('--no-sandbox')
|
|
|
|
|
|
|
|
driver = webdriver.Remote(
|
|
|
|
command_executor='http://localhost:3000/webdriver',
|
|
|
|
options=options
|
|
|
|
)
|
|
|
|
|
|
|
|
driver.get('https://www.google.com/')
|
|
|
|
|
|
|
|
WebDriverWait(driver, 10).until(
|
|
|
|
EC.presence_of_element_located((By.CSS_SELECTOR, 'img[alt="Google"]'))
|
|
|
|
)
|
|
|
|
|
|
|
|
driver.save_screenshot('google.png')
|
|
|
|
driver.close()
|
|
|
|
driver.quit()
|
|
|
|
```
|
|
|
|
|
2018-11-08 12:02:28 +02:00
|
|
|
## Up and Running
|
|
|
|
|
|
|
|
```bash
|
|
|
|
$ docker-compose up -d
|
2018-11-13 11:38:17 +02:00
|
|
|
|
2024-04-15 09:00:40 +02:00
|
|
|
$ crontab -l
|
|
|
|
0 */4 * * * docker restart browserless-browserless-1
|
|
|
|
5 * * * * docker exec browserless-browserless-1 find /tmp -name '.com.google.Chrome.*' -mmin +60 -exec rm -rf {} \;
|
|
|
|
|
2023-02-22 12:38:45 +02:00
|
|
|
$ npm install puppeteer-core
|
2018-11-08 12:02:28 +02:00
|
|
|
$ node screenshot.js
|
|
|
|
$ imgcat google.png
|
2018-11-13 11:38:17 +02:00
|
|
|
|
2023-02-22 12:38:45 +02:00
|
|
|
$ pip install selenium
|
|
|
|
$ python screenshot.py
|
|
|
|
$ imgcat google.png
|
|
|
|
|
2024-02-23 18:23:20 +02:00
|
|
|
$ http :3000/screenshot \
|
2018-11-13 11:38:17 +02:00
|
|
|
url=https://www.youtube.com \
|
2023-02-22 12:38:45 +02:00
|
|
|
options[fullPage]:=true \
|
|
|
|
gotoOptions[waitUntil]=networkidle2 > youtube.png
|
2018-11-13 11:38:17 +02:00
|
|
|
$ imgcat youtube.png
|
2024-02-23 18:23:20 +02:00
|
|
|
|
|
|
|
$ http :3000/scrape url=https://www.youtube.com elements[0][selector]=title debug[network]:=true |
|
|
|
|
jq -r '.debug.network.outbound[].url' |
|
|
|
|
xargs -r tldextract -j |
|
|
|
|
jq -r 'select(.fqdn|length>0).fqdn' |
|
|
|
|
sort -u
|
|
|
|
accounts.google.com
|
|
|
|
fonts.googleapis.com
|
|
|
|
fonts.gstatic.com
|
|
|
|
googleads.g.doubleclick.net
|
|
|
|
i.ytimg.com
|
|
|
|
www.youtube.com
|
2018-11-08 12:02:28 +02:00
|
|
|
```
|
2018-11-13 11:38:17 +02:00
|
|
|
|
2024-06-06 09:24:58 +02:00
|
|
|
## Waiting for Condition (v2)
|
|
|
|
|
|
|
|
[async][2] functions are supported
|
|
|
|
|
|
|
|
```bash
|
|
|
|
$ cat fn.js
|
|
|
|
async () => {
|
|
|
|
await new Promise(resolve => {
|
|
|
|
const interval = setInterval(() => {
|
|
|
|
var e = document.querySelector('#tryit-data');
|
|
|
|
if (e && e.innerText.includes('country')) {
|
|
|
|
resolve();
|
|
|
|
clearInterval(interval);
|
|
|
|
}
|
|
|
|
}, 1000)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
$ http '127.0.0.1:3000/content?token=1234567890&stealth&--proxy-server=http://x.x.x.x:8080' url='https://ipinfo.io' waitForFunction[fn]=@fn.js waitForFunction[timeout]:=10000 > ipinfo.html
|
|
|
|
```
|
|
|
|
|
2024-06-06 10:03:48 +02:00
|
|
|
## Intercepting HTTP traffic (v2)
|
|
|
|
|
|
|
|
```bash
|
|
|
|
$ cat function.js
|
|
|
|
export default async function ({ page }) {
|
|
|
|
const url = 'https://ipinfo.io/widget/demo/';
|
|
|
|
const promise = page.waitForResponse(res => res.url().includes(url));
|
|
|
|
await page.goto("https://ipinfo.io/");
|
|
|
|
const res = await promise;
|
|
|
|
const txt = await res.text();
|
|
|
|
return JSON.parse(txt);
|
|
|
|
}
|
|
|
|
|
|
|
|
$ http '127.0.0.1:3000/function?token=1234567890' Content-Type:application/javascript < function.js > ipinfo.json
|
|
|
|
```
|
|
|
|
|
2018-11-13 11:38:17 +02:00
|
|
|
[1]: https://docs.browserless.io/
|
2024-06-06 09:24:58 +02:00
|
|
|
[2]: https://docs.browserless.io/HTTP-APIs/content#waitforfunction
|