2016-09-01 18:46:56 -04:00
|
|
|
# frozen_string_literal: true
|
2018-02-28 19:11:41 -05:00
|
|
|
|
2016-09-01 18:46:56 -04:00
|
|
|
require 'spec_helper'
|
2018-07-10 17:18:39 -04:00
|
|
|
require 'selenium-webdriver'
|
2016-09-01 18:46:56 -04:00
|
|
|
require 'shared_selenium_session'
|
2016-10-13 22:50:03 -04:00
|
|
|
require 'rspec/shared_spec_matchers'
|
|
|
|
|
2018-02-28 19:11:41 -05:00
|
|
|
browser_options = ::Selenium::WebDriver::Firefox::Options.new
|
2018-10-19 13:09:52 -04:00
|
|
|
browser_options.headless! if ENV['HEADLESS']
|
2017-08-11 14:37:43 -04:00
|
|
|
# browser_options.add_option("log", {"level": "trace"})
|
2017-08-02 15:10:56 -04:00
|
|
|
|
2018-06-17 13:29:46 -04:00
|
|
|
browser_options.profile = Selenium::WebDriver::Firefox::Profile.new.tap do |profile|
|
|
|
|
profile['browser.download.dir'] = Capybara.save_path
|
|
|
|
profile['browser.download.folderList'] = 2
|
|
|
|
profile['browser.helperApps.neverAsk.saveToDisk'] = 'text/csv'
|
|
|
|
end
|
|
|
|
|
2018-11-18 16:27:00 -05:00
|
|
|
Capybara.register_driver :selenium_firefox do |app|
|
2017-08-07 16:21:11 -04:00
|
|
|
# ::Selenium::WebDriver.logger.level = "debug"
|
2016-10-13 22:50:03 -04:00
|
|
|
Capybara::Selenium::Driver.new(
|
|
|
|
app,
|
2017-08-11 14:37:43 -04:00
|
|
|
browser: :firefox,
|
2018-06-17 13:29:46 -04:00
|
|
|
options: browser_options,
|
2018-12-20 15:37:14 -05:00
|
|
|
timeout: 31
|
2017-08-07 16:21:11 -04:00
|
|
|
# Get a trace level log from geckodriver
|
|
|
|
# :driver_opts => { args: ['-vv'] }
|
2016-10-13 22:50:03 -04:00
|
|
|
)
|
2016-09-01 18:46:56 -04:00
|
|
|
end
|
|
|
|
|
2018-11-20 12:44:52 -05:00
|
|
|
Capybara.register_driver :selenium_firefox_not_clear_storage do |app|
|
2016-10-28 16:08:20 -04:00
|
|
|
Capybara::Selenium::Driver.new(
|
|
|
|
app,
|
|
|
|
browser: :firefox,
|
2018-11-20 12:44:52 -05:00
|
|
|
clear_local_storage: false,
|
|
|
|
clear_session_storage: false,
|
2017-08-11 14:37:43 -04:00
|
|
|
options: browser_options
|
2016-10-28 16:08:20 -04:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2016-09-01 18:46:56 -04:00
|
|
|
module TestSessions
|
2018-11-18 16:27:00 -05:00
|
|
|
SeleniumFirefox = Capybara::Session.new(:selenium_firefox, TestApp)
|
2016-09-01 18:46:56 -04:00
|
|
|
end
|
|
|
|
|
2018-02-28 19:11:41 -05:00
|
|
|
skipped_tests = %i[response_headers status_code trigger]
|
2016-09-01 18:46:56 -04:00
|
|
|
|
2019-04-19 15:22:17 -04:00
|
|
|
Capybara::SpecHelper.log_selenium_driver_version(Selenium::WebDriver::Firefox) if ENV['CI']
|
2018-03-13 17:22:50 -04:00
|
|
|
|
2018-11-18 16:27:00 -05:00
|
|
|
Capybara::SpecHelper.run_specs TestSessions::SeleniumFirefox, 'selenium', capybara_skip: skipped_tests do |example|
|
2018-07-05 14:50:43 -04:00
|
|
|
case example.metadata[:full_description]
|
|
|
|
when 'Capybara::Session selenium node #click should allow multiple modifiers'
|
2018-11-18 16:27:00 -05:00
|
|
|
pending "Firefox doesn't generate an event for shift+control+click" if firefox_gte?(62, @session) && !Gem.win_platform?
|
2018-07-05 14:50:43 -04:00
|
|
|
when /^Capybara::Session selenium node #double_click/
|
2018-11-18 16:27:00 -05:00
|
|
|
pending "selenium-webdriver/geckodriver doesn't generate double click event" if firefox_lt?(59, @session)
|
2018-07-05 14:50:43 -04:00
|
|
|
when 'Capybara::Session selenium #accept_prompt should accept the prompt with a blank response when there is a default'
|
2018-11-18 16:27:00 -05:00
|
|
|
pending "Geckodriver doesn't set a blank response in FF < 63 - https://bugzilla.mozilla.org/show_bug.cgi?id=1486485" if firefox_lt?(63, @session)
|
2018-07-24 17:13:03 -04:00
|
|
|
when 'Capybara::Session selenium #attach_file with multipart form should fire change once when uploading multiple files from empty'
|
2018-11-18 16:27:00 -05:00
|
|
|
pending "FF < 62 doesn't support setting all files at once" if firefox_lt?(62, @session)
|
2018-09-05 21:19:07 -04:00
|
|
|
when 'Capybara::Session selenium #accept_confirm should work with nested modals'
|
2018-11-18 16:27:00 -05:00
|
|
|
skip 'Broken in FF 63 - https://bugzilla.mozilla.org/show_bug.cgi?id=1487358' if firefox_gte?(63, @session)
|
2018-09-14 16:00:29 -04:00
|
|
|
when 'Capybara::Session selenium #click_link can download a file'
|
|
|
|
skip 'Need to figure out testing of file downloading on windows platform' if Gem.win_platform?
|
2018-10-02 17:26:31 -04:00
|
|
|
when 'Capybara::Session selenium #reset_session! removes ALL cookies'
|
|
|
|
pending "Geckodriver doesn't provide a way to remove cookies outside the current domain"
|
2019-02-27 13:42:29 -05:00
|
|
|
when 'Capybara::Session selenium #attach_file with a block can upload by clicking the file input'
|
2019-03-06 14:05:43 -05:00
|
|
|
pending "Geckodriver doesn't allow clicking on file inputs"
|
2018-07-05 14:50:43 -04:00
|
|
|
end
|
|
|
|
end
|
2016-09-01 18:46:56 -04:00
|
|
|
|
2018-07-10 17:18:39 -04:00
|
|
|
RSpec.describe 'Capybara::Session with firefox' do # rubocop:disable RSpec/MultipleDescribes
|
2017-05-08 18:12:33 -04:00
|
|
|
include Capybara::SpecHelper
|
2018-11-18 16:27:00 -05:00
|
|
|
include_examples 'Capybara::Session', TestSessions::SeleniumFirefox, :selenium_firefox
|
|
|
|
include_examples Capybara::RSpecMatchers, TestSessions::SeleniumFirefox, :selenium_firefox
|
2019-03-07 16:50:59 -05:00
|
|
|
|
|
|
|
describe 'filling in Firefox-specific date and time fields with keystrokes' do
|
|
|
|
let(:datetime) { Time.new(1983, 6, 19, 6, 30) }
|
|
|
|
let(:session) { TestSessions::SeleniumFirefox }
|
|
|
|
|
|
|
|
before do
|
|
|
|
session.visit('/form')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should fill in a date input with a String' do
|
|
|
|
session.fill_in('form_date', with: datetime.to_date.iso8601)
|
|
|
|
session.click_button('awesome')
|
|
|
|
expect(Date.parse(extract_results(session)['date'])).to eq datetime.to_date
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should fill in a time input with a String' do
|
|
|
|
session.fill_in('form_time', with: datetime.to_time.strftime('%T'))
|
|
|
|
session.click_button('awesome')
|
|
|
|
results = extract_results(session)['time']
|
|
|
|
expect(Time.parse(results).strftime('%r')).to eq datetime.strftime('%r')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should fill in a datetime input with a String' do
|
|
|
|
# FF doesn't currently support datetime-local so this is really just a text input
|
|
|
|
session.fill_in('form_datetime', with: datetime.iso8601)
|
|
|
|
session.click_button('awesome')
|
|
|
|
expect(Time.parse(extract_results(session)['datetime'])).to eq datetime
|
|
|
|
end
|
|
|
|
end
|
2016-09-01 18:46:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
RSpec.describe Capybara::Selenium::Driver do
|
2019-02-25 18:50:24 -05:00
|
|
|
let(:driver) { Capybara::Selenium::Driver.new(TestApp, browser: :firefox, options: browser_options) }
|
2016-09-01 18:46:56 -04:00
|
|
|
|
|
|
|
describe '#quit' do
|
2018-07-10 17:18:39 -04:00
|
|
|
it 'should reset browser when quit' do
|
2019-02-25 18:50:24 -05:00
|
|
|
expect(driver.browser).to be_truthy
|
|
|
|
driver.quit
|
2018-02-28 19:11:41 -05:00
|
|
|
# access instance variable directly so we don't create a new browser instance
|
2019-02-25 18:50:24 -05:00
|
|
|
expect(driver.instance_variable_get(:@browser)).to be_nil
|
2016-09-01 18:46:56 -04:00
|
|
|
end
|
2016-11-23 14:53:57 -05:00
|
|
|
|
2018-07-10 17:18:39 -04:00
|
|
|
context 'with errors' do
|
2019-02-25 18:50:24 -05:00
|
|
|
let!(:original_browser) { driver.browser }
|
2018-06-19 16:57:42 -04:00
|
|
|
|
2016-11-23 14:53:57 -05:00
|
|
|
after do
|
|
|
|
# Ensure browser is actually quit so we don't leave hanging processe
|
2019-02-25 18:50:24 -05:00
|
|
|
RSpec::Mocks.space.proxy_for(original_browser).reset
|
|
|
|
original_browser.quit
|
2016-11-23 14:53:57 -05:00
|
|
|
end
|
|
|
|
|
2018-07-10 17:18:39 -04:00
|
|
|
it 'warns UnknownError returned during quit because the browser is probably already gone' do
|
2019-02-25 18:50:24 -05:00
|
|
|
allow(driver).to receive(:warn)
|
|
|
|
allow(driver.browser).to(
|
2016-11-23 14:53:57 -05:00
|
|
|
receive(:quit)
|
2018-07-10 17:18:39 -04:00
|
|
|
.and_raise(Selenium::WebDriver::Error::UnknownError, 'random message')
|
2016-11-23 14:53:57 -05:00
|
|
|
)
|
|
|
|
|
2019-02-25 18:50:24 -05:00
|
|
|
expect { driver.quit }.not_to raise_error
|
|
|
|
expect(driver.instance_variable_get(:@browser)).to be_nil
|
|
|
|
expect(driver).to have_received(:warn).with(/random message/)
|
2016-11-23 14:53:57 -05:00
|
|
|
end
|
|
|
|
|
2018-07-10 17:18:39 -04:00
|
|
|
it 'ignores silenced UnknownError returned during quit because the browser is almost definitely already gone' do
|
2019-02-25 18:50:24 -05:00
|
|
|
allow(driver).to receive(:warn)
|
|
|
|
allow(driver.browser).to(
|
2016-11-23 14:53:57 -05:00
|
|
|
receive(:quit)
|
2018-07-10 17:18:39 -04:00
|
|
|
.and_raise(Selenium::WebDriver::Error::UnknownError, 'Error communicating with the remote browser')
|
2016-11-23 14:53:57 -05:00
|
|
|
)
|
|
|
|
|
2019-02-25 18:50:24 -05:00
|
|
|
expect { driver.quit }.not_to raise_error
|
|
|
|
expect(driver.instance_variable_get(:@browser)).to be_nil
|
|
|
|
expect(driver).not_to have_received(:warn)
|
2016-11-23 14:53:57 -05:00
|
|
|
end
|
|
|
|
end
|
2016-09-01 18:46:56 -04:00
|
|
|
end
|
2016-10-28 16:08:20 -04:00
|
|
|
|
2018-07-10 17:18:39 -04:00
|
|
|
context 'storage' do
|
|
|
|
describe '#reset!' do
|
2018-11-20 12:44:52 -05:00
|
|
|
it 'clears storage by default' do
|
2019-02-25 18:50:24 -05:00
|
|
|
session = TestSessions::SeleniumFirefox
|
|
|
|
session.visit('/with_js')
|
|
|
|
session.find(:css, '#set-storage').click
|
|
|
|
session.reset!
|
|
|
|
session.visit('/with_js')
|
|
|
|
expect(session.driver.browser.local_storage.keys).to be_empty
|
|
|
|
expect(session.driver.browser.session_storage.keys).to be_empty
|
2016-10-28 16:08:20 -04:00
|
|
|
end
|
|
|
|
|
2018-11-20 12:44:52 -05:00
|
|
|
it 'does not clear storage when false' do
|
2019-02-25 18:50:24 -05:00
|
|
|
session = Capybara::Session.new(:selenium_firefox_not_clear_storage, TestApp)
|
|
|
|
session.visit('/with_js')
|
|
|
|
session.find(:css, '#set-storage').click
|
|
|
|
session.reset!
|
|
|
|
session.visit('/with_js')
|
|
|
|
expect(session.driver.browser.local_storage.keys).not_to be_empty
|
|
|
|
expect(session.driver.browser.session_storage.keys).not_to be_empty
|
2016-10-28 16:08:20 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-11-20 17:20:46 -05:00
|
|
|
|
|
|
|
context 'timeout' do
|
|
|
|
it 'sets the http client read timeout' do
|
|
|
|
expect(TestSessions::SeleniumFirefox.driver.browser.send(:bridge).http.read_timeout).to eq 31
|
|
|
|
end
|
|
|
|
end
|
2016-09-01 18:46:56 -04:00
|
|
|
end
|
2018-06-06 16:21:33 -04:00
|
|
|
|
|
|
|
RSpec.describe Capybara::Selenium::Node do
|
2018-07-10 17:18:39 -04:00
|
|
|
context '#click' do
|
|
|
|
it 'warns when attempting on a table row' do
|
2018-11-18 16:27:00 -05:00
|
|
|
session = TestSessions::SeleniumFirefox
|
2018-06-06 16:21:33 -04:00
|
|
|
session.visit('/tables')
|
|
|
|
tr = session.find(:css, '#agent_table tr:first-child')
|
|
|
|
allow(tr.base).to receive(:warn)
|
|
|
|
tr.click
|
2018-06-06 19:11:47 -04:00
|
|
|
expect(tr.base).to have_received(:warn).with(/Clicking the first cell in the row instead/)
|
2018-06-06 16:21:33 -04:00
|
|
|
end
|
2018-07-03 16:31:31 -04:00
|
|
|
|
2018-07-10 17:18:39 -04:00
|
|
|
it 'should allow multiple modifiers', requires: [:js] do
|
2018-11-18 16:27:00 -05:00
|
|
|
session = TestSessions::SeleniumFirefox
|
2018-07-03 16:31:31 -04:00
|
|
|
session.visit('with_js')
|
2018-07-05 19:29:26 -04:00
|
|
|
# Firefox v62+ doesn't generate an event for control+shift+click
|
2018-07-03 16:31:31 -04:00
|
|
|
session.find(:css, '#click-test').click(:alt, :ctrl, :meta)
|
2018-07-05 19:29:26 -04:00
|
|
|
# it also triggers a contextmenu event when control is held so don't check click type
|
2018-07-10 17:18:39 -04:00
|
|
|
expect(session).to have_link('Has been alt control meta')
|
2018-07-03 16:31:31 -04:00
|
|
|
end
|
2018-06-06 16:21:33 -04:00
|
|
|
end
|
2018-09-23 13:09:38 -04:00
|
|
|
|
|
|
|
context '#send_keys' do
|
|
|
|
it 'should process space' do
|
2018-11-18 16:27:00 -05:00
|
|
|
session = TestSessions::SeleniumFirefox
|
2018-09-23 13:09:38 -04:00
|
|
|
session.visit('/form')
|
|
|
|
session.find(:css, '#address1_city').send_keys('ocean', [:shift, :space, 'side'])
|
|
|
|
expect(session.find(:css, '#address1_city').value).to eq 'ocean SIDE'
|
|
|
|
end
|
|
|
|
end
|
2018-06-06 16:21:33 -04:00
|
|
|
end
|