cypress-on-rails 1.18.0 → 1.20.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.github/workflows/claude-code-review.yml +57 -0
- data/.github/workflows/claude.yml +50 -0
- data/CHANGELOG.md +399 -98
- data/README.md +139 -19
- data/RELEASING.md +200 -0
- data/Rakefile +1 -4
- data/cypress-on-rails.gemspec +3 -2
- data/docs/BEST_PRACTICES.md +678 -0
- data/docs/DX_IMPROVEMENTS.md +163 -0
- data/docs/PLAYWRIGHT_GUIDE.md +554 -0
- data/docs/RELEASE.md +124 -0
- data/docs/TROUBLESHOOTING.md +351 -0
- data/docs/VCR_GUIDE.md +499 -0
- data/lib/cypress_on_rails/command_executor.rb +24 -0
- data/lib/cypress_on_rails/configuration.rb +32 -0
- data/lib/cypress_on_rails/railtie.rb +7 -0
- data/lib/cypress_on_rails/server.rb +258 -0
- data/lib/cypress_on_rails/state_reset_middleware.rb +58 -0
- data/lib/cypress_on_rails/version.rb +1 -1
- data/lib/generators/cypress_on_rails/install_generator.rb +2 -2
- data/lib/generators/cypress_on_rails/templates/config/initializers/cypress_on_rails.rb.erb +14 -2
- data/lib/generators/cypress_on_rails/templates/spec/cypress/e2e/rails_examples/using_factory_bot.cy.js +2 -2
- data/lib/generators/cypress_on_rails/templates/spec/cypress/e2e/rails_examples/using_scenarios.cy.js +1 -1
- data/lib/generators/cypress_on_rails/templates/spec/cypress/support/on-rails.js +1 -1
- data/lib/tasks/cypress.rake +33 -0
- data/rakelib/release.rake +80 -0
- data/rakelib/task_helpers.rb +23 -0
- data/rakelib/update_changelog.rake +63 -0
- data/spec/cypress_on_rails/configuration_spec.rb +6 -0
- data/spec/generators/install_generator_spec.rb +222 -0
- metadata +34 -4
    
        data/docs/VCR_GUIDE.md
    ADDED
    
    | @@ -0,0 +1,499 @@ | |
| 1 | 
            +
            # VCR Integration Guide
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            Complete guide for recording and replaying HTTP interactions in your tests using VCR with cypress-playwright-on-rails.
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            ## Table of Contents
         | 
