diff --git a/.prettierrc.json b/.prettierrc.json new file mode 100644 index 0000000..cba9b4d --- /dev/null +++ b/.prettierrc.json @@ -0,0 +1,34 @@ +{ + "printWidth": 80, + "tabWidth": 2, + "semi": true, + "singleQuote": true, + "quoteProps": "as-needed", + "jsxSingleQuote": true, + "trailingComma": "es5", + "bracketSpacing": true, + "bracketSameLine": false, + "arrowParens": "always", + "proseWrap": "preserve", + "htmlWhitespaceSensitivity": "css", + "endOfLine": "auto", + "embeddedLanguageFormatting": "auto", + "singleAttributePerLine": false, + "plugins": ["@ianvs/prettier-plugin-sort-imports"], + "importOrder": [ + "^(node:)", + "", + "^[.]", + "", + "", + "^[.]" + ], + "overrides": [ + { + "files": "*.jsonc", + "options": { + "trailingComma": "none" + } + } + ] +} diff --git a/CHANGELOG.md b/CHANGELOG.md index 022de42..dc6e42d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,7 +9,6 @@ ## 1.0.0 (2026-03-31) - ### Bug Fixes -* clarify Deno compatibility guidance in README ([#2](https://github.com/Lojhan/poku-react-testing/issues/2)) ([ee28792](https://github.com/Lojhan/poku-react-testing/commit/ee28792965b9cc9dacc8bee2fc96a5f84088f2d1)) +- clarify Deno compatibility guidance in README ([#2](https://github.com/Lojhan/poku-react-testing/issues/2)) ([ee28792](https://github.com/Lojhan/poku-react-testing/commit/ee28792965b9cc9dacc8bee2fc96a5f84088f2d1)) diff --git a/README.md b/README.md index 6fce352..0f1bde7 100644 --- a/README.md +++ b/README.md @@ -115,11 +115,11 @@ test('renders a heading', () => { ### Runtime × DOM Adapter -| | Node.js ≥ 20 | Bun ≥ 1 | Deno ≥ 2 | -|---|:---:|:---:|:---:| -| **happy-dom** | ✅ | ✅ | ✅ | -| **jsdom** | ✅ | ✅ | ⚠️ | -| **custom setup** | ✅ | ✅ | ✅ | +| | Node.js ≥ 20 | Bun ≥ 1 | Deno ≥ 2 | +| ---------------- | :----------: | :-----: | :------: | +| **happy-dom** | ✅ | ✅ | ✅ | +| **jsdom** | ✅ | ✅ | ⚠️ | +| **custom setup** | ✅ | ✅ | ✅ | > [!NOTE] > diff --git a/benchmark/REPORT.md b/benchmark/REPORT.md new file mode 100644 index 0000000..6896647 --- /dev/null +++ b/benchmark/REPORT.md @@ -0,0 +1,124 @@ +# React Testing Framework Benchmark Report + +> Generated: Tue, 31 Mar 2026 21:00:50 GMT + +## Environment + +| Property | Value | +|---|---| +| Node.js | v22.5.1 | +| Platform | darwin 25.4.0 | +| CPU | Apple M3 Pro | +| CPU Cores | 12 | +| Total RAM | 18.0 GB | +| Runs/scenario | 7 (trim ±1) | + +## Scenarios + +Each scenario runs the **same 9 React tests** across 5 test files: + +| Test File | Tests | +|---|---| +| 'counter.test.jsx' | 1 — stateful counter, event interaction | +| 'hooks.test.jsx' | 2 — custom hook harness + `renderHook` | +| 'lifecycle.test.jsx' | 2 — `rerender`, `unmount` + effect cleanup | +| 'context.test.jsx' | 1 — `createContext` + wrapper injection | +| 'concurrency.test.jsx' | 2 — React 19 `use()` + `useTransition` | + +### Frameworks under test + +| Combination | DOM layer | Assertion style | +|---|---|---| +| poku + poku-react-testing | happy-dom | `assert.strictEqual` | +| poku + poku-react-testing | jsdom | `assert.strictEqual` | +| jest 29 + @testing-library/react | jsdom (jest-environment-jsdom) | `expect().toBe()` | +| vitest 3 + @testing-library/react | jsdom | `expect().toBe()` | +| vitest 3 + @testing-library/react | happy-dom | `expect().toBe()` | + +## Results + +| Scenario | Mean | Min | Max | Stdev | Peak RSS | vs poku+happy-dom | +|--------------------|--------|--------|--------|--------|----------|-------------------| +| poku + happy-dom | 1.073s | 0.996s | 1.230s | 0.085s | 163.3 MB | *(baseline)* | +| poku + jsdom | 1.060s | 1.007s | 1.177s | 0.060s | 163.4 MB | -1% | +| jest + jsdom | 0.859s | 0.779s | 0.929s | 0.050s | 206.2 MB | -20% | +| vitest + jsdom | 0.964s | 0.950s | 0.987s | 0.017s | 148.0 MB | -10% | +| vitest + happy-dom | 0.838s | 0.812s | 0.864s | 0.021s | 116.3 MB | -22% | + +> **Wall-clock time** is measured with `performance.now()` around the child-process spawn. +> **Peak RSS** is captured via `/usr/bin/time -l` on macOS (bytes → MB). +> The baseline for relative comparisons is **poku + happy-dom**. + +## Analysis + +### Overall ranking (mean wall-clock time) + +1. **vitest + happy-dom** — 0.838s +2. **jest + jsdom** — 0.859s +3. **vitest + jsdom** — 0.964s +4. **poku + jsdom** — 1.060s +5. **poku + happy-dom** — 1.073s + +### Speed comparison + +- poku+happy-dom vs jest+jsdom: jest is **-20% faster** +- poku+happy-dom vs vitest+jsdom: vitest is **-10% faster** +- jest+jsdom vs vitest+jsdom: vitest is **12% slower** than jest + +### DOM adapter impact + +- **poku**: happy-dom vs jsdom — jsdom is **-1% faster** +- **vitest**: happy-dom vs jsdom — jsdom is **15% slower** + +### Memory footprint + +- **vitest + happy-dom**: 116.3 MB peak RSS +- **vitest + jsdom**: 148.0 MB peak RSS +- **poku + happy-dom**: 163.3 MB peak RSS +- **poku + jsdom**: 163.4 MB peak RSS +- **jest + jsdom**: 206.2 MB peak RSS + +### Consistency (lower stdev = more predictable) + +- **vitest + jsdom**: σ = 0.017s +- **vitest + happy-dom**: σ = 0.021s +- **jest + jsdom**: σ = 0.050s +- **poku + jsdom**: σ = 0.060s +- **poku + happy-dom**: σ = 0.085s + +## Key findings + +- **Fastest**: vitest + happy-dom — 0.838s mean +- **Slowest**: poku + happy-dom — 1.073s mean +- **Speed spread**: 28% difference between fastest and slowest + +### Interpretation + +**poku + poku-react-testing** avoids the multi-process or bundler startup that jest (babel transform +pipeline) and vitest (Vite + module graph) require. Its architecture — isolated per-file Node.js +processes with minimal bootstrap — means cold-start overhead is proportional to the number of test +files, not to the framework's own initialization. + +**jest** carries the heaviest startup cost due to: +1. Babel transformation of every TSX file on first run (no persistent cache in this benchmark) +2. 'jest-worker' process pool initialisation +3. JSDOM environment setup per test file + +**vitest** starts faster than jest because Vite's module graph is more efficient, and the +esbuild/Rollup pipeline is faster than Babel. However, the Vite dev server and HMR machinery still +contribute to startup overhead compared to a zero-bundler approach. + +**DOM adapter choice** (happy-dom vs jsdom) has a measurable but smaller effect than the choice of +framework. happy-dom is generally lighter and initialises faster; jsdom is more spec-complete. + +## Reproducibility + +```sh +# Install benchmark deps (one-time) +cd benchmark && npm install && cd .. + +# Re-run with custom run count +BENCH_RUNS=10 node benchmark/run.mjs +``` + +Results are saved to `benchmark/results.json` for programmatic analysis. diff --git a/benchmark/babel.config.cjs b/benchmark/babel.config.cjs new file mode 100644 index 0000000..9349078 --- /dev/null +++ b/benchmark/babel.config.cjs @@ -0,0 +1,6 @@ +module.exports = { + presets: [ + ['@babel/preset-env', { targets: { node: 'current' } }], + ['@babel/preset-react', { runtime: 'automatic' }], + ], +}; diff --git a/benchmark/jest.config.mjs b/benchmark/jest.config.mjs new file mode 100644 index 0000000..445dcdf --- /dev/null +++ b/benchmark/jest.config.mjs @@ -0,0 +1,8 @@ +export default { + testEnvironment: 'jsdom', + transform: { + '^.+\\.[jt]sx?$': ['babel-jest', { configFile: './babel.config.cjs' }], + }, + testMatch: ['/tests/jest/**/*.test.jsx'], + moduleFileExtensions: ['js', 'jsx', 'json'], +}; diff --git a/benchmark/package-lock.json b/benchmark/package-lock.json new file mode 100644 index 0000000..35767b9 --- /dev/null +++ b/benchmark/package-lock.json @@ -0,0 +1,7718 @@ +{ + "name": "poku-react-benchmark", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "poku-react-benchmark", + "version": "1.0.0", + "devDependencies": { + "@babel/core": "^7.29.0", + "@babel/preset-env": "^7.29.2", + "@babel/preset-react": "^7.28.5", + "@testing-library/react": "^16.3.2", + "@vitejs/plugin-react": "^4.7.0", + "babel-jest": "^30.3.0", + "happy-dom": "^20.8.9", + "jest": "^30.3.0", + "jest-environment-jsdom": "^30.3.0", + "jsdom": "26.1.0", + "poku": "4.2.0", + "poku-react-testing": "file:..", + "react": "^19.2.4", + "react-dom": "^19.2.4", + "vitest": "^3.2.4" + } + }, + "..": { + "name": "poku-react-testing", + "version": "1.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@testing-library/dom": "^10.4.1" + }, + "devDependencies": { + "@happy-dom/global-registrator": "^20.8.9", + "@types/jsdom": "^28.0.1", + "@types/node": "^25.5.0", + "@types/react": "^19.2.14", + "@types/react-dom": "^19.2.3", + "cross-env": "^10.1.0", + "happy-dom": "^20.8.9", + "jsdom": "^26.1.0", + "poku": "4.1.0", + "react": "^19.2.4", + "react-dom": "^19.2.4", + "rimraf": "^6.0.1", + "tsup": "^8.5.0", + "tsx": "^4.21.0", + "typescript": "^6.0.2" + }, + "engines": { + "bun": ">=1.x.x", + "deno": ">=2.x.x", + "node": ">=20.x.x", + "typescript": ">=6.x.x" + }, + "peerDependencies": { + "@happy-dom/global-registrator": ">=20", + "happy-dom": ">=20", + "jsdom": ">=22", + "poku": ">=4.1.0", + "react": ">=18", + "react-dom": ">=18" + }, + "peerDependenciesMeta": { + "@happy-dom/global-registrator": { + "optional": true + }, + "happy-dom": { + "optional": true + }, + "jsdom": { + "optional": true + } + } + }, + "node_modules/@babel/code-frame": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.28.5", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helpers": "^7.28.6", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/traverse": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.29.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-annotate-as-pure": { + "version": "7.27.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.27.3" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.28.6", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-create-class-features-plugin": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.3", + "@babel/helper-member-expression-to-functions": "^7.28.5", + "@babel/helper-optimise-call-expression": "^7.27.1", + "@babel/helper-replace-supers": "^7.28.6", + "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1", + "@babel/traverse": "^7.28.6", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-create-regexp-features-plugin": { + "version": "7.28.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.3", + "regexpu-core": "^6.3.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-define-polyfill-provider": { + "version": "0.6.8", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6", + "debug": "^4.4.3", + "lodash.debounce": "^4.0.8", + "resolve": "^1.22.11" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-member-expression-to-functions": { + "version": "7.28.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.28.5", + "@babel/types": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-validator-identifier": "^7.28.5", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-optimise-call-expression": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-remap-async-to-generator": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.1", + "@babel/helper-wrap-function": "^7.27.1", + "@babel/traverse": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-replace-supers": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-member-expression-to-functions": "^7.28.5", + "@babel/helper-optimise-call-expression": "^7.27.1", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-skip-transparent-expression-wrappers": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.27.1", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-wrap-function": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.28.6", + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.29.2", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.29.2", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.29.0" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-bugfix-firefox-class-in-computed-class-key": { + "version": "7.28.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/traverse": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-bugfix-safari-class-field-initializer-scope": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1", + "@babel/plugin-transform-optional-chaining": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.13.0" + } + }, + "node_modules/@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-proposal-private-property-in-object": { + "version": "7.21.0-placeholder-for-preset-env.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-static-block": { + "version": "7.14.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-assertions": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-attributes": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-private-property-in-object": { + "version": "7.14.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-unicode-sets-regex": { + "version": "7.18.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.18.6", + "@babel/helper-plugin-utils": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-arrow-functions": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-async-generator-functions": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-remap-async-to-generator": "^7.27.1", + "@babel/traverse": "^7.29.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-async-to-generator": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-remap-async-to-generator": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoped-functions": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoping": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-class-properties": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-class-static-block": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.12.0" + } + }, + "node_modules/@babel/plugin-transform-classes": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.3", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-globals": "^7.28.0", + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-replace-supers": "^7.28.6", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-computed-properties": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/template": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-destructuring": { + "version": "7.28.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/traverse": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-dotall-regex": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.28.5", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-duplicate-keys": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-duplicate-named-capturing-groups-regex": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.28.5", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-dynamic-import": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-explicit-resource-management": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/plugin-transform-destructuring": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-exponentiation-operator": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-export-namespace-from": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-for-of": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-function-name": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-compilation-targets": "^7.27.1", + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/traverse": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-json-strings": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-literals": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-logical-assignment-operators": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-member-expression-literals": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-amd": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-transforms": "^7.27.1", + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-commonjs": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-systemjs": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-validator-identifier": "^7.28.5", + "@babel/traverse": "^7.29.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-umd": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-transforms": "^7.27.1", + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-named-capturing-groups-regex": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.28.5", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-new-target": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-nullish-coalescing-operator": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-numeric-separator": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-object-rest-spread": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/plugin-transform-destructuring": "^7.28.5", + "@babel/plugin-transform-parameters": "^7.27.7", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-object-super": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/helper-replace-supers": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-optional-catch-binding": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-optional-chaining": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-parameters": { + "version": "7.27.7", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-private-methods": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-private-property-in-object": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.3", + "@babel/helper-create-class-features-plugin": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-property-literals": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-display-name": { + "version": "7.28.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.3", + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/plugin-syntax-jsx": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-development": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/plugin-transform-react-jsx": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-self": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz", + "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-source": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz", + "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-pure-annotations": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.27.1", + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-regenerator": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-regexp-modifiers": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.28.5", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-reserved-words": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-shorthand-properties": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-spread": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-sticky-regex": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-template-literals": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-typeof-symbol": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-escapes": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-property-regex": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.28.5", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-regex": { + "version": "7.27.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.27.1", + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-sets-regex": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.28.5", + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/preset-env": { + "version": "7.29.2", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.29.0", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-plugin-utils": "^7.28.6", + "@babel/helper-validator-option": "^7.27.1", + "@babel/plugin-bugfix-firefox-class-in-computed-class-key": "^7.28.5", + "@babel/plugin-bugfix-safari-class-field-initializer-scope": "^7.27.1", + "@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression": "^7.27.1", + "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining": "^7.27.1", + "@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly": "^7.28.6", + "@babel/plugin-proposal-private-property-in-object": "7.21.0-placeholder-for-preset-env.2", + "@babel/plugin-syntax-import-assertions": "^7.28.6", + "@babel/plugin-syntax-import-attributes": "^7.28.6", + "@babel/plugin-syntax-unicode-sets-regex": "^7.18.6", + "@babel/plugin-transform-arrow-functions": "^7.27.1", + "@babel/plugin-transform-async-generator-functions": "^7.29.0", + "@babel/plugin-transform-async-to-generator": "^7.28.6", + "@babel/plugin-transform-block-scoped-functions": "^7.27.1", + "@babel/plugin-transform-block-scoping": "^7.28.6", + "@babel/plugin-transform-class-properties": "^7.28.6", + "@babel/plugin-transform-class-static-block": "^7.28.6", + "@babel/plugin-transform-classes": "^7.28.6", + "@babel/plugin-transform-computed-properties": "^7.28.6", + "@babel/plugin-transform-destructuring": "^7.28.5", + "@babel/plugin-transform-dotall-regex": "^7.28.6", + "@babel/plugin-transform-duplicate-keys": "^7.27.1", + "@babel/plugin-transform-duplicate-named-capturing-groups-regex": "^7.29.0", + "@babel/plugin-transform-dynamic-import": "^7.27.1", + "@babel/plugin-transform-explicit-resource-management": "^7.28.6", + "@babel/plugin-transform-exponentiation-operator": "^7.28.6", + "@babel/plugin-transform-export-namespace-from": "^7.27.1", + "@babel/plugin-transform-for-of": "^7.27.1", + "@babel/plugin-transform-function-name": "^7.27.1", + "@babel/plugin-transform-json-strings": "^7.28.6", + "@babel/plugin-transform-literals": "^7.27.1", + "@babel/plugin-transform-logical-assignment-operators": "^7.28.6", + "@babel/plugin-transform-member-expression-literals": "^7.27.1", + "@babel/plugin-transform-modules-amd": "^7.27.1", + "@babel/plugin-transform-modules-commonjs": "^7.28.6", + "@babel/plugin-transform-modules-systemjs": "^7.29.0", + "@babel/plugin-transform-modules-umd": "^7.27.1", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.29.0", + "@babel/plugin-transform-new-target": "^7.27.1", + "@babel/plugin-transform-nullish-coalescing-operator": "^7.28.6", + "@babel/plugin-transform-numeric-separator": "^7.28.6", + "@babel/plugin-transform-object-rest-spread": "^7.28.6", + "@babel/plugin-transform-object-super": "^7.27.1", + "@babel/plugin-transform-optional-catch-binding": "^7.28.6", + "@babel/plugin-transform-optional-chaining": "^7.28.6", + "@babel/plugin-transform-parameters": "^7.27.7", + "@babel/plugin-transform-private-methods": "^7.28.6", + "@babel/plugin-transform-private-property-in-object": "^7.28.6", + "@babel/plugin-transform-property-literals": "^7.27.1", + "@babel/plugin-transform-regenerator": "^7.29.0", + "@babel/plugin-transform-regexp-modifiers": "^7.28.6", + "@babel/plugin-transform-reserved-words": "^7.27.1", + "@babel/plugin-transform-shorthand-properties": "^7.27.1", + "@babel/plugin-transform-spread": "^7.28.6", + "@babel/plugin-transform-sticky-regex": "^7.27.1", + "@babel/plugin-transform-template-literals": "^7.27.1", + "@babel/plugin-transform-typeof-symbol": "^7.27.1", + "@babel/plugin-transform-unicode-escapes": "^7.27.1", + "@babel/plugin-transform-unicode-property-regex": "^7.28.6", + "@babel/plugin-transform-unicode-regex": "^7.27.1", + "@babel/plugin-transform-unicode-sets-regex": "^7.28.6", + "@babel/preset-modules": "0.1.6-no-external-plugins", + "babel-plugin-polyfill-corejs2": "^0.4.15", + "babel-plugin-polyfill-corejs3": "^0.14.0", + "babel-plugin-polyfill-regenerator": "^0.6.6", + "core-js-compat": "^3.48.0", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-modules": { + "version": "0.1.6-no-external-plugins", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/types": "^7.4.4", + "esutils": "^2.0.2" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/@babel/preset-react": { + "version": "7.28.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1", + "@babel/helper-validator-option": "^7.27.1", + "@babel/plugin-transform-react-display-name": "^7.28.0", + "@babel/plugin-transform-react-jsx": "^7.27.1", + "@babel/plugin-transform-react-jsx-development": "^7.27.1", + "@babel/plugin-transform-react-pure-annotations": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/runtime": { + "version": "7.29.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/template": { + "version": "7.28.6", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.29.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "dev": true, + "license": "MIT" + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz", + "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz", + "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz", + "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz", + "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz", + "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz", + "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz", + "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz", + "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz", + "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz", + "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz", + "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz", + "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz", + "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz", + "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz", + "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz", + "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz", + "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz", + "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz", + "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz", + "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz", + "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz", + "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz", + "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz", + "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz", + "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz", + "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@isaacs/cliui": { + "version": "8.0.2", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^5.1.2", + "string-width-cjs": "npm:string-width@^4.2.0", + "strip-ansi": "^7.0.1", + "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", + "wrap-ansi": "^8.1.0", + "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "dev": true, + "license": "ISC", + "dependencies": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "30.3.0", + "@types/node": "*", + "chalk": "^4.1.2", + "jest-message-util": "30.3.0", + "jest-util": "30.3.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/core": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "30.3.0", + "@jest/pattern": "30.0.1", + "@jest/reporters": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "ansi-escapes": "^4.3.2", + "chalk": "^4.1.2", + "ci-info": "^4.2.0", + "exit-x": "^0.2.2", + "graceful-fs": "^4.2.11", + "jest-changed-files": "30.3.0", + "jest-config": "30.3.0", + "jest-haste-map": "30.3.0", + "jest-message-util": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-resolve": "30.3.0", + "jest-resolve-dependencies": "30.3.0", + "jest-runner": "30.3.0", + "jest-runtime": "30.3.0", + "jest-snapshot": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "jest-watcher": "30.3.0", + "pretty-format": "30.3.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/core/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/core/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/core/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/@jest/diff-sequences": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/environment": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/fake-timers": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "jest-mock": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/environment-jsdom-abstract": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/fake-timers": "30.3.0", + "@jest/types": "30.3.0", + "@types/jsdom": "^21.1.7", + "@types/node": "*", + "jest-mock": "30.3.0", + "jest-util": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "canvas": "^3.0.0", + "jsdom": "*" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/@jest/expect": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "30.3.0", + "jest-snapshot": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/expect-utils": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/get-type": "30.1.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/fake-timers": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "30.3.0", + "@sinonjs/fake-timers": "^15.0.0", + "@types/node": "*", + "jest-message-util": "30.3.0", + "jest-mock": "30.3.0", + "jest-util": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/get-type": { + "version": "30.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/globals": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/expect": "30.3.0", + "@jest/types": "30.3.0", + "jest-mock": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/pattern": { + "version": "30.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "jest-regex-util": "30.0.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/reporters": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "@jridgewell/trace-mapping": "^0.3.25", + "@types/node": "*", + "chalk": "^4.1.2", + "collect-v8-coverage": "^1.0.2", + "exit-x": "^0.2.2", + "glob": "^10.5.0", + "graceful-fs": "^4.2.11", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^6.0.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^5.0.0", + "istanbul-reports": "^3.1.3", + "jest-message-util": "30.3.0", + "jest-util": "30.3.0", + "jest-worker": "30.3.0", + "slash": "^3.0.0", + "string-length": "^4.0.2", + "v8-to-istanbul": "^9.0.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/schemas": { + "version": "30.0.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@sinclair/typebox": "^0.34.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/snapshot-utils": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "30.3.0", + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "natural-compare": "^1.4.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/source-map": { + "version": "30.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.25", + "callsites": "^3.1.0", + "graceful-fs": "^4.2.11" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/test-result": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "30.3.0", + "@jest/types": "30.3.0", + "@types/istanbul-lib-coverage": "^2.0.6", + "collect-v8-coverage": "^1.0.2" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/test-sequencer": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "30.3.0", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/transform": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.27.4", + "@jest/types": "30.3.0", + "@jridgewell/trace-mapping": "^0.3.25", + "babel-plugin-istanbul": "^7.0.1", + "chalk": "^4.1.2", + "convert-source-map": "^2.0.0", + "fast-json-stable-stringify": "^2.1.0", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-util": "30.3.0", + "pirates": "^4.0.7", + "slash": "^3.0.0", + "write-file-atomic": "^5.0.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/types": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/pattern": "30.0.1", + "@jest/schemas": "30.0.5", + "@types/istanbul-lib-coverage": "^2.0.6", + "@types/istanbul-reports": "^3.0.4", + "@types/node": "*", + "@types/yargs": "^17.0.33", + "chalk": "^4.1.2" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@pkgjs/parseargs": { + "version": "0.11.0", + "dev": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/@pkgr/core": { + "version": "0.2.9", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/pkgr" + } + }, + "node_modules/@rolldown/pluginutils": { + "version": "1.0.0-beta.27", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.27.tgz", + "integrity": "sha512-+d0F4MKMCbeVUJwG96uQ4SgAznZNSq93I3V+9NHA4OpvqG8mRCpGdKmK8l/dl02h2CCDHwW2FqilnTyDcAnqjA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.60.1.tgz", + "integrity": "sha512-d6FinEBLdIiK+1uACUttJKfgZREXrF0Qc2SmLII7W2AD8FfiZ9Wjd+rD/iRuf5s5dWrr1GgwXCvPqOuDquOowA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.60.1.tgz", + "integrity": "sha512-YjG/EwIDvvYI1YvYbHvDz/BYHtkY4ygUIXHnTdLhG+hKIQFBiosfWiACWortsKPKU/+dUwQQCKQM3qrDe8c9BA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.60.1.tgz", + "integrity": "sha512-mjCpF7GmkRtSJwon+Rq1N8+pI+8l7w5g9Z3vWj4T7abguC4Czwi3Yu/pFaLvA3TTeMVjnu3ctigusqWUfjZzvw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.60.1.tgz", + "integrity": "sha512-haZ7hJ1JT4e9hqkoT9R/19XW2QKqjfJVv+i5AGg57S+nLk9lQnJ1F/eZloRO3o9Scy9CM3wQ9l+dkXtcBgN5Ew==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.60.1.tgz", + "integrity": "sha512-czw90wpQq3ZsAVBlinZjAYTKduOjTywlG7fEeWKUA7oCmpA8xdTkxZZlwNJKWqILlq0wehoZcJYfBvOyhPTQ6w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.60.1.tgz", + "integrity": "sha512-KVB2rqsxTHuBtfOeySEyzEOB7ltlB/ux38iu2rBQzkjbwRVlkhAGIEDiiYnO2kFOkJp+Z7pUXKyrRRFuFUKt+g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.60.1.tgz", + "integrity": "sha512-L+34Qqil+v5uC0zEubW7uByo78WOCIrBvci69E7sFASRl0X7b/MB6Cqd1lky/CtcSVTydWa2WZwFuWexjS5o6g==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.60.1.tgz", + "integrity": "sha512-n83O8rt4v34hgFzlkb1ycniJh7IR5RCIqt6mz1VRJD6pmhRi0CXdmfnLu9dIUS6buzh60IvACM842Ffb3xd6Gg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.60.1.tgz", + "integrity": "sha512-Nql7sTeAzhTAja3QXeAI48+/+GjBJ+QmAH13snn0AJSNL50JsDqotyudHyMbO2RbJkskbMbFJfIJKWA6R1LCJQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.60.1.tgz", + "integrity": "sha512-+pUymDhd0ys9GcKZPPWlFiZ67sTWV5UU6zOJat02M1+PiuSGDziyRuI/pPue3hoUwm2uGfxdL+trT6Z9rxnlMA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.60.1.tgz", + "integrity": "sha512-VSvgvQeIcsEvY4bKDHEDWcpW4Yw7BtlKG1GUT4FzBUlEKQK0rWHYBqQt6Fm2taXS+1bXvJT6kICu5ZwqKCnvlQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.60.1.tgz", + "integrity": "sha512-4LqhUomJqwe641gsPp6xLfhqWMbQV04KtPp7/dIp0nzPxAkNY1AbwL5W0MQpcalLYk07vaW9Kp1PBhdpZYYcEw==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.60.1.tgz", + "integrity": "sha512-tLQQ9aPvkBxOc/EUT6j3pyeMD6Hb8QF2BTBnCQWP/uu1lhc9AIrIjKnLYMEroIz/JvtGYgI9dF3AxHZNaEH0rw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.60.1.tgz", + "integrity": "sha512-RMxFhJwc9fSXP6PqmAz4cbv3kAyvD1etJFjTx4ONqFP9DkTkXsAMU4v3Vyc5BgzC+anz7nS/9tp4obsKfqkDHg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.60.1.tgz", + "integrity": "sha512-QKgFl+Yc1eEk6MmOBfRHYF6lTxiiiV3/z/BRrbSiW2I7AFTXoBFvdMEyglohPj//2mZS4hDOqeB0H1ACh3sBbg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.60.1.tgz", + "integrity": "sha512-RAjXjP/8c6ZtzatZcA1RaQr6O1TRhzC+adn8YZDnChliZHviqIjmvFwHcxi4JKPSDAt6Uhf/7vqcBzQJy0PDJg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.60.1.tgz", + "integrity": "sha512-wcuocpaOlaL1COBYiA89O6yfjlp3RwKDeTIA0hM7OpmhR1Bjo9j31G1uQVpDlTvwxGn2nQs65fBFL5UFd76FcQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.60.1.tgz", + "integrity": "sha512-77PpsFQUCOiZR9+LQEFg9GClyfkNXj1MP6wRnzYs0EeWbPcHs02AXu4xuUbM1zhwn3wqaizle3AEYg5aeoohhg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.60.1.tgz", + "integrity": "sha512-5cIATbk5vynAjqqmyBjlciMJl1+R/CwX9oLk/EyiFXDWd95KpHdrOJT//rnUl4cUcskrd0jCCw3wpZnhIHdD9w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-openbsd-x64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.60.1.tgz", + "integrity": "sha512-cl0w09WsCi17mcmWqqglez9Gk8isgeWvoUZ3WiJFYSR3zjBQc2J5/ihSjpl+VLjPqjQ/1hJRcqBfLjssREQILw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ] + }, + "node_modules/@rollup/rollup-openharmony-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.60.1.tgz", + "integrity": "sha512-4Cv23ZrONRbNtbZa37mLSueXUCtN7MXccChtKpUnQNgF010rjrjfHx3QxkS2PI7LqGT5xXyYs1a7LbzAwT0iCA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.60.1.tgz", + "integrity": "sha512-i1okWYkA4FJICtr7KpYzFpRTHgy5jdDbZiWfvny21iIKky5YExiDXP+zbXzm3dUcFpkEeYNHgQ5fuG236JPq0g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.60.1.tgz", + "integrity": "sha512-u09m3CuwLzShA0EYKMNiFgcjjzwqtUMLmuCJLeZWjjOYA3IT2Di09KaxGBTP9xVztWyIWjVdsB2E9goMjZvTQg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.60.1.tgz", + "integrity": "sha512-k+600V9Zl1CM7eZxJgMyTUzmrmhB/0XZnF4pRypKAlAgxmedUA+1v9R+XOFv56W4SlHEzfeMtzujLJD22Uz5zg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.60.1.tgz", + "integrity": "sha512-lWMnixq/QzxyhTV6NjQJ4SFo1J6PvOX8vUx5Wb4bBPsEb+8xZ89Bz6kOXpfXj9ak9AHTQVQzlgzBEc1SyM27xQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@sinclair/typebox": { + "version": "0.34.49", + "dev": true, + "license": "MIT" + }, + "node_modules/@sinonjs/commons": { + "version": "3.0.1", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "15.2.0", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@sinonjs/commons": "^3.0.1" + } + }, + "node_modules/@testing-library/dom": { + "version": "10.4.1", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.1.tgz", + "integrity": "sha512-o4PXJQidqJl82ckFaXUeoAW+XysPLauYI43Abki5hABd853iMhitooc6znOnczgbTYmEP6U6/y1ZyKAIsvMKGg==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/runtime": "^7.12.5", + "@types/aria-query": "^5.0.1", + "aria-query": "5.3.0", + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "picocolors": "1.1.1", + "pretty-format": "^27.0.2" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@testing-library/react": { + "version": "16.3.2", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.12.5" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@testing-library/dom": "^10.0.0", + "@types/react": "^18.0.0 || ^19.0.0", + "@types/react-dom": "^18.0.0 || ^19.0.0", + "react": "^18.0.0 || ^19.0.0", + "react-dom": "^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@types/aria-query": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.4.tgz", + "integrity": "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==", + "dev": true, + "license": "MIT", + "peer": true + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/chai": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz", + "integrity": "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/deep-eql": "*", + "assertion-error": "^2.0.1" + } + }, + "node_modules/@types/deep-eql": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", + "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/istanbul-lib-coverage": { + "version": "2.0.6", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/istanbul-lib-report": { + "version": "3.0.3", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-coverage": "*" + } + }, + "node_modules/@types/istanbul-reports": { + "version": "3.0.4", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-report": "*" + } + }, + "node_modules/@types/jsdom": { + "version": "21.1.7", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "@types/tough-cookie": "*", + "parse5": "^7.0.0" + } + }, + "node_modules/@types/node": { + "version": "25.5.0", + "dev": true, + "license": "MIT", + "dependencies": { + "undici-types": "~7.18.0" + } + }, + "node_modules/@types/stack-utils": { + "version": "2.0.3", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/tough-cookie": { + "version": "4.0.5", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/whatwg-mimetype": { + "version": "3.0.2", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/ws": { + "version": "8.18.1", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/yargs": { + "version": "17.0.35", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/yargs-parser": { + "version": "21.0.3", + "dev": true, + "license": "MIT" + }, + "node_modules/@ungap/structured-clone": { + "version": "1.3.0", + "dev": true, + "license": "ISC" + }, + "node_modules/@unrs/resolver-binding-darwin-arm64": { + "version": "1.11.1", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@vitejs/plugin-react": { + "version": "4.7.0", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.7.0.tgz", + "integrity": "sha512-gUu9hwfWvvEDBBmgtAowQCojwZmJ5mcLn3aufeCsitijs3+f2NsrPtlAWIR6OPiqljl96GVCUbLe0HyqIpVaoA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.28.0", + "@babel/plugin-transform-react-jsx-self": "^7.27.1", + "@babel/plugin-transform-react-jsx-source": "^7.27.1", + "@rolldown/pluginutils": "1.0.0-beta.27", + "@types/babel__core": "^7.20.5", + "react-refresh": "^0.17.0" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "peerDependencies": { + "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" + } + }, + "node_modules/@vitest/expect": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-3.2.4.tgz", + "integrity": "sha512-Io0yyORnB6sikFlt8QW5K7slY4OjqNX9jmJQ02QDda8lyM6B5oNgVWoSoKPac8/kgnCUzuHQKrSLtu/uOqqrig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/chai": "^5.2.2", + "@vitest/spy": "3.2.4", + "@vitest/utils": "3.2.4", + "chai": "^5.2.0", + "tinyrainbow": "^2.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/mocker": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-3.2.4.tgz", + "integrity": "sha512-46ryTE9RZO/rfDd7pEqFl7etuyzekzEhUbTW3BvmeO/BcCMEgq59BKhek3dXDWgAj4oMK6OZi+vRr1wPW6qjEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/spy": "3.2.4", + "estree-walker": "^3.0.3", + "magic-string": "^0.30.17" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "msw": "^2.4.9", + "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0" + }, + "peerDependenciesMeta": { + "msw": { + "optional": true + }, + "vite": { + "optional": true + } + } + }, + "node_modules/@vitest/pretty-format": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-3.2.4.tgz", + "integrity": "sha512-IVNZik8IVRJRTr9fxlitMKeJeXFFFN0JaB9PHPGQ8NKQbGpfjlTx9zO4RefN8gp7eqjNy8nyK3NZmBzOPeIxtA==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyrainbow": "^2.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/runner": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-3.2.4.tgz", + "integrity": "sha512-oukfKT9Mk41LreEW09vt45f8wx7DordoWUZMYdY/cyAk7w5TWkTRCNZYF7sX7n2wB7jyGAl74OxgwhPgKaqDMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/utils": "3.2.4", + "pathe": "^2.0.3", + "strip-literal": "^3.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/snapshot": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-3.2.4.tgz", + "integrity": "sha512-dEYtS7qQP2CjU27QBC5oUOxLE/v5eLkGqPE0ZKEIDGMs4vKWe7IjgLOeauHsR0D5YuuycGRO5oSRXnwnmA78fQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "3.2.4", + "magic-string": "^0.30.17", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/spy": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-3.2.4.tgz", + "integrity": "sha512-vAfasCOe6AIK70iP5UD11Ac4siNUNJ9i/9PZ3NKx07sG6sUxeag1LWdNrMWeKKYBLlzuK+Gn65Yd5nyL6ds+nw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyspy": "^4.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/utils": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-3.2.4.tgz", + "integrity": "sha512-fB2V0JFrQSMsCo9HiSq3Ezpdv4iYaXRG1Sx8edX3MwxfyNn83mKiGzOcH+Fkxt4MHxr3y42fQi1oeAInqgX2QA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "3.2.4", + "loupe": "^3.1.4", + "tinyrainbow": "^2.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/agent-base": { + "version": "7.1.4", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14" + } + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "dev": true, + "license": "ISC", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/anymatch/node_modules/picomatch": { + "version": "2.3.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/argparse": { + "version": "1.0.10", + "dev": true, + "license": "MIT", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/aria-query": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "dequal": "^2.0.3" + } + }, + "node_modules/assertion-error": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", + "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/babel-jest": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/transform": "30.3.0", + "@types/babel__core": "^7.20.5", + "babel-plugin-istanbul": "^7.0.1", + "babel-preset-jest": "30.3.0", + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "slash": "^3.0.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.11.0 || ^8.0.0-0" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "7.0.1", + "dev": true, + "license": "BSD-3-Clause", + "workspaces": [ + "test/babel-8" + ], + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-instrument": "^6.0.2", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/babel-plugin-jest-hoist": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/babel__core": "^7.20.5" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/babel-plugin-polyfill-corejs2": { + "version": "0.4.17", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.28.6", + "@babel/helper-define-polyfill-provider": "^0.6.8", + "semver": "^6.3.1" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/babel-plugin-polyfill-corejs3": { + "version": "0.14.2", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.6.8", + "core-js-compat": "^3.48.0" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/babel-plugin-polyfill-regenerator": { + "version": "0.6.8", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.6.8" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.12.13", + "@babel/plugin-syntax-class-static-block": "^7.14.5", + "@babel/plugin-syntax-import-attributes": "^7.24.7", + "@babel/plugin-syntax-import-meta": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5", + "@babel/plugin-syntax-top-level-await": "^7.14.5" + }, + "peerDependencies": { + "@babel/core": "^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/babel-preset-jest": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "babel-plugin-jest-hoist": "30.3.0", + "babel-preset-current-node-syntax": "^1.2.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.11.0 || ^8.0.0-beta.1" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "dev": true, + "license": "MIT" + }, + "node_modules/baseline-browser-mapping": { + "version": "2.10.13", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.cjs" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/brace-expansion": { + "version": "2.0.3", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/browserslist": { + "version": "4.28.2", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "baseline-browser-mapping": "^2.10.12", + "caniuse-lite": "^1.0.30001782", + "electron-to-chromium": "^1.5.328", + "node-releases": "^2.0.36", + "update-browserslist-db": "^1.2.3" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "dev": true, + "license": "MIT" + }, + "node_modules/cac": { + "version": "6.7.14", + "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz", + "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001782", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chai": { + "version": "5.3.3", + "resolved": "https://registry.npmjs.org/chai/-/chai-5.3.3.tgz", + "integrity": "sha512-4zNhdJD/iOjSH0A05ea+Ke6MU5mmpQcbQsSOkgdaUMJ9zTlDTD/GYlwohmIE2u0gaxHYiVHEn1Fw9mZ/ktJWgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "assertion-error": "^2.0.1", + "check-error": "^2.1.1", + "deep-eql": "^5.0.1", + "loupe": "^3.1.0", + "pathval": "^2.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "dev": true, + "license": "MIT", + "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/char-regex": { + "version": "1.0.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/check-error": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/check-error/-/check-error-2.1.3.tgz", + "integrity": "sha512-PAJdDJusoxnwm1VwW07VWwUN1sl7smmC3OKggvndJFadxxDRyFJBX/ggnu/KE4kQAB7a3Dp8f/YXC1FlUprWmA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 16" + } + }, + "node_modules/ci-info": { + "version": "4.4.0", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/sibiraj-s" + } + ], + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/cjs-module-lexer": { + "version": "2.2.0", + "dev": true, + "license": "MIT" + }, + "node_modules/cliui": { + "version": "8.0.1", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/cliui/node_modules/emoji-regex": { + "version": "8.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/cliui/node_modules/string-width": { + "version": "4.2.3", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/strip-ansi": { + "version": "6.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/wrap-ansi": { + "version": "7.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/co": { + "version": "4.6.0", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.3", + "dev": true, + "license": "MIT" + }, + "node_modules/color-convert": { + "version": "2.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "dev": true, + "license": "MIT" + }, + "node_modules/concat-map": { + "version": "0.0.1", + "dev": true, + "license": "MIT" + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/core-js-compat": { + "version": "3.49.0", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.28.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/cssstyle": { + "version": "4.6.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@asamuzakjp/css-color": "^3.2.0", + "rrweb-cssom": "^0.8.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/cssstyle/node_modules/@asamuzakjp/css-color": { + "version": "3.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@csstools/css-calc": "^2.1.3", + "@csstools/css-color-parser": "^3.0.9", + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3", + "lru-cache": "^10.4.3" + } + }, + "node_modules/cssstyle/node_modules/@csstools/color-helpers": { + "version": "5.1.0", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT-0", + "engines": { + "node": ">=18" + } + }, + "node_modules/cssstyle/node_modules/@csstools/css-calc": { + "version": "2.1.4", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-parser-algorithms": "^3.0.5", + "@csstools/css-tokenizer": "^3.0.4" + } + }, + "node_modules/cssstyle/node_modules/@csstools/css-color-parser": { + "version": "3.1.0", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "dependencies": { + "@csstools/color-helpers": "^5.1.0", + "@csstools/css-calc": "^2.1.4" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-parser-algorithms": "^3.0.5", + "@csstools/css-tokenizer": "^3.0.4" + } + }, + "node_modules/cssstyle/node_modules/@csstools/css-parser-algorithms": { + "version": "3.0.5", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-tokenizer": "^3.0.4" + } + }, + "node_modules/cssstyle/node_modules/@csstools/css-tokenizer": { + "version": "3.0.4", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/cssstyle/node_modules/lru-cache": { + "version": "10.4.3", + "dev": true, + "license": "ISC" + }, + "node_modules/data-urls": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-5.0.0.tgz", + "integrity": "sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg==", + "dev": true, + "license": "MIT", + "dependencies": { + "whatwg-mimetype": "^4.0.0", + "whatwg-url": "^14.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/data-urls/node_modules/whatwg-mimetype": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", + "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decimal.js": { + "version": "10.6.0", + "dev": true, + "license": "MIT" + }, + "node_modules/dedent": { + "version": "1.7.2", + "dev": true, + "license": "MIT", + "peerDependencies": { + "babel-plugin-macros": "^3.1.0" + }, + "peerDependenciesMeta": { + "babel-plugin-macros": { + "optional": true + } + } + }, + "node_modules/deep-eql": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz", + "integrity": "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/detect-libc": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", + "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", + "dev": true, + "license": "Apache-2.0", + "optional": true, + "peer": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/detect-newline": { + "version": "3.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/dom-accessibility-api": { + "version": "0.5.16", + "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", + "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==", + "dev": true, + "license": "MIT", + "peer": true + }, + "node_modules/eastasianwidth": { + "version": "0.2.0", + "dev": true, + "license": "MIT" + }, + "node_modules/electron-to-chromium": { + "version": "1.5.329", + "dev": true, + "license": "ISC" + }, + "node_modules/emittery": { + "version": "0.13.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/emoji-regex": { + "version": "9.2.2", + "dev": true, + "license": "MIT" + }, + "node_modules/entities": { + "version": "7.0.1", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/error-ex": { + "version": "1.3.4", + "dev": true, + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/es-module-lexer": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz", + "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==", + "dev": true, + "license": "MIT" + }, + "node_modules/esbuild": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz", + "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.25.12", + "@esbuild/android-arm": "0.25.12", + "@esbuild/android-arm64": "0.25.12", + "@esbuild/android-x64": "0.25.12", + "@esbuild/darwin-arm64": "0.25.12", + "@esbuild/darwin-x64": "0.25.12", + "@esbuild/freebsd-arm64": "0.25.12", + "@esbuild/freebsd-x64": "0.25.12", + "@esbuild/linux-arm": "0.25.12", + "@esbuild/linux-arm64": "0.25.12", + "@esbuild/linux-ia32": "0.25.12", + "@esbuild/linux-loong64": "0.25.12", + "@esbuild/linux-mips64el": "0.25.12", + "@esbuild/linux-ppc64": "0.25.12", + "@esbuild/linux-riscv64": "0.25.12", + "@esbuild/linux-s390x": "0.25.12", + "@esbuild/linux-x64": "0.25.12", + "@esbuild/netbsd-arm64": "0.25.12", + "@esbuild/netbsd-x64": "0.25.12", + "@esbuild/openbsd-arm64": "0.25.12", + "@esbuild/openbsd-x64": "0.25.12", + "@esbuild/openharmony-arm64": "0.25.12", + "@esbuild/sunos-x64": "0.25.12", + "@esbuild/win32-arm64": "0.25.12", + "@esbuild/win32-ia32": "0.25.12", + "@esbuild/win32-x64": "0.25.12" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "2.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "dev": true, + "license": "BSD-2-Clause", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/estree-walker": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", + "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "^1.0.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "dev": true, + "license": "MIT", + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/execa/node_modules/signal-exit": { + "version": "3.0.7", + "dev": true, + "license": "ISC" + }, + "node_modules/exit-x": { + "version": "0.2.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/expect-utils": "30.3.0", + "@jest/get-type": "30.1.0", + "jest-matcher-utils": "30.3.0", + "jest-message-util": "30.3.0", + "jest-mock": "30.3.0", + "jest-util": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/expect-type": { + "version": "1.3.0", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "dev": true, + "license": "MIT" + }, + "node_modules/fb-watchman": { + "version": "2.0.2", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "bser": "2.1.1" + } + }, + "node_modules/fdir": { + "version": "6.5.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/find-up": { + "version": "4.1.0", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/foreground-child": { + "version": "3.3.1", + "dev": true, + "license": "ISC", + "dependencies": { + "cross-spawn": "^7.0.6", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/glob": { + "version": "10.5.0", + "dev": true, + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "dev": true, + "license": "ISC" + }, + "node_modules/happy-dom": { + "version": "20.8.9", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": ">=20.0.0", + "@types/whatwg-mimetype": "^3.0.2", + "@types/ws": "^8.18.1", + "entities": "^7.0.1", + "whatwg-mimetype": "^3.0.0", + "ws": "^8.18.3" + }, + "engines": { + "node": ">=20.0.0" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "dev": true, + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/html-encoding-sniffer": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-4.0.0.tgz", + "integrity": "sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "whatwg-encoding": "^3.1.1" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "dev": true, + "license": "MIT" + }, + "node_modules/http-proxy-agent": { + "version": "7.0.2", + "dev": true, + "license": "MIT", + "dependencies": { + "agent-base": "^7.1.0", + "debug": "^4.3.4" + }, + "engines": { + "node": ">= 14" + } + }, + "node_modules/https-proxy-agent": { + "version": "7.0.6", + "dev": true, + "license": "MIT", + "dependencies": { + "agent-base": "^7.1.2", + "debug": "4" + }, + "engines": { + "node": ">= 14" + } + }, + "node_modules/human-signals": { + "version": "2.1.0", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "dev": true, + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/import-local": { + "version": "3.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "dev": true, + "license": "ISC", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "dev": true, + "license": "ISC" + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "dev": true, + "license": "MIT" + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-potential-custom-element-name": { + "version": "1.0.1", + "dev": true, + "license": "MIT" + }, + "node_modules/is-stream": { + "version": "2.0.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "dev": true, + "license": "ISC" + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.2.2", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "6.0.3", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.23.9", + "@babel/parser": "^7.23.9", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-instrument/node_modules/semver": { + "version": "7.7.4", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "5.0.6", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.23", + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-reports": { + "version": "3.2.0", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jackspeak": { + "version": "3.4.3", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + }, + "optionalDependencies": { + "@pkgjs/parseargs": "^0.11.0" + } + }, + "node_modules/jest": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "30.3.0", + "@jest/types": "30.3.0", + "import-local": "^3.2.0", + "jest-cli": "30.3.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "execa": "^5.1.1", + "jest-util": "30.3.0", + "p-limit": "^3.1.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-circus": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/expect": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "chalk": "^4.1.2", + "co": "^4.6.0", + "dedent": "^1.6.0", + "is-generator-fn": "^2.1.0", + "jest-each": "30.3.0", + "jest-matcher-utils": "30.3.0", + "jest-message-util": "30.3.0", + "jest-runtime": "30.3.0", + "jest-snapshot": "30.3.0", + "jest-util": "30.3.0", + "p-limit": "^3.1.0", + "pretty-format": "30.3.0", + "pure-rand": "^7.0.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.6" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-circus/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-circus/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-circus/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-cli": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/types": "30.3.0", + "chalk": "^4.1.2", + "exit-x": "^0.2.2", + "import-local": "^3.2.0", + "jest-config": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "yargs": "^17.7.2" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-config": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.27.4", + "@jest/get-type": "30.1.0", + "@jest/pattern": "30.0.1", + "@jest/test-sequencer": "30.3.0", + "@jest/types": "30.3.0", + "babel-jest": "30.3.0", + "chalk": "^4.1.2", + "ci-info": "^4.2.0", + "deepmerge": "^4.3.1", + "glob": "^10.5.0", + "graceful-fs": "^4.2.11", + "jest-circus": "30.3.0", + "jest-docblock": "30.2.0", + "jest-environment-node": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-resolve": "30.3.0", + "jest-runner": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "parse-json": "^5.2.0", + "pretty-format": "30.3.0", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "@types/node": "*", + "esbuild-register": ">=3.4.0", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "esbuild-register": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-config/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-config/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-config/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-diff": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/diff-sequences": "30.3.0", + "@jest/get-type": "30.1.0", + "chalk": "^4.1.2", + "pretty-format": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-diff/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-diff/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-diff/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-docblock": { + "version": "30.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "detect-newline": "^3.1.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-each": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/get-type": "30.1.0", + "@jest/types": "30.3.0", + "chalk": "^4.1.2", + "jest-util": "30.3.0", + "pretty-format": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-each/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-each/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-each/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-environment-jsdom": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/environment-jsdom-abstract": "30.3.0", + "jsdom": "^26.1.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "peerDependencies": { + "canvas": "^3.0.0" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/jest-environment-node": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/fake-timers": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "jest-mock": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "30.3.0", + "@types/node": "*", + "anymatch": "^3.1.3", + "fb-watchman": "^2.0.2", + "graceful-fs": "^4.2.11", + "jest-regex-util": "30.0.1", + "jest-util": "30.3.0", + "jest-worker": "30.3.0", + "picomatch": "^4.0.3", + "walker": "^1.0.8" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.3" + } + }, + "node_modules/jest-leak-detector": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/get-type": "30.1.0", + "pretty-format": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-leak-detector/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-leak-detector/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-leak-detector/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-matcher-utils": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/get-type": "30.1.0", + "chalk": "^4.1.2", + "jest-diff": "30.3.0", + "pretty-format": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-matcher-utils/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-message-util": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@jest/types": "30.3.0", + "@types/stack-utils": "^2.0.3", + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "picomatch": "^4.0.3", + "pretty-format": "30.3.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.6" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-message-util/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-message-util/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-message-util/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-mock": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "30.3.0", + "@types/node": "*", + "jest-util": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "30.0.1", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "jest-pnp-resolver": "^1.2.3", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "slash": "^3.0.0", + "unrs-resolver": "^1.7.11" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-regex-util": "30.0.1", + "jest-snapshot": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-runner": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "30.3.0", + "@jest/environment": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "chalk": "^4.1.2", + "emittery": "^0.13.1", + "exit-x": "^0.2.2", + "graceful-fs": "^4.2.11", + "jest-docblock": "30.2.0", + "jest-environment-node": "30.3.0", + "jest-haste-map": "30.3.0", + "jest-leak-detector": "30.3.0", + "jest-message-util": "30.3.0", + "jest-resolve": "30.3.0", + "jest-runtime": "30.3.0", + "jest-util": "30.3.0", + "jest-watcher": "30.3.0", + "jest-worker": "30.3.0", + "p-limit": "^3.1.0", + "source-map-support": "0.5.13" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-runtime": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/fake-timers": "30.3.0", + "@jest/globals": "30.3.0", + "@jest/source-map": "30.0.1", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "chalk": "^4.1.2", + "cjs-module-lexer": "^2.1.0", + "collect-v8-coverage": "^1.0.2", + "glob": "^10.5.0", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "jest-message-util": "30.3.0", + "jest-mock": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-resolve": "30.3.0", + "jest-snapshot": "30.3.0", + "jest-util": "30.3.0", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-snapshot": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.27.4", + "@babel/generator": "^7.27.5", + "@babel/plugin-syntax-jsx": "^7.27.1", + "@babel/plugin-syntax-typescript": "^7.27.1", + "@babel/types": "^7.27.3", + "@jest/expect-utils": "30.3.0", + "@jest/get-type": "30.1.0", + "@jest/snapshot-utils": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "babel-preset-current-node-syntax": "^1.2.0", + "chalk": "^4.1.2", + "expect": "30.3.0", + "graceful-fs": "^4.2.11", + "jest-diff": "30.3.0", + "jest-matcher-utils": "30.3.0", + "jest-message-util": "30.3.0", + "jest-util": "30.3.0", + "pretty-format": "30.3.0", + "semver": "^7.7.2", + "synckit": "^0.11.8" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-snapshot/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-snapshot/node_modules/semver": { + "version": "7.7.4", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jest-util": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "30.3.0", + "@types/node": "*", + "chalk": "^4.1.2", + "ci-info": "^4.2.0", + "graceful-fs": "^4.2.11", + "picomatch": "^4.0.3" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-validate": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/get-type": "30.1.0", + "@jest/types": "30.3.0", + "camelcase": "^6.3.0", + "chalk": "^4.1.2", + "leven": "^3.1.0", + "pretty-format": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-validate/node_modules/ansi-styles": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-validate/node_modules/camelcase": { + "version": "6.3.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-validate/node_modules/pretty-format": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-validate/node_modules/react-is": { + "version": "18.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-watcher": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "ansi-escapes": "^4.3.2", + "chalk": "^4.1.2", + "emittery": "^0.13.1", + "jest-util": "30.3.0", + "string-length": "^4.0.2" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-worker": { + "version": "30.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "@ungap/structured-clone": "^1.3.0", + "jest-util": "30.3.0", + "merge-stream": "^2.0.0", + "supports-color": "^8.1.1" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "8.1.1", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "3.14.2", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsdom": { + "version": "26.1.0", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-26.1.0.tgz", + "integrity": "sha512-Cvc9WUhxSMEo4McES3P7oK3QaXldCfNWp7pl2NNeiIFlCoLr3kfq9kb1fxftiwk1FLV7CvpvDfonxtzUDeSOPg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssstyle": "^4.2.1", + "data-urls": "^5.0.0", + "decimal.js": "^10.5.0", + "html-encoding-sniffer": "^4.0.0", + "http-proxy-agent": "^7.0.2", + "https-proxy-agent": "^7.0.6", + "is-potential-custom-element-name": "^1.0.1", + "nwsapi": "^2.2.16", + "parse5": "^7.2.1", + "rrweb-cssom": "^0.8.0", + "saxes": "^6.0.0", + "symbol-tree": "^3.2.4", + "tough-cookie": "^5.1.1", + "w3c-xmlserializer": "^5.0.0", + "webidl-conversions": "^7.0.0", + "whatwg-encoding": "^3.1.1", + "whatwg-mimetype": "^4.0.0", + "whatwg-url": "^14.1.1", + "ws": "^8.18.0", + "xml-name-validator": "^5.0.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "canvas": "^3.0.0" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/jsdom/node_modules/whatwg-mimetype": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", + "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "dev": true, + "license": "MIT" + }, + "node_modules/json5": { + "version": "2.2.3", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/leven": { + "version": "3.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/lightningcss": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", + "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==", + "dev": true, + "license": "MPL-2.0", + "optional": true, + "peer": true, + "dependencies": { + "detect-libc": "^2.0.3" + }, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "lightningcss-android-arm64": "1.32.0", + "lightningcss-darwin-arm64": "1.32.0", + "lightningcss-darwin-x64": "1.32.0", + "lightningcss-freebsd-x64": "1.32.0", + "lightningcss-linux-arm-gnueabihf": "1.32.0", + "lightningcss-linux-arm64-gnu": "1.32.0", + "lightningcss-linux-arm64-musl": "1.32.0", + "lightningcss-linux-x64-gnu": "1.32.0", + "lightningcss-linux-x64-musl": "1.32.0", + "lightningcss-win32-arm64-msvc": "1.32.0", + "lightningcss-win32-x64-msvc": "1.32.0" + } + }, + "node_modules/lightningcss-android-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz", + "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "android" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz", + "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz", + "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz", + "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "freebsd" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz", + "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz", + "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz", + "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz", + "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz", + "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-arm64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz", + "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz", + "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "peer": true, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "dev": true, + "license": "MIT" + }, + "node_modules/locate-path": { + "version": "5.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lodash.debounce": { + "version": "4.0.8", + "dev": true, + "license": "MIT" + }, + "node_modules/loupe": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.2.1.tgz", + "integrity": "sha512-CdzqowRJCeLU72bHvWqwRBBlLcMEtIvGrlvef74kMnV2AolS9Y8xUv1I0U/MNAWMhBlKIoyuEgoJ0t/bbwHbLQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/lz-string": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", + "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "lz-string": "bin/bin.js" + } + }, + "node_modules/magic-string": { + "version": "0.30.21", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", + "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.5" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-dir/node_modules/semver": { + "version": "7.7.4", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/makeerror": { + "version": "1.0.12", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/minimatch": { + "version": "9.0.9", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.2" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minipass": { + "version": "7.1.3", + "dev": true, + "license": "BlueOak-1.0.0", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "dev": true, + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.11", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/napi-postinstall": { + "version": "0.3.4", + "dev": true, + "license": "MIT", + "bin": { + "napi-postinstall": "lib/cli.js" + }, + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/napi-postinstall" + } + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "dev": true, + "license": "MIT" + }, + "node_modules/node-int64": { + "version": "0.4.0", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.36", + "dev": true, + "license": "MIT" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/nwsapi": { + "version": "2.2.23", + "dev": true, + "license": "MIT" + }, + "node_modules/once": { + "version": "1.4.0", + "dev": true, + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "dev": true, + "license": "MIT", + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "4.1.0", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-locate/node_modules/p-limit": { + "version": "2.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/package-json-from-dist": { + "version": "1.0.1", + "dev": true, + "license": "BlueOak-1.0.0" + }, + "node_modules/parse-json": { + "version": "5.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parse5": { + "version": "7.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "entities": "^6.0.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, + "node_modules/parse5/node_modules/entities": { + "version": "6.0.1", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "dev": true, + "license": "MIT" + }, + "node_modules/path-scurry": { + "version": "1.11.1", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^10.2.0", + "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" + }, + "engines": { + "node": ">=16 || 14 >=14.18" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/path-scurry/node_modules/lru-cache": { + "version": "10.4.3", + "dev": true, + "license": "ISC" + }, + "node_modules/pathe": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz", + "integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==", + "dev": true, + "license": "MIT" + }, + "node_modules/pathval": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pathval/-/pathval-2.0.1.tgz", + "integrity": "sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14.16" + } + }, + "node_modules/picocolors": { + "version": "1.1.1", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "4.0.4", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "version": "4.0.7", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/poku": { + "version": "4.2.0", + "dev": true, + "license": "MIT", + "bin": { + "poku": "lib/bin/index.js" + }, + "engines": { + "bun": ">=1.x.x", + "deno": ">=2.x.x", + "node": ">=16.x.x", + "typescript": ">=5.x.x" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wellwelwel" + } + }, + "node_modules/poku-react-testing": { + "resolved": "..", + "link": true + }, + "node_modules/postcss": { + "version": "8.5.8", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "nanoid": "^3.3.11", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "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, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/pure-rand": { + "version": "7.0.1", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/dubzzz" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fast-check" + } + ], + "license": "MIT" + }, + "node_modules/react": { + "version": "19.2.4", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "19.2.4", + "dev": true, + "license": "MIT", + "dependencies": { + "scheduler": "^0.27.0" + }, + "peerDependencies": { + "react": "^19.2.4" + } + }, + "node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "license": "MIT", + "peer": true + }, + "node_modules/react-refresh": { + "version": "0.17.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.17.0.tgz", + "integrity": "sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/regenerate": { + "version": "1.4.2", + "dev": true, + "license": "MIT" + }, + "node_modules/regenerate-unicode-properties": { + "version": "10.2.2", + "dev": true, + "license": "MIT", + "dependencies": { + "regenerate": "^1.4.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regexpu-core": { + "version": "6.4.0", + "dev": true, + "license": "MIT", + "dependencies": { + "regenerate": "^1.4.2", + "regenerate-unicode-properties": "^10.2.2", + "regjsgen": "^0.8.0", + "regjsparser": "^0.13.0", + "unicode-match-property-ecmascript": "^2.0.0", + "unicode-match-property-value-ecmascript": "^2.2.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regjsgen": { + "version": "0.8.0", + "dev": true, + "license": "MIT" + }, + "node_modules/regjsparser": { + "version": "0.13.0", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "jsesc": "~3.1.0" + }, + "bin": { + "regjsparser": "bin/parser" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.11", + "dev": true, + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-from": { + "version": "5.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/rollup": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.1.tgz", + "integrity": "sha512-VmtB2rFU/GroZ4oL8+ZqXgSA38O6GR8KSIvWmEFv63pQ0G6KaBH9s07PO8XTXP4vI+3UJUEypOfjkGfmSBBR0w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.8" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.60.1", + "@rollup/rollup-android-arm64": "4.60.1", + "@rollup/rollup-darwin-arm64": "4.60.1", + "@rollup/rollup-darwin-x64": "4.60.1", + "@rollup/rollup-freebsd-arm64": "4.60.1", + "@rollup/rollup-freebsd-x64": "4.60.1", + "@rollup/rollup-linux-arm-gnueabihf": "4.60.1", + "@rollup/rollup-linux-arm-musleabihf": "4.60.1", + "@rollup/rollup-linux-arm64-gnu": "4.60.1", + "@rollup/rollup-linux-arm64-musl": "4.60.1", + "@rollup/rollup-linux-loong64-gnu": "4.60.1", + "@rollup/rollup-linux-loong64-musl": "4.60.1", + "@rollup/rollup-linux-ppc64-gnu": "4.60.1", + "@rollup/rollup-linux-ppc64-musl": "4.60.1", + "@rollup/rollup-linux-riscv64-gnu": "4.60.1", + "@rollup/rollup-linux-riscv64-musl": "4.60.1", + "@rollup/rollup-linux-s390x-gnu": "4.60.1", + "@rollup/rollup-linux-x64-gnu": "4.60.1", + "@rollup/rollup-linux-x64-musl": "4.60.1", + "@rollup/rollup-openbsd-x64": "4.60.1", + "@rollup/rollup-openharmony-arm64": "4.60.1", + "@rollup/rollup-win32-arm64-msvc": "4.60.1", + "@rollup/rollup-win32-ia32-msvc": "4.60.1", + "@rollup/rollup-win32-x64-gnu": "4.60.1", + "@rollup/rollup-win32-x64-msvc": "4.60.1", + "fsevents": "~2.3.2" + } + }, + "node_modules/rrweb-cssom": { + "version": "0.8.0", + "dev": true, + "license": "MIT" + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "dev": true, + "license": "MIT" + }, + "node_modules/saxes": { + "version": "6.0.0", + "dev": true, + "license": "ISC", + "dependencies": { + "xmlchars": "^2.2.0" + }, + "engines": { + "node": ">=v12.22.7" + } + }, + "node_modules/scheduler": { + "version": "0.27.0", + "dev": true, + "license": "MIT" + }, + "node_modules/semver": { + "version": "6.3.1", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/siginfo": { + "version": "2.0.0", + "dev": true, + "license": "ISC" + }, + "node_modules/signal-exit": { + "version": "4.1.0", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/slash": { + "version": "3.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map": { + "version": "0.6.1", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.13", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/stack-utils": { + "version": "2.0.6", + "dev": true, + "license": "MIT", + "dependencies": { + "escape-string-regexp": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/stackback": { + "version": "0.0.2", + "dev": true, + "license": "MIT" + }, + "node_modules/std-env": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.10.0.tgz", + "integrity": "sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==", + "dev": true, + "license": "MIT" + }, + "node_modules/string-length": { + "version": "4.0.2", + "dev": true, + "license": "MIT", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/string-length/node_modules/strip-ansi": { + "version": "6.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width": { + "version": "5.1.2", + "dev": true, + "license": "MIT", + "dependencies": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/string-width-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "7.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^6.2.2" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" + } + }, + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi/node_modules/ansi-regex": { + "version": "6.2.2", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" + } + }, + "node_modules/strip-bom": { + "version": "4.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/strip-literal": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/strip-literal/-/strip-literal-3.1.0.tgz", + "integrity": "sha512-8r3mkIM/2+PpjHoOtiAW8Rg3jJLHaV7xPwG+YRGrv6FP0wwk/toTpATxWYOW0BKdWwl82VT2tFYi5DlROa0Mxg==", + "dev": true, + "license": "MIT", + "dependencies": { + "js-tokens": "^9.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/antfu" + } + }, + "node_modules/strip-literal/node_modules/js-tokens": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz", + "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/supports-color": { + "version": "7.2.0", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/symbol-tree": { + "version": "3.2.4", + "dev": true, + "license": "MIT" + }, + "node_modules/synckit": { + "version": "0.11.12", + "dev": true, + "license": "MIT", + "dependencies": { + "@pkgr/core": "^0.2.9" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/synckit" + } + }, + "node_modules/test-exclude": { + "version": "6.0.0", + "dev": true, + "license": "ISC", + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/test-exclude/node_modules/brace-expansion": { + "version": "1.1.13", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/test-exclude/node_modules/glob": { + "version": "7.2.3", + "dev": true, + "license": "ISC", + "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/test-exclude/node_modules/minimatch": { + "version": "3.1.5", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/tinybench": { + "version": "2.9.0", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyexec": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz", + "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyglobby": { + "version": "0.2.15", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.3" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "node_modules/tinypool": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/tinypool/-/tinypool-1.1.1.tgz", + "integrity": "sha512-Zba82s87IFq9A9XmjiX5uZA/ARWDrB03OHlq+Vw1fSdt0I+4/Kutwy8BP4Y/y/aORMo61FQ0vIb5j44vSo5Pkg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.0.0 || >=20.0.0" + } + }, + "node_modules/tinyrainbow": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-2.0.0.tgz", + "integrity": "sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tinyspy": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/tinyspy/-/tinyspy-4.0.4.tgz", + "integrity": "sha512-azl+t0z7pw/z958Gy9svOTuzqIk6xq+NSheJzn5MMWtWTFywIacg2wUlzKFGtt3cthx0r2SxMK0yzJOR0IES7Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tldts": { + "version": "6.1.86", + "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.86.tgz", + "integrity": "sha512-WMi/OQ2axVTf/ykqCQgXiIct+mSQDFdH2fkwhPwgEwvJ1kSzZRiinb0zF2Xb8u4+OqPChmyI6MEu4EezNJz+FQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "tldts-core": "^6.1.86" + }, + "bin": { + "tldts": "bin/cli.js" + } + }, + "node_modules/tldts-core": { + "version": "6.1.86", + "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-6.1.86.tgz", + "integrity": "sha512-Je6p7pkk+KMzMv2XXKmAE3McmolOQFdxkKw0R8EYNr7sELW46JqnNeTX8ybPiQgvg1ymCoF8LXs5fzFaZvJPTA==", + "dev": true, + "license": "MIT" + }, + "node_modules/tmpl": { + "version": "1.0.5", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/tough-cookie": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-5.1.2.tgz", + "integrity": "sha512-FVDYdxtnj0G6Qm/DhNPSb8Ju59ULcup3tuJxkFb5K8Bv2pUXILbf0xZWU8PX8Ov19OXljbUyveOFwRMwkXzO+A==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tldts": "^6.1.32" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/tr46": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-5.1.1.tgz", + "integrity": "sha512-hdF5ZgjTqgAntKkklYw0R03MG2x/bSzTtkxmIRw/sTNV8YXsCJ1tfLAX23lhxhHJlEf3CRCOCGGWw3vI3GaSPw==", + "dev": true, + "license": "MIT", + "dependencies": { + "punycode": "^2.3.1" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/type-detect": { + "version": "4.0.8", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/type-fest": { + "version": "0.21.3", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/undici-types": { + "version": "7.18.2", + "dev": true, + "license": "MIT" + }, + "node_modules/unicode-canonical-property-names-ecmascript": { + "version": "2.0.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-match-property-ecmascript": { + "version": "2.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "unicode-canonical-property-names-ecmascript": "^2.0.0", + "unicode-property-aliases-ecmascript": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-match-property-value-ecmascript": { + "version": "2.2.1", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-property-aliases-ecmascript": { + "version": "2.2.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/unrs-resolver": { + "version": "1.11.1", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "dependencies": { + "napi-postinstall": "^0.3.0" + }, + "funding": { + "url": "https://opencollective.com/unrs-resolver" + }, + "optionalDependencies": { + "@unrs/resolver-binding-android-arm-eabi": "1.11.1", + "@unrs/resolver-binding-android-arm64": "1.11.1", + "@unrs/resolver-binding-darwin-arm64": "1.11.1", + "@unrs/resolver-binding-darwin-x64": "1.11.1", + "@unrs/resolver-binding-freebsd-x64": "1.11.1", + "@unrs/resolver-binding-linux-arm-gnueabihf": "1.11.1", + "@unrs/resolver-binding-linux-arm-musleabihf": "1.11.1", + "@unrs/resolver-binding-linux-arm64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-arm64-musl": "1.11.1", + "@unrs/resolver-binding-linux-ppc64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-riscv64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-riscv64-musl": "1.11.1", + "@unrs/resolver-binding-linux-s390x-gnu": "1.11.1", + "@unrs/resolver-binding-linux-x64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-x64-musl": "1.11.1", + "@unrs/resolver-binding-wasm32-wasi": "1.11.1", + "@unrs/resolver-binding-win32-arm64-msvc": "1.11.1", + "@unrs/resolver-binding-win32-ia32-msvc": "1.11.1", + "@unrs/resolver-binding-win32-x64-msvc": "1.11.1" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.2.3", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/v8-to-istanbul": { + "version": "9.3.0", + "dev": true, + "license": "ISC", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.12", + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^2.0.0" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/vite": { + "version": "6.4.1", + "resolved": "https://registry.npmjs.org/vite/-/vite-6.4.1.tgz", + "integrity": "sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.25.0", + "fdir": "^6.4.4", + "picomatch": "^4.0.2", + "postcss": "^8.5.3", + "rollup": "^4.34.9", + "tinyglobby": "^0.2.13" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", + "jiti": ">=1.21.0", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "sass-embedded": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/vite-node": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/vite-node/-/vite-node-3.2.4.tgz", + "integrity": "sha512-EbKSKh+bh1E1IFxeO0pg1n4dvoOTt0UDiXMd/qn++r98+jPO1xtJilvXldeuQ8giIB5IkpjCgMleHMNEsGH6pg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cac": "^6.7.14", + "debug": "^4.4.1", + "es-module-lexer": "^1.7.0", + "pathe": "^2.0.3", + "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0" + }, + "bin": { + "vite-node": "vite-node.mjs" + }, + "engines": { + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/vitest": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/vitest/-/vitest-3.2.4.tgz", + "integrity": "sha512-LUCP5ev3GURDysTWiP47wRRUpLKMOfPh+yKTx3kVIEiu5KOMeqzpnYNsKyOoVrULivR8tLcks4+lga33Whn90A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/chai": "^5.2.2", + "@vitest/expect": "3.2.4", + "@vitest/mocker": "3.2.4", + "@vitest/pretty-format": "^3.2.4", + "@vitest/runner": "3.2.4", + "@vitest/snapshot": "3.2.4", + "@vitest/spy": "3.2.4", + "@vitest/utils": "3.2.4", + "chai": "^5.2.0", + "debug": "^4.4.1", + "expect-type": "^1.2.1", + "magic-string": "^0.30.17", + "pathe": "^2.0.3", + "picomatch": "^4.0.2", + "std-env": "^3.9.0", + "tinybench": "^2.9.0", + "tinyexec": "^0.3.2", + "tinyglobby": "^0.2.14", + "tinypool": "^1.1.1", + "tinyrainbow": "^2.0.0", + "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0", + "vite-node": "3.2.4", + "why-is-node-running": "^2.3.0" + }, + "bin": { + "vitest": "vitest.mjs" + }, + "engines": { + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@edge-runtime/vm": "*", + "@types/debug": "^4.1.12", + "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", + "@vitest/browser": "3.2.4", + "@vitest/ui": "3.2.4", + "happy-dom": "*", + "jsdom": "*" + }, + "peerDependenciesMeta": { + "@edge-runtime/vm": { + "optional": true + }, + "@types/debug": { + "optional": true + }, + "@types/node": { + "optional": true + }, + "@vitest/browser": { + "optional": true + }, + "@vitest/ui": { + "optional": true + }, + "happy-dom": { + "optional": true + }, + "jsdom": { + "optional": true + } + } + }, + "node_modules/w3c-xmlserializer": { + "version": "5.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "xml-name-validator": "^5.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/walker": { + "version": "1.0.8", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "makeerror": "1.0.12" + } + }, + "node_modules/webidl-conversions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", + "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + } + }, + "node_modules/whatwg-encoding": { + "version": "3.1.1", + "dev": true, + "license": "MIT", + "dependencies": { + "iconv-lite": "0.6.3" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/whatwg-mimetype": { + "version": "3.0.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/whatwg-url": { + "version": "14.2.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.2.0.tgz", + "integrity": "sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tr46": "^5.1.0", + "webidl-conversions": "^7.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/which": { + "version": "2.0.2", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/why-is-node-running": { + "version": "2.3.0", + "dev": true, + "license": "MIT", + "dependencies": { + "siginfo": "^2.0.0", + "stackback": "0.0.2" + }, + "bin": { + "why-is-node-running": "cli.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi": { + "version": "8.1.0", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^6.1.0", + "string-width": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/wrap-ansi-cjs/node_modules/string-width": { + "version": "4.2.3", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-styles": { + "version": "6.2.3", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "dev": true, + "license": "ISC" + }, + "node_modules/write-file-atomic": { + "version": "5.0.1", + "dev": true, + "license": "ISC", + "dependencies": { + "imurmurhash": "^0.1.4", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": "^14.17.0 || ^16.13.0 || >=18.0.0" + } + }, + "node_modules/ws": { + "version": "8.20.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/xml-name-validator": { + "version": "5.0.0", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18" + } + }, + "node_modules/xmlchars": { + "version": "2.2.0", + "dev": true, + "license": "MIT" + }, + "node_modules/y18n": { + "version": "5.0.8", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "dev": true, + "license": "ISC" + }, + "node_modules/yargs": { + "version": "17.7.2", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs/node_modules/emoji-regex": { + "version": "8.0.0", + "dev": true, + "license": "MIT" + }, + "node_modules/yargs/node_modules/string-width": { + "version": "4.2.3", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/strip-ansi": { + "version": "6.0.1", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} diff --git a/benchmark/package.json b/benchmark/package.json new file mode 100644 index 0000000..1b98bf6 --- /dev/null +++ b/benchmark/package.json @@ -0,0 +1,29 @@ +{ + "name": "poku-react-benchmark", + "version": "1.0.0", + "private": true, + "type": "module", + "scripts": { + "setup": "npm install", + "test:jest": "node node_modules/.bin/jest --config jest.config.mjs", + "test:vitest:jsdom": "node node_modules/.bin/vitest run --config vitest.jsdom.config.js", + "test:vitest:happy": "node node_modules/.bin/vitest run --config vitest.happy.config.js" + }, + "devDependencies": { + "@babel/core": "^7.29.0", + "@babel/preset-env": "^7.29.2", + "@babel/preset-react": "^7.28.5", + "@testing-library/react": "^16.3.2", + "@vitejs/plugin-react": "^4.7.0", + "babel-jest": "^30.3.0", + "happy-dom": "^20.8.9", + "jest": "^30.3.0", + "jest-environment-jsdom": "^30.3.0", + "jsdom": "26.1.0", + "poku": "4.2.0", + "poku-react-testing": "file:..", + "react": "^19.2.4", + "react-dom": "^19.2.4", + "vitest": "^3.2.4" + } +} diff --git a/benchmark/poku.config.js b/benchmark/poku.config.js new file mode 100644 index 0000000..03ae003 --- /dev/null +++ b/benchmark/poku.config.js @@ -0,0 +1,9 @@ +import { defineConfig } from 'poku'; +import { reactTestingPlugin } from 'poku-react-testing/plugin'; + +const configuredDom = process.env.POKU_REACT_TEST_DOM; +const dom = configuredDom === 'jsdom' ? 'jsdom' : 'happy-dom'; + +export default defineConfig({ + plugins: [reactTestingPlugin({ dom })], +}); diff --git a/benchmark/results.json b/benchmark/results.json new file mode 100644 index 0000000..1402af6 --- /dev/null +++ b/benchmark/results.json @@ -0,0 +1,81 @@ +{ + "timestamp": "2026-03-31T21:00:50.386Z", + "system": { + "nodeVersion": "v22.5.1", + "platform": "darwin 25.4.0", + "cpuModel": "Apple M3 Pro", + "cpuCount": 12, + "totalMemGb": "18.0" + }, + "config": { + "runs": 7, + "trim": 1 + }, + "results": [ + { + "label": "poku + happy-dom", + "mean": 1073.1924668, + "min": 995.5031250000002, + "max": 1229.6369170000003, + "stddev": 84.66710558647982, + "meanRss": 171232460.8, + "meanUserCpu": 4396, + "meanSysCpu": 1584, + "runs": 7, + "validRuns": 7, + "failures": 0 + }, + { + "label": "poku + jsdom", + "mean": 1060.1384749999997, + "min": 1007.1474579999995, + "max": 1176.5342500000006, + "stddev": 60.012456221527664, + "meanRss": 171294720, + "meanUserCpu": 3228, + "meanSysCpu": 1440, + "runs": 7, + "validRuns": 7, + "failures": 0 + }, + { + "label": "jest + jsdom", + "mean": 858.7310334000001, + "min": 778.734042, + "max": 929.1150420000013, + "stddev": 50.38267849622563, + "meanRss": 216186880, + "meanUserCpu": 778, + "meanSysCpu": 176, + "runs": 7, + "validRuns": 7, + "failures": 0 + }, + { + "label": "vitest + jsdom", + "mean": 964.3196584000004, + "min": 949.5501660000009, + "max": 986.9945420000004, + "stddev": 16.60550682561403, + "meanRss": 155179417.6, + "meanUserCpu": 3336, + "meanSysCpu": 1102, + "runs": 7, + "validRuns": 7, + "failures": 0 + }, + { + "label": "vitest + happy-dom", + "mean": 837.8697498000001, + "min": 812.1916249999995, + "max": 864.1250409999993, + "stddev": 20.924101623413893, + "meanRss": 121916620.8, + "meanUserCpu": 2942, + "meanSysCpu": 1006, + "runs": 7, + "validRuns": 7, + "failures": 0 + } + ] +} \ No newline at end of file diff --git a/benchmark/run.mjs b/benchmark/run.mjs new file mode 100644 index 0000000..ae076d3 --- /dev/null +++ b/benchmark/run.mjs @@ -0,0 +1,592 @@ +#!/usr/bin/env node +/** + * React Testing Framework Benchmark + * Compares: poku+poku-react-testing vs jest+RTL vs vitest+RTL + * + * Metrics: wall-clock time, user CPU, sys CPU, peak RSS (macOS /usr/bin/time -l) + * Strategy: RUNS per scenario, drop 1 slowest + 1 fastest, report trimmed mean. + * + * NOTE ON k6: k6 is an HTTP/API load-testing tool; it has no mechanism to execute + * test-framework processes or measure startup/runtime overhead. It is not applicable + * to this class of benchmark. + */ +import { spawnSync } from 'child_process'; +import { + copyFileSync, + existsSync, + mkdirSync, + readdirSync, + rmSync, + writeFileSync, +} from 'fs'; +import os from 'os'; +import { dirname, join, resolve } from 'path'; +import { performance } from 'perf_hooks'; +import { fileURLToPath } from 'url'; + +const __dirname = dirname(fileURLToPath(import.meta.url)); +const ROOT = resolve(__dirname, '..'); +const BENCH = __dirname; + +const RUNS = parseInt(process.env.BENCH_RUNS ?? '7', 10); +const TRIM = 1; // drop 1 slowest + 1 fastest per scenario +const POKU_BIN = join(BENCH, 'node_modules', 'poku', 'lib', 'bin', 'index.js'); + +// ─── local build sync ────────────────────────────────────────────────────── + +/** + * Replace the benchmark's poku-react-testing installation with a clean copy + * of the local dist/ so that Node resolves react/react-dom from + * benchmark/node_modules instead of following a symlink to the workspace root. + */ +function syncLocalBuild() { + const dest = join(BENCH, 'node_modules', 'poku-react-testing'); + rmSync(dest, { recursive: true, force: true }); + mkdirSync(join(dest, 'dist'), { recursive: true }); + copyFileSync(join(ROOT, 'package.json'), join(dest, 'package.json')); + for (const file of readdirSync(join(ROOT, 'dist'))) { + copyFileSync(join(ROOT, 'dist', file), join(dest, 'dist', file)); + } +} + +// ─── helpers ─────────────────────────────────────────────────────────────── + +function mb(bytes) { + return bytes == null ? 'N/A' : (bytes / 1024 / 1024).toFixed(1); +} +function sec(ms) { + return ms == null ? 'N/A' : (ms / 1000).toFixed(3); +} +function pct(base, val) { + if (base == null || val == null) return ''; + const diff = ((val - base) / base) * 100; + return diff >= 0 ? `+${diff.toFixed(0)}%` : `${diff.toFixed(0)}%`; +} + +function pad(str, n) { + return String(str).padEnd(n); +} +function rpad(str, n) { + return String(str).padStart(n); +} + +// ─── single run measurement ───────────────────────────────────────────────── + +function runOnce(args, cwd, extraEnv = {}) { + const env = { ...process.env, ...extraEnv, FORCE_COLOR: '0' }; + const useTime = process.platform === 'darwin' && existsSync('/usr/bin/time'); + + const [cmd, cmdArgs] = useTime + ? ['/usr/bin/time', ['-l', ...args]] + : [args[0], args.slice(1)]; + + const t0 = performance.now(); + const result = spawnSync(cmd, cmdArgs, { + cwd, + env, + encoding: 'utf8', + stdio: ['ignore', 'pipe', 'pipe'], + maxBuffer: 10 * 1024 * 1024, + }); + const elapsed = performance.now() - t0; + + let maxRssBytes = null; + let userCpuMs = null; + let sysCpuMs = null; + + if (useTime) { + const stderr = result.stderr ?? ''; + // macOS /usr/bin/time -l format: + // 0.71 real 0.65 user 0.22 sys + // 3014656 maximum resident set size + const rssMatch = stderr.match(/(\d+)\s+maximum resident set size/); + if (rssMatch) maxRssBytes = parseInt(rssMatch[1], 10); + + const timeMatch = stderr.match( + /([\d.]+)\s+real\s+([\d.]+)\s+user\s+([\d.]+)\s+sys/ + ); + if (timeMatch) { + userCpuMs = parseFloat(timeMatch[2]) * 1000; + sysCpuMs = parseFloat(timeMatch[3]) * 1000; + } + } + + return { + elapsed, + maxRssBytes, + userCpuMs, + sysCpuMs, + exitCode: result.status ?? 1, + failed: result.status !== 0, + stdout: result.stdout ?? '', + stderr: result.stderr ?? '', + }; +} + +// ─── scenario runner ──────────────────────────────────────────────────────── + +function benchmarkScenario(label, args, cwd, extraEnv = {}) { + process.stdout.write(` ${pad(label, 36)}`); + + const measurements = []; + let failures = 0; + + for (let i = 0; i < RUNS; i++) { + const m = runOnce(args, cwd, extraEnv); + if (m.failed) { + failures++; + process.stdout.write('✗'); + } else { + measurements.push(m); + process.stdout.write('·'); + } + } + + const needed = RUNS - TRIM * 2; + if (measurements.length < needed) { + process.stdout.write(` FAILED (${failures}/${RUNS} runs failed)\n`); + // Print stderr of first failure for diagnosis + const lastRun = runOnce(args, cwd, extraEnv); + if (lastRun.failed) { + const diagStderr = lastRun.stderr.slice(-800).trim(); + const diagStdout = lastRun.stdout.slice(-400).trim(); + if (diagStderr) console.error('\n stderr:', diagStderr); + if (diagStdout) console.error('\n stdout:', diagStdout); + } + return null; + } + + measurements.sort((a, b) => a.elapsed - b.elapsed); + const trimmed = measurements.slice(TRIM, measurements.length - TRIM); + + const times = trimmed.map((m) => m.elapsed); + const rssList = trimmed.map((m) => m.maxRssBytes).filter((v) => v != null); + const userCpus = trimmed.map((m) => m.userCpuMs).filter((v) => v != null); + const sysCpus = trimmed.map((m) => m.sysCpuMs).filter((v) => v != null); + + const mean = times.reduce((a, b) => a + b, 0) / times.length; + const min = times[0]; + const max = times[times.length - 1]; + const variance = + times.reduce((a, t) => a + (t - mean) ** 2, 0) / times.length; + const stddev = Math.sqrt(variance); + + const meanRss = + rssList.length > 0 + ? rssList.reduce((a, b) => a + b, 0) / rssList.length + : null; + const meanUserCpu = + userCpus.length > 0 + ? userCpus.reduce((a, b) => a + b, 0) / userCpus.length + : null; + const meanSysCpu = + sysCpus.length > 0 + ? sysCpus.reduce((a, b) => a + b, 0) / sysCpus.length + : null; + + process.stdout.write(` ${sec(mean)}s\n`); + + return { + label, + mean, + min, + max, + stddev, + meanRss, + meanUserCpu, + meanSysCpu, + runs: RUNS, + validRuns: measurements.length, + failures, + }; +} + +// ─── scenario definitions ─────────────────────────────────────────────────── + +const scenarios = [ + { + label: 'poku + happy-dom', + args: ['node', POKU_BIN, 'tests/poku'], + cwd: BENCH, + env: { POKU_REACT_TEST_DOM: 'happy-dom' }, + }, + { + label: 'poku + jsdom', + args: ['node', POKU_BIN, 'tests/poku'], + cwd: BENCH, + env: { POKU_REACT_TEST_DOM: 'jsdom' }, + }, + { + label: 'jest + jsdom', + args: [ + 'node', + join(BENCH, 'node_modules', '.bin', 'jest'), + '--config', + join(BENCH, 'jest.config.mjs'), + ], + cwd: BENCH, + env: {}, + }, + { + label: 'vitest + jsdom', + args: [ + join(BENCH, 'node_modules', '.bin', 'vitest'), + 'run', + '--config', + join(BENCH, 'vitest.jsdom.config.js'), + ], + cwd: BENCH, + env: {}, + }, + { + label: 'vitest + happy-dom', + args: [ + join(BENCH, 'node_modules', '.bin', 'vitest'), + 'run', + '--config', + join(BENCH, 'vitest.happy.config.js'), + ], + cwd: BENCH, + env: {}, + }, +]; + +// ─── pre-flight checks ─────────────────────────────────────────────────────── + +if (!existsSync(join(BENCH, 'node_modules'))) { + console.error('\n❌ benchmark/node_modules not found.'); + console.error(' Run: cd benchmark && npm install\n'); + process.exit(1); +} +if (!existsSync(join(BENCH, 'node_modules', 'poku'))) { + console.error('\n❌ benchmark/node_modules/poku not found.'); + console.error(' Run: cd benchmark && npm install\n'); + process.exit(1); +} + +// ─── sync local build ──────────────────────────────────────────────────────── + +if (existsSync(join(ROOT, 'dist'))) { + process.stdout.write(' Syncing local poku-react-testing build... '); + syncLocalBuild(); + process.stdout.write('done\n\n'); +} + +// ─── system info ──────────────────────────────────────────────────────────── + +const cpuModel = os.cpus()[0]?.model ?? 'unknown'; +const cpuCount = os.cpus().length; +const totalMemGb = (os.totalmem() / 1024 ** 3).toFixed(1); +const nodeVersion = process.version; +const platform = `${os.platform()} ${os.release()}`; + +// ─── run ───────────────────────────────────────────────────────────────────── + +console.log('\n╔══════════════════════════════════════════════════════╗'); +console.log('║ React Testing Framework Benchmark ║'); +console.log('╚══════════════════════════════════════════════════════╝\n'); +console.log(` Node: ${nodeVersion}`); +console.log(` Platform: ${platform}`); +console.log(` CPU: ${cpuModel} (${cpuCount} cores)`); +console.log(` RAM: ${totalMemGb} GB`); +console.log(` Runs: ${RUNS} per scenario (trim ±${TRIM})\n`); +console.log(' Scenarios:'); +scenarios.forEach((s) => console.log(` · ${s.label}`)); +console.log('\n Legend: · = pass ✗ = fail\n'); + +const results = []; + +for (const s of scenarios) { + const r = benchmarkScenario(s.label, s.args, s.cwd, s.env); + if (r) results.push(r); +} + +if (results.length === 0) { + console.error('\n❌ All scenarios failed. Check configuration.\n'); + process.exit(1); +} + +// ─── console table ─────────────────────────────────────────────────────────── + +const baseline = results[0]; // poku + happy-dom + +console.log( + '\n┌─────────────────────────────────────────────────────────────────────────────┐' +); +console.log( + '│ Results Summary │' +); +console.log( + '└─────────────────────────────────────────────────────────────────────────────┘\n' +); + +const header = [ + pad('Scenario', 22), + rpad('Mean', 8), + rpad('Min', 8), + rpad('Max', 8), + rpad('Stdev', 7), + rpad('RSS MB', 7), + rpad('vs poku+hd', 11), +].join(' '); + +console.log(' ' + header); +console.log(' ' + '-'.repeat(header.length)); + +for (const r of results) { + const row = [ + pad(r.label, 22), + rpad(sec(r.mean) + 's', 8), + rpad(sec(r.min) + 's', 8), + rpad(sec(r.max) + 's', 8), + rpad(sec(r.stddev) + 's', 7), + rpad(mb(r.meanRss), 7), + rpad( + r.label === baseline.label ? '(baseline)' : pct(baseline.mean, r.mean), + 11 + ), + ].join(' '); + console.log(' ' + row); +} + +console.log(); + +// ─── save results JSON ─────────────────────────────────────────────────────── + +const timestamp = new Date().toISOString(); +const jsonOut = { + timestamp, + system: { nodeVersion, platform, cpuModel, cpuCount, totalMemGb }, + config: { runs: RUNS, trim: TRIM }, + results, +}; + +const jsonPath = join(BENCH, 'results.json'); +writeFileSync(jsonPath, JSON.stringify(jsonOut, null, 2)); +console.log(` Results saved → benchmark/results.json`); + +// ─── generate REPORT.md ────────────────────────────────────────────────────── + +function formatTable(rows, headers) { + const widths = headers.map((h, i) => + Math.max(h.length, ...rows.map((r) => String(r[i]).length)) + ); + const hr = widths.map((w) => '-'.repeat(w + 2)).join('|'); + const fmt = (row) => + row.map((v, i) => ` ${String(v).padEnd(widths[i])} `).join('|'); + return [ + '|' + fmt(headers) + '|', + '|' + hr + '|', + ...rows.map((r) => '|' + fmt(r) + '|'), + ].join('\n'); +} + +const tableRows = results.map((r) => [ + r.label, + sec(r.mean) + 's', + sec(r.min) + 's', + sec(r.max) + 's', + sec(r.stddev) + 's', + mb(r.meanRss) === 'N/A' ? 'N/A' : mb(r.meanRss) + ' MB', + r.label === baseline.label ? '*(baseline)*' : pct(baseline.mean, r.mean), +]); + +const resultsTable = formatTable(tableRows, [ + 'Scenario', + 'Mean', + 'Min', + 'Max', + 'Stdev', + 'Peak RSS', + 'vs poku+happy-dom', +]); + +// Build per-metric analysis +function findResult(label) { + return results.find((r) => r.label === label); +} + +const pokuHappy = findResult('poku + happy-dom'); +const pokuJsdom = findResult('poku + jsdom'); +const jestJsdom = findResult('jest + jsdom'); +const vitestJsdom = findResult('vitest + jsdom'); +const vitestHappy = findResult('vitest + happy-dom'); + +const happyVsJsdomPoku = + pokuHappy && pokuJsdom + ? (((pokuJsdom.mean - pokuHappy.mean) / pokuHappy.mean) * 100).toFixed(0) + : null; + +const happyVsJsdomVitest = + vitestHappy && vitestJsdom + ? ( + ((vitestJsdom.mean - vitestHappy.mean) / vitestHappy.mean) * + 100 + ).toFixed(0) + : null; + +const pokuVsJest = + pokuHappy && jestJsdom + ? (((jestJsdom.mean - pokuHappy.mean) / pokuHappy.mean) * 100).toFixed(0) + : null; + +const pokuVsVitest = + pokuHappy && vitestJsdom + ? (((vitestJsdom.mean - pokuHappy.mean) / pokuHappy.mean) * 100).toFixed(0) + : null; + +const jestVsVitest = + jestJsdom && vitestJsdom + ? (((vitestJsdom.mean - jestJsdom.mean) / jestJsdom.mean) * 100).toFixed(0) + : null; + +const fastest = results.reduce((a, b) => (a.mean < b.mean ? a : b)); +const slowest = results.reduce((a, b) => (a.mean > b.mean ? a : b)); + +const report = `# React Testing Framework Benchmark Report + +> Generated: ${new Date(timestamp).toUTCString()} + +## Environment + +| Property | Value | +|---|---| +| Node.js | ${nodeVersion} | +| Platform | ${platform} | +| CPU | ${cpuModel} | +| CPU Cores | ${cpuCount} | +| Total RAM | ${totalMemGb} GB | +| Runs/scenario | ${RUNS} (trim ±${TRIM}) | + +## Scenarios + +Each scenario runs the **same 9 React tests** across 5 test files: + +| Test File | Tests | +|---|---| +| 'counter.test.jsx' | 1 — stateful counter, event interaction | +| 'hooks.test.jsx' | 2 — custom hook harness + \`renderHook\` | +| 'lifecycle.test.jsx' | 2 — \`rerender\`, \`unmount\` + effect cleanup | +| 'context.test.jsx' | 1 — \`createContext\` + wrapper injection | +| 'concurrency.test.jsx' | 2 — React 19 \`use()\` + \`useTransition\` | + +### Frameworks under test + +| Combination | DOM layer | Assertion style | +|---|---|---| +| poku + poku-react-testing | happy-dom | \`assert.strictEqual\` | +| poku + poku-react-testing | jsdom | \`assert.strictEqual\` | +| jest 29 + @testing-library/react | jsdom (jest-environment-jsdom) | \`expect().toBe()\` | +| vitest 3 + @testing-library/react | jsdom | \`expect().toBe()\` | +| vitest 3 + @testing-library/react | happy-dom | \`expect().toBe()\` | + +## Results + +${resultsTable} + +> **Wall-clock time** is measured with \`performance.now()\` around the child-process spawn. +> **Peak RSS** is captured via \`/usr/bin/time -l\` on macOS (bytes → MB). +> The baseline for relative comparisons is **poku + happy-dom**. + +## Analysis + +### Overall ranking (mean wall-clock time) + +${results + .slice() + .sort((a, b) => a.mean - b.mean) + .map((r, i) => `${i + 1}. **${r.label}** — ${sec(r.mean)}s`) + .join('\n')} + +### Speed comparison + +${ + pokuVsJest != null + ? `- poku+happy-dom vs jest+jsdom: jest is **${pokuVsJest}% ${Number(pokuVsJest) > 0 ? 'slower' : 'faster'}**` + : '- jest data unavailable' +} +${ + pokuVsVitest != null + ? `- poku+happy-dom vs vitest+jsdom: vitest is **${pokuVsVitest}% ${Number(pokuVsVitest) > 0 ? 'slower' : 'faster'}**` + : '- vitest data unavailable' +} +${ + jestVsVitest != null + ? `- jest+jsdom vs vitest+jsdom: vitest is **${jestVsVitest}% ${Number(jestVsVitest) > 0 ? 'slower' : 'faster'}** than jest` + : '- jest/vitest comparison unavailable' +} + +### DOM adapter impact + +${ + happyVsJsdomPoku != null + ? `- **poku**: happy-dom vs jsdom — jsdom is **${happyVsJsdomPoku}% ${Number(happyVsJsdomPoku) > 0 ? 'slower' : 'faster'}**` + : '- poku DOM comparison unavailable' +} +${ + happyVsJsdomVitest != null + ? `- **vitest**: happy-dom vs jsdom — jsdom is **${happyVsJsdomVitest}% ${Number(happyVsJsdomVitest) > 0 ? 'slower' : 'faster'}**` + : '- vitest DOM comparison unavailable' +} + +### Memory footprint + +${ + results.filter((r) => r.meanRss != null).length > 0 + ? results + .filter((r) => r.meanRss != null) + .sort((a, b) => a.meanRss - b.meanRss) + .map((r) => `- **${r.label}**: ${mb(r.meanRss)} MB peak RSS`) + .join('\n') + : '_Peak RSS data not available on this platform._' +} + +### Consistency (lower stdev = more predictable) + +${results + .slice() + .sort((a, b) => a.stddev - b.stddev) + .map((r) => `- **${r.label}**: σ = ${sec(r.stddev)}s`) + .join('\n')} + +## Key findings + +- **Fastest**: ${fastest.label} — ${sec(fastest.mean)}s mean +- **Slowest**: ${slowest.label} — ${sec(slowest.mean)}s mean +- **Speed spread**: ${(((slowest.mean - fastest.mean) / fastest.mean) * 100).toFixed(0)}% difference between fastest and slowest + +### Interpretation + +**poku + poku-react-testing** avoids the multi-process or bundler startup that jest (babel transform +pipeline) and vitest (Vite + module graph) require. Its architecture — isolated per-file Node.js +processes with minimal bootstrap — means cold-start overhead is proportional to the number of test +files, not to the framework's own initialization. + +**jest** carries the heaviest startup cost due to: +1. Babel transformation of every TSX file on first run (no persistent cache in this benchmark) +2. 'jest-worker' process pool initialisation +3. JSDOM environment setup per test file + +**vitest** starts faster than jest because Vite's module graph is more efficient, and the +esbuild/Rollup pipeline is faster than Babel. However, the Vite dev server and HMR machinery still +contribute to startup overhead compared to a zero-bundler approach. + +**DOM adapter choice** (happy-dom vs jsdom) has a measurable but smaller effect than the choice of +framework. happy-dom is generally lighter and initialises faster; jsdom is more spec-complete. + +## Reproducibility + +\`\`\`sh +# Install benchmark deps (one-time) +cd benchmark && npm install && cd .. + +# Re-run with custom run count +BENCH_RUNS=10 node benchmark/run.mjs +\`\`\` + +Results are saved to \`benchmark/results.json\` for programmatic analysis. +`; + +const reportPath = join(BENCH, 'REPORT.md'); +writeFileSync(reportPath, report); +console.log(` Report saved → benchmark/REPORT.md\n`); diff --git a/benchmark/tests/jest/concurrency.test.jsx b/benchmark/tests/jest/concurrency.test.jsx new file mode 100644 index 0000000..6ba0b5a --- /dev/null +++ b/benchmark/tests/jest/concurrency.test.jsx @@ -0,0 +1,71 @@ +import { afterEach, expect, test } from '@jest/globals'; +import { cleanup, fireEvent, render, screen } from '@testing-library/react'; +import { Suspense, use, useState, useTransition } from 'react'; + +afterEach(cleanup); + +const ResourceView = ({ resource }) => { + const value = use(resource); + return

