Как запустить тесты Capybara с использованием Selenium & Chrome в среде Dockerised Rails на Mac - PullRequest
0 голосов
/ 24 апреля 2020

Мне кажется, что я близко, но врезался в стену и поэтому был бы признателен за несколько советов.

Моя цель - иметь возможность запускать системные тесты и (делать снимки экрана), используя Chrome.

Тесты, отличные от JS, проходят с использованием :rack_test.

При использовании :remote_browser (моя настройка Chrome) я могу посетить путь, сделать снимок экрана, на котором отображается Этот сайт недоступен 127.0.0.1 отказано в подключении.

Также стоит упомянуть, если я пытаюсь использовать vn c с open vnc://localhost:5900 Я получаю ошибку Не удалось подключиться к “ localhost » Невозможно связаться с« localhost ». Убедитесь, что удаленный компьютер доступен и брандмауэр не блокирует совместное использование экрана.

Я знаю, что что-то упустил, но застрял в том, что.

Это мой docker-compose.yml файл. Я использую автономное Chrome изображение.

version: '3'

services:

  app: &app
    build: .
    tmpfs:
      - /tmp

  backend: &backend
    <<: *app
    stdin_open: true
    tty: true
    volumes:
      - .:/usr/src/app:cached
      - rails_cache:/usr/src/app/tmp/cache
      - bundle:/usr/local/bundle
      - node_modules:/app/node_modules
      - packs:/app/public/packs
    env_file:
      - .env/development/database
      - .env/development/web
    environment:
      - WEBPACKER_DEV_SERVER_HOST=webpack-dev-server

  web:
    <<: *backend
    entrypoint: ./bin/docker-entrypoint.sh
    command: bundle exec rails server -b 0.0.0.0
    ports:
      - "3000:3000"

  chrome:
    image: selenium/standalone-chrome:3.141.59
    ports:
      - "4444:4444"
      - "5900:5900"

  webpack-dev-server:
    <<: *app
    command: ./bin/webpack-dev-server
    volumes:
      - .:/usr/src/app:cached
      - bundle:/usr/local/bundle
      - node_modules:/app/node_modules
      - packs:/app/public/packs
    ports:
      - "3035:3035"
    environment:
      - WEBPACKER_DEV_SERVER_HOST=0.0.0.0

  database:
    image: postgres
    env_file:
      - .env/development/database
    volumes:
      - db_data:/var/lib/postgresql/data

volumes:
  db_data:
  rails_cache:
  bundle:
  node_modules:
  packs:

Мой Gemfile включает capybara и selenium-webdriver

group :test do
  gem 'capybara', '~> 3.32', '>= 3.32.1'
  gem 'selenium-webdriver', '~> 3.142', '>= 3.142.7'
end

Тогда мой файл rails_helper.rb включает это

require 'capybara/rspec'
require 'selenium-webdriver'

Capybara.server_host = '0.0.0.0'
Capybara.server_port = 3000
Capybara.app_host = "http://localhost:3000"

Capybara.register_driver :remote_browser do |app|
  capabilities = Selenium::WebDriver::Remote::Capabilities.chrome(
    chromeOptions: { args: %w(disable-gpu no-sandbox) }
  )

  Capybara::Selenium::Driver.new(app,
    :browser => :remote,
    url: "http://chrome:4444/wd/hub",
    desired_capabilities: capabilities
  )
end

Capybara.javascript_driver = :remote_browser

Когда я запускаю тесты, я запускаю Docker с

docker-compose up chrome database

В своих журналах я вижу это о запуске службы Chrome.

chrome_1              | 2020-04-24 16:17:05,059 INFO spawned: 'xvfb' with pid 10
chrome_1              | 2020-04-24 16:17:05,060 INFO spawned: 'selenium-standalone' with pid 11
chrome_1              | 16:17:05.229 INFO [GridLauncherV3.parse] - Selenium server version: 3.141.59, revision: e82be7d358
chrome_1              | 2020-04-24 16:17:05,230 INFO success: xvfb entered RUNNING state, process has stayed up for > than 0 seconds (startsecs)
chrome_1              | 2020-04-24 16:17:05,230 INFO success: selenium-standalone entered RUNNING state, process has stayed up for > than 0 seconds (startsecs)
chrome_1              | 16:17:05.289 INFO [GridLauncherV3.lambda$buildLaunchers$3] - Launching a standalone Selenium Server on port 4444
chrome_1              | 2020-04-24 16:17:05.328:INFO::main: Logging initialized @260ms to org.seleniumhq.jetty9.util.log.StdErrLog
chrome_1              | 16:17:05.486 INFO [WebDriverServlet.<init>] - Initialising WebDriverServlet
chrome_1              | 16:17:05.548 INFO [SeleniumServer.boot] - Selenium Server is up and running on port 4444

Затем я запускаю системный тест с этим

docker-compose run --rm web bundle exec rspec spec/system/example_spec.rb

Мой пример теста очень прост

require 'rails_helper'

RSpec.describe "Example", type: :system do
  before do
    driven_by(:remote_browser)
  end

  it "should take a screenshot of root path" do
    visit root_path
    take_screenshot
  end
end

