TL;DR
Chrome bez interfejsu graficznego jest dostępna w wersji Chrome 59. Umożliwia on uruchamianie przeglądarki Chrome w środowisku bez interfejsu graficznego. Zasadniczo bieganie Chrome bez Chrome! Zawiera wszystkie nowoczesne funkcje platform internetowych za pomocą Chromium i mechanizmu renderowania Blink do wiersza poleceń.
Dlaczego jest to przydatne?
Przeglądarka bez interfejsu graficznego to świetne narzędzie do automatycznego testowania oraz tworzenia środowisk serwerowych, w których nie potrzebują widocznej powłoki UI. Warto na przykład przeprowadzić testy pod kątem utworzyć prawdziwą stronę internetową, utworzyć jej plik PDF lub po prostu sprawdzić, jak przeglądarka renderuje URL.
Uruchamiam interfejs bez interfejsu graficznego (CLI)
Aby zacząć korzystać z trybu bez interfejsu graficznego, najlepiej otwórz plik binarny Chrome.
w wierszu poleceń. Jeśli masz zainstalowaną przeglądarkę Chrome 59 lub nowszą, uruchom ją z flagą --headless
:
chrome \
--headless \ # Runs Chrome in headless mode.
--disable-gpu \ # Temporarily needed if running on Windows.
--remote-debugging-port=9222 \
https://www.chromestatus.com # URL to open. Defaults to about:blank.
Adres chrome
powinien wskazywać Twoją instalację Chrome. Dokładna lokalizacja
różnią się w zależności od platformy. Ponieważ używam komputera Mac, utworzyłem wygodne aliasy
dla każdej zainstalowanej przeze mnie wersji Chrome.
Jeśli używasz stabilnej wersji Chrome, a nie możesz pobrać wersji beta, zalecamy
przy użyciu chrome-canary
:
alias chrome="/Applications/Google\ Chrome.app/Contents/MacOS/Google\ Chrome"
alias chrome-canary="/Applications/Google\ Chrome\ Canary.app/Contents/MacOS/Google\ Chrome\ Canary"
alias chromium="/Applications/Chromium.app/Contents/MacOS/Chromium"
Chrome Canary możesz pobrać stąd.
Funkcje wiersza poleceń
W niektórych przypadkach nie trzeba programować tworzenia skryptów w Chrome bez interfejsu graficznego. W wierszu poleceń można używać przydatnych flag do wykonywania typowych zadań.
Drukowanie DOM
Flaga --dump-dom
pokazuje document.body.innerHTML
do standardowego protokołu:
chrome --headless --disable-gpu --dump-dom https://www.chromestatus.com/
Utwórz PDF
Flaga --print-to-pdf
tworzy plik PDF strony:
chrome --headless --disable-gpu --print-to-pdf https://www.chromestatus.com/
Robienie zrzutów ekranu
Aby zrobić zrzut ekranu strony, użyj flagi --screenshot
:
chrome --headless --disable-gpu --screenshot https://www.chromestatus.com/
# Size of a standard letterhead.
chrome --headless --disable-gpu --screenshot --window-size=1280,1696 https://www.chromestatus.com/
# Nexus 5x
chrome --headless --disable-gpu --screenshot --window-size=412,732 https://www.chromestatus.com/
Uruchomienie polecenia --screenshot
spowoduje utworzenie wscreenshot.png
w bieżącym katalogu roboczym. Jeśli szukasz zrzutów ekranu na całą stronę,
są nieco bardziej zaangażowani. Mamy świetny blog
post Davida Schnurra, na temat którego wszystko pomoże. Wymelduj się
Używanie Chrome bez interfejsu graficznego jako zautomatyzowanego narzędzia do tworzenia zrzutów ekranu
Tryb REPL (pętla odczytu-eval-print)
Flaga --repl
uruchamia się w trybie bez interfejsu graficznego w trybie, w którym można oceniać wyrażenia JS.
w przeglądarce, bezpośrednio z wiersza poleceń:
$ chrome --headless --disable-gpu --repl --crash-dumps-dir=./tmp https://www.chromestatus.com/
[0608/112805.245285:INFO:headless_shell.cc(278)] Type a Javascript expression to evaluate or "quit" to exit.
>>> location.href
{"result":{"type":"string","value":"https://www.chromestatus.com/features"}}
>>> quit
$
debugujesz Chrome bez użycia interfejsu przeglądarki?
Gdy uruchomisz Chrome z --remote-debugging-port=9222
, uruchomi się instancja
z włączonym protokołem Narzędzi deweloperskich.
jest używany do komunikowania się z Chrome i obsługi
z instancji przeglądarki. Zapewnia też narzędzia takie jak Sublime, VS Code i Node
na zdalne debugowanie aplikacji. #synergy
Nie masz interfejsu przeglądarki, aby wyświetlić tę stronę, więc otwórz http://localhost:9222
w innej przeglądarce, aby sprawdzić, czy wszystko działa. Zobaczysz listę
które można wyświetlić, aby sprawdzić, co renderuje bez interfejsu graficznego:
Tutaj możesz używać znanych funkcji Narzędzi deweloperskich do sprawdzania, debugowania i dostosowywania na stronie w zwykły sposób. Jeśli używasz interfejsu bez interfejsu graficznego, jest także zaawansowanym narzędziem do debugowania, które umożliwia śledzenie wszystkich przechodzących w ten sposób, komunikując się z przeglądarką.
Korzystanie z automatyzacji (węzeł)
Animator
Puppeteer to biblioteka węzłów opracowanych przez zespół Chrome. Zapewnia interfejs API wysokiego poziomu do sterowania bez interfejsu graficznego (lub w pełni) Chrome. Jest podobny do innych bibliotek testowania automatycznego, takich jak Phantom oraz NightmareJS, ale działa on tylko w najnowszych wersjach Chrome.
Aplikacja Puppeteer umożliwia m.in. łatwe robienie zrzutów ekranu, tworzenie plików PDF jak poruszać się po stronach i pobierać o nich informacje. Polecam bibliotekę jeśli chcesz szybko zautomatyzować testowanie przeglądarki. Ukrywa zawiłości, i wykonywał zbędne zadania, takie jak uruchamianie do debugowania instancji Chrome.
Zainstaluj:
npm i --save puppeteer
Przykład – wydrukowanie informacji o kliencie użytkownika.
const puppeteer = require('puppeteer');
(async() => {
const browser = await puppeteer.launch();
console.log(await browser.version());
await browser.close();
})();
Przykład – zrzut ekranu strony
const puppeteer = require('puppeteer');
(async() => {
const browser = await puppeteer.launch();
const page = await browser.newPage();
await page.goto('https://www.chromestatus.com', {waitUntil: 'networkidle2'});
await page.pdf({path: 'page.pdf', format: 'A4'});
await browser.close();
})();
Zapoznaj się z dokumentacją Puppeteer , aby dowiedzieć się więcej o pełnym interfejsie API.
Biblioteka CRI
chrome-remote-interface to biblioteka niższego poziomu niż API Puppeteer. Polecam, jeśli macie w pobliżu metalu i bezpośrednio przy użyciu protokołu DevTools.
Uruchamiam Chrome
chrome-remote-interface nie uruchamia za Ciebie Chrome, więc musisz i zadbać o to.
W sekcji interfejsu wiersza poleceń uruchomiliśmy Chrome ręcznie, używając
--headless --remote-debugging-port=9222
Aby jednak w pełni zautomatyzować testy, trzeba będzie
chcesz wywołać Chrome z Twojej aplikacji.
Jednym ze sposobów jest użycie polecenia child_process
:
const execFile = require('child_process').execFile;
function launchHeadlessChrome(url, callback) {
// Assuming MacOSx.
const CHROME = '/Applications/Google\ Chrome.app/Contents/MacOS/Google\ Chrome';
execFile(CHROME, ['--headless', '--disable-gpu', '--remote-debugging-port=9222', url], callback);
}
launchHeadlessChrome('https://www.chromestatus.com', (err, stdout, stderr) => {
...
});
Jednak jeśli potrzebujesz przenośnego rozwiązania, które działa w wielu różnych miejscach, platform. Spójrzmy tylko na zakodowaną na stałe ścieżkę do Chrome :(
Korzystanie z ChromeLauncher
Lighthouse jest cudowny,
narzędzie do testowania jakości aplikacji internetowych. Solidny moduł do uruchamiania
Przeglądarka Chrome została opracowana w ramach narzędzia Lighthouse, a teraz jest wyodrębniona do użytku samodzielnego.
Moduł NPM chrome-launcher
znajdzie gdzie
Przeglądarka Chrome jest zainstalowana. Skonfiguruj instancję debugowania, uruchom przeglądarkę i zamknij ją.
po zakończeniu programu. A najlepsze jest to, że działa na wielu platformach dzięki
Węzeł
Domyślnie chrome-launcher
będzie próbowała uruchomić Chrome Canary (jeśli jest
zainstalowane), ale możesz to zmienić i ręcznie wybrać przeglądarkę Chrome, której chcesz używać. Do
użyj go, najpierw zainstaluj z npm:
npm i --save chrome-launcher
Przykład – użycie interfejsu chrome-launcher
do uruchomienia interfejsu bez interfejsu graficznego
const chromeLauncher = require('chrome-launcher');
// Optional: set logging level of launcher to see its output.
// Install it using: npm i --save lighthouse-logger
// const log = require('lighthouse-logger');
// log.setLevel('info');
/**
* Launches a debugging instance of Chrome.
* @param {boolean=} headless True (default) launches Chrome in headless mode.
* False launches a full version of Chrome.
* @return {Promise<ChromeLauncher>}
*/
function launchChrome(headless=true) {
return chromeLauncher.launch({
// port: 9222, // Uncomment to force a specific port of your choice.
chromeFlags: [
'--window-size=412,732',
'--disable-gpu',
headless ? '--headless' : ''
]
});
}
launchChrome().then(chrome => {
console.log(`Chrome debuggable on port: ${chrome.port}`);
...
// chrome.kill();
});
Uruchamianie tego skryptu niewiele daje, ale powinieneś zobaczyć wystąpienie
Chrome uruchomił się w menedżerze zadań, który wczytał about:blank
. Pamiętaj, że
nie będzie działać jako interfejs przeglądarki. Nie mamy głowy.
Do kontrolowania przeglądarki potrzebny jest protokół Narzędzi deweloperskich.
Pobieranie informacji o stronie
Zainstalujmy bibliotekę:
npm i --save chrome-remote-interface
Przykłady
Przykład – wydrukowanie informacji o kliencie użytkownika.
const CDP = require('chrome-remote-interface');
...
launchChrome().then(async chrome => {
const version = await CDP.Version({port: chrome.port});
console.log(version['User-Agent']);
});
Wyniki w rodzaju: HeadlessChrome/60.0.3082.0
Przykład – sprawdź, czy witryna ma manifest aplikacji internetowej
const CDP = require('chrome-remote-interface');
...
(async function() {
const chrome = await launchChrome();
const protocol = await CDP({port: chrome.port});
// Extract the DevTools protocol domains we need and enable them.
// See API docs: https://chromedevtools.github.io/devtools-protocol/
const {Page} = protocol;
await Page.enable();
Page.navigate({url: 'https://www.chromestatus.com/'});
// Wait for window.onload before doing stuff.
Page.loadEventFired(async () => {
const manifest = await Page.getAppManifest();
if (manifest.url) {
console.log('Manifest: ' + manifest.url);
console.log(manifest.data);
} else {
console.log('Site has no app manifest');
}
protocol.close();
chrome.kill(); // Kill Chrome.
});
})();
Przykład: wyodrębnij <title>
strony za pomocą interfejsów DOM API.
const CDP = require('chrome-remote-interface');
...
(async function() {
const chrome = await launchChrome();
const protocol = await CDP({port: chrome.port});
// Extract the DevTools protocol domains we need and enable them.
// See API docs: https://chromedevtools.github.io/devtools-protocol/
const {Page, Runtime} = protocol;
await Promise.all([Page.enable(), Runtime.enable()]);
Page.navigate({url: 'https://www.chromestatus.com/'});
// Wait for window.onload before doing stuff.
Page.loadEventFired(async () => {
const js = "document.querySelector('title').textContent";
// Evaluate the JS expression in the page.
const result = await Runtime.evaluate({expression: js});
console.log('Title of page: ' + result.result.value);
protocol.close();
chrome.kill(); // Kill Chrome.
});
})();
Używanie Selenium, WebDriver i ChromeDriver
Obecnie Selenium otwiera pełną instancję Chrome. Innymi słowy, jest to zautomatyzowanego rozwiązania, ale nie całkowicie bez interfejsu graficznego. Selen może być jednak skonfigurowany do uruchamiania Chrome bez interfejsu graficznego z niewielkim nakładem pracy. Polecam Uruchamianie Selenium z Chrome bez interfejsu graficznego jeśli chcesz, konfiguracji znajdziesz w kilku szczegółowych instrukcjach, poniższe przykłady.
Przy użyciu ChromeDriver
ChromeDriver 2.32 korzysta z Chrome 61 i działa dobrze w przeglądarkach bez interfejsu graficznego.
Zainstaluj:
npm i --save-dev selenium-webdriver chromedriver
Przykład:
const fs = require('fs');
const webdriver = require('selenium-webdriver');
const chromedriver = require('chromedriver');
const chromeCapabilities = webdriver.Capabilities.chrome();
chromeCapabilities.set('chromeOptions', {args: ['--headless']});
const driver = new webdriver.Builder()
.forBrowser('chrome')
.withCapabilities(chromeCapabilities)
.build();
// Navigate to google.com, enter a search.
driver.get('https://www.google.com/');
driver.findElement({name: 'q'}).sendKeys('webdriver');
driver.findElement({name: 'btnG'}).click();
driver.wait(webdriver.until.titleIs('webdriver - Google Search'), 1000);
// Take screenshot of results page. Save to disk.
driver.takeScreenshot().then(base64png => {
fs.writeFileSync('screenshot.png', new Buffer(base64png, 'base64'));
});
driver.quit();
Korzystanie z WebDriverIO
WebDriverIO to interfejs API wyższego poziomu będący alternatywą dla Selenium WebDriver.
Zainstaluj:
npm i --save-dev webdriverio chromedriver
Przykład: filtrowanie funkcji CSS na stronie chromestatus.com
const webdriverio = require('webdriverio');
const chromedriver = require('chromedriver');
const PORT = 9515;
chromedriver.start([
'--url-base=wd/hub',
`--port=${PORT}`,
'--verbose'
]);
(async () => {
const opts = {
port: PORT,
desiredCapabilities: {
browserName: 'chrome',
chromeOptions: {args: ['--headless']}
}
};
const browser = webdriverio.remote(opts).init();
await browser.url('https://www.chromestatus.com/features');
const title = await browser.getTitle();
console.log(`Title: ${title}`);
await browser.waitForText('.num-features', 3000);
let numFeatures = await browser.getText('.num-features');
console.log(`Chrome has ${numFeatures} total features`);
await browser.setValue('input[type="search"]', 'CSS');
console.log('Filtering features...');
await browser.pause(1000);
numFeatures = await browser.getText('.num-features');
console.log(`Chrome has ${numFeatures} CSS features`);
const buffer = await browser.saveScreenshot('screenshot.png');
console.log('Saved screenshot...');
chromedriver.stop();
browser.end();
})();
Dodatkowe zasoby
Oto kilka przydatnych zasobów na początek:
Dokumenty
- Wyświetlający protokoły DevTools – dokumentacja API
Narzędzia
- chrome-remote-interface – węzeł który obejmuje protokół Narzędzi deweloperskich
- Lighthouse – zautomatyzowane narzędzie do testowania, jakość aplikacji internetowej, często używa protokołu
- chrome-launcher - moduł węzłów umożliwiający uruchomienie Chrome, gotowy do automatyzacji
Prezentacje
- „Sieć bez interfejsu graficznego” – świetny blog Paula Kinlana post na temat korzystania z interfejsu api.ai bez interfejsu graficznego.
Najczęstsze pytania
Czy potrzebuję flagi --disable-gpu
?
Tylko w systemie Windows. Na innych platformach nie jest on już wymagany. Flaga --disable-gpu
to
tymczasowo obejść
kilka błędów. Nie będziesz potrzebować tej flagi w kolejnych wersjach
Chrome. Więcej informacji: crbug.com/737678
.
Czy w dalszym ciągu potrzebuję Xvfb?
Nie. Chrome bez interfejsu graficznego nie używa okna, więc serwer wyświetlania taki jak Xvfb już nie są potrzebne. Możesz bez niego przeprowadzać automatyczne testy.
Co to jest Xvfb? Xvfb to wewnętrzny serwer wyświetlania dla systemów typu Unix, który pozwala do uruchamiania aplikacji graficznych (takich jak Chrome) bez fizycznego wyświetlacza. Wiele osób używa Xvfb do uruchamiania starszych wersji Chrome w celu korzystania z technologii bez interfejsu graficznego i testowania.
Jak utworzyć kontener Dockera, który będzie obsługiwać Chrome bez interfejsu graficznego?
Spójrz na to miejsce: Lighthouse-ci. Zawiera
przykładowy plik Dockerfile
używający node:8-slim
jako obrazu podstawowego, instaluje +
obsługuje Lighthouse
w App Engine Flex.
Czy mogę używać tego rozwiązania z usługą Selenium / WebDriver / ChromeDriver?
Tak. Zobacz Using Selenium, WebDriver i ChromeDriver.
Jaki jest związek z PhantomJS?
Chrome bez interfejsu graficznego jest podobny do narzędzi takich jak PhantomJS. Obie opcje można używać do testów automatycznych w środowisku bez interfejsu graficznego. Główna różnica że Phantom używa starszej wersji WebKit do renderowania Chrome bez interfejsu graficznego korzysta z najnowszej wersji Blink.
Obecnie Phantom zapewnia też interfejs API wyższego poziomu niż protokół narzędzi deweloperskich.
Gdzie zgłaszać błędy?
Błędy dotyczące Chrome bez interfejsu graficznego możesz zgłaszać na stronie crbug.com.
Błędy w protokole narzędzi deweloperskich możesz zgłaszać na stronie github.com/ChromeDevTools/devtools-protocol.