{value}

; +}; + +test('renders a resolved use() resource under Suspense', () => { + const value = 'Loaded from use() resource'; + const resolvedResource = { + status: 'fulfilled', + value, + then: (onFulfilled) => + Promise.resolve(onFulfilled ? onFulfilled(value) : value), + }; + + render( + Resource pending...}> + + + ); + + expect(screen.getByRole('heading', { level: 2 }).textContent).toBe(value); +}); + +test('runs urgent and transition update pipeline', () => { + const TransitionPipeline = () => { + const [urgentState, setUrgentState] = useState('idle'); + const [deferredState, setDeferredState] = useState('idle'); + const [isPending, startTransition] = useTransition(); + + return ( +
+ + {urgentState} + {deferredState} + + {isPending ? 'pending' : 'settled'} + +
+ ); + }; + + render(); + + expect(screen.getByLabelText('urgent-state').textContent).toBe('idle'); + expect(screen.getByLabelText('deferred-state').textContent).toBe('idle'); + + fireEvent.click(screen.getByRole('button', { name: 'Run pipeline' })); + + expect(screen.getByLabelText('urgent-state').textContent).toBe( + 'urgent-updated' + ); + expect(screen.getByLabelText('deferred-state').textContent).toBe( + 'transition-updated' + ); +}); diff --git a/benchmark/tests/jest/context.test.jsx b/benchmark/tests/jest/context.test.jsx new file mode 100644 index 0000000..56f719a --- /dev/null +++ b/benchmark/tests/jest/context.test.jsx @@ -0,0 +1,22 @@ +import { afterEach, expect, test } from '@jest/globals'; +import { cleanup, render, screen } from '@testing-library/react'; +import { createContext, useContext } from 'react'; + +afterEach(cleanup); + +const ThemeContext = createContext('light'); + +const ThemeLabel = () => { + const theme = useContext(ThemeContext); + return

Theme: {theme}

; +}; + +test('injects context values via wrapper', () => { + const ThemeWrapper = ({ children }) => ( + {children} + ); + + render(, { wrapper: ThemeWrapper }); + + expect(screen.getByText('Theme: dark').textContent).toBe('Theme: dark'); +}); diff --git a/benchmark/tests/jest/counter.test.jsx b/benchmark/tests/jest/counter.test.jsx new file mode 100644 index 0000000..34cc121 --- /dev/null +++ b/benchmark/tests/jest/counter.test.jsx @@ -0,0 +1,28 @@ +import { afterEach, expect, test } from '@jest/globals'; +import { cleanup, fireEvent, render, screen } from '@testing-library/react'; +import { useState } from 'react'; + +afterEach(cleanup); + +const Counter = ({ initialCount = 0 }) => { + const [count, setCount] = useState(initialCount); + + return ( +
+

Count: {count}

+ +
+ ); +}; + +test('renders and updates a React component', () => { + render(); + + expect(screen.getByRole('heading', { name: 'Count: 1' }).textContent).toBe( + 'Count: 1' + ); + fireEvent.click(screen.getByRole('button', { name: 'Increment' })); + expect(screen.getByRole('heading').textContent).toBe('Count: 2'); +}); diff --git a/benchmark/tests/jest/hooks.test.jsx b/benchmark/tests/jest/hooks.test.jsx new file mode 100644 index 0000000..8a3270a --- /dev/null +++ b/benchmark/tests/jest/hooks.test.jsx @@ -0,0 +1,49 @@ +import { afterEach, expect, test } from '@jest/globals'; +import { + cleanup, + fireEvent, + render, + renderHook, + screen, +} from '@testing-library/react'; +import { useMemo, useState } from 'react'; + +afterEach(cleanup); + +const useToggle = (initialValue = false) => { + const [enabled, setEnabled] = useState(initialValue); + const toggle = () => setEnabled((current) => !current); + + return useMemo(() => ({ enabled, toggle }), [enabled]); +}; + +const HookHarness = () => { + const { enabled, toggle } = useToggle(); + + return ( +
+ + {enabled ? 'enabled' : 'disabled'} + + +
+ ); +}; + +test('tests custom hooks through a component harness', () => { + render(); + + expect(screen.getByLabelText('toggle-state').textContent).toBe('disabled'); + fireEvent.click(screen.getByRole('button', { name: 'Toggle' })); + expect(screen.getByLabelText('toggle-state').textContent).toBe('enabled'); +}); + +test('tests hook logic directly with renderHook', () => { + const { result } = renderHook(({ initial }) => useToggle(initial), { + initialProps: { initial: true }, + }); + + expect(result.current.enabled).toBe(true); +}); diff --git a/benchmark/tests/jest/lifecycle.test.jsx b/benchmark/tests/jest/lifecycle.test.jsx new file mode 100644 index 0000000..29245cc --- /dev/null +++ b/benchmark/tests/jest/lifecycle.test.jsx @@ -0,0 +1,39 @@ +import { afterEach, expect, test } from '@jest/globals'; +import { cleanup, render, screen } from '@testing-library/react'; +import { useEffect } from 'react'; + +afterEach(cleanup); + +const Greeting = ({ name }) =>

