Skip to content

Commit 981f8e8

Browse files
feat: init support for e2e libs with Playwright (#131)
Abstract Middleware logic from dependency on Cypress Add support for Playwright --------- Signed-off-by: Khaled Emara <[email protected]>
1 parent 2dbf3bf commit 981f8e8

39 files changed

+452
-111
lines changed

.github/workflows/ruby.yml

+3-3
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ jobs:
1616
uses: ruby/setup-ruby@v1
1717
with:
1818
ruby-version: 2.3
19-
bundler-cache: true
19+
bundler-cache: true
2020
- name: Run tests
2121
run: bundle exec rake
2222
- run: gem uninstall -v '>= 2' -ax bundler || true
@@ -54,10 +54,10 @@ jobs:
5454
uses: ruby/setup-ruby@v1
5555
with:
5656
ruby-version: 2.6
57-
bundler-cache: true
57+
bundler-cache: true
5858
- name: Run tests
5959
run: bundle exec rake
6060
- name: Run interaction tests
6161
run: ./specs_e2e/rails_5_2/test.sh
6262
env:
63-
CYPRESS_RECORD_KEY: ${{ secrets.CYPRESS_RECORD_KEY }}
63+
CYPRESS_RECORD_KEY: ${{ secrets.CYPRESS_RECORD_KEY }}

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
### Changed
2+
* Add support for any e2e testing framewrok starting with Playwright [PR 131](https://github.com/shakacode/cypress-on-rails/pull/131) by [KhaledEmaraDev]
3+
14
## [1.14.0]
25
[Compare]: https://github.com/shakacode/cypress-on-rails/compare/v1.13.1...v1.14.0
36

README.md

+62-18
Original file line numberDiff line numberDiff line change
@@ -14,20 +14,21 @@ Need help with cypress-on-rails? Contact [ShakaCode](mailto:[email protected]
1414

1515
----
1616

17-
# Playwright.dev support
18-
We're working on full support for [Playwright.dev](https://playwright.dev/). See [issue #116](https://github.com/shakacode/cypress-on-rails/issues/116#issuecomment-1523946478) for details.
19-
2017
# Totally new to Cypress?
2118
Suggest you first learn the basics of Cypress before attempting to integrate with Ruby on Rails
2219

2320
* [Good start Here](https://docs.cypress.io/examples/examples/tutorials.html#Best-Practices)
2421

22+
# Totally new to Playwright?
23+
Suggest you first learn the basics of Playwright before attempting to integrate with Ruby on Rails
24+
25+
* [Good start Here](https://playwright.dev/docs/writing-tests)
26+
2527
## Overview
2628

27-
Gem for using [cypress.io](http://github.com/cypress-io/) in Rails and Ruby Rack applications
28-
with the goal of controlling state as mentioned in [Cypress Best Practices](https://docs.cypress.io/guides/references/best-practices.html#Organizing-Tests-Logging-In-Controlling-State)
29+
Gem for using [cypress.io](http://github.com/cypress-io/) or [playwright.dev](https://playwright.dev/) in Rails and Ruby Rack applications with the goal of controlling state as mentioned in [Cypress Best Practices](https://docs.cypress.io/guides/references/best-practices.html#Organizing-Tests-Logging-In-Controlling-State)
2930

30-
It allows you to run code in the application context when executing cypress tests.
31+
It allows you to run code in the application context when executing cypress or playwright tests.
3132
Do things like:
3233
* use database_cleaner before each test
3334
* seed the database with default data for each test
@@ -57,11 +58,18 @@ end
5758
Generate the boilerplate code using:
5859

5960
```shell
61+
# by default installs only cypress
6062
bin/rails g cypress_on_rails:install
6163

6264
# if you have/want a different cypress folder (default is cypress)
6365
bin/rails g cypress_on_rails:install --cypress_folder=spec/cypress
6466

67+
# to install both cypress and playwright
68+
bin/rails g cypress_on_rails:install --install_cypress --install_playwright --playwright_folder=playwright
69+
70+
# to change where the Ruby files reside (default is e2e)
71+
bin/rails g cypress_on_rails:install --install_folder=test/e2e
72+
6573
# if you target the Rails server with a path prefix to your URL
6674
bin/rails g cypress_on_rails:install --api_prefix=/api
6775

@@ -76,21 +84,22 @@ bin/rails g cypress_on_rails:update
7684
```
7785

7886
The generator modifies/adds the following files/directory in your application:
79-
* `config/environments/test.rb`
8087
* `config/initializers/cypress_on_rails.rb` used to configure Cypress on Rails
8188
* `spec/cypress/e2e/` contains your cypress tests
89+
* `spec/playwright/e2e/` contains your playwright tests
8290
* `spec/cypress/support/on-rails.js` contains Cypress on Rails support code
83-
* `spec/cypress/app_commands/scenarios/` contains your Cypress on Rails scenario definitions
84-
* `spec/cypress/cypress_helper.rb` contains helper code for Cypress on Rails app commands
91+
* `spec/playwright/support/on-rails.js` contains Playwright on Rails support code
92+
* `spec/e2e/app_commands/scenarios/` contains your Cypress on Rails scenario definitions
93+
* `spec/e2e/cypress_helper.rb` contains helper code for Cypress on Rails app commands
8594

86-
If you are not using `database_cleaner` look at `spec/cypress/app_commands/clean.rb`.
87-
If you are not using `factory_bot` look at `spec/cypress/app_commands/factory_bot.rb`.
95+
If you are not using `database_cleaner` look at `spec/e2e/app_commands/clean.rb`.
96+
If you are not using `factory_bot` look at `spec/e2e/app_commands/factory_bot.rb`.
8897

8998
Now you can create scenarios and commands that are plain Ruby files that get loaded through middleware, the ruby sky is your limit.
9099

91100
### Update your database.yml
92101

93-
When running `cypress test` on your local computer it's recommended to start your server in development mode so that changes you
102+
When running `cypress test` or `playwright test` on your local computer it's recommended to start your server in development mode so that changes you
94103
make are picked up without having to restart the server.
95104
It's recommended you update your `database.yml` to check if the `CYPRESS` environment variable is set and switch it to the test database
96105
otherwise cypress will keep clearing your development database.
@@ -123,6 +132,10 @@ yarn cypress open
123132
node_modules/.bin/cypress open
124133
# or if you changed the cypress folder to spec/cypress
125134
yarn cypress open --project ./spec
135+
# or for playwright
136+
yarn playwright test --ui
137+
# or using npm
138+
npx playwright test --ui
126139
```
127140

128141
How to run cypress on CI
@@ -133,7 +146,7 @@ How to run cypress on CI
133146

134147
yarn run cypress run
135148
# or for npm
136-
node_modules/.bin/cypress run
149+
npx cypress run
137150
```
138151

139152
### Example of using factory bot
@@ -203,7 +216,7 @@ describe('My First Test', () => {
203216

204217
### Example of loading Rails test fixtures
205218
```ruby
206-
# spec/cypress/app_commands/activerecord_fixtures.rb
219+
# spec/e2e/app_commands/activerecord_fixtures.rb
207220
require "active_record/fixtures"
208221

209222
fixtures_dir = ActiveRecord::Tasks::DatabaseTasks.fixtures_path
@@ -233,7 +246,7 @@ describe('My First Test', () => {
233246

234247
Scenarios are named `before` blocks that you can reference in your test.
235248

236-
You define a scenario in the `spec/cypress/app_commands/scenarios` directory:
249+
You define a scenario in the `spec/e2e/app_commands/scenarios` directory:
237250
```ruby
238251
# spec/cypress/app_commands/scenarios/basic.rb
239252
Profile.create name: "Cypress Hill"
@@ -259,9 +272,9 @@ describe('My First Test', () => {
259272

260273
### Example of using app commands
261274

262-
Create a Ruby file in the `spec/cypress/app_commands` directory:
275+
Create a Ruby file in the `spec/e2e/app_commands` directory:
263276
```ruby
264-
# spec/cypress/app_commands/load_seed.rb
277+
# spec/e2e/app_commands/load_seed.rb
265278
load "#{Rails.root}/db/seeds.rb"
266279
```
267280

@@ -279,6 +292,37 @@ describe('My First Test', () => {
279292
})
280293
```
281294

295+
### Example of using scenario with Playwright
296+
297+
Scenarios are named `before` blocks that you can reference in your test.
298+
299+
You define a scenario in the `spec/e2e/app_commands/scenarios` directory:
300+
```ruby
301+
# spec/e2e/app_commands/scenarios/basic.rb
302+
Profile.create name: "Cypress Hill"
303+
304+
# or if you have factory_bot enabled in your cypress_helper
305+
CypressOnRails::SmartFactoryWrapper.create(:profile, name: "Cypress Hill")
306+
```
307+
308+
Then reference the scenario in your test:
309+
```js
310+
// spec/playwright/e2e/scenario_example.spec.js
311+
import { test, expect } from "@playwright/test";
312+
import { app, appScenario } from '../../support/on-rails';
313+
314+
test.describe("Rails using scenarios examples", () => {
315+
test.beforeEach(async ({ page }) => {
316+
await app('clean');
317+
});
318+
319+
test("setup basic scenario", async ({ page }) => {
320+
await appScenario('basic');
321+
await page.goto("/");
322+
});
323+
});
324+
```
325+
282326
## Experimental Features (matching npm package)
283327

284328
Please test and give feedback.
@@ -334,7 +378,7 @@ describe('My First Test', () => {
334378
beforeEach(() => { cy.app('load_seed') })
335379

336380
it('visit root', () => {
337-
cy.app('clean') // have a look at cypress/app_commands/clean.rb
381+
cy.app('clean') // have a look at e2e/app_commands/clean.rb
338382

339383
cy.vcr_insert_cassette('cats', { record: "new_episodes" })
340384
cy.visit('/using_vcr/index')

lib/cypress_on_rails/command_executor.rb

+9-5
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ module CypressOnRails
44
# loads and evals the command files
55
class CommandExecutor
66
def self.perform(file,command_options = nil)
7-
load_cypress_helper
7+
load_e2e_helper
88
file_data = File.read(file)
99
eval file_data, binding, file
1010
rescue => e
@@ -13,12 +13,16 @@ def self.perform(file,command_options = nil)
1313
raise e
1414
end
1515

16-
def self.load_cypress_helper
17-
cypress_helper_file = "#{configuration.cypress_folder}/cypress_helper"
18-
if File.exist?("#{cypress_helper_file}.rb")
16+
def self.load_e2e_helper
17+
e2e_helper_file = "#{configuration.install_folder}/e2e_helper.rb"
18+
cypress_helper_file = "#{configuration.cypress_folder}/cypress_helper.rb"
19+
if File.exist?(e2e_helper_file)
20+
Kernel.require e2e_helper_file
21+
elsif File.exist?(cypress_helper_file)
1922
Kernel.require cypress_helper_file
23+
warn "cypress_helper.rb and cypress_folder are deprecated, please use the install generator to create e2e_helper.rb using install_folder"
2024
else
21-
logger.warn "could not find #{cypress_helper_file}.rb"
25+
logger.warn "could not find #{e2e_helper_file} nor #{cypress_helper_file}"
2226
end
2327
end
2428

lib/cypress_on_rails/configuration.rb

+12-2
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,22 @@
22

33
module CypressOnRails
44
class Configuration
5-
attr_accessor :cypress_folder
65
attr_accessor :api_prefix
6+
attr_accessor :install_folder
77
attr_accessor :use_middleware
88
attr_accessor :use_vcr_middleware
99
attr_accessor :logger
1010

11+
# Attributes for backwards compatibility
12+
def cypress_folder
13+
warn "cypress_folder is deprecated, please use install_folder"
14+
install_folder
15+
end
16+
def cypress_folder=(v)
17+
warn "cypress_folder= is deprecated, please use install_folder"
18+
self.install_folder = v
19+
end
20+
1121
def initialize
1222
reset
1323
end
@@ -16,8 +26,8 @@ def initialize
1626
alias :use_vcr_middleware? :use_vcr_middleware
1727

1828
def reset
19-
self.cypress_folder = 'spec/cypress'
2029
self.api_prefix = ''
30+
self.install_folder = 'spec/e2e'
2131
self.use_middleware = true
2232
self.use_vcr_middleware = false
2333
self.logger = Logger.new(STDOUT)

lib/cypress_on_rails/middleware.rb

+8-5
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
require 'cypress_on_rails/command_executor'
55

66
module CypressOnRails
7-
# Middleware to handle cypress commands and eval
7+
# Middleware to handle testing framework commands and eval
88
class Middleware
99
include MiddlewareConfig
1010

@@ -16,16 +16,19 @@ def initialize(app, command_executor = CommandExecutor, file = ::File)
1616

1717
def call(env)
1818
request = Rack::Request.new(env)
19-
if request.path.start_with?("#{configuration.api_prefix}/__cypress__/command")
19+
if request.path.start_with?("#{configuration.api_prefix}/__e2e__/command")
2020
configuration.tagged_logged { handle_command(request) }
21+
elsif request.path.start_with?("#{configuration.api_prefix}/__cypress__/command")
22+
configuration.tagged_logged { handle_command(request) }
23+
warn "/__cypress__/command is deprecated. Please use the install generator to use /__e2e__/command instead."
2124
else
2225
@app.call(env)
2326
end
2427
end
2528

2629
private
2730

28-
Command = Struct.new(:name, :options, :cypress_folder) do
31+
Command = Struct.new(:name, :options, :install_folder) do
2932
# @return [Array<Cypress::Middleware::Command>]
3033
def self.from_body(body, configuration)
3134
if body.is_a?(Array)
@@ -34,12 +37,12 @@ def self.from_body(body, configuration)
3437
command_params = [body]
3538
end
3639
command_params.map do |params|
37-
new(params.fetch('name'), params['options'], configuration.cypress_folder)
40+
new(params.fetch('name'), params['options'], configuration.install_folder)
3841
end
3942
end
4043

4144
def file_path
42-
"#{cypress_folder}/app_commands/#{name}.rb"
45+
"#{install_folder}/app_commands/#{name}.rb"
4346
end
4447
end
4548

lib/cypress_on_rails/vcr_middleware.rb

+3-3
Original file line numberDiff line numberDiff line change
@@ -15,9 +15,9 @@ def initialize(app, vcr = nil)
1515

1616
def call(env)
1717
request = Rack::Request.new(env)
18-
if request.path.start_with?('/__cypress__/vcr/insert')
18+
if request.path.start_with?('/__e2e__/vcr/insert')
1919
configuration.tagged_logged { handle_insert(request) }
20-
elsif request.path.start_with?('/__cypress__/vcr/eject')
20+
elsif request.path.start_with?('/__e2e__/vcr/eject')
2121
configuration.tagged_logged { handle_eject }
2222
else
2323
do_first_call unless @first_call
@@ -57,7 +57,7 @@ def vcr
5757
return @vcr if @vcr
5858
require 'vcr'
5959
VCR.configure do |config|
60-
config.cassette_library_dir = "#{configuration.cypress_folder}/fixtures/vcr_cassettes"
60+
config.cassette_library_dir = "#{configuration.install_folder}/fixtures/vcr_cassettes"
6161
end
6262
@vcr = VCR
6363
end

0 commit comments

Comments
 (0)