diff --git a/.github/workflows/benchmark.yml b/.github/workflows/benchmark.yml index 9379367..76dec13 100644 --- a/.github/workflows/benchmark.yml +++ b/.github/workflows/benchmark.yml @@ -79,31 +79,31 @@ jobs: const fs = require('fs'); let prResults = null; let mainResults = null; - + // Check if PR results exist if (fs.existsSync('/tmp/pr-results.json')) { prResults = JSON.parse(fs.readFileSync('/tmp/pr-results.json', 'utf8')); } else { console.log('PR benchmark results not found'); } - + // Check if main results exist if (fs.existsSync('/tmp/main-results.json')) { mainResults = JSON.parse(fs.readFileSync('/tmp/main-results.json', 'utf8')); } else { console.log('Main benchmark results not found'); } - + // Exit early if we don't have both results if (!prResults || !mainResults) { console.log('Missing benchmark results, skipping comparison'); process.exit(0); } - + const REGRESSION_THRESHOLD = 0.20; // 20% const regressions = []; const improvements = []; - + for (const prBench of prResults.results || []) { const mainBench = (mainResults.results || []).find(b => b.name === prBench.name); if (!mainBench) continue; @@ -131,20 +131,20 @@ jobs: }); } } - + console.log(JSON.stringify({ regressions, improvements }, null, 2)); - + if (regressions.length > 0) { process.exit(1); } EOF - + node /tmp/compare.js | tee /tmp/comparison.json - + const REGRESSION_THRESHOLD = 0.20; // 20% const regressions = []; const improvements = []; - + for (const prBench of prResults.results || []) { const mainBench = (mainResults.results || []).find(b => b.name === prBench.name); if (!mainBench) continue; @@ -172,14 +172,14 @@ jobs: }); } } - + console.log(JSON.stringify({ regressions, improvements }, null, 2)); - + if (regressions.length > 0) { process.exit(1); } EOF - + node /tmp/compare.js | tee /tmp/comparison.json - name: Comment on PR (regressions) @@ -188,13 +188,13 @@ jobs: with: script: | const fs = require('fs'); - + // Check if comparison file exists if (!fs.existsSync('/tmp/comparison.json')) { console.log('No comparison data available - benchmark results may be missing'); return; } - + let comparison; try { comparison = JSON.parse(fs.readFileSync('/tmp/comparison.json', 'utf8')); @@ -202,29 +202,29 @@ jobs: console.log('Failed to parse comparison data:', error.message); return; } - + const { regressions, improvements } = comparison; - + if (regressions.length === 0) { console.log('No regressions found'); return; } - + let comment = '## 🚨 Benchmark Regression Detected\n\n'; comment += 'The following benchmarks regressed by more than 20%:\n\n'; comment += '| Benchmark | Main (p50) | PR (p50) | Change |\n'; comment += '|-----------|-----------|---------|--------|\n'; - + for (const reg of regressions) { comment += `| ${reg.name} | ${reg.mainMedian}ms | ${reg.prMedian}ms | **+${reg.change}%** |\n`; } - + comment += '\n**Action**: Please investigate the performance regression and either:\n'; comment += '1. Optimize the code to meet baseline\n'; comment += '2. Update the baseline if the regression is acceptable\n'; comment += '3. File a perf follow-up if the change is necessary\n\n'; comment += 'See `bench/README.md` for interpretation guidance and `bench/baseline.md` for baseline numbers.\n'; - + if (improvements.length > 0) { comment += '\n### āœ… Improvements Detected\n\n'; comment += '| Benchmark | Main (p50) | PR (p50) | Change |\n'; @@ -234,7 +234,7 @@ jobs: comment += `| ${imp.name} | ${imp.mainMedian}ms | ${imp.prMedian}ms | ${imp.change}% |\n`; } } - + github.rest.issues.createComment({ issue_number: context.issue.number, owner: context.repo.owner, @@ -254,10 +254,10 @@ jobs: } const comparison = JSON.parse(fs.readFileSync('/tmp/comparison.json', 'utf8')); const { improvements } = comparison; - + let comment = '## āœ… Benchmarks Passed\n\n'; comment += 'All benchmarks are within regression threshold (20%).\n'; - + if (improvements.length > 0) { comment += '\n### šŸŽ‰ Improvements Detected\n\n'; comment += '| Benchmark | Main (p50) | PR (p50) | Change |\n'; @@ -267,7 +267,7 @@ jobs: comment += `| ${imp.name} | ${imp.mainMedian}ms | ${imp.prMedian}ms | ${imp.change}% |\n`; } } - + github.rest.issues.createComment({ issue_number: context.issue.number, owner: context.repo.owner, diff --git a/BENCHMARK_IMPLEMENTATION.md b/BENCHMARK_IMPLEMENTATION.md index e48955f..1a146a9 100644 --- a/BENCHMARK_IMPLEMENTATION.md +++ b/BENCHMARK_IMPLEMENTATION.md @@ -6,6 +6,7 @@ ## Deliverables ### āœ… 1. Benchmark Harness + - **Location**: `test/chains/stellar/bench/stellar.bench.ts` - **Tool**: Vitest benchmark mode (native, no external dependency needed; tinybench added as fallback) - **Run command**: `pnpm bench` (or `pnpm bench:watch` for watch mode) @@ -13,26 +14,28 @@ ### āœ… 2. Comprehensive Benchmarks for Stellar -| Operation | Coverage | -|-----------|----------| -| Key Derivation | deriveStealthKeys (single) | -| Address Generation | generateStealthAddress (single) | -| Meta-addressing | encodeStealthMetaAddress, decodeStealthMetaAddress, round-trip | -| Private Key | deriveStealthPrivateScalar (single) | -| Signing | signWithScalar (single) | +| Operation | Coverage | +| --------------------- | -------------------------------------------------------------------- | +| Key Derivation | deriveStealthKeys (single) | +| Address Generation | generateStealthAddress (single) | +| Meta-addressing | encodeStealthMetaAddress, decodeStealthMetaAddress, round-trip | +| Private Key | deriveStealthPrivateScalar (single) | +| Signing | signWithScalar (single) | | Announcement Scanning | checkStealthAddress, scanAnnouncements at N={10, 100, 1K, 10K, 100K} | -| Network | fetchAnnouncements (mocked RPC) | +| Network | fetchAnnouncements (mocked RPC) | **Total benchmarks**: 15 individual test cases ### āœ… 3. Configuration Updates **package.json**: + - Added `bench` script: `vitest bench --run` - Added `bench:watch` script: `vitest bench` - Added dev dependencies: `tinybench@^2.9.0` **vitest.config.ts**: + - Configured benchmark discovery: `test/chains/**/bench/**/*.bench.ts` - Output: JSON results to `bench/results.json` - Excluded bench files from unit tests @@ -40,6 +43,7 @@ ### āœ… 4. Documentation **bench/README.md** (comprehensive guide): + - Hardware baseline specifications - How to interpret benchmark results (hz, min, max, p50, p99) - How to compare against previous runs @@ -50,6 +54,7 @@ - CI integration overview **bench/baseline.md** (baseline report): + - Full hardware specifications (CPU, RAM, OS, Node.js version) - Per-benchmark results with p50/p99 statistics - Summary table showing linear scaling for scanAnnouncements @@ -64,6 +69,7 @@ **Location**: `.github/workflows/benchmark.yml` **Functionality**: + - Triggers on every PR to `main` and `develop` - Runs benchmarks on PR branch - Checks out and runs benchmarks on main branch @@ -75,6 +81,7 @@ - Improvements detected (celebrates wins) **Comments include**: + - Table of regressions/improvements with exact numbers - Actionable guidance for developers - Link to documentation @@ -82,6 +89,7 @@ ### āœ… 6. Identified Hot Path **Primary**: `scanAnnouncements` ECDH Loop + - **Problem**: Calls ECDH scalar multiplication N times (once per announcement) - **Current cost**: ~2.2 seconds for 100k announcements - **Root cause**: `computeSharedSecret()` uses Curve25519 scalar mult, which is expensive @@ -93,12 +101,12 @@ ## Acceptance Criteria Met -| Criterion | Status | Evidence | -|-----------|--------|----------| -| Bench harness committed and runnable via pnpm bench | āœ… | `pnpm bench` configured in package.json; runs stellar.bench.ts | -| Baseline report with hardware spec and per-benchmark p50/p99 | āœ… | bench/baseline.md includes full specs and all metrics | -| CI regression check wired up | āœ… | .github/workflows/benchmark.yml with 20% threshold and PR comments | -| Hot path documented with expected speedup | āœ… | bench/baseline.md documents scanAnnouncements ECDH loop (2–3x expected) | +| Criterion | Status | Evidence | +| ------------------------------------------------------------ | ------ | ----------------------------------------------------------------------- | +| Bench harness committed and runnable via pnpm bench | āœ… | `pnpm bench` configured in package.json; runs stellar.bench.ts | +| Baseline report with hardware spec and per-benchmark p50/p99 | āœ… | bench/baseline.md includes full specs and all metrics | +| CI regression check wired up | āœ… | .github/workflows/benchmark.yml with 20% threshold and PR comments | +| Hot path documented with expected speedup | āœ… | bench/baseline.md documents scanAnnouncements ECDH loop (2–3x expected) | ## Next Steps (Out of Scope) @@ -136,6 +144,7 @@ pnpm bench -- --include="scanAnnouncements" ### Reviewing PR Regression Results GitHub Actions will automatically post a comment on your PR showing: + - Which benchmarks regressed (if any) - By how much (%) - Links to baseline for comparison diff --git a/CONTEXT_BENCHMARKS_COMPLETE.md b/CONTEXT_BENCHMARKS_COMPLETE.md index f2dd34c..eed565a 100644 --- a/CONTEXT_BENCHMARKS_COMPLETE.md +++ b/CONTEXT_BENCHMARKS_COMPLETE.md @@ -13,15 +13,18 @@ Successfully implemented a comprehensive benchmark harness for the Wraith Protoc ## What Was Built ### 1. **Benchmark Harness** āœ… + - **Tool**: Vitest native benchmark mode (no compilation overhead) - **Command**: `pnpm bench` (or `pnpm bench:watch`) - **Results**: JSON output to `bench/results.json` - **Coverage**: 15 individual test cases across 11 benchmark suites ### 2. **Stellar Benchmarks** āœ… + Comprehensive coverage of all key cryptographic operations: **Single Operations**: + - Key derivation (deriveStealthKeys) — 0.028 ms p50 - Address generation (generateStealthAddress) — 0.80 ms p50 - Meta-address encoding/decoding — < 0.02 ms p50 @@ -29,16 +32,19 @@ Comprehensive coverage of all key cryptographic operations: - Signing (signWithScalar) — 0.925 ms p50 **At Scale**: + - Announcement scanning at N={10, 100, 1K, 10K, 100K} - Linear scaling confirmed (doubling N doubles time) - **Hot path identified**: 100k announcements scan in ~2.2 seconds **Network**: + - fetchAnnouncements (mocked Soroban RPC) ### 3. **Documentation** āœ… **bench/README.md** (200+ lines): + - Hardware baseline specifications - How to interpret p50/p99 metrics - How to compare against previous runs @@ -49,6 +55,7 @@ Comprehensive coverage of all key cryptographic operations: - CI integration guide **bench/baseline.md** (150+ lines): + - Full hardware specs (CPU, RAM, OS, Node.js) - Per-benchmark p50/p99 statistics - Summary table with scaling analysis @@ -58,6 +65,7 @@ Comprehensive coverage of all key cryptographic operations: - Next steps recommendations **BENCHMARK_IMPLEMENTATION.md**: + - This implementation summary - Quick start guide - Deliverables checklist @@ -65,6 +73,7 @@ Comprehensive coverage of all key cryptographic operations: ### 4. **CI Regression Testing** āœ… **.github/workflows/benchmark.yml**: + - Triggers on every PR to main/develop - Runs benchmarks on PR branch + main branch - Compares results automatically @@ -78,15 +87,15 @@ Comprehensive coverage of all key cryptographic operations: ## Performance Baseline -| Operation | p50 | Status | -|-----------|-----|--------| -| deriveStealthKeys | 0.028 ms | āœ“ Fast | -| generateStealthAddress | 0.80 ms | āœ“ OK | -| deriveStealthPrivateScalar | 0.835 ms | āœ“ OK | -| signWithScalar | 0.925 ms | āœ“ OK | -| scanAnnouncements (100) | 2.23 ms | āœ“ OK | -| scanAnnouncements (1K) | 22.3 ms | ⚠ Noticed | -| scanAnnouncements (10K) | 223 ms | šŸ”“ SLOW | +| Operation | p50 | Status | +| ---------------------------- | ------------ | --------------- | +| deriveStealthKeys | 0.028 ms | āœ“ Fast | +| generateStealthAddress | 0.80 ms | āœ“ OK | +| deriveStealthPrivateScalar | 0.835 ms | āœ“ OK | +| signWithScalar | 0.925 ms | āœ“ OK | +| scanAnnouncements (100) | 2.23 ms | āœ“ OK | +| scanAnnouncements (1K) | 22.3 ms | ⚠ Noticed | +| scanAnnouncements (10K) | 223 ms | šŸ”“ SLOW | | **scanAnnouncements (100K)** | **2,230 ms** | **šŸ”“ HOT PATH** | ### Hot Path Analysis @@ -96,6 +105,7 @@ Comprehensive coverage of all key cryptographic operations: **Current impact**: 100k announcements take ~2.2 seconds (p50) **Optimization opportunities**: + 1. **Batch ECDH** (3–5x speedup) — Use crypto library batch operations 2. **SIMD acceleration** (2–3x speedup) — Hardware acceleration via WebAssembly 3. **Hash composition** (10–20% speedup) — Pre-compute contexts @@ -107,10 +117,12 @@ Comprehensive coverage of all key cryptographic operations: ## Files Delivered ### Modified + - `package.json` — Added bench scripts and dev dependencies - `vitest.config.ts` — Benchmark configuration ### Created + - `.github/workflows/benchmark.yml` — CI regression testing (140 lines) - `bench/README.md` — User guide (200+ lines) - `bench/baseline.md` — Baseline report (150+ lines) @@ -123,18 +135,19 @@ Comprehensive coverage of all key cryptographic operations: ## Acceptance Criteria Status -| Criterion | Status | Evidence | -|-----------|--------|----------| -| **Bench harness committed and runnable via pnpm bench** | āœ… | `pnpm bench` script, stellar.bench.ts with 15 test cases | -| **Baseline report with hardware spec and per-benchmark p50/p99** | āœ… | bench/baseline.md with full specs and all metrics | -| **CI regression check wired up** | āœ… | .github/workflows/benchmark.yml with 20% threshold, PR comments | -| **Hot path documented with expected speedup** | āœ… | bench/baseline.md "Identified Hot Paths" (2–3x ECDH speedup) | +| Criterion | Status | Evidence | +| ---------------------------------------------------------------- | ------ | --------------------------------------------------------------- | +| **Bench harness committed and runnable via pnpm bench** | āœ… | `pnpm bench` script, stellar.bench.ts with 15 test cases | +| **Baseline report with hardware spec and per-benchmark p50/p99** | āœ… | bench/baseline.md with full specs and all metrics | +| **CI regression check wired up** | āœ… | .github/workflows/benchmark.yml with 20% threshold, PR comments | +| **Hot path documented with expected speedup** | āœ… | bench/baseline.md "Identified Hot Paths" (2–3x ECDH speedup) | --- ## How to Use ### Install & Run + ```bash # First time setup pnpm install @@ -147,6 +160,7 @@ pnpm bench:watch ``` ### Interpret Results + ```bash # Results saved to bench/results.json # See bench/README.md for p50/p99 explanation @@ -154,6 +168,7 @@ pnpm bench:watch ``` ### On PRs + - GitHub Actions automatically runs benchmarks - Posts comment if regression > 20% - Blocks merge until resolved or investigated @@ -181,11 +196,13 @@ pnpm bench:watch ## Why This Matters **Before**: 134 correctness tests but no performance data + - Spectre's background scanner "feels slow at scale" but no measurements - Demo's receive page "feels slow" but no numbers - No way to verify performance claims or detect regressions -**After**: +**After**: + - āœ… Comprehensive performance baseline established - āœ… Hot paths identified and quantified (ECDH loop) - āœ… Every optimization claim is now verifiable @@ -201,11 +218,12 @@ pnpm bench:watch āœ… **Comprehensive documentation included** āœ… **CI regression testing configured** āœ… **Performance baseline established** -āœ… **Hot path identified and quantified** +āœ… **Hot path identified and quantified** --- **Created**: June 1, 2026 -**Commits**: +**Commits**: + - 7b20812: perf: Add comprehensive benchmark harness for Stellar stealth address operations - 984e433: docs: Add benchmark implementation summary diff --git a/bench/README.md b/bench/README.md index 08d9e16..e41b225 100644 --- a/bench/README.md +++ b/bench/README.md @@ -28,6 +28,7 @@ The baseline numbers in `baseline.md` were measured on: ## Interpreting Results Each benchmark reports: + - **hz**: Operations per second (higher is better) - **min**: Minimum execution time (ms) - **max**: Maximum execution time (ms) @@ -35,6 +36,7 @@ Each benchmark reports: - **p99**: 99th percentile execution time (ms) — rare slow cases Example output: + ``` scanAnnouncements - 1,000 announcements hz min max p50 p99 @@ -42,6 +44,7 @@ scanAnnouncements - 1,000 announcements ``` This means: + - We can scan 1,000 announcements **45.2 times per second** - Typical scan takes **22.3 ms** - 1 in 100 scans takes longer than **24.1 ms** @@ -81,6 +84,7 @@ diff <(jq '.results[] | {name: .name, hz: .hz}' bench/results.json) \ ### Regression Budget We use a **20% regression threshold** in CI. This means: + - If a benchmark gets 20% slower, it will be flagged for review - Small regressions (< 20%) are acceptable and expected as features evolve - Threshold is tunable in `.github/workflows/benchmark.yml` @@ -89,13 +93,13 @@ We use a **20% regression threshold** in CI. This means: Announcement scanning is the primary performance concern. We benchmark at multiple scales: -| Scale | Purpose | -|-------|---------| -| 10 | Minimum useful scan (e.g., last block's transactions) | -| 100 | Typical daily inbox | -| 1,000 | Weekly backlog | -| 10,000 | Monthly backlog | -| 100,000 | Full network sync or stress test | +| Scale | Purpose | +| ------- | ----------------------------------------------------- | +| 10 | Minimum useful scan (e.g., last block's transactions) | +| 100 | Typical daily inbox | +| 1,000 | Weekly backlog | +| 10,000 | Monthly backlog | +| 100,000 | Full network sync or stress test | **Expected behavior**: Time should grow linearly with N. If time grows faster (quadratic), it suggests a fixable algorithm bottleneck. @@ -104,11 +108,12 @@ Announcement scanning is the primary performance concern. We benchmark at multip ### Hot Path: `scanAnnouncements` Outer Loop The scanning function: + 1. Iterates through all N announcements (O(N)) 2. For each announcement, computes ECDH (Curve25519 scalar mult) — **expensive** 3. Filters by view-tag first (eliminates ~99.6% of announcements) — **cheap** -**Issue**: Even with view-tag filtering, ECDH is repeated 1,000+ times. +**Issue**: Even with view-tag filtering, ECDH is repeated 1,000+ times. **Optimization opportunity**: Batch ECDH operations or use a more cache-friendly implementation. @@ -117,6 +122,7 @@ The scanning function: ### Secondary: Hash Function Composition Each announcement scan computes: + - SHA-256("wraith:scalar:" || shared_secret) — called many times - Could benefit from streaming or pre-computed context @@ -130,6 +136,7 @@ Each announcement scan computes: 4. Run `pnpm bench` and add results to `baseline.md` Example: + ```typescript bench('newFunction() on 1MB input', () => { const input = generateLargeInput(1024 * 1024); @@ -140,6 +147,7 @@ bench('newFunction() on 1MB input', () => { ## CI Integration Every PR runs benchmarks and compares against main. If any benchmark regresses > 20%, a comment is posted to the PR with: + - Which benchmarks regressed - By how much (%) - Link to the regression tracking issue diff --git a/bench/baseline.md b/bench/baseline.md index e99a963..67cd897 100644 --- a/bench/baseline.md +++ b/bench/baseline.md @@ -17,58 +17,59 @@ All times in milliseconds (ms). Lower is better. ### Key Derivation -| Benchmark | hz | min | max | p50 | p99 | -|-----------|----|----|-----|-----|-----| +| Benchmark | hz | min | max | p50 | p99 | +| ------------------------------------------ | ------------ | ----- | ----- | ----- | ----- | | deriveStealthKeys (from 64-byte signature) | 35,850 ops/s | 0.026 | 0.082 | 0.028 | 0.055 | **Notes**: Very fast; deterministic function using SHA-256 and clamping. ### Address Generation -| Benchmark | hz | min | max | p50 | p99 | -|-----------|----|----|-----|-----|-----| +| Benchmark | hz | min | max | p50 | p99 | +| ---------------------- | ----------- | ---- | ---- | ---- | ---- | | generateStealthAddress | 1,245 ops/s | 0.78 | 1.24 | 0.80 | 1.05 | **Notes**: Involves ECDH (Curve25519) and point addition; this is the baseline for payment generation. ### Meta-address Encoding/Decoding -| Benchmark | hz | min | max | p50 | p99 | -|-----------|----|----|-----|-----|-----| -| encodeStealthMetaAddress | 185,200 ops/s | 0.0051 | 0.015 | 0.0054 | 0.012 | -| decodeStealthMetaAddress | 105,300 ops/s | 0.0090 | 0.025 | 0.0095 | 0.022 | -| encode + decode round-trip | 65,400 ops/s | 0.015 | 0.033 | 0.0153 | 0.030 | +| Benchmark | hz | min | max | p50 | p99 | +| -------------------------- | ------------- | ------ | ----- | ------ | ----- | +| encodeStealthMetaAddress | 185,200 ops/s | 0.0051 | 0.015 | 0.0054 | 0.012 | +| decodeStealthMetaAddress | 105,300 ops/s | 0.0090 | 0.025 | 0.0095 | 0.022 | +| encode + decode round-trip | 65,400 ops/s | 0.015 | 0.033 | 0.0153 | 0.030 | **Notes**: Mostly string parsing and validation; very cheap operations. ### Private Key Derivation -| Benchmark | hz | min | max | p50 | p99 | -|-----------|----|----|-----|-----|-----| +| Benchmark | hz | min | max | p50 | p99 | +| -------------------------- | ----------- | ---- | ---- | ----- | ---- | | deriveStealthPrivateScalar | 1,198 ops/s | 0.82 | 1.31 | 0.835 | 1.12 | **Notes**: ECDH + hash; called once per matched announcement. ### Signing -| Benchmark | hz | min | max | p50 | p99 | -|-----------|----|----|-----|-----|-----| +| Benchmark | hz | min | max | p50 | p99 | +| -------------- | ----------- | ---- | ---- | ----- | ---- | | signWithScalar | 1,086 ops/s | 0.91 | 1.48 | 0.925 | 1.35 | **Notes**: Full ed25519 signature; used to sign transactions. ### Announcement Scanning (View-Only) -| Benchmark | Count | hz | min | max | p50 | p99 | -|-----------|-------|----|----|-----|-----|-----| -| checkStealthAddress (single match check) | 1 | 4,750 ops/s | 0.208 | 0.325 | 0.211 | 0.298 | -| scanAnnouncements | 10 | 4,542 ops/s | 0.220 | 0.336 | 0.223 | 0.310 | -| scanAnnouncements | 100 | 452 ops/s | 2.20 | 3.36 | 2.23 | 3.10 | -| scanAnnouncements | 1,000 | 45.2 ops/s | 22.0 | 33.6 | 22.3 | 31.0 | -| scanAnnouncements | 10,000 | 4.52 ops/s | 220 | 336 | 223 | 310 | -| scanAnnouncements | 100,000 | 0.452 ops/s | 2,200 | 3,360 | 2,230 | 3,100 | +| Benchmark | Count | hz | min | max | p50 | p99 | +| ---------------------------------------- | ------- | ----------- | ----- | ----- | ----- | ----- | +| checkStealthAddress (single match check) | 1 | 4,750 ops/s | 0.208 | 0.325 | 0.211 | 0.298 | +| scanAnnouncements | 10 | 4,542 ops/s | 0.220 | 0.336 | 0.223 | 0.310 | +| scanAnnouncements | 100 | 452 ops/s | 2.20 | 3.36 | 2.23 | 3.10 | +| scanAnnouncements | 1,000 | 45.2 ops/s | 22.0 | 33.6 | 22.3 | 31.0 | +| scanAnnouncements | 10,000 | 4.52 ops/s | 220 | 336 | 223 | 310 | +| scanAnnouncements | 100,000 | 0.452 ops/s | 2,200 | 3,360 | 2,230 | 3,100 | + +**Notes**: -**Notes**: - Scales linearly with N (as expected for a view-only filter) - 100k announcements scan in ~2.2 seconds (p50) - View-tag filtering reduces per-announcement cost by ~255x before full ECDH @@ -76,8 +77,8 @@ All times in milliseconds (ms). Lower is better. ### HTTP / Network -| Benchmark | hz | min | max | p50 | p99 | -|-----------|----|----|-----|-----|-----| +| Benchmark | hz | min | max | p50 | p99 | +| ---------------------------------------- | ---------- | ---- | ---- | ---- | ---- | | fetchAnnouncements (mocked RPC response) | 58.3 ops/s | 16.8 | 28.5 | 17.2 | 26.1 | **Notes**: Mocked to avoid network variance; real calls will depend on RPC latency. @@ -87,17 +88,20 @@ All times in milliseconds (ms). Lower is better. ### šŸ”“ Priority: `scanAnnouncements` Outer Loop **Current cost**: O(N) iterations through announcements, each involving: + 1. View-tag quick filter (99.6% reject rate) — cheap āœ“ 2. ECDH (Curve25519 scalar mult) — **expensive** āœ— 3. SHA-256 hash — moderate cost **Bottleneck**: ECDH is the primary cost driver. For 100k announcements, we compute 100k ECDH operations. -**Impact**: +**Impact**: + - Background sync (Spectre) becomes slow at 10k+ announcements - Receive page blocks UI when scanning > 1k recent announcements **Optimization opportunities**: + 1. **Batch ECDH** (3–5x speedup expected) - Use batched scalar multiplication if underlying crypto library supports it - Group operations to improve cache locality @@ -111,6 +115,7 @@ All times in milliseconds (ms). Lower is better. - Trade-off: less privacy but faster scanning **Recommended next steps**: + - Profile the exact time spent in ECDH vs. SHA-256 - Benchmark @noble/curves with vs. without SIMD (if available) - File a follow-up: `perf: Optimize scanAnnouncements ECDH loop via batching` @@ -120,6 +125,7 @@ All times in milliseconds (ms). Lower is better. **Current cost**: SHA-256 called multiple times per scan loop **Potential gain**: 10–20% via: + - Pre-computed context to reduce allocations - Streaming hash API if available diff --git a/docs/fee-estimation.md b/docs/fee-estimation.md index 85028f3..308914e 100644 --- a/docs/fee-estimation.md +++ b/docs/fee-estimation.md @@ -8,10 +8,10 @@ Stellar has two fee components: -| Component | Applies to | Source | -|---|---|---| -| **Inclusion fee** (base fee Ɨ op count) | All transactions | Horizon `/fee_stats` → recent ledger p50/p99 | -| **Resource fee** | Soroban (smart contract) only | `simulateTransaction` RPC call | +| Component | Applies to | Source | +| --------------------------------------- | ----------------------------- | -------------------------------------------- | +| **Inclusion fee** (base fee Ɨ op count) | All transactions | Horizon `/fee_stats` → recent ledger p50/p99 | +| **Resource fee** | Soroban (smart contract) only | `simulateTransaction` RPC call | The inclusion fee is set by network congestion — it can spike between the time you estimate and submit. The resource fee reflects ledger state at simulation time and may drift before submission. @@ -38,25 +38,25 @@ All fee values are in **stroops** (1 XLM = 10,000,000 stroops). ### Parameters -| Field | Type | Required | Description | -|---|---|---|---| -| `operationCount` | `number` | āœ… | Number of operations in the transaction | -| `sorobanResources` | `SorobanResources` | āŒ | Provide for Soroban invocations | -| `sorobanResources.transactionXdr` | `string` | āœ… (Soroban) | Serialised transaction XDR to simulate | -| `sorobanResources.simulationResult` | `SimulateTransactionResponse` | āŒ | Pre-fetched simulation (skips RPC call) | -| `network` | `'mainnet' \| 'testnet'` | āŒ | Defaults to `'testnet'` | -| `feeStats` | `Horizon.FeeStatsResponse` | āŒ | Pre-fetched fee stats (skips Horizon call) | -| `feeBump` | `boolean` | āŒ | Set `true` when wrapping in a fee-bump envelope | -| `rpcUrl` | `string` | āŒ | Custom Soroban RPC URL | -| `horizonUrl` | `string` | āŒ | Custom Horizon URL | +| Field | Type | Required | Description | +| ----------------------------------- | ----------------------------- | ------------ | ----------------------------------------------- | +| `operationCount` | `number` | āœ… | Number of operations in the transaction | +| `sorobanResources` | `SorobanResources` | āŒ | Provide for Soroban invocations | +| `sorobanResources.transactionXdr` | `string` | āœ… (Soroban) | Serialised transaction XDR to simulate | +| `sorobanResources.simulationResult` | `SimulateTransactionResponse` | āŒ | Pre-fetched simulation (skips RPC call) | +| `network` | `'mainnet' \| 'testnet'` | āŒ | Defaults to `'testnet'` | +| `feeStats` | `Horizon.FeeStatsResponse` | āŒ | Pre-fetched fee stats (skips Horizon call) | +| `feeBump` | `boolean` | āŒ | Set `true` when wrapping in a fee-bump envelope | +| `rpcUrl` | `string` | āŒ | Custom Soroban RPC URL | +| `horizonUrl` | `string` | āŒ | Custom Horizon URL | ### Return value ```ts interface FeeEstimate { - low: number; // Protocol minimum — likely rejected under congestion + low: number; // Protocol minimum — likely rejected under congestion expected: number; // p50 fee rate — good for non-urgent transactions - high: number; // p99 Ɨ 2Ɨ surge buffer — maximises inclusion probability + high: number; // p99 Ɨ 2Ɨ surge buffer — maximises inclusion probability breakdown: { networkBaseFee: number; operationCount: number; @@ -64,7 +64,7 @@ interface FeeEstimate { p50Fee?: number; p99Fee?: number; sorobanResourceFee?: number; // Soroban only - sorobanPadding?: number; // 25% padding on resource fee for "high" tier + sorobanPadding?: number; // 25% padding on resource fee for "high" tier uncertainty: string; }; } @@ -122,7 +122,7 @@ const tx = new TransactionBuilder(sourceAccount, { fee: '100', networkPassphrase: Networks.TESTNET, }) - .addOperation(contract.call('send_stealth', /* args */)) + .addOperation(contract.call('send_stealth' /* args */)) .setTimeout(30) .build(); @@ -158,9 +158,9 @@ const estimate = await estimateStellarFee({ ```ts const estimate = await estimateStellarFee({ - operationCount: 2, // ops in the INNER transaction + operationCount: 2, // ops in the INNER transaction network: 'mainnet', - feeBump: true, // adds 1 for the outer envelope + feeBump: true, // adds 1 for the outer envelope }); // breakdown.operationCount === 3 @@ -187,9 +187,9 @@ async function getFeeOptions(txXdr: string) { const toXlm = (stroops: number) => (stroops / 10_000_000).toFixed(7); return [ - { label: 'Slow', fee: estimate.low, display: toXlm(estimate.low) }, - { label: 'Normal', fee: estimate.expected, display: toXlm(estimate.expected) }, - { label: 'Priority', fee: estimate.high, display: toXlm(estimate.high) }, + { label: 'Slow', fee: estimate.low, display: toXlm(estimate.low) }, + { label: 'Normal', fee: estimate.expected, display: toXlm(estimate.expected) }, + { label: 'Priority', fee: estimate.high, display: toXlm(estimate.high) }, ]; } ``` @@ -218,4 +218,4 @@ pnpm test # Integration tests against testnet $env:INTEGRATION="1"; pnpm exec vitest run test/chains/stellar/fee-estimation.integration.test.ts -``` \ No newline at end of file +``` diff --git a/examples/react-native-stellar/App.tsx b/examples/react-native-stellar/App.tsx index 51fa323..b79bbea 100644 --- a/examples/react-native-stellar/App.tsx +++ b/examples/react-native-stellar/App.tsx @@ -15,7 +15,11 @@ function bytesToHex(bytes: Uint8Array): string { const sampleSignature = new Uint8Array(Array.from({ length: 64 }, (_, i) => i + 1)); -const announcementsFixture = (stealthAddress: string, ephemeralPubKey: Uint8Array, viewTag: number) => [ +const announcementsFixture = ( + stealthAddress: string, + ephemeralPubKey: Uint8Array, + viewTag: number, +) => [ { schemeId: 1, stealthAddress, @@ -28,9 +32,22 @@ const announcementsFixture = (stealthAddress: string, ephemeralPubKey: Uint8Arra export default function App() { const { keys, stealth, matches } = useMemo(() => { const keys = deriveStealthKeys(sampleSignature); - const stealth = generateStealthAddress(keys.spendingPubKey, keys.viewingPubKey, new Uint8Array(32).fill(0x42)); - const announcements = announcementsFixture(stealth.stealthAddress, stealth.ephemeralPubKey, stealth.viewTag); - const matches = scanAnnouncements(announcements, keys.viewingKey, keys.spendingPubKey, keys.spendingScalar); + const stealth = generateStealthAddress( + keys.spendingPubKey, + keys.viewingPubKey, + new Uint8Array(32).fill(0x42), + ); + const announcements = announcementsFixture( + stealth.stealthAddress, + stealth.ephemeralPubKey, + stealth.viewTag, + ); + const matches = scanAnnouncements( + announcements, + keys.viewingKey, + keys.spendingPubKey, + keys.spendingScalar, + ); return { keys, stealth, matches }; }, []); @@ -51,7 +68,9 @@ export default function App() { Scan Result Matches found: {matches.length} - {matches.length > 0 ? matches[0].stealthAddress : 'none'} + + {matches.length > 0 ? matches[0].stealthAddress : 'none'} + diff --git a/examples/react-native-stellar/polyfills.ts b/examples/react-native-stellar/polyfills.ts index 8a72d92..b64410c 100644 --- a/examples/react-native-stellar/polyfills.ts +++ b/examples/react-native-stellar/polyfills.ts @@ -35,12 +35,17 @@ if (typeof globalThis.btoa === 'undefined') { let str = input; let output = ''; - for (let block = 0, charCode, idx = 0, map = chars; str.charAt(idx | 0) || ((map = '='), idx % 1); ) { + for ( + let block = 0, charCode, idx = 0, map = chars; + str.charAt(idx | 0) || ((map = '='), idx % 1); + ) { charCode = str.charCodeAt((idx += 3 / 4)); if (charCode > 0xff) { - throw new Error('Failed to execute btoa: The string to be encoded contains characters outside of the Latin1 range.'); + throw new Error( + 'Failed to execute btoa: The string to be encoded contains characters outside of the Latin1 range.', + ); } - output += map.charAt((block = (block << 8) | charCode) >> ((3.5 - (idx % 1)) * 8) & 0x3f); + output += map.charAt(((block = (block << 8) | charCode) >> ((3.5 - (idx % 1)) * 8)) & 0x3f); } return output; diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 036518b..a77fb84 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -5,7 +5,6 @@ settings: excludeLinksFromLockfile: false importers: - .: dependencies: '@noble/curves': @@ -56,499 +55,810 @@ importers: version: 3.2.6(@types/node@26.0.0)(jiti@2.6.1) packages: - '@adraffy/ens-normalize@1.11.1': - resolution: {integrity: sha512-nhCBV3quEgesuf7c7KYfperqSS14T8bYuvJ8PcLJp6znkZpFc0AuW4qBtr8eKVyPPe/8RSr7sglCWPU5eaxwKQ==} + resolution: + { + integrity: sha512-nhCBV3quEgesuf7c7KYfperqSS14T8bYuvJ8PcLJp6znkZpFc0AuW4qBtr8eKVyPPe/8RSr7sglCWPU5eaxwKQ==, + } '@babel/code-frame@7.29.7': - resolution: {integrity: sha512-Aup7aUOfpbAUg2ROOJN6Iw5f9DMBlzu0mIkm/malLQFN/YQgO48wCj0Kxa3sEHJvPVFg7siR+qRInwXd2qhQKw==} - engines: {node: '>=6.9.0'} + resolution: + { + integrity: sha512-Aup7aUOfpbAUg2ROOJN6Iw5f9DMBlzu0mIkm/malLQFN/YQgO48wCj0Kxa3sEHJvPVFg7siR+qRInwXd2qhQKw==, + } + engines: { node: '>=6.9.0' } '@babel/helper-validator-identifier@7.29.7': - resolution: {integrity: sha512-qehxGkRj55h/ff8EMaJ+cYhyaKlHIxqYDn682wQD7RNp9UujOQsHog2uS0r2vzr4pW+sXf90NeeayjcNaX3fFg==} - engines: {node: '>=6.9.0'} + resolution: + { + integrity: sha512-qehxGkRj55h/ff8EMaJ+cYhyaKlHIxqYDn682wQD7RNp9UujOQsHog2uS0r2vzr4pW+sXf90NeeayjcNaX3fFg==, + } + engines: { node: '>=6.9.0' } '@babel/runtime@7.29.7': - resolution: {integrity: sha512-Nq8OhGWiZIZGV6hLHoyAKLLcJihP/xFeBMGJoUrxTX2psI8dCifzLhZISFb+VWS3wFMRDmCGw5R+dOySCqPLhw==} - engines: {node: '>=6.9.0'} + resolution: + { + integrity: sha512-Nq8OhGWiZIZGV6hLHoyAKLLcJihP/xFeBMGJoUrxTX2psI8dCifzLhZISFb+VWS3wFMRDmCGw5R+dOySCqPLhw==, + } + engines: { node: '>=6.9.0' } '@commitlint/cli@19.8.1': - resolution: {integrity: sha512-LXUdNIkspyxrlV6VDHWBmCZRtkEVRpBKxi2Gtw3J54cGWhLCTouVD/Q6ZSaSvd2YaDObWK8mDjrz3TIKtaQMAA==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-LXUdNIkspyxrlV6VDHWBmCZRtkEVRpBKxi2Gtw3J54cGWhLCTouVD/Q6ZSaSvd2YaDObWK8mDjrz3TIKtaQMAA==, + } + engines: { node: '>=v18' } hasBin: true '@commitlint/config-conventional@19.8.1': - resolution: {integrity: sha512-/AZHJL6F6B/G959CsMAzrPKKZjeEiAVifRyEwXxcT6qtqbPwGw+iQxmNS+Bu+i09OCtdNRW6pNpBvgPrtMr9EQ==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-/AZHJL6F6B/G959CsMAzrPKKZjeEiAVifRyEwXxcT6qtqbPwGw+iQxmNS+Bu+i09OCtdNRW6pNpBvgPrtMr9EQ==, + } + engines: { node: '>=v18' } '@commitlint/config-validator@19.8.1': - resolution: {integrity: sha512-0jvJ4u+eqGPBIzzSdqKNX1rvdbSU1lPNYlfQQRIFnBgLy26BtC0cFnr7c/AyuzExMxWsMOte6MkTi9I3SQ3iGQ==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-0jvJ4u+eqGPBIzzSdqKNX1rvdbSU1lPNYlfQQRIFnBgLy26BtC0cFnr7c/AyuzExMxWsMOte6MkTi9I3SQ3iGQ==, + } + engines: { node: '>=v18' } '@commitlint/ensure@19.8.1': - resolution: {integrity: sha512-mXDnlJdvDzSObafjYrOSvZBwkD01cqB4gbnnFuVyNpGUM5ijwU/r/6uqUmBXAAOKRfyEjpkGVZxaDsCVnHAgyw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-mXDnlJdvDzSObafjYrOSvZBwkD01cqB4gbnnFuVyNpGUM5ijwU/r/6uqUmBXAAOKRfyEjpkGVZxaDsCVnHAgyw==, + } + engines: { node: '>=v18' } '@commitlint/execute-rule@19.8.1': - resolution: {integrity: sha512-YfJyIqIKWI64Mgvn/sE7FXvVMQER/Cd+s3hZke6cI1xgNT/f6ZAz5heND0QtffH+KbcqAwXDEE1/5niYayYaQA==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-YfJyIqIKWI64Mgvn/sE7FXvVMQER/Cd+s3hZke6cI1xgNT/f6ZAz5heND0QtffH+KbcqAwXDEE1/5niYayYaQA==, + } + engines: { node: '>=v18' } '@commitlint/format@19.8.1': - resolution: {integrity: sha512-kSJj34Rp10ItP+Eh9oCItiuN/HwGQMXBnIRk69jdOwEW9llW9FlyqcWYbHPSGofmjsqeoxa38UaEA5tsbm2JWw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-kSJj34Rp10ItP+Eh9oCItiuN/HwGQMXBnIRk69jdOwEW9llW9FlyqcWYbHPSGofmjsqeoxa38UaEA5tsbm2JWw==, + } + engines: { node: '>=v18' } '@commitlint/is-ignored@19.8.1': - resolution: {integrity: sha512-AceOhEhekBUQ5dzrVhDDsbMaY5LqtN8s1mqSnT2Kz1ERvVZkNihrs3Sfk1Je/rxRNbXYFzKZSHaPsEJJDJV8dg==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-AceOhEhekBUQ5dzrVhDDsbMaY5LqtN8s1mqSnT2Kz1ERvVZkNihrs3Sfk1Je/rxRNbXYFzKZSHaPsEJJDJV8dg==, + } + engines: { node: '>=v18' } '@commitlint/lint@19.8.1': - resolution: {integrity: sha512-52PFbsl+1EvMuokZXLRlOsdcLHf10isTPlWwoY1FQIidTsTvjKXVXYb7AvtpWkDzRO2ZsqIgPK7bI98x8LRUEw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-52PFbsl+1EvMuokZXLRlOsdcLHf10isTPlWwoY1FQIidTsTvjKXVXYb7AvtpWkDzRO2ZsqIgPK7bI98x8LRUEw==, + } + engines: { node: '>=v18' } '@commitlint/load@19.8.1': - resolution: {integrity: sha512-9V99EKG3u7z+FEoe4ikgq7YGRCSukAcvmKQuTtUyiYPnOd9a2/H9Ak1J9nJA1HChRQp9OA/sIKPugGS+FK/k1A==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-9V99EKG3u7z+FEoe4ikgq7YGRCSukAcvmKQuTtUyiYPnOd9a2/H9Ak1J9nJA1HChRQp9OA/sIKPugGS+FK/k1A==, + } + engines: { node: '>=v18' } '@commitlint/message@19.8.1': - resolution: {integrity: sha512-+PMLQvjRXiU+Ae0Wc+p99EoGEutzSXFVwQfa3jRNUZLNW5odZAyseb92OSBTKCu+9gGZiJASt76Cj3dLTtcTdg==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-+PMLQvjRXiU+Ae0Wc+p99EoGEutzSXFVwQfa3jRNUZLNW5odZAyseb92OSBTKCu+9gGZiJASt76Cj3dLTtcTdg==, + } + engines: { node: '>=v18' } '@commitlint/parse@19.8.1': - resolution: {integrity: sha512-mmAHYcMBmAgJDKWdkjIGq50X4yB0pSGpxyOODwYmoexxxiUCy5JJT99t1+PEMK7KtsCtzuWYIAXYAiKR+k+/Jw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-mmAHYcMBmAgJDKWdkjIGq50X4yB0pSGpxyOODwYmoexxxiUCy5JJT99t1+PEMK7KtsCtzuWYIAXYAiKR+k+/Jw==, + } + engines: { node: '>=v18' } '@commitlint/read@19.8.1': - resolution: {integrity: sha512-03Jbjb1MqluaVXKHKRuGhcKWtSgh3Jizqy2lJCRbRrnWpcM06MYm8th59Xcns8EqBYvo0Xqb+2DoZFlga97uXQ==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-03Jbjb1MqluaVXKHKRuGhcKWtSgh3Jizqy2lJCRbRrnWpcM06MYm8th59Xcns8EqBYvo0Xqb+2DoZFlga97uXQ==, + } + engines: { node: '>=v18' } '@commitlint/resolve-extends@19.8.1': - resolution: {integrity: sha512-GM0mAhFk49I+T/5UCYns5ayGStkTt4XFFrjjf0L4S26xoMTSkdCf9ZRO8en1kuopC4isDFuEm7ZOm/WRVeElVg==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-GM0mAhFk49I+T/5UCYns5ayGStkTt4XFFrjjf0L4S26xoMTSkdCf9ZRO8en1kuopC4isDFuEm7ZOm/WRVeElVg==, + } + engines: { node: '>=v18' } '@commitlint/rules@19.8.1': - resolution: {integrity: sha512-Hnlhd9DyvGiGwjfjfToMi1dsnw1EXKGJNLTcsuGORHz6SS9swRgkBsou33MQ2n51/boIDrbsg4tIBbRpEWK2kw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-Hnlhd9DyvGiGwjfjfToMi1dsnw1EXKGJNLTcsuGORHz6SS9swRgkBsou33MQ2n51/boIDrbsg4tIBbRpEWK2kw==, + } + engines: { node: '>=v18' } '@commitlint/to-lines@19.8.1': - resolution: {integrity: sha512-98Mm5inzbWTKuZQr2aW4SReY6WUukdWXuZhrqf1QdKPZBCCsXuG87c+iP0bwtD6DBnmVVQjgp4whoHRVixyPBg==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-98Mm5inzbWTKuZQr2aW4SReY6WUukdWXuZhrqf1QdKPZBCCsXuG87c+iP0bwtD6DBnmVVQjgp4whoHRVixyPBg==, + } + engines: { node: '>=v18' } '@commitlint/top-level@19.8.1': - resolution: {integrity: sha512-Ph8IN1IOHPSDhURCSXBz44+CIu+60duFwRsg6HqaISFHQHbmBtxVw4ZrFNIYUzEP7WwrNPxa2/5qJ//NK1FGcw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-Ph8IN1IOHPSDhURCSXBz44+CIu+60duFwRsg6HqaISFHQHbmBtxVw4ZrFNIYUzEP7WwrNPxa2/5qJ//NK1FGcw==, + } + engines: { node: '>=v18' } '@commitlint/types@19.8.1': - resolution: {integrity: sha512-/yCrWGCoA1SVKOks25EGadP9Pnj0oAIHGpl2wH2M2Y46dPM2ueb8wyCVOD7O3WCTkaJ0IkKvzhl1JY7+uCT2Dw==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-/yCrWGCoA1SVKOks25EGadP9Pnj0oAIHGpl2wH2M2Y46dPM2ueb8wyCVOD7O3WCTkaJ0IkKvzhl1JY7+uCT2Dw==, + } + engines: { node: '>=v18' } '@esbuild/aix-ppc64@0.27.7': - resolution: {integrity: sha512-EKX3Qwmhz1eMdEJokhALr0YiD0lhQNwDqkPYyPhiSwKrh7/4KRjQc04sZ8db+5DVVnZ1LmbNDI1uAMPEUBnQPg==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-EKX3Qwmhz1eMdEJokhALr0YiD0lhQNwDqkPYyPhiSwKrh7/4KRjQc04sZ8db+5DVVnZ1LmbNDI1uAMPEUBnQPg==, + } + engines: { node: '>=18' } cpu: [ppc64] os: [aix] '@esbuild/android-arm64@0.27.7': - resolution: {integrity: sha512-62dPZHpIXzvChfvfLJow3q5dDtiNMkwiRzPylSCfriLvZeq0a1bWChrGx/BbUbPwOrsWKMn8idSllklzBy+dgQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-62dPZHpIXzvChfvfLJow3q5dDtiNMkwiRzPylSCfriLvZeq0a1bWChrGx/BbUbPwOrsWKMn8idSllklzBy+dgQ==, + } + engines: { node: '>=18' } cpu: [arm64] os: [android] '@esbuild/android-arm@0.27.7': - resolution: {integrity: sha512-jbPXvB4Yj2yBV7HUfE2KHe4GJX51QplCN1pGbYjvsyCZbQmies29EoJbkEc+vYuU5o45AfQn37vZlyXy4YJ8RQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-jbPXvB4Yj2yBV7HUfE2KHe4GJX51QplCN1pGbYjvsyCZbQmies29EoJbkEc+vYuU5o45AfQn37vZlyXy4YJ8RQ==, + } + engines: { node: '>=18' } cpu: [arm] os: [android] '@esbuild/android-x64@0.27.7': - resolution: {integrity: sha512-x5VpMODneVDb70PYV2VQOmIUUiBtY3D3mPBG8NxVk5CogneYhkR7MmM3yR/uMdITLrC1ml/NV1rj4bMJuy9MCg==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-x5VpMODneVDb70PYV2VQOmIUUiBtY3D3mPBG8NxVk5CogneYhkR7MmM3yR/uMdITLrC1ml/NV1rj4bMJuy9MCg==, + } + engines: { node: '>=18' } cpu: [x64] os: [android] '@esbuild/darwin-arm64@0.27.7': - resolution: {integrity: sha512-5lckdqeuBPlKUwvoCXIgI2D9/ABmPq3Rdp7IfL70393YgaASt7tbju3Ac+ePVi3KDH6N2RqePfHnXkaDtY9fkw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-5lckdqeuBPlKUwvoCXIgI2D9/ABmPq3Rdp7IfL70393YgaASt7tbju3Ac+ePVi3KDH6N2RqePfHnXkaDtY9fkw==, + } + engines: { node: '>=18' } cpu: [arm64] os: [darwin] '@esbuild/darwin-x64@0.27.7': - resolution: {integrity: sha512-rYnXrKcXuT7Z+WL5K980jVFdvVKhCHhUwid+dDYQpH+qu+TefcomiMAJpIiC2EM3Rjtq0sO3StMV/+3w3MyyqQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-rYnXrKcXuT7Z+WL5K980jVFdvVKhCHhUwid+dDYQpH+qu+TefcomiMAJpIiC2EM3Rjtq0sO3StMV/+3w3MyyqQ==, + } + engines: { node: '>=18' } cpu: [x64] os: [darwin] '@esbuild/freebsd-arm64@0.27.7': - resolution: {integrity: sha512-B48PqeCsEgOtzME2GbNM2roU29AMTuOIN91dsMO30t+Ydis3z/3Ngoj5hhnsOSSwNzS+6JppqWsuhTp6E82l2w==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-B48PqeCsEgOtzME2GbNM2roU29AMTuOIN91dsMO30t+Ydis3z/3Ngoj5hhnsOSSwNzS+6JppqWsuhTp6E82l2w==, + } + engines: { node: '>=18' } cpu: [arm64] os: [freebsd] '@esbuild/freebsd-x64@0.27.7': - resolution: {integrity: sha512-jOBDK5XEjA4m5IJK3bpAQF9/Lelu/Z9ZcdhTRLf4cajlB+8VEhFFRjWgfy3M1O4rO2GQ/b2dLwCUGpiF/eATNQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-jOBDK5XEjA4m5IJK3bpAQF9/Lelu/Z9ZcdhTRLf4cajlB+8VEhFFRjWgfy3M1O4rO2GQ/b2dLwCUGpiF/eATNQ==, + } + engines: { node: '>=18' } cpu: [x64] os: [freebsd] '@esbuild/linux-arm64@0.27.7': - resolution: {integrity: sha512-RZPHBoxXuNnPQO9rvjh5jdkRmVizktkT7TCDkDmQ0W2SwHInKCAV95GRuvdSvA7w4VMwfCjUiPwDi0ZO6Nfe9A==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-RZPHBoxXuNnPQO9rvjh5jdkRmVizktkT7TCDkDmQ0W2SwHInKCAV95GRuvdSvA7w4VMwfCjUiPwDi0ZO6Nfe9A==, + } + engines: { node: '>=18' } cpu: [arm64] os: [linux] '@esbuild/linux-arm@0.27.7': - resolution: {integrity: sha512-RkT/YXYBTSULo3+af8Ib0ykH8u2MBh57o7q/DAs3lTJlyVQkgQvlrPTnjIzzRPQyavxtPtfg0EopvDyIt0j1rA==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-RkT/YXYBTSULo3+af8Ib0ykH8u2MBh57o7q/DAs3lTJlyVQkgQvlrPTnjIzzRPQyavxtPtfg0EopvDyIt0j1rA==, + } + engines: { node: '>=18' } cpu: [arm] os: [linux] '@esbuild/linux-ia32@0.27.7': - resolution: {integrity: sha512-GA48aKNkyQDbd3KtkplYWT102C5sn/EZTY4XROkxONgruHPU72l+gW+FfF8tf2cFjeHaRbWpOYa/uRBz/Xq1Pg==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-GA48aKNkyQDbd3KtkplYWT102C5sn/EZTY4XROkxONgruHPU72l+gW+FfF8tf2cFjeHaRbWpOYa/uRBz/Xq1Pg==, + } + engines: { node: '>=18' } cpu: [ia32] os: [linux] '@esbuild/linux-loong64@0.27.7': - resolution: {integrity: sha512-a4POruNM2oWsD4WKvBSEKGIiWQF8fZOAsycHOt6JBpZ+JN2n2JH9WAv56SOyu9X5IqAjqSIPTaJkqN8F7XOQ5Q==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-a4POruNM2oWsD4WKvBSEKGIiWQF8fZOAsycHOt6JBpZ+JN2n2JH9WAv56SOyu9X5IqAjqSIPTaJkqN8F7XOQ5Q==, + } + engines: { node: '>=18' } cpu: [loong64] os: [linux] '@esbuild/linux-mips64el@0.27.7': - resolution: {integrity: sha512-KabT5I6StirGfIz0FMgl1I+R1H73Gp0ofL9A3nG3i/cYFJzKHhouBV5VWK1CSgKvVaG4q1RNpCTR2LuTVB3fIw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-KabT5I6StirGfIz0FMgl1I+R1H73Gp0ofL9A3nG3i/cYFJzKHhouBV5VWK1CSgKvVaG4q1RNpCTR2LuTVB3fIw==, + } + engines: { node: '>=18' } cpu: [mips64el] os: [linux] '@esbuild/linux-ppc64@0.27.7': - resolution: {integrity: sha512-gRsL4x6wsGHGRqhtI+ifpN/vpOFTQtnbsupUF5R5YTAg+y/lKelYR1hXbnBdzDjGbMYjVJLJTd2OFmMewAgwlQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-gRsL4x6wsGHGRqhtI+ifpN/vpOFTQtnbsupUF5R5YTAg+y/lKelYR1hXbnBdzDjGbMYjVJLJTd2OFmMewAgwlQ==, + } + engines: { node: '>=18' } cpu: [ppc64] os: [linux] '@esbuild/linux-riscv64@0.27.7': - resolution: {integrity: sha512-hL25LbxO1QOngGzu2U5xeXtxXcW+/GvMN3ejANqXkxZ/opySAZMrc+9LY/WyjAan41unrR3YrmtTsUpwT66InQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-hL25LbxO1QOngGzu2U5xeXtxXcW+/GvMN3ejANqXkxZ/opySAZMrc+9LY/WyjAan41unrR3YrmtTsUpwT66InQ==, + } + engines: { node: '>=18' } cpu: [riscv64] os: [linux] '@esbuild/linux-s390x@0.27.7': - resolution: {integrity: sha512-2k8go8Ycu1Kb46vEelhu1vqEP+UeRVj2zY1pSuPdgvbd5ykAw82Lrro28vXUrRmzEsUV0NzCf54yARIK8r0fdw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-2k8go8Ycu1Kb46vEelhu1vqEP+UeRVj2zY1pSuPdgvbd5ykAw82Lrro28vXUrRmzEsUV0NzCf54yARIK8r0fdw==, + } + engines: { node: '>=18' } cpu: [s390x] os: [linux] '@esbuild/linux-x64@0.27.7': - resolution: {integrity: sha512-hzznmADPt+OmsYzw1EE33ccA+HPdIqiCRq7cQeL1Jlq2gb1+OyWBkMCrYGBJ+sxVzve2ZJEVeePbLM2iEIZSxA==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-hzznmADPt+OmsYzw1EE33ccA+HPdIqiCRq7cQeL1Jlq2gb1+OyWBkMCrYGBJ+sxVzve2ZJEVeePbLM2iEIZSxA==, + } + engines: { node: '>=18' } cpu: [x64] os: [linux] '@esbuild/netbsd-arm64@0.27.7': - resolution: {integrity: sha512-b6pqtrQdigZBwZxAn1UpazEisvwaIDvdbMbmrly7cDTMFnw/+3lVxxCTGOrkPVnsYIosJJXAsILG9XcQS+Yu6w==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-b6pqtrQdigZBwZxAn1UpazEisvwaIDvdbMbmrly7cDTMFnw/+3lVxxCTGOrkPVnsYIosJJXAsILG9XcQS+Yu6w==, + } + engines: { node: '>=18' } cpu: [arm64] os: [netbsd] '@esbuild/netbsd-x64@0.27.7': - resolution: {integrity: sha512-OfatkLojr6U+WN5EDYuoQhtM+1xco+/6FSzJJnuWiUw5eVcicbyK3dq5EeV/QHT1uy6GoDhGbFpprUiHUYggrw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-OfatkLojr6U+WN5EDYuoQhtM+1xco+/6FSzJJnuWiUw5eVcicbyK3dq5EeV/QHT1uy6GoDhGbFpprUiHUYggrw==, + } + engines: { node: '>=18' } cpu: [x64] os: [netbsd] '@esbuild/openbsd-arm64@0.27.7': - resolution: {integrity: sha512-AFuojMQTxAz75Fo8idVcqoQWEHIXFRbOc1TrVcFSgCZtQfSdc1RXgB3tjOn/krRHENUB4j00bfGjyl2mJrU37A==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-AFuojMQTxAz75Fo8idVcqoQWEHIXFRbOc1TrVcFSgCZtQfSdc1RXgB3tjOn/krRHENUB4j00bfGjyl2mJrU37A==, + } + engines: { node: '>=18' } cpu: [arm64] os: [openbsd] '@esbuild/openbsd-x64@0.27.7': - resolution: {integrity: sha512-+A1NJmfM8WNDv5CLVQYJ5PshuRm/4cI6WMZRg1by1GwPIQPCTs1GLEUHwiiQGT5zDdyLiRM/l1G0Pv54gvtKIg==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-+A1NJmfM8WNDv5CLVQYJ5PshuRm/4cI6WMZRg1by1GwPIQPCTs1GLEUHwiiQGT5zDdyLiRM/l1G0Pv54gvtKIg==, + } + engines: { node: '>=18' } cpu: [x64] os: [openbsd] '@esbuild/openharmony-arm64@0.27.7': - resolution: {integrity: sha512-+KrvYb/C8zA9CU/g0sR6w2RBw7IGc5J2BPnc3dYc5VJxHCSF1yNMxTV5LQ7GuKteQXZtspjFbiuW5/dOj7H4Yw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-+KrvYb/C8zA9CU/g0sR6w2RBw7IGc5J2BPnc3dYc5VJxHCSF1yNMxTV5LQ7GuKteQXZtspjFbiuW5/dOj7H4Yw==, + } + engines: { node: '>=18' } cpu: [arm64] os: [openharmony] '@esbuild/sunos-x64@0.27.7': - resolution: {integrity: sha512-ikktIhFBzQNt/QDyOL580ti9+5mL/YZeUPKU2ivGtGjdTYoqz6jObj6nOMfhASpS4GU4Q/Clh1QtxWAvcYKamA==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-ikktIhFBzQNt/QDyOL580ti9+5mL/YZeUPKU2ivGtGjdTYoqz6jObj6nOMfhASpS4GU4Q/Clh1QtxWAvcYKamA==, + } + engines: { node: '>=18' } cpu: [x64] os: [sunos] '@esbuild/win32-arm64@0.27.7': - resolution: {integrity: sha512-7yRhbHvPqSpRUV7Q20VuDwbjW5kIMwTHpptuUzV+AA46kiPze5Z7qgt6CLCK3pWFrHeNfDd1VKgyP4O+ng17CA==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-7yRhbHvPqSpRUV7Q20VuDwbjW5kIMwTHpptuUzV+AA46kiPze5Z7qgt6CLCK3pWFrHeNfDd1VKgyP4O+ng17CA==, + } + engines: { node: '>=18' } cpu: [arm64] os: [win32] '@esbuild/win32-ia32@0.27.7': - resolution: {integrity: sha512-SmwKXe6VHIyZYbBLJrhOoCJRB/Z1tckzmgTLfFYOfpMAx63BJEaL9ExI8x7v0oAO3Zh6D/Oi1gVxEYr5oUCFhw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-SmwKXe6VHIyZYbBLJrhOoCJRB/Z1tckzmgTLfFYOfpMAx63BJEaL9ExI8x7v0oAO3Zh6D/Oi1gVxEYr5oUCFhw==, + } + engines: { node: '>=18' } cpu: [ia32] os: [win32] '@esbuild/win32-x64@0.27.7': - resolution: {integrity: sha512-56hiAJPhwQ1R4i+21FVF7V8kSD5zZTdHcVuRFMW0hn753vVfQN8xlx4uOPT4xoGH0Z/oVATuR82AiqSTDIpaHg==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-56hiAJPhwQ1R4i+21FVF7V8kSD5zZTdHcVuRFMW0hn753vVfQN8xlx4uOPT4xoGH0Z/oVATuR82AiqSTDIpaHg==, + } + engines: { node: '>=18' } cpu: [x64] os: [win32] '@jridgewell/gen-mapping@0.3.13': - resolution: {integrity: sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==} + resolution: + { + integrity: sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==, + } '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} + resolution: + { + integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==, + } + engines: { node: '>=6.0.0' } '@jridgewell/sourcemap-codec@1.5.5': - resolution: {integrity: sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==} + resolution: + { + integrity: sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==, + } '@jridgewell/trace-mapping@0.3.31': - resolution: {integrity: sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==} + resolution: + { + integrity: sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==, + } '@noble/ciphers@1.3.0': - resolution: {integrity: sha512-2I0gnIVPtfnMw9ee9h1dJG7tp81+8Ob3OJb3Mv37rx5L40/b0i7djjCVvGOVqc9AEIQyvyu1i6ypKdFw8R8gQw==} - engines: {node: ^14.21.3 || >=16} + resolution: + { + integrity: sha512-2I0gnIVPtfnMw9ee9h1dJG7tp81+8Ob3OJb3Mv37rx5L40/b0i7djjCVvGOVqc9AEIQyvyu1i6ypKdFw8R8gQw==, + } + engines: { node: ^14.21.3 || >=16 } '@noble/curves@1.9.1': - resolution: {integrity: sha512-k11yZxZg+t+gWvBbIswW0yoJlu8cHOC7dhunwOzoWH/mXGBiYyR4YY6hAEK/3EUs4UpB8la1RfdRpeGsFHkWsA==} - engines: {node: ^14.21.3 || >=16} + resolution: + { + integrity: sha512-k11yZxZg+t+gWvBbIswW0yoJlu8cHOC7dhunwOzoWH/mXGBiYyR4YY6hAEK/3EUs4UpB8la1RfdRpeGsFHkWsA==, + } + engines: { node: ^14.21.3 || >=16 } '@noble/curves@1.9.7': - resolution: {integrity: sha512-gbKGcRUYIjA3/zCCNaWDciTMFI0dCkvou3TL8Zmy5Nc7sJ47a0jtOeZoTaMxkuqRo9cRhjOdZJXegxYE5FN/xw==} - engines: {node: ^14.21.3 || >=16} + resolution: + { + integrity: sha512-gbKGcRUYIjA3/zCCNaWDciTMFI0dCkvou3TL8Zmy5Nc7sJ47a0jtOeZoTaMxkuqRo9cRhjOdZJXegxYE5FN/xw==, + } + engines: { node: ^14.21.3 || >=16 } '@noble/hashes@1.8.0': - resolution: {integrity: sha512-jCs9ldd7NwzpgXDIf6P3+NrHh9/sD6CQdxHyjQI+h/6rDNo88ypBxxz45UDuZHz9r3tNz7N/VInSVoVdtXEI4A==} - engines: {node: ^14.21.3 || >=16} + resolution: + { + integrity: sha512-jCs9ldd7NwzpgXDIf6P3+NrHh9/sD6CQdxHyjQI+h/6rDNo88ypBxxz45UDuZHz9r3tNz7N/VInSVoVdtXEI4A==, + } + engines: { node: ^14.21.3 || >=16 } '@rollup/rollup-android-arm-eabi@4.62.2': - resolution: {integrity: sha512-6o7ZLZK+BeenkZCFNDXqpbjw9bD6nuWonvS/lwQJp7NoVVxm6p3qE7qQ5jGuBjiFsgvqjD8mZAU5oWxTmbOeOg==} + resolution: + { + integrity: sha512-6o7ZLZK+BeenkZCFNDXqpbjw9bD6nuWonvS/lwQJp7NoVVxm6p3qE7qQ5jGuBjiFsgvqjD8mZAU5oWxTmbOeOg==, + } cpu: [arm] os: [android] '@rollup/rollup-android-arm64@4.62.2': - resolution: {integrity: sha512-BaH7BllCACHoH1LguOU56UItGfUWjujlO65kS9LAodViaN4bwIKd7oeW/ZHJ/4ljr/7MIiENnNy3HJ0zXv8Zkw==} + resolution: + { + integrity: sha512-BaH7BllCACHoH1LguOU56UItGfUWjujlO65kS9LAodViaN4bwIKd7oeW/ZHJ/4ljr/7MIiENnNy3HJ0zXv8Zkw==, + } cpu: [arm64] os: [android] '@rollup/rollup-darwin-arm64@4.62.2': - resolution: {integrity: sha512-v39RCCvj4He82I9sFmk+M1VZ0PLM9sfsLVikjfx2hYBNALhrrOR2D3JjQA6AhlaSOgcR+RzrKY7e1+bT6SUO/A==} + resolution: + { + integrity: sha512-v39RCCvj4He82I9sFmk+M1VZ0PLM9sfsLVikjfx2hYBNALhrrOR2D3JjQA6AhlaSOgcR+RzrKY7e1+bT6SUO/A==, + } cpu: [arm64] os: [darwin] '@rollup/rollup-darwin-x64@4.62.2': - resolution: {integrity: sha512-yl0y2vq3S3lHeuXhEdss6TWfKW8vkujImO12tn4ZkG/4oghr09LvdYm2RElVjokTQiUvDUGXLGsYeLqUMCKpGA==} + resolution: + { + integrity: sha512-yl0y2vq3S3lHeuXhEdss6TWfKW8vkujImO12tn4ZkG/4oghr09LvdYm2RElVjokTQiUvDUGXLGsYeLqUMCKpGA==, + } cpu: [x64] os: [darwin] '@rollup/rollup-freebsd-arm64@4.62.2': - resolution: {integrity: sha512-tT4pvt4qXD+vEoezupCWi+a1F0vvDiksiHc+PxRlYTOH1I6/X4id9jPxTP+Fg+545euaFT1jJVs4CEdHZAU1vw==} + resolution: + { + integrity: sha512-tT4pvt4qXD+vEoezupCWi+a1F0vvDiksiHc+PxRlYTOH1I6/X4id9jPxTP+Fg+545euaFT1jJVs4CEdHZAU1vw==, + } cpu: [arm64] os: [freebsd] '@rollup/rollup-freebsd-x64@4.62.2': - resolution: {integrity: sha512-6nU5F2wCW+qvCBhTn1pdIU3bzsIoF7EUwsCDRxilWGprQR6yd508YnH9+OKFCwpfS8pjZqDUmnCAr7exax0XCg==} + resolution: + { + integrity: sha512-6nU5F2wCW+qvCBhTn1pdIU3bzsIoF7EUwsCDRxilWGprQR6yd508YnH9+OKFCwpfS8pjZqDUmnCAr7exax0XCg==, + } cpu: [x64] os: [freebsd] '@rollup/rollup-linux-arm-gnueabihf@4.62.2': - resolution: {integrity: sha512-n1GJHPOvpIfhi3TmrCeh6S6URt9BFCt0KQE3qvexyGCTAKpR4Lg+eWvNZEqu7epxwus/8ElT3hacYEucm49SZg==} + resolution: + { + integrity: sha512-n1GJHPOvpIfhi3TmrCeh6S6URt9BFCt0KQE3qvexyGCTAKpR4Lg+eWvNZEqu7epxwus/8ElT3hacYEucm49SZg==, + } cpu: [arm] os: [linux] libc: [glibc] '@rollup/rollup-linux-arm-musleabihf@4.62.2': - resolution: {integrity: sha512-JqgflS8wEB+UXV/vS1RpRbifGBeN4D5lz8D8oOFbFZw4vedvdOgCFAjfBmIMdW3yL10XpQQ0Ambepw6MXrhOnA==} + resolution: + { + integrity: sha512-JqgflS8wEB+UXV/vS1RpRbifGBeN4D5lz8D8oOFbFZw4vedvdOgCFAjfBmIMdW3yL10XpQQ0Ambepw6MXrhOnA==, + } cpu: [arm] os: [linux] libc: [musl] '@rollup/rollup-linux-arm64-gnu@4.62.2': - resolution: {integrity: sha512-wnFJkogWvN4jm/hQRF2UBaeUmk20j5+DmHvoyWii2b8HJDyvz1MF2OU/6ynXt2KR63rbZLWkFpoytpdc/yBuSA==} + resolution: + { + integrity: sha512-wnFJkogWvN4jm/hQRF2UBaeUmk20j5+DmHvoyWii2b8HJDyvz1MF2OU/6ynXt2KR63rbZLWkFpoytpdc/yBuSA==, + } cpu: [arm64] os: [linux] libc: [glibc] '@rollup/rollup-linux-arm64-musl@4.62.2': - resolution: {integrity: sha512-HVu2bp0zhvJ8xHEV9+UUs7S90VadmBSY3LcIMvozbPo4AuMGDWlz3ymHLHZPX4hR67TKTt8Qp5PJ5RBg/i+RMQ==} + resolution: + { + integrity: sha512-HVu2bp0zhvJ8xHEV9+UUs7S90VadmBSY3LcIMvozbPo4AuMGDWlz3ymHLHZPX4hR67TKTt8Qp5PJ5RBg/i+RMQ==, + } cpu: [arm64] os: [linux] libc: [musl] '@rollup/rollup-linux-loong64-gnu@4.62.2': - resolution: {integrity: sha512-mQqqAV8QaoSgr9I2fKDLY2BAVvmKjWoGiu/cSYQonsLvtqwEn1E4QYfnCOcp5zoEqNhsDYin1s6jx/VJmrxlZg==} + resolution: + { + integrity: sha512-mQqqAV8QaoSgr9I2fKDLY2BAVvmKjWoGiu/cSYQonsLvtqwEn1E4QYfnCOcp5zoEqNhsDYin1s6jx/VJmrxlZg==, + } cpu: [loong64] os: [linux] libc: [glibc] '@rollup/rollup-linux-loong64-musl@4.62.2': - resolution: {integrity: sha512-IxKLoxCQ2IWi6bT2akyDUBGsOImDKB+sPp4EsTmwFQ/fMwpCKm8uLSSgP/Kx/QYUgKis6SEZ5/Nlhup0DIA0PQ==} + resolution: + { + integrity: sha512-IxKLoxCQ2IWi6bT2akyDUBGsOImDKB+sPp4EsTmwFQ/fMwpCKm8uLSSgP/Kx/QYUgKis6SEZ5/Nlhup0DIA0PQ==, + } cpu: [loong64] os: [linux] libc: [musl] '@rollup/rollup-linux-ppc64-gnu@4.62.2': - resolution: {integrity: sha512-Mk5ha2RQSgyFfmYYLkBpPnUk8D8FriBxesO1u9O75X0mHgXL1UQcH5Itl2lurWL2tj0RxV9b9tJgipac0hRY9A==} + resolution: + { + integrity: sha512-Mk5ha2RQSgyFfmYYLkBpPnUk8D8FriBxesO1u9O75X0mHgXL1UQcH5Itl2lurWL2tj0RxV9b9tJgipac0hRY9A==, + } cpu: [ppc64] os: [linux] libc: [glibc] '@rollup/rollup-linux-ppc64-musl@4.62.2': - resolution: {integrity: sha512-CjvEnqJL/0/TQ3TXX3OPIJ/kmBellrWd4heXUmHeJlTnmwjKpSJzoehLaL6Xk0ZnMHBu9dZuFADNOrtjF4v+2w==} + resolution: + { + integrity: sha512-CjvEnqJL/0/TQ3TXX3OPIJ/kmBellrWd4heXUmHeJlTnmwjKpSJzoehLaL6Xk0ZnMHBu9dZuFADNOrtjF4v+2w==, + } cpu: [ppc64] os: [linux] libc: [musl] '@rollup/rollup-linux-riscv64-gnu@4.62.2': - resolution: {integrity: sha512-1SiZbzwdkaDURsew/tSOrooKiYy7EQGT6m8ufavAi9NEyQb/6VuIxFXAL1fqa4iZe3g4NbNk4P7J32z2tw5Mgg==} + resolution: + { + integrity: sha512-1SiZbzwdkaDURsew/tSOrooKiYy7EQGT6m8ufavAi9NEyQb/6VuIxFXAL1fqa4iZe3g4NbNk4P7J32z2tw5Mgg==, + } cpu: [riscv64] os: [linux] libc: [glibc] '@rollup/rollup-linux-riscv64-musl@4.62.2': - resolution: {integrity: sha512-nQts12zJ3NQRoE6uYljOH89v7szzLDvG2JD/vsX+vGXU8w/At1GowTZ5/7qeFQ8m7L55rpR8Okugnuo5bgjy2Q==} + resolution: + { + integrity: sha512-nQts12zJ3NQRoE6uYljOH89v7szzLDvG2JD/vsX+vGXU8w/At1GowTZ5/7qeFQ8m7L55rpR8Okugnuo5bgjy2Q==, + } cpu: [riscv64] os: [linux] libc: [musl] '@rollup/rollup-linux-s390x-gnu@4.62.2': - resolution: {integrity: sha512-E9/ll019jhPIJgpzfZoIkBGhcz+kKNgVWYRY0zr9srBdPPFVpvOKW8VaJKUbeK+eZXyQF9ltME+Kk6affeaPgg==} + resolution: + { + integrity: sha512-E9/ll019jhPIJgpzfZoIkBGhcz+kKNgVWYRY0zr9srBdPPFVpvOKW8VaJKUbeK+eZXyQF9ltME+Kk6affeaPgg==, + } cpu: [s390x] os: [linux] libc: [glibc] '@rollup/rollup-linux-x64-gnu@4.62.2': - resolution: {integrity: sha512-5BqxR/pshjey51iliyzTD5Xi3EN0aLmQ2lZ3lvefVV9c82BvrLo2/6OT55iifpWBufs6kdwWbuOKS841DrmK9A==} + resolution: + { + integrity: sha512-5BqxR/pshjey51iliyzTD5Xi3EN0aLmQ2lZ3lvefVV9c82BvrLo2/6OT55iifpWBufs6kdwWbuOKS841DrmK9A==, + } cpu: [x64] os: [linux] libc: [glibc] '@rollup/rollup-linux-x64-musl@4.62.2': - resolution: {integrity: sha512-uNN83XxQrRAh/w0/pmAfibcwyb6YWt4gP+dpnQKPVJshAloQ785ii8CT8ZCIxkGg9opVsvAlGhFitSm6D1Jjpg==} + resolution: + { + integrity: sha512-uNN83XxQrRAh/w0/pmAfibcwyb6YWt4gP+dpnQKPVJshAloQ785ii8CT8ZCIxkGg9opVsvAlGhFitSm6D1Jjpg==, + } cpu: [x64] os: [linux] libc: [musl] '@rollup/rollup-openbsd-x64@4.62.2': - resolution: {integrity: sha512-srjEIxSH3LRnJN6THczDHWQplqEMFiAJrTab0msUryh9kwNpkICf3Ea6q6MN/2cZwRFUNx5w+h6Hpi4QuHS6Zg==} + resolution: + { + integrity: sha512-srjEIxSH3LRnJN6THczDHWQplqEMFiAJrTab0msUryh9kwNpkICf3Ea6q6MN/2cZwRFUNx5w+h6Hpi4QuHS6Zg==, + } cpu: [x64] os: [openbsd] '@rollup/rollup-openharmony-arm64@4.62.2': - resolution: {integrity: sha512-8hOJnxgbyObnCm5AlRA3A931xX19xq80RjVTKgJOvEKWqJruP/Uf12IbAOaDjjEXYRewwHLfmF0YRIdK3OwKWA==} + resolution: + { + integrity: sha512-8hOJnxgbyObnCm5AlRA3A931xX19xq80RjVTKgJOvEKWqJruP/Uf12IbAOaDjjEXYRewwHLfmF0YRIdK3OwKWA==, + } cpu: [arm64] os: [openharmony] '@rollup/rollup-win32-arm64-msvc@4.62.2': - resolution: {integrity: sha512-mmF4AY1i0hG/bLWUctUq59gtmgaSIRa3cu/A3JFRp/sCNEme2bgDEiDS22P9FbnJB8NJNF4jPJiSP5RHQpUTDg==} + resolution: + { + integrity: sha512-mmF4AY1i0hG/bLWUctUq59gtmgaSIRa3cu/A3JFRp/sCNEme2bgDEiDS22P9FbnJB8NJNF4jPJiSP5RHQpUTDg==, + } cpu: [arm64] os: [win32] '@rollup/rollup-win32-ia32-msvc@4.62.2': - resolution: {integrity: sha512-DZgkknc6jhHrk46V25vbAM0zZkyP0nSDkJB8/dRkLTxv470dOmWDqGoEJl/9A0dFfS7yE3REOwNDxpHwSLSt0Q==} + resolution: + { + integrity: sha512-DZgkknc6jhHrk46V25vbAM0zZkyP0nSDkJB8/dRkLTxv470dOmWDqGoEJl/9A0dFfS7yE3REOwNDxpHwSLSt0Q==, + } cpu: [ia32] os: [win32] '@rollup/rollup-win32-x64-gnu@4.62.2': - resolution: {integrity: sha512-T6xr6ucWSFto+VGajA8YH26LdpHRuP4YLHEKAtCWvJDOlnmWcDZVCI2Jmjr+IFHDlt2zRaTAKE4tfjTaWLgJBg==} + resolution: + { + integrity: sha512-T6xr6ucWSFto+VGajA8YH26LdpHRuP4YLHEKAtCWvJDOlnmWcDZVCI2Jmjr+IFHDlt2zRaTAKE4tfjTaWLgJBg==, + } cpu: [x64] os: [win32] '@rollup/rollup-win32-x64-msvc@4.62.2': - resolution: {integrity: sha512-BfzEnDJOt9T8M989/lA37EcJgat01wLRnoi5dQf3QzOH7jzpqTAzdDbVfRljVr5r+jzKqpbHeyOfAaXxAd0PAA==} + resolution: + { + integrity: sha512-BfzEnDJOt9T8M989/lA37EcJgat01wLRnoi5dQf3QzOH7jzpqTAzdDbVfRljVr5r+jzKqpbHeyOfAaXxAd0PAA==, + } cpu: [x64] os: [win32] '@scure/base@1.2.6': - resolution: {integrity: sha512-g/nm5FgUa//MCj1gV09zTJTaM6KBAHqLN907YVQqf7zC49+DcO4B1so4ZX07Ef10Twr6nuqYEH9GEggFXA4Fmg==} + resolution: + { + integrity: sha512-g/nm5FgUa//MCj1gV09zTJTaM6KBAHqLN907YVQqf7zC49+DcO4B1so4ZX07Ef10Twr6nuqYEH9GEggFXA4Fmg==, + } '@scure/bip32@1.7.0': - resolution: {integrity: sha512-E4FFX/N3f4B80AKWp5dP6ow+flD1LQZo/w8UnLGYZO674jS6YnYeepycOOksv+vLPSpgN35wgKgy+ybfTb2SMw==} + resolution: + { + integrity: sha512-E4FFX/N3f4B80AKWp5dP6ow+flD1LQZo/w8UnLGYZO674jS6YnYeepycOOksv+vLPSpgN35wgKgy+ybfTb2SMw==, + } '@scure/bip39@1.6.0': - resolution: {integrity: sha512-+lF0BbLiJNwVlev4eKelw1WWLaiKXw7sSl8T6FvBlWkdX+94aGJ4o8XjUdlyhTCjd8c+B3KT3JfS8P0bLRNU6A==} + resolution: + { + integrity: sha512-+lF0BbLiJNwVlev4eKelw1WWLaiKXw7sSl8T6FvBlWkdX+94aGJ4o8XjUdlyhTCjd8c+B3KT3JfS8P0bLRNU6A==, + } '@solana/buffer-layout@4.0.1': - resolution: {integrity: sha512-E1ImOIAD1tBZFRdjeM4/pzTiTApC0AOBGwyAMS4fwIodCWArzJ3DWdoh8cKxeFM2fElkxBh2Aqts1BPC373rHA==} - engines: {node: '>=5.10'} + resolution: + { + integrity: sha512-E1ImOIAD1tBZFRdjeM4/pzTiTApC0AOBGwyAMS4fwIodCWArzJ3DWdoh8cKxeFM2fElkxBh2Aqts1BPC373rHA==, + } + engines: { node: '>=5.10' } '@solana/codecs-core@2.3.0': - resolution: {integrity: sha512-oG+VZzN6YhBHIoSKgS5ESM9VIGzhWjEHEGNPSibiDTxFhsFWxNaz8LbMDPjBUE69r9wmdGLkrQ+wVPbnJcZPvw==} - engines: {node: '>=20.18.0'} + resolution: + { + integrity: sha512-oG+VZzN6YhBHIoSKgS5ESM9VIGzhWjEHEGNPSibiDTxFhsFWxNaz8LbMDPjBUE69r9wmdGLkrQ+wVPbnJcZPvw==, + } + engines: { node: '>=20.18.0' } peerDependencies: typescript: '>=5.3.3' '@solana/codecs-numbers@2.3.0': - resolution: {integrity: sha512-jFvvwKJKffvG7Iz9dmN51OGB7JBcy2CJ6Xf3NqD/VP90xak66m/Lg48T01u5IQ/hc15mChVHiBm+HHuOFDUrQg==} - engines: {node: '>=20.18.0'} + resolution: + { + integrity: sha512-jFvvwKJKffvG7Iz9dmN51OGB7JBcy2CJ6Xf3NqD/VP90xak66m/Lg48T01u5IQ/hc15mChVHiBm+HHuOFDUrQg==, + } + engines: { node: '>=20.18.0' } peerDependencies: typescript: '>=5.3.3' '@solana/errors@2.3.0': - resolution: {integrity: sha512-66RI9MAbwYV0UtP7kGcTBVLxJgUxoZGm8Fbc0ah+lGiAw17Gugco6+9GrJCV83VyF2mDWyYnYM9qdI3yjgpnaQ==} - engines: {node: '>=20.18.0'} + resolution: + { + integrity: sha512-66RI9MAbwYV0UtP7kGcTBVLxJgUxoZGm8Fbc0ah+lGiAw17Gugco6+9GrJCV83VyF2mDWyYnYM9qdI3yjgpnaQ==, + } + engines: { node: '>=20.18.0' } hasBin: true peerDependencies: typescript: '>=5.3.3' '@solana/web3.js@1.98.4': - resolution: {integrity: sha512-vv9lfnvjUsRiq//+j5pBdXig0IQdtzA0BRZ3bXEP4KaIyF1CcaydWqgyzQgfZMNIsWNWmG+AUHwPy4AHOD6gpw==} + resolution: + { + integrity: sha512-vv9lfnvjUsRiq//+j5pBdXig0IQdtzA0BRZ3bXEP4KaIyF1CcaydWqgyzQgfZMNIsWNWmG+AUHwPy4AHOD6gpw==, + } '@stellar/js-xdr@3.1.2': - resolution: {integrity: sha512-VVolPL5goVEIsvuGqDc5uiKxV03lzfWdvYg1KikvwheDmTBO68CKDji3bAZ/kppZrx5iTA8z3Ld5yuytcvhvOQ==} + resolution: + { + integrity: sha512-VVolPL5goVEIsvuGqDc5uiKxV03lzfWdvYg1KikvwheDmTBO68CKDji3bAZ/kppZrx5iTA8z3Ld5yuytcvhvOQ==, + } '@stellar/stellar-base@13.1.0': - resolution: {integrity: sha512-90EArG+eCCEzDGj3OJNoCtwpWDwxjv+rs/RNPhvg4bulpjN/CSRj+Ys/SalRcfM4/WRC5/qAfjzmJBAuquWhkA==} - engines: {node: '>=18.0.0'} + resolution: + { + integrity: sha512-90EArG+eCCEzDGj3OJNoCtwpWDwxjv+rs/RNPhvg4bulpjN/CSRj+Ys/SalRcfM4/WRC5/qAfjzmJBAuquWhkA==, + } + engines: { node: '>=18.0.0' } deprecated: This package is now rolled into @stellar/stellar-sdk. Please use @stellar/stellar-sdk to continue receiving updates and support. '@stellar/stellar-sdk@13.3.0': - resolution: {integrity: sha512-8+GHcZLp+mdin8gSjcgfb/Lb6sSMYRX6Nf/0LcSJxvjLQR0XHpjGzOiRbYb2jSXo51EnA6kAV5j+4Pzh5OUKUg==} - engines: {node: '>=18.0.0'} + resolution: + { + integrity: sha512-8+GHcZLp+mdin8gSjcgfb/Lb6sSMYRX6Nf/0LcSJxvjLQR0XHpjGzOiRbYb2jSXo51EnA6kAV5j+4Pzh5OUKUg==, + } + engines: { node: '>=18.0.0' } '@swc/helpers@0.5.23': - resolution: {integrity: sha512-5lSsMOTXURePglDfvuAQUqkGek9Hg2kksOYay2m0+XR++b2NWYL/4sWyuvVBIs8oKnJaxkdi9whaL/sqN13afw==} + resolution: + { + integrity: sha512-5lSsMOTXURePglDfvuAQUqkGek9Hg2kksOYay2m0+XR++b2NWYL/4sWyuvVBIs8oKnJaxkdi9whaL/sqN13afw==, + } '@types/chai@5.2.3': - resolution: {integrity: sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==} + resolution: + { + integrity: sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==, + } '@types/connect@3.4.38': - resolution: {integrity: sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==} + resolution: + { + integrity: sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==, + } '@types/conventional-commits-parser@5.0.2': - resolution: {integrity: sha512-BgT2szDXnVypgpNxOK8aL5SGjUdaQbC++WZNjF1Qge3Og2+zhHj+RWhmehLhYyvQwqAmvezruVfOf8+3m74W+g==} + resolution: + { + integrity: sha512-BgT2szDXnVypgpNxOK8aL5SGjUdaQbC++WZNjF1Qge3Og2+zhHj+RWhmehLhYyvQwqAmvezruVfOf8+3m74W+g==, + } '@types/deep-eql@4.0.2': - resolution: {integrity: sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==} + resolution: + { + integrity: sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==, + } '@types/estree@1.0.9': - resolution: {integrity: sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg==} + resolution: + { + integrity: sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg==, + } '@types/node@12.20.55': - resolution: {integrity: sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==} + resolution: + { + integrity: sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==, + } '@types/node@26.0.0': - resolution: {integrity: sha512-vf2YFi1iY9lHGwNJMs01biZFbKJkrZR1T6/MlzjhJLPdntOHLhTrDSnSVcdtvjihi4VQNlrFRIxLsDBlQpAipA==} + resolution: + { + integrity: sha512-vf2YFi1iY9lHGwNJMs01biZFbKJkrZR1T6/MlzjhJLPdntOHLhTrDSnSVcdtvjihi4VQNlrFRIxLsDBlQpAipA==, + } '@types/uuid@10.0.0': - resolution: {integrity: sha512-7gqG38EyHgyP1S+7+xomFtL+ZNHcKv6DwNaCZmJmo1vgMugyF3TCnXVg4t1uk89mLNwnLtnY3TpOpCOyp1/xHQ==} + resolution: + { + integrity: sha512-7gqG38EyHgyP1S+7+xomFtL+ZNHcKv6DwNaCZmJmo1vgMugyF3TCnXVg4t1uk89mLNwnLtnY3TpOpCOyp1/xHQ==, + } '@types/ws@7.4.7': - resolution: {integrity: sha512-JQbbmxZTZehdc2iszGKs5oC3NFnjeay7mtAWrdt7qNtAVK0g19muApzAy4bm9byz79xa2ZnO/BOBC2R8RC5Lww==} + resolution: + { + integrity: sha512-JQbbmxZTZehdc2iszGKs5oC3NFnjeay7mtAWrdt7qNtAVK0g19muApzAy4bm9byz79xa2ZnO/BOBC2R8RC5Lww==, + } '@types/ws@8.18.1': - resolution: {integrity: sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg==} + resolution: + { + integrity: sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg==, + } '@vitest/expect@3.2.6': - resolution: {integrity: sha512-1+7q9BtaKzEmO+fmNT3kYvoNn5Y71XWAx2Q5HRim4tTVRQVRv4uJFAQ5FbK0OPUeNP/WmVCpxYxoJdvuHVjzBQ==} + resolution: + { + integrity: sha512-1+7q9BtaKzEmO+fmNT3kYvoNn5Y71XWAx2Q5HRim4tTVRQVRv4uJFAQ5FbK0OPUeNP/WmVCpxYxoJdvuHVjzBQ==, + } '@vitest/mocker@3.2.6': - resolution: {integrity: sha512-EZOrpDbkKotFAP7wPAQV1UIyoGOk4oX7ynWhBhLB7v+meMHbQhU16oPpIYGTTe4oFlhpryGpgpcZP/sin3hYuw==} + resolution: + { + integrity: sha512-EZOrpDbkKotFAP7wPAQV1UIyoGOk4oX7ynWhBhLB7v+meMHbQhU16oPpIYGTTe4oFlhpryGpgpcZP/sin3hYuw==, + } peerDependencies: msw: ^2.4.9 vite: ^5.0.0 || ^6.0.0 || ^7.0.0-0 @@ -559,26 +869,47 @@ packages: optional: true '@vitest/pretty-format@3.2.6': - resolution: {integrity: sha512-lb7XXXzmm2h2ASzFnRvQpDo6onT1NmMJA3tkGTWiBFtRJ9lxGY3d3mm/Apt36gej2bkkOVLL/yTOtufDaFa/jA==} + resolution: + { + integrity: sha512-lb7XXXzmm2h2ASzFnRvQpDo6onT1NmMJA3tkGTWiBFtRJ9lxGY3d3mm/Apt36gej2bkkOVLL/yTOtufDaFa/jA==, + } '@vitest/runner@3.2.6': - resolution: {integrity: sha512-HYcoSj1w5tcgUnzoF0HcyaAQjpA1gj9ftUJ7iSJSuipc02jW9gKkigwZbjFldAfYHA1fa8UZVRftdMY5msWM9Q==} + resolution: + { + integrity: sha512-HYcoSj1w5tcgUnzoF0HcyaAQjpA1gj9ftUJ7iSJSuipc02jW9gKkigwZbjFldAfYHA1fa8UZVRftdMY5msWM9Q==, + } '@vitest/snapshot@3.2.6': - resolution: {integrity: sha512-H+ZjNTWGpObenh0YnlBctAPnJSI20P81PL8BPzWpx54YXLLTm8hEsWawtcYLMrwvpK48hGxLLbCS+1KRXhsKhw==} + resolution: + { + integrity: sha512-H+ZjNTWGpObenh0YnlBctAPnJSI20P81PL8BPzWpx54YXLLTm8hEsWawtcYLMrwvpK48hGxLLbCS+1KRXhsKhw==, + } '@vitest/spy@3.2.6': - resolution: {integrity: sha512-oq6BbH68WzcWmwtBrU9nqLeaXTR4XwJF7FSLkKEZo4i6eoXcrxjcwSuTvWBIRUTC6VC72nXYunzqgZA+IKdtxg==} + resolution: + { + integrity: sha512-oq6BbH68WzcWmwtBrU9nqLeaXTR4XwJF7FSLkKEZo4i6eoXcrxjcwSuTvWBIRUTC6VC72nXYunzqgZA+IKdtxg==, + } '@vitest/utils@3.2.6': - resolution: {integrity: sha512-lI23nIs4bnT3T8NIoh+vFaz5s2/DdP0Jgt2jxwgWljvwn82cLJtyi/If+fjFyoLMGIOz0U/fKvWE0d4jsNQEfg==} + resolution: + { + integrity: sha512-lI23nIs4bnT3T8NIoh+vFaz5s2/DdP0Jgt2jxwgWljvwn82cLJtyi/If+fjFyoLMGIOz0U/fKvWE0d4jsNQEfg==, + } JSONStream@1.3.5: - resolution: {integrity: sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==} + resolution: + { + integrity: sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==, + } hasBin: true abitype@1.2.3: - resolution: {integrity: sha512-Ofer5QUnuUdTFsBRwARMoWKOH1ND5ehwYhJ3OJ/BQO+StkwQjHw0XyVh4vDttzHB7QOFhPHa/o413PJ82gU/Tg==} + resolution: + { + integrity: sha512-Ofer5QUnuUdTFsBRwARMoWKOH1ND5ehwYhJ3OJ/BQO+StkwQjHw0XyVh4vDttzHB7QOFhPHa/o413PJ82gU/Tg==, + } peerDependencies: typescript: '>=5.0.4' zod: ^3.22.0 || ^4.0.0 @@ -589,54 +920,96 @@ packages: optional: true acorn@8.17.0: - resolution: {integrity: sha512-xRQbDb9BnwDafYNn6Vwl839DYVjqXYb1XVGtWAZ1kcDc6iwAL4hg3B1dZlRiuENFeO2H53gFG3in621AdERVAg==} - engines: {node: '>=0.4.0'} + resolution: + { + integrity: sha512-xRQbDb9BnwDafYNn6Vwl839DYVjqXYb1XVGtWAZ1kcDc6iwAL4hg3B1dZlRiuENFeO2H53gFG3in621AdERVAg==, + } + engines: { node: '>=0.4.0' } hasBin: true agent-base@6.0.2: - resolution: {integrity: sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==} - engines: {node: '>= 6.0.0'} + resolution: + { + integrity: sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==, + } + engines: { node: '>= 6.0.0' } agentkeepalive@4.6.0: - resolution: {integrity: sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ==} - engines: {node: '>= 8.0.0'} + resolution: + { + integrity: sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ==, + } + engines: { node: '>= 8.0.0' } ajv@8.20.0: - resolution: {integrity: sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA==} + resolution: + { + integrity: sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA==, + } ansi-regex@5.0.1: - resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==, + } + engines: { node: '>=8' } ansi-styles@4.3.0: - resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==, + } + engines: { node: '>=8' } any-promise@1.3.0: - resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} + resolution: + { + integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==, + } argparse@2.0.1: - resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} + resolution: + { + integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==, + } array-ify@1.0.0: - resolution: {integrity: sha512-c5AMf34bKdvPhQ7tBGhqkgKNUzMr4WUs+WDtC2ZUGOUncbxKMTvqxYctiseW3+L4bA8ec+GcZ6/A/FW4m8ukng==} + resolution: + { + integrity: sha512-c5AMf34bKdvPhQ7tBGhqkgKNUzMr4WUs+WDtC2ZUGOUncbxKMTvqxYctiseW3+L4bA8ec+GcZ6/A/FW4m8ukng==, + } assertion-error@2.0.1: - resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==, + } + engines: { node: '>=12' } asynckit@0.4.0: - resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} + resolution: + { + integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==, + } available-typed-arrays@1.0.7: - resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==, + } + engines: { node: '>= 0.4' } axios@1.18.1: - resolution: {integrity: sha512-3nTvFlvpn9Zu/RkHUqtc7/+al4UpRW5az71ap5zccp6e8RAYEzhMTecX8Dz1wWDYrPpUoB1HAQEGEAEvUr7S9g==} + resolution: + { + integrity: sha512-3nTvFlvpn9Zu/RkHUqtc7/+al4UpRW5az71ap5zccp6e8RAYEzhMTecX8Dz1wWDYrPpUoB1HAQEGEAEvUr7S9g==, + } bare-addon-resolve@1.10.0: - resolution: {integrity: sha512-sSd0jieRJlDaODOzj0oe0RjFVC1QI0ZIjGIdPkbrTXsdVVtENg14c+lHHAhHwmWCZ2nQlMhy8jA3Y5LYPc/isA==} + resolution: + { + integrity: sha512-sSd0jieRJlDaODOzj0oe0RjFVC1QI0ZIjGIdPkbrTXsdVVtENg14c+lHHAhHwmWCZ2nQlMhy8jA3Y5LYPc/isA==, + } peerDependencies: bare-url: '*' peerDependenciesMeta: @@ -644,7 +1017,10 @@ packages: optional: true bare-module-resolve@1.12.2: - resolution: {integrity: sha512-j+hiD5k99qec4KjJvYsI67q5AOBifmy9JG3oeMVxTmvrhn2sIdp8StrUvZu4YNgwTpO+NhniQG16N1ETDe1k5w==} + resolution: + { + integrity: sha512-j+hiD5k99qec4KjJvYsI67q5AOBifmy9JG3oeMVxTmvrhn2sIdp8StrUvZu4YNgwTpO+NhniQG16N1ETDe1k5w==, + } peerDependencies: bare-url: '*' peerDependenciesMeta: @@ -652,139 +1028,244 @@ packages: optional: true bare-semver@1.1.0: - resolution: {integrity: sha512-1Hw5qJ7hXdVt3uPUqjeFTuxyvBUJauvz5A1I2jk8gzjZMHp04n//6nV9MDbG9CMw78JHY2lGV0w6s//LrASm2w==} + resolution: + { + integrity: sha512-1Hw5qJ7hXdVt3uPUqjeFTuxyvBUJauvz5A1I2jk8gzjZMHp04n//6nV9MDbG9CMw78JHY2lGV0w6s//LrASm2w==, + } base-x@3.0.11: - resolution: {integrity: sha512-xz7wQ8xDhdyP7tQxwdteLYeFfS68tSMNCZ/Y37WJ4bhGfKPpqEIlmIyueQHqOyoPhE6xNUqjzRr8ra0eF9VRvA==} + resolution: + { + integrity: sha512-xz7wQ8xDhdyP7tQxwdteLYeFfS68tSMNCZ/Y37WJ4bhGfKPpqEIlmIyueQHqOyoPhE6xNUqjzRr8ra0eF9VRvA==, + } base32.js@0.1.0: - resolution: {integrity: sha512-n3TkB02ixgBOhTvANakDb4xaMXnYUVkNoRFJjQflcqMQhyEKxEHdj3E6N8t8sUQ0mjH/3/JxzlXuz3ul/J90pQ==} - engines: {node: '>=0.12.0'} + resolution: + { + integrity: sha512-n3TkB02ixgBOhTvANakDb4xaMXnYUVkNoRFJjQflcqMQhyEKxEHdj3E6N8t8sUQ0mjH/3/JxzlXuz3ul/J90pQ==, + } + engines: { node: '>=0.12.0' } base64-js@1.5.1: - resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} + resolution: + { + integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==, + } bignumber.js@9.3.1: - resolution: {integrity: sha512-Ko0uX15oIUS7wJ3Rb30Fs6SkVbLmPBAKdlm7q9+ak9bbIeFf0MwuBsQV6z7+X768/cHsfg+WlysDWJcmthjsjQ==} + resolution: + { + integrity: sha512-Ko0uX15oIUS7wJ3Rb30Fs6SkVbLmPBAKdlm7q9+ak9bbIeFf0MwuBsQV6z7+X768/cHsfg+WlysDWJcmthjsjQ==, + } bn.js@5.2.3: - resolution: {integrity: sha512-EAcmnPkxpntVL+DS7bO1zhcZNvCkxqtkd0ZY53h06GNQ3DEkkGZ/gKgmDv6DdZQGj9BgfSPKtJJ7Dp1GPP8f7w==} + resolution: + { + integrity: sha512-EAcmnPkxpntVL+DS7bO1zhcZNvCkxqtkd0ZY53h06GNQ3DEkkGZ/gKgmDv6DdZQGj9BgfSPKtJJ7Dp1GPP8f7w==, + } borsh@0.7.0: - resolution: {integrity: sha512-CLCsZGIBCFnPtkNnieW/a8wmreDmfUtjU2m9yHrzPXIlNbqVs0AQrSatSG6vdNYUqdc83tkQi2eHfF98ubzQLA==} + resolution: + { + integrity: sha512-CLCsZGIBCFnPtkNnieW/a8wmreDmfUtjU2m9yHrzPXIlNbqVs0AQrSatSG6vdNYUqdc83tkQi2eHfF98ubzQLA==, + } bs58@4.0.1: - resolution: {integrity: sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==} + resolution: + { + integrity: sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==, + } buffer@6.0.3: - resolution: {integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==} + resolution: + { + integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==, + } bufferutil@4.1.0: - resolution: {integrity: sha512-ZMANVnAixE6AWWnPzlW2KpUrxhm9woycYvPOo67jWHyFowASTEd9s+QN1EIMsSDtwhIxN4sWE1jotpuDUIgyIw==} - engines: {node: '>=6.14.2'} + resolution: + { + integrity: sha512-ZMANVnAixE6AWWnPzlW2KpUrxhm9woycYvPOo67jWHyFowASTEd9s+QN1EIMsSDtwhIxN4sWE1jotpuDUIgyIw==, + } + engines: { node: '>=6.14.2' } bundle-require@5.1.0: - resolution: {integrity: sha512-3WrrOuZiyaaZPWiEt4G3+IffISVC9HYlWueJEBWED4ZH4aIAC2PnkdnuRrR94M+w6yGWn4AglWtJtBI8YqvgoA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + resolution: + { + integrity: sha512-3WrrOuZiyaaZPWiEt4G3+IffISVC9HYlWueJEBWED4ZH4aIAC2PnkdnuRrR94M+w6yGWn4AglWtJtBI8YqvgoA==, + } + engines: { node: ^12.20.0 || ^14.13.1 || >=16.0.0 } peerDependencies: esbuild: '>=0.18' cac@6.7.14: - resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==, + } + engines: { node: '>=8' } call-bind-apply-helpers@1.0.2: - resolution: {integrity: sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==, + } + engines: { node: '>= 0.4' } call-bind@1.0.9: - resolution: {integrity: sha512-a/hy+pNsFUTR+Iz8TCJvXudKVLAnz/DyeSUo10I5yvFDQJBFU2s9uqQpoSrJlroHUKoKqzg+epxyP9lqFdzfBQ==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-a/hy+pNsFUTR+Iz8TCJvXudKVLAnz/DyeSUo10I5yvFDQJBFU2s9uqQpoSrJlroHUKoKqzg+epxyP9lqFdzfBQ==, + } + engines: { node: '>= 0.4' } call-bound@1.0.4: - resolution: {integrity: sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==, + } + engines: { node: '>= 0.4' } callsites@3.1.0: - resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} - engines: {node: '>=6'} + resolution: + { + integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==, + } + engines: { node: '>=6' } chai@5.3.3: - resolution: {integrity: sha512-4zNhdJD/iOjSH0A05ea+Ke6MU5mmpQcbQsSOkgdaUMJ9zTlDTD/GYlwohmIE2u0gaxHYiVHEn1Fw9mZ/ktJWgw==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-4zNhdJD/iOjSH0A05ea+Ke6MU5mmpQcbQsSOkgdaUMJ9zTlDTD/GYlwohmIE2u0gaxHYiVHEn1Fw9mZ/ktJWgw==, + } + engines: { node: '>=18' } chalk@5.6.2: - resolution: {integrity: sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA==} - engines: {node: ^12.17.0 || ^14.13 || >=16.0.0} + resolution: + { + integrity: sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA==, + } + engines: { node: ^12.17.0 || ^14.13 || >=16.0.0 } check-error@2.1.3: - resolution: {integrity: sha512-PAJdDJusoxnwm1VwW07VWwUN1sl7smmC3OKggvndJFadxxDRyFJBX/ggnu/KE4kQAB7a3Dp8f/YXC1FlUprWmA==} - engines: {node: '>= 16'} + resolution: + { + integrity: sha512-PAJdDJusoxnwm1VwW07VWwUN1sl7smmC3OKggvndJFadxxDRyFJBX/ggnu/KE4kQAB7a3Dp8f/YXC1FlUprWmA==, + } + engines: { node: '>= 16' } chokidar@4.0.3: - resolution: {integrity: sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==} - engines: {node: '>= 14.16.0'} + resolution: + { + integrity: sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==, + } + engines: { node: '>= 14.16.0' } cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==, + } + engines: { node: '>=12' } color-convert@2.0.1: - resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} - engines: {node: '>=7.0.0'} + resolution: + { + integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==, + } + engines: { node: '>=7.0.0' } color-name@1.1.4: - resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + resolution: + { + integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==, + } combined-stream@1.0.8: - resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} - engines: {node: '>= 0.8'} + resolution: + { + integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==, + } + engines: { node: '>= 0.8' } commander@14.0.3: - resolution: {integrity: sha512-H+y0Jo/T1RZ9qPP4Eh1pkcQcLRglraJaSLoyOtHxu6AapkjWVCy2Sit1QQ4x3Dng8qDlSsZEet7g5Pq06MvTgw==} - engines: {node: '>=20'} + resolution: + { + integrity: sha512-H+y0Jo/T1RZ9qPP4Eh1pkcQcLRglraJaSLoyOtHxu6AapkjWVCy2Sit1QQ4x3Dng8qDlSsZEet7g5Pq06MvTgw==, + } + engines: { node: '>=20' } commander@2.20.3: - resolution: {integrity: sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==} + resolution: + { + integrity: sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==, + } commander@4.1.1: - resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} - engines: {node: '>= 6'} + resolution: + { + integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==, + } + engines: { node: '>= 6' } compare-func@2.0.0: - resolution: {integrity: sha512-zHig5N+tPWARooBnb0Zx1MFcdfpyJrfTJ3Y5L+IFvUm8rM74hHz66z0gw0x4tijh5CorKkKUCnW82R2vmpeCRA==} + resolution: + { + integrity: sha512-zHig5N+tPWARooBnb0Zx1MFcdfpyJrfTJ3Y5L+IFvUm8rM74hHz66z0gw0x4tijh5CorKkKUCnW82R2vmpeCRA==, + } confbox@0.1.8: - resolution: {integrity: sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w==} + resolution: + { + integrity: sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w==, + } consola@3.4.2: - resolution: {integrity: sha512-5IKcdX0nnYavi6G7TtOhwkYzyjfJlatbjMjuLSfE2kYT5pMDOilZ4OvMhi637CcDICTmz3wARPoyhqyX1Y+XvA==} - engines: {node: ^14.18.0 || >=16.10.0} + resolution: + { + integrity: sha512-5IKcdX0nnYavi6G7TtOhwkYzyjfJlatbjMjuLSfE2kYT5pMDOilZ4OvMhi637CcDICTmz3wARPoyhqyX1Y+XvA==, + } + engines: { node: ^14.18.0 || >=16.10.0 } conventional-changelog-angular@7.0.0: - resolution: {integrity: sha512-ROjNchA9LgfNMTTFSIWPzebCwOGFdgkEq45EnvvrmSLvCtAw0HSmrCs7/ty+wAeYUZyNay0YMUNYFTRL72PkBQ==} - engines: {node: '>=16'} + resolution: + { + integrity: sha512-ROjNchA9LgfNMTTFSIWPzebCwOGFdgkEq45EnvvrmSLvCtAw0HSmrCs7/ty+wAeYUZyNay0YMUNYFTRL72PkBQ==, + } + engines: { node: '>=16' } conventional-changelog-conventionalcommits@7.0.2: - resolution: {integrity: sha512-NKXYmMR/Hr1DevQegFB4MwfM5Vv0m4UIxKZTTYuD98lpTknaZlSRrDOG4X7wIXpGkfsYxZTghUN+Qq+T0YQI7w==} - engines: {node: '>=16'} + resolution: + { + integrity: sha512-NKXYmMR/Hr1DevQegFB4MwfM5Vv0m4UIxKZTTYuD98lpTknaZlSRrDOG4X7wIXpGkfsYxZTghUN+Qq+T0YQI7w==, + } + engines: { node: '>=16' } conventional-commits-parser@5.0.0: - resolution: {integrity: sha512-ZPMl0ZJbw74iS9LuX9YIAiW8pfM5p3yh2o/NbXHbkFuZzY5jvdi5jFycEOkmBW5H5I7nA+D6f3UcsCLP2vvSEA==} - engines: {node: '>=16'} + resolution: + { + integrity: sha512-ZPMl0ZJbw74iS9LuX9YIAiW8pfM5p3yh2o/NbXHbkFuZzY5jvdi5jFycEOkmBW5H5I7nA+D6f3UcsCLP2vvSEA==, + } + engines: { node: '>=16' } hasBin: true cosmiconfig-typescript-loader@6.3.0: - resolution: {integrity: sha512-Akr82WH1Wfqatyiqpj8HDkO2o2KmJRu1FhKfSNJP3K4IdXwHfEyL7MOb62i1AGQVLtIQM+iCE9CGOtrfhR+mmA==} - engines: {node: '>=v18'} + resolution: + { + integrity: sha512-Akr82WH1Wfqatyiqpj8HDkO2o2KmJRu1FhKfSNJP3K4IdXwHfEyL7MOb62i1AGQVLtIQM+iCE9CGOtrfhR+mmA==, + } + engines: { node: '>=v18' } peerDependencies: '@types/node': '*' cosmiconfig: '>=9' typescript: '>=5' cosmiconfig@9.0.2: - resolution: {integrity: sha512-gtTZxTDau1wL7Y7zifc2dd8jHSK/k6BTx/2Xp/BpdlAdnlYWFVt7qhJqgwi7637yRwRQ3qL4ZidbB4I8tA5VOg==} - engines: {node: '>=14'} + resolution: + { + integrity: sha512-gtTZxTDau1wL7Y7zifc2dd8jHSK/k6BTx/2Xp/BpdlAdnlYWFVt7qhJqgwi7637yRwRQ3qL4ZidbB4I8tA5VOg==, + } + engines: { node: '>=14' } peerDependencies: typescript: '>=4.9.5' peerDependenciesMeta: @@ -792,12 +1273,18 @@ packages: optional: true dargs@8.1.0: - resolution: {integrity: sha512-wAV9QHOsNbwnWdNW2FYvE1P56wtgSbM+3SZcdGiWQILwVjACCXDCI3Ai8QlCjMDB8YK5zySiXZYBiwGmNY3lnw==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-wAV9QHOsNbwnWdNW2FYvE1P56wtgSbM+3SZcdGiWQILwVjACCXDCI3Ai8QlCjMDB8YK5zySiXZYBiwGmNY3lnw==, + } + engines: { node: '>=12' } debug@4.4.3: - resolution: {integrity: sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==} - engines: {node: '>=6.0'} + resolution: + { + integrity: sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==, + } + engines: { node: '>=6.0' } peerDependencies: supports-color: '*' peerDependenciesMeta: @@ -805,114 +1292,204 @@ packages: optional: true deep-eql@5.0.2: - resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} - engines: {node: '>=6'} + resolution: + { + integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==, + } + engines: { node: '>=6' } define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==, + } + engines: { node: '>= 0.4' } delay@5.0.0: - resolution: {integrity: sha512-ReEBKkIfe4ya47wlPYf/gu5ib6yUG0/Aez0JQZQz94kiWtRQvZIQbTiehsnwHvLSWJnQdhVeqYue7Id1dKr0qw==} - engines: {node: '>=10'} + resolution: + { + integrity: sha512-ReEBKkIfe4ya47wlPYf/gu5ib6yUG0/Aez0JQZQz94kiWtRQvZIQbTiehsnwHvLSWJnQdhVeqYue7Id1dKr0qw==, + } + engines: { node: '>=10' } delayed-stream@1.0.0: - resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} - engines: {node: '>=0.4.0'} + resolution: + { + integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==, + } + engines: { node: '>=0.4.0' } dot-prop@5.3.0: - resolution: {integrity: sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==, + } + engines: { node: '>=8' } dunder-proto@1.0.1: - resolution: {integrity: sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==, + } + engines: { node: '>= 0.4' } emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} + resolution: + { + integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==, + } env-paths@2.2.1: - resolution: {integrity: sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==} - engines: {node: '>=6'} + resolution: + { + integrity: sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==, + } + engines: { node: '>=6' } error-ex@1.3.4: - resolution: {integrity: sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==} + resolution: + { + integrity: sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==, + } es-define-property@1.0.1: - resolution: {integrity: sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==, + } + engines: { node: '>= 0.4' } es-errors@1.3.0: - resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==, + } + engines: { node: '>= 0.4' } es-module-lexer@1.7.0: - resolution: {integrity: sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==} + resolution: + { + integrity: sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==, + } es-object-atoms@1.1.2: - resolution: {integrity: sha512-HWcBoN6NileqtSydK2FqHbS/LoDd2pqrnQHLyJzBj4kOp/ky2MWMN694xOfkK8/SnUsW2DH7EfyVlydKCsm1Zw==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-HWcBoN6NileqtSydK2FqHbS/LoDd2pqrnQHLyJzBj4kOp/ky2MWMN694xOfkK8/SnUsW2DH7EfyVlydKCsm1Zw==, + } + engines: { node: '>= 0.4' } es-set-tostringtag@2.1.0: - resolution: {integrity: sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==, + } + engines: { node: '>= 0.4' } es6-promise@4.2.8: - resolution: {integrity: sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==} + resolution: + { + integrity: sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==, + } es6-promisify@5.0.0: - resolution: {integrity: sha512-C+d6UdsYDk0lMebHNR4S2NybQMMngAOnOwYBQjTOiv0MkoJMP0Myw2mgpDLBcpfCmRLxyFqYhS/CfOENq4SJhQ==} + resolution: + { + integrity: sha512-C+d6UdsYDk0lMebHNR4S2NybQMMngAOnOwYBQjTOiv0MkoJMP0Myw2mgpDLBcpfCmRLxyFqYhS/CfOENq4SJhQ==, + } esbuild@0.27.7: - resolution: {integrity: sha512-IxpibTjyVnmrIQo5aqNpCgoACA/dTKLTlhMHihVHhdkxKyPO1uBBthumT0rdHmcsk9uMonIWS0m4FljWzILh3w==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-IxpibTjyVnmrIQo5aqNpCgoACA/dTKLTlhMHihVHhdkxKyPO1uBBthumT0rdHmcsk9uMonIWS0m4FljWzILh3w==, + } + engines: { node: '>=18' } hasBin: true escalade@3.2.0: - resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} - engines: {node: '>=6'} + resolution: + { + integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==, + } + engines: { node: '>=6' } estree-walker@3.0.3: - resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} + resolution: + { + integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==, + } eventemitter3@5.0.1: - resolution: {integrity: sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==} + resolution: + { + integrity: sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==, + } eventemitter3@5.0.4: - resolution: {integrity: sha512-mlsTRyGaPBjPedk6Bvw+aqbsXDtoAyAzm5MO7JgU+yVRyMQ5O8bD4Kcci7BS85f93veegeCPkL8R4GLClnjLFw==} + resolution: + { + integrity: sha512-mlsTRyGaPBjPedk6Bvw+aqbsXDtoAyAzm5MO7JgU+yVRyMQ5O8bD4Kcci7BS85f93veegeCPkL8R4GLClnjLFw==, + } eventsource@2.0.2: - resolution: {integrity: sha512-IzUmBGPR3+oUG9dUeXynyNmf91/3zUSJg1lCktzKw47OXuhco54U3r9B7O4XX+Rb1Itm9OZ2b0RkTs10bICOxA==} - engines: {node: '>=12.0.0'} + resolution: + { + integrity: sha512-IzUmBGPR3+oUG9dUeXynyNmf91/3zUSJg1lCktzKw47OXuhco54U3r9B7O4XX+Rb1Itm9OZ2b0RkTs10bICOxA==, + } + engines: { node: '>=12.0.0' } expect-type@1.3.0: - resolution: {integrity: sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==} - engines: {node: '>=12.0.0'} + resolution: + { + integrity: sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==, + } + engines: { node: '>=12.0.0' } eyes@0.1.8: - resolution: {integrity: sha512-GipyPsXO1anza0AOZdy69Im7hGFCNB7Y/NGjDlZGJ3GJJLtwNSb2vrzYrTYJRrRloVx7pl+bhUaTB8yiccPvFQ==} - engines: {node: '> 0.1.90'} + resolution: + { + integrity: sha512-GipyPsXO1anza0AOZdy69Im7hGFCNB7Y/NGjDlZGJ3GJJLtwNSb2vrzYrTYJRrRloVx7pl+bhUaTB8yiccPvFQ==, + } + engines: { node: '> 0.1.90' } fake-indexeddb@6.2.5: - resolution: {integrity: sha512-CGnyrvbhPlWYMngksqrSSUT1BAVP49dZocrHuK0SvtR0D5TMs5wP0o3j7jexDJW01KSadjBp1M/71o/KR3nD1w==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-CGnyrvbhPlWYMngksqrSSUT1BAVP49dZocrHuK0SvtR0D5TMs5wP0o3j7jexDJW01KSadjBp1M/71o/KR3nD1w==, + } + engines: { node: '>=18' } fast-check@4.8.0: - resolution: {integrity: sha512-GOJ158CUMnN6cSahsv4+ExARvIDuzzinFjkp0E9WtiBa5zcVeLozVkWaE4IzFcc+Y48Wp1EDlUZsXRyAztQcSg==} - engines: {node: '>=12.17.0'} + resolution: + { + integrity: sha512-GOJ158CUMnN6cSahsv4+ExARvIDuzzinFjkp0E9WtiBa5zcVeLozVkWaE4IzFcc+Y48Wp1EDlUZsXRyAztQcSg==, + } + engines: { node: '>=12.17.0' } fast-deep-equal@3.1.3: - resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} + resolution: + { + integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==, + } fast-stable-stringify@1.0.0: - resolution: {integrity: sha512-wpYMUmFu5f00Sm0cj2pfivpmawLZ0NKdviQ4w9zJeR8JVtOpOxHmLaJuj0vxvGqMJQWyP/COUkF75/57OKyRag==} + resolution: + { + integrity: sha512-wpYMUmFu5f00Sm0cj2pfivpmawLZ0NKdviQ4w9zJeR8JVtOpOxHmLaJuj0vxvGqMJQWyP/COUkF75/57OKyRag==, + } fast-uri@3.1.2: - resolution: {integrity: sha512-rVjf7ArG3LTk+FS6Yw81V1DLuZl1bRbNrev6Tmd/9RaroeeRRJhAt7jg/6YFxbvAQXUCavSoZhPPj6oOx+5KjQ==} + resolution: + { + integrity: sha512-rVjf7ArG3LTk+FS6Yw81V1DLuZl1bRbNrev6Tmd/9RaroeeRRJhAt7jg/6YFxbvAQXUCavSoZhPPj6oOx+5KjQ==, + } fdir@6.5.0: - resolution: {integrity: sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==} - engines: {node: '>=12.0.0'} + resolution: + { + integrity: sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==, + } + engines: { node: '>=12.0.0' } peerDependencies: picomatch: ^3 || ^4 peerDependenciesMeta: @@ -920,18 +1497,30 @@ packages: optional: true feaxios@0.0.23: - resolution: {integrity: sha512-eghR0A21fvbkcQBgZuMfQhrXxJzC0GNUGC9fXhBge33D+mFDTwl0aJ35zoQQn575BhyjQitRc5N4f+L4cP708g==} + resolution: + { + integrity: sha512-eghR0A21fvbkcQBgZuMfQhrXxJzC0GNUGC9fXhBge33D+mFDTwl0aJ35zoQQn575BhyjQitRc5N4f+L4cP708g==, + } find-up@7.0.0: - resolution: {integrity: sha512-YyZM99iHrqLKjmt4LJDj58KI+fYyufRLBSYcqycxf//KpBk9FoewoGX0450m9nB44qrZnovzC2oeP5hUibxc/g==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-YyZM99iHrqLKjmt4LJDj58KI+fYyufRLBSYcqycxf//KpBk9FoewoGX0450m9nB44qrZnovzC2oeP5hUibxc/g==, + } + engines: { node: '>=18' } fix-dts-default-cjs-exports@1.0.1: - resolution: {integrity: sha512-pVIECanWFC61Hzl2+oOCtoJ3F17kglZC/6N94eRWycFgBH35hHx0Li604ZIzhseh97mf2p0cv7vVrOZGoqhlEg==} + resolution: + { + integrity: sha512-pVIECanWFC61Hzl2+oOCtoJ3F17kglZC/6N94eRWycFgBH35hHx0Li604ZIzhseh97mf2p0cv7vVrOZGoqhlEg==, + } follow-redirects@1.16.0: - resolution: {integrity: sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==} - engines: {node: '>=4.0'} + resolution: + { + integrity: sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==, + } + engines: { node: '>=4.0' } peerDependencies: debug: '*' peerDependenciesMeta: @@ -939,251 +1528,452 @@ packages: optional: true for-each@0.3.5: - resolution: {integrity: sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==, + } + engines: { node: '>= 0.4' } form-data@4.0.6: - resolution: {integrity: sha512-vKatAh4SlVfgbv+YtmhiRjhEMJsYpsG1Y2rMQtR+SVSbytsSD1YGzDIcrAJmdFec88u/+VoGmxnl+80gL1tRCQ==} - engines: {node: '>= 6'} + resolution: + { + integrity: sha512-vKatAh4SlVfgbv+YtmhiRjhEMJsYpsG1Y2rMQtR+SVSbytsSD1YGzDIcrAJmdFec88u/+VoGmxnl+80gL1tRCQ==, + } + engines: { node: '>= 6' } fsevents@2.3.3: - resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} - engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + resolution: + { + integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==, + } + engines: { node: ^8.16.0 || ^10.6.0 || >=11.0.0 } os: [darwin] function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} + resolution: + { + integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==, + } get-caller-file@2.0.5: - resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} - engines: {node: 6.* || 8.* || >= 10.*} + resolution: + { + integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==, + } + engines: { node: 6.* || 8.* || >= 10.* } get-intrinsic@1.3.0: - resolution: {integrity: sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==, + } + engines: { node: '>= 0.4' } get-proto@1.0.1: - resolution: {integrity: sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==, + } + engines: { node: '>= 0.4' } git-raw-commits@4.0.0: - resolution: {integrity: sha512-ICsMM1Wk8xSGMowkOmPrzo2Fgmfo4bMHLNX6ytHjajRJUqvHOw/TFapQ+QG75c3X/tTDDhOSRPGC52dDbNM8FQ==} - engines: {node: '>=16'} + resolution: + { + integrity: sha512-ICsMM1Wk8xSGMowkOmPrzo2Fgmfo4bMHLNX6ytHjajRJUqvHOw/TFapQ+QG75c3X/tTDDhOSRPGC52dDbNM8FQ==, + } + engines: { node: '>=16' } deprecated: This package is no longer maintained. For the JavaScript API, please use @conventional-changelog/git-client instead. hasBin: true global-directory@4.0.1: - resolution: {integrity: sha512-wHTUcDUoZ1H5/0iVqEudYW4/kAlN5cZ3j/bXn0Dpbizl9iaUVeWSHqiOjsgk6OW2bkLclbBjzewBz6weQ1zA2Q==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-wHTUcDUoZ1H5/0iVqEudYW4/kAlN5cZ3j/bXn0Dpbizl9iaUVeWSHqiOjsgk6OW2bkLclbBjzewBz6weQ1zA2Q==, + } + engines: { node: '>=18' } gopd@1.2.0: - resolution: {integrity: sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==, + } + engines: { node: '>= 0.4' } has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} + resolution: + { + integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==, + } has-symbols@1.1.0: - resolution: {integrity: sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==, + } + engines: { node: '>= 0.4' } has-tostringtag@1.0.2: - resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==, + } + engines: { node: '>= 0.4' } hasown@2.0.4: - resolution: {integrity: sha512-T2UbfbBEF32wiepXIsMlTW9+dDYC6wMh/t/vYA4tuOMKqWz/n3vr1NFSxQiyP+zk2mXsoMA/i/7qV6LKut1t1A==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-T2UbfbBEF32wiepXIsMlTW9+dDYC6wMh/t/vYA4tuOMKqWz/n3vr1NFSxQiyP+zk2mXsoMA/i/7qV6LKut1t1A==, + } + engines: { node: '>= 0.4' } https-proxy-agent@5.0.1: - resolution: {integrity: sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==} - engines: {node: '>= 6'} + resolution: + { + integrity: sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==, + } + engines: { node: '>= 6' } humanize-ms@1.2.1: - resolution: {integrity: sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==} + resolution: + { + integrity: sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==, + } husky@9.1.7: - resolution: {integrity: sha512-5gs5ytaNjBrh5Ow3zrvdUUY+0VxIuWVL4i9irt6friV+BqdCfmV11CQTWMiBYWHbXhco+J1kHfTOUkePhCDvMA==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-5gs5ytaNjBrh5Ow3zrvdUUY+0VxIuWVL4i9irt6friV+BqdCfmV11CQTWMiBYWHbXhco+J1kHfTOUkePhCDvMA==, + } + engines: { node: '>=18' } hasBin: true ieee754@1.2.1: - resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} + resolution: + { + integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==, + } import-fresh@3.3.1: - resolution: {integrity: sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==} - engines: {node: '>=6'} + resolution: + { + integrity: sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==, + } + engines: { node: '>=6' } import-meta-resolve@4.2.0: - resolution: {integrity: sha512-Iqv2fzaTQN28s/FwZAoFq0ZSs/7hMAHJVX+w8PZl3cY19Pxk6jFFalxQoIfW2826i/fDLXv8IiEZRIT0lDuWcg==} + resolution: + { + integrity: sha512-Iqv2fzaTQN28s/FwZAoFq0ZSs/7hMAHJVX+w8PZl3cY19Pxk6jFFalxQoIfW2826i/fDLXv8IiEZRIT0lDuWcg==, + } inherits@2.0.4: - resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} + resolution: + { + integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==, + } ini@4.1.1: - resolution: {integrity: sha512-QQnnxNyfvmHFIsj7gkPcYymR8Jdw/o7mp5ZFihxn6h8Ci6fh3Dx4E1gPjpQEpIuPo9XVNY/ZUwh4BPMjGyL01g==} - engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + resolution: + { + integrity: sha512-QQnnxNyfvmHFIsj7gkPcYymR8Jdw/o7mp5ZFihxn6h8Ci6fh3Dx4E1gPjpQEpIuPo9XVNY/ZUwh4BPMjGyL01g==, + } + engines: { node: ^14.17.0 || ^16.13.0 || >=18.0.0 } is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} + resolution: + { + integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==, + } is-callable@1.2.7: - resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==, + } + engines: { node: '>= 0.4' } is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==, + } + engines: { node: '>=8' } is-obj@2.0.0: - resolution: {integrity: sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==, + } + engines: { node: '>=8' } is-retry-allowed@3.0.0: - resolution: {integrity: sha512-9xH0xvoggby+u0uGF7cZXdrutWiBiaFG8ZT4YFPXL8NzkyAwX3AKGLeFQLvzDpM430+nDFBZ1LHkie/8ocL06A==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-9xH0xvoggby+u0uGF7cZXdrutWiBiaFG8ZT4YFPXL8NzkyAwX3AKGLeFQLvzDpM430+nDFBZ1LHkie/8ocL06A==, + } + engines: { node: '>=12' } is-text-path@2.0.0: - resolution: {integrity: sha512-+oDTluR6WEjdXEJMnC2z6A4FRwFoYuvShVVEGsS7ewc0UTi2QtAKMDJuL4BDEVt+5T7MjFo12RP8ghOM75oKJw==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-+oDTluR6WEjdXEJMnC2z6A4FRwFoYuvShVVEGsS7ewc0UTi2QtAKMDJuL4BDEVt+5T7MjFo12RP8ghOM75oKJw==, + } + engines: { node: '>=8' } is-typed-array@1.1.15: - resolution: {integrity: sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==, + } + engines: { node: '>= 0.4' } isarray@2.0.5: - resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} + resolution: + { + integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==, + } isomorphic-ws@4.0.1: - resolution: {integrity: sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w==} + resolution: + { + integrity: sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w==, + } peerDependencies: ws: '*' isows@1.0.7: - resolution: {integrity: sha512-I1fSfDCZL5P0v33sVqeTDSpcstAg/N+wF5HS033mogOVIp4B+oHC7oOCsA3axAbBSGTJ8QubbNmnIRN/h8U7hg==} + resolution: + { + integrity: sha512-I1fSfDCZL5P0v33sVqeTDSpcstAg/N+wF5HS033mogOVIp4B+oHC7oOCsA3axAbBSGTJ8QubbNmnIRN/h8U7hg==, + } peerDependencies: ws: '*' jayson@4.3.0: - resolution: {integrity: sha512-AauzHcUcqs8OBnCHOkJY280VaTiCm57AbuO7lqzcw7JapGj50BisE3xhksye4zlTSR1+1tAz67wLTl8tEH1obQ==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-AauzHcUcqs8OBnCHOkJY280VaTiCm57AbuO7lqzcw7JapGj50BisE3xhksye4zlTSR1+1tAz67wLTl8tEH1obQ==, + } + engines: { node: '>=8' } hasBin: true jiti@2.6.1: - resolution: {integrity: sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ==} + resolution: + { + integrity: sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ==, + } hasBin: true joycon@3.1.1: - resolution: {integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==} - engines: {node: '>=10'} + resolution: + { + integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==, + } + engines: { node: '>=10' } js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} + resolution: + { + integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==, + } js-tokens@9.0.1: - resolution: {integrity: sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==} + resolution: + { + integrity: sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==, + } js-yaml@4.2.0: - resolution: {integrity: sha512-ePWsvanv0DWuDRsW8dnt+R4jQ31SCRCQ7hhNcPXZPsoBZiemuZNYGf7adZdqX2D86j6rvKp3RpCxVTSb8WQlOw==} + resolution: + { + integrity: sha512-ePWsvanv0DWuDRsW8dnt+R4jQ31SCRCQ7hhNcPXZPsoBZiemuZNYGf7adZdqX2D86j6rvKp3RpCxVTSb8WQlOw==, + } hasBin: true json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} + resolution: + { + integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==, + } json-schema-traverse@1.0.0: - resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} + resolution: + { + integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==, + } json-stringify-safe@5.0.1: - resolution: {integrity: sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==} + resolution: + { + integrity: sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==, + } jsonparse@1.3.1: - resolution: {integrity: sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==} - engines: {'0': node >= 0.2.0} + resolution: + { + integrity: sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==, + } + engines: { '0': node >= 0.2.0 } lilconfig@3.1.3: - resolution: {integrity: sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw==} - engines: {node: '>=14'} + resolution: + { + integrity: sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw==, + } + engines: { node: '>=14' } lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} + resolution: + { + integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==, + } load-tsconfig@0.2.5: - resolution: {integrity: sha512-IXO6OCs9yg8tMKzfPZ1YmheJbZCiEsnBdcB03l0OcfK9prKnJb96siuHCr5Fl37/yo9DnKU+TLpxzTUspw9shg==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + resolution: + { + integrity: sha512-IXO6OCs9yg8tMKzfPZ1YmheJbZCiEsnBdcB03l0OcfK9prKnJb96siuHCr5Fl37/yo9DnKU+TLpxzTUspw9shg==, + } + engines: { node: ^12.20.0 || ^14.13.1 || >=16.0.0 } locate-path@7.2.0: - resolution: {integrity: sha512-gvVijfZvn7R+2qyPX8mAuKcFGDf6Nc61GdvGafQsHL0sBIxfKzA+usWn4GFC/bk+QdwPUD4kWFJLhElipq+0VA==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + resolution: + { + integrity: sha512-gvVijfZvn7R+2qyPX8mAuKcFGDf6Nc61GdvGafQsHL0sBIxfKzA+usWn4GFC/bk+QdwPUD4kWFJLhElipq+0VA==, + } + engines: { node: ^12.20.0 || ^14.13.1 || >=16.0.0 } lodash.camelcase@4.3.0: - resolution: {integrity: sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==} + resolution: + { + integrity: sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==, + } lodash.isplainobject@4.0.6: - resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==} + resolution: + { + integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==, + } lodash.kebabcase@4.1.1: - resolution: {integrity: sha512-N8XRTIMMqqDgSy4VLKPnJ/+hpGZN+PHQiJnSenYqPaVV/NCqEogTnAdZLQiGKhxX+JCs8waWq2t1XHWKOmlY8g==} + resolution: + { + integrity: sha512-N8XRTIMMqqDgSy4VLKPnJ/+hpGZN+PHQiJnSenYqPaVV/NCqEogTnAdZLQiGKhxX+JCs8waWq2t1XHWKOmlY8g==, + } lodash.merge@4.6.2: - resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} + resolution: + { + integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==, + } lodash.mergewith@4.6.2: - resolution: {integrity: sha512-GK3g5RPZWTRSeLSpgP8Xhra+pnjBC56q9FZYe1d5RN3TJ35dbkGy3YqBSMbyCrlbi+CM9Z3Jk5yTL7RCsqboyQ==} + resolution: + { + integrity: sha512-GK3g5RPZWTRSeLSpgP8Xhra+pnjBC56q9FZYe1d5RN3TJ35dbkGy3YqBSMbyCrlbi+CM9Z3Jk5yTL7RCsqboyQ==, + } lodash.snakecase@4.1.1: - resolution: {integrity: sha512-QZ1d4xoBHYUeuouhEq3lk3Uq7ldgyFXGBhg04+oRLnIz8o9T65Eh+8YdroUwn846zchkA9yDsDl5CVVaV2nqYw==} + resolution: + { + integrity: sha512-QZ1d4xoBHYUeuouhEq3lk3Uq7ldgyFXGBhg04+oRLnIz8o9T65Eh+8YdroUwn846zchkA9yDsDl5CVVaV2nqYw==, + } lodash.startcase@4.4.0: - resolution: {integrity: sha512-+WKqsK294HMSc2jEbNgpHpd0JfIBhp7rEV4aqXWqFr6AlXov+SlcgB1Fv01y2kGe3Gc8nMW7VA0SrGuSkRfIEg==} + resolution: + { + integrity: sha512-+WKqsK294HMSc2jEbNgpHpd0JfIBhp7rEV4aqXWqFr6AlXov+SlcgB1Fv01y2kGe3Gc8nMW7VA0SrGuSkRfIEg==, + } lodash.uniq@4.5.0: - resolution: {integrity: sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==} + resolution: + { + integrity: sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==, + } lodash.upperfirst@4.3.1: - resolution: {integrity: sha512-sReKOYJIJf74dhJONhU4e0/shzi1trVbSWDOhKYE5XV2O+H7Sb2Dihwuc7xWxVl+DgFPyTqIN3zMfT9cq5iWDg==} + resolution: + { + integrity: sha512-sReKOYJIJf74dhJONhU4e0/shzi1trVbSWDOhKYE5XV2O+H7Sb2Dihwuc7xWxVl+DgFPyTqIN3zMfT9cq5iWDg==, + } loupe@3.2.1: - resolution: {integrity: sha512-CdzqowRJCeLU72bHvWqwRBBlLcMEtIvGrlvef74kMnV2AolS9Y8xUv1I0U/MNAWMhBlKIoyuEgoJ0t/bbwHbLQ==} + resolution: + { + integrity: sha512-CdzqowRJCeLU72bHvWqwRBBlLcMEtIvGrlvef74kMnV2AolS9Y8xUv1I0U/MNAWMhBlKIoyuEgoJ0t/bbwHbLQ==, + } magic-string@0.30.21: - resolution: {integrity: sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==} + resolution: + { + integrity: sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==, + } math-intrinsics@1.1.0: - resolution: {integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==, + } + engines: { node: '>= 0.4' } meow@12.1.1: - resolution: {integrity: sha512-BhXM0Au22RwUneMPwSCnyhTOizdWoIEPU9sp0Aqa1PnDMR5Wv2FGXYDjuzJEIX+Eo2Rb8xuYe5jrnm5QowQFkw==} - engines: {node: '>=16.10'} + resolution: + { + integrity: sha512-BhXM0Au22RwUneMPwSCnyhTOizdWoIEPU9sp0Aqa1PnDMR5Wv2FGXYDjuzJEIX+Eo2Rb8xuYe5jrnm5QowQFkw==, + } + engines: { node: '>=16.10' } mime-db@1.52.0: - resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} - engines: {node: '>= 0.6'} + resolution: + { + integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==, + } + engines: { node: '>= 0.6' } mime-types@2.1.35: - resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} - engines: {node: '>= 0.6'} + resolution: + { + integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==, + } + engines: { node: '>= 0.6' } minimist@1.2.8: - resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + resolution: + { + integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==, + } mlly@1.8.2: - resolution: {integrity: sha512-d+ObxMQFmbt10sretNDytwt85VrbkhhUA/JBGm1MPaWJ65Cl4wOgLaB1NYvJSZ0Ef03MMEU/0xpPMXUIQ29UfA==} + resolution: + { + integrity: sha512-d+ObxMQFmbt10sretNDytwt85VrbkhhUA/JBGm1MPaWJ65Cl4wOgLaB1NYvJSZ0Ef03MMEU/0xpPMXUIQ29UfA==, + } ms@2.1.3: - resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} + resolution: + { + integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==, + } mz@2.7.0: - resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} + resolution: + { + integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==, + } nanoid@3.3.15: - resolution: {integrity: sha512-y7Wygv/7mEOvxTuEQDB8StXdMRBWf1kR/tlhAzBRUFkB2jfcLOAxO/SHmOO2zgz1pVgK29/kyupn059/bCHdjA==} - engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} + resolution: + { + integrity: sha512-y7Wygv/7mEOvxTuEQDB8StXdMRBWf1kR/tlhAzBRUFkB2jfcLOAxO/SHmOO2zgz1pVgK29/kyupn059/bCHdjA==, + } + engines: { node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1 } hasBin: true node-fetch@2.7.0: - resolution: {integrity: sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==} - engines: {node: 4.x || >=6.0.0} + resolution: + { + integrity: sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==, + } + engines: { node: 4.x || >=6.0.0 } peerDependencies: encoding: ^0.1.0 peerDependenciesMeta: @@ -1191,15 +1981,24 @@ packages: optional: true node-gyp-build@4.8.4: - resolution: {integrity: sha512-LA4ZjwlnUblHVgq0oBF3Jl/6h/Nvs5fzBLwdEF4nuxnFdsfajde4WfxtJr3CaiH+F6ewcIB/q4jQ4UzPyid+CQ==} + resolution: + { + integrity: sha512-LA4ZjwlnUblHVgq0oBF3Jl/6h/Nvs5fzBLwdEF4nuxnFdsfajde4WfxtJr3CaiH+F6ewcIB/q4jQ4UzPyid+CQ==, + } hasBin: true object-assign@4.1.1: - resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} - engines: {node: '>=0.10.0'} + resolution: + { + integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==, + } + engines: { node: '>=0.10.0' } ox@0.14.29: - resolution: {integrity: sha512-M5j87Ec4V99MQdRct/g09eWXW60g6zhHTUs1lr4deUtrPDnezBdCJTgKd7pxqTpSZBFveV0ALi9jMMuT1qKyNg==} + resolution: + { + integrity: sha512-M5j87Ec4V99MQdRct/g09eWXW60g6zhHTUs1lr4deUtrPDnezBdCJTgKd7pxqTpSZBFveV0ALi9jMMuT1qKyNg==, + } peerDependencies: typescript: '>=5.4.0' peerDependenciesMeta: @@ -1207,53 +2006,92 @@ packages: optional: true p-limit@4.0.0: - resolution: {integrity: sha512-5b0R4txpzjPWVw/cXXUResoD4hb6U/x9BH08L7nw+GN1sezDzPdxeRvpc9c433fZhBan/wusjbCsqwqm4EIBIQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + resolution: + { + integrity: sha512-5b0R4txpzjPWVw/cXXUResoD4hb6U/x9BH08L7nw+GN1sezDzPdxeRvpc9c433fZhBan/wusjbCsqwqm4EIBIQ==, + } + engines: { node: ^12.20.0 || ^14.13.1 || >=16.0.0 } p-locate@6.0.0: - resolution: {integrity: sha512-wPrq66Llhl7/4AGC6I+cqxT07LhXvWL08LNXz1fENOw0Ap4sRZZ/gZpTTJ5jpurzzzfS2W/Ge9BY3LgLjCShcw==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + resolution: + { + integrity: sha512-wPrq66Llhl7/4AGC6I+cqxT07LhXvWL08LNXz1fENOw0Ap4sRZZ/gZpTTJ5jpurzzzfS2W/Ge9BY3LgLjCShcw==, + } + engines: { node: ^12.20.0 || ^14.13.1 || >=16.0.0 } parent-module@1.0.1: - resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} - engines: {node: '>=6'} + resolution: + { + integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==, + } + engines: { node: '>=6' } parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==, + } + engines: { node: '>=8' } path-exists@5.0.0: - resolution: {integrity: sha512-RjhtfwJOxzcFmNOi6ltcbcu4Iu+FL3zEj83dk4kAS+fVpTxXLO1b38RvJgT/0QwvV/L3aY9TAnyv0EOqW4GoMQ==} - engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + resolution: + { + integrity: sha512-RjhtfwJOxzcFmNOi6ltcbcu4Iu+FL3zEj83dk4kAS+fVpTxXLO1b38RvJgT/0QwvV/L3aY9TAnyv0EOqW4GoMQ==, + } + engines: { node: ^12.20.0 || ^14.13.1 || >=16.0.0 } pathe@2.0.3: - resolution: {integrity: sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==} + resolution: + { + integrity: sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==, + } pathval@2.0.1: - resolution: {integrity: sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ==} - engines: {node: '>= 14.16'} + resolution: + { + integrity: sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ==, + } + engines: { node: '>= 14.16' } picocolors@1.1.1: - resolution: {integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==} + resolution: + { + integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==, + } picomatch@4.0.4: - resolution: {integrity: sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==, + } + engines: { node: '>=12' } pirates@4.0.7: - resolution: {integrity: sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==} - engines: {node: '>= 6'} + resolution: + { + integrity: sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==, + } + engines: { node: '>= 6' } pkg-types@1.3.1: - resolution: {integrity: sha512-/Jm5M4RvtBFVkKWRu2BLUTNP8/M2a+UwuAX+ae4770q1qVGtfjG+WTCupoZixokjmHiry8uI+dlY8KXYV5HVVQ==} + resolution: + { + integrity: sha512-/Jm5M4RvtBFVkKWRu2BLUTNP8/M2a+UwuAX+ae4770q1qVGtfjG+WTCupoZixokjmHiry8uI+dlY8KXYV5HVVQ==, + } possible-typed-array-names@1.1.0: - resolution: {integrity: sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==, + } + engines: { node: '>= 0.4' } postcss-load-config@6.0.1: - resolution: {integrity: sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==} - engines: {node: '>= 18'} + resolution: + { + integrity: sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==, + } + engines: { node: '>= 18' } peerDependencies: jiti: '>=1.21.0' postcss: '>=8.0.9' @@ -1270,189 +2108,339 @@ packages: optional: true postcss@8.5.15: - resolution: {integrity: sha512-FfR8sjd4em2T6fb3I2MwAJU7HWVMr9zba+enmQeeWFfCbm+UOC/0X4DS8XtpUTMwWMGbjKYP7xjfNekzyGmB3A==} - engines: {node: ^10 || ^12 || >=14} + resolution: + { + integrity: sha512-FfR8sjd4em2T6fb3I2MwAJU7HWVMr9zba+enmQeeWFfCbm+UOC/0X4DS8XtpUTMwWMGbjKYP7xjfNekzyGmB3A==, + } + engines: { node: ^10 || ^12 || >=14 } prettier@3.8.4: - resolution: {integrity: sha512-N2MylSdi48+5N/6S5j+maeHbUSIzzZ5uOcX5Hm4QpV8Dkb1HFjfAKTKX6yNPJQD9AhcT3ifHNB66tWTTJDi11Q==} - engines: {node: '>=14'} + resolution: + { + integrity: sha512-N2MylSdi48+5N/6S5j+maeHbUSIzzZ5uOcX5Hm4QpV8Dkb1HFjfAKTKX6yNPJQD9AhcT3ifHNB66tWTTJDi11Q==, + } + engines: { node: '>=14' } hasBin: true proxy-from-env@2.1.0: - resolution: {integrity: sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA==} - engines: {node: '>=10'} + resolution: + { + integrity: sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA==, + } + engines: { node: '>=10' } pure-rand@8.4.0: - resolution: {integrity: sha512-IoM8YF/jY0hiugFo/wOWqfmarlE6J0wc6fDK1PhftMk7MGhVZl88sZimmqBBFomLOCSmcCCpsfj7wXASCpvK9A==} + resolution: + { + integrity: sha512-IoM8YF/jY0hiugFo/wOWqfmarlE6J0wc6fDK1PhftMk7MGhVZl88sZimmqBBFomLOCSmcCCpsfj7wXASCpvK9A==, + } randombytes@2.1.0: - resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} + resolution: + { + integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==, + } readdirp@4.1.2: - resolution: {integrity: sha512-GDhwkLfywWL2s6vEjyhri+eXmfH6j1L7JE27WhqLeYzoh/A3DBaYGEj2H/HFZCn/kMfim73FXxEJTw06WtxQwg==} - engines: {node: '>= 14.18.0'} + resolution: + { + integrity: sha512-GDhwkLfywWL2s6vEjyhri+eXmfH6j1L7JE27WhqLeYzoh/A3DBaYGEj2H/HFZCn/kMfim73FXxEJTw06WtxQwg==, + } + engines: { node: '>= 14.18.0' } require-addon@1.2.0: - resolution: {integrity: sha512-VNPDZlYgIYQwWp9jMTzljx+k0ZtatKlcvOhktZ/anNPI3dQ9NXk7cq2U4iJ1wd9IrytRnYhyEocFWbkdPb+MYA==} - engines: {bare: '>=1.10.0'} + resolution: + { + integrity: sha512-VNPDZlYgIYQwWp9jMTzljx+k0ZtatKlcvOhktZ/anNPI3dQ9NXk7cq2U4iJ1wd9IrytRnYhyEocFWbkdPb+MYA==, + } + engines: { bare: '>=1.10.0' } require-directory@2.1.1: - resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} - engines: {node: '>=0.10.0'} + resolution: + { + integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==, + } + engines: { node: '>=0.10.0' } require-from-string@2.0.2: - resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} - engines: {node: '>=0.10.0'} + resolution: + { + integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==, + } + engines: { node: '>=0.10.0' } resolve-from@4.0.0: - resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} - engines: {node: '>=4'} + resolution: + { + integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==, + } + engines: { node: '>=4' } resolve-from@5.0.0: - resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==, + } + engines: { node: '>=8' } rollup@4.62.2: - resolution: {integrity: sha512-RFnrW4lhXA3s3eqHDZvN654g8OTjzRfqpIRJYczCGB6HzphckVAi/Qh4tbPUbRuDi7s1Llv8g/NspLkttY3gTA==} - engines: {node: '>=18.0.0', npm: '>=8.0.0'} + resolution: + { + integrity: sha512-RFnrW4lhXA3s3eqHDZvN654g8OTjzRfqpIRJYczCGB6HzphckVAi/Qh4tbPUbRuDi7s1Llv8g/NspLkttY3gTA==, + } + engines: { node: '>=18.0.0', npm: '>=8.0.0' } hasBin: true rpc-websockets@9.3.9: - resolution: {integrity: sha512-2iQDaTB4g5fDB2ihrTFSJSibCEuxaRi1q7qTW7ZO9/M5/TC+ToHA4D9/ffNLEbAoHNNrcdeP05oATNk44SKZXA==} + resolution: + { + integrity: sha512-2iQDaTB4g5fDB2ihrTFSJSibCEuxaRi1q7qTW7ZO9/M5/TC+ToHA4D9/ffNLEbAoHNNrcdeP05oATNk44SKZXA==, + } safe-buffer@5.2.1: - resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} + resolution: + { + integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==, + } semver@7.8.5: - resolution: {integrity: sha512-Y7/KDsb8LjooZpwaqGyulO6DQlksgCncchHGk+sZIY4SBvUocMBEFH5Ur1fI4dV+Jvl0w6cjvucaIi40puRioA==} - engines: {node: '>=10'} + resolution: + { + integrity: sha512-Y7/KDsb8LjooZpwaqGyulO6DQlksgCncchHGk+sZIY4SBvUocMBEFH5Ur1fI4dV+Jvl0w6cjvucaIi40puRioA==, + } + engines: { node: '>=10' } hasBin: true set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==, + } + engines: { node: '>= 0.4' } sha.js@2.4.12: - resolution: {integrity: sha512-8LzC5+bvI45BjpfXU8V5fdU2mfeKiQe1D1gIMn7XUlF3OTUrpdJpPPH4EMAnF0DsHHdSZqCdSss5qCmJKuiO3w==} - engines: {node: '>= 0.10'} + resolution: + { + integrity: sha512-8LzC5+bvI45BjpfXU8V5fdU2mfeKiQe1D1gIMn7XUlF3OTUrpdJpPPH4EMAnF0DsHHdSZqCdSss5qCmJKuiO3w==, + } + engines: { node: '>= 0.10' } hasBin: true siginfo@2.0.0: - resolution: {integrity: sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==} + resolution: + { + integrity: sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==, + } sodium-native@4.3.3: - resolution: {integrity: sha512-OnxSlN3uyY8D0EsLHpmm2HOFmKddQVvEMmsakCrXUzSd8kjjbzL413t4ZNF3n0UxSwNgwTyUvkmZHTfuCeiYSw==} + resolution: + { + integrity: sha512-OnxSlN3uyY8D0EsLHpmm2HOFmKddQVvEMmsakCrXUzSd8kjjbzL413t4ZNF3n0UxSwNgwTyUvkmZHTfuCeiYSw==, + } source-map-js@1.2.1: - resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} - engines: {node: '>=0.10.0'} + resolution: + { + integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==, + } + engines: { node: '>=0.10.0' } source-map@0.7.6: - resolution: {integrity: sha512-i5uvt8C3ikiWeNZSVZNWcfZPItFQOsYTUAOkcUPGd8DqDy1uOUikjt5dG+uRlwyvR108Fb9DOd4GvXfT0N2/uQ==} - engines: {node: '>= 12'} + resolution: + { + integrity: sha512-i5uvt8C3ikiWeNZSVZNWcfZPItFQOsYTUAOkcUPGd8DqDy1uOUikjt5dG+uRlwyvR108Fb9DOd4GvXfT0N2/uQ==, + } + engines: { node: '>= 12' } split2@4.2.0: - resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} - engines: {node: '>= 10.x'} + resolution: + { + integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==, + } + engines: { node: '>= 10.x' } stackback@0.0.2: - resolution: {integrity: sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==} + resolution: + { + integrity: sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==, + } std-env@3.10.0: - resolution: {integrity: sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==} + resolution: + { + integrity: sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==, + } stream-chain@2.2.5: - resolution: {integrity: sha512-1TJmBx6aSWqZ4tx7aTpBDXK0/e2hhcNSTV8+CbFJtDjbb+I1mZ8lHit0Grw9GRT+6JbIrrDd8esncgBi8aBXGA==} + resolution: + { + integrity: sha512-1TJmBx6aSWqZ4tx7aTpBDXK0/e2hhcNSTV8+CbFJtDjbb+I1mZ8lHit0Grw9GRT+6JbIrrDd8esncgBi8aBXGA==, + } stream-json@1.9.1: - resolution: {integrity: sha512-uWkjJ+2Nt/LO9Z/JyKZbMusL8Dkh97uUBTv3AJQ74y07lVahLY4eEFsPsE97pxYBwr8nnjMAIch5eqI0gPShyw==} + resolution: + { + integrity: sha512-uWkjJ+2Nt/LO9Z/JyKZbMusL8Dkh97uUBTv3AJQ74y07lVahLY4eEFsPsE97pxYBwr8nnjMAIch5eqI0gPShyw==, + } string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==, + } + engines: { node: '>=8' } strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==, + } + engines: { node: '>=8' } strip-literal@3.1.0: - resolution: {integrity: sha512-8r3mkIM/2+PpjHoOtiAW8Rg3jJLHaV7xPwG+YRGrv6FP0wwk/toTpATxWYOW0BKdWwl82VT2tFYi5DlROa0Mxg==} + resolution: + { + integrity: sha512-8r3mkIM/2+PpjHoOtiAW8Rg3jJLHaV7xPwG+YRGrv6FP0wwk/toTpATxWYOW0BKdWwl82VT2tFYi5DlROa0Mxg==, + } sucrase@3.35.1: - resolution: {integrity: sha512-DhuTmvZWux4H1UOnWMB3sk0sbaCVOoQZjv8u1rDoTV0HTdGem9hkAZtl4JZy8P2z4Bg0nT+YMeOFyVr4zcG5Tw==} - engines: {node: '>=16 || 14 >=14.17'} + resolution: + { + integrity: sha512-DhuTmvZWux4H1UOnWMB3sk0sbaCVOoQZjv8u1rDoTV0HTdGem9hkAZtl4JZy8P2z4Bg0nT+YMeOFyVr4zcG5Tw==, + } + engines: { node: '>=16 || 14 >=14.17' } hasBin: true superstruct@2.0.2: - resolution: {integrity: sha512-uV+TFRZdXsqXTL2pRvujROjdZQ4RAlBUS5BTh9IGm+jTqQntYThciG/qu57Gs69yjnVUSqdxF9YLmSnpupBW9A==} - engines: {node: '>=14.0.0'} + resolution: + { + integrity: sha512-uV+TFRZdXsqXTL2pRvujROjdZQ4RAlBUS5BTh9IGm+jTqQntYThciG/qu57Gs69yjnVUSqdxF9YLmSnpupBW9A==, + } + engines: { node: '>=14.0.0' } text-encoding-utf-8@1.0.2: - resolution: {integrity: sha512-8bw4MY9WjdsD2aMtO0OzOCY3pXGYNx2d2FfHRVUKkiCPDWjKuOlhLVASS+pD7VkLTVjW268LYJHwsnPFlBpbAg==} + resolution: + { + integrity: sha512-8bw4MY9WjdsD2aMtO0OzOCY3pXGYNx2d2FfHRVUKkiCPDWjKuOlhLVASS+pD7VkLTVjW268LYJHwsnPFlBpbAg==, + } text-extensions@2.4.0: - resolution: {integrity: sha512-te/NtwBwfiNRLf9Ijqx3T0nlqZiQ2XrrtBvu+cLL8ZRrGkO0NHTug8MYFKyoSrv/sHTaSKfilUkizV6XhxMJ3g==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-te/NtwBwfiNRLf9Ijqx3T0nlqZiQ2XrrtBvu+cLL8ZRrGkO0NHTug8MYFKyoSrv/sHTaSKfilUkizV6XhxMJ3g==, + } + engines: { node: '>=8' } thenify-all@1.6.0: - resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} - engines: {node: '>=0.8'} + resolution: + { + integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==, + } + engines: { node: '>=0.8' } thenify@3.3.1: - resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} + resolution: + { + integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==, + } through@2.3.8: - resolution: {integrity: sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==} + resolution: + { + integrity: sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==, + } tinybench@2.9.0: - resolution: {integrity: sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==} + resolution: + { + integrity: sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==, + } tinyexec@0.3.2: - resolution: {integrity: sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==} + resolution: + { + integrity: sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==, + } tinyexec@1.2.4: - resolution: {integrity: sha512-SHf/r48b7vOrjve9PxJo3MN5v5yuyjHvdUcrQffT3WXMUfnGmHDVbC4k3sHJaJTgZCwpUplIaAo5ANtMyp3YHg==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-SHf/r48b7vOrjve9PxJo3MN5v5yuyjHvdUcrQffT3WXMUfnGmHDVbC4k3sHJaJTgZCwpUplIaAo5ANtMyp3YHg==, + } + engines: { node: '>=18' } tinyglobby@0.2.17: - resolution: {integrity: sha512-wXR/dYpcqKmfWpEdZjiKJOwCNFndD0DMnrW/cYjVGttEkBfVgcLFHoNrlj47mjOVic9yyNu65alsgF4NQyTa2g==} - engines: {node: '>=12.0.0'} + resolution: + { + integrity: sha512-wXR/dYpcqKmfWpEdZjiKJOwCNFndD0DMnrW/cYjVGttEkBfVgcLFHoNrlj47mjOVic9yyNu65alsgF4NQyTa2g==, + } + engines: { node: '>=12.0.0' } tinypool@1.1.1: - resolution: {integrity: sha512-Zba82s87IFq9A9XmjiX5uZA/ARWDrB03OHlq+Vw1fSdt0I+4/Kutwy8BP4Y/y/aORMo61FQ0vIb5j44vSo5Pkg==} - engines: {node: ^18.0.0 || >=20.0.0} + resolution: + { + integrity: sha512-Zba82s87IFq9A9XmjiX5uZA/ARWDrB03OHlq+Vw1fSdt0I+4/Kutwy8BP4Y/y/aORMo61FQ0vIb5j44vSo5Pkg==, + } + engines: { node: ^18.0.0 || >=20.0.0 } tinyrainbow@2.0.0: - resolution: {integrity: sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==} - engines: {node: '>=14.0.0'} + resolution: + { + integrity: sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==, + } + engines: { node: '>=14.0.0' } tinyspy@4.0.4: - resolution: {integrity: sha512-azl+t0z7pw/z958Gy9svOTuzqIk6xq+NSheJzn5MMWtWTFywIacg2wUlzKFGtt3cthx0r2SxMK0yzJOR0IES7Q==} - engines: {node: '>=14.0.0'} + resolution: + { + integrity: sha512-azl+t0z7pw/z958Gy9svOTuzqIk6xq+NSheJzn5MMWtWTFywIacg2wUlzKFGtt3cthx0r2SxMK0yzJOR0IES7Q==, + } + engines: { node: '>=14.0.0' } to-buffer@1.2.2: - resolution: {integrity: sha512-db0E3UJjcFhpDhAF4tLo03oli3pwl3dbnzXOUIlRKrp+ldk/VUxzpWYZENsw2SZiuBjHAk7DfB0VU7NKdpb6sw==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-db0E3UJjcFhpDhAF4tLo03oli3pwl3dbnzXOUIlRKrp+ldk/VUxzpWYZENsw2SZiuBjHAk7DfB0VU7NKdpb6sw==, + } + engines: { node: '>= 0.4' } toml@3.0.0: - resolution: {integrity: sha512-y/mWCZinnvxjTKYhJ+pYxwD0mRLVvOtdS2Awbgxln6iEnt4rk0yBxeSBHkGJcPucRiG0e55mwWp+g/05rsrd6w==} + resolution: + { + integrity: sha512-y/mWCZinnvxjTKYhJ+pYxwD0mRLVvOtdS2Awbgxln6iEnt4rk0yBxeSBHkGJcPucRiG0e55mwWp+g/05rsrd6w==, + } tr46@0.0.3: - resolution: {integrity: sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==} + resolution: + { + integrity: sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==, + } tree-kill@1.2.2: - resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} + resolution: + { + integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==, + } hasBin: true ts-interface-checker@0.1.13: - resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} + resolution: + { + integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==, + } tslib@2.8.1: - resolution: {integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==} + resolution: + { + integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==, + } tsup@8.5.1: - resolution: {integrity: sha512-xtgkqwdhpKWr3tKPmCkvYmS9xnQK3m3XgxZHwSUjvfTjp7YfXe5tT3GgWi0F2N+ZSMsOeWeZFh7ZZFg5iPhing==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-xtgkqwdhpKWr3tKPmCkvYmS9xnQK3m3XgxZHwSUjvfTjp7YfXe5tT3GgWi0F2N+ZSMsOeWeZFh7ZZFg5iPhing==, + } + engines: { node: '>=18' } hasBin: true peerDependencies: '@microsoft/api-extractor': ^7.36.0 @@ -1470,45 +2458,78 @@ packages: optional: true tweetnacl@1.0.3: - resolution: {integrity: sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==} + resolution: + { + integrity: sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==, + } typed-array-buffer@1.0.3: - resolution: {integrity: sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==, + } + engines: { node: '>= 0.4' } typescript@5.9.3: - resolution: {integrity: sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==} - engines: {node: '>=14.17'} + resolution: + { + integrity: sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==, + } + engines: { node: '>=14.17' } hasBin: true ufo@1.6.4: - resolution: {integrity: sha512-JFNbkD1Svwe0KvGi8GOeLcP4kAWQ609twvCdcHxq1oSL8svv39ZuSvajcD8B+5D0eL4+s1Is2D/O6KN3qcTeRA==} + resolution: + { + integrity: sha512-JFNbkD1Svwe0KvGi8GOeLcP4kAWQ609twvCdcHxq1oSL8svv39ZuSvajcD8B+5D0eL4+s1Is2D/O6KN3qcTeRA==, + } undici-types@8.3.0: - resolution: {integrity: sha512-j375ScV60dom+YkPFIfTLcOiPxkN/buHz5GobjLhixFuANaNs3C9l4GmrWqejgXWJ7BbJcFYpTEUkS1Ge8bpZQ==} + resolution: + { + integrity: sha512-j375ScV60dom+YkPFIfTLcOiPxkN/buHz5GobjLhixFuANaNs3C9l4GmrWqejgXWJ7BbJcFYpTEUkS1Ge8bpZQ==, + } unicorn-magic@0.1.0: - resolution: {integrity: sha512-lRfVq8fE8gz6QMBuDM6a+LO3IAzTi05H6gCVaUpir2E1Rwpo4ZUog45KpNXKC/Mn3Yb9UDuHumeFTo9iV/D9FQ==} - engines: {node: '>=18'} + resolution: + { + integrity: sha512-lRfVq8fE8gz6QMBuDM6a+LO3IAzTi05H6gCVaUpir2E1Rwpo4ZUog45KpNXKC/Mn3Yb9UDuHumeFTo9iV/D9FQ==, + } + engines: { node: '>=18' } urijs@1.19.11: - resolution: {integrity: sha512-HXgFDgDommxn5/bIv0cnQZsPhHDA90NPHD6+c/v21U5+Sx5hoP8+dP9IZXBU1gIfvdRfhG8cel9QNPeionfcCQ==} + resolution: + { + integrity: sha512-HXgFDgDommxn5/bIv0cnQZsPhHDA90NPHD6+c/v21U5+Sx5hoP8+dP9IZXBU1gIfvdRfhG8cel9QNPeionfcCQ==, + } utf-8-validate@6.0.6: - resolution: {integrity: sha512-q3l3P9UtEEiAHcsgsqTgf9PPjctrDWoIXW3NpOHFdRDbLvu4DLIcxHangJ4RLrWkBcKjmcs/6NkerI8T/rE4LA==} - engines: {node: '>=6.14.2'} + resolution: + { + integrity: sha512-q3l3P9UtEEiAHcsgsqTgf9PPjctrDWoIXW3NpOHFdRDbLvu4DLIcxHangJ4RLrWkBcKjmcs/6NkerI8T/rE4LA==, + } + engines: { node: '>=6.14.2' } uuid@14.0.1: - resolution: {integrity: sha512-6ZxzVpzDXDa3bJWaHilVayA+BH/1zmxCJoVgvmqJnid/gPoKHxUrS/aC/T6LGQtNHT+XHG9fXPJB4d+IrU30Ew==} + resolution: + { + integrity: sha512-6ZxzVpzDXDa3bJWaHilVayA+BH/1zmxCJoVgvmqJnid/gPoKHxUrS/aC/T6LGQtNHT+XHG9fXPJB4d+IrU30Ew==, + } hasBin: true uuid@8.3.2: - resolution: {integrity: sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==} + resolution: + { + integrity: sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==, + } deprecated: uuid@10 and below is no longer supported. For ESM codebases, update to uuid@latest. For CommonJS codebases, use uuid@11 (but be aware this version will likely be deprecated in 2028). hasBin: true viem@2.53.1: - resolution: {integrity: sha512-FhfJ/SW73CVosiyVLmIMVgKDRKYV1AGCLzZoHYvmNayyVff63Qi1ocPCk59LqC/cNw244RbBJjHnmxqXkE7NpA==} + resolution: + { + integrity: sha512-FhfJ/SW73CVosiyVLmIMVgKDRKYV1AGCLzZoHYvmNayyVff63Qi1ocPCk59LqC/cNw244RbBJjHnmxqXkE7NpA==, + } peerDependencies: typescript: '>=5.0.4' peerDependenciesMeta: @@ -1516,13 +2537,19 @@ packages: optional: true vite-node@3.2.4: - resolution: {integrity: sha512-EbKSKh+bh1E1IFxeO0pg1n4dvoOTt0UDiXMd/qn++r98+jPO1xtJilvXldeuQ8giIB5IkpjCgMleHMNEsGH6pg==} - engines: {node: ^18.0.0 || ^20.0.0 || >=22.0.0} + resolution: + { + integrity: sha512-EbKSKh+bh1E1IFxeO0pg1n4dvoOTt0UDiXMd/qn++r98+jPO1xtJilvXldeuQ8giIB5IkpjCgMleHMNEsGH6pg==, + } + engines: { node: ^18.0.0 || ^20.0.0 || >=22.0.0 } hasBin: true vite@7.3.5: - resolution: {integrity: sha512-KuOaNhcnGFN2zIPGA7wRmzF+lJA1sea7rHq17aiJ++9lzY1WWG6Jpwqwe1KNbRVPIqHmr8GLYx7jbrQcN/7/ww==} - engines: {node: ^20.19.0 || >=22.12.0} + resolution: + { + integrity: sha512-KuOaNhcnGFN2zIPGA7wRmzF+lJA1sea7rHq17aiJ++9lzY1WWG6Jpwqwe1KNbRVPIqHmr8GLYx7jbrQcN/7/ww==, + } + engines: { node: ^20.19.0 || >=22.12.0 } hasBin: true peerDependencies: '@types/node': ^20.19.0 || >=22.12.0 @@ -1561,8 +2588,11 @@ packages: optional: true vitest@3.2.6: - resolution: {integrity: sha512-xejya+bT/j/+R/AGa1XOfRxLmNUlLtlwjRsFUILF+xHfzElmGcmFydy2gqqIrd62ptIEfwVMofd19uNWD9L7Nw==} - engines: {node: ^18.0.0 || ^20.0.0 || >=22.0.0} + resolution: + { + integrity: sha512-xejya+bT/j/+R/AGa1XOfRxLmNUlLtlwjRsFUILF+xHfzElmGcmFydy2gqqIrd62ptIEfwVMofd19uNWD9L7Nw==, + } + engines: { node: ^18.0.0 || ^20.0.0 || >=22.0.0 } hasBin: true peerDependencies: '@edge-runtime/vm': '*' @@ -1589,27 +2619,45 @@ packages: optional: true webidl-conversions@3.0.1: - resolution: {integrity: sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==} + resolution: + { + integrity: sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==, + } whatwg-url@5.0.0: - resolution: {integrity: sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==} + resolution: + { + integrity: sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==, + } which-typed-array@1.1.22: - resolution: {integrity: sha512-fvO4ExWMFsqyhG3AiPAObMuY1lxaqgYcxbc49CNdWDDECOJNgQyvsOWVwbZc+qf3rzRtxojBK+CMEv0Ld5CYpw==} - engines: {node: '>= 0.4'} + resolution: + { + integrity: sha512-fvO4ExWMFsqyhG3AiPAObMuY1lxaqgYcxbc49CNdWDDECOJNgQyvsOWVwbZc+qf3rzRtxojBK+CMEv0Ld5CYpw==, + } + engines: { node: '>= 0.4' } why-is-node-running@2.3.0: - resolution: {integrity: sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==} - engines: {node: '>=8'} + resolution: + { + integrity: sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==, + } + engines: { node: '>=8' } hasBin: true wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} + resolution: + { + integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==, + } + engines: { node: '>=10' } ws@7.5.11: - resolution: {integrity: sha512-zS54Oen9bITtp7kp2XM3AydrCIq1D+HwJOuH+c+e4LfpL/lotP5osijd+UoMnxwAam1GN8R4KtLAyIrIcBNpiA==} - engines: {node: '>=8.3.0'} + resolution: + { + integrity: sha512-zS54Oen9bITtp7kp2XM3AydrCIq1D+HwJOuH+c+e4LfpL/lotP5osijd+UoMnxwAam1GN8R4KtLAyIrIcBNpiA==, + } + engines: { node: '>=8.3.0' } peerDependencies: bufferutil: ^4.0.1 utf-8-validate: ^5.0.2 @@ -1620,8 +2668,11 @@ packages: optional: true ws@8.20.1: - resolution: {integrity: sha512-It4dO0K5v//JtTXuPkfEOaI3uUN87iYPnqo/ZzqCoG3g8uhA66QUMs/SrM0YK7/NAu+r4LMh/9dq2A7k+rHs+w==} - engines: {node: '>=10.0.0'} + resolution: + { + integrity: sha512-It4dO0K5v//JtTXuPkfEOaI3uUN87iYPnqo/ZzqCoG3g8uhA66QUMs/SrM0YK7/NAu+r4LMh/9dq2A7k+rHs+w==, + } + engines: { node: '>=10.0.0' } peerDependencies: bufferutil: ^4.0.1 utf-8-validate: '>=5.0.2' @@ -1632,8 +2683,11 @@ packages: optional: true ws@8.21.0: - resolution: {integrity: sha512-Vsp28b7DRcimFQvrqu2Wek3z1iYxDCWqHYB8Qsnk/S4RfaCQzPGPyBNuVjJV3cd6UiKtUtp6sNM77gWvzcCH+g==} - engines: {node: '>=10.0.0'} + resolution: + { + integrity: sha512-Vsp28b7DRcimFQvrqu2Wek3z1iYxDCWqHYB8Qsnk/S4RfaCQzPGPyBNuVjJV3cd6UiKtUtp6sNM77gWvzcCH+g==, + } + engines: { node: '>=10.0.0' } peerDependencies: bufferutil: ^4.0.1 utf-8-validate: '>=5.0.2' @@ -1644,23 +2698,34 @@ packages: optional: true y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} + resolution: + { + integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==, + } + engines: { node: '>=10' } yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==, + } + engines: { node: '>=12' } yargs@17.7.3: - resolution: {integrity: sha512-GZtjxm/J/4TSxuL3FNYjCmLktBTnIw/rVmKSIyKeYAZpmJB2ig9VauCC5xsa82GNKVKDAqpOn3KVzNt0zmrU0g==} - engines: {node: '>=12'} + resolution: + { + integrity: sha512-GZtjxm/J/4TSxuL3FNYjCmLktBTnIw/rVmKSIyKeYAZpmJB2ig9VauCC5xsa82GNKVKDAqpOn3KVzNt0zmrU0g==, + } + engines: { node: '>=12' } yocto-queue@1.2.2: - resolution: {integrity: sha512-4LCcse/U2MHZ63HAJVE+v71o7yOdIe4cZ70Wpf8D/IyjDKYQLV5GD46B+hSTjJsvV5PztjvHoU580EftxjDZFQ==} - engines: {node: '>=12.20'} + resolution: + { + integrity: sha512-4LCcse/U2MHZ63HAJVE+v71o7yOdIe4cZ70Wpf8D/IyjDKYQLV5GD46B+hSTjJsvV5PztjvHoU580EftxjDZFQ==, + } + engines: { node: '>=12.20' } snapshots: - '@adraffy/ens-normalize@1.11.1': {} '@babel/code-frame@7.29.7': diff --git a/pnpm-workspace.yaml b/pnpm-workspace.yaml index 789f832..0835062 100644 --- a/pnpm-workspace.yaml +++ b/pnpm-workspace.yaml @@ -1,4 +1,4 @@ allowBuilds: bufferutil: true esbuild: true - utf-8-validate: true \ No newline at end of file + utf-8-validate: true diff --git a/src/chains/stellar/announcements.ts b/src/chains/stellar/announcements.ts index eb6b9d8..0d68325 100644 --- a/src/chains/stellar/announcements.ts +++ b/src/chains/stellar/announcements.ts @@ -60,36 +60,7 @@ function getDefaultCache(): AnnouncementCache { return _defaultCache; } -export interface FetchAnnouncementsOptions { - /** Override the Soroban RPC URL. */ - sorobanUrl?: string; - /** - * Skip the cache and always fetch from RPC. - * Use after sending a payment to ensure the new announcement is visible. - */ - bypassCache?: boolean; - /** Provide a custom cache implementation (e.g., for testing). */ - cache?: AnnouncementCache; -} - /** - * Fetches stealth address announcements from the Soroban RPC for the given - * Stellar network, using an announcement cache to avoid redundant RPC traffic. - * - * On the first call the full available window is fetched. On subsequent calls - * only the delta since the last seen ledger is fetched, and results are merged - * with cached data before being returned. - * - * @param chain The chain identifier (default: `"stellar"`). - * @param options Fetch options including cache bypass and custom cache. - * @returns Array of all known announcements (cached + fresh). - */ -export async function fetchAnnouncements( - chain: string = 'stellar', - options: FetchAnnouncementsOptions = {}, -): Promise { - const { sorobanUrl, bypassCache = false, cache = getDefaultCache() } = options; - * Fetches Stellar stealth announcements from the configured Soroban RPC. * * Use this before {@link scanAnnouncements} when a recipient wants to discover @@ -302,51 +273,41 @@ export async function* fetchAnnouncementsStream( } } - // ------------------------------------------------------------------ - // Cache integration - // ------------------------------------------------------------------ - let fetchFromLedger = windowStart; - let resumeCursor: string | undefined; - let cached: Announcement[] = []; - - if (!bypassCache) { - const lastSeen = await cache.getLastSeen(network); - if (lastSeen) { - // Only fetch the delta. - fetchFromLedger = lastSeen.ledger + 1; - resumeCursor = lastSeen.cursor; - const hit = await cache.get(network, windowStart, lastSeen.ledger); - cached = hit ?? []; - let startLedger = 1; + // Page through events, yielding each announcement as it arrives. + let cursor: string | undefined; + let hasMore = true; - if (probeData.error?.message) { - const range = parseLedgerRange(probeData.error.message); - if (range) { - startLedger = Math.max(range.oldest, range.latest - 5000); - } else { - return; + while (hasMore) { + const params: Record = { + filters: [{ type: 'contract', contractIds: [announcerContract] }], + pagination: cursor ? { limit: 1000, cursor } : { limit: 1000 }, + }; + + if (!cursor) { + params.startLedger = windowStart; } - } - // Nothing new to fetch. - if (fetchFromLedger > windowEnd) return cached; + const res = await fetch(url, { + method: 'POST', + headers: { 'Content-Type': 'application/json' }, + body: JSON.stringify({ jsonrpc: '2.0', id: 2, method: 'getEvents', params }), + }); - // ------------------------------------------------------------------ - // Fetch delta from RPC - // ------------------------------------------------------------------ - const { announcements: fresh, lastLedger, lastCursor } = await fetchRange( - url, - announcerContract, - fetchFromLedger, - resumeCursor, - ); - - if (!bypassCache && fresh.length > 0 && lastLedger !== undefined && lastCursor) { - await cache.put(network, fresh); - await cache.setLastSeen(network, lastLedger, lastCursor); - } + const data = await res.json(); + const events: Record[] = data.result?.events ?? []; + + for (const event of events) { + const ann = parseAnnouncementEvent(event); + if (ann) yield ann; + } - return [...cached, ...fresh]; + if (events.length < 1000) { + hasMore = false; + } else { + cursor = data.result?.cursor as string | undefined; + if (!cursor) hasMore = false; + } + } } // --------------------------------------------------------------------------- @@ -398,7 +359,6 @@ async function fetchRange( lastLedger = ann.ledger; } } - if (ann) yield ann; } if (events.length < 1000) { @@ -409,9 +369,9 @@ async function fetchRange( else lastCursor = cursor; } } -} return { announcements: all, lastLedger, lastCursor }; +} async function getSorobanLedgerWindow( sorobanUrl: string, announcerContract: string, @@ -523,31 +483,8 @@ function parseLedgerRange(message: string): { oldest: number; latest: number } | export function parseAnnouncementEvent(event: Record): Announcement | null { try { const topics = event.topic as string[] | undefined; - if (!topics || topics.length < 3) return null; - - const schemeIdScVal = xdr.ScVal.fromXDR(topics[1], 'base64'); - const stealthScVal = xdr.ScVal.fromXDR(topics[2], 'base64'); - const stealthAddress = Address.fromScAddress(stealthScVal.address()).toString(); - - const valueScVal = xdr.ScVal.fromXDR(event.value as string, 'base64'); - const valueVec = valueScVal.vec(); - if (!valueVec || valueVec.length < 3) return null; - - const caller = Address.fromScAddress(valueVec[0].address()).toString(); - const ephPubKeyBytes = valueVec[1].bytes(); - const viewTagBytes = valueVec[2].bytes(); - if (!ephPubKeyBytes || !viewTagBytes) return null; - - const ledger = typeof event.ledger === 'number' ? event.ledger : typeof event.ledger === 'string' ? parseInt(event.ledger, 10) : undefined; - - return { - schemeId: schemeIdScVal.u32(), - stealthAddress, - caller, - ephemeralPubKey: bytesToHex(new Uint8Array(ephPubKeyBytes)), - metadata: bytesToHex(new Uint8Array(viewTagBytes)), - ledger, - }; + if (!topics) return null; + if (topics.length === 3) { return parseV1AnnouncementEvent(event, topics); } diff --git a/src/chains/stellar/cache.ts b/src/chains/stellar/cache.ts index ea0540e..283b65c 100644 --- a/src/chains/stellar/cache.ts +++ b/src/chains/stellar/cache.ts @@ -58,7 +58,11 @@ export class MemoryCache implements AnnouncementCache { this.maxBytes = maxBytes; } - async get(network: Network, fromLedger: number, toLedger: number): Promise { + async get( + network: Network, + fromLedger: number, + toLedger: number, + ): Promise { const ns = this.store.get(network); if (!ns || ns.size === 0) return null; const results = [...ns.values()].filter( @@ -166,7 +170,11 @@ export class IndexedDBCache implements AnnouncementCache { return this.dbPromise; } - async get(network: Network, fromLedger: number, toLedger: number): Promise { + async get( + network: Network, + fromLedger: number, + toLedger: number, + ): Promise { const db = await this.openDB(); return new Promise((resolve, reject) => { const tx = db.transaction(STORE_ANN, 'readonly'); diff --git a/src/chains/stellar/federation.ts b/src/chains/stellar/federation.ts new file mode 100644 index 0000000..d717c8a --- /dev/null +++ b/src/chains/stellar/federation.ts @@ -0,0 +1,298 @@ +/** + * Stellar Federation Address Resolution (SEP-0002) + * + * Resolves human-readable federation addresses (`name*domain.com`) to Stellar + * account IDs and optional memos via the two-step TOML → federation-server + * lookup defined in SEP-0002. + * + * Resolution results are cached in-memory (default TTL: 1 hour). Pass a custom + * `cacheTtl` or `fetch` to the {@link FederationResolver} constructor when you + * need a different policy or want to inject a mock in tests. + * + * @see https://github.com/stellar/stellar-protocol/blob/master/ecosystem/sep-0002.md + */ + +// --------------------------------------------------------------------------- +// Types +// --------------------------------------------------------------------------- + +export type FederationMemoType = 'text' | 'id' | 'hash'; + +/** Resolved Stellar federation address. */ +export interface FederationResult { + /** The Stellar account ID (`G…`). */ + accountId: string; + /** + * Optional memo to attach to payments sent to this address. + * Only present when the federation server returned both `memo_type` and `memo`. + */ + memo?: { type: FederationMemoType; value: string }; +} + +/** Discriminated error codes for federation resolution failures. */ +export type FederationErrorCode = + | 'INVALID_ADDRESS' + | 'TOML_FETCH_FAILED' + | 'TOML_PARSE_FAILED' + | 'NO_FEDERATION_SERVER' + | 'FEDERATION_FETCH_FAILED' + | 'NOT_FOUND' + | 'INVALID_RESPONSE'; + +export class FederationError extends Error { + readonly code: FederationErrorCode; + + constructor(code: FederationErrorCode, message: string) { + super(message); + this.name = 'FederationError'; + this.code = code; + } +} + +/** Minimal fetch signature accepted by {@link FederationResolver}. */ +export type FederationFetchFn = (url: string) => Promise; + +export interface FederationResolverOptions { + /** + * Cache TTL in milliseconds. Defaults to `3_600_000` (1 hour). + * Set to `0` to disable caching. + */ + cacheTtl?: number; + /** + * Custom fetch function. Defaults to `globalThis.fetch`. + * Inject a mock here in unit tests. + */ + fetch?: FederationFetchFn; +} + +// --------------------------------------------------------------------------- +// Implementation +// --------------------------------------------------------------------------- + +interface CacheEntry { + result: FederationResult; + expiresAt: number; +} + +const DEFAULT_TTL_MS = 3_600_000; +const MEMO_TYPES = new Set(['text', 'id', 'hash']); +// name must not contain `*` or whitespace; domain must contain a dot +const FED_ADDRESS_RE = /^[^*\s]+\*[^*\s]+\.[^*\s]+$/; + +/** + * Stateful resolver with its own in-memory cache. Prefer this class when you + * need isolated cache scopes, non-default TTLs, or a mock fetch in tests. + * + * For most callers, use the module-level {@link resolveStellarFederation} instead. + * + * **Caching:** Resolved results are stored until `expiresAt = Date.now() + cacheTtl`. + * Use {@link invalidate} to drop a single entry or {@link clear} to flush all entries. + */ +export class FederationResolver { + private readonly cache = new Map(); + private readonly ttl: number; + private readonly fetchFn: FederationFetchFn; + + constructor(options: FederationResolverOptions = {}) { + this.ttl = options.cacheTtl ?? DEFAULT_TTL_MS; + this.fetchFn = options.fetch ?? ((url) => globalThis.fetch(url)); + } + + /** + * Resolves `address` (`name*domain.com`) to a {@link FederationResult}. + * + * Flow: + * 1. Validate address format. + * 2. Return cached result if still fresh. + * 3. Fetch `https:///.well-known/stellar.toml` and extract `FEDERATION_SERVER`. + * 4. Query `?q=
&type=name`. + * 5. Parse JSON, cache, and return. + * + * @throws {@link FederationError} on any failure (see `code` for the reason). + */ + async resolve(address: string): Promise { + const normalised = address.trim().toLowerCase(); + + if (!FED_ADDRESS_RE.test(normalised)) { + throw new FederationError('INVALID_ADDRESS', `Not a valid federation address: "${address}"`); + } + + if (this.ttl > 0) { + const cached = this.cache.get(normalised); + if (cached && Date.now() < cached.expiresAt) return cached.result; + } + + const domain = normalised.slice(normalised.indexOf('*') + 1); + const federationUrl = await this.resolveFederationServer(domain); + // Use the normalised (lowercase) address for the query so cache keying and + // federation server lookups are consistent regardless of caller casing. + const result = await this.queryFederationServer(federationUrl, normalised); + + if (this.ttl > 0) { + this.cache.set(normalised, { result, expiresAt: Date.now() + this.ttl }); + } + return result; + } + + /** Drops the cached entry for `address` so the next call re-fetches. */ + invalidate(address: string): void { + this.cache.delete(address.trim().toLowerCase()); + } + + /** Clears all cached entries. */ + clear(): void { + this.cache.clear(); + } + + // ------------------------------------------------------------------------- + + private async resolveFederationServer(domain: string): Promise { + const tomlUrl = `https://${domain}/.well-known/stellar.toml`; + let tomlText: string; + try { + const resp = await this.fetchFn(tomlUrl); + if (!resp.ok) { + throw new FederationError( + 'TOML_FETCH_FAILED', + `stellar.toml returned HTTP ${resp.status} for domain "${domain}"`, + ); + } + tomlText = await resp.text(); + } catch (err) { + if (err instanceof FederationError) throw err; + throw new FederationError( + 'TOML_FETCH_FAILED', + `Failed to fetch stellar.toml from "${domain}": ${(err as Error).message}`, + ); + } + return parseFederationServerUrl(tomlText, domain); + } + + private async queryFederationServer( + federationUrl: string, + address: string, + ): Promise { + // Append query params whether or not the URL already has a query string. + // encodeURIComponent leaves `*` unencoded; force-encode it so the `name*domain` + // separator is unambiguous in the query string. + const encoded = encodeURIComponent(address).replace(/\*/g, '%2A'); + const sep = federationUrl.includes('?') ? '&' : '?'; + const queryUrl = `${federationUrl}${sep}q=${encoded}&type=name`; + try { + const resp = await this.fetchFn(queryUrl); + if (resp.status === 404) { + throw new FederationError('NOT_FOUND', `Federation address not found: "${address}"`); + } + if (!resp.ok) { + const body = await resp.text().catch(() => ''); + const detail = tryParseErrorBody(body) ?? `HTTP ${resp.status}`; + throw new FederationError('FEDERATION_FETCH_FAILED', `Federation server error: ${detail}`); + } + const body = await resp.text(); + let data: Record; + try { + data = JSON.parse(body) as Record; + } catch { + throw new FederationError( + 'INVALID_RESPONSE', + 'Federation server returned non-JSON response', + ); + } + return parseFederationResponse(data, address); + } catch (err) { + if (err instanceof FederationError) throw err; + throw new FederationError( + 'FEDERATION_FETCH_FAILED', + `Failed to reach federation server: ${(err as Error).message}`, + ); + } + } +} + +/** Shared singleton with default 1-hour cache used by {@link resolveStellarFederation}. */ +const _defaultResolver = new FederationResolver(); + +/** + * Resolves a Stellar federation address (`name*domain.com`) to a Stellar + * account ID and optional memo, following SEP-0002. + * + * Results are cached for 1 hour by default. For custom TTL, mock fetch, or + * isolated cache scopes, instantiate {@link FederationResolver} directly. + * + * @example + * ```ts + * const { accountId, memo } = await resolveStellarFederation('alice*example.com'); + * // Send to accountId, attach memo if present + * ``` + * + * @throws {@link FederationError} — check `.code` for the failure reason. + */ +export function resolveStellarFederation(address: string): Promise { + return _defaultResolver.resolve(address); +} + +// --------------------------------------------------------------------------- +// Internal helpers +// --------------------------------------------------------------------------- + +function parseFederationServerUrl(toml: string, domain: string): string { + // Match both single- and double-quoted values on any line. + const match = toml.match(/^\s*FEDERATION_SERVER\s*=\s*["']([^"']+)["']/m); + if (!match) { + throw new FederationError( + 'NO_FEDERATION_SERVER', + `stellar.toml for "${domain}" does not define FEDERATION_SERVER`, + ); + } + const url = match[1].trim(); + try { + const parsed = new URL(url); + if (parsed.protocol !== 'https:') { + throw new FederationError( + 'TOML_PARSE_FAILED', + `FEDERATION_SERVER must use HTTPS, got "${url}"`, + ); + } + } catch (err) { + if (err instanceof FederationError) throw err; + throw new FederationError( + 'TOML_PARSE_FAILED', + `FEDERATION_SERVER is not a valid URL: "${url}"`, + ); + } + return url; +} + +function parseFederationResponse(data: Record, address: string): FederationResult { + if (typeof data['account_id'] !== 'string' || !data['account_id']) { + throw new FederationError( + 'INVALID_RESPONSE', + `Federation response missing account_id for "${address}"`, + ); + } + const result: FederationResult = { accountId: data['account_id'] }; + + const memoType = typeof data['memo_type'] === 'string' ? data['memo_type'].toLowerCase() : null; + const memoValue = typeof data['memo'] === 'string' ? data['memo'] : null; + + if (memoType !== null && memoValue !== null) { + if (!MEMO_TYPES.has(memoType)) { + throw new FederationError('INVALID_RESPONSE', `Unknown memo_type "${data['memo_type']}"`); + } + result.memo = { type: memoType as FederationMemoType, value: memoValue }; + } + + return result; +} + +function tryParseErrorBody(body: string): string | null { + try { + const parsed = JSON.parse(body) as Record; + if (typeof parsed['detail'] === 'string') return parsed['detail']; + if (typeof parsed['message'] === 'string') return parsed['message']; + if (typeof parsed['error'] === 'string') return parsed['error']; + } catch { + // not JSON — fall through + } + return body.slice(0, 200) || null; +} diff --git a/src/chains/stellar/fee-estimation.ts b/src/chains/stellar/fee-estimation.ts index 25d18e0..bbf7fae 100644 --- a/src/chains/stellar/fee-estimation.ts +++ b/src/chains/stellar/fee-estimation.ts @@ -17,7 +17,7 @@ * - All values are in stroops (1 XLM = 10,000,000 stroops). */ -import type { Horizon, Soroban } from '@stellar/stellar-sdk'; +import type { Horizon, rpc } from '@stellar/stellar-sdk'; import type { Network } from './types'; // --------------------------------------------------------------------------- @@ -34,7 +34,7 @@ export interface SorobanResources { /** * Pre-fetched simulation result. If provided, no RPC call is made. */ - simulationResult?: Soroban.Api.SimulateTransactionResponse; + simulationResult?: rpc.Api.SimulateTransactionResponse; } /** Parameters for fee estimation. */ @@ -52,7 +52,7 @@ export interface EstimateFeeParams { * Pre-fetched fee stats from Horizon (/fee_stats). * If omitted, the helper fetches them itself. */ - feeStats?: Horizon.FeeStatsResponse; + feeStats?: Horizon.HorizonApi.FeeStatsResponse; /** * Set to true when the transaction will be wrapped in a fee-bump envelope. * Adds one extra base-fee unit for the outer transaction (CAP-0015). @@ -141,28 +141,21 @@ async function fetchFeeStats( ): Promise<{ baseFee: number; p50: number; p99: number }> { const res = await fetch(`${horizonUrl}/fee_stats`); if (!res.ok) { - throw new Error( - `Horizon /fee_stats request failed: ${res.status} ${res.statusText}`, - ); + throw new Error(`Horizon /fee_stats request failed: ${res.status} ${res.statusText}`); } - const data = (await res.json()) as Horizon.FeeStatsResponse; + const data = (await res.json()) as Horizon.HorizonApi.FeeStatsResponse; return parseFeeStats(data); } /** Parse a Horizon FeeStatsResponse into the numbers we need. */ -export function parseFeeStats(data: Horizon.FeeStatsResponse): { +export function parseFeeStats(data: Horizon.HorizonApi.FeeStatsResponse): { baseFee: number; p50: number; p99: number; } { - const baseFee = - parseInt(data.last_ledger_base_fee, 10) || PROTOCOL_MIN_BASE_FEE; - const p50 = - parseInt(data.fee_charged?.p50 ?? data.last_ledger_base_fee, 10) || - baseFee; - const p99 = - parseInt(data.fee_charged?.p99 ?? data.last_ledger_base_fee, 10) || - baseFee; + const baseFee = parseInt(data.last_ledger_base_fee, 10) || PROTOCOL_MIN_BASE_FEE; + const p50 = parseInt(data.fee_charged?.p50 ?? data.last_ledger_base_fee, 10) || baseFee; + const p99 = parseInt(data.fee_charged?.p99 ?? data.last_ledger_base_fee, 10) || baseFee; return { baseFee, p50, p99 }; } @@ -170,39 +163,29 @@ export function parseFeeStats(data: Horizon.FeeStatsResponse): { async function runSimulation( transactionXdr: string, rpcUrl: string, -): Promise { - const { SorobanRpc: SorobanRpcModule, TransactionBuilder } = await import( - '@stellar/stellar-sdk' - ); - const server = new SorobanRpcModule.Server(rpcUrl); +): Promise { + const { rpc: rpcModule, TransactionBuilder } = await import('@stellar/stellar-sdk'); + const server = new rpcModule.Server(rpcUrl); const tx = TransactionBuilder.fromXDR(transactionXdr, 'base64'); - return server.simulateTransaction( - tx as Parameters[0], - ); + return server.simulateTransaction(tx as Parameters[0]); } /** Extract resource fee in stroops from a simulation result. Throws on error. */ -function extractSorobanResourceFee( - result: Soroban.Api.SimulateTransactionResponse, -): number { +function extractSorobanResourceFee(result: rpc.Api.SimulateTransactionResponse): number { if ('error' in result) { throw new Error( - `Soroban simulation failed: ${(result as Soroban.Api.SimulateTransactionErrorResponse).error}`, + `Soroban simulation failed: ${(result as rpc.Api.SimulateTransactionErrorResponse).error}`, ); } if ('restorePreamble' in result) { return ( parseInt( - (result as Soroban.Api.SimulateTransactionRestoreResponse) - .restorePreamble.minResourceFee, + (result as rpc.Api.SimulateTransactionRestoreResponse).restorePreamble.minResourceFee, 10, ) || 0 ); } - const fee = parseInt( - (result as Soroban.Api.SimulateTransactionSuccessResponse).minResourceFee, - 10, - ); + const fee = parseInt((result as rpc.Api.SimulateTransactionSuccessResponse).minResourceFee, 10); return isNaN(fee) ? 0 : fee; } @@ -226,9 +209,7 @@ function extractSorobanResourceFee( * console.log(estimate.expected); // e.g. 1000 stroops * ``` */ -export async function estimateStellarFee( - params: EstimateFeeParams, -): Promise { +export async function estimateStellarFee(params: EstimateFeeParams): Promise { const { operationCount, sorobanResources, @@ -275,8 +256,7 @@ export async function estimateStellarFee( const inclusionLow = baseFee * safeOps; const inclusionExpected = Math.max(p50, baseFee) * safeOps; - const inclusionHigh = - Math.max(p99, baseFee) * HIGH_SURGE_MULTIPLIER * safeOps; + const inclusionHigh = Math.max(p99, baseFee) * HIGH_SURGE_MULTIPLIER * safeOps; // Step 4: add Soroban resource fee const sorobanPadding = simulationUsed @@ -320,14 +300,10 @@ function buildUncertaintyNote(opts: { ); } if (opts.feeBump) { - parts.push( - 'Fee-bump outer envelope adds 1 extra base-fee unit (CAP-0015).', - ); + parts.push('Fee-bump outer envelope adds 1 extra base-fee unit (CAP-0015).'); } if (opts.network === 'mainnet') { - parts.push( - 'Mainnet congestion can spike beyond p99 during high-traffic periods.', - ); + parts.push('Mainnet congestion can spike beyond p99 during high-traffic periods.'); } return parts.join(' '); -} \ No newline at end of file +} diff --git a/src/chains/stellar/index.ts b/src/chains/stellar/index.ts index cb4ac2d..aee573d 100644 --- a/src/chains/stellar/index.ts +++ b/src/chains/stellar/index.ts @@ -9,7 +9,12 @@ export { META_ADDRESS_PREFIX, } from './constants'; export { encodeStealthMetaAddress, decodeStealthMetaAddress } from './meta-address'; -export { generateStealthAddress, computeSharedSecret, computeViewTag } from './stealth'; +export { + generateStealthAddress, + computeSharedSecret, + computeViewTag, + computeAnnouncementViewTag, +} from './stealth'; export { checkStealthAddress, scanAnnouncements, scanAnnouncementsStream } from './scan'; export { deriveStealthPrivateScalar, signStellarTransaction } from './spend'; export { buildStealthPayment, buildStealthAnnouncement } from './builders'; @@ -23,8 +28,6 @@ export { L, } from './scalar'; export { bytesToHex, hexToBytes } from './utils'; -export { fetchAnnouncements } from './announcements'; -export type { FetchAnnouncementsOptions } from './announcements'; export { MemoryCache, IndexedDBCache, autoSelectCache } from './cache'; export type { AnnouncementCache } from './cache'; export { @@ -66,3 +69,11 @@ export type { } from './fee-estimation'; export { buildStellarSwapAndStealth } from './swap'; export type { BuildStellarSwapAndStealthOptions, SwapAndStealthResult } from './swap'; +export { resolveStellarFederation, FederationResolver, FederationError } from './federation'; +export type { + FederationResult, + FederationMemoType, + FederationErrorCode, + FederationFetchFn, + FederationResolverOptions, +} from './federation'; diff --git a/src/chains/stellar/stealth.ts b/src/chains/stellar/stealth.ts index fb781b0..b4156fc 100644 --- a/src/chains/stellar/stealth.ts +++ b/src/chains/stellar/stealth.ts @@ -111,3 +111,25 @@ export function computeViewTag(sharedSecret: Uint8Array): number { input.set(sharedSecret, LEGACY_VIEW_TAG_PREFIX.length); return sha256(input)[0]; } + +/** + * Computes the v2 view tag for a Stellar stealth announcement from public keys only. + * + * Unlike the legacy {@link computeViewTag} (which requires the ECDH shared secret), + * the v2 tag hashes the ephemeral and viewing public keys directly. This lets senders + * and recipients compute the same tag without exposing the shared secret. + * + * @param ephPubKey - 32-byte ephemeral ed25519 public key generated by the sender. + * @param viewingPubKey - 32-byte recipient ed25519 viewing public key. + * @returns Integer view tag in the range 0–255. + */ +export function computeAnnouncementViewTag( + ephPubKey: Uint8Array, + viewingPubKey: Uint8Array, +): number { + const input = new Uint8Array(VIEW_TAG_PREFIX.length + ephPubKey.length + viewingPubKey.length); + input.set(VIEW_TAG_PREFIX); + input.set(ephPubKey, VIEW_TAG_PREFIX.length); + input.set(viewingPubKey, VIEW_TAG_PREFIX.length + ephPubKey.length); + return sha256(input)[0]; +} diff --git a/src/compat/react-native.ts b/src/compat/react-native.ts index 67bf442..8f8add4 100644 --- a/src/compat/react-native.ts +++ b/src/compat/react-native.ts @@ -1,18 +1,27 @@ function btoaPolyfill(input: string): string { - const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; - let str = input; + const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; let output = ''; + let i = 0; + + while (i < input.length) { + const b0 = input.charCodeAt(i++); + const b1 = i < input.length ? input.charCodeAt(i++) : 0; + const b2 = i < input.length ? input.charCodeAt(i++) : 0; - for (let block = 0, charCode, idx = 0, map = chars; str.charAt(idx | 0) || ((map = '='), idx % 1); ) { - charCode = str.charCodeAt((idx += 3 / 4)); - if (charCode > 0xff) { - throw new Error("Unable to encode character as binary data"); + if (b0 > 0xff || b1 > 0xff || b2 > 0xff) { + throw new Error('Unable to encode character as binary data'); } + + const triplet = (b0 << 16) | (b1 << 8) | b2; output += - map.charAt((block = (block << 8) | charCode) >> ((3.5 - (idx % 1)) * 8) & 0x3f); + chars[(triplet >> 18) & 0x3f] + + chars[(triplet >> 12) & 0x3f] + + chars[(triplet >> 6) & 0x3f] + + chars[triplet & 0x3f]; } - return output; + const padLen = (3 - (input.length % 3)) % 3; + return output.slice(0, output.length - padLen) + '='.repeat(padLen); } function atobPolyfill(input: string): string { @@ -21,7 +30,7 @@ function atobPolyfill(input: string): string { let output = ''; if (str.length % 4 === 1) { - throw new Error("InvalidCharacterError: Incorrect base64 string length"); + throw new Error('InvalidCharacterError: Incorrect base64 string length'); } for (let bc = 0, bs = 0, buffer, idx = 0; (buffer = str.charAt(idx++)); ) { diff --git a/test/chains/stellar/bench/stellar.bench.ts b/test/chains/stellar/bench/stellar.bench.ts index ca80d19..58e7186 100644 --- a/test/chains/stellar/bench/stellar.bench.ts +++ b/test/chains/stellar/bench/stellar.bench.ts @@ -4,7 +4,10 @@ import { deriveStealthKeys } from '../../../../src/chains/stellar/keys'; import { generateStealthAddress } from '../../../../src/chains/stellar/stealth'; import { scanAnnouncements, checkStealthAddress } from '../../../../src/chains/stellar/scan'; import { deriveStealthPrivateScalar } from '../../../../src/chains/stellar/spend'; -import { encodeStealthMetaAddress, decodeStealthMetaAddress } from '../../../../src/chains/stellar/meta-address'; +import { + encodeStealthMetaAddress, + decodeStealthMetaAddress, +} from '../../../../src/chains/stellar/meta-address'; import { signWithScalar } from '../../../../src/chains/stellar/scalar'; import { fetchAnnouncements } from '../../../../src/chains/stellar/announcements'; import { bytesToHex } from '../../../../src/chains/stellar/utils'; @@ -103,19 +106,39 @@ describe('Stellar Stealth Benchmarks', () => { }); bench('scanAnnouncements - 100 announcements', () => { - scanAnnouncements(announcements100, keys.viewingKey, keys.spendingPubKey, keys.spendingScalar); + scanAnnouncements( + announcements100, + keys.viewingKey, + keys.spendingPubKey, + keys.spendingScalar, + ); }); bench('scanAnnouncements - 1,000 announcements', () => { - scanAnnouncements(announcements1000, keys.viewingKey, keys.spendingPubKey, keys.spendingScalar); + scanAnnouncements( + announcements1000, + keys.viewingKey, + keys.spendingPubKey, + keys.spendingScalar, + ); }); bench('scanAnnouncements - 10,000 announcements', () => { - scanAnnouncements(announcements10000, keys.viewingKey, keys.spendingPubKey, keys.spendingScalar); + scanAnnouncements( + announcements10000, + keys.viewingKey, + keys.spendingPubKey, + keys.spendingScalar, + ); }); bench('scanAnnouncements - 100,000 announcements', () => { - scanAnnouncements(announcements100000, keys.viewingKey, keys.spendingPubKey, keys.spendingScalar); + scanAnnouncements( + announcements100000, + keys.viewingKey, + keys.spendingPubKey, + keys.spendingScalar, + ); }); }); diff --git a/test/chains/stellar/cache.test.ts b/test/chains/stellar/cache.test.ts index b6bea2c..ab560a3 100644 --- a/test/chains/stellar/cache.test.ts +++ b/test/chains/stellar/cache.test.ts @@ -115,4 +115,3 @@ describe('MemoryCache', () => { expect(result).toHaveLength(1); }); }); - diff --git a/test/chains/stellar/federation.integration.test.ts b/test/chains/stellar/federation.integration.test.ts new file mode 100644 index 0000000..0a1102f --- /dev/null +++ b/test/chains/stellar/federation.integration.test.ts @@ -0,0 +1,61 @@ +/** + * Integration tests: resolveStellarFederation against live Stellar testnet. + * + * Makes real HTTP requests to: + * - https://testanchor.stellar.org/.well-known/stellar.toml + * - https://testanchor.stellar.org/federation (discovered from the TOML) + * + * Run with: + * INTEGRATION=1 pnpm exec vitest run test/chains/stellar/federation.integration.test.ts + * + * Skipped by default in CI unless INTEGRATION=1 is set. + */ + +import { describe, it, expect } from 'vitest'; +import { FederationResolver, FederationError } from '../../../src/chains/stellar/federation'; + +const SKIP = process.env['INTEGRATION'] !== '1'; + +describe('Integration: federation resolution (testanchor.stellar.org)', { skip: SKIP }, () => { + it('discovers FEDERATION_SERVER from stellar.toml', async () => { + const resp = await fetch('https://testanchor.stellar.org/.well-known/stellar.toml'); + expect(resp.ok).toBe(true); + const toml = await resp.text(); + expect(toml).toMatch(/FEDERATION_SERVER/); + console.log('[Integration] stellar.toml (first 400 chars):\n', toml.slice(0, 400)); + }, 10_000); + + it('resolves demo*testanchor.stellar.org to a valid Stellar account ID', async () => { + const resolver = new FederationResolver({ cacheTtl: 0 }); + const result = await resolver.resolve('demo*testanchor.stellar.org'); + // Stellar account IDs start with G and are 56 chars (base32) + expect(result.accountId).toMatch(/^G[A-Z2-7]{55}$/); + console.log('[Integration] demo*testanchor.stellar.org →', result); + }, 15_000); + + it('returns cached result on second call (no extra HTTP calls)', async () => { + const calls: string[] = []; + const resolver = new FederationResolver({ + cacheTtl: 60_000, + fetch: async (url) => { + calls.push(url); + return globalThis.fetch(url); + }, + }); + await resolver.resolve('demo*testanchor.stellar.org'); + const callsAfterFirst = calls.length; + await resolver.resolve('demo*testanchor.stellar.org'); + // Second resolve must not add any new calls + expect(calls.length).toBe(callsAfterFirst); + }, 15_000); + + it('throws NOT_FOUND or FEDERATION_FETCH_FAILED for a non-existent address', async () => { + const resolver = new FederationResolver({ cacheTtl: 0 }); + const err = await resolver + .resolve('nonexistent-user-zzzzzzzz*testanchor.stellar.org') + .catch((e) => e); + expect(err).toBeInstanceOf(FederationError); + expect(['NOT_FOUND', 'FEDERATION_FETCH_FAILED']).toContain(err.code); + console.log('[Integration] Expected error for non-existent address:', err.code, err.message); + }, 15_000); +}); diff --git a/test/chains/stellar/federation.test.ts b/test/chains/stellar/federation.test.ts new file mode 100644 index 0000000..48ebfcd --- /dev/null +++ b/test/chains/stellar/federation.test.ts @@ -0,0 +1,437 @@ +import { describe, it, expect, vi, beforeEach } from 'vitest'; +import { + FederationResolver, + FederationError, + resolveStellarFederation, +} from '../../../src/chains/stellar/federation'; + +// --------------------------------------------------------------------------- +// Helpers +// --------------------------------------------------------------------------- + +function makeToml(federationUrl = 'https://federation.example.com'): string { + return `NETWORK_PASSPHRASE="Test SDF Network ; September 2015"\nFEDERATION_SERVER="${federationUrl}"\n`; +} + +function makeFedResponse(accountId: string, extra: Record = {}): string { + return JSON.stringify({ stellar_address: 'alice*example.com', account_id: accountId, ...extra }); +} + +type MockEntry = { status: number; body: string }; + +function mockFetch(responses: Record) { + return vi.fn(async (url: string) => { + const entry = responses[url]; + if (!entry) throw new Error(`Unexpected fetch call: ${url}`); + return { + ok: entry.status >= 200 && entry.status < 300, + status: entry.status, + text: async () => entry.body, + } as Response; + }); +} + +const FAKE_ACCOUNT = 'GCEZWKCA5VLDNRLN3RPRJMRZOX3Z6G5CHCGSHV1NFOU3FAKE'; + +// Default responses for happy-path tests +function defaultResponses(): Record { + return { + 'https://example.com/.well-known/stellar.toml': { + status: 200, + body: makeToml('https://federation.example.com'), + }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: makeFedResponse(FAKE_ACCOUNT), + }, + }; +} + +// --------------------------------------------------------------------------- +// Happy path +// --------------------------------------------------------------------------- + +describe('FederationResolver — happy path', () => { + let fetchFn: ReturnType; + let resolver: FederationResolver; + + beforeEach(() => { + fetchFn = mockFetch(defaultResponses()); + resolver = new FederationResolver({ fetch: fetchFn as never, cacheTtl: 60_000 }); + }); + + it('resolves a valid federation address', async () => { + const result = await resolver.resolve('alice*example.com'); + expect(result.accountId).toBe(FAKE_ACCOUNT); + expect(result.memo).toBeUndefined(); + }); + + it('fetches stellar.toml first, then queries the federation server', async () => { + await resolver.resolve('alice*example.com'); + expect(fetchFn).toHaveBeenCalledTimes(2); + expect(fetchFn).toHaveBeenNthCalledWith(1, 'https://example.com/.well-known/stellar.toml'); + expect(fetchFn).toHaveBeenNthCalledWith( + 2, + 'https://federation.example.com?q=alice%2Aexample.com&type=name', + ); + }); + + it('encodes the federation address in the query string', async () => { + // `*` must be percent-encoded as %2A in the q= parameter + await resolver.resolve('alice*example.com'); + const calledUrl = fetchFn.mock.calls[1][0] as string; + expect(calledUrl).toContain('q=alice%2Aexample.com'); + }); + + it('caches the result — only two HTTP calls for repeated resolves', async () => { + await resolver.resolve('alice*example.com'); + await resolver.resolve('alice*example.com'); + expect(fetchFn).toHaveBeenCalledTimes(2); + }); + + it('re-fetches after invalidate()', async () => { + await resolver.resolve('alice*example.com'); + resolver.invalidate('alice*example.com'); + await resolver.resolve('alice*example.com'); + expect(fetchFn).toHaveBeenCalledTimes(4); + }); + + it('re-fetches after clear()', async () => { + await resolver.resolve('alice*example.com'); + resolver.clear(); + await resolver.resolve('alice*example.com'); + expect(fetchFn).toHaveBeenCalledTimes(4); + }); + + it('normalises address to lowercase for cache keying', async () => { + await resolver.resolve('Alice*Example.com'); + await resolver.resolve('ALICE*EXAMPLE.COM'); + // Both resolve to the same cache key — only 2 real HTTP calls total + expect(fetchFn).toHaveBeenCalledTimes(2); + }); + + it('skips the cache when cacheTtl=0', async () => { + const f = mockFetch(defaultResponses()); + const r = new FederationResolver({ fetch: f as never, cacheTtl: 0 }); + await r.resolve('alice*example.com'); + await r.resolve('alice*example.com'); + expect(f).toHaveBeenCalledTimes(4); // 2 HTTP calls Ɨ 2 resolves + }); +}); + +// --------------------------------------------------------------------------- +// Memo handling +// --------------------------------------------------------------------------- + +describe('FederationResolver — memo handling', () => { + it.each([ + ['text', 'payment-ref-123'], + ['id', '99887766'], + ['hash', 'deadbeefcafe'], + ] as const)('parses memo_type=%s', async (memoType, memoValue) => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { + status: 200, + body: makeToml(), + }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: makeFedResponse(FAKE_ACCOUNT, { memo_type: memoType, memo: memoValue }), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + const result = await r.resolve('alice*example.com'); + expect(result.memo).toEqual({ type: memoType, value: memoValue }); + }); + + it('omits memo when only memo_type is present (no memo value)', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: JSON.stringify({ account_id: FAKE_ACCOUNT, memo_type: 'text' }), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + const result = await r.resolve('alice*example.com'); + expect(result.memo).toBeUndefined(); + }); + + it('omits memo when only memo is present (no memo_type)', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: JSON.stringify({ account_id: FAKE_ACCOUNT, memo: 'orphan-memo' }), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + const result = await r.resolve('alice*example.com'); + expect(result.memo).toBeUndefined(); + }); +}); + +// --------------------------------------------------------------------------- +// TOML parsing edge cases +// --------------------------------------------------------------------------- + +describe('FederationResolver — TOML parsing', () => { + function makeResolver(tomlBody: string) { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: tomlBody }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: makeFedResponse(FAKE_ACCOUNT), + }, + }); + return new FederationResolver({ fetch: f as never }); + } + + it('accepts single-quoted FEDERATION_SERVER', async () => { + const r = makeResolver(`FEDERATION_SERVER='https://federation.example.com'\n`); + const result = await r.resolve('alice*example.com'); + expect(result.accountId).toBe(FAKE_ACCOUNT); + }); + + it('ignores other TOML keys before FEDERATION_SERVER', async () => { + const toml = [ + `NETWORK_PASSPHRASE="Test"`, + `ACCOUNTS=["GABC"]`, + `FEDERATION_SERVER="https://federation.example.com"`, + `HORIZON_URL="https://horizon.example.com"`, + ].join('\n'); + const r = makeResolver(toml); + const result = await r.resolve('alice*example.com'); + expect(result.accountId).toBe(FAKE_ACCOUNT); + }); + + it('handles FEDERATION_SERVER with leading whitespace', async () => { + const r = makeResolver(` FEDERATION_SERVER = "https://federation.example.com"\n`); + const result = await r.resolve('alice*example.com'); + expect(result.accountId).toBe(FAKE_ACCOUNT); + }); + + it('appends &q= when federation URL already contains a query string', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { + status: 200, + body: `FEDERATION_SERVER="https://federation.example.com/api?version=2"\n`, + }, + 'https://federation.example.com/api?version=2&q=alice%2Aexample.com&type=name': { + status: 200, + body: makeFedResponse(FAKE_ACCOUNT), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + const result = await r.resolve('alice*example.com'); + expect(result.accountId).toBe(FAKE_ACCOUNT); + }); +}); + +// --------------------------------------------------------------------------- +// Error cases +// --------------------------------------------------------------------------- + +describe('FederationResolver — errors', () => { + it('throws INVALID_ADDRESS when * is absent', async () => { + const r = new FederationResolver({ fetch: vi.fn() as never }); + await expect(r.resolve('aliceexample.com')).rejects.toMatchObject({ + code: 'INVALID_ADDRESS', + }); + }); + + it('throws INVALID_ADDRESS for multiple * separators', async () => { + const r = new FederationResolver({ fetch: vi.fn() as never }); + await expect(r.resolve('alice*foo*example.com')).rejects.toMatchObject({ + code: 'INVALID_ADDRESS', + }); + }); + + it('throws INVALID_ADDRESS when domain has no dot', async () => { + const r = new FederationResolver({ fetch: vi.fn() as never }); + await expect(r.resolve('alice*examplecom')).rejects.toMatchObject({ + code: 'INVALID_ADDRESS', + }); + }); + + it('throws INVALID_ADDRESS for whitespace in name', async () => { + const r = new FederationResolver({ fetch: vi.fn() as never }); + await expect(r.resolve('ali ce*example.com')).rejects.toMatchObject({ + code: 'INVALID_ADDRESS', + }); + }); + + it('throws TOML_FETCH_FAILED when stellar.toml returns non-2xx', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 500, body: '' }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'TOML_FETCH_FAILED', + }); + }); + + it('throws TOML_FETCH_FAILED when fetch rejects with a network error', async () => { + const f = vi.fn().mockRejectedValue(new Error('network unreachable')); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'TOML_FETCH_FAILED', + message: expect.stringContaining('network unreachable'), + }); + }); + + it('throws NO_FEDERATION_SERVER when stellar.toml lacks FEDERATION_SERVER', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { + status: 200, + body: 'NETWORK_PASSPHRASE="Test"\n', + }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'NO_FEDERATION_SERVER', + }); + }); + + it('throws TOML_PARSE_FAILED when FEDERATION_SERVER uses HTTP', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { + status: 200, + body: `FEDERATION_SERVER="http://federation.example.com"\n`, + }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'TOML_PARSE_FAILED', + }); + }); + + it('throws TOML_PARSE_FAILED when FEDERATION_SERVER is not a valid URL', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { + status: 200, + body: `FEDERATION_SERVER="not-a-url"\n`, + }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'TOML_PARSE_FAILED', + }); + }); + + it('throws NOT_FOUND when federation server returns 404', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { status: 404, body: '' }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'NOT_FOUND', + }); + }); + + it('throws FEDERATION_FETCH_FAILED on a non-404 server error', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { status: 503, body: '' }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'FEDERATION_FETCH_FAILED', + }); + }); + + it('surfaces the error detail field from a JSON error body', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 400, + body: JSON.stringify({ detail: 'rate limit exceeded' }), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + const err = (await r.resolve('alice*example.com').catch((e) => e)) as FederationError; + expect(err.message).toContain('rate limit exceeded'); + }); + + it('throws INVALID_RESPONSE when account_id is missing from federation response', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: JSON.stringify({ stellar_address: 'alice*example.com' }), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'INVALID_RESPONSE', + }); + }); + + it('throws INVALID_RESPONSE for an unknown memo_type', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: JSON.stringify({ account_id: FAKE_ACCOUNT, memo_type: 'binary', memo: '0xff' }), + }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'INVALID_RESPONSE', + }); + }); + + it('throws INVALID_RESPONSE when the federation server returns non-JSON', async () => { + const f = mockFetch({ + 'https://example.com/.well-known/stellar.toml': { status: 200, body: makeToml() }, + 'https://federation.example.com?q=alice%2Aexample.com&type=name': { + status: 200, + body: 'error', + }, + }); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'INVALID_RESPONSE', + }); + }); + + it('FederationError has the correct name property', async () => { + const r = new FederationResolver({ fetch: vi.fn() as never }); + const err = (await r.resolve('bad').catch((e) => e)) as FederationError; + expect(err).toBeInstanceOf(FederationError); + expect(err.name).toBe('FederationError'); + }); + + it('throws FEDERATION_FETCH_FAILED when the fetch call to the federation server rejects', async () => { + const f = vi + .fn() + .mockResolvedValueOnce({ + ok: true, + status: 200, + text: async () => makeToml(), + } as Response) + .mockRejectedValueOnce(new Error('connection refused')); + const r = new FederationResolver({ fetch: f as never }); + await expect(r.resolve('alice*example.com')).rejects.toMatchObject({ + code: 'FEDERATION_FETCH_FAILED', + message: expect.stringContaining('connection refused'), + }); + }); +}); + +// --------------------------------------------------------------------------- +// Module-level export +// --------------------------------------------------------------------------- + +describe('resolveStellarFederation', () => { + it('is exported and is a function', () => { + expect(typeof resolveStellarFederation).toBe('function'); + }); + + it('rejects invalid addresses with INVALID_ADDRESS', async () => { + await expect(resolveStellarFederation('notafedaddress')).rejects.toMatchObject({ + code: 'INVALID_ADDRESS', + }); + }); +}); diff --git a/test/chains/stellar/fee-estimation.integration.test.ts b/test/chains/stellar/fee-estimation.integration.test.ts index d1530c3..f2502fd 100644 --- a/test/chains/stellar/fee-estimation.integration.test.ts +++ b/test/chains/stellar/fee-estimation.integration.test.ts @@ -61,4 +61,4 @@ describe('Integration: estimateStellarFee (testnet)', { skip: SKIP }, () => { expect(bumped.breakdown.feeBumpApplied).toBe(true); expect(bumped.low).toBeGreaterThanOrEqual(300); }, 15_000); -}); \ No newline at end of file +}); diff --git a/test/chains/stellar/fee-estimation.test.ts b/test/chains/stellar/fee-estimation.test.ts index 49d3d21..cadb5b9 100644 --- a/test/chains/stellar/fee-estimation.test.ts +++ b/test/chains/stellar/fee-estimation.test.ts @@ -1,17 +1,12 @@ import { describe, it, expect, vi, beforeEach } from 'vitest'; import type { Horizon, SorobanRpc } from '@stellar/stellar-sdk'; -import { - estimateStellarFee, - parseFeeStats, -} from '../../../src/chains/stellar/fee-estimation'; +import { estimateStellarFee, parseFeeStats } from '../../../src/chains/stellar/fee-estimation'; // --------------------------------------------------------------------------- // Fixtures // --------------------------------------------------------------------------- -function makeFeeStats( - overrides?: Partial, -): Horizon.FeeStatsResponse { +function makeFeeStats(overrides?: Partial): Horizon.FeeStatsResponse { return { last_ledger: '12345', last_ledger_base_fee: '100', @@ -52,23 +47,20 @@ function makeFeeStats( } as Horizon.FeeStatsResponse; } -function makeSimResult( - minResourceFee = '5000', -): SorobanRpc.Api.SimulateTransactionSuccessResponse { +function makeSimResult(minResourceFee = '5000'): SorobanRpc.Api.SimulateTransactionSuccessResponse { return { id: 'sim-1', latestLedger: 9999, minResourceFee, results: [], - transactionData: '' as unknown as SorobanRpc.Api.SimulateTransactionSuccessResponse['transactionData'], + transactionData: + '' as unknown as SorobanRpc.Api.SimulateTransactionSuccessResponse['transactionData'], events: [], cost: { cpuInsns: '1000', memBytes: '512' }, } as unknown as SorobanRpc.Api.SimulateTransactionSuccessResponse; } -function makeSimError( - error = 'out of gas', -): SorobanRpc.Api.SimulateTransactionErrorResponse { +function makeSimError(error = 'out of gas'): SorobanRpc.Api.SimulateTransactionErrorResponse { return { id: 'sim-err', latestLedger: 9999, @@ -136,9 +128,9 @@ describe('estimateStellarFee — classic ops', () => { feeStats: makeFeeStats(), }); - expect(estimate.low).toBe(100); // baseFee Ɨ 1 - expect(estimate.expected).toBe(300); // p50 Ɨ 1 - expect(estimate.high).toBe(10_000); // p99 Ɨ 2 Ɨ 1 + expect(estimate.low).toBe(100); // baseFee Ɨ 1 + expect(estimate.expected).toBe(300); // p50 Ɨ 1 + expect(estimate.high).toBe(10_000); // p99 Ɨ 2 Ɨ 1 }); it('scales linearly with op count', async () => { @@ -155,9 +147,7 @@ describe('estimateStellarFee — classic ops', () => { it('fetches fee stats from Horizon when not pre-fetched', async () => { mockHorizonFeeStats(); const estimate = await estimateStellarFee({ operationCount: 1 }); - expect(mockFetch).toHaveBeenCalledWith( - expect.stringContaining('fee_stats'), - ); + expect(mockFetch).toHaveBeenCalledWith(expect.stringContaining('fee_stats')); expect(estimate.low).toBeGreaterThan(0); }); @@ -167,16 +157,14 @@ describe('estimateStellarFee — classic ops', () => { operationCount: 1, horizonUrl: 'https://my-horizon.example.com', }); - expect(mockFetch).toHaveBeenCalledWith( - 'https://my-horizon.example.com/fee_stats', - ); + expect(mockFetch).toHaveBeenCalledWith('https://my-horizon.example.com/fee_stats'); }); it('throws on Horizon error', async () => { mockHorizonError(); - await expect( - estimateStellarFee({ operationCount: 1 }), - ).rejects.toThrow('Horizon /fee_stats request failed: 503'); + await expect(estimateStellarFee({ operationCount: 1 })).rejects.toThrow( + 'Horizon /fee_stats request failed: 503', + ); }); it('throws on negative operationCount', async () => { @@ -263,7 +251,9 @@ describe('estimateStellarFee — Soroban (pre-fetched simulation)', () => { feeStats: makeFeeStats(), sorobanResources: { transactionXdr: 'AAAAAQ==', - simulationResult: makeSimError('wasm trap') as unknown as SorobanRpc.Api.SimulateTransactionResponse, + simulationResult: makeSimError( + 'wasm trap', + ) as unknown as SorobanRpc.Api.SimulateTransactionResponse, }, }), ).rejects.toThrow('Soroban simulation failed: wasm trap'); @@ -304,4 +294,4 @@ describe('estimateStellarFee — Soroban (pre-fetched simulation)', () => { expect(estimate.high).toBe(highInclusion + resourceFee + padding); }); -}); \ No newline at end of file +}); diff --git a/test/chains/stellar/scan.test.ts b/test/chains/stellar/scan.test.ts index 64a6d87..0fb026d 100644 --- a/test/chains/stellar/scan.test.ts +++ b/test/chains/stellar/scan.test.ts @@ -1,10 +1,16 @@ import { describe, test, expect } from 'vitest'; import { deriveStealthKeys } from '../../../src/chains/stellar/keys'; -import { generateStealthAddress } from '../../../src/chains/stellar/stealth'; +import { + generateStealthAddress, + computeAnnouncementViewTag, + computeSharedSecret, + computeViewTag, +} from '../../../src/chains/stellar/stealth'; import { checkStealthAddress, scanAnnouncements, scanAnnouncementsStream, + scanAnnouncementsLegacySharedSecretTag, } from '../../../src/chains/stellar/scan'; import { SCHEME_ID } from '../../../src/chains/stellar/constants'; import { bytesToHex } from '../../../src/chains/stellar/utils';