Hello {name}

; + +test('rerender updates component props in place', () => { + const view = render(); + + expect(screen.getByRole('heading', { level: 3 }).textContent).toBe( + 'Hello Ada' + ); + view.rerender(); + expect(screen.getByRole('heading', { level: 3 }).textContent).toBe( + 'Hello Grace' + ); +}); + +test('unmount runs effect cleanup logic', () => { + let cleaned = false; + + const WithEffect = () => { + useEffect(() => { + return () => { + cleaned = true; + }; + }, []); + + return Mounted; + }; + + const view = render(); + expect(cleaned).toBe(false); + + view.unmount(); + expect(cleaned).toBe(true); +}); diff --git a/benchmark/tests/poku/concurrency.test.jsx b/benchmark/tests/poku/concurrency.test.jsx new file mode 100644 index 0000000..6a68b10 --- /dev/null +++ b/benchmark/tests/poku/concurrency.test.jsx @@ -0,0 +1,79 @@ +import { afterEach, assert, test } from 'poku'; +import { cleanup, fireEvent, render, screen } from 'poku-react-testing'; +import React, { Suspense, use, useState, useTransition } from 'react'; + +afterEach(cleanup); + +const ResourceView = ({ resource }) => { + const value = use(resource); + return

{value}

; +}; + +test('renders a resolved use() resource under Suspense', () => { + const value = 'Loaded from use() resource'; + const resolvedResource = { + status: 'fulfilled', + value, + then: (onFulfilled) => + Promise.resolve(onFulfilled ? onFulfilled(value) : value), + }; + + render( + Resource pending...}> + + + ); + + assert.strictEqual( + screen.getByRole('heading', { level: 2 }).textContent, + value + ); +}); + +await test('runs urgent and transition update pipeline', async () => { + const TransitionPipeline = () => { + const [urgentState, setUrgentState] = useState('idle'); + const [deferredState, setDeferredState] = useState('idle'); + const [isPending, startTransition] = useTransition(); + + return ( +
+ + {urgentState} + {deferredState} + + {isPending ? 'pending' : 'settled'} + +
+ ); + }; + + render(); + + assert.strictEqual(screen.getByLabelText('urgent-state').textContent, 'idle'); + assert.strictEqual( + screen.getByLabelText('deferred-state').textContent, + 'idle' + ); + + fireEvent.click(screen.getByRole('button', { name: 'Run pipeline' })); + + assert.strictEqual( + screen.getByLabelText('urgent-state').textContent, + 'urgent-updated' + ); + assert.strictEqual( + screen.getByLabelText('deferred-state').textContent, + 'transition-updated' + ); +}); diff --git a/benchmark/tests/poku/context.test.jsx b/benchmark/tests/poku/context.test.jsx new file mode 100644 index 0000000..8366e60 --- /dev/null +++ b/benchmark/tests/poku/context.test.jsx @@ -0,0 +1,25 @@ +import { afterEach, assert, test } from 'poku'; +import { cleanup, render, screen } from 'poku-react-testing'; +import React, { createContext, useContext } from 'react'; + +afterEach(cleanup); + +const ThemeContext = createContext('light'); + +const ThemeLabel = () => { + const theme = useContext(ThemeContext); + return

Theme: {theme}

; +}; + +test('injects context values via wrapper', () => { + const ThemeWrapper = ({ children }) => ( + {children} + ); + + render(, { wrapper: ThemeWrapper }); + + assert.strictEqual( + screen.getByText('Theme: dark').textContent, + 'Theme: dark' + ); +}); diff --git a/benchmark/tests/poku/counter.test.jsx b/benchmark/tests/poku/counter.test.jsx new file mode 100644 index 0000000..49e5938 --- /dev/null +++ b/benchmark/tests/poku/counter.test.jsx @@ -0,0 +1,29 @@ +import { afterEach, assert, test } from 'poku'; +import { cleanup, fireEvent, render, screen } from 'poku-react-testing'; +import React, { useState } from 'react'; + +afterEach(cleanup); + +const Counter = ({ initialCount = 0 }) => { + const [count, setCount] = useState(initialCount); + + return ( +
+

Count: {count}

+ +
+ ); +}; + +test('renders and updates a React component', () => { + render(); + + assert.strictEqual( + screen.getByRole('heading', { name: 'Count: 1' }).textContent, + 'Count: 1' + ); + fireEvent.click(screen.getByRole('button', { name: 'Increment' })); + assert.strictEqual(screen.getByRole('heading').textContent, 'Count: 2'); +}); diff --git a/benchmark/tests/poku/hooks.test.jsx b/benchmark/tests/poku/hooks.test.jsx new file mode 100644 index 0000000..dd8782b --- /dev/null +++ b/benchmark/tests/poku/hooks.test.jsx @@ -0,0 +1,55 @@ +import { afterEach, assert, test } from 'poku'; +import { + cleanup, + fireEvent, + render, + renderHook, + screen, +} from 'poku-react-testing'; +import React, { useMemo, useState } from 'react'; + +afterEach(cleanup); + +const useToggle = (initialValue = false) => { + const [enabled, setEnabled] = useState(initialValue); + const toggle = () => setEnabled((current) => !current); + + return useMemo(() => ({ enabled, toggle }), [enabled]); +}; + +const HookHarness = () => { + const { enabled, toggle } = useToggle(); + + return ( +
+ + {enabled ? 'enabled' : 'disabled'} + + +
+ ); +}; + +test('tests custom hooks through a component harness', () => { + render(); + + assert.strictEqual( + screen.getByLabelText('toggle-state').textContent, + 'disabled' + ); + fireEvent.click(screen.getByRole('button', { name: 'Toggle' })); + assert.strictEqual( + screen.getByLabelText('toggle-state').textContent, + 'enabled' + ); +}); + +test('tests hook logic directly with renderHook', () => { + const { result } = renderHook(({ initial }) => useToggle(initial), { + initialProps: { initial: true }, + }); + + assert.strictEqual(result.current.enabled, true); +}); diff --git a/benchmark/tests/poku/lifecycle.test.jsx b/benchmark/tests/poku/lifecycle.test.jsx new file mode 100644 index 0000000..87f56f9 --- /dev/null +++ b/benchmark/tests/poku/lifecycle.test.jsx @@ -0,0 +1,41 @@ +import { afterEach, assert, test } from 'poku'; +import { cleanup, render, screen } from 'poku-react-testing'; +import React, { useEffect } from 'react'; + +afterEach(cleanup); + +const Greeting = ({ name }) =>

