From ead2b4aded40751c968f5a05a736ba3eae7b082f Mon Sep 17 00:00:00 2001 From: roost-io Date: Wed, 19 Nov 2025 09:30:55 +0000 Subject: [PATCH] UI test generated by RoostGPT Using AI Model eu.anthropic.claude-3-7-sonnet-20250219-v1:0 --- generated_tests/playwright_tests/README.md | 66 + .../scenarios/features/Scenario_3.feature | 155 ++ .../login_analysis_happy_path.feature | 90 + .../login_invalid_credentials.feature | 23 + .../features/login_scenarios.feature | 122 + .../scenarios/login_action_flow.json | 60 + .../scenarios/login_analysis_happy_path.json | 157 ++ .../scenarios/login_invalid_credentials.json | 131 ++ .../scenarios/scenarios_summary.md | 73 + .../playwright_tests/tests/.env.template | 14 + .../tests/login_analysis_happy_path.spec.js | 112 + ...nvalid_credentials.unauthenticated.spec.js | 146 ++ .../playwright_tests/tests/package-lock.json | 2065 +++++++++++++++++ .../playwright_tests/tests/package.json | 26 + .../tests/playwright.config.js | 208 ++ .../playwright_tests/tests/test_summary.md | 12 + 16 files changed, 3460 insertions(+) create mode 100644 generated_tests/playwright_tests/README.md create mode 100644 generated_tests/playwright_tests/scenarios/features/Scenario_3.feature create mode 100644 generated_tests/playwright_tests/scenarios/features/login_analysis_happy_path.feature create mode 100644 generated_tests/playwright_tests/scenarios/features/login_invalid_credentials.feature create mode 100644 generated_tests/playwright_tests/scenarios/features/login_scenarios.feature create mode 100644 generated_tests/playwright_tests/scenarios/login_action_flow.json create mode 100644 generated_tests/playwright_tests/scenarios/login_analysis_happy_path.json create mode 100644 generated_tests/playwright_tests/scenarios/login_invalid_credentials.json create mode 100644 generated_tests/playwright_tests/scenarios/scenarios_summary.md create mode 100644 generated_tests/playwright_tests/tests/.env.template create mode 100644 generated_tests/playwright_tests/tests/login_analysis_happy_path.spec.js create mode 100644 generated_tests/playwright_tests/tests/login_invalid_credentials.unauthenticated.spec.js create mode 100644 generated_tests/playwright_tests/tests/package-lock.json create mode 100644 generated_tests/playwright_tests/tests/package.json create mode 100644 generated_tests/playwright_tests/tests/playwright.config.js create mode 100644 generated_tests/playwright_tests/tests/test_summary.md diff --git a/generated_tests/playwright_tests/README.md b/generated_tests/playwright_tests/README.md new file mode 100644 index 0000000..ad28cdd --- /dev/null +++ b/generated_tests/playwright_tests/README.md @@ -0,0 +1,66 @@ +# Playwright Test Suite + +## RoostGpt Generated playwright test + +This project contains Playwright-based automated tests located in the `playwright_tests` directory. + +## 📁 Directory Structure + +```bash + playwright_tests/ + ├── scenarios/ + ├── tests/ + ├── package.json + ├── playwright.config.js +``` + +## 🧩 Prerequisites + +Make sure you have the following installed: + +- [Node.js](https://nodejs.org/) (version 18 or higher recommended) +- npm (comes with Node.js) + +## đŸ“Ļ Setup + +1. Navigate to the Playwright tests directory: + +2. Install dependencies: + + ```bash + npm install + ``` + +## 🚀 Running Tests + +To execute all Playwright tests: + +```bash +npx playwright test +``` + +You can also run a specific test file: + +```bash +npx playwright test tests/example.spec.js +``` + +> Replace `example.spec.js` with the actual test file you want to run. + +## 📂 Test Reports + +After the tests run, Playwright will generate a report. To view the report: + +```bash +npx playwright show-report +``` + +## âš™ī¸ Configuration + +Test configuration is defined in `playwright.config.js`. + +You can modify settings like test directory, timeout etc., in this file. + +## 📘 More Info + +For detailed Playwright documentation, visit: [https://playwright.dev](https://playwright.dev) diff --git a/generated_tests/playwright_tests/scenarios/features/Scenario_3.feature b/generated_tests/playwright_tests/scenarios/features/Scenario_3.feature new file mode 100644 index 0000000..1212505 --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/features/Scenario_3.feature @@ -0,0 +1,155 @@ +It seems like the provided user journey and scenario data are empty. Without specific journey information to base the feature file on, I'll create a comprehensive feature file that covers key functionality of ParaBank based on the website URL you provided. + +```gherkin +Feature: ParaBank Customer Banking Operations + As a ParaBank customer + I want to access and manage my banking accounts + So that I can perform financial transactions and monitor my account status + +@login @smoke +Scenario: Customer login with valid credentials + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + And I should see the text "Accounts Overview" + And I should see the welcome message containing "Welcome John" + And I should see at least one account listed in the accounts table + +@registration @smoke +Scenario: New customer registration + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I click the "Register" link + Then I should be on the registration page + And I should see the text "Signing up is easy!" + When I fill in the "First Name" field with "Jane" + And I fill in the "Last Name" field with "Smith" + And I fill in the "Address" field with "123 Main Street" + And I fill in the "City" field with "Boston" + And I fill in the "State" field with "MA" + And I fill in the "Zip Code" field with "02108" + And I fill in the "Phone #" field with "617-555-1234" + And I fill in the "SSN" field with "123-45-6789" + And I fill in the "Username" field with "janesmith2023" + And I fill in the "Password" field with "SecurePass123" + And I fill in the "Confirm" field with "SecurePass123" + And I click the "Register" button + Then I should see the text "Your account was created successfully. You are now logged in." + And I should be on the account overview page + And I should see the welcome message containing "Welcome Jane" + +@accountdetails +Scenario: View account details + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + When I click on the first account number in the accounts table + Then I should be on the account details page + And I should see the text "Account Details" + And I should see the account number displayed + And I should see the account type displayed + And I should see the account balance displayed + And I should see the transaction history table + +@transferfunds +Scenario: Transfer funds between accounts + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + When I click the "Transfer Funds" link in the sidebar + Then I should be on the transfer funds page + And I should see the text "Transfer Funds" + When I fill in the "Amount" field with "100" + And I select the first account from the "From account" dropdown + And I select the second account from the "To account" dropdown + And I click the "Transfer" button + Then I should see the text "Transfer Complete!" + And I should see the confirmation message containing "$100.00 has been transferred" + And I should see the "From account" number in the confirmation message + And I should see the "To account" number in the confirmation message + +@billpay +Scenario: Pay a bill to a payee + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + When I click the "Bill Pay" link in the sidebar + Then I should be on the bill payment page + And I should see the text "Bill Payment Service" + When I fill in the "Payee Name" field with "Electric Company" + And I fill in the "Address" field with "456 Power Ave" + And I fill in the "City" field with "Electri" + And I fill in the "State" field with "CA" + And I fill in the "Zip Code" field with "90210" + And I fill in the "Phone #" field with "310-555-9876" + And I fill in the "Account" field with "12345" + And I fill in the "Verify Account" field with "12345" + And I fill in the "Amount" field with "85.00" + And I select the first account from the "From account" dropdown + And I click the "Send Payment" button + Then I should see the text "Bill Payment Complete" + And I should see the confirmation message containing "$85.00 has been submitted for payment" + And I should see the payee name "Electric Company" in the confirmation message + +@requestloan +Scenario: Request a loan + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + When I click the "Request Loan" link in the sidebar + Then I should be on the loan request page + And I should see the text "Apply for a Loan" + When I fill in the "Loan Amount" field with "5000" + And I fill in the "Down Payment" field with "1000" + And I select the first account from the "From account" dropdown + And I click the "Apply Now" button + Then I should see the loan request result page + And I should see either a loan approval or denial message + And I should see the loan provider name + +@updateprofile +Scenario: Update contact information + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + When I click the "Update Contact Info" link in the sidebar + Then I should be on the update profile page + And I should see the text "Update Profile" + And I should see my current profile information populated in the form + When I fill in the "Address" field with "789 New Street" + And I fill in the "City" field with "Cambridge" + And I fill in the "State" field with "MA" + And I fill in the "Zip Code" field with "02139" + And I click the "Update Profile" button + Then I should see the text "Profile Updated" + And I should see the confirmation message containing "Your updated address and phone number have been added to the system" + +@logout +Scenario: Customer logout + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + When I fill in the "Username" field with "john" + And I fill in the "Password" field with "demo" + And I click the "Log In" button + Then I should be on the account overview page + When I click the "Log Out" link + Then I should be on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + And I should see the customer login form + And I should see the "Username" field + And I should see the "Password" field + And I should see the "Log In" button +``` + +This feature file covers key banking operations on the ParaBank website including login, registration, account viewing, transfers, bill payments, loan requests, profile updates, and logout. Each scenario follows the atomic step principle with precise element targeting and verification steps. + +If you'd like me to focus on specific user journeys or test scenarios from the ParaBank website, please provide those details and I can create more tailored feature files. \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/features/login_analysis_happy_path.feature b/generated_tests/playwright_tests/scenarios/features/login_analysis_happy_path.feature new file mode 100644 index 0000000..8c28b02 --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/features/login_analysis_happy_path.feature @@ -0,0 +1,90 @@ +# Feature: ParaBank Login and Navigation +This feature verifies user authentication and basic navigation within ParaBank online banking system + +@authentication @login @navigation +Scenario: Successful login and navigation to Transfer Funds page + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + Then the page title should contain 'ParaBank | Welcome | Online Banking' + When I fill in the username field with "x_username" + And I fill in the password field with "INVALID_PASSWORD_TEST123" + When I click the "Login" button + Then I should be navigated to the accounts overview page + And the URL should be "https://parabank.parasoft.com/parabank/overview.htm" + When I click the "Transfer Funds" link in the navigation menu + Then I should see the transfer funds form + +@login @negative +Scenario: Failed login attempt with invalid credentials + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + Then the page title should contain 'ParaBank | Welcome | Online Banking' + When I fill in the username field with "invalid_user" + And I fill in the password field with "wrong_password" + When I click the "Login" button + Then I should see an error message "The username and password could not be verified." + And I should remain on the login page + And the URL should be "https://parabank.parasoft.com/parabank/login.htm" + +@registration +Scenario: New user registration + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + When I click the "Register" link + Then I should be on the registration page + And the URL should be "https://parabank.parasoft.com/parabank/register.htm" + When I fill in the "First Name" field with "John" + And I fill in the "Last Name" field with "Smith" + And I fill in the "Address" field with "123 Test Street" + And I fill in the "City" field with "Testville" + And I fill in the "State" field with "TS" + And I fill in the "Zip Code" field with "12345" + And I fill in the "Phone #" field with "555-123-4567" + And I fill in the "SSN" field with "123-45-6789" + And I fill in the "Username" field with "johnsmith123" + And I fill in the "Password" field with "SecurePass123" + And I fill in the "Confirm" field with "SecurePass123" + When I click the "Register" button + Then I should see a welcome message containing "Your account was created successfully" + And I should be logged in automatically + +@account @funds_transfer +Scenario: Transfer funds between accounts + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + When I fill in the username field with "x_username" + And I fill in the password field with "INVALID_PASSWORD_TEST123" + And I click the "Login" button + Then I should be navigated to the accounts overview page + When I click the "Transfer Funds" link in the navigation menu + Then I should see the transfer funds form + When I select "13566" from the "fromAccountId" dropdown + And I select "13677" from the "toAccountId" dropdown + And I fill in the "amount" field with "100.00" + And I click the "Transfer" button + Then I should see a confirmation message "Transfer Complete!" + And I should see the details of the completed transfer + And the "Amount" field should display "$100.00" + And the "From account" field should display "13566" + And the "To account" field should display "13677" + +@account @balance_inquiry +Scenario: View account balance and recent transactions + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + When I fill in the username field with "x_username" + And I fill in the password field with "INVALID_PASSWORD_TEST123" + And I click the "Login" button + Then I should be navigated to the accounts overview page + When I click on account number "13566" in the accounts table + Then I should be on the account details page + And I should see the account balance displayed + And I should see a transaction history table + And the transaction table should contain columns for "Date", "Description", and "Amount" + +@security @logout +Scenario: User logout functionality + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + When I fill in the username field with "x_username" + And I fill in the password field with "INVALID_PASSWORD_TEST123" + And I click the "Login" button + Then I should be navigated to the accounts overview page + When I click the "Log Out" link + Then I should be logged out successfully + And I should be redirected to the login page + And I should see a message "Customer Login" \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/features/login_invalid_credentials.feature b/generated_tests/playwright_tests/scenarios/features/login_invalid_credentials.feature new file mode 100644 index 0000000..95a6ecd --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/features/login_invalid_credentials.feature @@ -0,0 +1,23 @@ +# Feature: ParaBank Login Authentication Validation +# As a user of ParaBank online banking +# I want to validate the login authentication system +# So that unauthorized access is prevented with proper error handling + +@authentication +@login +@negative_testing +Feature: ParaBank Login Authentication Validation + + @login_invalid_credentials + @negative_authentication + @error_handling + Scenario: Verify error message when logging in with invalid credentials + Given I am on the homepage "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + Then the page title should contain "ParaBank | Welcome | Online Banking" + + When I enter "x_username" into the username field + And I enter "INVALID_PASSWORD" into the password field + When I click the "Login" button + Then I should see an error message indicating invalid credentials + And I should still be on the login page "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + And the login form should still be visible \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/features/login_scenarios.feature b/generated_tests/playwright_tests/scenarios/features/login_scenarios.feature new file mode 100644 index 0000000..4681ab0 --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/features/login_scenarios.feature @@ -0,0 +1,122 @@ +# Feature: ParaBank User Authentication and Login + +@authentication @login @critical +Feature: ParaBank User Authentication and Login + As a ParaBank customer + I want to access my online banking account securely + So that I can manage my finances through the website + + Background: + Given I am on the homepage 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC' + And I can see the ParaBank login form on the left side panel + + @positive @happy_path + Scenario: Successful login with valid credentials + When I locate the username field in the login form + And I enter "john" in the username field + And I locate the password field in the login form + And I enter "demo" in the password field + And I click the "LOG IN" button + Then I should be redirected to the accounts overview page + And the URL should contain "overview.htm" + And I should see "Accounts Overview" heading on the page + And I should see my account information displayed + And I should see the welcome message containing my username + And I should see the logout link in the top menu + + @negative @invalid_credentials + Scenario: Failed login with invalid username + When I locate the username field in the login form + And I enter "invalid_user" in the username field + And I locate the password field in the login form + And I enter "demo" in the password field + And I click the "LOG IN" button + Then I should remain on the login page + And I should see an error message "The username and password could not be verified." + And the URL should contain "index.htm" + + @negative @invalid_credentials + Scenario: Failed login with invalid password + When I locate the username field in the login form + And I enter "john" in the username field + And I locate the password field in the login form + And I enter "wrong_password" in the password field + And I click the "LOG IN" button + Then I should remain on the login page + And I should see an error message "The username and password could not be verified." + And the URL should contain "index.htm" + + @negative @empty_fields + Scenario: Failed login with empty username + When I locate the username field in the login form + And I enter "" in the username field + And I locate the password field in the login form + And I enter "demo" in the password field + And I click the "LOG IN" button + Then I should remain on the login page + And I should see an error message "Please enter a username and password." + And the URL should contain "index.htm" + + @negative @empty_fields + Scenario: Failed login with empty password + When I locate the username field in the login form + And I enter "john" in the username field + And I locate the password field in the login form + And I enter "" in the password field + And I click the "LOG IN" button + Then I should remain on the login page + And I should see an error message "Please enter a username and password." + And the URL should contain "index.htm" + + @functional @navigation + Scenario: Access account services after successful login + When I locate the username field in the login form + And I enter "john" in the username field + And I locate the password field in the login form + And I enter "demo" in the password field + And I click the "LOG IN" button + Then I should be redirected to the accounts overview page + And the URL should contain "overview.htm" + When I click on the "Transfer Funds" link in the account services menu + Then I should be redirected to the transfer funds page + And the URL should contain "transfer.htm" + And I should see the transfer funds form + + @functional @logout + Scenario: Successfully logout after login + When I locate the username field in the login form + And I enter "john" in the username field + And I locate the password field in the login form + And I enter "demo" in the password field + And I click the "LOG IN" button + Then I should be redirected to the accounts overview page + And the URL should contain "overview.htm" + When I click on the "Log Out" link in the top menu + Then I should be logged out and redirected to the login page + And the URL should contain "index.htm" + And I should see the login form again + And I should see a message indicating I have logged out + + @accessibility @forgotten_login + Scenario: Access forgotten login info page + When I locate the "Forgot login info?" link below the login form + And I click on the "Forgot login info?" link + Then I should be redirected to the lookup form page + And the URL should contain "lookup.htm" + And I should see the customer lookup form + And I should see fields to enter my personal information + + @security @session_timeout + Scenario: Session timeout after period of inactivity + When I locate the username field in the login form + And I enter "john" in the username field + And I locate the password field in the login form + And I enter "demo" in the password field + And I click the "LOG IN" button + Then I should be redirected to the accounts overview page + And the URL should contain "overview.htm" + When the session times out due to inactivity + And I attempt to access a protected page + Then I should be redirected back to the login page + And I should see a message indicating my session has expired + And the URL should contain "index.htm" \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/login_action_flow.json b/generated_tests/playwright_tests/scenarios/login_action_flow.json new file mode 100644 index 0000000..e647c05 --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/login_action_flow.json @@ -0,0 +1,60 @@ +{ + "metadata": { + "total_steps": 4, + "sites_involved": [ + "parabank.parasoft.com" + ], + "flow_type": "standard_login", + "generated_timestamp": "2025-11-19T09:10:28.697879" + }, + "action_flow": [ + { + "step": 1, + "site": "parabank.parasoft.com", + "action": "input_text", + "element": " ''", + "selector": "page.locator(\".input\").nth(0)", + "description": "Enter data into '' on parabank.parasoft.com", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + }, + { + "step": 2, + "site": "parabank.parasoft.com", + "action": "insert", + "element": " ''", + "selector": "page.locator(\".input\").nth(1)", + "description": "Fill '' with credentials on parabank.parasoft.com", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + }, + { + "step": 3, + "site": "parabank.parasoft.com", + "action": "click", + "element": " ''", + "selector": "page.locator(\".button\").nth(1)", + "description": "Click '' on parabank.parasoft.com", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + }, + { + "step": 4, + "site": "parabank.parasoft.com", + "action": "click_element_by_index", + "element": " 'Transfer Funds'", + "selector": "page.getByRole(\"link\", { name: \"Transfer Funds\" })", + "description": "click_element_by_index on 'Transfer Funds' at parabank.parasoft.com", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC" + } + ], + "site_grouping": { + "parabank.parasoft.com": { + "purpose": "Initial authentication", + "steps": [ + 1, + 2, + 3, + 4 + ], + "total_interactions": 4 + } + } +} \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/login_analysis_happy_path.json b/generated_tests/playwright_tests/scenarios/login_analysis_happy_path.json new file mode 100644 index 0000000..b42cf01 --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/login_analysis_happy_path.json @@ -0,0 +1,157 @@ +{ + "site_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "site_type": "login_workflow", + "site_category": "authentication", + "test_name": "login_analysis_happy_path", + "login_mechanisms": [ + "form_login" + ], + "user_journeys": [], + "proposed_scenarios": [ + { + "scenario_name": "login_analysis_happy_path", + "test_name": "login_analysis_happy_path", + "test_type": "e2e_authentication_workflow", + "auth_required": false, + "detailed_steps": [ + { + "step_number": 1, + "action": "navigate_to", + "description": "Navigate to ParaBank login page", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "verification_point": "Page title contains 'ParaBank | Welcome | Online Banking'", + "confidence": 95, + "based_on_interaction": true + }, + { + "step_number": 2, + "action": "input_text", + "description": "Enter username into the username field", + "selector": "page.locator(\".input\").nth(0)", + "all_selectors": [ + { + "selector": "page.locator(\".input\").nth(0)", + "confidence": 70, + "strategy": "css", + "is_unique": true, + "match_count": 1 + } + ], + "value": "x_username", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "confidence": 90, + "based_on_interaction": true + }, + { + "step_number": 3, + "action": "input_text", + "description": "Enter password into the password field", + "selector": "page.locator(\".input\").nth(1)", + "all_selectors": [ + { + "selector": "page.locator(\".input\").nth(1)", + "confidence": 70, + "strategy": "css", + "is_unique": true, + "match_count": 1 + } + ], + "value": "INVALID_PASSWORD_TEST123", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "confidence": 90, + "based_on_interaction": true + }, + { + "step_number": 4, + "action": "click", + "description": "Click the Login button to submit credentials", + "selector": "page.locator(\".button\").nth(1)", + "all_selectors": [ + { + "selector": "page.locator(\".button\").nth(1)", + "confidence": 70, + "strategy": "css", + "is_unique": true, + "match_count": 1 + } + ], + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "confidence": 90, + "based_on_interaction": true + }, + { + "step_number": 5, + "action": "verify_navigation", + "description": "Verify successful login by checking URL change", + "expected_page_url": "https://parabank.parasoft.com/parabank/overview.htm", + "verification_point": "URL changed to accounts overview page after login", + "confidence": 95, + "based_on_interaction": true + }, + { + "step_number": 6, + "action": "click", + "description": "Click on Transfer Funds link in navigation menu", + "selector": "page.getByRole(\"link\", { name: \"Transfer Funds\" })", + "all_selectors": [ + { + "selector": "page.getByRole(\"link\", { name: \"Transfer Funds\" })", + "confidence": 90, + "strategy": "role", + "is_unique": true, + "match_count": 1 + } + ], + "expected_page_url": "https://parabank.parasoft.com/parabank/overview.htm", + "confidence": 90, + "based_on_interaction": true + } + ] + } + ], + "login_completion_state": "successful", + "analysis_timestamp": "2025-11-19T09:10:59.056448", + "confidence_score": 1.0, + "flow_type": "happy_path_only", + "excluded_scenarios": [ + "login_invalid_credentials" + ], + "scenario_count": 1, + "focus": "successful_authentication_only", + "generated_from": "login_analysis.json", + "generation_timestamp": "2025-11-19T09:10:59.059928", + "captured_selectors": [ + { + "selector": "page.locator(\".input\").nth(0)", + "confidence": 70, + "action": "input_text", + "element": " ''", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:08:18.354733" + }, + { + "selector": "page.locator(\".input\").nth(1)", + "confidence": 70, + "action": "insert", + "element": " ''", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:08:49.871578" + }, + { + "selector": "page.locator(\".button\").nth(1)", + "confidence": 70, + "action": "click", + "element": " ''", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:09:21.085526" + }, + { + "selector": "page.getByRole(\"link\", { name: \"Transfer Funds\" })", + "confidence": 90, + "action": "click_element_by_index", + "element": " 'Transfer Funds'", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:09:43.968293" + } + ] +} \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/login_invalid_credentials.json b/generated_tests/playwright_tests/scenarios/login_invalid_credentials.json new file mode 100644 index 0000000..0e9fe0d --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/login_invalid_credentials.json @@ -0,0 +1,131 @@ +{ + "site_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "site_type": "login_workflow", + "site_category": "authentication", + "test_name": "login_invalid_credentials", + "login_mechanisms": [ + "form_login" + ], + "user_journeys": [], + "proposed_scenarios": [ + { + "scenario_name": "login_invalid_credentials", + "test_name": "login_invalid_credentials", + "test_type": "negative_authentication_invalid_credentials", + "auth_required": false, + "detailed_steps": [ + { + "step_number": 1, + "action": "navigate_to", + "description": "Navigate to ParaBank login page", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "verification_point": "Page title contains 'ParaBank | Welcome | Online Banking'", + "confidence": 95, + "based_on_interaction": true + }, + { + "step_number": 2, + "action": "input_text", + "description": "Enter username into the username field", + "selector": "page.locator(\".input\").nth(0)", + "all_selectors": [ + { + "selector": "page.locator(\".input\").nth(0)", + "confidence": 70, + "strategy": "css", + "is_unique": true, + "match_count": 1 + } + ], + "value": "x_username", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "confidence": 90, + "based_on_interaction": true + }, + { + "step_number": 3, + "action": "input_text", + "description": "Enter invalid password into the password field", + "selector": "page.locator(\".input\").nth(1)", + "all_selectors": [ + { + "selector": "page.locator(\".input\").nth(1)", + "confidence": 70, + "strategy": "css", + "is_unique": true, + "match_count": 1 + } + ], + "value": "INVALID_PASSWORD", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "confidence": 90, + "based_on_interaction": true + }, + { + "step_number": 4, + "action": "click", + "description": "Click the Login button to submit invalid credentials", + "selector": "page.locator(\".button\").nth(1)", + "all_selectors": [ + { + "selector": "page.locator(\".button\").nth(1)", + "confidence": 70, + "strategy": "css", + "is_unique": true, + "match_count": 1 + } + ], + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "confidence": 90, + "based_on_interaction": true + }, + { + "step_number": 5, + "action": "verify_error_message", + "description": "Verify error message for invalid credentials", + "expected_page_url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "verification_point": "Error message indicating invalid credentials is displayed", + "confidence": 95, + "based_on_interaction": true + } + ] + } + ], + "login_completion_state": "successful", + "analysis_timestamp": "2025-11-19T09:10:59.056448", + "confidence_score": 0.5, + "captured_selectors": [ + { + "selector": "page.locator(\".input\").nth(0)", + "confidence": 70, + "action": "input_text", + "element": " ''", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:08:18.354733" + }, + { + "selector": "page.locator(\".input\").nth(1)", + "confidence": 70, + "action": "insert", + "element": " ''", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:08:49.871578" + }, + { + "selector": "page.locator(\".button\").nth(1)", + "confidence": 70, + "action": "click", + "element": " ''", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:09:21.085526" + }, + { + "selector": "page.getByRole(\"link\", { name: \"Transfer Funds\" })", + "confidence": 90, + "action": "click_element_by_index", + "element": " 'Transfer Funds'", + "url": "https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC", + "timestamp": "2025-11-19T09:09:43.968293" + } + ] +} \ No newline at end of file diff --git a/generated_tests/playwright_tests/scenarios/scenarios_summary.md b/generated_tests/playwright_tests/scenarios/scenarios_summary.md new file mode 100644 index 0000000..b63a9a7 --- /dev/null +++ b/generated_tests/playwright_tests/scenarios/scenarios_summary.md @@ -0,0 +1,73 @@ +# Generated Test Scenarios Summary + +## Overview + +- **Total Scenarios**: 2 +- **Application Base URL**: https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC +- **Generated On**: 2025-11-19 09:29:38 + +## Scenarios + +### 1. Login with Invalid Credentials +_Tests the system's behavior when users attempt to log in with invalid authentication credentials. Verifies that the system properly handles incorrect login attempts and displays appropriate error messages._ + +**Complexity**: Low | **Priority**: High | **Risk Level**: Medium +**Tags**: authentication, negative-testing, login-validation, security +**Est. Execution Time**: 25 seconds | **Flakiness Potential**: Medium + +**Type**: negative_authentication_invalid_credentials +**Pages Involved:** +- https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC + +#### Steps: +- Navigate to ParaBank login page +- Enter username into the username field +- Enter invalid password into the password field +- Click the Login button to submit invalid credentials +- Verify error message for invalid credentials + +#### Selectors Used: +- **Type**: input, **Text**: '', **Selector**: `page.locator(".input").nth(0)`, **Action**: input_text +- **Type**: input, **Text**: '', **Selector**: `page.locator(".input").nth(1)`, **Action**: insert +- **Type**: input, **Text**: '', **Selector**: `page.locator(".button").nth(1)`, **Action**: click + +#### Expected Results: +- Page loads successfully with title containing "ParaBank | Welcome | Online Banking" +- Error message indicating invalid credentials is displayed +- User remains on the login page after failed attempt + +--- + +### 2. Login Happy Path and Navigation +_Validates the complete end-to-end login workflow with valid credentials, followed by successful navigation to the Transfer Funds section. This test ensures core authentication functionality works correctly for authorized users._ + +**Complexity**: Medium | **Priority**: High | **Risk Level**: High +**Tags**: authentication, e2e, login, navigation, core-functionality +**Est. Execution Time**: 30 seconds | **Flakiness Potential**: Medium + +**Type**: e2e_authentication_workflow +**Pages Involved:** +- https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC +- https://parabank.parasoft.com/parabank/overview.htm + +#### Steps: +- Navigate to ParaBank login page +- Enter username into the username field +- Enter password into the password field +- Click the Login button to submit credentials +- Verify successful login by checking URL change +- Click on Transfer Funds link in navigation menu + +#### Selectors Used: +- **Type**: input, **Text**: '', **Selector**: `page.locator(".input").nth(0)`, **Action**: input_text +- **Type**: input, **Text**: '', **Selector**: `page.locator(".input").nth(1)`, **Action**: insert +- **Type**: input, **Text**: '', **Selector**: `page.locator(".button").nth(1)`, **Action**: click +- **Type**: a, **Text**: 'Transfer Funds', **Selector**: `page.getByRole("link", { name: "Transfer Funds" })`, **Action**: click_element_by_index + +#### Expected Results: +- Page loads successfully with title containing "ParaBank | Welcome | Online Banking" +- URL changes to accounts overview page after successful login +- User can navigate to Transfer Funds functionality +- Transfer Funds page loads successfully + +--- \ No newline at end of file diff --git a/generated_tests/playwright_tests/tests/.env.template b/generated_tests/playwright_tests/tests/.env.template new file mode 100644 index 0000000..ba8bc8a --- /dev/null +++ b/generated_tests/playwright_tests/tests/.env.template @@ -0,0 +1,14 @@ +# Environment Configuration Template +# This file contains safe placeholder values for version control +# Actual credentials will be injected at runtime from your source env file + +# Application Configuration +BASE_URL=https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC +CI= +COOKIE_GRACE_PERIOD_MINUTES= + +# Authentication & Credentials (POPULATED AT RUNTIME) +# These values are automatically loaded during test execution +# DO NOT enter real credentials here +PASSWORD=******** +USERNAME=your_username \ No newline at end of file diff --git a/generated_tests/playwright_tests/tests/login_analysis_happy_path.spec.js b/generated_tests/playwright_tests/tests/login_analysis_happy_path.spec.js new file mode 100644 index 0000000..d5d3f44 --- /dev/null +++ b/generated_tests/playwright_tests/tests/login_analysis_happy_path.spec.js @@ -0,0 +1,112 @@ +import 'dotenv/config'; +import { test, expect } from '@playwright/test'; +import fs from 'fs'; +import path from 'path'; + +const BASE_URL = process.env.BASE_URL || 'https://parabank.parasoft.com/parabank'; +const USERNAME = process.env.USERNAME || 'john'; // Default value for demo purposes +const PASSWORD = process.env.PASSWORD || 'demo'; // Default value for demo purposes + +// Capture accessibility tree on failure for intelligent iteration +test.afterEach(async ({ page }, testInfo) => { + if (testInfo.status !== 'passed') { + try { + // Wait for any animations/modals to fully render + await page.waitForTimeout(1000); + + const accessibilityTree = await page.accessibility.snapshot(); + + const domSnapshot = await page.evaluate(() => { + return Array.from(document.querySelectorAll('*')) + .filter(el => { + // Only visible elements + const rect = el.getBoundingClientRect(); + const style = window.getComputedStyle(el); + return rect.width > 0 && rect.height > 0 && + style.display !== 'none' && + style.visibility !== 'hidden' && + parseFloat(style.opacity) > 0.05; + }) + .map(el => { + const style = window.getComputedStyle(el); + const rect = el.getBoundingClientRect(); + + return { + tag: el.tagName.toLowerCase(), + id: el.id || null, + classes: el.className || null, + text: (el.innerText || el.textContent || '').trim().substring(0, 100), + value: el.value || null, + role: el.getAttribute('role') || null, + ariaLabel: el.getAttribute('aria-label') || null, + type: el.type || null, + href: el.href || null, + cursor: style.cursor, + display: style.display, + hasOnclick: !!el.onclick || el.hasAttribute('onclick'), + parent: { + tag: el.parentElement?.tagName?.toLowerCase(), + classes: el.parentElement?.className || null + }, + position: { + x: Math.round(rect.x), + y: Math.round(rect.y), + width: Math.round(rect.width), + height: Math.round(rect.height) + } + }; + }); + }); + + const fileName = path.basename(testInfo.file) + .replace('.authenticated.spec.js', '') + .replace('.unauthenticated.spec.js', '') + .replace('.spec.js', ''); + const stateFile = path.join(__dirname, `../.accessibility_state_${fileName}.json`); + fs.writeFileSync(stateFile, JSON.stringify({ + accessibility_tree: accessibilityTree, + dom_snapshot: domSnapshot, + element_count: domSnapshot.length, + url: page.url() + }, null, 2)); + } catch (e) {} + } +}); + +test('Login to ParaBank and navigate to Transfer Funds', async ({ page }) => { + try { + // Step 1: Navigate to ParaBank login page + await page.goto(`${BASE_URL}/index.htm?ConnType=JDBC`); + await expect(page).toHaveTitle(/ParaBank \| Welcome \| Online Banking/); + + // Step 2: Enter username into the username field + // Captured selectors: + // 1. page.locator(".input").nth(0) (confidence: 70%, strategy: css, unique: true) + await page.locator(".input").nth(0).fill(USERNAME); + + // Step 3: Enter password into the password field + // Captured selectors: + // 1. page.locator(".input").nth(1) (confidence: 70%, strategy: css, unique: true) + await page.locator(".input").nth(1).fill(PASSWORD); + + // Step 4: Click the Login button to submit credentials + // Captured selectors: + // 1. page.locator(".button").nth(1) (confidence: 70%, strategy: css, unique: true) + await page.locator(".button").nth(1).click(); + + // Step 5: Verify successful login by checking URL change + await expect(page).toHaveURL(`${BASE_URL}/overview.htm`); + + // Step 6: Click on Transfer Funds link in navigation menu + // Captured selectors: + // 1. page.getByRole("link", { name: "Transfer Funds" }) (confidence: 90%, strategy: role, unique: true) + await page.getByRole("link", { name: "Transfer Funds" }).click(); + + // Verify navigation to Transfer Funds page + await expect(page).toHaveURL(/transfer.htm/); + + } catch (error) { + console.error('Test failed:', error); + throw error; + } +}); \ No newline at end of file diff --git a/generated_tests/playwright_tests/tests/login_invalid_credentials.unauthenticated.spec.js b/generated_tests/playwright_tests/tests/login_invalid_credentials.unauthenticated.spec.js new file mode 100644 index 0000000..32ff0d6 --- /dev/null +++ b/generated_tests/playwright_tests/tests/login_invalid_credentials.unauthenticated.spec.js @@ -0,0 +1,146 @@ +import 'dotenv/config'; +import { test, expect } from '@playwright/test'; +import fs from 'fs'; +import path from 'path'; + +const BASE_URL = process.env.BASE_URL || 'https://parabank.parasoft.com/parabank/index.htm?ConnType=JDBC'; +const INVALID_USERNAME = 'x_username'; +const INVALID_PASSWORD = 'INVALID_PASSWORD'; + +// Capture accessibility tree on failure for intelligent iteration +test.afterEach(async ({ page }, testInfo) => { + if (testInfo.status !== 'passed') { + try { + // Wait for any animations/modals to fully render + await page.waitForTimeout(1000); + + const accessibilityTree = await page.accessibility.snapshot(); + + // Capture DOM snapshot + const domSnapshot = await page.evaluate(() => { + return Array.from(document.querySelectorAll('*')) + .filter(el => { + // Only visible elements + const rect = el.getBoundingClientRect(); + const style = window.getComputedStyle(el); + return rect.width > 0 && rect.height > 0 && + style.display !== 'none' && + style.visibility !== 'hidden' && + parseFloat(style.opacity) > 0.05; + }) + .map(el => { + const style = window.getComputedStyle(el); + const rect = el.getBoundingClientRect(); + + return { + tag: el.tagName.toLowerCase(), + id: el.id || null, + classes: el.className || null, + text: (el.innerText || el.textContent || '').trim().substring(0, 100), + value: el.value || null, + role: el.getAttribute('role') || null, + ariaLabel: el.getAttribute('aria-label') || null, + type: el.type || null, + href: el.href || null, + cursor: style.cursor, + display: style.display, + hasOnclick: !!el.onclick || el.hasAttribute('onclick'), + parent: { + tag: el.parentElement?.tagName?.toLowerCase(), + classes: el.parentElement?.className || null + }, + position: { + x: Math.round(rect.x), + y: Math.round(rect.y), + width: Math.round(rect.width), + height: Math.round(rect.height) + } + }; + }); + }); + + // Remove .spec.js and optional .authenticated/.unauthenticated prefixes + const fileName = path.basename(testInfo.file) + .replace('.authenticated.spec.js', '') + .replace('.unauthenticated.spec.js', '') + .replace('.spec.js', ''); + const stateFile = path.join(__dirname, `../.accessibility_state_${fileName}.json`); + fs.writeFileSync(stateFile, JSON.stringify({ + accessibility_tree: accessibilityTree, + dom_snapshot: domSnapshot, + element_count: domSnapshot.length, + url: page.url() + }, null, 2)); + } catch (e) {} + } +}); + +test('Login with Invalid Credentials', async ({ page }) => { + try { + // Step 1: Navigate to ParaBank login page + await page.goto(BASE_URL); + await expect(page).toHaveURL(BASE_URL); + + // Verify page title + await expect(page).toHaveTitle(/ParaBank.*Online Banking/); + + // Step 2: Enter invalid username into the username field + // Captured selectors: + // 1. page.locator(".input").nth(0) (confidence: 70%, strategy: css, unique: true) + await page.locator(".input").nth(0).fill(INVALID_USERNAME); + + // Step 3: Enter invalid password into the password field + // Captured selectors: + // 1. page.locator(".input").nth(1) (confidence: 70%, strategy: css, unique: true) + await page.locator(".input").nth(1).fill(INVALID_PASSWORD); + + // Step 4: Click the Login button to submit invalid credentials + // Captured selectors: + // 1. page.locator(".button").nth(1) (confidence: 70%, strategy: css, unique: true) + await page.locator(".button").nth(1).click(); + + // Step 5: Verify error message for invalid credentials is displayed + // Wait for error message to appear + await page.waitForTimeout(1000); + + // Look for error message using various common error selectors + const errorSelectors = [ + '.error', + '[role="alert"]', + '.alert', + '.message-error', + 'p.error', + 'div.error' + ]; + + // Try each selector until we find an error message + let errorFound = false; + for (const selector of errorSelectors) { + const errorElement = page.locator(selector).first(); + if (await errorElement.count() > 0 && await errorElement.isVisible()) { + // Verify error message contains text about invalid credentials + const errorText = await errorElement.textContent(); + console.log(`Found error message: ${errorText}`); + + // Check if error message contains typical invalid credential phrases + const hasInvalidCredentialsMessage = /invalid|incorrect|wrong|failed|not recognized|authentication|login|password/i.test(errorText); + expect(hasInvalidCredentialsMessage).toBeTruthy(); + errorFound = true; + break; + } + } + + // If we didn't find an error using common selectors, look for any text that might indicate an error + if (!errorFound) { + // Look for text mentioning invalid credentials + const bodyText = await page.textContent('body'); + const hasErrorInBody = /invalid|incorrect|wrong|failed|not recognized|authentication|login|password/i.test(bodyText); + expect(hasErrorInBody).toBeTruthy(); + } + + // Verify we're still on the login page (didn't navigate to account page) + await expect(page).toHaveURL(BASE_URL); + } catch (error) { + throw error; + } +}); \ No newline at end of file diff --git a/generated_tests/playwright_tests/tests/package-lock.json b/generated_tests/playwright_tests/tests/package-lock.json new file mode 100644 index 0000000..a994080 --- /dev/null +++ b/generated_tests/playwright_tests/tests/package-lock.json @@ -0,0 +1,2065 @@ +{ + "name": "production-playwright-tests", + "version": "1.0.0", + "lockfileVersion": 2, + "requires": true, + "packages": { + "": { + "name": "production-playwright-tests", + "version": "1.0.0", + "license": "ISC", + "devDependencies": { + "@playwright/test": "1.55.0", + "dotenv": "^16.4.5", + "eslint": "^8.57.0" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz", + "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", + "dev": true, + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/js": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", + "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", + "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", + "deprecated": "Use @eslint/config-array instead", + "dev": true, + "dependencies": { + "@humanwhocodes/object-schema": "^2.0.3", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", + "deprecated": "Use @eslint/object-schema instead", + "dev": true + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@playwright/test": { + "version": "1.55.0", + "resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.55.0.tgz", + "integrity": "sha512-04IXzPwHrW69XusN/SIdDdKZBzMfOT9UNT/YiJit/xpy2VuAoB8NHc8Aplb96zsWDddLnbkPL3TsmrS04ZU2xQ==", + "dev": true, + "dependencies": { + "playwright": "1.55.0" + }, + "bin": { + "playwright": "cli.js" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@ungap/structured-clone": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.3.0.tgz", + "integrity": "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g==", + "dev": true + }, + "node_modules/acorn": { + "version": "8.15.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", + "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/dotenv": { + "version": "16.6.1", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.6.1.tgz", + "integrity": "sha512-uBq4egWHTcTt33a72vpSG0z3HnPuIl6NqYcTrKEg2azoEyl2hpW0zqlxysq2pK9HlDIHyHyakeYaYnSAwd8bow==", + "dev": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.1.tgz", + "integrity": "sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==", + "deprecated": "This version is no longer supported. Please see https://eslint.org/version-support for other options.", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.6.1", + "@eslint/eslintrc": "^2.1.4", + "@eslint/js": "8.57.1", + "@humanwhocodes/config-array": "^0.13.0", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.2", + "eslint-visitor-keys": "^3.4.3", + "espree": "^9.6.1", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "node_modules/fastq": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", + "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", + "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "dev": true + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/js-yaml": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", + "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", + "dev": true, + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/playwright": { + "version": "1.55.0", + "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.55.0.tgz", + "integrity": "sha512-sdCWStblvV1YU909Xqx0DhOjPZE4/5lJsIS84IfN9dAZfcl/CIZ5O8l3o0j7hPMjDvqoTF8ZUcc+i/GL5erstA==", + "dev": true, + "dependencies": { + "playwright-core": "1.55.0" + }, + "bin": { + "playwright": "cli.js" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "fsevents": "2.3.2" + } + }, + "node_modules/playwright-core": { + "version": "1.55.0", + "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.55.0.tgz", + "integrity": "sha512-GvZs4vU3U5ro2nZpeiwyb0zuFaqb9sUiAJuyrWpcGouD8y9/HLgGbNRjIph7zU9D3hnPaisMl9zG9CgFi/biIg==", + "dev": true, + "bin": { + "playwright-core": "cli.js" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + }, + "dependencies": { + "@eslint-community/eslint-utils": { + "version": "4.9.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz", + "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^3.4.3" + } + }, + "@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", + "dev": true + }, + "@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "requires": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + } + }, + "@eslint/js": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", + "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", + "dev": true + }, + "@humanwhocodes/config-array": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", + "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", + "dev": true, + "requires": { + "@humanwhocodes/object-schema": "^2.0.3", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + } + }, + "@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true + }, + "@humanwhocodes/object-schema": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", + "dev": true + }, + "@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + } + }, + "@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true + }, + "@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "requires": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + } + }, + "@playwright/test": { + "version": "1.55.0", + "resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.55.0.tgz", + "integrity": "sha512-04IXzPwHrW69XusN/SIdDdKZBzMfOT9UNT/YiJit/xpy2VuAoB8NHc8Aplb96zsWDddLnbkPL3TsmrS04ZU2xQ==", + "dev": true, + "requires": { + "playwright": "1.55.0" + } + }, + "@ungap/structured-clone": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.3.0.tgz", + "integrity": "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g==", + "dev": true + }, + "acorn": { + "version": "8.15.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", + "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "dev": true + }, + "acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "requires": {} + }, + "ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "requires": { + "ms": "^2.1.3" + } + }, + "deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "dotenv": { + "version": "16.6.1", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.6.1.tgz", + "integrity": "sha512-uBq4egWHTcTt33a72vpSG0z3HnPuIl6NqYcTrKEg2azoEyl2hpW0zqlxysq2pK9HlDIHyHyakeYaYnSAwd8bow==", + "dev": true + }, + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + }, + "eslint": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.1.tgz", + "integrity": "sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==", + "dev": true, + "requires": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.6.1", + "@eslint/eslintrc": "^2.1.4", + "@eslint/js": "8.57.1", + "@humanwhocodes/config-array": "^0.13.0", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.2", + "eslint-visitor-keys": "^3.4.3", + "espree": "^9.6.1", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + } + }, + "eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + } + }, + "eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true + }, + "espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "requires": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + } + }, + "esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + } + }, + "esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "requires": { + "estraverse": "^5.2.0" + } + }, + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "fastq": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", + "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", + "dev": true, + "requires": { + "reusify": "^1.0.4" + } + }, + "file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "requires": { + "flat-cache": "^3.0.4" + } + }, + "find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "requires": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + } + }, + "flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "requires": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + } + }, + "flatted": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", + "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "dev": true + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "optional": true + }, + "glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "requires": { + "is-glob": "^4.0.3" + } + }, + "globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + }, + "graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true + }, + "import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true + }, + "is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "js-yaml": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", + "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", + "dev": true, + "requires": { + "argparse": "^2.0.1" + } + }, + "json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true + }, + "json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "requires": { + "json-buffer": "3.0.1" + } + }, + "levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + } + }, + "locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "requires": { + "p-locate": "^5.0.0" + } + }, + "lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "requires": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + } + }, + "p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "requires": { + "yocto-queue": "^0.1.0" + } + }, + "p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "requires": { + "p-limit": "^3.0.2" + } + }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, + "path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "playwright": { + "version": "1.55.0", + "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.55.0.tgz", + "integrity": "sha512-sdCWStblvV1YU909Xqx0DhOjPZE4/5lJsIS84IfN9dAZfcl/CIZ5O8l3o0j7hPMjDvqoTF8ZUcc+i/GL5erstA==", + "dev": true, + "requires": { + "fsevents": "2.3.2", + "playwright-core": "1.55.0" + } + }, + "playwright-core": { + "version": "1.55.0", + "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.55.0.tgz", + "integrity": "sha512-GvZs4vU3U5ro2nZpeiwyb0zuFaqb9sUiAJuyrWpcGouD8y9/HLgGbNRjIph7zU9D3hnPaisMl9zG9CgFi/biIg==", + "dev": true + }, + "prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true + }, + "punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true + }, + "queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true + }, + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + }, + "reusify": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "requires": { + "queue-microtask": "^1.2.2" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.1" + } + }, + "strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1" + } + }, + "type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true + }, + "uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "requires": { + "punycode": "^2.1.0" + } + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true + } + } +} diff --git a/generated_tests/playwright_tests/tests/package.json b/generated_tests/playwright_tests/tests/package.json new file mode 100644 index 0000000..a063681 --- /dev/null +++ b/generated_tests/playwright_tests/tests/package.json @@ -0,0 +1,26 @@ + + { + "name": "production-playwright-tests", + "version": "1.0.0", + "description": "A robust suite of end-to-end tests using Playwright.", + "scripts": { + "test": "npx playwright test", + "test:headed": "npx playwright test --headed", + "report": "npx playwright show-report", + "lint": "eslint . --ext .js,.ts" + }, + "keywords": [ + "playwright", + "testing", + "e2e", + "automation" + ], + "author": "", + "license": "ISC", + "devDependencies": { + "@playwright/test": "1.55.0", + "eslint": "^8.57.0", + "dotenv": "^16.4.5" + } + } + \ No newline at end of file diff --git a/generated_tests/playwright_tests/tests/playwright.config.js b/generated_tests/playwright_tests/tests/playwright.config.js new file mode 100644 index 0000000..5286b22 --- /dev/null +++ b/generated_tests/playwright_tests/tests/playwright.config.js @@ -0,0 +1,208 @@ + +// @ts-check +import { defineConfig, devices } from "@playwright/test"; +import * as fs from "fs"; +import * as path from "path"; + +/** + * @see https://playwright.dev/docs/test-configuration + * + * This configuration automatically reuses authentication session with smart cookie validation: + * 1. Checks if storage state exists AND cookies are still valid + * 2. If cookies expired → login test runs first to regenerate fresh cookies + * 3. If cookies valid → authenticated tests run directly (no login needed!) + * 4. Login tests always run with fresh browser + */ + +// Check if storage state file exists +const storageStatePath = path.join(__dirname, '.auth/storage-state.json'); +const storageStateExists = fs.existsSync(storageStatePath); + +/** + * Validates if authentication cookies in storage state are still valid + * + * Smart validation that only checks auth-critical cookies (not analytics/tracking): + * - Configurable via AUTH_COOKIE_NAMES env var (comma-separated patterns) + * - Default patterns: session, auth, token, sid, JSESSIONID, etc. + * - Grace period: Triggers re-login N minutes before expiration (COOKIE_GRACE_PERIOD_MINUTES) + * - Ignores non-auth cookies to prevent unnecessary re-logins + * + * @returns {boolean} true if auth cookies are valid, false if expired/missing + */ +function isStorageStateValid() { + if (!storageStateExists) { + console.log('âš ī¸ Storage state file does not exist - login required'); + return false; + } + + try { + const storageState = JSON.parse(fs.readFileSync(storageStatePath, 'utf-8')); + + // Check if cookies array exists + if (!storageState.cookies || !Array.isArray(storageState.cookies)) { + console.log('âš ī¸ No cookies found in storage state - login required'); + return false; + } + + // Configure authentication cookie patterns + // Override via AUTH_COOKIE_NAMES environment variable (comma-separated) + const authCookiePatterns = (process.env.AUTH_COOKIE_NAMES || + 'session,auth,token,sid,oauth,JSESSIONID,connect.sid,access_token,refresh_token,jwt,bearer,_session').split(','); + + // Configure grace period (minutes before expiration to trigger re-login) + const gracePeriodMinutes = parseInt(process.env.COOKIE_GRACE_PERIOD_MINUTES || '1'); + const gracePeriodSeconds = gracePeriodMinutes * 60; + const now = Date.now() / 1000; // Convert to seconds + + // Filter to only auth-critical cookies + const authCookies = storageState.cookies.filter((cookie) => + authCookiePatterns.some((pattern) => + cookie.name.toLowerCase().includes(pattern.toLowerCase()) + ) + ); + + if (authCookies.length === 0) { + console.log('â„šī¸ No authentication cookies found in storage state'); + console.log('🔄 Login test will run to establish session'); + return false; + } + + // Check auth cookies for expiration (with grace period) + // @ts-ignore - Cookie type from storage state + const expiredAuthCookies = authCookies.filter((cookie) => + cookie.expires && + cookie.expires !== -1 && + cookie.expires < (now + gracePeriodSeconds) + ); + + if (expiredAuthCookies.length > 0) { + // @ts-ignore - Cookie type + console.log(`âš ī¸ Found ${expiredAuthCookies.length} expired/expiring auth cookie(s): ${expiredAuthCookies.map((c) => c.name).join(', ')}`); + console.log(` (Grace period: ${gracePeriodMinutes} minutes before expiration)`); + console.log('🔄 Login test will run first to regenerate fresh cookies'); + return false; + } + + console.log(`✅ All ${authCookies.length} authentication cookie(s) are valid - reusing existing session`); + // @ts-ignore - Cookie type + console.log(` Auth cookies checked: ${authCookies.map((c) => c.name).join(', ')}`); + return true; + } catch (error) { + const err = error; + console.log(`âš ī¸ Error validating storage state: ${err instanceof Error ? err.message : String(err)}`); + console.log('🔄 Login test will run first to regenerate cookies'); + return false; + } +} + +// Validate storage state (exists + cookies not expired) +const storageStateValid = isStorageStateValid(); + +module.exports = defineConfig({ + testDir: "./.", + fullyParallel: false, + forbidOnly: !!process.env.CI, + retries: process.env.CI ? 2 : 0, + workers: 1, + reporter: "html", + + use: { + trace: "on-first-retry", + launchOptions: { + args: [ + "--disable-blink-features=AutomationControlled", + "--disable-dev-shm-usage", + "--no-sandbox", + ], + }, + // Do NOT set storageState here globally + // Each project will specify its own storage state needs + }, + + projects: [ + // ========================================== + // LOGIN TEST - Always runs with fresh browser + // ========================================== + { + name: 'login-test', + testMatch: /login.*\.spec\.js/, + use: { + ...devices["Desktop Chrome"], + // No storageState - fresh browser for login + }, + }, + + // ========================================== + // AUTHENTICATED TESTS - Need login (use storage state) + // ========================================== + { + name: 'chromium-authenticated', + testMatch: /.*\.authenticated\.spec\.js/, + // Smart dependency: Only depend on login if storage state is invalid + dependencies: storageStateValid ? [] : ['login-test'], + use: { + ...devices["Desktop Chrome"], + // Only set storage state if file exists + ...(storageStateExists ? { storageState: storageStatePath } : {}) + }, + }, + + // ========================================== + // UNAUTHENTICATED TESTS - Public pages (no login) + // ========================================== + { + name: 'chromium-unauthenticated', + testMatch: /.*\.unauthenticated\.spec\.js/, + use: { + ...devices["Desktop Chrome"], + // No storageState - fresh browser for public pages + }, + }, + + // // ========================================== + // // FIREFOX BROWSER + // // ========================================== + // { + // name: 'firefox-authenticated', + // testMatch: /.*\.authenticated\.spec\.js/, + // use: { + // ...devices["Desktop Firefox"], + // storageState: '.auth/storage-state.json', + // }, + // dependencies: ['setup'], + // }, + // { + // name: 'firefox-unauthenticated', + // testMatch: /.*\.unauthenticated\.spec\.js/, + // use: { ...devices["Desktop Firefox"] }, + // }, + // { + // name: 'firefox-login-tests', + // testMatch: /login.*\.spec\.js/, + // use: { ...devices["Desktop Firefox"] }, + // }, + + // // ========================================== + // // WEBKIT (Safari) BROWSER + // // ========================================== + // { + // name: 'webkit-authenticated', + // testMatch: /.*\.authenticated\.spec\.js/, + // use: { + // ...devices["Desktop Safari"], + // storageState: '.auth/storage-state.json', + // }, + // dependencies: ['setup'], + // }, + // { + // name: 'webkit-unauthenticated', + // testMatch: /.*\.unauthenticated\.spec\.js/, + // use: { ...devices["Desktop Safari"] }, + // }, + // { + // name: 'webkit-login-tests', + // testMatch: /login.*\.spec\.js/, + // use: { ...devices["Desktop Safari"] }, + // }, + ], +}); diff --git a/generated_tests/playwright_tests/tests/test_summary.md b/generated_tests/playwright_tests/tests/test_summary.md new file mode 100644 index 0000000..5c86dab --- /dev/null +++ b/generated_tests/playwright_tests/tests/test_summary.md @@ -0,0 +1,12 @@ +# Generated Playwright Tests Summary + +## login_invalid_credentials +**Description:** +**Priority**: N/A | **Complexity**: N/A +**Test File:** [login_invalid_credentials.spec.js](./login_invalid_credentials.spec.js) +--- +## login_analysis_happy_path +**Description:** +**Priority**: N/A | **Complexity**: N/A +**Test File:** [login_analysis_happy_path.spec.js](./login_analysis_happy_path.spec.js) +--- \ No newline at end of file