| 6 | 
            +
            - [Overview](#overview)
         | 
| 7 | 
            +
            - [Installation](#installation)
         | 
| 8 | 
            +
            - [Configuration](#configuration)
         | 
| 9 | 
            +
            - [Insert/Eject Mode](#inserteject-mode)
         | 
| 10 | 
            +
            - [Use Cassette Mode](#use-cassette-mode)
         | 
| 11 | 
            +
            - [GraphQL Integration](#graphql-integration)
         | 
| 12 | 
            +
            - [Advanced Usage](#advanced-usage)
         | 
| 13 | 
            +
            - [Troubleshooting](#troubleshooting)
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            ## Overview
         | 
| 16 | 
            +
             | 
| 17 | 
            +
            VCR (Video Cassette Recorder) records your test suite's HTTP interactions and replays them during future test runs for fast, deterministic tests. This is particularly useful for:
         | 
| 18 | 
            +
            - Testing against third-party APIs
         | 
| 19 | 
            +
            - Avoiding rate limits
         | 
| 20 | 
            +
            - Testing without internet connection
         | 
| 21 | 
            +
            - Ensuring consistent test data
         | 
| 22 | 
            +
            - Speeding up test execution
         | 
| 23 | 
            +
             | 
| 24 | 
            +
            ## Installation
         | 
| 25 | 
            +
             | 
| 26 | 
            +
            ### 1. Add required gems
         | 
| 27 | 
            +
            ```ruby
         | 
| 28 | 
            +
            # Gemfile
         | 
| 29 | 
            +
            group :test, :development do
         | 
| 30 | 
            +
              gem 'vcr'
         | 
| 31 | 
            +
              gem 'webmock'
         | 
| 32 | 
            +
              gem 'cypress-on-rails', '~> 1.0'
         | 
| 33 | 
            +
            end
         | 
| 34 | 
            +
            ```
         | 
| 35 | 
            +
             | 
| 36 | 
            +
            ### 2. Install npm package (optional, for enhanced features)
         | 
| 37 | 
            +
            ```bash
         | 
| 38 | 
            +
            yarn add -D cypress-on-rails
         | 
| 39 | 
            +
            # or
         | 
| 40 | 
            +
            npm install --save-dev cypress-on-rails
         | 
| 41 | 
            +
            ```
         | 
| 42 | 
            +
             | 
| 43 | 
            +
            ## Configuration
         | 
| 44 | 
            +
             | 
| 45 | 
            +
            ### Basic VCR Setup
         | 
| 46 | 
            +
             | 
| 47 | 
            +
            ```ruby
         | 
| 48 | 
            +
            # config/initializers/cypress_on_rails.rb
         | 
| 49 | 
            +
            CypressOnRails.configure do |c|
         | 
| 50 | 
            +
              # Enable VCR middleware
         | 
| 51 | 
            +
              c.use_vcr_middleware = !Rails.env.production? && ENV['CYPRESS'].present?
         | 
| 52 | 
            +
              
         | 
| 53 | 
            +
              # VCR configuration options
         | 
| 54 | 
            +
              c.vcr_options = {
         | 
| 55 | 
            +
                # HTTP library to hook into
         | 
| 56 | 
            +
                hook_into: :webmock,
         | 
| 57 | 
            +
                
         | 
| 58 | 
            +
                # Default recording mode
         | 
| 59 | 
            +
                default_cassette_options: { 
         | 
| 60 | 
            +
                  record: :once,  # :once, :new_episodes, :none, :all
         | 
| 61 | 
            +
                  match_requests_on: [:method, :uri, :body],
         | 
| 62 | 
            +
                  allow_unused_http_interactions: false
         | 
| 63 | 
            +
                },
         | 
| 64 | 
            +
                
         | 
| 65 | 
            +
                # Where to save cassettes
         | 
| 66 | 
            +
                cassette_library_dir: Rails.root.join('spec/fixtures/vcr_cassettes'),
         | 
| 67 | 
            +
                
         | 
| 68 | 
            +
                # Configure which hosts to ignore
         | 
| 69 | 
            +
                ignore_hosts: ['localhost', '127.0.0.1', '0.0.0.0'],
         | 
| 70 | 
            +
                
         | 
| 71 | 
            +
                # Filter sensitive data
         | 
| 72 | 
            +
                filter_sensitive_data: {
         | 
| 73 | 
            +
                  '<API_KEY>' => ENV['EXTERNAL_API_KEY'],
         | 
| 74 | 
            +
                  '<AUTH_TOKEN>' => ENV['AUTH_TOKEN']
         | 
| 75 | 
            +
                },
         | 
| 76 | 
            +
                
         | 
| 77 | 
            +
                # Preserve exact body bytes for binary data
         | 
| 78 | 
            +
                preserve_exact_body_bytes: true,
         | 
| 79 | 
            +
                
         | 
| 80 | 
            +
                # Allow HTTP connections when no cassette
         | 
| 81 | 
            +
                allow_http_connections_when_no_cassette: false
         | 
| 82 | 
            +
              }
         | 
| 83 | 
            +
            end
         | 
| 84 | 
            +
            ```
         | 
| 85 | 
            +
             | 
| 86 | 
            +
            ### Cypress Setup
         | 
| 87 | 
            +
             | 
| 88 | 
            +
            ```js
         | 
| 89 | 
            +
            // cypress/support/index.js
         | 
| 90 | 
            +
            import 'cypress-on-rails/support/index'
         | 
| 91 | 
            +
             | 
| 92 | 
            +
            // Optional: Configure VCR commands
         | 
| 93 | 
            +
            Cypress.Commands.add('vcrInsert', (name, options = {}) => {
         | 
| 94 | 
            +
              cy.app('vcr_insert_cassette', { name, ...options });
         | 
| 95 | 
            +
            });
         | 
| 96 | 
            +
             | 
| 97 | 
            +
            Cypress.Commands.add('vcrEject', () => {
         | 
| 98 | 
            +
              cy.app('vcr_eject_cassette');
         | 
| 99 | 
            +
            });
         | 
| 100 | 
            +
            ```
         | 
| 101 | 
            +
             | 
| 102 | 
            +
            ### Clean Command Setup
         | 
| 103 | 
            +
             | 
| 104 | 
            +
            ```ruby
         | 
| 105 | 
            +
            # e2e/app_commands/clean.rb
         | 
| 106 | 
            +
            # Ensure cassettes are ejected between tests
         | 
| 107 | 
            +
            VCR.eject_cassette if VCR.current_cassette
         | 
| 108 | 
            +
            VCR.turn_off!
         | 
| 109 | 
            +
            WebMock.disable! if defined?(WebMock)
         | 
| 110 | 
            +
             | 
| 111 | 
            +
            # Your existing clean logic...
         | 
| 112 | 
            +
            DatabaseCleaner.clean
         | 
| 113 | 
            +
            ```
         | 
| 114 | 
            +
             | 
| 115 | 
            +
            ## Insert/Eject Mode
         | 
| 116 | 
            +
             | 
| 117 | 
            +
            Insert/eject mode gives you explicit control over when to start and stop recording.
         | 
| 118 | 
            +
             | 
| 119 | 
            +
            ### Configuration
         | 
| 120 | 
            +
            ```ruby
         | 
| 121 | 
            +
            CypressOnRails.configure do |c|
         | 
| 122 | 
            +
              c.use_vcr_middleware = !Rails.env.production? && ENV['CYPRESS'].present?
         | 
| 123 | 
            +
              # Don't enable use_cassette mode
         | 
| 124 | 
            +
            end
         | 
| 125 | 
            +
            ```
         | 
| 126 | 
            +
             | 
| 127 | 
            +
            ### Basic Usage
         | 
| 128 | 
            +
            ```js
         | 
| 129 | 
            +
            describe('External API Tests', () => {
         | 
| 130 | 
            +
              afterEach(() => {
         | 
| 131 | 
            +
                cy.vcr_eject_cassette();
         | 
| 132 | 
            +
              });
         | 
| 133 | 
            +
             | 
| 134 | 
            +
              it('fetches weather data', () => {
         | 
| 135 | 
            +
                // Start recording
         | 
| 136 | 
            +
                cy.vcr_insert_cassette('weather_api', { 
         | 
| 137 | 
            +
                  record: 'new_episodes' 
         | 
| 138 | 
            +
                });
         | 
| 139 | 
            +
                
         | 
| 140 | 
            +
                cy.visit('/weather');
         | 
| 141 | 
            +
                cy.contains('Current Temperature');
         | 
| 142 | 
            +
                
         | 
| 143 | 
            +
                // Recording continues until ejected
         | 
| 144 | 
            +
              });
         | 
| 145 | 
            +
             | 
| 146 | 
            +
              it('handles API errors', () => {
         | 
| 147 | 
            +
                // Use pre-recorded cassette
         | 
| 148 | 
            +
                cy.vcr_insert_cassette('weather_api_error', { 
         | 
| 149 | 
            +
                  record: 'none'  // Only replay, don't record
         | 
| 150 | 
            +
                });
         | 
| 151 | 
            +
                
         | 
| 152 | 
            +
                cy.visit('/weather?city=invalid');
         | 
| 153 | 
            +
                cy.contains('City not found');
         | 
| 154 | 
            +
              });
         | 
| 155 | 
            +
            });
         | 
| 156 | 
            +
            ```
         | 
| 157 | 
            +
             | 
| 158 | 
            +
            ### Advanced Options
         | 
| 159 | 
            +
            ```js
         | 
| 160 | 
            +
            cy.vcr_insert_cassette('api_calls', {
         | 
| 161 | 
            +
              record: 'new_episodes',           // Recording mode
         | 
| 162 | 
            +
              match_requests_on: ['method', 'uri', 'body'],  // Request matching
         | 
| 163 | 
            +
              erb: true,                         // Enable ERB in cassettes
         | 
| 164 | 
            +
              allow_playback_repeats: true,     // Allow multiple replays
         | 
| 165 | 
            +
              exclusive: true,                   // Disallow other cassettes
         | 
| 166 | 
            +
              serialize_with: 'json',           // Use JSON format
         | 
| 167 | 
            +
              preserve_exact_body_bytes: true,  // For binary data
         | 
| 168 | 
            +
              decode_compressed_response: true  // Handle gzipped responses
         | 
| 169 | 
            +
            });
         | 
| 170 | 
            +
            ```
         | 
| 171 | 
            +
             | 
| 172 | 
            +
            ## Use Cassette Mode
         | 
| 173 | 
            +
             | 
| 174 | 
            +
            Use cassette mode automatically wraps each request with VCR.use_cassette.
         | 
| 175 | 
            +
             | 
| 176 | 
            +
            ### Configuration
         | 
| 177 | 
            +
            ```ruby
         | 
| 178 | 
            +
            CypressOnRails.configure do |c|
         | 
| 179 | 
            +
              # Use this instead of use_vcr_middleware
         | 
| 180 | 
            +
              c.use_vcr_use_cassette_middleware = !Rails.env.production? && ENV['CYPRESS'].present?
         | 
| 181 | 
            +
              
         | 
| 182 | 
            +
              c.vcr_options = {
         | 
| 183 | 
            +
                hook_into: :webmock,
         | 
| 184 | 
            +
                default_cassette_options: { 
         | 
| 185 | 
            +
                  record: :once,
         | 
| 186 | 
            +
                  match_requests_on: [:method, :uri]
         | 
| 187 | 
            +
                },
         | 
| 188 | 
            +
                cassette_library_dir: Rails.root.join('spec/fixtures/vcr_cassettes')
         | 
| 189 | 
            +
              }
         | 
| 190 | 
            +
            end
         | 
| 191 | 
            +
            ```
         | 
| 192 | 
            +
             | 
| 193 | 
            +
            ### How It Works
         | 
| 194 | 
            +
            Each request is automatically wrapped with `VCR.use_cassette`. The cassette name is derived from the request URL or operation name.
         | 
| 195 | 
            +
             | 
| 196 | 
            +
            ### Directory Structure
         | 
| 197 | 
            +
            ```
         | 
| 198 | 
            +
            spec/fixtures/vcr_cassettes/
         | 
| 199 | 
            +
            ├── api/
         | 
| 200 | 
            +
            │   ├── users/
         | 
| 201 | 
            +
            │   │   └── index.yml
         | 
| 202 | 
            +
            │   └── products/
         | 
| 203 | 
            +
            │       ├── index.yml
         | 
| 204 | 
            +
            │       └── show.yml
         | 
| 205 | 
            +
            └── graphql/
         | 
| 206 | 
            +
                ├── GetUser.yml
         | 
| 207 | 
            +
                └── CreatePost.yml
         | 
| 208 | 
            +
            ```
         | 
| 209 | 
            +
             | 
| 210 | 
            +
            ## GraphQL Integration
         | 
| 211 | 
            +
             | 
| 212 | 
            +
            GraphQL requires special handling due to all requests going to the same endpoint.
         | 
| 213 | 
            +
             | 
| 214 | 
            +
            ### Setup for GraphQL
         | 
| 215 | 
            +
             | 
| 216 | 
            +
            ```js
         | 
| 217 | 
            +
            // cypress/support/commands.js
         | 
| 218 | 
            +
            Cypress.Commands.add('mockGraphQL', () => {
         | 
| 219 | 
            +
              cy.on('window:before:load', (win) => {
         | 
| 220 | 
            +
                const originalFetch = win.fetch;
         | 
| 221 | 
            +
                const fetch = (path, options, ...rest) => {
         | 
| 222 | 
            +
                  if (options && options.body) {
         | 
| 223 | 
            +
                    try {
         | 
| 224 | 
            +
                      const body = JSON.parse(options.body);
         | 
| 225 | 
            +
                      // Add operation name to URL for VCR matching
         | 
| 226 | 
            +
                      if (body.operationName) {
         | 
| 227 | 
            +
                        return originalFetch(
         | 
| 228 | 
            +
                          `${path}?operation=${body.operationName}`, 
         | 
| 229 | 
            +
                          options, 
         | 
| 230 | 
            +
                          ...rest
         | 
| 231 | 
            +
                        );
         | 
| 232 | 
            +
                      }
         | 
| 233 | 
            +
                    } catch (e) {
         | 
| 234 | 
            +
                      return originalFetch(path, options, ...rest);
         | 
| 235 | 
            +
                    }
         | 
| 236 | 
            +
                  }
         | 
| 237 | 
            +
                  return originalFetch(path, options, ...rest);
         | 
| 238 | 
            +
                };
         | 
| 239 | 
            +
                cy.stub(win, 'fetch', fetch);
         | 
| 240 | 
            +
              });
         | 
| 241 | 
            +
            });
         | 
| 242 | 
            +
             | 
| 243 | 
            +
            // cypress/support/index.js
         | 
| 244 | 
            +
            beforeEach(() => {
         | 
| 245 | 
            +
              cy.mockGraphQL();  // Enable GraphQL operation tracking
         | 
| 246 | 
            +
            });
         | 
| 247 | 
            +
            ```
         | 
| 248 | 
            +
             | 
| 249 | 
            +
            ### GraphQL Test Example
         | 
| 250 | 
            +
            ```js
         | 
| 251 | 
            +
            it('queries user data', () => {
         | 
| 252 | 
            +
              // Cassette will be saved as vcr_cassettes/graphql/GetUser.yml
         | 
| 253 | 
            +
              cy.visit('/profile');
         | 
| 254 | 
            +
              
         | 
| 255 | 
            +
              // The GraphQL query with operationName: 'GetUser' 
         | 
| 256 | 
            +
              // will be automatically recorded
         | 
| 257 | 
            +
              
         | 
| 258 | 
            +
              cy.contains('John Doe');
         | 
| 259 | 
            +
            });
         | 
| 260 | 
            +
            ```
         | 
| 261 | 
            +
             | 
| 262 | 
            +
            ### Custom GraphQL Matching
         | 
| 263 | 
            +
            ```ruby
         | 
| 264 | 
            +
            # config/initializers/cypress_on_rails.rb
         | 
| 265 | 
            +
            c.vcr_options = {
         | 
| 266 | 
            +
              match_requests_on: [:method, :uri, 
         | 
| 267 | 
            +
                lambda { |req1, req2|
         | 
| 268 | 
            +
                  # Custom matching for GraphQL requests
         | 
| 269 | 
            +
                  if req1.uri.path == '/graphql' && req2.uri.path == '/graphql'
         | 
| 270 | 
            +
                    body1 = JSON.parse(req1.body)
         | 
| 271 | 
            +
                    body2 = JSON.parse(req2.body)
         | 
| 272 | 
            +
                    
         | 
| 273 | 
            +
                    # Match by operation name and variables
         | 
| 274 | 
            +
                    body1['operationName'] == body2['operationName'] &&
         | 
| 275 | 
            +
                    body1['variables'] == body2['variables']
         | 
| 276 | 
            +
                  else
         | 
| 277 | 
            +
                    true
         | 
| 278 | 
            +
                  end
         | 
| 279 | 
            +
                }
         | 
| 280 | 
            +
              ]
         | 
| 281 | 
            +
            }
         | 
| 282 | 
            +
            ```
         | 
| 283 | 
            +
             | 
| 284 | 
            +
            ## Advanced Usage
         | 
| 285 | 
            +
             | 
| 286 | 
            +
            ### Dynamic Cassette Names
         | 
| 287 | 
            +
            ```js
         | 
| 288 | 
            +
            // Use test context for cassette names
         | 
| 289 | 
            +
            it('fetches user data', function() {
         | 
| 290 | 
            +
              const cassetteName = `${this.currentTest.parent.title}_${this.currentTest.title}`
         | 
| 291 | 
            +
                .replace(/\s+/g, '_')
         | 
| 292 | 
            +
                .toLowerCase();
         | 
| 293 | 
            +
              
         | 
| 294 | 
            +
              cy.vcr_insert_cassette(cassetteName, { record: 'once' });
         | 
| 295 | 
            +
              
         | 
| 296 | 
            +
              cy.visit('/users');
         | 
| 297 | 
            +
              // Test continues...
         | 
| 298 | 
            +
            });
         | 
| 299 | 
            +
            ```
         | 
| 300 | 
            +
             | 
| 301 | 
            +
            ### Conditional Recording
         | 
| 302 | 
            +
            ```js
         | 
| 303 | 
            +
            const shouldRecord = Cypress.env('RECORD_VCR') === 'true';
         | 
| 304 | 
            +
             | 
| 305 | 
            +
            cy.vcr_insert_cassette('api_calls', {
         | 
| 306 | 
            +
              record: shouldRecord ? 'new_episodes' : 'none'
         | 
| 307 | 
            +
            });
         | 
| 308 | 
            +
            ```
         | 
| 309 | 
            +
             | 
| 310 | 
            +
            ### Multiple Cassettes
         | 
| 311 | 
            +
            ```js
         | 
| 312 | 
            +
            it('combines multiple API sources', () => {
         | 
| 313 | 
            +
              // Stack multiple cassettes
         | 
| 314 | 
            +
              cy.vcr_insert_cassette('weather_api');
         | 
| 315 | 
            +
              cy.vcr_insert_cassette('news_api');
         | 
| 316 | 
            +
              
         | 
| 317 | 
            +
              cy.visit('/dashboard');
         | 
| 318 | 
            +
              
         | 
| 319 | 
            +
              // Both APIs will be recorded
         | 
| 320 | 
            +
              
         | 
| 321 | 
            +
              // Eject in reverse order
         | 
| 322 | 
            +
              cy.vcr_eject_cassette(); // Ejects news_api
         | 
| 323 | 
            +
              cy.vcr_eject_cassette(); // Ejects weather_api
         | 
| 324 | 
            +
            });
         | 
| 325 | 
            +
            ```
         | 
| 326 | 
            +
             | 
| 327 | 
            +
            ### Custom Matchers
         | 
| 328 | 
            +
            ```ruby
         | 
| 329 | 
            +
            # e2e/app_commands/vcr_custom.rb
         | 
| 330 | 
            +
            VCR.configure do |c|
         | 
| 331 | 
            +
              # Custom request matcher
         | 
| 332 | 
            +
              c.register_request_matcher :uri_ignoring_params do |req1, req2|
         | 
| 333 | 
            +
                URI(req1.uri).host == URI(req2.uri).host &&
         | 
| 334 | 
            +
                URI(req1.uri).path == URI(req2.uri).path
         | 
| 335 | 
            +
              end
         | 
| 336 | 
            +
            end
         | 
| 337 | 
            +
             | 
| 338 | 
            +
            # Use in test
         | 
| 339 | 
            +
            VCR.use_cassette('api_call', 
         | 
| 340 | 
            +
              match_requests_on: [:method, :uri_ignoring_params]
         | 
| 341 | 
            +
            )
         | 
| 342 | 
            +
            ```
         | 
| 343 | 
            +
             | 
| 344 | 
            +
            ### Filtering Sensitive Data
         | 
| 345 | 
            +
            ```ruby
         | 
| 346 | 
            +
            VCR.configure do |c|
         | 
| 347 | 
            +
              # Filter authorization headers
         | 
| 348 | 
            +
              c.filter_sensitive_data('<AUTHORIZATION>') do |interaction|
         | 
| 349 | 
            +
                interaction.request.headers['Authorization']&.first
         | 
| 350 | 
            +
              end
         | 
| 351 | 
            +
              
         | 
| 352 | 
            +
              # Filter API keys from URLs
         | 
| 353 | 
            +
              c.filter_sensitive_data('<API_KEY>') do |interaction|
         | 
| 354 | 
            +
                URI(interaction.request.uri).query
         | 
| 355 | 
            +
                  &.match(/api_key=([^&]+)/)
         | 
| 356 | 
            +
                  &.captures
         | 
| 357 | 
            +
                  &.first
         | 
| 358 | 
            +
              end
         | 
| 359 | 
            +
              
         | 
| 360 | 
            +
              # Filter response tokens
         | 
| 361 | 
            +
              c.filter_sensitive_data('<TOKEN>') do |interaction|
         | 
| 362 | 
            +
                JSON.parse(interaction.response.body)['token'] rescue nil
         | 
| 363 | 
            +
              end
         | 
| 364 | 
            +
            end
         | 
| 365 | 
            +
            ```
         | 
| 366 | 
            +
             | 
| 367 | 
            +
            ## Troubleshooting
         | 
| 368 | 
            +
             | 
| 369 | 
            +
            ### Issue: "No route matches [POST] '/api/__e2e__/vcr/insert'"
         | 
| 370 | 
            +
             | 
| 371 | 
            +
            **Solution:** Ensure VCR middleware is enabled:
         | 
| 372 | 
            +
            ```ruby
         | 
| 373 | 
            +
            # config/initializers/cypress_on_rails.rb
         | 
| 374 | 
            +
            c.use_vcr_middleware = !Rails.env.production? && ENV['CYPRESS'].present?
         | 
| 375 | 
            +
            ```
         | 
| 376 | 
            +
             | 
| 377 | 
            +
            And that the API prefix matches:
         | 
| 378 | 
            +
            ```ruby
         | 
| 379 | 
            +
            c.api_prefix = '/api'  # If your app uses /api prefix
         | 
| 380 | 
            +
            ```
         | 
| 381 | 
            +
             | 
| 382 | 
            +
            ### Issue: "VCR::Errors::UnhandledHTTPRequestError"
         | 
| 383 | 
            +
             | 
| 384 | 
            +
            **Cause:** Request not matching any cassette.
         | 
| 385 | 
            +
             | 
| 386 | 
            +
            **Solutions:**
         | 
| 387 | 
            +
            1. Re-record the cassette:
         | 
| 388 | 
            +
            ```js
         | 
| 389 | 
            +
            cy.vcr_insert_cassette('my_cassette', { record: 'new_episodes' });
         | 
| 390 | 
            +
            ```
         | 
| 391 | 
            +
             | 
| 392 | 
            +
            2. Adjust matching criteria:
         | 
| 393 | 
            +
            ```ruby
         | 
| 394 | 
            +
            c.vcr_options = {
         | 
| 395 | 
            +
              default_cassette_options: {
         | 
| 396 | 
            +
                match_requests_on: [:method, :host, :path]  # Ignore query params
         | 
| 397 | 
            +
              }
         | 
| 398 | 
            +
            }
         | 
| 399 | 
            +
            ```
         | 
| 400 | 
            +
             | 
| 401 | 
            +
            3. Allow new requests:
         | 
| 402 | 
            +
            ```ruby
         | 
| 403 | 
            +
            c.vcr_options = {
         | 
| 404 | 
            +
              default_cassette_options: {
         | 
| 405 | 
            +
                record: 'new_episodes',  # Record new requests
         | 
| 406 | 
            +
                allow_unused_http_interactions: true
         | 
| 407 | 
            +
              }
         | 
| 408 | 
            +
            }
         | 
| 409 | 
            +
            ```
         | 
| 410 | 
            +
             | 
| 411 | 
            +
            ### Issue: "Cassette not found"
         | 
| 412 | 
            +
             | 
| 413 | 
            +
            **Solution:** Check the cassette path:
         | 
| 414 | 
            +
            ```ruby
         | 
| 415 | 
            +
            # Verify the directory exists
         | 
| 416 | 
            +
            c.vcr_options = {
         | 
| 417 | 
            +
              cassette_library_dir: Rails.root.join('spec/fixtures/vcr_cassettes')
         | 
| 418 | 
            +
            }
         | 
| 419 | 
            +
            ```
         | 
| 420 | 
            +
             | 
| 421 | 
            +
            Create the directory if needed:
         | 
| 422 | 
            +
            ```bash
         | 
| 423 | 
            +
            mkdir -p spec/fixtures/vcr_cassettes
         | 
| 424 | 
            +
            ```
         | 
| 425 | 
            +
             | 
| 426 | 
            +
            ### Issue: "WebMock::NetConnectNotAllowedError"
         | 
| 427 | 
            +
             | 
| 428 | 
            +
            **Cause:** HTTP connection attempted without cassette.
         | 
| 429 | 
            +
             | 
| 430 | 
            +
            **Solutions:**
         | 
| 431 | 
            +
            1. Insert a cassette before the request:
         | 
| 432 | 
            +
            ```js
         | 
| 433 | 
            +
            cy.vcr_insert_cassette('api_calls');
         | 
| 434 | 
            +
            ```
         | 
| 435 | 
            +
             | 
| 436 | 
            +
            2. Allow connections to specific hosts:
         | 
| 437 | 
            +
            ```ruby
         | 
| 438 | 
            +
            WebMock.disable_net_connect!(
         | 
| 439 | 
            +
              allow_localhost: true,
         | 
| 440 | 
            +
              allow: ['chromedriver.storage.googleapis.com']
         | 
| 441 | 
            +
            )
         | 
| 442 | 
            +
            ```
         | 
| 443 | 
            +
             | 
| 444 | 
            +
            3. Disable WebMock for specific tests:
         | 
| 445 | 
            +
            ```js
         | 
| 446 | 
            +
            cy.app('eval', { code: 'WebMock.disable!' });
         | 
| 447 | 
            +
            // Run test
         | 
| 448 | 
            +
            cy.app('eval', { code: 'WebMock.enable!' });
         | 
| 449 | 
            +
            ```
         | 
| 450 | 
            +
             | 
| 451 | 
            +
            ### Issue: Binary/Encoded Response Issues
         | 
| 452 | 
            +
             | 
| 453 | 
            +
            **Solution:** Configure VCR to handle binary data:
         | 
| 454 | 
            +
            ```ruby
         | 
| 455 | 
            +
            c.vcr_options = {
         | 
| 456 | 
            +
              preserve_exact_body_bytes: true,
         | 
| 457 | 
            +
              decode_compressed_response: true
         | 
| 458 | 
            +
            }
         | 
| 459 | 
            +
            ```
         | 
| 460 | 
            +
             | 
| 461 | 
            +
            ### Issue: Timestamps in Recordings
         | 
| 462 | 
            +
             | 
| 463 | 
            +
            **Solution:** Filter dynamic timestamps:
         | 
| 464 | 
            +
            ```ruby
         | 
| 465 | 
            +
            VCR.configure do |c|
         | 
| 466 | 
            +
              c.before_record do |interaction|
         | 
| 467 | 
            +
                # Normalize timestamps in responses
         | 
| 468 | 
            +
                if interaction.response.headers['date']
         | 
| 469 | 
            +
                  interaction.response.headers['date'] = ['2024-01-01 00:00:00']
         | 
| 470 | 
            +
                end
         | 
| 471 | 
            +
              end
         | 
| 472 | 
            +
            end
         | 
| 473 | 
            +
            ```
         | 
| 474 | 
            +
             | 
| 475 | 
            +
            ## Best Practices
         | 
| 476 | 
            +
             | 
| 477 | 
            +
            1. **Organize cassettes by feature**: Use subdirectories for different features
         | 
| 478 | 
            +
            2. **Use descriptive names**: Make cassette names self-documenting
         | 
| 479 | 
            +
            3. **Commit cassettes to version control**: Share recordings with team
         | 
| 480 | 
            +
            4. **Periodically refresh cassettes**: Re-record to catch API changes
         | 
| 481 | 
            +
            5. **Filter sensitive data**: Never commit real API keys or tokens
         | 
| 482 | 
            +
            6. **Use appropriate record modes**:
         | 
| 483 | 
            +
               - `:once` for stable APIs
         | 
| 484 | 
            +
               - `:new_episodes` during development
         | 
| 485 | 
            +
               - `:none` for CI/production
         | 
| 486 | 
            +
            7. **Document external dependencies**: List which APIs are being mocked
         | 
| 487 | 
            +
            8. **Handle errors gracefully**: Record both success and error responses
         | 
| 488 | 
            +
             | 
| 489 | 
            +
            ## Summary
         | 
| 490 | 
            +
             | 
| 491 | 
            +
            VCR integration with cypress-playwright-on-rails provides powerful HTTP mocking capabilities. Choose between:
         | 
| 492 | 
            +
            - **Insert/Eject mode**: For explicit control over recording
         | 
| 493 | 
            +
            - **Use Cassette mode**: For automatic recording, especially with GraphQL
         | 
| 494 | 
            +
             | 
| 495 | 
            +
            Remember to:
         | 
| 496 | 
            +
            - Configure VCR appropriately for your needs
         | 
| 497 | 
            +
            - Filter sensitive data
         | 
| 498 | 
            +
            - Organize cassettes logically
         | 
| 499 | 
            +
            - Keep cassettes up to date
         | 
| @@ -16,11 +16,35 @@ module CypressOnRails | |
| 16 16 | 
             
                def self.load_e2e_helper
         | 
| 17 17 | 
             
                  e2e_helper_file = "#{configuration.install_folder}/e2e_helper.rb"
         | 
| 18 18 | 
             
                  cypress_helper_file = "#{configuration.install_folder}/cypress_helper.rb"
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                  # Check for old structure (files in framework subdirectory)
         | 
| 21 | 
            +
                  old_cypress_location = "#{configuration.install_folder}/cypress/e2e_helper.rb"
         | 
| 22 | 
            +
                  old_playwright_location = "#{configuration.install_folder}/playwright/e2e_helper.rb"
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                  # Try to load from the correct location first
         | 
| 19 25 | 
             
                  if File.exist?(e2e_helper_file)
         | 
| 20 26 | 
             
                    Kernel.require e2e_helper_file
         | 
| 21 27 | 
             
                  elsif File.exist?(cypress_helper_file)
         | 
| 22 28 | 
             
                    Kernel.require cypress_helper_file
         | 
| 23 29 | 
             
                    warn "cypress_helper.rb is deprecated, please rename the file to e2e_helper.rb"
         | 
| 30 | 
            +
                  # Fallback: load from old location if new location doesn't exist
         | 
| 31 | 
            +
                  elsif File.exist?(old_cypress_location) || File.exist?(old_playwright_location)
         | 
| 32 | 
            +
                    old_location = File.exist?(old_cypress_location) ? old_cypress_location : old_playwright_location
         | 
| 33 | 
            +
                    logger.warn "=" * 80
         | 
| 34 | 
            +
                    logger.warn "DEPRECATION WARNING: Old folder structure detected!"
         | 
| 35 | 
            +
                    logger.warn "Found e2e_helper.rb at: #{old_location}"
         | 
| 36 | 
            +
                    logger.warn "This file should be at: #{e2e_helper_file}"
         | 
| 37 | 
            +
                    logger.warn ""
         | 
| 38 | 
            +
                    logger.warn "Loading from old location for now, but this will stop working in a future version."
         | 
| 39 | 
            +
                    logger.warn "The generator now creates e2e_helper.rb and app_commands/ at the install_folder"
         | 
| 40 | 
            +
                    logger.warn "root, not inside the framework subdirectory."
         | 
| 41 | 
            +
                    logger.warn ""
         | 
| 42 | 
            +
                    logger.warn "To fix this, run: mv #{old_location} #{e2e_helper_file}"
         | 
| 43 | 
            +
                    logger.warn "Also move app_commands: mv #{File.dirname(old_location)}/app_commands #{configuration.install_folder}/"
         | 
| 44 | 
            +
                    logger.warn "See CHANGELOG.md for full migration guide."
         | 
| 45 | 
            +
                    logger.warn "=" * 80
         | 
| 46 | 
            +
                    # Load from old location as fallback
         | 
| 47 | 
            +
                    Kernel.require old_location
         | 
| 24 48 | 
             
                  else
         | 
| 25 49 | 
             
                    logger.warn "could not find #{e2e_helper_file} nor #{cypress_helper_file}"
         | 
| 26 50 | 
             
                  end
         | 
| @@ -10,6 +10,24 @@ module CypressOnRails | |
| 10 10 | 
             
                attr_accessor :before_request
         | 
| 11 11 | 
             
                attr_accessor :logger
         | 
| 12 12 | 
             
                attr_accessor :vcr_options
         | 
| 13 | 
            +
                
         | 
| 14 | 
            +
                # Server hooks for managing test lifecycle
         | 
| 15 | 
            +
                attr_accessor :before_server_start
         | 
| 16 | 
            +
                attr_accessor :after_server_start
         | 
| 17 | 
            +
                attr_accessor :after_transaction_start
         | 
| 18 | 
            +
                attr_accessor :after_state_reset
         | 
| 19 | 
            +
                attr_accessor :before_server_stop
         | 
| 20 | 
            +
                
         | 
| 21 | 
            +
                # Server configuration
         | 
| 22 | 
            +
                attr_accessor :server_host
         | 
| 23 | 
            +
                attr_accessor :server_port
         | 
| 24 | 
            +
                attr_accessor :transactional_server
         | 
| 25 | 
            +
                # HTTP path to check for server readiness (default: '/')
         | 
| 26 | 
            +
                # Can be set via CYPRESS_RAILS_READINESS_PATH environment variable
         | 
| 27 | 
            +
                attr_accessor :server_readiness_path
         | 
| 28 | 
            +
                # Timeout in seconds for individual HTTP readiness checks (default: 5)
         | 
| 29 | 
            +
                # Can be set via CYPRESS_RAILS_READINESS_TIMEOUT environment variable
         | 
| 30 | 
            +
                attr_accessor :server_readiness_timeout
         | 
| 13 31 |  | 
| 14 32 | 
             
                # Attributes for backwards compatibility
         | 
| 15 33 | 
             
                def cypress_folder
         | 
| @@ -38,6 +56,20 @@ module CypressOnRails | |
| 38 56 | 
             
                  self.before_request = -> (request) {}
         | 
| 39 57 | 
             
                  self.logger = Logger.new(STDOUT)
         | 
| 40 58 | 
             
                  self.vcr_options = {}
         | 
| 59 | 
            +
                  
         | 
| 60 | 
            +
                  # Server hooks
         | 
| 61 | 
            +
                  self.before_server_start = nil
         | 
| 62 | 
            +
                  self.after_server_start = nil
         | 
| 63 | 
            +
                  self.after_transaction_start = nil
         | 
| 64 | 
            +
                  self.after_state_reset = nil
         | 
| 65 | 
            +
                  self.before_server_stop = nil
         | 
| 66 | 
            +
                  
         | 
| 67 | 
            +
                  # Server configuration
         | 
| 68 | 
            +
                  self.server_host = ENV.fetch('CYPRESS_RAILS_HOST', 'localhost')
         | 
| 69 | 
            +
                  self.server_port = ENV.fetch('CYPRESS_RAILS_PORT', nil)
         | 
| 70 | 
            +
                  self.transactional_server = true
         | 
| 71 | 
            +
                  self.server_readiness_path = ENV.fetch('CYPRESS_RAILS_READINESS_PATH', '/')
         | 
| 72 | 
            +
                  self.server_readiness_timeout = ENV.fetch('CYPRESS_RAILS_READINESS_TIMEOUT', '5').to_i
         | 
| 41 73 | 
             
                end
         | 
| 42 74 |  | 
| 43 75 | 
             
                def tagged_logged
         | 
| @@ -3,10 +3,17 @@ require 'cypress_on_rails/configuration' | |
| 3 3 |  | 
| 4 4 | 
             
            module CypressOnRails
         | 
| 5 5 | 
             
              class Railtie < Rails::Railtie
         | 
| 6 | 
            +
                rake_tasks do
         | 
| 7 | 
            +
                  load 'tasks/cypress.rake'
         | 
| 8 | 
            +
                end
         | 
| 6 9 | 
             
                initializer :setup_cypress_middleware, after: :load_config_initializers do |app|
         | 
| 7 10 | 
             
                  if CypressOnRails.configuration.use_middleware?
         | 
| 8 11 | 
             
                    require 'cypress_on_rails/middleware'
         | 
| 9 12 | 
             
                    app.middleware.use Middleware
         | 
| 13 | 
            +
                    
         | 
| 14 | 
            +
                    # Add state reset middleware for compatibility with cypress-rails
         | 
| 15 | 
            +
                    require 'cypress_on_rails/state_reset_middleware'
         | 
| 16 | 
            +
                    app.middleware.use StateResetMiddleware
         | 
| 10 17 | 
             
                  end
         | 
| 11 18 | 
             
                  if CypressOnRails.configuration.use_vcr_middleware?
         | 
| 12 19 | 
             
                    require 'cypress_on_rails/vcr/insert_eject_middleware'
         |