Он запускается и проходит

[Screenshot]: /usr/src/app/tmp/screenshots/r_spec_example_598.png
.

Finished in 0.77216 seconds (files took 1.73 seconds to load)
1 example, 0 failures

Вот журнал

chrome_1              | 16:29:55.662 INFO [ActiveSessionFactory.apply] - Capabilities are: {
chrome_1              |   "browserName": "chrome",
chrome_1              |   "chromeOptions": {
chrome_1              |     "args": [
chrome_1              |       "disable-gpu",
chrome_1              |       "no-sandbox"
chrome_1              |     ]
chrome_1              |   },
chrome_1              |   "cssSelectorsEnabled": true,
chrome_1              |   "javascriptEnabled": true,
chrome_1              |   "nativeEvents": false,
chrome_1              |   "rotatable": false,
chrome_1              |   "takesScreenshot": false,
chrome_1              |   "version": ""
chrome_1              | }
chrome_1              | 16:29:55.662 INFO [ActiveSessionFactory.lambda$apply$11] - Matched factory org.openqa.selenium.grid.session.remote.ServicedSession$Factory (provider: org.openqa.selenium.chrome.ChromeDriverService)
chrome_1              | Starting ChromeDriver 81.0.4044.69 (6813546031a4bc83f717a2ef7cd4ac6ec1199132-refs/branch-heads/4044@{#776}) on port 13909
chrome_1              | Only local connections are allowed.
chrome_1              | Please protect ports used by ChromeDriver and related test frameworks to prevent access by malicious code.
chrome_1              | [1587745795.669][SEVERE]: bind() failed: Cannot assign requested address (99)
chrome_1              | 16:29:55.943 INFO [ProtocolHandshake.createSession] - Detected dialect: W3C
chrome_1              | 16:29:55.943 INFO [RemoteSession$Factory.lambda$performHandshake$0] - Started new session d8f42b8387f0160b95f333eac5c1b576 (org.openqa.selenium.chrome.ChromeDriverService)
chrome_1              | 16:29:56.287 INFO [ActiveSessions$1.onStop] - Removing session d8f42b8387f0160b95f333eac5c1b576 (org.openqa.selenium.chrome.ChromeDriverService)

И скриншот выглядит следующим образом ...

enter image description here

Кроме того, это сообщение об ошибке при попытке open vnc://localhost:5900

enter image description here

Дошло до того, что я близок к тому, чтобы признать поражение и не тестировать эти варианты использования, поэтому буду очень признателен за некоторые советы. Спасибо!

1 Ответ

0 голосов
/ 25 апреля 2020

Спасибо Томасу Уолполу за то, что он указал мне правильное направление.

У меня это работает сейчас, и я решил опубликовать решение на тот случай, если кто-то еще изо всех сил попытается заставить установку работать.

Первая точка - vn c не работает

Исправить несложно - используемая мной версия Chrome не включала это, и было необходимо использовать отладочный селен версия.

Служба selenium_chrome в docker-compose.yml становится такой

  selenium_chrome:
    image: selenium/standalone-chrome-debug:3.141.59
    ports:
      - "5900:5900"

Основная проблема сети

Это относится к Capybara.app_host как объяснил Томас Уолпол.

Решение состояло из нескольких шагов. Во-первых, тестовый сервер теперь вещает на другой порт, 4000.

Служба web в docker-compose.yml добавляет порт.

  web:
    <<: *backend
    entrypoint: ./bin/docker-entrypoint.sh
    command: bundle exec rails server -b 0.0.0.0
    ports:
      - "3000:3000"
      - "4000:4000"

rails_helper.rb использует этот порт web - это внутренний сетевой адрес Docker для службы web.

    Capybara.server_host = '0.0.0.0'
    Capybara.server_port = 4000
    Capybara.app_host = "http://web:4000"

Последней проблемой с моим кодом было местоположение этого кода. Он также должен находиться внутри блока Rspec.config.

Complete rails_helper.rb code

Таким образом, был полный рабочий окончательный соответствующий код в rails_helper.rb.

require 'capybara/rspec'
require 'selenium-webdriver'

Capybara.register_driver :headless_selenium_chrome_in_container do |app|
  capabilities = Selenium::WebDriver::Remote::Capabilities.chrome(
    chromeOptions: { args: %w(headless disable-gpu window-size=1440x768) }
  )

  Capybara::Selenium::Driver.new(app,
    :browser => :remote,
    url: "http://selenium_chrome:4444/wd/hub",
    desired_capabilities: capabilities
  )
end

Capybara.register_driver :selenium_chrome_in_container do |app|
  Capybara::Selenium::Driver.new(app,
    :browser => :remote,
    url: "http://selenium_chrome:4444/wd/hub",
    desired_capabilities: :chrome
  )
end

RSpec.configure do |config|
  ...

  config.before(:each, type: :system) do
    Capybara.server_host = '0.0.0.0'
    Capybara.server_port = 4000
    Capybara.app_host = "http://web:4000"
    driven_by(:headless_selenium_chrome_in_container)
  end
end

Примечание: это дает возможность запускать Chrome без головы или с помощью браузера, доступного через vnc://localhost:5900.

...