Как настроить Capybara для запуска тестов в докеризированной селеновой сетке?

У меня есть набор тестов, которые я хочу выполнить в докеризированной селеновой сетке. Тесты написаны в Ruby с использованием RSpec и Capybara. Также стоит отметить: я использую dinghy в качестве обертки для докер-машины.

Несколько недель назад я построил доказательство концепции, но с помощью Nightwatch вместо RSpec + Capybara. Это прекрасно работает, но получение Capybara для работы с докеризированной селеновой сеткой оказалось трудным.

Я пробовал множество конфигураций без успеха. Я думаю, что наиболее близким к успешной конфигурации является следующее...

# docker-compose.yml

web:
  image: web:latest  # built on my machine
  environment:
    VIRTUAL_HOST: app.under.test

rspec:
  build: .
  dockerfile: Dockerfile
  environment:
    APP_HOST: app.under.test
  volumes:
    - .:/usr/src/app
  links:
    - web
    - hub

hub:
  image: selenium/hub:latest
  environment:
    VIRTUAL_HOST: selenium.hub.docker
  ports:
    - 4444:4444

node-firefox:
  image: selenium/node-firefox:latest
  environment:
    VIRTUAL_HOST: firefox.docker
  links:
    - hub

node-chrome:
  image: selenium/node-chrome:latest
  environment:
    VIRTUAL_HOST: chrome.docker
  links:
    - hub
# Dockerfile for the rspec image

FROM instructure/ruby-passenger:2.3

USER root

ENV APP_HOME /usr/src/app
RUN mkdir -p $APP_HOME
WORKDIR $APP_HOME

COPY Gemfile Gemfile.lock $APP_HOME/
RUN chown -R docker:docker $APP_HOME

USER docker
RUN bundle install --quiet --jobs 8
USER root

COPY . $APP_HOME
RUN chown -R docker:docker $APP_HOME

USER docker
# spec/example_test.rb:

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

RSpec.configure do |config|
  config.include Capybara::DSL
end

def setup
  url = 'http://selenium.hub.docker'
  capabilities = Selenium::WebDriver::Remote::Capabilities.firefox
  Capybara.app_host = ENV['APP_HOST']

  Capybara.register_driver :remote_browser do |app|
    Capybara::Selenium::Driver.new(
      app,
      :browser => :remote,
      url: url,
      desired_capabilities: capabilities
    )
  end

  Capybara.default_driver = :remote_browser
  Capybara.javascript_driver = :remote_browser
end

describe 'This is an example' do
  it 'and it works' do
    setup
    visit '/'
    expect(page.title).to eq 'Home'
  end
end

Но эта конфигурация ^ дает следующую ошибку при запуске тестов:

1) This is an example and it works
     Failure/Error: visit '/'

 Selenium::WebDriver::Error::WebDriverError:
   unexpected response, code=200, content-type="text/html"
   <html><head><title>Selenium Grid2.0 help</title></head><body>You are using grid 2.52.0<br>Find help on the official selenium wiki : <a href='https://github.com/SeleniumHQ/selenium/wiki/Grid2' >more help here</a><br>default monitoring page : <a href='/grid/console' >console</a></body></html>
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/remote/http/common.rb:85:in `create_response'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/remote/http/default.rb:90:in `request'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/remote/http/common.rb:59:in `call'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/remote/bridge.rb:645:in `raw_execute'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/remote/bridge.rb:123:in `create_session'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/remote/bridge.rb:87:in `initialize'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/common/driver.rb:59:in `new'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver/common/driver.rb:59:in `for'
 # /home/docker/.gem/ruby/2.3.0/gems/selenium-webdriver-2.52.0/lib/selenium/webdriver.rb:86:in `for'
 # /home/docker/.gem/ruby/2.3.0/gems/capybara-2.6.2/lib/capybara/selenium/driver.rb:13:in `browser'
 # /home/docker/.gem/ruby/2.3.0/gems/capybara-2.6.2/lib/capybara/selenium/driver.rb:45:in `visit'
 # /home/docker/.gem/ruby/2.3.0/gems/capybara-2.6.2/lib/capybara/session.rb:232:in `visit'
 # /home/docker/.gem/ruby/2.3.0/gems/capybara-2.6.2/lib/capybara/dsl.rb:51:in `block (2 levels) in <module:DSL>'
 # ./spec/example_test.rb:31:in `block (2 levels) in <top (required)>'

Любые идеи? Что мне не хватает?

Update

Получил это!

def setup
  url = 'http://selenium.hub.docker/wd/hub'
  capabilities = Selenium::WebDriver::Remote::Capabilities.firefox
  Capybara.app_host = "http://#{ENV['APP_HOST']}"
  Capybara.run_server = false

  Capybara.register_driver :remote_browser do |app|
    Capybara::Selenium::Driver.new(
      app,
      :browser => :remote,
      url: url,
      desired_capabilities: capabilities
    )
  end

  Capybara.default_driver = :remote_browser
  Capybara.javascript_driver = :remote_browser
end

Ключ --- в дополнение к Capybara.run_server = false и добавлению http:// к app_host --- указывал /wd/hub на URL.

Рабочее решение здесь.

Ответ 1

Для настройки capybara есть несколько вещей, о которых вам нужно беспокоиться -

1 - где capybara подключается к браузере - в большинстве случаев для людей это локально, но вы используете удаленный доступ, и вы, кажется, правильно настроили

2 - где браузер пытается подключиться - базовый url для этого установлен Capybara.app_host - поэтому Capybara.app_host = " http://<ip/domain имя приложения будет запущено в > - у вас установлено значение ENV ['APP_HOST'], для которого установлено значение app.under.test. Я предполагаю, что это имя ip/domain, которое разрешает доступный интерфейс на машина capybara запускается?

3 - где Capybara связывает тестируемое приложение - это устанавливается через Capybara.server_host и Capybara.server_port - server_host по умолчанию 127.0.0.1, и порт выбирается случайным образом. Скорее всего, там, где ваша проблема заключается в том, что 127.0.0.1 на машине, на которой работает Capybara, обычно недоступен с других машин. Вам нужно будет изменить это как внешний доступный ip, который соответствует app.under.test из # 2. Вам также может потребоваться установить порт, если это требуется для конфигурации брандмауэра.