Replace selenium-webdriver with playwright (#34867)

This commit is contained in:
Matt Jankowski 2025-06-10 12:33:46 -04:00 committed by GitHub
parent b8cc9b3290
commit 629bb74451
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 45 additions and 43 deletions

View file

@ -332,6 +332,21 @@ jobs:
- name: Load database schema - name: Load database schema
run: './bin/rails db:create db:schema:load db:seed' run: './bin/rails db:create db:schema:load db:seed'
- name: Cache Playwright Chromium browser
id: playwright-cache
uses: actions/cache@v4
with:
path: ~/.cache/ms-playwright
key: playwright-browsers-${{ runner.os }}-${{ hashFiles('yarn.lock') }}
- name: Install Playwright Chromium browser (with deps)
if: steps.playwright-cache.outputs.cache-hit != 'true'
run: yarn run playwright install --with-deps chromium
- name: Install Playwright Chromium browser deps
if: steps.playwright-cache.outputs.cache-hit == 'true'
run: yarn run playwright install-deps chromium
- run: bin/rspec spec/system --tag streaming --tag js - run: bin/rspec spec/system --tag streaming --tag js
- name: Archive logs - name: Archive logs

View file

@ -137,7 +137,7 @@ group :test do
# Browser integration testing # Browser integration testing
gem 'capybara', '~> 3.39' gem 'capybara', '~> 3.39'
gem 'selenium-webdriver' gem 'capybara-playwright-driver'
# Used to reset the database between system tests # Used to reset the database between system tests
gem 'database_cleaner-active_record' gem 'database_cleaner-active_record'

View file

@ -142,6 +142,10 @@ GEM
rack-test (>= 0.6.3) rack-test (>= 0.6.3)
regexp_parser (>= 1.5, < 3.0) regexp_parser (>= 1.5, < 3.0)
xpath (~> 3.2) xpath (~> 3.2)
capybara-playwright-driver (0.5.6)
addressable
capybara
playwright-ruby-client (>= 1.16.0)
case_transform (0.2) case_transform (0.2)
activesupport activesupport
cbor (0.5.9.8) cbor (0.5.9.8)
@ -603,6 +607,9 @@ GEM
pg (1.5.9) pg (1.5.9)
pghero (3.7.0) pghero (3.7.0)
activerecord (>= 7.1) activerecord (>= 7.1)
playwright-ruby-client (1.52.0)
concurrent-ruby (>= 1.1.6)
mime-types (>= 3.0)
pp (0.6.2) pp (0.6.2)
prettyprint prettyprint
premailer (1.27.0) premailer (1.27.0)
@ -808,12 +815,6 @@ GEM
activerecord (>= 4.0.0) activerecord (>= 4.0.0)
railties (>= 4.0.0) railties (>= 4.0.0)
securerandom (0.4.1) securerandom (0.4.1)
selenium-webdriver (4.33.0)
base64 (~> 0.2)
logger (~> 1.4)
rexml (~> 3.2, >= 3.2.5)
rubyzip (>= 1.2.2, < 3.0)
websocket (~> 1.0)
shoulda-matchers (6.5.0) shoulda-matchers (6.5.0)
activesupport (>= 5.2.0) activesupport (>= 5.2.0)
sidekiq (7.3.9) sidekiq (7.3.9)
@ -927,7 +928,6 @@ GEM
crack (>= 0.3.2) crack (>= 0.3.2)
hashdiff (>= 0.4.0, < 2.0.0) hashdiff (>= 0.4.0, < 2.0.0)
webrick (1.9.1) webrick (1.9.1)
websocket (1.2.11)
websocket-driver (0.7.7) websocket-driver (0.7.7)
base64 base64
websocket-extensions (>= 0.1.0) websocket-extensions (>= 0.1.0)
@ -955,6 +955,7 @@ DEPENDENCIES
browser browser
bundler-audit (~> 0.9) bundler-audit (~> 0.9)
capybara (~> 3.39) capybara (~> 3.39)
capybara-playwright-driver
charlock_holmes (~> 0.7.7) charlock_holmes (~> 0.7.7)
chewy (~> 7.3) chewy (~> 7.3)
climate_control climate_control
@ -1070,7 +1071,6 @@ DEPENDENCIES
rubyzip (~> 2.3) rubyzip (~> 2.3)
sanitize (~> 7.0) sanitize (~> 7.0)
scenic (~> 1.7) scenic (~> 1.7)
selenium-webdriver
shoulda-matchers shoulda-matchers
sidekiq (< 8) sidekiq (< 8)
sidekiq-bulk (~> 0.2.0) sidekiq-bulk (~> 0.2.0)

View file

@ -9,29 +9,36 @@ end
RSpec.configure do |config| RSpec.configure do |config|
config.include BrowserErrorsHelpers, :js, type: :system config.include BrowserErrorsHelpers, :js, type: :system
config.before(:each, :js, type: :system) do config.before(:each, :js, type: :system) do |example|
@ignored_js_errors_for_spec = [] @ignored_js_errors_for_spec = []
example.metadata[:js_console_messages] ||= []
Capybara.current_session.driver.with_playwright_page do |page|
page.on('console', lambda { |msg|
example.metadata[:js_console_messages] << { type: msg.type, text: msg.text, location: msg.location }
})
end
end end
config.after(:each, :js, type: :system) do config.after(:each, :js, type: :system) do |example|
# Classes of intermittent ignorable errors # Classes of intermittent ignorable errors
ignored_errors = [ ignored_errors = [
/Error while trying to use the following icon from the Manifest/, # https://github.com/mastodon/mastodon/pull/30793 /Error while trying to use the following icon from the Manifest/, # https://github.com/mastodon/mastodon/pull/30793
/Manifest: Line: 1, column: 1, Syntax error/, # Similar parsing/interruption issue as above /Manifest: Line: 1, column: 1, Syntax error/, # Similar parsing/interruption issue as above
].concat(@ignored_js_errors_for_spec) ].concat(@ignored_js_errors_for_spec)
errors = page.driver.browser.logs.get(:browser).reject do |error| errors = example.metadata[:js_console_messages].reject do |msg|
ignored_errors.any? { |pattern| pattern.match(error.message) } ignored_errors.any? { |pattern| pattern.match(msg[:text]) }
end end
if errors.present? if errors.present?
aggregate_failures 'browser errrors' do aggregate_failures 'browser errrors' do
errors.each do |error| errors.each do |error|
expect(error.level).to_not eq('SEVERE'), error.message expect(error[:type]).to_not eq('error'), error[:text]
next unless error.level == 'WARNING' next unless error[:type] == 'warning'
warn 'WARN: browser warning' warn 'WARN: browser warning'
warn error.message warn error[:text]
end end
end end
end end

View file

@ -4,36 +4,16 @@ Capybara.server_host = 'localhost'
Capybara.server_port = 3000 Capybara.server_port = 3000
Capybara.app_host = "http://#{Capybara.server_host}:#{Capybara.server_port}" Capybara.app_host = "http://#{Capybara.server_host}:#{Capybara.server_port}"
require 'selenium/webdriver' Capybara.register_driver(:playwright) do |app|
Capybara::Playwright::Driver.new(app)
def common_chrome_options
options = Selenium::WebDriver::Chrome::Options.new
options.add_argument '--window-size=1680,1050'
options.add_argument '--disable-search-engine-choice-screen'
options
end end
Capybara.javascript_driver = :playwright
Capybara.register_driver :chrome do |app| if ENV['CI'].present?
Capybara::Selenium::Driver.new(app, browser: :chrome, options: common_chrome_options) # Reduce intermittent failures from slow CI runner environment
Capybara.default_max_wait_time = 2**3
end end
Capybara.register_driver :headless_chrome do |app|
options = common_chrome_options
options.add_argument '--headless=new'
Capybara::Selenium::Driver.new(
app,
browser: :chrome,
options: options
)
end
Capybara.javascript_driver = :headless_chrome
# Some of the flaky tests seem to be caused by github runners being too slow for the
# default timeout of 2 seconds
Capybara.default_max_wait_time = 8
RSpec.configure do |config| RSpec.configure do |config|
config.before(:each, type: :system) do config.before(:each, type: :system) do
driven_by :rack_test driven_by :rack_test

View file

@ -26,7 +26,7 @@ RSpec.describe 'Log out' do
describe 'Logging out from the JS app', :js, :streaming do describe 'Logging out from the JS app', :js, :streaming do
it 'logs the user out' do it 'logs the user out' do
# The frontend tries to load announcements after a short delay, but the session might be expired by then, and the browser will output an error. # The frontend tries to load announcements after a short delay, but the session might be expired by then, and the browser will output an error.
ignore_js_error(/Failed to load resource: the server responded with a status of 422/) ignore_js_error(/Failed to load resource: the server responded with a status/)
visit root_path visit root_path
expect(page) expect(page)