Hello {name}

; + +test('rerender updates component props in place', () => { + const view = render(); + + assert.strictEqual( + screen.getByRole('heading', { level: 3 }).textContent, + 'Hello Ada' + ); + view.rerender(); + assert.strictEqual( + screen.getByRole('heading', { level: 3 }).textContent, + 'Hello Grace' + ); +}); + +test('unmount runs effect cleanup logic', () => { + let cleaned = false; + + const WithEffect = () => { + useEffect(() => { + return () => { + cleaned = true; + }; + }, []); + + return Mounted; + }; + + const view = render(); + assert.strictEqual(cleaned, false); + + view.unmount(); + assert.strictEqual(cleaned, true); +}); diff --git a/benchmark/tests/vitest/concurrency.test.jsx b/benchmark/tests/vitest/concurrency.test.jsx new file mode 100644 index 0000000..9252283 --- /dev/null +++ b/benchmark/tests/vitest/concurrency.test.jsx @@ -0,0 +1,71 @@ +import { cleanup, fireEvent, render, screen } from '@testing-library/react'; +import { Suspense, use, useState, useTransition } from 'react'; +import { afterEach, expect, test } from 'vitest'; + +afterEach(cleanup); + +const ResourceView = ({ resource }) => { + const value = use(resource); + return

{value}

; +}; + +test('renders a resolved use() resource under Suspense', () => { + const value = 'Loaded from use() resource'; + const resolvedResource = { + status: 'fulfilled', + value, + then: (onFulfilled) => + Promise.resolve(onFulfilled ? onFulfilled(value) : value), + }; + + render( + Resource pending...}> + + + ); + + expect(screen.getByRole('heading', { level: 2 }).textContent).toBe(value); +}); + +test('runs urgent and transition update pipeline', () => { + const TransitionPipeline = () => { + const [urgentState, setUrgentState] = useState('idle'); + const [deferredState, setDeferredState] = useState('idle'); + const [isPending, startTransition] = useTransition(); + + return ( +
+ + {urgentState} + {deferredState} + + {isPending ? 'pending' : 'settled'} + +
+ ); + }; + + render(); + + expect(screen.getByLabelText('urgent-state').textContent).toBe('idle'); + expect(screen.getByLabelText('deferred-state').textContent).toBe('idle'); + + fireEvent.click(screen.getByRole('button', { name: 'Run pipeline' })); + + expect(screen.getByLabelText('urgent-state').textContent).toBe( + 'urgent-updated' + ); + expect(screen.getByLabelText('deferred-state').textContent).toBe( + 'transition-updated' + ); +}); diff --git a/benchmark/tests/vitest/context.test.jsx b/benchmark/tests/vitest/context.test.jsx new file mode 100644 index 0000000..f5375dd --- /dev/null +++ b/benchmark/tests/vitest/context.test.jsx @@ -0,0 +1,22 @@ +import { cleanup, render, screen } from '@testing-library/react'; +import { createContext, useContext } from 'react'; +import { afterEach, expect, test } from 'vitest'; + +afterEach(cleanup); + +const ThemeContext = createContext('light'); + +const ThemeLabel = () => { + const theme = useContext(ThemeContext); + return

Theme: {theme}

; +}; + +test('injects context values via wrapper', () => { + const ThemeWrapper = ({ children }) => ( + {children} + ); + + render(, { wrapper: ThemeWrapper }); + + expect(screen.getByText('Theme: dark').textContent).toBe('Theme: dark'); +}); diff --git a/benchmark/tests/vitest/counter.test.jsx b/benchmark/tests/vitest/counter.test.jsx new file mode 100644 index 0000000..07171e0 --- /dev/null +++ b/benchmark/tests/vitest/counter.test.jsx @@ -0,0 +1,28 @@ +import { cleanup, fireEvent, render, screen } from '@testing-library/react'; +import { useState } from 'react'; +import { afterEach, expect, test } from 'vitest'; + +afterEach(cleanup); + +const Counter = ({ initialCount = 0 }) => { + const [count, setCount] = useState(initialCount); + + return ( +
+

Count: {count}

+ +
+ ); +}; + +test('renders and updates a React component', () => { + render(); + + expect(screen.getByRole('heading', { name: 'Count: 1' }).textContent).toBe( + 'Count: 1' + ); + fireEvent.click(screen.getByRole('button', { name: 'Increment' })); + expect(screen.getByRole('heading').textContent).toBe('Count: 2'); +}); diff --git a/benchmark/tests/vitest/hooks.test.jsx b/benchmark/tests/vitest/hooks.test.jsx new file mode 100644 index 0000000..196c10a --- /dev/null +++ b/benchmark/tests/vitest/hooks.test.jsx @@ -0,0 +1,49 @@ +import { + cleanup, + fireEvent, + render, + renderHook, + screen, +} from '@testing-library/react'; +import { useMemo, useState } from 'react'; +import { afterEach, expect, test } from 'vitest'; + +afterEach(cleanup); + +const useToggle = (initialValue = false) => { + const [enabled, setEnabled] = useState(initialValue); + const toggle = () => setEnabled((current) => !current); + + return useMemo(() => ({ enabled, toggle }), [enabled]); +}; + +const HookHarness = () => { + const { enabled, toggle } = useToggle(); + + return ( +
+ + {enabled ? 'enabled' : 'disabled'} + + +
+ ); +}; + +test('tests custom hooks through a component harness', () => { + render(); + + expect(screen.getByLabelText('toggle-state').textContent).toBe('disabled'); + fireEvent.click(screen.getByRole('button', { name: 'Toggle' })); + expect(screen.getByLabelText('toggle-state').textContent).toBe('enabled'); +}); + +test('tests hook logic directly with renderHook', () => { + const { result } = renderHook(({ initial }) => useToggle(initial), { + initialProps: { initial: true }, + }); + + expect(result.current.enabled).toBe(true); +}); diff --git a/benchmark/tests/vitest/lifecycle.test.jsx b/benchmark/tests/vitest/lifecycle.test.jsx new file mode 100644 index 0000000..8c65a92 --- /dev/null +++ b/benchmark/tests/vitest/lifecycle.test.jsx @@ -0,0 +1,39 @@ +import { cleanup, render, screen } from '@testing-library/react'; +import { useEffect } from 'react'; +import { afterEach, expect, test } from 'vitest'; + +afterEach(cleanup); + +const Greeting = ({ name }) =>

Hello {name}

; + +test('rerender updates component props in place', () => { + const view = render(); + + expect(screen.getByRole('heading', { level: 3 }).textContent).toBe( + 'Hello Ada' + ); + view.rerender(); + expect(screen.getByRole('heading', { level: 3 }).textContent).toBe( + 'Hello Grace' + ); +}); + +test('unmount runs effect cleanup logic', () => { + let cleaned = false; + + const WithEffect = () => { + useEffect(() => { + return () => { + cleaned = true; + }; + }, []); + + return Mounted; + }; + + const view = render(); + expect(cleaned).toBe(false); + + view.unmount(); + expect(cleaned).toBe(true); +}); diff --git a/benchmark/vitest.happy.config.js b/benchmark/vitest.happy.config.js new file mode 100644 index 0000000..cc2e788 --- /dev/null +++ b/benchmark/vitest.happy.config.js @@ -0,0 +1,11 @@ +import react from '@vitejs/plugin-react'; +import { defineConfig } from 'vitest/config'; + +export default defineConfig({ + plugins: [react()], + test: { + environment: 'happy-dom', + include: ['tests/vitest/**/*.test.jsx'], + globals: false, + }, +}); diff --git a/benchmark/vitest.jsdom.config.js b/benchmark/vitest.jsdom.config.js new file mode 100644 index 0000000..bccc3b3 --- /dev/null +++ b/benchmark/vitest.jsdom.config.js @@ -0,0 +1,11 @@ +import react from '@vitejs/plugin-react'; +import { defineConfig } from 'vitest/config'; + +export default defineConfig({ + plugins: [react()], + test: { + environment: 'jsdom', + include: ['tests/vitest/**/*.test.jsx'], + globals: false, + }, +}); diff --git a/package-lock.json b/package-lock.json index 1f375ce..990ec8a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -7,12 +7,13 @@ "": { "name": "poku-react-testing", "version": "1.0.1", - "license": "ISC", + "license": "MIT", "dependencies": { "@testing-library/dom": "^10.4.1" }, "devDependencies": { "@happy-dom/global-registrator": "^20.8.9", + "@ianvs/prettier-plugin-sort-imports": "^4.7.0", "@types/jsdom": "^28.0.1", "@types/node": "^25.5.0", "@types/react": "^19.2.14", @@ -21,6 +22,7 @@ "happy-dom": "^20.8.9", "jsdom": "^26.1.0", "poku": "4.1.0", + "prettier": "^3.6.2", "react": "^19.2.4", "react-dom": "^19.2.4", "rimraf": "^6.0.1", @@ -89,6 +91,43 @@ "node": ">=6.9.0" } }, + "node_modules/@babel/generator": { + "version": "7.29.1", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz", + "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, "node_modules/@babel/helper-validator-identifier": { "version": "7.28.5", "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", @@ -98,6 +137,22 @@ "node": ">=6.9.0" } }, + "node_modules/@babel/parser": { + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.2.tgz", + "integrity": "sha512-4GgRzy/+fsBa72/RZVJmGKPmZu9Byn8o4MoLpmNe1m8ZfYnz5emHLQz3U4gLud6Zwl0RZIcgiLD7Uq7ySFuDLA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.29.0" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, "node_modules/@babel/runtime": { "version": "7.29.2", "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.2.tgz", @@ -107,6 +162,54 @@ "node": ">=6.9.0" } }, + "node_modules/@babel/template": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz", + "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz", + "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz", + "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, "node_modules/@csstools/color-helpers": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/@csstools/color-helpers/-/color-helpers-5.1.0.tgz", @@ -685,6 +788,41 @@ "node": ">=20.0.0" } }, + "node_modules/@ianvs/prettier-plugin-sort-imports": { + "version": "4.7.1", + "resolved": "https://registry.npmjs.org/@ianvs/prettier-plugin-sort-imports/-/prettier-plugin-sort-imports-4.7.1.tgz", + "integrity": "sha512-jmTNYGlg95tlsoG3JLCcuC4BrFELJtLirLAkQW/71lXSyOhVt/Xj7xWbbGcuVbNq1gwWgSyMrPjJc9Z30hynVw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@babel/generator": "^7.26.2", + "@babel/parser": "^7.26.2", + "@babel/traverse": "^7.25.9", + "@babel/types": "^7.26.0", + "semver": "^7.5.2" + }, + "peerDependencies": { + "@prettier/plugin-oxc": "^0.0.4 || ^0.1.0", + "@vue/compiler-sfc": "2.7.x || 3.x", + "content-tag": "^4.0.0", + "prettier": "2 || 3 || ^4.0.0-0", + "prettier-plugin-ember-template-tag": "^2.1.0" + }, + "peerDependenciesMeta": { + "@prettier/plugin-oxc": { + "optional": true + }, + "@vue/compiler-sfc": { + "optional": true + }, + "content-tag": { + "optional": true + }, + "prettier-plugin-ember-template-tag": { + "optional": true + } + } + }, "node_modules/@jridgewell/gen-mapping": { "version": "0.3.13", "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", @@ -1733,6 +1871,19 @@ "node": ">=18" } }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, "node_modules/lilconfig": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-3.1.3.tgz", @@ -2038,6 +2189,22 @@ } } }, + "node_modules/prettier": { + "version": "3.8.1", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.8.1.tgz", + "integrity": "sha512-UOnG6LftzbdaHZcKoPFtOcCKztrQ57WkHDeRD9t/PTQtmT0NHSeWWepj6pS0z/N7+08BHFDQVUrfmfMRcZwbMg==", + "dev": true, + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, "node_modules/pretty-format": { "version": "27.5.1", "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", @@ -2224,6 +2391,19 @@ "dev": true, "license": "MIT" }, + "node_modules/semver": { + "version": "7.7.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz", + "integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", diff --git a/package.json b/package.json index 12d9d26..7dc54e2 100644 --- a/package.json +++ b/package.json @@ -32,13 +32,15 @@ }, "scripts": { "test": "npm run test:happy && npm run test:jsdom", - "test:happy": "cross-env POKU_REACT_TEST_DOM=happy-dom node --import=tsx ./node_modules/poku/lib/bin/index.js --showLogs", - "test:jsdom": "cross-env POKU_REACT_TEST_DOM=jsdom node --import=tsx ./node_modules/poku/lib/bin/index.js --showLogs", - "test:bun": "cross-env POKU_REACT_TEST_DOM=happy-dom bun ./node_modules/poku/lib/bin/index.js --showLogs && cross-env POKU_REACT_TEST_DOM=jsdom bun ./node_modules/poku/lib/bin/index.js --showLogs", - "test:deno": "cross-env POKU_REACT_TEST_DOM=happy-dom deno run -A npm:poku --showLogs", + "test:happy": "cross-env POKU_REACT_TEST_DOM=happy-dom node --import=tsx ./node_modules/poku/lib/bin/index.js tests --showLogs", + "test:jsdom": "cross-env POKU_REACT_TEST_DOM=jsdom node --import=tsx ./node_modules/poku/lib/bin/index.js tests --showLogs", + "test:bun": "cross-env POKU_REACT_TEST_DOM=happy-dom bun ./node_modules/poku/lib/bin/index.js tests --showLogs && cross-env POKU_REACT_TEST_DOM=jsdom bun ./node_modules/poku/lib/bin/index.js tests --showLogs", + "test:deno": "cross-env POKU_REACT_TEST_DOM=happy-dom deno run -A npm:poku tests --showLogs", "clean": "rimraf dist", "build": "tsup src/index.ts src/plugin.ts src/react-testing.ts src/dom-setup-happy.ts src/dom-setup-jsdom.ts --format esm --dts --target node20 --sourcemap --clean --tsconfig tsconfig.tsup.json", "typecheck": "tsc -p tsconfig.build.json --noEmit", + "format": "prettier --write .", + "format:check": "prettier --check .", "lint": "npm run typecheck", "check": "npm run typecheck && npm test", "prepack": "npm run build" @@ -88,6 +90,7 @@ } }, "devDependencies": { + "@ianvs/prettier-plugin-sort-imports": "^4.7.0", "@happy-dom/global-registrator": "^20.8.9", "@types/node": "^25.5.0", "@types/jsdom": "^28.0.1", @@ -97,6 +100,7 @@ "happy-dom": "^20.8.9", "jsdom": "^26.1.0", "poku": "4.1.0", + "prettier": "^3.6.2", "react": "^19.2.4", "react-dom": "^19.2.4", "rimraf": "^6.0.1", diff --git a/src/dom-setup-happy.ts b/src/dom-setup-happy.ts index 31dc60e..22ac062 100644 --- a/src/dom-setup-happy.ts +++ b/src/dom-setup-happy.ts @@ -1,4 +1,5 @@ import { GlobalRegistrator } from '@happy-dom/global-registrator'; +import { parseRuntimeOptions } from './runtime-options.ts'; declare const Deno: unknown; @@ -8,8 +9,7 @@ type ReactActGlobal = typeof globalThis & { const reactGlobal = globalThis as ReactActGlobal; -const defaultUrl = 'http://localhost:3000/'; -const configuredUrl = process.env.POKU_REACT_DOM_URL || defaultUrl; +const configuredUrl = parseRuntimeOptions().domUrl; if (!globalThis.window || !globalThis.document) { const isDenoRuntime = typeof Deno !== 'undefined'; @@ -23,11 +23,12 @@ if (!globalThis.window || !globalThis.document) { }); if (isDenoRuntime) { - if (nativeEvent) (globalThis as unknown as Record).Event = nativeEvent; + if (nativeEvent) + (globalThis as unknown as Record).Event = nativeEvent; if (nativeDispatchEvent) globalThis.dispatchEvent = nativeDispatchEvent; } } if (typeof reactGlobal.IS_REACT_ACT_ENVIRONMENT === 'undefined') { reactGlobal.IS_REACT_ACT_ENVIRONMENT = true; -} \ No newline at end of file +} diff --git a/src/dom-setup-jsdom.ts b/src/dom-setup-jsdom.ts index f712ede..e5e80a3 100644 --- a/src/dom-setup-jsdom.ts +++ b/src/dom-setup-jsdom.ts @@ -1,5 +1,6 @@ -const defaultUrl = 'http://localhost:3000/'; -const configuredUrl = process.env.POKU_REACT_DOM_URL || defaultUrl; +import { parseRuntimeOptions } from './runtime-options.ts'; + +const configuredUrl = parseRuntimeOptions().domUrl; type ReactActGlobal = typeof globalThis & { IS_REACT_ACT_ENVIRONMENT?: boolean; @@ -22,7 +23,7 @@ const createJSDOMEnvironment = async () => { mod = await import('jsdom'); } catch { throw new Error( - '[poku-react-testing] DOM adapter "jsdom" requires the "jsdom" package. Install it with "npm install --save-dev jsdom".', + '[poku-react-testing] DOM adapter "jsdom" requires the "jsdom" package. Install it with "npm install --save-dev jsdom".' ); } @@ -48,4 +49,4 @@ if (!globalThis.window || !globalThis.document) { if (typeof reactGlobal.IS_REACT_ACT_ENVIRONMENT === 'undefined') { reactGlobal.IS_REACT_ACT_ENVIRONMENT = true; -} \ No newline at end of file +} diff --git a/src/index.ts b/src/index.ts index f8d7635..1376b6e 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,11 +1,18 @@ /** Plugin factory and alias for Poku integration. */ export { createReactTestingPlugin, reactTestingPlugin } from './plugin.ts'; export type { - ReactDomAdapter, - ReactMetricsOptions, - ReactMetricsSummary, - ReactTestingPluginOptions, + ReactDomAdapter, + ReactMetricsOptions, + ReactMetricsSummary, + ReactTestingPluginOptions, } from './plugin.ts'; /** React testing helpers and DX exports. */ -export { act, cleanup, fireEvent, render, renderHook, screen } from './react-testing.ts'; -export * from './react-testing.ts'; \ No newline at end of file +export { + act, + cleanup, + fireEvent, + render, + renderHook, + screen, +} from './react-testing.ts'; +export * from './react-testing.ts'; diff --git a/src/plugin.ts b/src/plugin.ts index 6eb063d..249b89b 100644 --- a/src/plugin.ts +++ b/src/plugin.ts @@ -1,14 +1,15 @@ -import { definePlugin } from 'poku/plugins'; -import { fileURLToPath } from 'node:url'; -import { dirname, extname, resolve } from 'node:path'; import { existsSync } from 'node:fs'; +import { dirname, extname, resolve } from 'node:path'; +import { fileURLToPath } from 'node:url'; +import { definePlugin } from 'poku/plugins'; +import { runtimeOptionArgPrefixes } from './runtime-options.ts'; const currentDir = dirname(fileURLToPath(import.meta.url)); const resolveSetupModulePath = (baseName: string) => { - const jsPath = resolve(currentDir, `${baseName}.js`); - if (existsSync(jsPath)) return jsPath; + const jsPath = resolve(currentDir, `${baseName}.js`); + if (existsSync(jsPath)) return jsPath; - return resolve(currentDir, `${baseName}.ts`); + return resolve(currentDir, `${baseName}.ts`); }; const happyDomSetupPath = resolveSetupModulePath('dom-setup-happy'); @@ -19,340 +20,430 @@ const reactExtensions = new Set(['.tsx', '.jsx']); export type ReactDomAdapter = 'happy-dom' | 'jsdom' | { setupModule: string }; export type ReactTestingPluginOptions = { - /** - * DOM implementation used by test file processes. - * - * - `happy-dom`: fast default suitable for most component tests. - * - `jsdom`: broader compatibility for browser-like APIs. - * - `{ setupModule }`: custom module that prepares globals. - */ - dom?: ReactDomAdapter; - /** - * URL assigned to the DOM environment. - */ - domUrl?: string; - /** - * Render metrics configuration. Disabled by default for production-safe behavior. - */ - metrics?: boolean | ReactMetricsOptions; + /** + * DOM implementation used by test file processes. + * + * - `happy-dom`: fast default suitable for most component tests. + * - `jsdom`: broader compatibility for browser-like APIs. + * - `{ setupModule }`: custom module that prepares globals. + */ + dom?: ReactDomAdapter; + /** + * URL assigned to the DOM environment. + */ + domUrl?: string; + /** + * Render metrics configuration. Disabled by default for production-safe behavior. + */ + metrics?: boolean | ReactMetricsOptions; }; export type ReactMetricsSummary = { - totalCaptured: number; - totalReported: number; - topSlowest: RenderMetric[]; + totalCaptured: number; + totalReported: number; + topSlowest: RenderMetric[]; }; export type ReactMetricsOptions = { - /** - * Enable or disable render metrics collection. - */ - enabled?: boolean; - /** - * Maximum number of rows to display/report. - * @default 5 - */ - topN?: number; - /** - * Minimum duration to include in the final report. - * @default 0 - */ - minDurationMs?: number; - /** - * Custom reporter. Falls back to console output when omitted. - */ - reporter?: (summary: ReactMetricsSummary) => void; + /** + * Enable or disable render metrics collection. + */ + enabled?: boolean; + /** + * Maximum number of rows to display/report. + * @default 5 + */ + topN?: number; + /** + * Minimum duration to include in the final report. + * @default 0 + */ + minDurationMs?: number; + /** + * Custom reporter. Falls back to console output when omitted. + */ + reporter?: (summary: ReactMetricsSummary) => void; }; type RenderMetricMessage = { - type: 'POKU_REACT_RENDER_METRIC'; - componentName?: string; - durationMs?: number; + type: 'POKU_REACT_RENDER_METRIC'; + componentName?: string; + durationMs?: number; +}; + +type RenderMetricBatchMessage = { + type: 'POKU_REACT_RENDER_METRIC_BATCH'; + metrics: Array<{ + componentName?: string; + durationMs?: number; + }>; }; type RenderMetric = { - file: string; - componentName: string; - durationMs: number; + file: string; + componentName: string; + durationMs: number; }; type NormalizedMetricsOptions = { - enabled: boolean; - topN: number; - minDurationMs: number; - reporter?: (summary: ReactMetricsSummary) => void; + enabled: boolean; + topN: number; + minDurationMs: number; + reporter?: (summary: ReactMetricsSummary) => void; }; type RuntimeSupport = { - supportsNodeLikeImport: boolean; - supportsDenoPreload: boolean; + supportsNodeLikeImport: boolean; + supportsDenoPreload: boolean; }; type BuildRunnerCommandInput = { - runtime: string; - command: string[]; - file: string; - domSetupPath: string; + runtime: string; + command: string[]; + file: string; + domSetupPath: string; + runtimeOptionArgs: string[]; }; type BuildRunnerCommandOutput = { - shouldHandle: boolean; - command: string[]; -}; - -type EnvSnapshot = { - previousDomUrl: string | undefined; - previousMetricsFlag: string | undefined; + shouldHandle: boolean; + command: string[]; }; const DEFAULT_TOP_N = 5; const DEFAULT_MIN_DURATION_MS = 0; -const isRenderMetricMessage = (message: unknown): message is RenderMetricMessage => { - if (!message || typeof message !== 'object') return false; - return (message as Record).type === 'POKU_REACT_RENDER_METRIC'; +const isRenderMetricMessage = ( + message: unknown +): message is RenderMetricMessage => { + if (!message || typeof message !== 'object') return false; + return ( + (message as Record).type === 'POKU_REACT_RENDER_METRIC' + ); +}; + +const isRenderMetricBatchMessage = ( + message: unknown +): message is RenderMetricBatchMessage => { + if (!message || typeof message !== 'object') return false; + + const record = message as Record; + return ( + record.type === 'POKU_REACT_RENDER_METRIC_BATCH' && + Array.isArray(record.metrics) + ); }; const getComponentName = (componentName: unknown) => - typeof componentName === 'string' && componentName.length > 0 - ? componentName - : 'AnonymousComponent'; + typeof componentName === 'string' && componentName.length > 0 + ? componentName + : 'AnonymousComponent'; -const isTsxImport = (arg: string) => arg === '--import=tsx' || arg === '--loader=tsx'; +const isTsxImport = (arg: string) => + arg === '--import=tsx' || arg === '--loader=tsx'; const isNodeRuntime = (runtime: string) => runtime === 'node'; const isBunRuntime = (runtime: string) => runtime === 'bun'; const isDenoRuntime = (runtime: string) => runtime === 'deno'; const getRuntimeSupport = (runtime: string): RuntimeSupport => ({ - supportsNodeLikeImport: isNodeRuntime(runtime) || isBunRuntime(runtime), - supportsDenoPreload: isDenoRuntime(runtime), + supportsNodeLikeImport: isNodeRuntime(runtime) || isBunRuntime(runtime), + supportsDenoPreload: isDenoRuntime(runtime), }); const canHandleRuntime = (runtime: string) => { - const support = getRuntimeSupport(runtime); - return support.supportsNodeLikeImport || support.supportsDenoPreload; + const support = getRuntimeSupport(runtime); + return support.supportsNodeLikeImport || support.supportsDenoPreload; }; const resolveDomSetupPath = (adapter: ReactDomAdapter | undefined) => { - if (!adapter || adapter === 'happy-dom') return happyDomSetupPath; - if (adapter === 'jsdom') return jsdomSetupPath; + if (!adapter || adapter === 'happy-dom') return happyDomSetupPath; + if (adapter === 'jsdom') return jsdomSetupPath; - return resolve(process.cwd(), adapter.setupModule); + return resolve(process.cwd(), adapter.setupModule); }; const getPositiveIntegerOrDefault = (value: unknown, fallback: number) => { - const numeric = Number(value); - if (!Number.isFinite(numeric) || numeric <= 0) return fallback; - return Math.floor(numeric); + const numeric = + typeof value === 'number' + ? value + : typeof value === 'string' && value.trim().length > 0 + ? Number(value.trim()) + : NaN; + + if (!Number.isFinite(numeric) || numeric <= 0) return fallback; + return Math.floor(numeric); }; const getNonNegativeNumberOrDefault = (value: unknown, fallback: number) => { - const numeric = Number(value); - if (!Number.isFinite(numeric) || numeric < 0) return fallback; - return numeric; -}; - -const normalizeMetricsOptions = ( - metrics: ReactTestingPluginOptions['metrics'], -): NormalizedMetricsOptions => { - if (metrics === true) { - return { - enabled: true, - topN: DEFAULT_TOP_N, - minDurationMs: DEFAULT_MIN_DURATION_MS, - }; - } - - if (!metrics) { - return { - enabled: false, - topN: DEFAULT_TOP_N, - minDurationMs: DEFAULT_MIN_DURATION_MS, - }; - } - - const normalized: NormalizedMetricsOptions = { - enabled: metrics.enabled ?? true, - topN: getPositiveIntegerOrDefault(metrics.topN, DEFAULT_TOP_N), - minDurationMs: getNonNegativeNumberOrDefault(metrics.minDurationMs, DEFAULT_MIN_DURATION_MS), - }; - - if (metrics.reporter) normalized.reporter = metrics.reporter; - - return normalized; + const numeric = + typeof value === 'number' + ? value + : typeof value === 'string' && value.trim().length > 0 + ? Number(value.trim()) + : NaN; + + if (!Number.isFinite(numeric) || numeric < 0) return fallback; + return numeric; }; -const buildRunnerCommand = ({ - runtime, - command, - file, - domSetupPath, -}: BuildRunnerCommandInput): BuildRunnerCommandOutput => { - const support = getRuntimeSupport(runtime); - - if (!support.supportsNodeLikeImport && !support.supportsDenoPreload) { - return { shouldHandle: false, command }; - } +const buildRuntimeOptionArgs = ( + options: ReactTestingPluginOptions, + metricsOptions: NormalizedMetricsOptions +) => { + const args: string[] = []; - if (!reactExtensions.has(extname(file))) { - return { shouldHandle: false, command }; - } + if (options.domUrl) { + args.push(`${runtimeOptionArgPrefixes.domUrl}${options.domUrl}`); + } - const fileIndex = command.lastIndexOf(file); - if (fileIndex === -1) return { shouldHandle: false, command }; + if (metricsOptions.enabled) { + args.push(`${runtimeOptionArgPrefixes.metrics}1`); + args.push( + `${runtimeOptionArgPrefixes.minMetricMs}${metricsOptions.minDurationMs}` + ); + } - const beforeFile = command.slice(1, fileIndex); - const afterFile = command.slice(fileIndex + 1); - - const hasTsx = beforeFile.some(isTsxImport); - const hasNodeLikeDomSetup = beforeFile.some((arg) => arg === `--import=${domSetupPath}`); - const hasDenoDomSetup = beforeFile.some((arg) => arg === `--preload=${domSetupPath}`); - - const extraImports: string[] = []; - if (isNodeRuntime(runtime) && !hasTsx) extraImports.push('--import=tsx'); - if (support.supportsNodeLikeImport && !hasNodeLikeDomSetup) extraImports.push(`--import=${domSetupPath}`); - if (support.supportsDenoPreload && !hasDenoDomSetup) extraImports.push(`--preload=${domSetupPath}`); - - return { - shouldHandle: true, - command: [runtime, ...beforeFile, ...extraImports, file, ...afterFile], - }; + return args; }; -const captureEnvSnapshot = (): EnvSnapshot => ({ - previousDomUrl: process.env.POKU_REACT_DOM_URL, - previousMetricsFlag: process.env.POKU_REACT_ENABLE_METRICS, -}); - -const applyEnvironmentOptions = (options: ReactTestingPluginOptions, metricsOptions: NormalizedMetricsOptions) => { - if (options.domUrl) { - process.env.POKU_REACT_DOM_URL = options.domUrl; - } - - if (metricsOptions.enabled) { - process.env.POKU_REACT_ENABLE_METRICS = '1'; - } +const normalizeMetricsOptions = ( + metrics: ReactTestingPluginOptions['metrics'] +): NormalizedMetricsOptions => { + if (metrics === true) { + return { + enabled: true, + topN: DEFAULT_TOP_N, + minDurationMs: DEFAULT_MIN_DURATION_MS, + }; + } + + if (!metrics) { + return { + enabled: false, + topN: DEFAULT_TOP_N, + minDurationMs: DEFAULT_MIN_DURATION_MS, + }; + } + + const normalized: NormalizedMetricsOptions = { + enabled: metrics.enabled ?? true, + topN: getPositiveIntegerOrDefault(metrics.topN, DEFAULT_TOP_N), + minDurationMs: getNonNegativeNumberOrDefault( + metrics.minDurationMs, + DEFAULT_MIN_DURATION_MS + ), + }; + + if (metrics.reporter) normalized.reporter = metrics.reporter; + + return normalized; }; -const restoreEnvironmentOptions = (snapshot: EnvSnapshot) => { - if (typeof snapshot.previousDomUrl === 'undefined') { - delete process.env.POKU_REACT_DOM_URL; - } else { - process.env.POKU_REACT_DOM_URL = snapshot.previousDomUrl; - } - - if (typeof snapshot.previousMetricsFlag === 'undefined') { - delete process.env.POKU_REACT_ENABLE_METRICS; - } else { - process.env.POKU_REACT_ENABLE_METRICS = snapshot.previousMetricsFlag; - } +const buildRunnerCommand = ({ + runtime, + command, + file, + domSetupPath, + runtimeOptionArgs, +}: BuildRunnerCommandInput): BuildRunnerCommandOutput => { + const support = getRuntimeSupport(runtime); + + if (!support.supportsNodeLikeImport && !support.supportsDenoPreload) { + return { shouldHandle: false, command }; + } + + if (!reactExtensions.has(extname(file))) { + return { shouldHandle: false, command }; + } + + // Optimization: find from the end to prevent false matches in directory names + const fileIndex = command.lastIndexOf(file); + if (fileIndex === -1) return { shouldHandle: false, command }; + + const nodeImportFlag = `--import=${domSetupPath}`; + const denoPreloadFlag = `--preload=${domSetupPath}`; + const beforeFile: string[] = []; + const afterFile: string[] = []; + + let hasTsx = false; + let hasNodeLikeDomSetup = false; + let hasDenoDomSetup = false; + const existingArgs = new Set(); + + for (let index = 1; index < command.length; index += 1) { + const arg = command[index]; + if (typeof arg !== 'string') continue; + + existingArgs.add(arg); + + if (index < fileIndex) { + beforeFile.push(arg); + + if (isTsxImport(arg)) hasTsx = true; + else if (arg === nodeImportFlag) hasNodeLikeDomSetup = true; + else if (arg === denoPreloadFlag) hasDenoDomSetup = true; + continue; + } + + if (index > fileIndex) { + afterFile.push(arg); + } + } + + const extraImports: string[] = []; + if (isNodeRuntime(runtime) && !hasTsx) extraImports.push('--import=tsx'); + if (support.supportsNodeLikeImport && !hasNodeLikeDomSetup) + extraImports.push(nodeImportFlag); + if (support.supportsDenoPreload && !hasDenoDomSetup) + extraImports.push(denoPreloadFlag); + + const runtimeArgsToInject: string[] = []; + for (const runtimeOptionArg of runtimeOptionArgs) { + if (existingArgs.has(runtimeOptionArg)) continue; + runtimeArgsToInject.push(runtimeOptionArg); + } + + return { + shouldHandle: true, + command: [ + runtime, + ...beforeFile, + ...extraImports, + file, + ...runtimeArgsToInject, + ...afterFile, + ], + }; }; -const selectTopSlowestMetrics = (metrics: RenderMetric[], options: NormalizedMetricsOptions) => - [...metrics] - .filter((metric) => metric.durationMs >= options.minDurationMs) - .sort((a, b) => b.durationMs - a.durationMs) - .slice(0, options.topN); +const selectTopSlowestMetrics = ( + metrics: RenderMetric[], + options: NormalizedMetricsOptions +) => + [...metrics] + .sort((a, b) => b.durationMs - a.durationMs) + .slice(0, options.topN); const createMetricsSummary = ( - metrics: RenderMetric[], - options: NormalizedMetricsOptions, + metrics: RenderMetric[], + options: NormalizedMetricsOptions ): ReactMetricsSummary | null => { - if (!options.enabled || metrics.length === 0) return null; + if (!options.enabled || metrics.length === 0) return null; - const topSlowest = selectTopSlowestMetrics(metrics, options); - if (topSlowest.length === 0) return null; + const topSlowest = selectTopSlowestMetrics(metrics, options); + if (topSlowest.length === 0) return null; - return { - totalCaptured: metrics.length, - totalReported: topSlowest.length, - topSlowest, - }; + return { + totalCaptured: metrics.length, // Note: Represents captured over threshold + totalReported: topSlowest.length, + topSlowest, + }; }; const printMetricsSummary = (summary: ReactMetricsSummary) => { - const lines = summary.topSlowest.map( - (metric) => - ` - ${metric.componentName} in ${metric.file}: ${metric.durationMs.toFixed(2)}ms`, - ); + const lines = summary.topSlowest.map( + (metric) => + ` - ${metric.componentName} in ${metric.file}: ${metric.durationMs.toFixed(2)}ms` + ); - console.log('\n[poku-react-testing] Slowest component renders'); - for (const line of lines) console.log(line); + console.log('\n[poku-react-testing] Slowest component renders'); + for (const line of lines) console.log(line); }; /** * Create a Poku plugin that prepares DOM globals and TSX execution for React tests. */ -export const createReactTestingPlugin = (options: ReactTestingPluginOptions = {}) => { - const metrics: RenderMetric[] = []; - const envSnapshot = captureEnvSnapshot(); - const domSetupPath = resolveDomSetupPath(options.dom); - const metricsOptions = normalizeMetricsOptions(options.metrics); - - applyEnvironmentOptions(options, metricsOptions); - - return definePlugin({ - name: 'react-testing', - ipc: metricsOptions.enabled, - - runner(command, file) { - const runtime = command[0]; - if (!runtime) return command; - const result = buildRunnerCommand({ - runtime, - command, - file, - domSetupPath, - }); - - return result.command; - }, - - onTestProcess(child, file) { - if (!metricsOptions.enabled) return; - - child.on('message', (message) => { - if (!isRenderMetricMessage(message)) return; - - metrics.push({ - file, - componentName: getComponentName(message.componentName), - durationMs: Number(message.durationMs) || 0, - }); - }); - }, - - teardown() { - restoreEnvironmentOptions(envSnapshot); - - const summary = createMetricsSummary(metrics, metricsOptions); - if (!summary) return; - - if (metricsOptions.reporter) { - metricsOptions.reporter(summary); - return; - } - - printMetricsSummary(summary); - }, - }); +export const createReactTestingPlugin = ( + options: ReactTestingPluginOptions = {} +) => { + let metrics: RenderMetric[] = []; + const domSetupPath = resolveDomSetupPath(options.dom); + const metricsOptions = normalizeMetricsOptions(options.metrics); + const runtimeOptionArgs = buildRuntimeOptionArgs(options, metricsOptions); + + return definePlugin({ + name: 'react-testing', + ipc: metricsOptions.enabled, + + runner(command, file) { + const runtime = command[0]; + if (!runtime) return command; + const result = buildRunnerCommand({ + runtime, + command, + file, + domSetupPath, + runtimeOptionArgs, + }); + + return result.command; + }, + + onTestProcess(child, file) { + if (!metricsOptions.enabled) return; + + child.on('message', (message) => { + if (isRenderMetricBatchMessage(message)) { + for (const metric of message.metrics) { + const durationMs = Number(metric.durationMs) || 0; + + metrics.push({ + file, + componentName: getComponentName(metric.componentName), + durationMs, + }); + } + + if (metrics.length > metricsOptions.topN * 10) { + metrics = selectTopSlowestMetrics(metrics, metricsOptions); + } + + return; + } + + if (!isRenderMetricMessage(message)) return; + + const durationMs = Number(message.durationMs) || 0; + + metrics.push({ + file, + componentName: getComponentName(message.componentName), + durationMs, + }); + + // Optimization: Prevent unbounded memory growth on massive suites + // Prune array back down periodically to keep only top candidates + if (metrics.length > metricsOptions.topN * 10) { + metrics = selectTopSlowestMetrics(metrics, metricsOptions); + } + }); + }, + + teardown() { + const summary = createMetricsSummary(metrics, metricsOptions); + if (!summary) return; + + if (metricsOptions.reporter) { + metricsOptions.reporter(summary); + return; + } + + printMetricsSummary(summary); + }, + }); }; -/** - * Alias for `createReactTestingPlugin`. - */ export const reactTestingPlugin = createReactTestingPlugin; export const __internal = { - buildRunnerCommand, - canHandleRuntime, - captureEnvSnapshot, - applyEnvironmentOptions, - restoreEnvironmentOptions, - normalizeMetricsOptions, - selectTopSlowestMetrics, - createMetricsSummary, - getComponentName, - isRenderMetricMessage, - resolveDomSetupPath, + buildRunnerCommand, + canHandleRuntime, + buildRuntimeOptionArgs, + normalizeMetricsOptions, + selectTopSlowestMetrics, + createMetricsSummary, + getComponentName, + isRenderMetricMessage, + isRenderMetricBatchMessage, + resolveDomSetupPath, }; diff --git a/src/react-testing.ts b/src/react-testing.ts index 221dcd1..a96493b 100644 --- a/src/react-testing.ts +++ b/src/react-testing.ts @@ -1,13 +1,15 @@ -import React, { type ComponentType, type PropsWithChildren, type ReactElement } from 'react'; -import { - getQueriesForElement, - type BoundFunctions, - type Screen, - type fireEvent as baseFireEvent, - queries, +import type { + fireEvent as baseFireEvent, + BoundFunctions, + Screen, } from '@testing-library/dom'; +import type { ComponentType, PropsWithChildren, ReactElement } from 'react'; +import type { Root } from 'react-dom/client'; +import { getQueriesForElement, queries } from '@testing-library/dom'; import * as domTestingLibrary from '@testing-library/dom'; -import { createRoot, type Root } from 'react-dom/client'; +import React from 'react'; +import { createRoot } from 'react-dom/client'; +import { parseRuntimeOptions } from './runtime-options.ts'; const { act } = React; @@ -17,17 +19,17 @@ export { act }; type WrapperComponent = ComponentType>; type InternalMounted = { - root: Root; - container: Element; + root: Root | null; + container: Element | null; ownsContainer: boolean; }; const mountedRoots = new Set(); -const getNow = () => +const getNow: () => number = typeof performance !== 'undefined' && typeof performance.now === 'function' - ? performance.now() - : Date.now(); + ? performance.now.bind(performance) + : Date.now.bind(Date); const getComponentName = (ui: ReactElement) => { const uiType = ui.type; @@ -38,22 +40,101 @@ const getComponentName = (ui: ReactElement) => { return typed.displayName || typed.name || 'AnonymousComponent'; }; -const emitRenderMetric = (componentName: string, durationMs: number) => { - const metricsEnabled = process.env.POKU_REACT_ENABLE_METRICS === '1'; - if (!metricsEnabled) return; - if (typeof process.send !== 'function') return; +const runtimeOptions = parseRuntimeOptions(); +const metricsEnabled = runtimeOptions.metricsEnabled; +const minMetricMs = runtimeOptions.minMetricMs; + +type QueuedRenderMetric = { + componentName: string; + durationMs: number; +}; + +const metricBuffer: QueuedRenderMetric[] = []; +let metricFlushTimer: ReturnType | undefined; +let metricsChannelClosed = false; + +const flushMetricBuffer = () => { + if (!metricsEnabled || typeof process.send !== 'function') return; + + if (process.connected === false) { + metricBuffer.length = 0; + metricsChannelClosed = true; + return; + } + + if (metricsChannelClosed || metricBuffer.length === 0) return; - const safeDuration = Number.isFinite(durationMs) && durationMs >= 0 ? durationMs : 0; + const payload = metricBuffer.splice(0, metricBuffer.length); try { process.send({ - type: 'POKU_REACT_RENDER_METRIC', - componentName, - durationMs: safeDuration, + type: 'POKU_REACT_RENDER_METRIC_BATCH', + metrics: payload, }); } catch { - // Ignore IPC failures when the process channel is already closed. + metricsChannelClosed = true; + metricBuffer.length = 0; + } +}; + +const clearMetricFlushTimer = () => { + if (!metricFlushTimer) return; + clearTimeout(metricFlushTimer); + metricFlushTimer = undefined; +}; + +const scheduleMetricFlush = () => { + if (metricFlushTimer) return; + + metricFlushTimer = setTimeout(() => { + metricFlushTimer = undefined; + flushMetricBuffer(); + }, runtimeOptions.metricFlushMs); + + metricFlushTimer.unref?.(); +}; + +if (metricsEnabled) { + process.on('beforeExit', () => { + clearMetricFlushTimer(); + flushMetricBuffer(); + }); + + process.on('disconnect', () => { + clearMetricFlushTimer(); + metricBuffer.length = 0; + metricsChannelClosed = true; + }); +} + +const emitRenderMetric = (componentName: string, durationMs: number) => { + if (!metricsEnabled || typeof process.send !== 'function') return; + + if (process.connected === false || metricsChannelClosed) { + metricBuffer.length = 0; + metricsChannelClosed = true; + clearMetricFlushTimer(); + return; + } + + const safeDuration = + Number.isFinite(durationMs) && durationMs >= 0 ? durationMs : 0; + + // Optimization: Drop metrics below the threshold to prevent IPC flooding + if (safeDuration < minMetricMs) return; + + metricBuffer.push({ + componentName, + durationMs: safeDuration, + }); + + if (metricBuffer.length >= runtimeOptions.metricBatchSize) { + clearMetricFlushTimer(); + flushMetricBuffer(); + return; } + + scheduleMetricFlush(); }; const wrapUi = (ui: ReactElement, Wrapper?: WrapperComponent) => @@ -76,7 +157,10 @@ export type RenderResult = BoundFunctions & { /** * Render a React element in an isolated container and return bound DOM queries. */ -export const render = (ui: ReactElement, options: RenderOptions = {}): RenderResult => { +export const render = ( + ui: ReactElement, + options: RenderOptions = {} +): RenderResult => { const baseElement = options.baseElement || document.body; const container = options.container || document.createElement('div'); const ownsContainer = !options.container; @@ -103,15 +187,19 @@ export const render = (ui: ReactElement, options: RenderOptions = {}): RenderRes const unmount = () => { if (!mountedRoots.has(mounted)) return; - act(() => { - root.unmount(); - }); - - if (mounted.ownsContainer && container.parentNode) { - container.parentNode.removeChild(container); + try { + act(() => { + mounted.root?.unmount(); + }); + } finally { + if (mounted.ownsContainer && mounted.container?.parentNode) { + mounted.container.parentNode.removeChild(mounted.container); + } + + mounted.root = null; + mounted.container = null; + mountedRoots.delete(mounted); } - - mountedRoots.delete(mounted); }; const rerender = (nextUi: ReactElement) => { @@ -149,9 +237,12 @@ export type RenderHookResult = { /** * Render a hook directly and expose the latest hook value via `result.current`. */ -export const renderHook = = Record>( +export const renderHook = < + Result, + Props extends Record = Record, +>( hook: (props: Props) => Result, - options: RenderHookOptions = {}, + options: RenderHookOptions = {} ): RenderHookResult => { let currentResult!: Result; @@ -160,7 +251,7 @@ export const renderHook = = Recor return null; }; - const initialProps = (options.initialProps || ({} as Props)); + const initialProps = options.initialProps || ({} as Props); const view = render(React.createElement(HookHarness, initialProps), options); return { @@ -178,54 +269,88 @@ export const renderHook = = Recor * Unmount all rendered roots and remove owned containers from the document. */ export const cleanup = () => { - for (const mounted of [...mountedRoots]) { - act(() => { - mounted.root.unmount(); - }); + const mountedEntries = [...mountedRoots]; - if (mounted.ownsContainer && mounted.container.parentNode) { - mounted.container.parentNode.removeChild(mounted.container); + for (const mounted of mountedEntries) { + try { + act(() => { + mounted.root?.unmount(); + }); + } finally { + if (mounted.ownsContainer && mounted.container?.parentNode) { + mounted.container.parentNode.removeChild(mounted.container); + } + + mounted.root = null; + mounted.container = null; + mountedRoots.delete(mounted); } - - mountedRoots.delete(mounted); } + + flushMetricBuffer(); }; /** * Global Testing Library `screen` bound to `document.body`. */ -export const screen = new Proxy({} as Screen, { - get(_, key: PropertyKey) { - const queries = getQueriesForElement(document.body) as Record; - const value = queries[key]; - return typeof value === 'function' ? value.bind(queries) : value; - }, -}) as Screen; +const baseScreenQueries = getQueriesForElement(document.body) as Record< + string, + unknown +>; + +const boundScreenQueries: Record = {}; +for (const key of Object.keys(baseScreenQueries)) { + const value = baseScreenQueries[key]; + boundScreenQueries[key] = + typeof value === 'function' + ? (value as (...args: unknown[]) => unknown).bind(baseScreenQueries) + : value; +} + +export const screen = boundScreenQueries as Screen; /** * Testing Library `fireEvent` wrapped in React `act` for synchronous state flushing. */ -export const fireEvent = new Proxy(domTestingLibrary.fireEvent, { - get(target, key: keyof typeof domTestingLibrary.fireEvent) { - const value = target[key]; - if (typeof value !== 'function') return value; +const baseFireEventInstance = domTestingLibrary.fireEvent; + +const wrappedFireEvent = ((...args: Parameters) => { + let result!: ReturnType; + act(() => { + result = baseFireEventInstance(...args); + }); + return result; +}) as typeof baseFireEvent; - return (...args: unknown[]) => { - let result: unknown; - act(() => { - result = Reflect.apply(value as (...innerArgs: unknown[]) => unknown, target, args); - }); - return result; - }; - }, +for (const key of Object.keys(baseFireEventInstance) as Array< + keyof typeof baseFireEventInstance +>) { + const value = baseFireEventInstance[key]; + + if (typeof value !== 'function') { + ( + wrappedFireEvent as unknown as Record< + keyof typeof baseFireEventInstance, + unknown + > + )[key] = value; + continue; + } - apply(target, thisArg, args) { + ( + wrappedFireEvent as unknown as Record< + keyof typeof baseFireEventInstance, + unknown + > + )[key] = (...args: unknown[]) => { let result: unknown; act(() => { - result = Reflect.apply(target, thisArg, args); + result = Reflect.apply(value, baseFireEventInstance, args); }); return result; - }, -}) as typeof baseFireEvent; + }; +} + +export const fireEvent = wrappedFireEvent; export * from '@testing-library/dom'; diff --git a/src/runtime-options.ts b/src/runtime-options.ts new file mode 100644 index 0000000..eb38d7b --- /dev/null +++ b/src/runtime-options.ts @@ -0,0 +1,79 @@ +const DEFAULT_DOM_URL = 'http://localhost:3000/'; +const DEFAULT_METRIC_BATCH_SIZE = 50; +const DEFAULT_METRIC_FLUSH_MS = 50; + +const METRICS_FLAG_PREFIX = '--poku-react-metrics='; +const MIN_METRIC_MS_PREFIX = '--poku-react-min-metric-ms='; +const DOM_URL_PREFIX = '--poku-react-dom-url='; +const METRIC_BATCH_SIZE_PREFIX = '--poku-react-metric-batch-size='; +const METRIC_FLUSH_MS_PREFIX = '--poku-react-metric-flush-ms='; + +export type RuntimeOptions = { + domUrl: string; + metricsEnabled: boolean; + minMetricMs: number; + metricBatchSize: number; + metricFlushMs: number; +}; + +const parseFlagValue = (prefix: string, argv: string[]) => { + for (const arg of argv) { + if (!arg.startsWith(prefix)) continue; + return arg.slice(prefix.length); + } + + return undefined; +}; + +const parseBooleanFlag = (value: string | undefined) => { + if (!value) return false; + return value === '1' || value.toLowerCase() === 'true'; +}; + +const parseFiniteNumber = (value: string | undefined, fallback: number) => { + if (!value || value.trim() === '') return fallback; + + const parsed = Number(value.trim()); + if (!Number.isFinite(parsed)) return fallback; + + return parsed; +}; + +const parsePositiveInteger = (value: string | undefined, fallback: number) => { + const parsed = parseFiniteNumber(value, fallback); + if (parsed <= 0) return fallback; + return Math.floor(parsed); +}; + +export const parseRuntimeOptions = ( + argv: string[] = process.argv +): RuntimeOptions => { + const metricsEnabled = parseBooleanFlag( + parseFlagValue(METRICS_FLAG_PREFIX, argv) + ); + + return { + domUrl: parseFlagValue(DOM_URL_PREFIX, argv) || DEFAULT_DOM_URL, + metricsEnabled, + minMetricMs: Math.max( + 0, + parseFiniteNumber(parseFlagValue(MIN_METRIC_MS_PREFIX, argv), 0) + ), + metricBatchSize: parsePositiveInteger( + parseFlagValue(METRIC_BATCH_SIZE_PREFIX, argv), + DEFAULT_METRIC_BATCH_SIZE + ), + metricFlushMs: parsePositiveInteger( + parseFlagValue(METRIC_FLUSH_MS_PREFIX, argv), + DEFAULT_METRIC_FLUSH_MS + ), + }; +}; + +export const runtimeOptionArgPrefixes = { + metrics: METRICS_FLAG_PREFIX, + minMetricMs: MIN_METRIC_MS_PREFIX, + domUrl: DOM_URL_PREFIX, + metricBatchSize: METRIC_BATCH_SIZE_PREFIX, + metricFlushMs: METRIC_FLUSH_MS_PREFIX, +}; diff --git a/tests/plugin-internals.test.ts b/tests/plugin-internals.test.ts index 397de89..032920f 100644 --- a/tests/plugin-internals.test.ts +++ b/tests/plugin-internals.test.ts @@ -1,23 +1,6 @@ import { assert, test } from 'poku'; import { __internal } from '../src/plugin.ts'; -const ORIGINAL_DOM_URL = process.env.POKU_REACT_DOM_URL; -const ORIGINAL_METRICS = process.env.POKU_REACT_ENABLE_METRICS; - -const restoreEnvBaseline = () => { - if (typeof ORIGINAL_DOM_URL === 'undefined') { - delete process.env.POKU_REACT_DOM_URL; - } else { - process.env.POKU_REACT_DOM_URL = ORIGINAL_DOM_URL; - } - - if (typeof ORIGINAL_METRICS === 'undefined') { - delete process.env.POKU_REACT_ENABLE_METRICS; - } else { - process.env.POKU_REACT_ENABLE_METRICS = ORIGINAL_METRICS; - } -}; - test('normalizes metrics defaults when disabled', () => { const normalized = __internal.normalizeMetricsOptions(undefined); @@ -50,6 +33,7 @@ test('buildRunnerCommand injects tsx and dom setup for node', () => { command: ['node', '--trace-warnings', 'tests/example.test.tsx'], file: 'tests/example.test.tsx', domSetupPath: '/tmp/dom-setup.ts', + runtimeOptionArgs: ['--poku-react-dom-url=http://example.local/'], }); assert.strictEqual(result.shouldHandle, true); @@ -59,6 +43,7 @@ test('buildRunnerCommand injects tsx and dom setup for node', () => { '--import=tsx', '--import=/tmp/dom-setup.ts', 'tests/example.test.tsx', + '--poku-react-dom-url=http://example.local/', ]); }); @@ -68,6 +53,7 @@ test('buildRunnerCommand injects dom setup for bun without tsx import', () => { command: ['bun', 'tests/example.test.tsx'], file: 'tests/example.test.tsx', domSetupPath: '/tmp/dom-setup.ts', + runtimeOptionArgs: ['--poku-react-metrics=1'], }); assert.strictEqual(result.shouldHandle, true); @@ -75,6 +61,7 @@ test('buildRunnerCommand injects dom setup for bun without tsx import', () => { 'bun', '--import=/tmp/dom-setup.ts', 'tests/example.test.tsx', + '--poku-react-metrics=1', ]); }); @@ -84,6 +71,7 @@ test('buildRunnerCommand injects preload for deno', () => { command: ['deno', 'run', '-A', 'tests/example.test.tsx'], file: 'tests/example.test.tsx', domSetupPath: '/tmp/dom-setup.ts', + runtimeOptionArgs: ['--poku-react-min-metric-ms=1.5'], }); assert.strictEqual(result.shouldHandle, true); @@ -93,6 +81,7 @@ test('buildRunnerCommand injects preload for deno', () => { '-A', '--preload=/tmp/dom-setup.ts', 'tests/example.test.tsx', + '--poku-react-min-metric-ms=1.5', ]); }); @@ -103,6 +92,7 @@ test('buildRunnerCommand leaves unsupported runtime unchanged', () => { command: original, file: 'tests/example.test.tsx', domSetupPath: '/tmp/dom-setup.ts', + runtimeOptionArgs: [], }); assert.strictEqual(result.shouldHandle, false); @@ -116,28 +106,59 @@ test('buildRunnerCommand leaves non-react extension unchanged', () => { command: original, file: 'tests/example.test.ts', domSetupPath: '/tmp/dom-setup.ts', + runtimeOptionArgs: [], }); assert.strictEqual(result.shouldHandle, false); assert.deepStrictEqual(result.command, original); }); -test('environment helpers apply and restore options', () => { - restoreEnvBaseline(); +test('buildRunnerCommand avoids duplicate runtime args', () => { + const result = __internal.buildRunnerCommand({ + runtime: 'node', + command: [ + 'node', + '--trace-warnings', + 'tests/example.test.tsx', + '--poku-react-metrics=1', + ], + file: 'tests/example.test.tsx', + domSetupPath: '/tmp/dom-setup.ts', + runtimeOptionArgs: ['--poku-react-metrics=1'], + }); + + assert.deepStrictEqual(result.command, [ + 'node', + '--trace-warnings', + '--import=tsx', + '--import=/tmp/dom-setup.ts', + 'tests/example.test.tsx', + '--poku-react-metrics=1', + ]); +}); - const snapshot = __internal.captureEnvSnapshot(); - __internal.applyEnvironmentOptions( +test('buildRuntimeOptionArgs creates argv-safe plugin flags', () => { + const args = __internal.buildRuntimeOptionArgs( { domUrl: 'http://example.local/', metrics: true }, - __internal.normalizeMetricsOptions(true), + __internal.normalizeMetricsOptions({ enabled: true, minDurationMs: 2.75 }) ); - assert.strictEqual(process.env.POKU_REACT_DOM_URL, 'http://example.local/'); - assert.strictEqual(process.env.POKU_REACT_ENABLE_METRICS, '1'); + assert.deepStrictEqual(args, [ + '--poku-react-dom-url=http://example.local/', + '--poku-react-metrics=1', + '--poku-react-min-metric-ms=2.75', + ]); +}); - __internal.restoreEnvironmentOptions(snapshot); +test('normalizeMetricsOptions ignores invalid non-number containers', () => { + const normalized = __internal.normalizeMetricsOptions({ + enabled: true, + topN: [42] as unknown as number, + minDurationMs: ['4'] as unknown as number, + }); - assert.strictEqual(process.env.POKU_REACT_DOM_URL, ORIGINAL_DOM_URL); - assert.strictEqual(process.env.POKU_REACT_ENABLE_METRICS, ORIGINAL_METRICS); + assert.strictEqual(normalized.topN, 5); + assert.strictEqual(normalized.minDurationMs, 0); }); test('createMetricsSummary returns ordered top metrics with filters', () => { @@ -149,7 +170,11 @@ test('createMetricsSummary returns ordered top metrics with filters', () => { const summary = __internal.createMetricsSummary( metrics, - __internal.normalizeMetricsOptions({ enabled: true, topN: 2, minDurationMs: 1 }), + __internal.normalizeMetricsOptions({ + enabled: true, + topN: 2, + minDurationMs: 1, + }) ); assert.ok(summary); @@ -157,14 +182,14 @@ test('createMetricsSummary returns ordered top metrics with filters', () => { assert.strictEqual(summary?.totalReported, 2); assert.deepStrictEqual( summary?.topSlowest.map((item) => item.componentName), - ['B', 'C'], + ['B', 'C'] ); }); test('createMetricsSummary returns null when disabled', () => { const summary = __internal.createMetricsSummary( [{ file: 'a', componentName: 'A', durationMs: 8 }], - __internal.normalizeMetricsOptions(false), + __internal.normalizeMetricsOptions(false) ); assert.strictEqual(summary, null); @@ -177,15 +202,41 @@ test('getComponentName falls back for non-string values', () => { }); test('isRenderMetricMessage validates expected payloads', () => { - assert.strictEqual(__internal.isRenderMetricMessage({ type: 'POKU_REACT_RENDER_METRIC' }), true); - assert.strictEqual(__internal.isRenderMetricMessage({ type: 'OTHER' }), false); + assert.strictEqual( + __internal.isRenderMetricMessage({ type: 'POKU_REACT_RENDER_METRIC' }), + true + ); + assert.strictEqual( + __internal.isRenderMetricMessage({ type: 'OTHER' }), + false + ); assert.strictEqual(__internal.isRenderMetricMessage(null), false); }); +test('isRenderMetricBatchMessage validates batched payloads', () => { + assert.strictEqual( + __internal.isRenderMetricBatchMessage({ + type: 'POKU_REACT_RENDER_METRIC_BATCH', + metrics: [{ componentName: 'A', durationMs: 1.2 }], + }), + true + ); + + assert.strictEqual( + __internal.isRenderMetricBatchMessage({ + type: 'POKU_REACT_RENDER_METRIC_BATCH', + }), + false + ); + assert.strictEqual(__internal.isRenderMetricBatchMessage(null), false); +}); + test('resolveDomSetupPath resolves built-in and custom adapters', () => { const happyPath = __internal.resolveDomSetupPath('happy-dom'); const jsdomPath = __internal.resolveDomSetupPath('jsdom'); - const customPath = __internal.resolveDomSetupPath({ setupModule: 'tests/setup/custom.ts' }); + const customPath = __internal.resolveDomSetupPath({ + setupModule: 'tests/setup/custom.ts', + }); assert.ok(happyPath.includes('dom-setup-happy')); assert.ok(jsdomPath.includes('dom-setup-jsdom')); diff --git a/tests/react-concurrency.test.tsx b/tests/react-concurrency.test.tsx index 63bf2bb..e0b2ac6 100644 --- a/tests/react-concurrency.test.tsx +++ b/tests/react-concurrency.test.tsx @@ -1,5 +1,5 @@ -import { Suspense, use, useState, useTransition } from 'react'; import { afterEach, assert, test } from 'poku'; +import { Suspense, use, useState, useTransition } from 'react'; import { cleanup, fireEvent, render, screen } from '../src/index.ts'; afterEach(cleanup); @@ -19,15 +19,18 @@ test('renders a resolved use() resource under Suspense', () => { } as unknown as Promise; render( - Resource pending...}> + Resource pending...}> - , + ); - assert.strictEqual(screen.getByRole('heading', { level: 2 }).textContent, value); + assert.strictEqual( + screen.getByRole('heading', { level: 2 }).textContent, + value + ); }); -test('runs urgent and transition update pipeline', () => { +await test('runs urgent and transition update pipeline', async () => { const TransitionPipeline = () => { const [urgentState, setUrgentState] = useState('idle'); const [deferredState, setDeferredState] = useState('idle'); @@ -36,7 +39,7 @@ test('runs urgent and transition update pipeline', () => { return (
- {urgentState} - {deferredState} - {isPending ? 'pending' : 'settled'} + {urgentState} + {deferredState} + + {isPending ? 'pending' : 'settled'} +
); }; @@ -56,10 +61,19 @@ test('runs urgent and transition update pipeline', () => { render(); assert.strictEqual(screen.getByLabelText('urgent-state').textContent, 'idle'); - assert.strictEqual(screen.getByLabelText('deferred-state').textContent, 'idle'); + assert.strictEqual( + screen.getByLabelText('deferred-state').textContent, + 'idle' + ); fireEvent.click(screen.getByRole('button', { name: 'Run pipeline' })); - assert.strictEqual(screen.getByLabelText('urgent-state').textContent, 'urgent-updated'); - assert.strictEqual(screen.getByLabelText('deferred-state').textContent, 'transition-updated'); + assert.strictEqual( + screen.getByLabelText('urgent-state').textContent, + 'urgent-updated' + ); + assert.strictEqual( + screen.getByLabelText('deferred-state').textContent, + 'transition-updated' + ); }); diff --git a/tests/react-context.test.tsx b/tests/react-context.test.tsx index 4649195..f675afa 100644 --- a/tests/react-context.test.tsx +++ b/tests/react-context.test.tsx @@ -1,5 +1,5 @@ -import { createContext, useContext } from 'react'; import { afterEach, assert, test } from 'poku'; +import { createContext, useContext } from 'react'; import { cleanup, render, screen } from '../src/index.ts'; afterEach(cleanup); @@ -13,10 +13,13 @@ const ThemeLabel = () => { test('injects context values via wrapper', () => { const ThemeWrapper = ({ children }: { children?: React.ReactNode }) => ( - {children} + {children} ); render(, { wrapper: ThemeWrapper }); - assert.strictEqual(screen.getByText('Theme: dark').textContent, 'Theme: dark'); + assert.strictEqual( + screen.getByText('Theme: dark').textContent, + 'Theme: dark' + ); }); diff --git a/tests/react-hooks.test.tsx b/tests/react-hooks.test.tsx index ff3568c..acd0ad2 100644 --- a/tests/react-hooks.test.tsx +++ b/tests/react-hooks.test.tsx @@ -1,6 +1,12 @@ -import { useMemo, useState } from 'react'; import { afterEach, assert, test } from 'poku'; -import { cleanup, fireEvent, render, renderHook, screen } from '../src/index.ts'; +import { useMemo, useState } from 'react'; +import { + cleanup, + fireEvent, + render, + renderHook, + screen, +} from '../src/index.ts'; afterEach(cleanup); @@ -16,8 +22,10 @@ const HookHarness = () => { return (
- {enabled ? 'enabled' : 'disabled'} -
@@ -27,15 +35,24 @@ const HookHarness = () => { test('tests custom hooks through a component harness', () => { render(); - assert.strictEqual(screen.getByLabelText('toggle-state').textContent, 'disabled'); + assert.strictEqual( + screen.getByLabelText('toggle-state').textContent, + 'disabled' + ); fireEvent.click(screen.getByRole('button', { name: 'Toggle' })); - assert.strictEqual(screen.getByLabelText('toggle-state').textContent, 'enabled'); + assert.strictEqual( + screen.getByLabelText('toggle-state').textContent, + 'enabled' + ); }); test('tests hook logic directly with renderHook', () => { - const { result } = renderHook(({ initial }: { initial: boolean }) => useToggle(initial), { - initialProps: { initial: true }, - }); + const { result } = renderHook( + ({ initial }: { initial: boolean }) => useToggle(initial), + { + initialProps: { initial: true }, + } + ); assert.strictEqual(result.current.enabled, true); }); diff --git a/tests/react-lifecycle.test.tsx b/tests/react-lifecycle.test.tsx index 5941c1f..43be2a3 100644 --- a/tests/react-lifecycle.test.tsx +++ b/tests/react-lifecycle.test.tsx @@ -1,5 +1,5 @@ -import { useEffect } from 'react'; import { afterEach, assert, test } from 'poku'; +import { useEffect } from 'react'; import { cleanup, render, screen } from '../src/index.ts'; afterEach(cleanup); @@ -7,13 +7,19 @@ afterEach(cleanup); const Greeting = ({ name }: { name: string }) =>

Hello {name}

; test('rerender updates component props in place', () => { - const view = render(); + const view = render(); - assert.strictEqual(screen.getByRole('heading', { level: 3 }).textContent, 'Hello Ada'); + assert.strictEqual( + screen.getByRole('heading', { level: 3 }).textContent, + 'Hello Ada' + ); - view.rerender(); + view.rerender(); - assert.strictEqual(screen.getByRole('heading', { level: 3 }).textContent, 'Hello Grace'); + assert.strictEqual( + screen.getByRole('heading', { level: 3 }).textContent, + 'Hello Grace' + ); }); test('unmount runs effect cleanup logic', () => { diff --git a/tests/react-plugin.test.tsx b/tests/react-plugin.test.tsx index 6a25e78..ea97302 100644 --- a/tests/react-plugin.test.tsx +++ b/tests/react-plugin.test.tsx @@ -1,7 +1,7 @@ -import { useState } from 'react'; import { afterEach, assert, test } from 'poku'; +import { useState } from 'react'; import { cleanup, fireEvent, render, screen } from '../src/index.ts'; - + afterEach(cleanup); type CounterProps = { @@ -18,18 +18,20 @@ const Counter = ({ initialCount = 0 }: CounterProps) => { return (

Count: {count}

-
); }; - test('renders and updates a React component', () => { render(); - assert.strictEqual(screen.getByRole('heading', { name: 'Count: 1' }).textContent, 'Count: 1'); + assert.strictEqual( + screen.getByRole('heading', { name: 'Count: 1' }).textContent, + 'Count: 1' + ); fireEvent.click(screen.getByRole('button', { name: 'Increment' })); diff --git a/tsconfig.json b/tsconfig.json index 90f1150..107d614 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -17,5 +17,11 @@ "resolveJsonModule": true, "types": ["node", "react", "react-dom"] }, - "include": ["src/**/*.ts", "src/**/*.tsx", "src/**/*.d.ts", "tests/**/*.ts", "tests/**/*.tsx"] -} \ No newline at end of file + "include": [ + "src/**/*.ts", + "src/**/*.tsx", + "src/**/*.d.ts", + "tests/**/*.ts", + "tests/**/*.tsx" + ] +}