diff --git a/.bazelversion b/.bazelversion
index f9c71a52e2fd..acd405b1d62e 100644
--- a/.bazelversion
+++ b/.bazelversion
@@ -1 +1 @@
-8.5.1
+8.6.0
diff --git a/.github/workflows/assistant-to-the-branch-manager.yml b/.github/workflows/assistant-to-the-branch-manager.yml
index 6e0a4cb8c7d7..eada34703004 100644
--- a/.github/workflows/assistant-to-the-branch-manager.yml
+++ b/.github/workflows/assistant-to-the-branch-manager.yml
@@ -17,6 +17,6 @@ jobs:
- uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
with:
persist-credentials: false
- - uses: angular/dev-infra/github-actions/branch-manager@469708f109a90884ca403d150d33079a3a5a8769
+ - uses: angular/dev-infra/github-actions/branch-manager@9cc477855b9788df6257301074a1629bc3545722
with:
angular-robot-key: ${{ secrets.ANGULAR_ROBOT_PRIVATE_KEY }}
diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml
index 5f62362d4410..2760c53a94ee 100644
--- a/.github/workflows/ci.yml
+++ b/.github/workflows/ci.yml
@@ -21,9 +21,9 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Generate JSON schema types
@@ -44,11 +44,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Install node modules
@@ -61,11 +61,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Install node modules
@@ -84,13 +84,13 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Run CLI E2E tests
@@ -100,11 +100,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Install node modules
@@ -137,7 +137,7 @@ jobs:
runs-on: windows-2025
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Download built Windows E2E tests
@@ -164,13 +164,13 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Run CLI E2E tests
@@ -188,13 +188,13 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Run CLI E2E tests
@@ -208,13 +208,13 @@ jobs:
SAUCE_TUNNEL_IDENTIFIER: angular-cli-${{ github.workflow }}-${{ github.run_number }}
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
with:
google_credential: ${{ secrets.RBE_TRUSTED_BUILDS_USER }}
- name: Run E2E Browser tests
@@ -244,11 +244,11 @@ jobs:
CIRCLE_BRANCH: ${{ github.ref_name }}
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- run: pnpm admin snapshots --verbose
env:
SNAPSHOT_BUILDS_GITHUB_TOKEN: ${{ secrets.SNAPSHOT_BUILDS_GITHUB_TOKEN }}
diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml
index 0b070c5aae16..8e9ed17f7796 100644
--- a/.github/workflows/codeql.yml
+++ b/.github/workflows/codeql.yml
@@ -23,12 +23,12 @@ jobs:
with:
persist-credentials: false
- name: Initialize CodeQL
- uses: github/codeql-action/init@45cbd0c69e560cd9e7cd7f8c32362050c9b7ded2 # v4.32.2
+ uses: github/codeql-action/init@89a39a4e59826350b863aa6b6252a07ad50cf83e # v4.32.4
with:
languages: javascript-typescript
build-mode: none
config-file: .github/codeql/config.yml
- name: Perform CodeQL Analysis
- uses: github/codeql-action/analyze@45cbd0c69e560cd9e7cd7f8c32362050c9b7ded2 # v4.32.2
+ uses: github/codeql-action/analyze@89a39a4e59826350b863aa6b6252a07ad50cf83e # v4.32.4
with:
category: '/language:javascript-typescript'
diff --git a/.github/workflows/dev-infra.yml b/.github/workflows/dev-infra.yml
index d8d2f0b7bb6c..55b7a545c05b 100644
--- a/.github/workflows/dev-infra.yml
+++ b/.github/workflows/dev-infra.yml
@@ -13,13 +13,13 @@ jobs:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
- - uses: angular/dev-infra/github-actions/pull-request-labeling@469708f109a90884ca403d150d33079a3a5a8769
+ - uses: angular/dev-infra/github-actions/pull-request-labeling@9cc477855b9788df6257301074a1629bc3545722
with:
angular-robot-key: ${{ secrets.ANGULAR_ROBOT_PRIVATE_KEY }}
post_approval_changes:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
- - uses: angular/dev-infra/github-actions/post-approval-changes@469708f109a90884ca403d150d33079a3a5a8769
+ - uses: angular/dev-infra/github-actions/post-approval-changes@9cc477855b9788df6257301074a1629bc3545722
with:
angular-robot-key: ${{ secrets.ANGULAR_ROBOT_PRIVATE_KEY }}
diff --git a/.github/workflows/feature-requests.yml b/.github/workflows/feature-requests.yml
index 4e22f1a5eda8..dc02ea5f6ee2 100644
--- a/.github/workflows/feature-requests.yml
+++ b/.github/workflows/feature-requests.yml
@@ -16,6 +16,6 @@ jobs:
if: github.repository == 'angular/angular-cli'
runs-on: ubuntu-latest
steps:
- - uses: angular/dev-infra/github-actions/feature-request@469708f109a90884ca403d150d33079a3a5a8769
+ - uses: angular/dev-infra/github-actions/feature-request@9cc477855b9788df6257301074a1629bc3545722
with:
angular-robot-key: ${{ secrets.ANGULAR_ROBOT_PRIVATE_KEY }}
diff --git a/.github/workflows/perf.yml b/.github/workflows/perf.yml
index 9dbf67608b8c..9cfe7d19ebed 100644
--- a/.github/workflows/perf.yml
+++ b/.github/workflows/perf.yml
@@ -23,7 +23,7 @@ jobs:
workflows: ${{ steps.workflows.outputs.workflows }}
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- id: workflows
@@ -38,9 +38,9 @@ jobs:
workflow: ${{ fromJSON(needs.list.outputs.workflows) }}
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
# We utilize the google-github-actions/auth action to allow us to get an active credential using workflow
diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml
index 12195b663fda..e3da41b25686 100644
--- a/.github/workflows/pr.yml
+++ b/.github/workflows/pr.yml
@@ -34,9 +34,9 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup ESLint Caching
uses: actions/cache@cdf6c1fa76f9f475f3d7449005a359c84ca0f306 # v5.0.3
with:
@@ -66,17 +66,17 @@ jobs:
# it has been merged.
run: pnpm ng-dev format changed --check ${{ github.event.pull_request.base.sha }}
- name: Check Package Licenses
- uses: angular/dev-infra/github-actions/linting/licenses@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/linting/licenses@9cc477855b9788df6257301074a1629bc3545722
build:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Build release targets
@@ -93,11 +93,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Run module and package tests
@@ -114,13 +114,13 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
- name: Run CLI E2E tests
run: pnpm bazel test --test_env=E2E_SHARD_TOTAL=6 --test_env=E2E_SHARD_INDEX=${{ matrix.shard }} --config=e2e //tests:e2e.${{ matrix.subset }}_node${{ matrix.node }}
@@ -128,11 +128,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Build E2E tests for Windows on Linux
@@ -156,7 +156,7 @@ jobs:
runs-on: windows-2025
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Download built Windows E2E tests
@@ -183,13 +183,13 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
- name: Run CLI E2E tests
run: pnpm bazel test --test_env=E2E_SHARD_TOTAL=3 --test_env=E2E_SHARD_INDEX=${{ matrix.shard }} --config=e2e //tests:e2e.${{ matrix.subset }}_node${{ matrix.node }}
@@ -205,12 +205,12 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Initialize environment
- uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/npm/checkout-and-setup-node@9cc477855b9788df6257301074a1629bc3545722
- name: Install node modules
run: pnpm install --frozen-lockfile
- name: Setup Bazel
- uses: angular/dev-infra/github-actions/bazel/setup@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/setup@9cc477855b9788df6257301074a1629bc3545722
- name: Setup Bazel RBE
- uses: angular/dev-infra/github-actions/bazel/configure-remote@469708f109a90884ca403d150d33079a3a5a8769
+ uses: angular/dev-infra/github-actions/bazel/configure-remote@9cc477855b9788df6257301074a1629bc3545722
- name: Run CLI E2E tests
run: pnpm bazel test --test_env=E2E_SHARD_TOTAL=6 --test_env=E2E_SHARD_INDEX=${{ matrix.shard }} --config=e2e //tests:e2e.snapshots.${{ matrix.subset }}_node${{ matrix.node }}
diff --git a/.github/workflows/scorecard.yml b/.github/workflows/scorecard.yml
index 4c9465d0cf9b..6e75cf9b2dc3 100644
--- a/.github/workflows/scorecard.yml
+++ b/.github/workflows/scorecard.yml
@@ -46,6 +46,6 @@ jobs:
# Upload the results to GitHub's code scanning dashboard.
- name: 'Upload to code-scanning'
- uses: github/codeql-action/upload-sarif@45cbd0c69e560cd9e7cd7f8c32362050c9b7ded2 # v4.32.2
+ uses: github/codeql-action/upload-sarif@89a39a4e59826350b863aa6b6252a07ad50cf83e # v4.32.4
with:
sarif_file: results.sarif
diff --git a/CHANGELOG.md b/CHANGELOG.md
index b11c764b3618..ece818ff8c88 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,26 +1,54 @@
-
+
-# 19.2.20 (2026-02-13)
+# 21.2.0 (2026-02-25)
-### @angular-devkit/build-angular
+### @angular/cli
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | ------------------------- |
-| [0e5421ba7](https://github.com/angular/angular-cli/commit/0e5421ba78814cf11e4d4510e930eaacc6458662) | fix | update webpack to 5.105.0 |
+| Commit | Type | Description |
+| --------------------------------------------------------------------------------------------------- | ---- | ------------------------------------------------------------------------------------------------------------- |
+| [0dd04f289](https://github.com/angular/angular-cli/commit/0dd04f289e555a4a8af7bdadabe300da74701e3b) | feat | add markdown files to Prettier's formatting list |
+| [fbae1b6ab](https://github.com/angular/angular-cli/commit/fbae1b6ab384186ae69e804c54815cea80e6a600) | feat | automatic formatting files modified by schematics |
+| [91b9d281f](https://github.com/angular/angular-cli/commit/91b9d281fc88a242aa6e5dd5495e275990d926ef) | feat | integrate file formatting into update migrations |
+| [98a24d040](https://github.com/angular/angular-cli/commit/98a24d0401f36f484dc9c4d8b0f5284ffa524f19) | feat | standardize MCP tools around workspace/project options |
+| [d9cd609c5](https://github.com/angular/angular-cli/commit/d9cd609c5d13fe492b1f31973d9be518f8529387) | fix | correctly parse scoped packages in yarn classic list output |
+| [5b05f2500](https://github.com/angular/angular-cli/commit/5b05f25005621828565585692b1d7a67c5f0fec8) | fix | enable shell option for Prettier execution on Windows platforms |
+| [25b8a157d](https://github.com/angular/angular-cli/commit/25b8a157df70fb0d2c4e6c5438a50ec12e3abc0c) | fix | quote complex range specifiers in package manager |
+| [6f29a8c35](https://github.com/angular/angular-cli/commit/6f29a8c35abb8928d4e7ea01958192dd2a83491d) | fix | renamed files by their new path in the schematic workflow |
+| [201a036f2](https://github.com/angular/angular-cli/commit/201a036f204a6940f70a36a507a4a53d144b5768) | fix | simplify Angular version compatibility checks and add special handling for local builds of new major versions |
+| [cdd26bb66](https://github.com/angular/angular-cli/commit/cdd26bb66d8ab334f76323c2b5cae1aa8ce815f6) | fix | validate package manager version using `semver.valid` and throw an error if invalid |
+| [bc363af8b](https://github.com/angular/angular-cli/commit/bc363af8bc40f117a4e35ec9eb7eedf69f5b5b37) | perf | optimize package manager discovery with stat-based probing |
+
+### @schematics/angular
+
+| Commit | Type | Description |
+| --------------------------------------------------------------------------------------------------- | ---- | ----------------------------------------------------------------------------------- |
+| [aa7381efd](https://github.com/angular/angular-cli/commit/aa7381efd213eff70a8004731a7e2b06a60cb8c2) | feat | add a '.prettierrc' file to generated workspaces and add Prettier as dev dependency |
+| [f80db6fb7](https://github.com/angular/angular-cli/commit/f80db6fb714aa326f6ed03a8a51090ca59ad0955) | feat | add ng-add support for Vitest browser providers |
+| [5d1df50d8](https://github.com/angular/angular-cli/commit/5d1df50d8b84b453570ae5fd9ab6f949bbc11649) | fix | add actionable feedback to vitest-browser schematic |
+
+### @angular/build
+
+| Commit | Type | Description |
+| --------------------------------------------------------------------------------------------------- | ---- | -------------------------------------------------------- |
+| [ece30f235](https://github.com/angular/angular-cli/commit/ece30f2359c2dc794b0c9272447f623a121e88b0) | feat | add headless option to unit-test builder |
+| [cad7a7c0f](https://github.com/angular/angular-cli/commit/cad7a7c0ff3778f04820a99ad0aa9d74f1067fd5) | feat | run vitest browser with playwright with OS theme |
+| [0b4982720](https://github.com/angular/angular-cli/commit/0b4982720e111bf5029bcf97f7e0ce2658c42d43) | fix | adjust sourcemap sources when Vitest wrapper is bypassed |
+| [1f114a9e8](https://github.com/angular/angular-cli/commit/1f114a9e8b9bddd53e01016a2d7cb211a04eee48) | fix | bundle setup files in unit-test builder for Vitest |
+| [fd5cb28c8](https://github.com/angular/angular-cli/commit/fd5cb28c8082417288a896b89bde659bb0dc92e2) | fix | explicitly fail when using Vitest runtime mocking |
+| [dc899e8a5](https://github.com/angular/angular-cli/commit/dc899e8a530979de8e9579f2281b681e6f737a62) | fix | normalize `allowedHosts` in dev-server |
+| [26bbea12f](https://github.com/angular/angular-cli/commit/26bbea12f872c18e59de05d3c51cc11dd0a09cda) | fix | serve extensionless assets without transformation |
-
+
-# 21.2.0-next.2 (2026-02-11)
+# 19.2.20 (2026-02-13)
-### @angular/build
+### @angular-devkit/build-angular
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | ----------------------------------------------------- |
-| [cad7a7c0f](https://github.com/angular/angular-cli/commit/cad7a7c0ff3778f04820a99ad0aa9d74f1067fd5) | feat | run vitest browser with playwright with OS theme |
-| [8ae7f59e6](https://github.com/angular/angular-cli/commit/8ae7f59e6f988489fda8c1346e3d2c3768d7a5f0) | fix | correctly resolve absolute setup file paths in Vitest |
-| [fd5cb28c8](https://github.com/angular/angular-cli/commit/fd5cb28c8082417288a896b89bde659bb0dc92e2) | fix | explicitly fail when using Vitest runtime mocking |
+| Commit | Type | Description |
+| --------------------------------------------------------------------------------------------------- | ---- | ------------------------- |
+| [0e5421ba7](https://github.com/angular/angular-cli/commit/0e5421ba78814cf11e4d4510e930eaacc6458662) | fix | update webpack to 5.105.0 |
@@ -48,34 +76,6 @@
-
-
-# 21.2.0-next.1 (2026-02-05)
-
-### @angular/cli
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | ---------------------------------------------------------- |
-| [91b9d281f](https://github.com/angular/angular-cli/commit/91b9d281fc88a242aa6e5dd5495e275990d926ef) | feat | integrate file formatting into update migrations |
-| [6f29a8c35](https://github.com/angular/angular-cli/commit/6f29a8c35abb8928d4e7ea01958192dd2a83491d) | fix | renamed files by their new path in the schematic workflow |
-| [bc363af8b](https://github.com/angular/angular-cli/commit/bc363af8bc40f117a4e35ec9eb7eedf69f5b5b37) | perf | optimize package manager discovery with stat-based probing |
-
-### @schematics/angular
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | ---------------------------------------------------------------- |
-| [5d1df50d8](https://github.com/angular/angular-cli/commit/5d1df50d8b84b453570ae5fd9ab6f949bbc11649) | fix | add actionable feedback to vitest-browser schematic |
-| [51fc77828](https://github.com/angular/angular-cli/commit/51fc77828a33fdf35051b7e18d79ad43f90cba1d) | fix | warn when production configuration is missing for service worker |
-
-### @angular/build
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | -------------------------------------------------- |
-| [ece30f235](https://github.com/angular/angular-cli/commit/ece30f2359c2dc794b0c9272447f623a121e88b0) | feat | add headless option to unit-test builder |
-| [1f114a9e8](https://github.com/angular/angular-cli/commit/1f114a9e8b9bddd53e01016a2d7cb211a04eee48) | fix | bundle setup files in unit-test builder for Vitest |
-
-
-
# 21.1.3 (2026-02-05)
@@ -94,41 +94,6 @@
-
-
-# 21.2.0-next.0 (2026-01-28)
-
-### @angular/cli
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | --------------------------------------------------------------- |
-| [0dd04f289](https://github.com/angular/angular-cli/commit/0dd04f289e555a4a8af7bdadabe300da74701e3b) | feat | add markdown files to Prettier's formatting list |
-| [fbae1b6ab](https://github.com/angular/angular-cli/commit/fbae1b6ab384186ae69e804c54815cea80e6a600) | feat | automatic formatting files modified by schematics |
-| [98a24d040](https://github.com/angular/angular-cli/commit/98a24d0401f36f484dc9c4d8b0f5284ffa524f19) | feat | standardize MCP tools around workspace/project options |
-| [d9cd609c5](https://github.com/angular/angular-cli/commit/d9cd609c5d13fe492b1f31973d9be518f8529387) | fix | correctly parse scoped packages in yarn classic list output |
-| [5b05f2500](https://github.com/angular/angular-cli/commit/5b05f25005621828565585692b1d7a67c5f0fec8) | fix | enable shell option for Prettier execution on Windows platforms |
-
-### @schematics/angular
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | ----------------------------------------------------------------------------------- |
-| [aa7381efd](https://github.com/angular/angular-cli/commit/aa7381efd213eff70a8004731a7e2b06a60cb8c2) | feat | add a '.prettierrc' file to generated workspaces and add Prettier as dev dependency |
-| [f80db6fb7](https://github.com/angular/angular-cli/commit/f80db6fb714aa326f6ed03a8a51090ca59ad0955) | feat | add ng-add support for Vitest browser providers |
-
-### @angular-devkit/build-angular
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | ----------------------------------- |
-| [b4a8d198c](https://github.com/angular/angular-cli/commit/b4a8d198c78aaf0cac7671f26162ce5818a5704c) | fix | address Node.js deprecation DEP0190 |
-
-### @angular/build
-
-| Commit | Type | Description |
-| --------------------------------------------------------------------------------------------------- | ---- | -------------------------------------------------------- |
-| [0b4982720](https://github.com/angular/angular-cli/commit/0b4982720e111bf5029bcf97f7e0ce2658c42d43) | fix | adjust sourcemap sources when Vitest wrapper is bypassed |
-
-
-
# 21.1.2 (2026-01-28)
@@ -2401,7 +2366,6 @@
- Protractor is no longer supported.
Protractor was marked end-of-life in August 2023 (see https://protractortest.org/). Projects still relying on Protractor should consider migrating to another E2E testing framework, several support solid migration paths from Protractor.
-
- https://angular.dev/tools/cli/end-to-end
- https://blog.angular.dev/the-state-of-end-to-end-testing-with-angular-d175f751cb9c
@@ -6036,7 +6000,6 @@ Alan Agius, Charles Lyding and Doug Parker
### @angular/cli
- Several changes to the `ng analytics` command syntax.
-
- `ng analytics project ` has been replaced with `ng analytics `
- `ng analytics ` has been replaced with `ng analytics --global`
@@ -6066,7 +6029,6 @@ Alan Agius, Charles Lyding and Doug Parker
- `browser` and `karma` builders `script` and `styles` options input files extensions are now validated.
Valid extensions for `scripts` are:
-
- `.js`
- `.cjs`
- `.mjs`
@@ -6075,7 +6037,6 @@ Alan Agius, Charles Lyding and Doug Parker
- `.mjsx`
Valid extensions for `styles` are:
-
- `.css`
- `.less`
- `.sass`
diff --git a/MODULE.bazel b/MODULE.bazel
index e218349c3d69..07ebd69baf1e 100644
--- a/MODULE.bazel
+++ b/MODULE.bazel
@@ -5,27 +5,27 @@ module(
)
bazel_dep(name = "platforms", version = "1.0.0")
-bazel_dep(name = "yq.bzl", version = "0.3.4")
+bazel_dep(name = "yq.bzl", version = "0.3.5")
bazel_dep(name = "rules_nodejs", version = "6.7.3")
bazel_dep(name = "aspect_rules_js", version = "2.9.2")
-bazel_dep(name = "aspect_rules_ts", version = "3.8.4")
+bazel_dep(name = "aspect_rules_ts", version = "3.8.5")
bazel_dep(name = "rules_pkg", version = "1.2.0")
-bazel_dep(name = "rules_cc", version = "0.2.16")
+bazel_dep(name = "rules_cc", version = "0.2.17")
bazel_dep(name = "aspect_bazel_lib", version = "2.22.5")
bazel_dep(name = "bazel_skylib", version = "1.9.0")
bazel_dep(name = "aspect_rules_esbuild", version = "0.25.0")
-bazel_dep(name = "aspect_rules_jasmine", version = "2.0.2")
+bazel_dep(name = "aspect_rules_jasmine", version = "2.0.3")
bazel_dep(name = "rules_angular")
git_override(
module_name = "rules_angular",
commit = "d746c4f75e42cffe389d1ab077f4639be2bc78d1",
- remote = "https://github.com/devversion/rules_angular.git",
+ remote = "https://github.com/angular/rules_angular.git",
)
bazel_dep(name = "devinfra")
git_override(
module_name = "devinfra",
- commit = "469708f109a90884ca403d150d33079a3a5a8769",
+ commit = "9cc477855b9788df6257301074a1629bc3545722",
remote = "https://github.com/angular/dev-infra.git",
)
@@ -33,14 +33,14 @@ bazel_dep(name = "rules_sass")
git_override(
module_name = "rules_sass",
commit = "1184a80751a21af8348f308abc5b38a41f26850e",
- remote = "https://github.com/devversion/rules_sass.git",
+ remote = "https://github.com/angular/rules_sass.git",
)
bazel_dep(name = "rules_browsers")
git_override(
module_name = "rules_browsers",
- commit = "e08ae33c679d07b3b2fcc136658b787a81995bc5",
- remote = "https://github.com/devversion/rules_browsers.git",
+ commit = "8231142fc5516323b040883bf774ba8362645387",
+ remote = "https://github.com/angular/rules_browsers.git",
)
node = use_extension("@rules_nodejs//nodejs:extensions.bzl", "node")
@@ -110,8 +110,8 @@ use_repo(
pnpm = use_extension("@aspect_rules_js//npm:extensions.bzl", "pnpm")
pnpm.pnpm(
name = "pnpm",
- pnpm_version = "10.29.3",
- pnpm_version_integrity = "sha512-SY4ftMylqgbB3PJhHm+vxQly/+cYmZjECekN50VmREKY/+Q+bNKs3Hdboap8xeCSqLcFTIEbqMV3D4RpPTPS3A==",
+ pnpm_version = "10.30.2",
+ pnpm_version_integrity = "sha512-Ns3HB+e3lAqYjJwez4jQhPhRS1w/CF9TouJEwpIdOyVFvCDdTr4fwkX+7EY7spiuzqemPtH3aAuHfcY3nY0MtA==",
)
use_repo(pnpm, "pnpm")
diff --git a/MODULE.bazel.lock b/MODULE.bazel.lock
index 8af79006afc6..0c44f36a633a 100644
--- a/MODULE.bazel.lock
+++ b/MODULE.bazel.lock
@@ -21,8 +21,8 @@
"https://bcr.bazel.build/modules/aspect_bazel_lib/2.9.3/MODULE.bazel": "66baf724dbae7aff4787bf2245cc188d50cb08e07789769730151c0943587c14",
"https://bcr.bazel.build/modules/aspect_rules_esbuild/0.25.0/MODULE.bazel": "5fef5ec709c837312823f9bcf0f276661e2cb48ad52f17c4e01176bbf1e9bf58",
"https://bcr.bazel.build/modules/aspect_rules_esbuild/0.25.0/source.json": "5e42968c6d23ab8bd95c02634b16864d866334347827cb6a8425b86c11cc4363",
- "https://bcr.bazel.build/modules/aspect_rules_jasmine/2.0.2/MODULE.bazel": "45f054400ff242c4433f6d7f20f6123a9a72739cb7a1f44247d738db1644f46c",
- "https://bcr.bazel.build/modules/aspect_rules_jasmine/2.0.2/source.json": "3ed399a5654259a822448f9cdbf21f6c738f16ccd7f89249c7507e374cbdd1e3",
+ "https://bcr.bazel.build/modules/aspect_rules_jasmine/2.0.3/MODULE.bazel": "1c2e7a2908dbf6640dce9b242369cf48b7018af666485cbae1470b49a9672591",
+ "https://bcr.bazel.build/modules/aspect_rules_jasmine/2.0.3/source.json": "5619d28f3e99eca53a0ae077cff6c66966d2da2ee8621bc0542af3fb85df128f",
"https://bcr.bazel.build/modules/aspect_rules_js/2.0.0/MODULE.bazel": "b45b507574aa60a92796e3e13c195cd5744b3b8aff516a9c0cb5ae6a048161c5",
"https://bcr.bazel.build/modules/aspect_rules_js/2.4.2/MODULE.bazel": "0d01db38b96d25df7ed952a5e96eac4b3802723d146961974bf020f6dd07591d",
"https://bcr.bazel.build/modules/aspect_rules_js/2.6.2/MODULE.bazel": "ed2a871f4ab8fbde0cab67c425745069d84ea64b64313fa1a2954017326511f5",
@@ -30,9 +30,8 @@
"https://bcr.bazel.build/modules/aspect_rules_js/2.9.2/source.json": "4bff7c03ab387b60deb15649ba575688e62f2a71a7544cbc7a660b19ec473808",
"https://bcr.bazel.build/modules/aspect_rules_ts/3.6.3/MODULE.bazel": "d09db394970f076176ce7bab5b5fa7f0d560fd4f30b8432ea5e2c2570505b130",
"https://bcr.bazel.build/modules/aspect_rules_ts/3.7.0/MODULE.bazel": "5aace216caf88638950ef061245d23c36f57c8359e56e97f02a36f70bb09c50f",
- "https://bcr.bazel.build/modules/aspect_rules_ts/3.8.3/MODULE.bazel": "a26c28ebcd0c0d50ab0708ac21fa48bd2dced3a4dad4c31a2fa48588b42ad762",
- "https://bcr.bazel.build/modules/aspect_rules_ts/3.8.4/MODULE.bazel": "a50254ac3add6232d0f9f93103836f9afaf614315589a13abf74183982c4101d",
- "https://bcr.bazel.build/modules/aspect_rules_ts/3.8.4/source.json": "f786e0763f3ea5de7ea6d4c4e38fccb48bf4d9c5eafaf95091c0e1590502510e",
+ "https://bcr.bazel.build/modules/aspect_rules_ts/3.8.5/MODULE.bazel": "bcf8f0b6b9375f0f74451e2f70671efae9bb366acef8fdc04675305eaf137f06",
+ "https://bcr.bazel.build/modules/aspect_rules_ts/3.8.5/source.json": "fa35e43f6359f654e4b70ce55efdf280d0b06c0b3ef9fc0b06ba52327a0e6311",
"https://bcr.bazel.build/modules/aspect_tools_telemetry/0.2.3/MODULE.bazel": "20f53b145f40957a51077ae90b37b7ce83582a1daf9350349f0f86179e19dd0d",
"https://bcr.bazel.build/modules/aspect_tools_telemetry/0.2.6/MODULE.bazel": "cafb8781ad591bc57cc765dca5fefab08cf9f65af363d162b79d49205c7f8af7",
"https://bcr.bazel.build/modules/aspect_tools_telemetry/0.2.8/MODULE.bazel": "aa975a83e72bcaac62ee61ab12b788ea324a1d05c4aab28aadb202f647881679",
@@ -75,7 +74,8 @@
"https://bcr.bazel.build/modules/buildozer/7.1.2/MODULE.bazel": "2e8dd40ede9c454042645fd8d8d0cd1527966aa5c919de86661e62953cd73d84",
"https://bcr.bazel.build/modules/buildozer/7.1.2/source.json": "c9028a501d2db85793a6996205c8de120944f50a0d570438fcae0457a5f9d1f8",
"https://bcr.bazel.build/modules/gawk/5.3.2.bcr.1/MODULE.bazel": "cdf8cbe5ee750db04b78878c9633cc76e80dcf4416cbe982ac3a9222f80713c8",
- "https://bcr.bazel.build/modules/gawk/5.3.2.bcr.1/source.json": "fa7b512dfcb5eafd90ce3959cf42a2a6fe96144ebbb4b3b3928054895f2afac2",
+ "https://bcr.bazel.build/modules/gawk/5.3.2.bcr.3/MODULE.bazel": "f1b7bb2dd53e8f2ef984b39485ec8a44e9076dda5c4b8efd2fb4c6a6e856a31d",
+ "https://bcr.bazel.build/modules/gawk/5.3.2.bcr.3/source.json": "ebe931bfe362e4b41e59ee00a528db6074157ff2ced92eb9e970acab2e1089c9",
"https://bcr.bazel.build/modules/google_benchmark/1.8.2/MODULE.bazel": "a70cf1bba851000ba93b58ae2f6d76490a9feb74192e57ab8e8ff13c34ec50cb",
"https://bcr.bazel.build/modules/googletest/1.11.0/MODULE.bazel": "3a83f095183f66345ca86aa13c58b59f9f94a2f81999c093d4eeaa2d262d12f4",
"https://bcr.bazel.build/modules/googletest/1.14.0.bcr.1/MODULE.bazel": "22c31a561553727960057361aa33bf20fb2e98584bc4fec007906e27053f80c6",
@@ -123,7 +123,9 @@
"https://bcr.bazel.build/modules/rules_cc/0.0.9/MODULE.bazel": "836e76439f354b89afe6a911a7adf59a6b2518fafb174483ad78a2a2fde7b1c5",
"https://bcr.bazel.build/modules/rules_cc/0.1.1/MODULE.bazel": "2f0222a6f229f0bf44cd711dc13c858dad98c62d52bd51d8fc3a764a83125513",
"https://bcr.bazel.build/modules/rules_cc/0.2.16/MODULE.bazel": "9242fa89f950c6ef7702801ab53922e99c69b02310c39fb6e62b2bd30df2a1d4",
- "https://bcr.bazel.build/modules/rules_cc/0.2.16/source.json": "d03d5cde49376d87e14ec14b666c56075e5e3926930327fd5d0484a1ff2ac1cc",
+ "https://bcr.bazel.build/modules/rules_cc/0.2.17/MODULE.bazel": "1849602c86cb60da8613d2de887f9566a6d354a6df6d7009f9d04a14402f9a84",
+ "https://bcr.bazel.build/modules/rules_cc/0.2.17/source.json": "3832f45d145354049137c0090df04629d9c2b5493dc5c2bf46f1834040133a07",
+ "https://bcr.bazel.build/modules/rules_cc/0.2.4/MODULE.bazel": "1ff1223dfd24f3ecf8f028446d4a27608aa43c3f41e346d22838a4223980b8cc",
"https://bcr.bazel.build/modules/rules_foreign_cc/0.9.0/MODULE.bazel": "c9e8c682bf75b0e7c704166d79b599f93b72cfca5ad7477df596947891feeef6",
"https://bcr.bazel.build/modules/rules_fuzzing/0.5.2/MODULE.bazel": "40c97d1144356f52905566c55811f13b299453a14ac7769dfba2ac38192337a8",
"https://bcr.bazel.build/modules/rules_fuzzing/0.5.2/source.json": "c8b1e2c717646f1702290959a3302a178fb639d987ab61d548105019f11e527e",
@@ -196,14 +198,14 @@
"https://bcr.bazel.build/modules/stardoc/0.7.2/source.json": "58b029e5e901d6802967754adf0a9056747e8176f017cfe3607c0851f4d42216",
"https://bcr.bazel.build/modules/tar.bzl/0.2.1/MODULE.bazel": "52d1c00a80a8cc67acbd01649e83d8dd6a9dc426a6c0b754a04fe8c219c76468",
"https://bcr.bazel.build/modules/tar.bzl/0.5.1/MODULE.bazel": "7c2eb3dcfc53b0f3d6f9acdfd911ca803eaf92aadf54f8ca6e4c1f3aee288351",
- "https://bcr.bazel.build/modules/tar.bzl/0.8.1/MODULE.bazel": "6ffe8907ed4c555bc94bd35a5a01411cc4470c6dace84f9cf487815409e077d1",
- "https://bcr.bazel.build/modules/tar.bzl/0.8.1/source.json": "835f83b482facf6205ad8708cf2b2f6524d1d7b1075a90fe9bb540da761d6d2e",
+ "https://bcr.bazel.build/modules/tar.bzl/0.9.0/MODULE.bazel": "452a22d7f02b1c9d7a22ab25edf20f46f3e1101f0f67dc4bfbf9a474ddf02445",
+ "https://bcr.bazel.build/modules/tar.bzl/0.9.0/source.json": "c732760a374831a2cf5b08839e4be75017196b4d796a5aa55235272ee17cd839",
"https://bcr.bazel.build/modules/upb/0.0.0-20220923-a547704/MODULE.bazel": "7298990c00040a0e2f121f6c32544bab27d4452f80d9ce51349b1a28f3005c43",
"https://bcr.bazel.build/modules/yq.bzl/0.1.1/MODULE.bazel": "9039681f9bcb8958ee2c87ffc74bdafba9f4369096a2b5634b88abc0eaefa072",
"https://bcr.bazel.build/modules/yq.bzl/0.2.0/MODULE.bazel": "6f3a675677db8885be4d607fde14cc51829715e3a879fb016eb9bf336786ce6d",
"https://bcr.bazel.build/modules/yq.bzl/0.3.2/MODULE.bazel": "0384efa70e8033d842ea73aa4b7199fa099709e236a7264345c03937166670b6",
- "https://bcr.bazel.build/modules/yq.bzl/0.3.4/MODULE.bazel": "d3a270662f5d766cd7229732d65a5a5bc485240c3007343dd279edfb60c9ae27",
- "https://bcr.bazel.build/modules/yq.bzl/0.3.4/source.json": "786dafdc2843722da3416e4343ee1a05237227f068590779a6e8496a2064c0f9",
+ "https://bcr.bazel.build/modules/yq.bzl/0.3.5/MODULE.bazel": "130c603e54be717bdf84100210f06598a0d2b4b4e01888fb01b70f50f41767ec",
+ "https://bcr.bazel.build/modules/yq.bzl/0.3.5/source.json": "1ae7bdc03cb26aaa8bd2bceadf65e90d90f0b2d03008ba9a0564da2e21396c39",
"https://bcr.bazel.build/modules/zlib/1.2.11/MODULE.bazel": "07b389abc85fdbca459b69e2ec656ae5622873af3f845e1c9d80fe179f3effa0",
"https://bcr.bazel.build/modules/zlib/1.3.1.bcr.5/MODULE.bazel": "eec517b5bbe5492629466e11dae908d043364302283de25581e3eb944326c4ca",
"https://bcr.bazel.build/modules/zlib/1.3.1.bcr.5/source.json": "22bc55c47af97246cfc093d0acf683a7869377de362b5d1c552c2c2e16b7a806",
@@ -213,7 +215,7 @@
"moduleExtensions": {
"@@aspect_rules_esbuild+//esbuild:extensions.bzl%esbuild": {
"general": {
- "bzlTransitiveDigest": "c4i5gawrp4Au9UMb55EzQCePYwkrFqD9tFBN7GdHG5g=",
+ "bzlTransitiveDigest": "fzdwsWU1Px4bYL57Cw1rvU/mgOqB7f0AF51JxT5kAs8=",
"usagesDigest": "ToTaCONCN/E05krnHXLM1kpV1zrHNxHrGpUip973II4=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
@@ -414,8 +416,8 @@
},
"@@aspect_rules_js+//npm:extensions.bzl%pnpm": {
"general": {
- "bzlTransitiveDigest": "HC+l+mTivq1p/KbcVQ+iV5QwYR+oKESJh827FY68SH8=",
- "usagesDigest": "PvqSdyUvIknVzZ66q+9FjDqiPWbKoaSj5J3EB+Z3ZAs=",
+ "bzlTransitiveDigest": "9Zmqluz9ltvCoc6pskOslTkQpkH6CIINjWVkgam9aa4=",
+ "usagesDigest": "ZvjRdptHQkMA7CP3Q06p9iboasgFVjpW2zKBrKW0xCA=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
"envVariables": {},
@@ -424,11 +426,11 @@
"repoRuleId": "@@aspect_rules_js+//npm/private:npm_import.bzl%npm_import_rule",
"attributes": {
"package": "pnpm",
- "version": "10.29.3",
+ "version": "10.30.2",
"root_package": "",
"link_workspace": "",
"link_packages": {},
- "integrity": "sha512-SY4ftMylqgbB3PJhHm+vxQly/+cYmZjECekN50VmREKY/+Q+bNKs3Hdboap8xeCSqLcFTIEbqMV3D4RpPTPS3A==",
+ "integrity": "sha512-Ns3HB+e3lAqYjJwez4jQhPhRS1w/CF9TouJEwpIdOyVFvCDdTr4fwkX+7EY7spiuzqemPtH3aAuHfcY3nY0MtA==",
"url": "",
"commit": "",
"patch_args": [
@@ -451,7 +453,7 @@
"repoRuleId": "@@aspect_rules_js+//npm/private:npm_import.bzl%npm_import_links",
"attributes": {
"package": "pnpm",
- "version": "10.29.3",
+ "version": "10.30.2",
"dev": false,
"root_package": "",
"link_packages": {},
@@ -568,10 +570,10 @@
},
"@@aspect_rules_ts+//ts:extensions.bzl%ext": {
"general": {
- "bzlTransitiveDigest": "QDTi1Wl/eEY4IgbXjRhegUQfHj+bB8ZEVyiSGLZc6qo=",
- "usagesDigest": "aaqqxEFKCRGFkeAf0pKmXvZZTLGYIk3pQsDFG28ZbNg=",
+ "bzlTransitiveDigest": "6NyLUdrb79stFKm8oWIBK2q65cT5Upbr8IR2Qwsywu8=",
+ "usagesDigest": "ohKYRsVTeRZRfv+0QiN+5H2ycAbbirey745XNBkWqkk=",
"recordedFileInputs": {
- "@@rules_browsers+//package.json": "84dc1ba9b1c667a25894e97218bd8f247d54f24bb694efb397a881be3c06a4c5"
+ "@@rules_browsers+//package.json": "772d873d450a539e2133635aeb5e63744cf1cec86e6b37aeecd9267a147fb0d7"
},
"recordedDirentsInputs": {},
"envVariables": {},
@@ -635,7 +637,7 @@
"@@aspect_tools_telemetry+//:extension.bzl%telemetry": {
"general": {
"bzlTransitiveDigest": "cl5A2O84vDL6Tt+Qga8FCj1DUDGqn+e7ly5rZ+4xvcc=",
- "usagesDigest": "PRLGsERE1Dznyx/OIAl7BPo8mzMvOklnNnZ8zdCpPTE=",
+ "usagesDigest": "/e/KYM5sjJPlgbDVQclkrYIlSJA/gt68f4vYfSZSvkQ=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
"envVariables": {},
@@ -645,9 +647,9 @@
"attributes": {
"deps": {
"aspect_rules_js": "2.9.2",
- "aspect_rules_ts": "3.8.4",
+ "aspect_rules_ts": "3.8.5",
"aspect_rules_esbuild": "0.25.0",
- "aspect_rules_jasmine": "2.0.2",
+ "aspect_rules_jasmine": "2.0.3",
"aspect_tools_telemetry": "0.3.3"
}
}
@@ -669,7 +671,7 @@
},
"@@pybind11_bazel+//:python_configure.bzl%extension": {
"general": {
- "bzlTransitiveDigest": "c9ZWWeXeu6bctL4/SsY2otFWyeFN0JJ20+ymGyJZtWk=",
+ "bzlTransitiveDigest": "D2/qWHU6yQFwRG7Bb+caqrYMha5avsASao2vERrxK24=",
"usagesDigest": "fycyB39YnXIJkfWCIXLUKJMZzANcuLy9ZE73hRucjFk=",
"recordedFileInputs": {
"@@pybind11_bazel+//MODULE.bazel": "88af1c246226d87e65be78ed49ecd1e6f5e98648558c14ce99176da041dc378e"
@@ -736,8 +738,8 @@
},
"@@rules_browsers+//browsers:extensions.bzl%browsers": {
"general": {
- "bzlTransitiveDigest": "agkaLQ8wE1r/5IX6pkERzFxI/z0M42Em+ICNO6TXsVo=",
- "usagesDigest": "FS7q5WaIwg3KirS3njhuPFkTIBYvDaTInVGrlzu0XL8=",
+ "bzlTransitiveDigest": "Bm6fiKpWy96aLohOlLCP36ARVxRLZm/R+smhsb2HzmI=",
+ "usagesDigest": "FmXYJVoVJlnfUU8x8gObSvu4qWcco/9Faw61aC/wBF0=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
"envVariables": {},
@@ -745,9 +747,9 @@
"rules_browsers_chrome_linux": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "0a2ff0fc9eb5958b7b420f20e3968f424be7423fef89739e71565a48aa073a57",
+ "sha256": "1ac33f89306327af43be159c03ca4a26486de0858f42fe52394acdef50364143",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/linux64/chrome-headless-shell-linux64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/linux64/chrome-headless-shell-linux64.zip"
],
"named_files": {
"CHROME-HEADLESS-SHELL": "chrome-headless-shell-linux64/chrome-headless-shell"
@@ -763,9 +765,9 @@
"rules_browsers_chrome_mac": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "e6076b1201d86f74c5eab982a239d5af83e66b1aa4d780bcb792698790e01d87",
+ "sha256": "169ff49c465cfda52931395e61861e146dfc5013e92c01ca792db5acea858d0b",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/mac-x64/chrome-headless-shell-mac-x64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/mac-x64/chrome-headless-shell-mac-x64.zip"
],
"named_files": {
"CHROME-HEADLESS-SHELL": "chrome-headless-shell-mac-x64/chrome-headless-shell"
@@ -781,9 +783,9 @@
"rules_browsers_chrome_mac_arm": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "b74dbcf5543d916b02d0a133e2e7c6a4de251f06733f72c2c15ea8c42213f63b",
+ "sha256": "aeaaaaa4d68193a21bed04c44ddeb1230232707b4ea1d845a92925787509cd8e",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/mac-arm64/chrome-headless-shell-mac-arm64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/mac-arm64/chrome-headless-shell-mac-arm64.zip"
],
"named_files": {
"CHROME-HEADLESS-SHELL": "chrome-headless-shell-mac-arm64/chrome-headless-shell"
@@ -799,9 +801,9 @@
"rules_browsers_chrome_win64": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "df1e612dc3b1615e182a1f11821052995913c39df37caa52699de21a68d030d2",
+ "sha256": "4d6d79bcbcb22084df6e3a3d3a2caff67d6c0fa488d63f0c7ec1526f9553db8c",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/win64/chrome-headless-shell-win64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/win64/chrome-headless-shell-win64.zip"
],
"named_files": {
"CHROME-HEADLESS-SHELL": "chrome-headless-shell-win64/chrome-headless-shell.exe"
@@ -817,9 +819,9 @@
"rules_browsers_chromedriver_linux": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "69c504306399d979a2766fea603c3fb9d3d87d46c75bddc9f2a049b4f636d57c",
+ "sha256": "0607ccf6810a07ae08cac6443beac8b23f88dd53c7f1e0299e22d65f7cd2d020",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/linux64/chromedriver-linux64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/linux64/chromedriver-linux64.zip"
],
"named_files": {
"CHROMEDRIVER": "chromedriver-linux64/chromedriver"
@@ -833,9 +835,9 @@
"rules_browsers_chromedriver_mac": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "5fc9d6f594fc5f2568a15145f25116dd8e9c9a60baa8da4bb21a17650fb00e7e",
+ "sha256": "0f512a9dd683ed4c41e609d8d02c07807497dbad3ab2f95f0d583486be7b8cff",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/mac-x64/chromedriver-mac-x64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/mac-x64/chromedriver-mac-x64.zip"
],
"named_files": {
"CHROMEDRIVER": "chromedriver-mac-x64/chromedriver"
@@ -849,9 +851,9 @@
"rules_browsers_chromedriver_mac_arm": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "14e92294c2c3639ca4e7d27e850588b619d698e2f8905cee368f07db2e1bf1e9",
+ "sha256": "7d6fc6d17de1733eb6739d1ea16d085c8df1568bcf9fa0d130c2784b27f38268",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/mac-arm64/chromedriver-mac-arm64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/mac-arm64/chromedriver-mac-arm64.zip"
],
"named_files": {
"CHROMEDRIVER": "chromedriver-mac-arm64/chromedriver"
@@ -865,9 +867,9 @@
"rules_browsers_chromedriver_win64": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "cf641d2e176db95bcc158cd90eafd347ad4928fa0458a5f3bfd56c6d983e70db",
+ "sha256": "f4e9fb7bbf692fde7979b24e8d737b3cef4baafbc7a370e5d0abc4a8450fd830",
"urls": [
- "https://storage.googleapis.com/chrome-for-testing-public/145.0.7586.0/win64/chromedriver-win64.zip"
+ "https://storage.googleapis.com/chrome-for-testing-public/147.0.7687.0/win64/chromedriver-win64.zip"
],
"named_files": {
"CHROMEDRIVER": "chromedriver-win64/chromedriver.exe"
@@ -881,9 +883,9 @@
"rules_browsers_firefox_linux": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "8d56f479cc398a537a60a3fa20dca92d8a41925113d3a67f534881a4e4d7e344",
+ "sha256": "f055b9c0d7346a10d22edc7f10e08679af2ea495367381ab2be9cab3ec6add97",
"urls": [
- "https://archive.mozilla.org/pub/firefox/releases/146.0/linux-x86_64/en-US/firefox-146.0.tar.xz"
+ "https://archive.mozilla.org/pub/firefox/releases/147.0/linux-x86_64/en-US/firefox-147.0.tar.xz"
],
"named_files": {
"FIREFOX": "firefox/firefox"
@@ -897,9 +899,9 @@
"rules_browsers_firefox_mac": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "4b1645313887972d466cd82166ea571485c2c40a167f84624e3f3ca739993cc9",
+ "sha256": "48485e2068bc726e2f30cf5855fc2da1fc75c1272bc243a5394f428ffae3ba35",
"urls": [
- "https://archive.mozilla.org/pub/firefox/releases/146.0/mac/en-US/Firefox%20146.0.dmg"
+ "https://archive.mozilla.org/pub/firefox/releases/147.0/mac/en-US/Firefox%20147.0.dmg"
],
"named_files": {
"FIREFOX": "Firefox.app/Contents/MacOS/firefox"
@@ -913,9 +915,9 @@
"rules_browsers_firefox_mac_arm": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "4b1645313887972d466cd82166ea571485c2c40a167f84624e3f3ca739993cc9",
+ "sha256": "48485e2068bc726e2f30cf5855fc2da1fc75c1272bc243a5394f428ffae3ba35",
"urls": [
- "https://archive.mozilla.org/pub/firefox/releases/146.0/mac/en-US/Firefox%20146.0.dmg"
+ "https://archive.mozilla.org/pub/firefox/releases/147.0/mac/en-US/Firefox%20147.0.dmg"
],
"named_files": {
"FIREFOX": "Firefox.app/Contents/MacOS/firefox"
@@ -929,9 +931,9 @@
"rules_browsers_firefox_win64": {
"repoRuleId": "@@rules_browsers+//browsers/private:browser_repo.bzl%browser_repo",
"attributes": {
- "sha256": "216870c89648f32450cfefb5cec417fcd66d480d5dc83f894bf99f5fd7f38dbb",
+ "sha256": "36ff9e150875aa48a0af9eec3eb67f66dddd8efac5c743265371a72ae3e796c4",
"urls": [
- "https://archive.mozilla.org/pub/firefox/releases/146.0/win64/en-US/Firefox%20Setup%20146.0.exe"
+ "https://archive.mozilla.org/pub/firefox/releases/147.0/win64/en-US/Firefox%20Setup%20147.0.exe"
],
"named_files": {
"FIREFOX": "core/firefox.exe"
@@ -948,7 +950,7 @@
},
"@@rules_fuzzing+//fuzzing/private:extensions.bzl%non_module_dependencies": {
"general": {
- "bzlTransitiveDigest": "WHRlQQnxW7e7XMRBhq7SARkDarLDOAbg6iLaJpk5QYM=",
+ "bzlTransitiveDigest": "4LouzhF/yT117s7peGnNs9ROomiJXC6Zl5R0oI21jho=",
"usagesDigest": "wy6ISK6UOcBEjj/mvJ/S3WeXoO67X+1llb9yPyFtPgc=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
@@ -1031,7 +1033,7 @@
},
"@@rules_kotlin+//src/main/starlark/core/repositories:bzlmod_setup.bzl%rules_kotlin_extensions": {
"general": {
- "bzlTransitiveDigest": "rL/34P1aFDq2GqVC2zCFgQ8nTuOC6ziogocpvG50Qz8=",
+ "bzlTransitiveDigest": "nvW/NrBXlAmiQw99EMGKkLaD2KbNp2mQDlxdfpr+0Ls=",
"usagesDigest": "QI2z8ZUR+mqtbwsf2fLqYdJAkPOHdOV+tF2yVAUgRzw=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
@@ -1096,7 +1098,7 @@
"@@rules_nodejs+//nodejs:extensions.bzl%node": {
"general": {
"bzlTransitiveDigest": "4pUxCNc22K4I+6+4Nxu52Hur12tFRfa1JMsN5mdDv60=",
- "usagesDigest": "6UAmdIABVpqhlkQ3A3NGscf00ds9dFEt+lei3DibyqM=",
+ "usagesDigest": "quEcEHfGUyYHUdxVbXCKK5PPlFD8GM/jDelnql58cSw=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
"envVariables": {},
@@ -1883,7 +1885,7 @@
},
"@@rules_python+//python/extensions:pip.bzl%pip": {
"general": {
- "bzlTransitiveDigest": "d3ENjFH8qMwmOrkcb3c9JYqQ5hJ6owjfbSr24KY0Ugg=",
+ "bzlTransitiveDigest": "LFkevJ0u48WgTVCyU07LkRD8iYwsRUs2M7uSFHkQAW0=",
"usagesDigest": "AK1R124YPWwAs8z1CQYyjYuci8RO5Ofot+EP5ZCNQDc=",
"recordedFileInputs": {
"@@protobuf+//python/requirements.txt": "983be60d3cec4b319dcab6d48aeb3f5b2f7c3350f26b3a9e97486c37967c73c5",
@@ -4677,8 +4679,8 @@
},
"@@yq.bzl+//yq:extensions.bzl%yq": {
"general": {
- "bzlTransitiveDigest": "tDqk+ntWTdxNAWPDjRY1uITgHbti2jcXR5ZdinltBs0=",
- "usagesDigest": "OQwtwmKiZAvI0n0B86XlM4tmQHq4zcjFjAEiRGPhXVI=",
+ "bzlTransitiveDigest": "UfFMy8CWK4/dVo/tfaSAIYUiDGNAPes5eRllx9O9Q9Q=",
+ "usagesDigest": "OKtzWFv/1OIvlyc7JZVttov74lm6Ldv/V6oOd4SGa1U=",
"recordedFileInputs": {},
"recordedDirentsInputs": {},
"envVariables": {},
diff --git a/constants.bzl b/constants.bzl
index d4be4fc34b84..d9a42e40631f 100644
--- a/constants.bzl
+++ b/constants.bzl
@@ -3,10 +3,10 @@ RELEASE_ENGINES_NODE = "^20.19.0 || ^22.12.0 || >=24.0.0"
RELEASE_ENGINES_NPM = "^6.11.0 || ^7.5.6 || >=8.0.0"
RELEASE_ENGINES_YARN = ">= 1.13.0"
-NG_PACKAGR_VERSION = "^21.2.0-next.0"
-ANGULAR_FW_VERSION = "^21.2.0-next.0"
-ANGULAR_FW_PEER_DEP = "^21.0.0 || ^21.2.0-next.0"
-NG_PACKAGR_PEER_DEP = "^21.0.0 || ^21.2.0-next.0"
+NG_PACKAGR_VERSION = "^21.2.0"
+ANGULAR_FW_VERSION = "^21.2.0"
+ANGULAR_FW_PEER_DEP = "^21.0.0"
+NG_PACKAGR_PEER_DEP = "^21.0.0"
# Baseline widely-available date in `YYYY-MM-DD` format which defines Angular's
# browser support. This date serves as the source of truth for the Angular CLI's
diff --git a/docs/DEVELOPER.md b/docs/DEVELOPER.md
index 8204bd7dcbce..075ddbcf9ca3 100644
--- a/docs/DEVELOPER.md
+++ b/docs/DEVELOPER.md
@@ -56,7 +56,7 @@ project](#building-and-installing-the-cli), then run the desired `ng` command
as:
```shell
-node --inspect-brk node_modules/.bin/ng ...
+node --inspect-brk node_modules/.bin/ng
```
This will trigger a breakpoint as the CLI starts up. You can connect to this
diff --git a/goldens/public-api/angular/ssr/index.api.md b/goldens/public-api/angular/ssr/index.api.md
index 81764fcc1f62..cbdacabfd7f6 100644
--- a/goldens/public-api/angular/ssr/index.api.md
+++ b/goldens/public-api/angular/ssr/index.api.md
@@ -11,11 +11,17 @@ import { Type } from '@angular/core';
// @public
export class AngularAppEngine {
+ constructor(options?: AngularAppEngineOptions);
handle(request: Request, requestContext?: unknown): Promise;
static ɵallowStaticRouteRender: boolean;
static ɵhooks: Hooks;
}
+// @public
+export interface AngularAppEngineOptions {
+ allowedHosts?: readonly string[];
+}
+
// @public
export function createRequestHandler(handler: RequestHandlerFunction): RequestHandlerFunction;
diff --git a/goldens/public-api/angular/ssr/node/index.api.md b/goldens/public-api/angular/ssr/node/index.api.md
index eccb6396938e..2c52c06b47c1 100644
--- a/goldens/public-api/angular/ssr/node/index.api.md
+++ b/goldens/public-api/angular/ssr/node/index.api.md
@@ -15,8 +15,12 @@ import { Type } from '@angular/core';
// @public
export class AngularNodeAppEngine {
- constructor();
- handle(request: IncomingMessage | Http2ServerRequest, requestContext?: unknown): Promise;
+ constructor(options?: AngularNodeAppEngineOptions);
+ handle(request: IncomingMessage | Http2ServerRequest | Request, requestContext?: unknown): Promise;
+}
+
+// @public
+export interface AngularNodeAppEngineOptions extends AngularAppEngineOptions {
}
// @public
@@ -27,6 +31,7 @@ export class CommonEngine {
// @public (undocumented)
export interface CommonEngineOptions {
+ allowedHosts?: readonly string[];
bootstrap?: Type<{}> | ((context: BootstrapContext) => Promise);
enablePerformanceProfiler?: boolean;
providers?: StaticProvider[];
diff --git a/goldens/public-api/angular_devkit/architect/index.api.md b/goldens/public-api/angular_devkit/architect/index.api.md
index 0ae8751719b5..747b7010f580 100644
--- a/goldens/public-api/angular_devkit/architect/index.api.md
+++ b/goldens/public-api/angular_devkit/architect/index.api.md
@@ -530,7 +530,7 @@ export type Target = json.JsonObject & Target_2;
export function targetFromTargetString(specifier: string, abbreviatedProjectName?: string, abbreviatedTargetName?: string): Target;
// @public
-export function targetStringFromTarget({ project, target, configuration }: Target): string;
+export function targetStringFromTarget(input: Target): string;
// @public
export type TypedBuilderProgress = {
diff --git a/package.json b/package.json
index 39fba551b317..369b67efb526 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "@angular/devkit-repo",
- "version": "21.2.0-next.2",
+ "version": "21.2.0",
"private": true,
"description": "Software Development Kit for Angular",
"keywords": [
@@ -28,12 +28,12 @@
"type": "git",
"url": "https://github.com/angular/angular-cli.git"
},
- "packageManager": "pnpm@10.29.3",
+ "packageManager": "pnpm@10.30.2",
"engines": {
"node": "^20.19.0 || ^22.12.0 || >=24.0.0",
"npm": "Please use pnpm instead of NPM to install dependencies",
"yarn": "Please use pnpm instead of Yarn to install dependencies",
- "pnpm": "10.29.3"
+ "pnpm": "10.30.2"
},
"author": "Angular Authors",
"license": "MIT",
@@ -42,20 +42,20 @@
},
"homepage": "https://github.com/angular/angular-cli",
"devDependencies": {
- "@angular/animations": "21.2.0-next.3",
- "@angular/cdk": "21.2.0-next.4",
- "@angular/common": "21.2.0-next.3",
- "@angular/compiler": "21.2.0-next.3",
- "@angular/compiler-cli": "21.2.0-next.3",
- "@angular/core": "21.2.0-next.3",
- "@angular/forms": "21.2.0-next.3",
- "@angular/localize": "21.2.0-next.3",
- "@angular/material": "21.2.0-next.4",
- "@angular/ng-dev": "https://github.com/angular/dev-infra-private-ng-dev-builds.git#2a5e8e5b5398ae13a8d9a963ac980707313a6c9e",
- "@angular/platform-browser": "21.2.0-next.3",
- "@angular/platform-server": "21.2.0-next.3",
- "@angular/router": "21.2.0-next.3",
- "@angular/service-worker": "21.2.0-next.3",
+ "@angular/animations": "21.2.0",
+ "@angular/cdk": "21.2.0",
+ "@angular/common": "21.2.0",
+ "@angular/compiler": "21.2.0",
+ "@angular/compiler-cli": "21.2.0",
+ "@angular/core": "21.2.0",
+ "@angular/forms": "21.2.0",
+ "@angular/localize": "21.2.0",
+ "@angular/material": "21.2.0",
+ "@angular/ng-dev": "https://github.com/angular/dev-infra-private-ng-dev-builds.git#c78d7a03ae1ca76d2946651d0dd7f476f09b4374",
+ "@angular/platform-browser": "21.2.0",
+ "@angular/platform-server": "21.2.0",
+ "@angular/router": "21.2.0",
+ "@angular/service-worker": "21.2.0",
"@babel/core": "7.29.0",
"@bazel/bazelisk": "1.28.1",
"@bazel/buildifier": "8.2.1",
diff --git a/packages/angular/build/package.json b/packages/angular/build/package.json
index c4d847ae5a8e..ab3558baa04d 100644
--- a/packages/angular/build/package.json
+++ b/packages/angular/build/package.json
@@ -54,7 +54,7 @@
"@angular/ssr": "workspace:*",
"jsdom": "28.1.0",
"less": "4.4.2",
- "ng-packagr": "21.2.0-next.0",
+ "ng-packagr": "21.2.0",
"postcss": "8.5.6",
"rxjs": "7.8.2",
"vitest": "4.0.18"
diff --git a/packages/angular/build/src/builders/application/execute-build.ts b/packages/angular/build/src/builders/application/execute-build.ts
index 0654cd965558..aaddc5b6ef7e 100644
--- a/packages/angular/build/src/builders/application/execute-build.ts
+++ b/packages/angular/build/src/builders/application/execute-build.ts
@@ -56,6 +56,7 @@ export async function executeBuild(
verbose,
colors,
jsonLogs,
+ security,
} = options;
// TODO: Consider integrating into watch mode. Would require full rebuild on target changes.
@@ -263,7 +264,7 @@ export async function executeBuild(
if (serverEntryPoint) {
executionResult.addOutputFile(
SERVER_APP_ENGINE_MANIFEST_FILENAME,
- generateAngularServerAppEngineManifest(i18nOptions, baseHref),
+ generateAngularServerAppEngineManifest(i18nOptions, security.allowedHosts, baseHref),
BuildOutputFileType.ServerRoot,
);
}
diff --git a/packages/angular/build/src/builders/application/i18n.ts b/packages/angular/build/src/builders/application/i18n.ts
index ae37efa674e4..081be50e7a9f 100644
--- a/packages/angular/build/src/builders/application/i18n.ts
+++ b/packages/angular/build/src/builders/application/i18n.ts
@@ -123,48 +123,48 @@ export async function inlineI18n(
inlineResult.prerenderedRoutes = { ...inlineResult.prerenderedRoutes, ...generatedRoutes };
updatedOutputFiles.push(...localeOutputFiles);
}
- } finally {
- await inliner.close();
- }
- // Update the result with all localized files.
- executionResult.outputFiles = [
- // Root and SSR entry files are not modified.
- ...unModifiedOutputFiles,
- // Updated files for each locale.
- ...updatedOutputFiles,
- ];
-
- // Assets are only changed if not using the flat output option
- if (!i18nOptions.flatOutput) {
- executionResult.assetFiles = updatedAssetFiles;
- }
-
- // Inline any template updates if present
- if (executionResult.templateUpdates?.size) {
- // The development server only allows a single locale but issue a warning if used programmatically (experimental)
- // with multiple locales and template HMR.
- if (i18nOptions.inlineLocales.size > 1) {
- inlineResult.warnings.push(
- `Component HMR updates can only be inlined with a single locale. The first locale will be used.`,
- );
+ // Update the result with all localized files.
+ executionResult.outputFiles = [
+ // Root and SSR entry files are not modified.
+ ...unModifiedOutputFiles,
+ // Updated files for each locale.
+ ...updatedOutputFiles,
+ ];
+
+ // Assets are only changed if not using the flat output option
+ if (!i18nOptions.flatOutput) {
+ executionResult.assetFiles = updatedAssetFiles;
}
- const firstLocale = [...i18nOptions.inlineLocales][0];
-
- for (const [id, content] of executionResult.templateUpdates) {
- const templateUpdateResult = await inliner.inlineTemplateUpdate(
- firstLocale,
- i18nOptions.locales[firstLocale].translation,
- content,
- id,
- );
- executionResult.templateUpdates.set(id, templateUpdateResult.code);
- inlineResult.errors.push(...templateUpdateResult.errors);
- inlineResult.warnings.push(...templateUpdateResult.warnings);
+
+ // Inline any template updates if present
+ if (executionResult.templateUpdates?.size) {
+ // The development server only allows a single locale but issue a warning if used programmatically (experimental)
+ // with multiple locales and template HMR.
+ if (i18nOptions.inlineLocales.size > 1) {
+ inlineResult.warnings.push(
+ `Component HMR updates can only be inlined with a single locale. The first locale will be used.`,
+ );
+ }
+ const firstLocale = [...i18nOptions.inlineLocales][0];
+
+ for (const [id, content] of executionResult.templateUpdates) {
+ const templateUpdateResult = await inliner.inlineTemplateUpdate(
+ firstLocale,
+ i18nOptions.locales[firstLocale].translation,
+ content,
+ id,
+ );
+ executionResult.templateUpdates.set(id, templateUpdateResult.code);
+ inlineResult.errors.push(...templateUpdateResult.errors);
+ inlineResult.warnings.push(...templateUpdateResult.warnings);
+ }
}
- }
- return inlineResult;
+ return inlineResult;
+ } finally {
+ await inliner.close();
+ }
}
/**
diff --git a/packages/angular/build/src/builders/application/options.ts b/packages/angular/build/src/builders/application/options.ts
index 83b7ea428f35..4f0d1295a7e3 100644
--- a/packages/angular/build/src/builders/application/options.ts
+++ b/packages/angular/build/src/builders/application/options.ts
@@ -400,8 +400,9 @@ export async function normalizeOptions(
}
}
- const autoCsp = options.security?.autoCsp;
+ const { autoCsp, allowedHosts = [] } = options.security ?? {};
const security = {
+ allowedHosts,
autoCsp: autoCsp
? {
unsafeEval: autoCsp === true ? false : !!autoCsp.unsafeEval,
diff --git a/packages/angular/build/src/builders/application/schema.json b/packages/angular/build/src/builders/application/schema.json
index 8db4e6145b3f..5498a21fe004 100644
--- a/packages/angular/build/src/builders/application/schema.json
+++ b/packages/angular/build/src/builders/application/schema.json
@@ -52,6 +52,14 @@
"type": "object",
"additionalProperties": false,
"properties": {
+ "allowedHosts": {
+ "description": "A list of hostnames that are allowed to access the server-side application. For more information, see https://angular.dev/best-practices/security#preventing-server-side-request-forgery-ssrf.",
+ "type": "array",
+ "uniqueItems": true,
+ "items": {
+ "type": "string"
+ }
+ },
"autoCsp": {
"description": "Enables automatic generation of a hash-based Strict Content Security Policy (https://web.dev/articles/strict-csp#choose-hash) based on scripts in index.html. Will default to true once we are out of experimental/preview phases.",
"default": false,
diff --git a/packages/angular/build/src/builders/dev-server/tests/behavior/build-assets_spec.ts b/packages/angular/build/src/builders/dev-server/tests/behavior/build-assets_spec.ts
index f7c7a0acb33a..7d3e1ffc414b 100644
--- a/packages/angular/build/src/builders/dev-server/tests/behavior/build-assets_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/behavior/build-assets_spec.ts
@@ -21,6 +21,26 @@ describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupT
"import {foo} from 'unresolved'; /* a comment */const foo = `bar`;\n\n\n";
describe('Behavior: "browser builder assets"', () => {
+ it('serves a project extensionless asset unmodified', async () => {
+ await harness.writeFile('src/extensionless', javascriptFileContent);
+
+ setupTarget(harness, {
+ assets: ['src/extensionless'],
+ optimization: {
+ scripts: true,
+ },
+ });
+
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ });
+
+ const { result, response } = await executeOnceAndFetch(harness, 'extensionless');
+
+ expect(result?.success).toBeTrue();
+ expect(await response?.text()).toContain(javascriptFileContent);
+ });
+
it('serves a project JavaScript asset unmodified', async () => {
await harness.writeFile('src/extra.js', javascriptFileContent);
diff --git a/packages/angular/build/src/builders/dev-server/tests/behavior/build-budgets_spec.ts b/packages/angular/build/src/builders/dev-server/tests/behavior/build-budgets_spec.ts
index aee551e78b48..97a68506fcf5 100644
--- a/packages/angular/build/src/builders/dev-server/tests/behavior/build-budgets_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/behavior/build-budgets_spec.ts
@@ -11,29 +11,25 @@ import { executeDevServer } from '../../index';
import { describeServeBuilder } from '../jasmine-helpers';
import { BASE_OPTIONS, DEV_SERVER_BUILDER_INFO } from '../setup';
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- // TODO(fix-vite): currently this is broken in vite.
- (isViteRun ? xdescribe : describe)('Behavior: "browser builder budgets"', () => {
- beforeEach(() => {
- setupTarget(harness, {
- // Add a budget error for any file over 100 bytes
- budgets: [{ type: BudgetType.All, maximumError: '100b' }],
- optimization: true,
- });
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ // TODO(fix-vite): currently this is broken in vite.
+ xdescribe('Behavior: "browser builder budgets"', () => {
+ beforeEach(() => {
+ setupTarget(harness, {
+ // Add a budget error for any file over 100 bytes
+ budgets: [{ type: BudgetType.All, maximumError: '100b' }],
+ optimization: true,
});
+ });
- it('should ignore budgets defined in the "buildTarget" options', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- });
+ it('should ignore budgets defined in the "buildTarget" options', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ });
- const { result } = await harness.executeOnce();
+ const { result } = await harness.executeOnce();
- expect(result?.success).toBe(true);
- });
+ expect(result?.success).toBe(true);
});
- },
-);
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/tests/behavior/build-conditions_spec.ts b/packages/angular/build/src/builders/dev-server/tests/behavior/build-conditions_spec.ts
index 2a7d59d8d574..aef1973d4a48 100644
--- a/packages/angular/build/src/builders/dev-server/tests/behavior/build-conditions_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/behavior/build-conditions_spec.ts
@@ -15,86 +15,74 @@ import { executeOnceAndFetch } from '../execute-fetch';
import { describeServeBuilder } from '../jasmine-helpers';
import { BASE_OPTIONS, DEV_SERVER_BUILDER_INFO } from '../setup';
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isApplicationBuilder) => {
- describe('Behavior: "conditional imports"', () => {
- if (!isApplicationBuilder) {
- it('requires esbuild', () => {
- expect(true).toBeTrue();
- });
-
- return;
- }
-
- beforeEach(async () => {
- setupTarget(harness);
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ describe('Behavior: "conditional imports"', () => {
+ beforeEach(async () => {
+ setupTarget(harness);
- await setupConditionImport(harness);
- });
+ await setupConditionImport(harness);
+ });
- interface ImportsTestCase {
- name: string;
- mapping: unknown;
- output?: string;
- }
+ interface ImportsTestCase {
+ name: string;
+ mapping: unknown;
+ output?: string;
+ }
- const GOOD_TARGET = './src/good.js';
- const BAD_TARGET = './src/bad.js';
+ const GOOD_TARGET = './src/good.js';
+ const BAD_TARGET = './src/bad.js';
- const testCases: ImportsTestCase[] = [
- { name: 'simple string', mapping: GOOD_TARGET },
- {
- name: 'default fallback without matching condition',
- mapping: {
- 'never': BAD_TARGET,
- 'default': GOOD_TARGET,
- },
+ const testCases: ImportsTestCase[] = [
+ { name: 'simple string', mapping: GOOD_TARGET },
+ {
+ name: 'default fallback without matching condition',
+ mapping: {
+ 'never': BAD_TARGET,
+ 'default': GOOD_TARGET,
},
- {
- name: 'development condition',
- mapping: {
- 'development': GOOD_TARGET,
- 'default': BAD_TARGET,
- },
+ },
+ {
+ name: 'development condition',
+ mapping: {
+ 'development': GOOD_TARGET,
+ 'default': BAD_TARGET,
},
- {
- name: 'production condition',
- mapping: {
- 'production': BAD_TARGET,
- 'default': GOOD_TARGET,
- },
+ },
+ {
+ name: 'production condition',
+ mapping: {
+ 'production': BAD_TARGET,
+ 'default': GOOD_TARGET,
},
- {
- name: 'browser condition (in browser)',
- mapping: {
- 'browser': GOOD_TARGET,
- 'default': BAD_TARGET,
- },
+ },
+ {
+ name: 'browser condition (in browser)',
+ mapping: {
+ 'browser': GOOD_TARGET,
+ 'default': BAD_TARGET,
},
- ];
+ },
+ ];
- for (const testCase of testCases) {
- describe(testCase.name, () => {
- beforeEach(async () => {
- await setTargetMapping(harness, testCase.mapping);
- });
+ for (const testCase of testCases) {
+ describe(testCase.name, () => {
+ beforeEach(async () => {
+ await setTargetMapping(harness, testCase.mapping);
+ });
- it('resolves to expected target', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- });
+ it('resolves to expected target', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ });
- const { result, response } = await executeOnceAndFetch(harness, '/main.js');
+ const { result, response } = await executeOnceAndFetch(harness, '/main.js');
- expect(result?.success).toBeTrue();
- const output = await response?.text();
- expect(output).toContain('good-value');
- expect(output).not.toContain('bad-value');
- });
+ expect(result?.success).toBeTrue();
+ const output = await response?.text();
+ expect(output).toContain('good-value');
+ expect(output).not.toContain('bad-value');
});
- }
- });
- },
-);
+ });
+ }
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/tests/behavior/build_translation_watch_spec.ts b/packages/angular/build/src/builders/dev-server/tests/behavior/build_translation_watch_spec.ts
index b7d65e52e966..24dca8a6a5dc 100644
--- a/packages/angular/build/src/builders/dev-server/tests/behavior/build_translation_watch_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/behavior/build_translation_watch_spec.ts
@@ -12,70 +12,66 @@ import { executeDevServer } from '../../index';
import { describeServeBuilder } from '../jasmine-helpers';
import { BASE_OPTIONS, DEV_SERVER_BUILDER_INFO } from '../setup';
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- // TODO(fix-vite): currently this is broken in vite.
- (isViteRun ? xdescribe : describe)('Behavior: "i18n translation file watching"', () => {
- beforeEach(() => {
- harness.useProject('test', {
- root: '.',
- sourceRoot: 'src',
- cli: {
- cache: {
- enabled: false,
- },
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ // TODO(fix-vite): currently this is broken in vite.
+ xdescribe('Behavior: "i18n translation file watching"', () => {
+ beforeEach(() => {
+ harness.useProject('test', {
+ root: '.',
+ sourceRoot: 'src',
+ cli: {
+ cache: {
+ enabled: false,
},
- i18n: {
- locales: {
- fr: 'src/locales/messages.fr.xlf',
- },
+ },
+ i18n: {
+ locales: {
+ fr: 'src/locales/messages.fr.xlf',
},
- });
-
- setupTarget(harness, { localize: ['fr'] });
+ },
});
- it('watches i18n translation files by default', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- watch: true,
- });
+ setupTarget(harness, { localize: ['fr'] });
+ });
+
+ it('watches i18n translation files by default', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ watch: true,
+ });
- await harness.writeFile(
- 'src/app/app.component.html',
- `
+ await harness.writeFile(
+ 'src/app/app.component.html',
+ `
Hello {{ title }}!
`,
- );
+ );
- await harness.writeFile('src/locales/messages.fr.xlf', TRANSLATION_FILE_CONTENT);
+ await harness.writeFile('src/locales/messages.fr.xlf', TRANSLATION_FILE_CONTENT);
- await harness.executeWithCases([
- async ({ result }) => {
- expect(result?.success).toBe(true);
+ await harness.executeWithCases([
+ async ({ result }) => {
+ expect(result?.success).toBe(true);
- const mainUrl = new URL('main.js', `${result?.baseUrl}`);
- const response = await fetch(mainUrl);
- expect(await response?.text()).toContain('Bonjour');
+ const mainUrl = new URL('main.js', `${result?.baseUrl}`);
+ const response = await fetch(mainUrl);
+ expect(await response?.text()).toContain('Bonjour');
- await harness.modifyFile('src/locales/messages.fr.xlf', (content) =>
- content.replace('Bonjour', 'Salut'),
- );
- },
- async ({ result }) => {
- expect(result?.success).toBe(true);
+ await harness.modifyFile('src/locales/messages.fr.xlf', (content) =>
+ content.replace('Bonjour', 'Salut'),
+ );
+ },
+ async ({ result }) => {
+ expect(result?.success).toBe(true);
- const mainUrl = new URL('main.js', `${result?.baseUrl}`);
- const response = await fetch(mainUrl);
- expect(await response?.text()).toContain('Salut');
- },
- ]);
- });
+ const mainUrl = new URL('main.js', `${result?.baseUrl}`);
+ const response = await fetch(mainUrl);
+ expect(await response?.text()).toContain('Salut');
+ },
+ ]);
});
- },
-);
+ });
+});
const TRANSLATION_FILE_CONTENT = `
diff --git a/packages/angular/build/src/builders/dev-server/tests/behavior/serve-live-reload-proxies_spec.ts b/packages/angular/build/src/builders/dev-server/tests/behavior/serve-live-reload-proxies_spec.ts
index efdd749de258..6c41769d5aea 100644
--- a/packages/angular/build/src/builders/dev-server/tests/behavior/serve-live-reload-proxies_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/behavior/serve-live-reload-proxies_spec.ts
@@ -133,145 +133,138 @@ async function goToPageAndWaitForWS(page: Page, url: string): Promise {
await client.detach();
}
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- // TODO(fix-vite): currently this is broken in vite.
- (isViteRun ? xdescribe : describe)(
- 'Behavior: "Dev-server builder live-reload with proxies"',
- () => {
- let browser: Browser;
- let page: Page;
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ // TODO(fix-vite): currently this is broken in vite.
+ xdescribe('Behavior: "Dev-server builder live-reload with proxies"', () => {
+ let browser: Browser;
+ let page: Page;
- const SERVE_OPTIONS = Object.freeze({
- ...BASE_OPTIONS,
- hmr: false,
- watch: true,
- liveReload: true,
- });
+ const SERVE_OPTIONS = Object.freeze({
+ ...BASE_OPTIONS,
+ hmr: false,
+ watch: true,
+ liveReload: true,
+ });
- beforeAll(async () => {
- browser = await puppeteer.launch({
- // MacOSX users need to set the local binary manually because Chrome has lib files with
- // spaces in them which Bazel does not support in runfiles
- // See: https://github.com/angular/angular-cli/pull/17624
- // eslint-disable-next-line max-len
- // executablePath: '/Users//git/angular-cli/node_modules/puppeteer/.local-chromium/mac-818858/chrome-mac/Chromium.app/Contents/MacOS/Chromium',
- ignoreHTTPSErrors: true,
- args: ['--no-sandbox', '--disable-gpu'],
- });
- });
+ beforeAll(async () => {
+ browser = await puppeteer.launch({
+ // MacOSX users need to set the local binary manually because Chrome has lib files with
+ // spaces in them which Bazel does not support in runfiles
+ // See: https://github.com/angular/angular-cli/pull/17624
+ // eslint-disable-next-line max-len
+ // executablePath: '/Users//git/angular-cli/node_modules/puppeteer/.local-chromium/mac-818858/chrome-mac/Chromium.app/Contents/MacOS/Chromium',
+ ignoreHTTPSErrors: true,
+ args: ['--no-sandbox', '--disable-gpu'],
+ });
+ });
- afterAll(async () => {
- await browser.close();
- });
+ afterAll(async () => {
+ await browser.close();
+ });
- beforeEach(async () => {
- setupTarget(harness, {
- polyfills: ['src/polyfills.ts'],
- });
+ beforeEach(async () => {
+ setupTarget(harness, {
+ polyfills: ['src/polyfills.ts'],
+ });
- page = await browser.newPage();
- });
+ page = await browser.newPage();
+ });
- afterEach(async () => {
- await page.close();
- });
+ afterEach(async () => {
+ await page.close();
+ });
- it('works without proxy', async () => {
- harness.useTarget('serve', {
- ...SERVE_OPTIONS,
- });
+ it('works without proxy', async () => {
+ harness.useTarget('serve', {
+ ...SERVE_OPTIONS,
+ });
- await harness.writeFile('src/app/app.component.html', '{{ title }}
');
+ await harness.writeFile('src/app/app.component.html', '{{ title }}
');
- await harness.executeWithCases([
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- if (typeof result?.baseUrl !== 'string') {
- throw new Error('Expected "baseUrl" to be a string.');
- }
+ await harness.executeWithCases([
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ if (typeof result?.baseUrl !== 'string') {
+ throw new Error('Expected "baseUrl" to be a string.');
+ }
- await goToPageAndWaitForWS(page, result.baseUrl);
- await harness.modifyFile('src/app/app.component.ts', (content) =>
- content.replace(`'app'`, `'app-live-reload'`),
- );
- },
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- const innerText = await page.evaluate(() => document.querySelector('p').innerText);
- expect(innerText).toBe('app-live-reload');
- },
- ]);
- });
+ await goToPageAndWaitForWS(page, result.baseUrl);
+ await harness.modifyFile('src/app/app.component.ts', (content) =>
+ content.replace(`'app'`, `'app-live-reload'`),
+ );
+ },
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ const innerText = await page.evaluate(() => document.querySelector('p').innerText);
+ expect(innerText).toBe('app-live-reload');
+ },
+ ]);
+ });
- it('works without http -> http proxy', async () => {
- harness.useTarget('serve', {
- ...SERVE_OPTIONS,
- });
+ it('works without http -> http proxy', async () => {
+ harness.useTarget('serve', {
+ ...SERVE_OPTIONS,
+ });
- await harness.writeFile('src/app/app.component.html', '{{ title }}
');
+ await harness.writeFile('src/app/app.component.html', '{{ title }}
');
- let proxy: ProxyInstance | undefined;
- try {
- await harness.executeWithCases([
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- if (typeof result?.baseUrl !== 'string') {
- throw new Error('Expected "baseUrl" to be a string.');
- }
+ let proxy: ProxyInstance | undefined;
+ try {
+ await harness.executeWithCases([
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ if (typeof result?.baseUrl !== 'string') {
+ throw new Error('Expected "baseUrl" to be a string.');
+ }
- proxy = await createProxy(result.baseUrl, false);
- await goToPageAndWaitForWS(page, proxy.url);
- await harness.modifyFile('src/app/app.component.ts', (content) =>
- content.replace(`'app'`, `'app-live-reload'`),
- );
- },
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- const innerText = await page.evaluate(() => document.querySelector('p').innerText);
- expect(innerText).toBe('app-live-reload');
- },
- ]);
- } finally {
- proxy?.server.close();
- }
- });
+ proxy = await createProxy(result.baseUrl, false);
+ await goToPageAndWaitForWS(page, proxy.url);
+ await harness.modifyFile('src/app/app.component.ts', (content) =>
+ content.replace(`'app'`, `'app-live-reload'`),
+ );
+ },
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ const innerText = await page.evaluate(() => document.querySelector('p').innerText);
+ expect(innerText).toBe('app-live-reload');
+ },
+ ]);
+ } finally {
+ proxy?.server.close();
+ }
+ });
- it('works without https -> http proxy', async () => {
- harness.useTarget('serve', {
- ...SERVE_OPTIONS,
- });
+ it('works without https -> http proxy', async () => {
+ harness.useTarget('serve', {
+ ...SERVE_OPTIONS,
+ });
- await harness.writeFile('src/app/app.component.html', '{{ title }}
');
+ await harness.writeFile('src/app/app.component.html', '{{ title }}
');
- let proxy: ProxyInstance | undefined;
- try {
- await harness.executeWithCases([
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- if (typeof result?.baseUrl !== 'string') {
- throw new Error('Expected "baseUrl" to be a string.');
- }
+ let proxy: ProxyInstance | undefined;
+ try {
+ await harness.executeWithCases([
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ if (typeof result?.baseUrl !== 'string') {
+ throw new Error('Expected "baseUrl" to be a string.');
+ }
- proxy = await createProxy(result.baseUrl, true);
- await goToPageAndWaitForWS(page, proxy.url);
- await harness.modifyFile('src/app/app.component.ts', (content) =>
- content.replace(`'app'`, `'app-live-reload'`),
- );
- },
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- const innerText = await page.evaluate(() => document.querySelector('p').innerText);
- expect(innerText).toBe('app-live-reload');
- },
- ]);
- } finally {
- proxy?.server.close();
- }
- });
- },
- );
- },
-);
+ proxy = await createProxy(result.baseUrl, true);
+ await goToPageAndWaitForWS(page, proxy.url);
+ await harness.modifyFile('src/app/app.component.ts', (content) =>
+ content.replace(`'app'`, `'app-live-reload'`),
+ );
+ },
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ const innerText = await page.evaluate(() => document.querySelector('p').innerText);
+ expect(innerText).toBe('app-live-reload');
+ },
+ ]);
+ } finally {
+ proxy?.server.close();
+ }
+ });
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/tests/behavior/serve_service-worker_spec.ts b/packages/angular/build/src/builders/dev-server/tests/behavior/serve_service-worker_spec.ts
index b3b63c3a3093..10e2cee70465 100644
--- a/packages/angular/build/src/builders/dev-server/tests/behavior/serve_service-worker_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/behavior/serve_service-worker_spec.ts
@@ -36,180 +36,174 @@ const manifest = {
],
};
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- describe('Behavior: "dev-server builder serves service worker"', () => {
- beforeEach(async () => {
- // Application code is not needed for these tests
- await harness.writeFile('src/main.ts', '');
- await harness.writeFile('src/polyfills.ts', '');
-
- harness.useProject('test', {
- root: '.',
- sourceRoot: 'src',
- cli: {
- cache: {
- enabled: false,
- },
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ describe('Behavior: "dev-server builder serves service worker"', () => {
+ beforeEach(async () => {
+ // Application code is not needed for these tests
+ await harness.writeFile('src/main.ts', '');
+ await harness.writeFile('src/polyfills.ts', '');
+
+ harness.useProject('test', {
+ root: '.',
+ sourceRoot: 'src',
+ cli: {
+ cache: {
+ enabled: false,
},
- i18n: {
- sourceLocale: {
- code: 'fr',
- },
+ },
+ i18n: {
+ sourceLocale: {
+ code: 'fr',
},
- });
+ },
+ });
+ });
+
+ it('works with service worker', async () => {
+ setupTarget(harness, {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ serviceWorker: 'ngsw-config.json',
+ assets: ['src/favicon.ico', 'src/assets'],
+ styles: ['src/styles.css'],
});
- it('works with service worker', async () => {
- setupTarget(harness, {
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
- serviceWorker: (isViteRun ? 'ngsw-config.json' : true) as any,
- assets: ['src/favicon.ico', 'src/assets'],
- styles: ['src/styles.css'],
- });
-
- await harness.writeFiles({
- 'ngsw-config.json': JSON.stringify(manifest),
- 'src/assets/folder-asset.txt': 'folder-asset.txt',
- 'src/styles.css': `body { background: url(./spectrum.png); }`,
- });
-
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- });
-
- const { result, response } = await executeOnceAndFetch(harness, '/ngsw.json');
-
- expect(result?.success).toBeTrue();
-
- expect(await response?.json()).toEqual(
- jasmine.objectContaining({
- configVersion: 1,
- index: '/index.html',
- navigationUrls: [
- { positive: true, regex: '^\\/.*$' },
- { positive: false, regex: '^\\/(?:.+\\/)?[^/]*\\.[^/]*$' },
- { positive: false, regex: '^\\/(?:.+\\/)?[^/]*__[^/]*$' },
- { positive: false, regex: '^\\/(?:.+\\/)?[^/]*__[^/]*\\/.*$' },
- ],
- assetGroups: [
- {
- name: 'app',
- installMode: 'prefetch',
- updateMode: 'prefetch',
- urls: ['/favicon.ico', '/index.html'],
- cacheQueryOptions: {
- ignoreVary: true,
- },
- patterns: [],
+ await harness.writeFiles({
+ 'ngsw-config.json': JSON.stringify(manifest),
+ 'src/assets/folder-asset.txt': 'folder-asset.txt',
+ 'src/styles.css': `body { background: url(./spectrum.png); }`,
+ });
+
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ });
+
+ const { result, response } = await executeOnceAndFetch(harness, '/ngsw.json');
+
+ expect(result?.success).toBeTrue();
+
+ expect(await response?.json()).toEqual(
+ jasmine.objectContaining({
+ configVersion: 1,
+ index: '/index.html',
+ navigationUrls: [
+ { positive: true, regex: '^\\/.*$' },
+ { positive: false, regex: '^\\/(?:.+\\/)?[^/]*\\.[^/]*$' },
+ { positive: false, regex: '^\\/(?:.+\\/)?[^/]*__[^/]*$' },
+ { positive: false, regex: '^\\/(?:.+\\/)?[^/]*__[^/]*\\/.*$' },
+ ],
+ assetGroups: [
+ {
+ name: 'app',
+ installMode: 'prefetch',
+ updateMode: 'prefetch',
+ urls: ['/favicon.ico', '/index.html'],
+ cacheQueryOptions: {
+ ignoreVary: true,
},
- {
- name: 'assets',
- installMode: 'lazy',
- updateMode: 'prefetch',
- urls: ['/assets/folder-asset.txt', '/media/spectrum.png'],
- cacheQueryOptions: {
- ignoreVary: true,
- },
- patterns: [],
+ patterns: [],
+ },
+ {
+ name: 'assets',
+ installMode: 'lazy',
+ updateMode: 'prefetch',
+ urls: ['/assets/folder-asset.txt', '/media/spectrum.png'],
+ cacheQueryOptions: {
+ ignoreVary: true,
},
- ],
- dataGroups: [],
- hashTable: {
- '/favicon.ico': '84161b857f5c547e3699ddfbffc6d8d737542e01',
- '/assets/folder-asset.txt': '617f202968a6a81050aa617c2e28e1dca11ce8d4',
- '/index.html': isViteRun
- ? 'e5b73e6798d2782bf59dd5272d254d5bde364695'
- : '9d232e3e13b4605d197037224a2a6303dd337480',
- '/media/spectrum.png': '8d048ece46c0f3af4b598a95fd8e4709b631c3c0',
+ patterns: [],
},
- }),
- );
+ ],
+ dataGroups: [],
+ hashTable: {
+ '/favicon.ico': '84161b857f5c547e3699ddfbffc6d8d737542e01',
+ '/assets/folder-asset.txt': '617f202968a6a81050aa617c2e28e1dca11ce8d4',
+ '/index.html': 'e5b73e6798d2782bf59dd5272d254d5bde364695',
+ '/media/spectrum.png': '8d048ece46c0f3af4b598a95fd8e4709b631c3c0',
+ },
+ }),
+ );
+ });
+
+ it('works with localize', async () => {
+ setupTarget(harness, {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ serviceWorker: 'ngsw-config.json' as any,
+ assets: ['src/favicon.ico', 'src/assets'],
+ styles: ['src/styles.css'],
+ localize: ['fr'],
});
- it('works with localize', async () => {
- setupTarget(harness, {
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
- serviceWorker: (isViteRun ? 'ngsw-config.json' : true) as any,
- assets: ['src/favicon.ico', 'src/assets'],
- styles: ['src/styles.css'],
- localize: ['fr'],
- });
+ await harness.writeFiles({
+ 'ngsw-config.json': JSON.stringify(manifest),
+ 'src/assets/folder-asset.txt': 'folder-asset.txt',
+ 'src/styles.css': `body { background: url(./spectrum.png); }`,
+ });
+
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ });
- await harness.writeFiles({
- 'ngsw-config.json': JSON.stringify(manifest),
- 'src/assets/folder-asset.txt': 'folder-asset.txt',
- 'src/styles.css': `body { background: url(./spectrum.png); }`,
- });
+ const { result, response } = await executeOnceAndFetch(harness, '/ngsw.json');
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- });
+ expect(result?.success).toBeTrue();
- const { result, response } = await executeOnceAndFetch(harness, '/ngsw.json');
+ expect(await response?.json()).toBeDefined();
+ });
- expect(result?.success).toBeTrue();
+ // TODO(fix-vite): currently this is broken in vite due to watcher never terminates.
+ xit('works in watch mode', async () => {
+ setupTarget(harness, {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ serviceWorker: 'ngsw-config.json' as any,
+ assets: ['src/favicon.ico', 'src/assets'],
+ styles: ['src/styles.css'],
+ });
- expect(await response?.json()).toBeDefined();
+ await harness.writeFiles({
+ 'ngsw-config.json': JSON.stringify(manifest),
+ 'src/assets/folder-asset.txt': 'folder-asset.txt',
+ 'src/styles.css': `body { background: url(./spectrum.png); }`,
});
- // TODO(fix-vite): currently this is broken in vite due to watcher never terminates.
- (isViteRun ? xit : it)('works in watch mode', async () => {
- setupTarget(harness, {
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
- serviceWorker: (isViteRun ? 'ngsw-config.json' : true) as any,
- assets: ['src/favicon.ico', 'src/assets'],
- styles: ['src/styles.css'],
- });
-
- await harness.writeFiles({
- 'ngsw-config.json': JSON.stringify(manifest),
- 'src/assets/folder-asset.txt': 'folder-asset.txt',
- 'src/styles.css': `body { background: url(./spectrum.png); }`,
- });
-
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- watch: true,
- });
-
- await harness.executeWithCases([
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- const response = await fetch(new URL('ngsw.json', `${result?.baseUrl}`));
- const { hashTable } = (await response.json()) as { hashTable: object };
- const hashTableEntries = Object.keys(hashTable);
-
- expect(hashTableEntries).toEqual([
- '/assets/folder-asset.txt',
- '/favicon.ico',
- '/index.html',
- '/media/spectrum.png',
- ]);
-
- await harness.writeFile(
- 'src/assets/folder-new-asset.txt',
- harness.readFile('src/assets/folder-asset.txt'),
- );
- },
- async ({ result }) => {
- expect(result?.success).toBeTrue();
- const response = await fetch(new URL('ngsw.json', `${result?.baseUrl}`));
- const { hashTable } = (await response.json()) as { hashTable: object };
- const hashTableEntries = Object.keys(hashTable);
-
- expect(hashTableEntries).toEqual([
- '/assets/folder-asset.txt',
- '/assets/folder-new-asset.txt',
- '/favicon.ico',
- '/index.html',
- '/media/spectrum.png',
- ]);
- },
- ]);
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ watch: true,
});
+
+ await harness.executeWithCases([
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ const response = await fetch(new URL('ngsw.json', `${result?.baseUrl}`));
+ const { hashTable } = (await response.json()) as { hashTable: object };
+ const hashTableEntries = Object.keys(hashTable);
+
+ expect(hashTableEntries).toEqual([
+ '/assets/folder-asset.txt',
+ '/favicon.ico',
+ '/index.html',
+ '/media/spectrum.png',
+ ]);
+
+ await harness.writeFile(
+ 'src/assets/folder-new-asset.txt',
+ harness.readFile('src/assets/folder-asset.txt'),
+ );
+ },
+ async ({ result }) => {
+ expect(result?.success).toBeTrue();
+ const response = await fetch(new URL('ngsw.json', `${result?.baseUrl}`));
+ const { hashTable } = (await response.json()) as { hashTable: object };
+ const hashTableEntries = Object.keys(hashTable);
+
+ expect(hashTableEntries).toEqual([
+ '/assets/folder-asset.txt',
+ '/assets/folder-new-asset.txt',
+ '/favicon.ico',
+ '/index.html',
+ '/media/spectrum.png',
+ ]);
+ },
+ ]);
});
- },
-);
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/tests/jasmine-helpers.ts b/packages/angular/build/src/builders/dev-server/tests/jasmine-helpers.ts
index c5a73446cf5a..e680fe0f62ab 100644
--- a/packages/angular/build/src/builders/dev-server/tests/jasmine-helpers.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/jasmine-helpers.ts
@@ -19,7 +19,6 @@ export function describeServeBuilder(
specDefinitions: (
harness: JasmineBuilderHarness,
setupTarget: typeof setupApplicationTarget,
- isViteRun: true,
) => void,
): void {
let optionSchema = optionSchemaCache.get(options.schemaPath);
@@ -36,6 +35,6 @@ export function describeServeBuilder(
beforeEach(() => host.initialize().toPromise());
afterEach(() => host.restore().toPromise());
- specDefinitions(harness, setupApplicationTarget, true);
+ specDefinitions(harness, setupApplicationTarget);
});
}
diff --git a/packages/angular/build/src/builders/dev-server/tests/options/port_spec.ts b/packages/angular/build/src/builders/dev-server/tests/options/port_spec.ts
index 83f3a4d1486b..8869dd20dcbb 100644
--- a/packages/angular/build/src/builders/dev-server/tests/options/port_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/options/port_spec.ts
@@ -26,87 +26,58 @@ function getResultPort(result: Record | undefined): string | un
}
}
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- describe('option: "port"', () => {
- beforeEach(async () => {
- setupTarget(harness);
-
- // Application code is not needed for these tests
- await harness.writeFile('src/main.ts', '');
- });
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ describe('option: "port"', () => {
+ beforeEach(async () => {
+ setupTarget(harness);
+
+ // Application code is not needed for these tests
+ await harness.writeFile('src/main.ts', '');
+ });
- it('uses default port (4200) when not present', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- // Base options set port to zero
- port: undefined,
- });
-
- const { result, response, logs } = await executeOnceAndFetch(harness, '/');
-
- expect(result?.success).toBeTrue();
- expect(getResultPort(result)).toBe('4200');
- expect(await response?.text()).toContain('');
-
- if (!isViteRun) {
- expect(logs).toContain(
- jasmine.objectContaining({
- message: jasmine.stringMatching(/:4200/),
- }),
- );
- }
+ it('uses default port (4200) when not present', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ // Base options set port to zero
+ port: undefined,
});
- it('uses a random free port when set to 0 (zero)', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- port: 0,
- });
-
- const { result, response, logs } = await executeOnceAndFetch(harness, '/');
-
- expect(result?.success).toBeTrue();
- const port = getResultPort(result);
- expect(port).not.toBe('4200');
- if (isViteRun) {
- // Should not be default Vite port either
- expect(port).not.toBe('5173');
- }
-
- expect(port).toMatch(/\d{4,6}/);
- expect(await response?.text()).toContain('');
-
- if (!isViteRun) {
- expect(logs).toContain(
- jasmine.objectContaining({
- message: jasmine.stringMatching(':' + port),
- }),
- );
- }
+ const { result, response, logs } = await executeOnceAndFetch(harness, '/');
+
+ expect(result?.success).toBeTrue();
+ expect(getResultPort(result)).toBe('4200');
+ expect(await response?.text()).toContain('');
+ });
+
+ it('uses a random free port when set to 0 (zero)', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ port: 0,
});
- it('uses specific port when a non-zero number is specified', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- port: 8000,
- });
-
- const { result, response, logs } = await executeOnceAndFetch(harness, '/');
-
- expect(result?.success).toBeTrue();
- expect(getResultPort(result)).toBe('8000');
- expect(await response?.text()).toContain('');
- if (!isViteRun) {
- expect(logs).toContain(
- jasmine.objectContaining({
- message: jasmine.stringMatching(':8000'),
- }),
- );
- }
+ const { result, response, logs } = await executeOnceAndFetch(harness, '/');
+
+ expect(result?.success).toBeTrue();
+ const port = getResultPort(result);
+ expect(port).not.toBe('4200');
+ // Should not be default Vite port either
+ expect(port).not.toBe('5173');
+
+ expect(port).toMatch(/\d{4,6}/);
+ expect(await response?.text()).toContain('');
+ });
+
+ it('uses specific port when a non-zero number is specified', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ port: 8000,
});
+
+ const { result, response, logs } = await executeOnceAndFetch(harness, '/');
+
+ expect(result?.success).toBeTrue();
+ expect(getResultPort(result)).toBe('8000');
+ expect(await response?.text()).toContain('');
});
- },
-);
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/tests/options/prebundle_spec.ts b/packages/angular/build/src/builders/dev-server/tests/options/prebundle_spec.ts
index 1e7c5fcd7322..80bab96d3bb8 100644
--- a/packages/angular/build/src/builders/dev-server/tests/options/prebundle_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/options/prebundle_spec.ts
@@ -12,88 +12,84 @@ import { describeServeBuilder } from '../jasmine-helpers';
import { BASE_OPTIONS, DEV_SERVER_BUILDER_INFO } from '../setup';
// TODO: Temporarily disabled pending investigation into test-only Vite not stopping when caching is enabled
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- // prebundling is not available in webpack
- (isViteRun ? xdescribe : xdescribe)('option: "prebundle"', () => {
- beforeEach(async () => {
- setupTarget(harness);
-
- harness.useProject('test', {
- cli: {
- cache: {
- enabled: true,
- },
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ // prebundling is not available in webpack
+ xdescribe('option: "prebundle"', () => {
+ beforeEach(async () => {
+ setupTarget(harness);
+
+ harness.useProject('test', {
+ cli: {
+ cache: {
+ enabled: true,
},
- });
+ },
+ });
- // Application code is not needed for these tests
- await harness.writeFile(
- 'src/main.ts',
- `
+ // Application code is not needed for these tests
+ await harness.writeFile(
+ 'src/main.ts',
+ `
import { VERSION as coreVersion } from '@angular/core';
import { VERSION as platformVersion } from '@angular/platform-browser';
console.log(coreVersion);
console.log(platformVersion);
`,
- );
+ );
+ });
+
+ it('should prebundle dependencies when option is not present', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
});
- it('should prebundle dependencies when option is not present', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- });
+ const { result, content } = await executeOnceAndFetch(harness, '/main.js');
- const { result, content } = await executeOnceAndFetch(harness, '/main.js');
+ expect(result?.success).toBeTrue();
+ expect(content).toContain('vite/deps/@angular_core.js');
+ expect(content).not.toContain('node_modules/@angular/core/');
+ });
- expect(result?.success).toBeTrue();
- expect(content).toContain('vite/deps/@angular_core.js');
- expect(content).not.toContain('node_modules/@angular/core/');
+ it('should prebundle dependencies when option is set to true', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ prebundle: true,
});
- it('should prebundle dependencies when option is set to true', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- prebundle: true,
- });
+ const { result, content } = await executeOnceAndFetch(harness, '/main.js');
- const { result, content } = await executeOnceAndFetch(harness, '/main.js');
+ expect(result?.success).toBeTrue();
+ expect(content).toContain('vite/deps/@angular_core.js');
+ expect(content).not.toContain('node_modules/@angular/core/');
+ });
- expect(result?.success).toBeTrue();
- expect(content).toContain('vite/deps/@angular_core.js');
- expect(content).not.toContain('node_modules/@angular/core/');
+ it('should not prebundle dependencies when option is set to false', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ prebundle: false,
});
- it('should not prebundle dependencies when option is set to false', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- prebundle: false,
- });
+ const { result, content } = await executeOnceAndFetch(harness, '/main.js');
- const { result, content } = await executeOnceAndFetch(harness, '/main.js');
+ expect(result?.success).toBeTrue();
+ expect(content).not.toContain('vite/deps/@angular_core.js');
+ expect(content).toContain('node_modules/@angular/core/');
+ });
- expect(result?.success).toBeTrue();
- expect(content).not.toContain('vite/deps/@angular_core.js');
- expect(content).toContain('node_modules/@angular/core/');
+ it('should not prebundle specified dependency if added to exclude list', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ prebundle: { exclude: ['@angular/platform-browser'] },
});
- it('should not prebundle specified dependency if added to exclude list', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- prebundle: { exclude: ['@angular/platform-browser'] },
- });
+ const { result, content } = await executeOnceAndFetch(harness, '/main.js');
- const { result, content } = await executeOnceAndFetch(harness, '/main.js');
-
- expect(result?.success).toBeTrue();
- expect(content).toContain('vite/deps/@angular_core.js');
- expect(content).not.toContain('node_modules/@angular/core/');
- expect(content).not.toContain('vite/deps/@angular_platform-browser.js');
- expect(content).toContain('node_modules/@angular/platform-browser/');
- });
+ expect(result?.success).toBeTrue();
+ expect(content).toContain('vite/deps/@angular_core.js');
+ expect(content).not.toContain('node_modules/@angular/core/');
+ expect(content).not.toContain('vite/deps/@angular_platform-browser.js');
+ expect(content).toContain('node_modules/@angular/platform-browser/');
});
- },
-);
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/tests/options/proxy-config_spec.ts b/packages/angular/build/src/builders/dev-server/tests/options/proxy-config_spec.ts
index 78f3323b97cc..1c6dfb60ca9d 100644
--- a/packages/angular/build/src/builders/dev-server/tests/options/proxy-config_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/options/proxy-config_spec.ts
@@ -12,7 +12,7 @@ import { executeOnceAndFetch } from '../execute-fetch';
import { describeServeBuilder } from '../jasmine-helpers';
import { BASE_OPTIONS, DEV_SERVER_BUILDER_INFO, BuilderHarness } from '../setup';
-describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget, isVite) => {
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
describe('option: "proxyConfig"', () => {
beforeEach(async () => {
setupTarget(harness);
@@ -236,14 +236,51 @@ describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupT
}
});
- /**
- * ****************************************************************************************************
- * ********************************** Below only Vite specific tests **********************************
- * ****************************************************************************************************
- */
- if (isVite) {
- viteOnlyTests(harness);
- }
+ it('proxies support regexp as context', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ proxyConfig: 'proxy.config.json',
+ });
+
+ const proxyServer = await createProxyServer();
+ try {
+ await harness.writeFiles({
+ 'proxy.config.json': `
+ { "^/api/.*": { "target": "http://127.0.0.1:${proxyServer.address.port}" } }
+ `,
+ });
+
+ const { result, response } = await executeOnceAndFetch(harness, '/api/test');
+
+ expect(result?.success).toBeTrue();
+ expect(await response?.text()).toContain('TEST_API_RETURN');
+ } finally {
+ await proxyServer.close();
+ }
+ });
+
+ it('proxies support negated regexp as context', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ proxyConfig: 'proxy.config.json',
+ });
+
+ const proxyServer = await createProxyServer();
+ try {
+ await harness.writeFiles({
+ 'proxy.config.json': `
+ { "^\\/(?!something).*": { "target": "http://127.0.0.1:${proxyServer.address.port}" } }
+ `,
+ });
+
+ const { result, response } = await executeOnceAndFetch(harness, '/api/test');
+
+ expect(result?.success).toBeTrue();
+ expect(await response?.text()).toContain('TEST_API_RETURN');
+ } finally {
+ await proxyServer.close();
+ }
+ });
});
});
@@ -270,54 +307,3 @@ async function createProxyServer() {
close: () => new Promise((resolve) => proxyServer.close(() => resolve())),
};
}
-
-/**
- * Vite specific tests
- */
-function viteOnlyTests(harness: BuilderHarness): void {
- it('proxies support regexp as context', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- proxyConfig: 'proxy.config.json',
- });
-
- const proxyServer = await createProxyServer();
- try {
- await harness.writeFiles({
- 'proxy.config.json': `
- { "^/api/.*": { "target": "http://127.0.0.1:${proxyServer.address.port}" } }
- `,
- });
-
- const { result, response } = await executeOnceAndFetch(harness, '/api/test');
-
- expect(result?.success).toBeTrue();
- expect(await response?.text()).toContain('TEST_API_RETURN');
- } finally {
- await proxyServer.close();
- }
- });
-
- it('proxies support negated regexp as context', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- proxyConfig: 'proxy.config.json',
- });
-
- const proxyServer = await createProxyServer();
- try {
- await harness.writeFiles({
- 'proxy.config.json': `
- { "^\\/(?!something).*": { "target": "http://127.0.0.1:${proxyServer.address.port}" } }
- `,
- });
-
- const { result, response } = await executeOnceAndFetch(harness, '/api/test');
-
- expect(result?.success).toBeTrue();
- expect(await response?.text()).toContain('TEST_API_RETURN');
- } finally {
- await proxyServer.close();
- }
- });
-}
diff --git a/packages/angular/build/src/builders/dev-server/tests/options/serve-path_spec.ts b/packages/angular/build/src/builders/dev-server/tests/options/serve-path_spec.ts
index 7570175c65d2..5917dcc8eeb4 100644
--- a/packages/angular/build/src/builders/dev-server/tests/options/serve-path_spec.ts
+++ b/packages/angular/build/src/builders/dev-server/tests/options/serve-path_spec.ts
@@ -12,109 +12,105 @@ import { executeOnceAndFetch } from '../execute-fetch';
import { describeServeBuilder } from '../jasmine-helpers';
import { BASE_OPTIONS, DEV_SERVER_BUILDER_INFO } from '../setup';
-describeServeBuilder(
- executeDevServer,
- DEV_SERVER_BUILDER_INFO,
- (harness, setupTarget, isViteRun) => {
- describe('option: "servePath"', () => {
- beforeEach(async () => {
- setupTarget(harness, {
- assets: ['src/assets'],
- });
-
- // Application code is not needed for these tests
- await harness.writeFile('src/main.ts', 'console.log("foo");');
+describeServeBuilder(executeDevServer, DEV_SERVER_BUILDER_INFO, (harness, setupTarget) => {
+ describe('option: "servePath"', () => {
+ beforeEach(async () => {
+ setupTarget(harness, {
+ assets: ['src/assets'],
});
- it('serves application at the root when option is not present', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- });
-
- const { result, response } = await executeOnceAndFetch(harness, '/main.js');
+ // Application code is not needed for these tests
+ await harness.writeFile('src/main.ts', 'console.log("foo");');
+ });
- expect(result?.success).toBeTrue();
- const baseUrl = new URL(`${result?.baseUrl}`);
- expect(baseUrl.pathname).toBe('/');
- expect(await response?.text()).toContain('console.log');
+ it('serves application at the root when option is not present', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
});
- it('serves application at specified path when option is used', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- servePath: 'test',
- });
+ const { result, response } = await executeOnceAndFetch(harness, '/main.js');
- const { result, response } = await executeOnceAndFetch(harness, '/test/main.js');
+ expect(result?.success).toBeTrue();
+ const baseUrl = new URL(`${result?.baseUrl}`);
+ expect(baseUrl.pathname).toBe('/');
+ expect(await response?.text()).toContain('console.log');
+ });
- expect(result?.success).toBeTrue();
- const baseUrl = new URL(`${result?.baseUrl}/`);
- expect(baseUrl.pathname).toBe('/test/');
- expect(await response?.text()).toContain('console.log');
+ it('serves application at specified path when option is used', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ servePath: 'test',
});
- // TODO(fix-vite): currently this is broken in vite.
- (isViteRun ? xit : it)('does not rewrite from root when option is used', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- servePath: 'test',
- });
+ const { result, response } = await executeOnceAndFetch(harness, '/test/main.js');
- const { result, response } = await executeOnceAndFetch(harness, '/', {
- // fallback processing requires an accept header
- request: { headers: { accept: 'text/html' } },
- });
+ expect(result?.success).toBeTrue();
+ const baseUrl = new URL(`${result?.baseUrl}/`);
+ expect(baseUrl.pathname).toBe('/test/');
+ expect(await response?.text()).toContain('console.log');
+ });
- expect(result?.success).toBeTrue();
- expect(response?.status).toBe(404);
+ // TODO(fix-vite): currently this is broken in vite.
+ xit('does not rewrite from root when option is used', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ servePath: 'test',
});
- it('does not rewrite from path outside serve path when option is used', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- servePath: 'test',
- });
+ const { result, response } = await executeOnceAndFetch(harness, '/', {
+ // fallback processing requires an accept header
+ request: { headers: { accept: 'text/html' } },
+ });
- const { result, response } = await executeOnceAndFetch(harness, '/api/', {
- // fallback processing requires an accept header
- request: { headers: { accept: 'text/html' } },
- });
+ expect(result?.success).toBeTrue();
+ expect(response?.status).toBe(404);
+ });
- expect(result?.success).toBeTrue();
- expect(response?.status).toBe(404);
+ it('does not rewrite from path outside serve path when option is used', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ servePath: 'test',
});
- it('rewrites from path inside serve path when option is used', async () => {
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- servePath: 'test',
- });
+ const { result, response } = await executeOnceAndFetch(harness, '/api/', {
+ // fallback processing requires an accept header
+ request: { headers: { accept: 'text/html' } },
+ });
- const { result, response } = await executeOnceAndFetch(harness, '/test/inside', {
- // fallback processing requires an accept header
- request: { headers: { accept: 'text/html' } },
- });
+ expect(result?.success).toBeTrue();
+ expect(response?.status).toBe(404);
+ });
+
+ it('rewrites from path inside serve path when option is used', async () => {
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ servePath: 'test',
+ });
- expect(result?.success).toBeTrue();
- expect(await response?.text()).toContain('');
+ const { result, response } = await executeOnceAndFetch(harness, '/test/inside', {
+ // fallback processing requires an accept header
+ request: { headers: { accept: 'text/html' } },
});
- it('serves assets at specified path when option is used', async () => {
- await harness.writeFile('src/assets/test.txt', 'hello world!');
+ expect(result?.success).toBeTrue();
+ expect(await response?.text()).toContain('');
+ });
- harness.useTarget('serve', {
- ...BASE_OPTIONS,
- servePath: 'test',
- });
+ it('serves assets at specified path when option is used', async () => {
+ await harness.writeFile('src/assets/test.txt', 'hello world!');
- const { result, response } = await executeOnceAndFetch(harness, '/test/assets/test.txt', {
- // fallback processing requires an accept header
- request: { headers: { accept: 'text/html' } },
- });
+ harness.useTarget('serve', {
+ ...BASE_OPTIONS,
+ servePath: 'test',
+ });
- expect(result?.success).toBeTrue();
- expect(await response?.text()).toContain('hello world');
+ const { result, response } = await executeOnceAndFetch(harness, '/test/assets/test.txt', {
+ // fallback processing requires an accept header
+ request: { headers: { accept: 'text/html' } },
});
+
+ expect(result?.success).toBeTrue();
+ expect(await response?.text()).toContain('hello world');
});
- },
-);
+ });
+});
diff --git a/packages/angular/build/src/builders/dev-server/vite/index.ts b/packages/angular/build/src/builders/dev-server/vite/index.ts
index 8129daac1ba1..083008f17050 100644
--- a/packages/angular/build/src/builders/dev-server/vite/index.ts
+++ b/packages/angular/build/src/builders/dev-server/vite/index.ts
@@ -9,7 +9,6 @@
import type { BuilderContext } from '@angular-devkit/architect';
import type { Plugin } from 'esbuild';
import assert from 'node:assert';
-import { builtinModules, isBuiltin } from 'node:module';
import { join } from 'node:path';
import type { Connect, ViteDevServer } from 'vite';
import type { ComponentStyleRecord } from '../../../tools/vite/middlewares';
@@ -21,7 +20,6 @@ import { Result, ResultKind } from '../../application/results';
import { OutputHashing } from '../../application/schema';
import {
type ApplicationBuilderInternalOptions,
- type ExternalResultMetadata,
JavaScriptTransformer,
getSupportedBrowsers,
isZonelessApp,
@@ -99,8 +97,18 @@ export async function* serveWithVite(
browserOptions.ssr ||= true;
}
- // Disable auto CSP.
+ // Vite allowedHost syntax doesn't allow `*.` but `.` acts as `*.`
+ // Angular SSR supports `*.`.
+ const allowedHosts = Array.isArray(serverOptions.allowedHosts)
+ ? serverOptions.allowedHosts.map((host) => (host[0] === '.' ? '*' + host : host))
+ : [];
+
+ // Always allow the dev server host
+ allowedHosts.push(serverOptions.host);
+
browserOptions.security = {
+ allowedHosts,
+ // Disable auto CSP.
autoCsp: false,
};
diff --git a/packages/angular/build/src/private.ts b/packages/angular/build/src/private.ts
index 4791a94a42d3..50d57d0e5a01 100644
--- a/packages/angular/build/src/private.ts
+++ b/packages/angular/build/src/private.ts
@@ -26,6 +26,10 @@ export { buildApplicationInternal } from './builders/application';
export type { ApplicationBuilderInternalOptions } from './builders/application/options';
export { type Result, type ResultFile, ResultKind } from './builders/application/results';
export { serveWithVite } from './builders/dev-server/vite';
+export {
+ normalizeOptions as normalizeDevServerOptions,
+ type NormalizedDevServerOptions,
+} from './builders/dev-server/options';
// Tools
export * from './tools/babel/plugins';
diff --git a/packages/angular/build/src/tools/vite/middlewares/assets-middleware.ts b/packages/angular/build/src/tools/vite/middlewares/assets-middleware.ts
index f0a137f578f8..e0074625afe0 100644
--- a/packages/angular/build/src/tools/vite/middlewares/assets-middleware.ts
+++ b/packages/angular/build/src/tools/vite/middlewares/assets-middleware.ts
@@ -45,8 +45,8 @@ export function createAngularAssetsMiddleware(
// Rewrite all build assets to a vite raw fs URL
const asset = assets.get(pathname);
if (asset) {
- // This is a workaround to serve CSS, JS and TS files without Vite transformations.
- if (JS_TS_REGEXP.test(extension) || CSS_PREPROCESSOR_REGEXP.test(extension)) {
+ // This is a workaround to serve extensionless, CSS, JS and TS files without Vite transformations.
+ if (!extension || JS_TS_REGEXP.test(extension) || CSS_PREPROCESSOR_REGEXP.test(extension)) {
const contents = readFileSync(asset.source);
const etag = `W/${createHash('sha256').update(contents).digest('hex')}`;
if (checkAndHandleEtag(req, res, etag)) {
diff --git a/packages/angular/build/src/utils/server-rendering/manifest.ts b/packages/angular/build/src/utils/server-rendering/manifest.ts
index b01bff38b58f..34c2e334b52c 100644
--- a/packages/angular/build/src/utils/server-rendering/manifest.ts
+++ b/packages/angular/build/src/utils/server-rendering/manifest.ts
@@ -53,11 +53,13 @@ function escapeUnsafeChars(str: string): string {
*
* @param i18nOptions - The internationalization options for the application build. This
* includes settings for inlining locales and determining the output structure.
+ * @param allowedHosts - A list of hosts that are allowed to access the server-side application.
* @param baseHref - The base HREF for the application. This is used to set the base URL
* for all relative URLs in the application.
*/
export function generateAngularServerAppEngineManifest(
i18nOptions: NormalizedApplicationBuildOptions['i18nOptions'],
+ allowedHosts: string[],
baseHref: string | undefined,
): string {
const entryPoints: Record = {};
@@ -84,6 +86,7 @@ export function generateAngularServerAppEngineManifest(
const manifestContent = `
export default {
basePath: '${basePath}',
+ allowedHosts: ${JSON.stringify(allowedHosts, undefined, 2)},
supportedLocales: ${JSON.stringify(supportedLocales, undefined, 2)},
entryPoints: {
${Object.entries(entryPoints)
diff --git a/packages/angular/build/src/utils/server-rendering/prerender.ts b/packages/angular/build/src/utils/server-rendering/prerender.ts
index f33f851f10c4..39d0f0934c92 100644
--- a/packages/angular/build/src/utils/server-rendering/prerender.ts
+++ b/packages/angular/build/src/utils/server-rendering/prerender.ts
@@ -225,6 +225,9 @@ async function renderPages(
hasSsrEntry: !!outputFilesForWorker['server.mjs'],
} as RenderWorkerData,
execArgv: workerExecArgv,
+ env: {
+ 'NG_ALLOWED_HOSTS': 'localhost',
+ },
});
try {
@@ -337,6 +340,9 @@ async function getAllRoutes(
hasSsrEntry: !!outputFilesForWorker['server.mjs'],
} as RoutesExtractorWorkerData,
execArgv: workerExecArgv,
+ env: {
+ 'NG_ALLOWED_HOSTS': 'localhost',
+ },
});
try {
diff --git a/packages/angular/build/src/utils/version.ts b/packages/angular/build/src/utils/version.ts
index 51f493bfe993..b9bbf5cd03b8 100644
--- a/packages/angular/build/src/utils/version.ts
+++ b/packages/angular/build/src/utils/version.ts
@@ -37,18 +37,35 @@ export function assertCompatibleAngularVersion(projectRoot: string): void | neve
process.exit(2);
}
+ const angularCoreSemVer = new SemVer(angularPkgJson['version']);
+ const { version, build, raw } = angularCoreSemVer;
const supportedAngularSemver = '0.0.0-ANGULAR-FW-PEER-DEP';
- if (angularPkgJson['version'] === '0.0.0' || supportedAngularSemver.startsWith('0.0.0')) {
+
+ if (version.startsWith('0.0.0') || supportedAngularSemver.startsWith('0.0.0')) {
// Internal CLI and FW testing version.
return;
}
- const angularVersion = new SemVer(angularPkgJson['version']);
+ if (build.length && version.endsWith('.0.0-next.0')) {
+ // Special handle for local builds only when it's prerelease of major version and it's the 0th version.
+ // This happends when we are bumping to a new major version. and the cli has not releated a verion.
+
+ // Example:
+ // raw: '22.0.0-next.0+sha-c7dc705-with-local-changes',
+ // major: 22,
+ // minor: 0,
+ // patch: 0,
+ // prerelease: [ 'next', 0 ],
+ // build: [ 'sha-c7dc705-with-local-changes' ],
+ // version: '22.0.0-next.0'
+
+ return;
+ }
- if (!satisfies(angularVersion, supportedAngularSemver, { includePrerelease: true })) {
+ if (!satisfies(angularCoreSemVer, supportedAngularSemver, { includePrerelease: true })) {
console.error(
`Error: The current version of "@angular/build" supports Angular versions ${supportedAngularSemver},\n` +
- `but detected Angular version ${angularVersion} instead.\n` +
+ `but detected Angular version ${raw} instead.\n` +
'Please visit the link below to find instructions on how to update Angular.\nhttps://update.angular.dev/',
);
diff --git a/packages/angular/cli/src/command-builder/utilities/json-schema_spec.ts b/packages/angular/cli/src/command-builder/utilities/json-schema_spec.ts
index d311373d69f0..11228e4adca0 100644
--- a/packages/angular/cli/src/command-builder/utilities/json-schema_spec.ts
+++ b/packages/angular/cli/src/command-builder/utilities/json-schema_spec.ts
@@ -9,7 +9,7 @@
import { JsonObject, schema } from '@angular-devkit/core';
import yargs from 'yargs';
-import { addSchemaOptionsToCommand, parseJsonSchemaToOptions } from './json-schema';
+import { Option, addSchemaOptionsToCommand, parseJsonSchemaToOptions } from './json-schema';
describe('parseJsonSchemaToOptions', () => {
describe('without required fields in schema', () => {
@@ -21,10 +21,9 @@ describe('parseJsonSchemaToOptions', () => {
};
let localYargs: yargs.Argv;
- beforeEach(async () => {
- // Create a fresh yargs for each call. The yargs object is stateful and
- // calling .parse multiple times on the same instance isn't safe.
- localYargs = yargs().exitProcess(false).strict().fail(false).wrap(1_000);
+ let options: Option[];
+
+ beforeAll(async () => {
const jsonSchema = {
'type': 'object',
'properties': {
@@ -118,12 +117,20 @@ describe('parseJsonSchemaToOptions', () => {
},
},
};
+
const registry = new schema.CoreSchemaRegistry();
- const options = await parseJsonSchemaToOptions(
+ options = await parseJsonSchemaToOptions(
registry,
jsonSchema as unknown as JsonObject,
false,
);
+ });
+
+ beforeEach(async () => {
+ // Create a fresh yargs for each call. The yargs object is stateful and
+ // calling .parse multiple times on the same instance isn't safe.
+ localYargs = yargs().exitProcess(false).strict().fail(false).wrap(1_000);
+
addSchemaOptionsToCommand(localYargs, options, true);
});
@@ -138,7 +145,15 @@ describe('parseJsonSchemaToOptions', () => {
});
describe('type=array, enum', () => {
- it('parses valid option value', async () => {
+ it('parses valid option value when specified once', async () => {
+ expect(await parse(['--arrayWithChoices', 'always', 'never'])).toEqual(
+ jasmine.objectContaining({
+ 'arrayWithChoices': ['always', 'never'],
+ }),
+ );
+ });
+
+ it('parses valid option value when specified multiple times', async () => {
expect(
await parse(['--arrayWithChoices', 'always', '--arrayWithChoices', 'never']),
).toEqual(
@@ -160,7 +175,15 @@ describe('parseJsonSchemaToOptions', () => {
});
describe('type=array, enum in oneOf', () => {
- it('parses valid option value', async () => {
+ it('parses valid option value when specified once', async () => {
+ expect(await parse(['--arrayWithChoicesInOneOf', 'default', 'verbose'])).toEqual(
+ jasmine.objectContaining({
+ 'arrayWithChoicesInOneOf': ['default', 'verbose'],
+ }),
+ );
+ });
+
+ it('parses valid option value when specified multiple times', async () => {
expect(
await parse([
'--arrayWithChoicesInOneOf',
@@ -183,7 +206,15 @@ describe('parseJsonSchemaToOptions', () => {
});
describe('type=array, anyOf', () => {
- it('parses valid option value', async () => {
+ it('parses valid option value when specified once', async () => {
+ expect(await parse(['--arrayWithComplexAnyOf', 'default', 'something-else'])).toEqual(
+ jasmine.objectContaining({
+ 'arrayWithComplexAnyOf': ['default', 'something-else'],
+ }),
+ );
+ });
+
+ it('parses valid option value when specified multiple times', async () => {
expect(
await parse([
'--arrayWithComplexAnyOf',
diff --git a/packages/angular/cli/src/package-managers/host.ts b/packages/angular/cli/src/package-managers/host.ts
index 433b54414f69..cee68015f677 100644
--- a/packages/angular/cli/src/package-managers/host.ts
+++ b/packages/angular/cli/src/package-managers/host.ts
@@ -24,6 +24,12 @@ import { PackageManagerError } from './error';
* An abstraction layer for side-effectful operations.
*/
export interface Host {
+ /**
+ * Whether shell quoting is required for package manager specifiers.
+ * This is typically true on Windows, where commands are executed in a shell.
+ */
+ readonly requiresQuoting?: boolean;
+
/**
* Creates a directory.
* @param path The path to the directory.
@@ -101,6 +107,7 @@ export interface Host {
*/
export const NodeJS_HOST: Host = {
stat,
+ requiresQuoting: platform() === 'win32',
mkdir,
readFile: (path: string) => readFile(path, { encoding: 'utf8' }),
copyFile: (src, dest) => copyFile(src, dest, constants.COPYFILE_FICLONE),
@@ -130,6 +137,9 @@ export const NodeJS_HOST: Host = {
env: {
...process.env,
...options.env,
+ // NPM updater notifier will prevents the child process from closing until it timeout after 3 minutes.
+ NO_UPDATE_NOTIFIER: '1',
+ NPM_CONFIG_UPDATE_NOTIFIER: 'false',
},
} satisfies SpawnOptions;
const childProcess = isWin32
diff --git a/packages/angular/cli/src/package-managers/package-manager.ts b/packages/angular/cli/src/package-managers/package-manager.ts
index fd3b0a663a79..de3172107d8d 100644
--- a/packages/angular/cli/src/package-managers/package-manager.ts
+++ b/packages/angular/cli/src/package-managers/package-manager.ts
@@ -14,7 +14,7 @@
import { join, relative, resolve } from 'node:path';
import npa from 'npm-package-arg';
-import { maxSatisfying } from 'semver';
+import { maxSatisfying, valid } from 'semver';
import { PackageManagerError } from './error';
import { Host } from './host';
import { Logger } from './logger';
@@ -34,7 +34,7 @@ const METADATA_FIELDS = ['name', 'dist-tags', 'versions', 'time'] as const;
* This is a performance optimization to avoid downloading unnecessary data.
* These fields are the ones required by the CLI for operations like `ng add` and `ng update`.
*/
-const MANIFEST_FIELDS = [
+export const MANIFEST_FIELDS = [
'name',
'version',
'deprecated',
@@ -370,6 +370,10 @@ export class PackageManager {
const { stdout } = await this.#run(this.descriptor.versionCommand);
this.#version = stdout.trim();
+ if (!valid(this.#version)) {
+ throw new Error(`Invalid semver version for ${this.name}: "${this.#version}"`);
+ }
+
return this.#version;
}
@@ -440,7 +444,9 @@ export class PackageManager {
version: string,
options: { timeout?: number; registry?: string; bypassCache?: boolean } = {},
): Promise {
- const specifier = `${packageName}@${version}`;
+ const specifier = this.host.requiresQuoting
+ ? `"${packageName}@${version}"`
+ : `${packageName}@${version}`;
const commandArgs = [...this.descriptor.getManifestCommand, specifier];
const formatter = this.descriptor.viewCommandFieldArgFormatter;
if (formatter) {
diff --git a/packages/angular/cli/src/package-managers/package-manager_spec.ts b/packages/angular/cli/src/package-managers/package-manager_spec.ts
index 802f50fa66ab..8d439d9b3b75 100644
--- a/packages/angular/cli/src/package-managers/package-manager_spec.ts
+++ b/packages/angular/cli/src/package-managers/package-manager_spec.ts
@@ -6,20 +6,54 @@
* found in the LICENSE file at https://angular.dev/license
*/
-import { Host } from './host';
-import { PackageManager } from './package-manager';
+import { MANIFEST_FIELDS, PackageManager } from './package-manager';
import { SUPPORTED_PACKAGE_MANAGERS } from './package-manager-descriptor';
import { MockHost } from './testing/mock-host';
describe('PackageManager', () => {
- let host: Host;
+ let host: MockHost;
let runCommandSpy: jasmine.Spy;
const descriptor = SUPPORTED_PACKAGE_MANAGERS['npm'];
beforeEach(() => {
host = new MockHost();
runCommandSpy = spyOn(host, 'runCommand').and.resolveTo({ stdout: '1.2.3', stderr: '' });
- host.runCommand = runCommandSpy;
+ });
+
+ describe('getRegistryManifest', () => {
+ it('should quote complex range specifiers when required by the host', async () => {
+ // Simulate a quoting host
+ Object.assign(host, { requiresQuoting: true });
+
+ const pm = new PackageManager(host, '/tmp', descriptor);
+ const manifest = { name: 'foo', version: '1.0.0' };
+ runCommandSpy.and.resolveTo({ stdout: JSON.stringify(manifest), stderr: '' });
+
+ await pm.getRegistryManifest('foo', '>=1.0.0 <2.0.0');
+
+ expect(runCommandSpy).toHaveBeenCalledWith(
+ descriptor.binary,
+ [...descriptor.getManifestCommand, '"foo@>=1.0.0 <2.0.0"', ...MANIFEST_FIELDS],
+ jasmine.anything(),
+ );
+ });
+
+ it('should NOT quote complex range specifiers when not required by the host', async () => {
+ // Simulate a non-quoting host
+ Object.assign(host, { requiresQuoting: false });
+
+ const pm = new PackageManager(host, '/tmp', descriptor);
+ const manifest = { name: 'foo', version: '1.0.0' };
+ runCommandSpy.and.resolveTo({ stdout: JSON.stringify(manifest), stderr: '' });
+
+ await pm.getRegistryManifest('foo', '>=1.0.0 <2.0.0');
+
+ expect(runCommandSpy).toHaveBeenCalledWith(
+ descriptor.binary,
+ [...descriptor.getManifestCommand, 'foo@>=1.0.0 <2.0.0', ...MANIFEST_FIELDS],
+ jasmine.anything(),
+ );
+ });
});
describe('getVersion', () => {
diff --git a/packages/angular/cli/src/package-managers/testing/mock-host.ts b/packages/angular/cli/src/package-managers/testing/mock-host.ts
index ae4476c6501d..46e71be3cf60 100644
--- a/packages/angular/cli/src/package-managers/testing/mock-host.ts
+++ b/packages/angular/cli/src/package-managers/testing/mock-host.ts
@@ -14,6 +14,7 @@ import { Host } from '../host';
* This class allows for simulating a file system in memory.
*/
export class MockHost implements Host {
+ readonly requiresQuoting = false;
private readonly fs = new Map();
constructor(files: Record = {}) {
diff --git a/packages/angular/cli/src/utilities/prettier.ts b/packages/angular/cli/src/utilities/prettier.ts
index e481d94a8b89..e2e516b82367 100644
--- a/packages/angular/cli/src/utilities/prettier.ts
+++ b/packages/angular/cli/src/utilities/prettier.ts
@@ -44,7 +44,7 @@ export async function formatFiles(cwd: string, files: Set): Promise {
- const webRequest = createWebRequestFromNodeRequest(request);
+ const webRequest =
+ request instanceof Request ? request : createWebRequestFromNodeRequest(request);
return this.angularAppEngine.handle(webRequest, requestContext);
}
diff --git a/packages/angular/ssr/node/src/common-engine/common-engine.ts b/packages/angular/ssr/node/src/common-engine/common-engine.ts
index 673156ee6b43..1c130d9abe86 100644
--- a/packages/angular/ssr/node/src/common-engine/common-engine.ts
+++ b/packages/angular/ssr/node/src/common-engine/common-engine.ts
@@ -12,6 +12,8 @@ import { renderApplication, renderModule, ɵSERVER_CONTEXT } from '@angular/plat
import * as fs from 'node:fs';
import { dirname, join, normalize, resolve } from 'node:path';
import { URL } from 'node:url';
+import { validateUrl } from '../../../src/utils/validation';
+import { getAllowedHostsFromEnv } from '../environment-options';
import { attachNodeGlobalErrorHandlers } from '../errors';
import { CommonEngineInlineCriticalCssProcessor } from './inline-css-processor';
import {
@@ -31,6 +33,9 @@ export interface CommonEngineOptions {
/** Enable request performance profiling data collection and printing the results in the server console. */
enablePerformanceProfiler?: boolean;
+
+ /** A set of hostnames that are allowed to access the server. */
+ allowedHosts?: readonly string[];
}
export interface CommonEngineRenderOptions {
@@ -64,8 +69,14 @@ export class CommonEngine {
private readonly templateCache = new Map();
private readonly inlineCriticalCssProcessor = new CommonEngineInlineCriticalCssProcessor();
private readonly pageIsSSG = new Map();
+ private readonly allowedHosts: ReadonlySet;
+
+ constructor(private options?: CommonEngineOptions) {
+ this.allowedHosts = new Set([
+ ...getAllowedHostsFromEnv(),
+ ...(this.options?.allowedHosts ?? []),
+ ]);
- constructor(private options?: CommonEngineOptions | undefined) {
attachNodeGlobalErrorHandlers();
}
@@ -74,6 +85,40 @@ export class CommonEngine {
* render options
*/
async render(opts: CommonEngineRenderOptions): Promise {
+ const { url } = opts;
+
+ if (url && URL.canParse(url)) {
+ const urlObj = new URL(url);
+ try {
+ validateUrl(urlObj, this.allowedHosts);
+ } catch (error) {
+ const isAllowedHostConfigured = this.allowedHosts.size > 0;
+ // eslint-disable-next-line no-console
+ console.error(
+ `ERROR: ${(error as Error).message}` +
+ 'Please provide a list of allowed hosts in the "allowedHosts" option in the "CommonEngine" constructor.',
+ isAllowedHostConfigured
+ ? ''
+ : '\nFalling back to client side rendering. This will become a 400 Bad Request in a future major version.',
+ );
+
+ if (!isAllowedHostConfigured) {
+ // Fallback to CSR to avoid a breaking change.
+ // TODO(alanagius): Return a 400 and remove this fallback in the next major version (v22).
+ let document = opts.document;
+ if (!document && opts.documentFilePath) {
+ document = opts.document ?? (await this.getDocument(opts.documentFilePath));
+ }
+
+ if (document) {
+ return document;
+ }
+ }
+
+ throw error;
+ }
+ }
+
const enablePerformanceProfiler = this.options?.enablePerformanceProfiler;
const runMethod = enablePerformanceProfiler
diff --git a/packages/angular/ssr/node/src/environment-options.ts b/packages/angular/ssr/node/src/environment-options.ts
new file mode 100644
index 000000000000..b18dcd08ebe6
--- /dev/null
+++ b/packages/angular/ssr/node/src/environment-options.ts
@@ -0,0 +1,29 @@
+/**
+ * @license
+ * Copyright Google LLC All Rights Reserved.
+ *
+ * Use of this source code is governed by an MIT-style license that can be
+ * found in the LICENSE file at https://angular.dev/license
+ */
+
+/**
+ * Retrieves the list of allowed hosts from the environment variable `NG_ALLOWED_HOSTS`.
+ * @returns An array of allowed hosts.
+ */
+export function getAllowedHostsFromEnv(): ReadonlyArray {
+ const allowedHosts: string[] = [];
+ const envNgAllowedHosts = process.env['NG_ALLOWED_HOSTS'];
+ if (!envNgAllowedHosts) {
+ return allowedHosts;
+ }
+
+ const hosts = envNgAllowedHosts.split(',');
+ for (const host of hosts) {
+ const trimmed = host.trim();
+ if (trimmed.length > 0) {
+ allowedHosts.push(trimmed);
+ }
+ }
+
+ return allowedHosts;
+}
diff --git a/packages/angular/ssr/node/src/request.ts b/packages/angular/ssr/node/src/request.ts
index 32d90d0029fc..402ec29ba56d 100644
--- a/packages/angular/ssr/node/src/request.ts
+++ b/packages/angular/ssr/node/src/request.ts
@@ -8,6 +8,7 @@
import type { IncomingHttpHeaders, IncomingMessage } from 'node:http';
import type { Http2ServerRequest } from 'node:http2';
+import { getFirstHeaderValue } from '../../src/utils/validation';
/**
* A set containing all the pseudo-headers defined in the HTTP/2 specification.
@@ -103,21 +104,3 @@ export function createRequestUrl(nodeRequest: IncomingMessage | Http2ServerReque
return new URL(`${protocol}://${hostnameWithPort}${originalUrl ?? url}`);
}
-
-/**
- * Extracts the first value from a multi-value header string.
- *
- * @param value - A string or an array of strings representing the header values.
- * If it's a string, values are expected to be comma-separated.
- * @returns The first trimmed value from the multi-value header, or `undefined` if the input is invalid or empty.
- *
- * @example
- * ```typescript
- * getFirstHeaderValue("value1, value2, value3"); // "value1"
- * getFirstHeaderValue(["value1", "value2"]); // "value1"
- * getFirstHeaderValue(undefined); // undefined
- * ```
- */
-function getFirstHeaderValue(value: string | string[] | undefined): string | undefined {
- return value?.toString().split(',', 1)[0]?.trim();
-}
diff --git a/packages/angular/ssr/package.json b/packages/angular/ssr/package.json
index af829b8adc1e..5cd45b9058d4 100644
--- a/packages/angular/ssr/package.json
+++ b/packages/angular/ssr/package.json
@@ -29,12 +29,12 @@
},
"devDependencies": {
"@angular-devkit/schematics": "workspace:*",
- "@angular/common": "21.2.0-next.3",
- "@angular/compiler": "21.2.0-next.3",
- "@angular/core": "21.2.0-next.3",
- "@angular/platform-browser": "21.2.0-next.3",
- "@angular/platform-server": "21.2.0-next.3",
- "@angular/router": "21.2.0-next.3",
+ "@angular/common": "21.2.0",
+ "@angular/compiler": "21.2.0",
+ "@angular/core": "21.2.0",
+ "@angular/platform-browser": "21.2.0",
+ "@angular/platform-server": "21.2.0",
+ "@angular/router": "21.2.0",
"@schematics/angular": "workspace:*",
"beasties": "0.4.1"
},
diff --git a/packages/angular/ssr/public_api.ts b/packages/angular/ssr/public_api.ts
index e685f4ceabe3..e566d8414f2f 100644
--- a/packages/angular/ssr/public_api.ts
+++ b/packages/angular/ssr/public_api.ts
@@ -8,7 +8,7 @@
export * from './private_export';
-export { AngularAppEngine } from './src/app-engine';
+export { AngularAppEngine, type AngularAppEngineOptions } from './src/app-engine';
export { createRequestHandler, type RequestHandlerFunction } from './src/handler';
export {
diff --git a/packages/angular/ssr/src/app-engine.ts b/packages/angular/ssr/src/app-engine.ts
index 0cb728e8535d..0ba82002dcef 100644
--- a/packages/angular/ssr/src/app-engine.ts
+++ b/packages/angular/ssr/src/app-engine.ts
@@ -11,6 +11,17 @@ import { Hooks } from './hooks';
import { getPotentialLocaleIdFromUrl, getPreferredLocale } from './i18n';
import { EntryPointExports, getAngularAppEngineManifest } from './manifest';
import { joinUrlParts } from './utils/url';
+import { cloneRequestAndPatchHeaders, validateRequest } from './utils/validation';
+
+/**
+ * Options for the Angular server application engine.
+ */
+export interface AngularAppEngineOptions {
+ /**
+ * A set of allowed hostnames for the server application.
+ */
+ allowedHosts?: readonly string[];
+}
/**
* Angular server application engine.
@@ -45,6 +56,11 @@ export class AngularAppEngine {
*/
private readonly manifest = getAngularAppEngineManifest();
+ /**
+ * A set of allowed hostnames for the server application.
+ */
+ private readonly allowedHosts: ReadonlySet;
+
/**
* A map of supported locales from the server application's manifest.
*/
@@ -57,6 +73,14 @@ export class AngularAppEngine {
*/
private readonly entryPointsCache = new Map>();
+ /**
+ * Creates a new instance of the Angular server application engine.
+ * @param options Options for the Angular server application engine.
+ */
+ constructor(options?: AngularAppEngineOptions) {
+ this.allowedHosts = new Set([...(options?.allowedHosts ?? []), ...this.manifest.allowedHosts]);
+ }
+
/**
* Handles an incoming HTTP request by serving prerendered content, performing server-side rendering,
* or delivering a static file for client-side rendered routes based on the `RenderMode` setting.
@@ -67,12 +91,38 @@ export class AngularAppEngine {
*
* @remarks A request to `https://www.example.com/page/index.html` will serve or render the Angular route
* corresponding to `https://www.example.com/page`.
+ *
+ * @remarks
+ * To prevent potential Server-Side Request Forgery (SSRF), this function verifies the hostname
+ * of the `request.url` against a list of authorized hosts.
+ * If the hostname is not recognized and `allowedHosts` is not empty, a Client-Side Rendered (CSR) version of the
+ * page is returned otherwise a 400 Bad Request is returned.
+ * Resolution:
+ * Authorize your hostname by configuring `allowedHosts` in `angular.json` in:
+ * `projects.[project-name].architect.build.options.security.allowedHosts`.
+ * Alternatively, you pass it directly through the configuration options of `AngularAppEngine`.
+ *
+ * For more information see: https://angular.dev/best-practices/security#preventing-server-side-request-forgery-ssrf
*/
async handle(request: Request, requestContext?: unknown): Promise {
- const serverApp = await this.getAngularServerAppForRequest(request);
+ const allowedHost = this.allowedHosts;
+ try {
+ validateRequest(request, allowedHost);
+ } catch (error) {
+ return this.handleValidationError(error as Error, request);
+ }
+
+ // Clone request with patched headers to prevent unallowed host header access.
+ const { request: securedRequest, onError: onHeaderValidationError } =
+ cloneRequestAndPatchHeaders(request, allowedHost);
+
+ const serverApp = await this.getAngularServerAppForRequest(securedRequest);
if (serverApp) {
- return serverApp.handle(request, requestContext);
+ return Promise.race([
+ onHeaderValidationError.then((error) => this.handleValidationError(error, securedRequest)),
+ serverApp.handle(securedRequest, requestContext),
+ ]);
}
if (this.supportedLocales.length > 1) {
@@ -201,4 +251,42 @@ export class AngularAppEngine {
return this.getEntryPointExports(potentialLocale) ?? this.getEntryPointExports('');
}
+
+ /**
+ * Handles validation errors by logging the error and returning an appropriate response.
+ *
+ * @param error - The validation error to handle.
+ * @param request - The HTTP request that caused the validation error.
+ * @returns A promise that resolves to a `Response` object with a 400 status code if allowed hosts are configured,
+ * or `null` if allowed hosts are not configured (in which case the request is served client-side).
+ */
+ private async handleValidationError(error: Error, request: Request): Promise {
+ const isAllowedHostConfigured = this.allowedHosts.size > 0;
+ const errorMessage = error.message;
+
+ // eslint-disable-next-line no-console
+ console.error(
+ `ERROR: Bad Request ("${request.url}").\n` +
+ errorMessage +
+ (isAllowedHostConfigured
+ ? ''
+ : '\nFalling back to client side rendering. This will become a 400 Bad Request in a future major version.') +
+ '\n\nFor more information, see https://angular.dev/best-practices/security#preventing-server-side-request-forgery-ssrf',
+ );
+
+ if (isAllowedHostConfigured) {
+ // Allowed hosts has been configured incorrectly, thus we can return a 400 bad request.
+ return new Response(errorMessage, {
+ status: 400,
+ statusText: 'Bad Request',
+ headers: { 'Content-Type': 'text/plain' },
+ });
+ }
+
+ // Fallback to CSR to avoid a breaking change.
+ // TODO(alanagius): Return a 400 and remove this fallback in the next major version (v22).
+ const serverApp = await this.getAngularServerAppForRequest(request);
+
+ return serverApp?.serveClientSidePage() ?? null;
+ }
}
diff --git a/packages/angular/ssr/src/app.ts b/packages/angular/ssr/src/app.ts
index 96afaa44c8d6..76296ebe737d 100644
--- a/packages/angular/ssr/src/app.ts
+++ b/packages/angular/ssr/src/app.ts
@@ -484,6 +484,27 @@ export class AngularServerApp {
return html;
}
+
+ /**
+ * Serves the client-side version of the application.
+ * TODO(alanagius): Remove this method in version 22.
+ * @internal
+ */
+ async serveClientSidePage(): Promise {
+ const {
+ manifest: { locale },
+ assets,
+ } = this;
+
+ const html = await assets.getServerAsset('index.csr.html').text();
+
+ return new Response(html, {
+ headers: new Headers({
+ 'Content-Type': 'text/html;charset=UTF-8',
+ ...(locale !== undefined ? { 'Content-Language': locale } : {}),
+ }),
+ });
+ }
}
let angularServerApp: AngularServerApp | undefined;
diff --git a/packages/angular/ssr/src/manifest.ts b/packages/angular/ssr/src/manifest.ts
index 0de603bba104..21ded49b3e10 100644
--- a/packages/angular/ssr/src/manifest.ts
+++ b/packages/angular/ssr/src/manifest.ts
@@ -6,7 +6,6 @@
* found in the LICENSE file at https://angular.dev/license
*/
-import type { BootstrapContext } from '@angular/platform-browser';
import type { SerializableRouteTreeNode } from './routes/route-tree';
import { AngularBootstrap } from './utils/ng';
@@ -74,6 +73,11 @@ export interface AngularAppEngineManifest {
* - `value`: The url segment associated with that locale.
*/
readonly supportedLocales: Readonly>;
+
+ /**
+ * A readonly array of allowed hostnames.
+ */
+ readonly allowedHosts: Readonly;
}
/**
diff --git a/packages/angular/ssr/src/utils/url.ts b/packages/angular/ssr/src/utils/url.ts
index 1fa756e19c19..d5e7c9ac2814 100644
--- a/packages/angular/ssr/src/utils/url.ts
+++ b/packages/angular/ssr/src/utils/url.ts
@@ -95,26 +95,32 @@ export function addTrailingSlash(url: string): string {
* ```
*/
export function joinUrlParts(...parts: string[]): string {
- const normalizeParts: string[] = [];
+ const normalizedParts: string[] = [];
+
for (const part of parts) {
if (part === '') {
// Skip any empty parts
continue;
}
- let normalizedPart = part;
- if (part[0] === '/') {
- normalizedPart = normalizedPart.slice(1);
+ let start = 0;
+ let end = part.length;
+
+ // Use "Pointers" to avoid intermediate slices
+ while (start < end && part[start] === '/') {
+ start++;
}
- if (part.at(-1) === '/') {
- normalizedPart = normalizedPart.slice(0, -1);
+
+ while (end > start && part[end - 1] === '/') {
+ end--;
}
- if (normalizedPart !== '') {
- normalizeParts.push(normalizedPart);
+
+ if (start < end) {
+ normalizedParts.push(part.slice(start, end));
}
}
- return addLeadingSlash(normalizeParts.join('/'));
+ return addLeadingSlash(normalizedParts.join('/'));
}
/**
diff --git a/packages/angular/ssr/src/utils/validation.ts b/packages/angular/ssr/src/utils/validation.ts
new file mode 100644
index 000000000000..c89cdd6a64ed
--- /dev/null
+++ b/packages/angular/ssr/src/utils/validation.ts
@@ -0,0 +1,268 @@
+/**
+ * @license
+ * Copyright Google LLC All Rights Reserved.
+ *
+ * Use of this source code is governed by an MIT-style license that can be
+ * found in the LICENSE file at https://angular.dev/license
+ */
+
+/**
+ * The set of headers that should be validated for host header injection attacks.
+ */
+const HOST_HEADERS_TO_VALIDATE: ReadonlySet = new Set(['host', 'x-forwarded-host']);
+
+/**
+ * Regular expression to validate that the port is a numeric value.
+ */
+const VALID_PORT_REGEX = /^\d+$/;
+
+/**
+ * Regular expression to validate that the protocol is either http or https (case-insensitive).
+ */
+const VALID_PROTO_REGEX = /^https?$/i;
+
+/**
+ * Regular expression to validate that the host is a valid hostname.
+ */
+const VALID_HOST_REGEX = /^[a-z0-9.:-]+$/i;
+
+/**
+ * Regular expression to validate that the prefix is valid.
+ */
+const INVALID_PREFIX_REGEX = /^[/\\]{2}|(?:^|[/\\])\.\.?(?:[/\\]|$)/;
+
+/**
+ * Extracts the first value from a multi-value header string.
+ *
+ * @param value - A string or an array of strings representing the header values.
+ * If it's a string, values are expected to be comma-separated.
+ * @returns The first trimmed value from the multi-value header, or `undefined` if the input is invalid or empty.
+ *
+ * @example
+ * ```typescript
+ * getFirstHeaderValue("value1, value2, value3"); // "value1"
+ * getFirstHeaderValue(["value1", "value2"]); // "value1"
+ * getFirstHeaderValue(undefined); // undefined
+ * ```
+ */
+export function getFirstHeaderValue(
+ value: string | string[] | undefined | null,
+): string | undefined {
+ return value?.toString().split(',', 1)[0]?.trim();
+}
+
+/**
+ * Validates a request.
+ *
+ * @param request - The incoming `Request` object to validate.
+ * @param allowedHosts - A set of allowed hostnames.
+ * @throws Error if any of the validated headers contain invalid values.
+ */
+export function validateRequest(request: Request, allowedHosts: ReadonlySet): void {
+ validateHeaders(request);
+ validateUrl(new URL(request.url), allowedHosts);
+}
+
+/**
+ * Validates that the hostname of a given URL is allowed.
+ *
+ * @param url - The URL object to validate.
+ * @param allowedHosts - A set of allowed hostnames.
+ * @throws Error if the hostname is not in the allowlist.
+ */
+export function validateUrl(url: URL, allowedHosts: ReadonlySet): void {
+ const { hostname } = url;
+ if (!isHostAllowed(hostname, allowedHosts)) {
+ throw new Error(`URL with hostname "${hostname}" is not allowed.`);
+ }
+}
+
+/**
+ * Clones a request and patches the `get` method of the request headers to validate the host headers.
+ * @param request - The request to validate.
+ * @param allowedHosts - A set of allowed hostnames.
+ * @returns An object containing the cloned request and a promise that resolves to an error
+ * if any of the validated headers contain invalid values.
+ */
+export function cloneRequestAndPatchHeaders(
+ request: Request,
+ allowedHosts: ReadonlySet,
+): { request: Request; onError: Promise } {
+ let onError: (value: Error) => void;
+ const onErrorPromise = new Promise((resolve) => {
+ onError = resolve;
+ });
+
+ const clonedReq = new Request(request.clone(), {
+ signal: request.signal,
+ });
+
+ const headers = clonedReq.headers;
+
+ const originalGet = headers.get;
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
+ (headers.get as typeof originalGet) = function (name) {
+ const value = originalGet.call(headers, name);
+ if (!value) {
+ return value;
+ }
+
+ validateHeader(name, value, allowedHosts, onError);
+
+ return value;
+ };
+
+ const originalValues = headers.values;
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
+ (headers.values as typeof originalValues) = function () {
+ for (const name of HOST_HEADERS_TO_VALIDATE) {
+ validateHeader(name, originalGet.call(headers, name), allowedHosts, onError);
+ }
+
+ return originalValues.call(headers);
+ };
+
+ const originalEntries = headers.entries;
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
+ (headers.entries as typeof originalEntries) = function () {
+ const iterator = originalEntries.call(headers);
+
+ return {
+ next() {
+ const result = iterator.next();
+ if (!result.done) {
+ const [key, value] = result.value;
+ validateHeader(key, value, allowedHosts, onError);
+ }
+
+ return result;
+ },
+ [Symbol.iterator]() {
+ return this;
+ },
+ };
+ };
+
+ // Ensure for...of loops use the new patched entries
+ (headers[Symbol.iterator] as typeof originalEntries) = headers.entries;
+
+ return { request: clonedReq, onError: onErrorPromise };
+}
+
+/**
+ * Validates a specific header value against the allowed hosts.
+ * @param name - The name of the header to validate.
+ * @param value - The value of the header to validate.
+ * @param allowedHosts - A set of allowed hostnames.
+ * @param onError - A callback function to call if the header value is invalid.
+ * @throws Error if the header value is invalid.
+ */
+function validateHeader(
+ name: string,
+ value: string | null,
+ allowedHosts: ReadonlySet,
+ onError: (value: Error) => void,
+): void {
+ if (!value) {
+ return;
+ }
+
+ if (!HOST_HEADERS_TO_VALIDATE.has(name.toLowerCase())) {
+ return;
+ }
+
+ try {
+ verifyHostAllowed(name, value, allowedHosts);
+ } catch (error) {
+ onError(error as Error);
+
+ throw error;
+ }
+}
+
+/**
+ * Validates a specific host header value against the allowed hosts.
+ *
+ * @param headerName - The name of the header to validate (e.g., 'host', 'x-forwarded-host').
+ * @param headerValue - The value of the header to validate.
+ * @param allowedHosts - A set of allowed hostnames.
+ * @throws Error if the header value is invalid or the hostname is not in the allowlist.
+ */
+function verifyHostAllowed(
+ headerName: string,
+ headerValue: string,
+ allowedHosts: ReadonlySet,
+): void {
+ const value = getFirstHeaderValue(headerValue);
+ if (!value) {
+ return;
+ }
+
+ const url = `http://${value}`;
+ if (!URL.canParse(url)) {
+ throw new Error(`Header "${headerName}" contains an invalid value and cannot be parsed.`);
+ }
+
+ const { hostname } = new URL(url);
+ if (!isHostAllowed(hostname, allowedHosts)) {
+ throw new Error(`Header "${headerName}" with value "${value}" is not allowed.`);
+ }
+}
+
+/**
+ * Checks if the hostname is allowed.
+ * @param hostname - The hostname to check.
+ * @param allowedHosts - A set of allowed hostnames.
+ * @returns `true` if the hostname is allowed, `false` otherwise.
+ */
+function isHostAllowed(hostname: string, allowedHosts: ReadonlySet): boolean {
+ if (allowedHosts.has(hostname)) {
+ return true;
+ }
+
+ for (const allowedHost of allowedHosts) {
+ if (!allowedHost.startsWith('*.')) {
+ continue;
+ }
+
+ const domain = allowedHost.slice(1);
+ if (hostname.endsWith(domain)) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+/**
+ * Validates the headers of an incoming request.
+ *
+ * @param request - The incoming `Request` object containing the headers to validate.
+ * @throws Error if any of the validated headers contain invalid values.
+ */
+function validateHeaders(request: Request): void {
+ const headers = request.headers;
+ for (const headerName of HOST_HEADERS_TO_VALIDATE) {
+ const headerValue = getFirstHeaderValue(headers.get(headerName));
+ if (headerValue && !VALID_HOST_REGEX.test(headerValue)) {
+ throw new Error(`Header "${headerName}" contains characters that are not allowed.`);
+ }
+ }
+
+ const xForwardedPort = getFirstHeaderValue(headers.get('x-forwarded-port'));
+ if (xForwardedPort && !VALID_PORT_REGEX.test(xForwardedPort)) {
+ throw new Error('Header "x-forwarded-port" must be a numeric value.');
+ }
+
+ const xForwardedProto = getFirstHeaderValue(headers.get('x-forwarded-proto'));
+ if (xForwardedProto && !VALID_PROTO_REGEX.test(xForwardedProto)) {
+ throw new Error('Header "x-forwarded-proto" must be either "http" or "https".');
+ }
+
+ const xForwardedPrefix = getFirstHeaderValue(headers.get('x-forwarded-prefix'));
+ if (xForwardedPrefix && INVALID_PREFIX_REGEX.test(xForwardedPrefix)) {
+ throw new Error(
+ 'Header "x-forwarded-prefix" must not start with multiple "/" or "\\" or contain ".", ".." path segments.',
+ );
+ }
+}
diff --git a/packages/angular/ssr/test/app-engine_spec.ts b/packages/angular/ssr/test/app-engine_spec.ts
index b08931b9400b..29d638a8c13f 100644
--- a/packages/angular/ssr/test/app-engine_spec.ts
+++ b/packages/angular/ssr/test/app-engine_spec.ts
@@ -11,13 +11,26 @@
import '@angular/compiler';
/* eslint-enable import/no-unassigned-import */
-import { Component } from '@angular/core';
+import { Component, REQUEST, inject } from '@angular/core';
import { destroyAngularServerApp, getOrCreateAngularServerApp } from '../src/app';
import { AngularAppEngine } from '../src/app-engine';
import { setAngularAppEngineManifest } from '../src/manifest';
import { RenderMode } from '../src/routes/route-config';
import { setAngularAppTestingManifest } from './testing-utils';
+@Component({
+ selector: 'app-home',
+ template: 'Home works',
+})
+class TestHomeComponent {
+ private request = inject(REQUEST);
+ constructor() {
+ // Force header access to trigger validation
+ this.request?.headers.get('host');
+ this.request?.headers.get('x-forwarded-host');
+ }
+}
+
function createEntryPoint(locale: string) {
return async () => {
@Component({
@@ -81,6 +94,7 @@ describe('AngularAppEngine', () => {
describe('Localized app', () => {
beforeAll(() => {
setAngularAppEngineManifest({
+ allowedHosts: ['example.com'],
// Note: Although we are testing only one locale, we need to configure two or more
// to ensure that we test a different code path.
entryPoints: {
@@ -160,6 +174,7 @@ describe('AngularAppEngine', () => {
describe('Localized app with single locale', () => {
beforeAll(() => {
setAngularAppEngineManifest({
+ allowedHosts: ['example.com'],
entryPoints: {
it: createEntryPoint('it'),
},
@@ -226,6 +241,7 @@ describe('AngularAppEngine', () => {
class HomeComponent {}
setAngularAppEngineManifest({
+ allowedHosts: ['example.com'],
entryPoints: {
'': async () => {
setAngularAppTestingManifest(
@@ -270,4 +286,144 @@ describe('AngularAppEngine', () => {
expect(await response?.text()).toContain('Home works');
});
});
+
+ describe('Invalid host headers', () => {
+ let consoleErrorSpy: jasmine.Spy;
+
+ describe('with allowed hosts configured', () => {
+ beforeAll(() => {
+ setAngularAppEngineManifest({
+ allowedHosts: ['example.com'],
+ entryPoints: {
+ '': async () => {
+ setAngularAppTestingManifest(
+ [{ path: 'home', component: TestHomeComponent }],
+ [{ path: '**', renderMode: RenderMode.Server }],
+ );
+
+ return {
+ ɵgetOrCreateAngularServerApp: getOrCreateAngularServerApp,
+ ɵdestroyAngularServerApp: destroyAngularServerApp,
+ };
+ },
+ },
+ basePath: '/',
+ supportedLocales: { 'en-US': '' },
+ });
+
+ appEngine = new AngularAppEngine();
+ });
+
+ beforeEach(() => {
+ consoleErrorSpy = spyOn(console, 'error');
+ });
+
+ it('should return 400 when disallowed host', async () => {
+ const request = new Request('https://evil.com');
+ const response = await appEngine.handle(request);
+ expect(response).not.toBeNull();
+ expect(response?.status).toBe(400);
+ expect(await response?.text()).toContain('URL with hostname "evil.com" is not allowed.');
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
+ jasmine.stringMatching('URL with hostname "evil.com" is not allowed.'),
+ );
+ });
+
+ it('should return 400 when disallowed host header', async () => {
+ const request = new Request('https://example.com/home', {
+ headers: { 'host': 'evil.com' },
+ });
+ const response = await appEngine.handle(request);
+ expect(response).not.toBeNull();
+ expect(response?.status).toBe(400);
+ expect(await response?.text()).toContain(
+ 'Header "host" with value "evil.com" is not allowed.',
+ );
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
+ jasmine.stringMatching('Header "host" with value "evil.com" is not allowed.'),
+ );
+ });
+
+ it('should return 400 when disallowed x-forwarded-host header', async () => {
+ const request = new Request('https://example.com/home', {
+ headers: { 'x-forwarded-host': 'evil.com' },
+ });
+ const response = await appEngine.handle(request);
+ expect(response).not.toBeNull();
+ expect(response?.status).toBe(400);
+ expect(await response?.text()).toContain(
+ 'Header "x-forwarded-host" with value "evil.com" is not allowed.',
+ );
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
+ jasmine.stringMatching('Header "x-forwarded-host" with value "evil.com" is not allowed.'),
+ );
+ });
+
+ it('should return 400 when host with path separator', async () => {
+ const request = new Request('https://example.com/home', {
+ headers: { 'host': 'example.com/evil' },
+ });
+ const response = await appEngine.handle(request);
+ expect(response).not.toBeNull();
+ expect(response?.status).toBe(400);
+ expect(await response?.text()).toContain(
+ 'Header "host" contains characters that are not allowed.',
+ );
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
+ jasmine.stringMatching('Header "host" contains characters that are not allowed.'),
+ );
+ });
+ });
+
+ describe('without allowed hosts configured', () => {
+ beforeAll(() => {
+ setAngularAppEngineManifest({
+ allowedHosts: [],
+ entryPoints: {
+ '': async () => {
+ setAngularAppTestingManifest(
+ [{ path: 'home', component: TestHomeComponent }],
+ [{ path: '**', renderMode: RenderMode.Server }],
+ );
+
+ return {
+ ɵgetOrCreateAngularServerApp: getOrCreateAngularServerApp,
+ ɵdestroyAngularServerApp: destroyAngularServerApp,
+ };
+ },
+ },
+ basePath: '/',
+ supportedLocales: { 'en-US': '' },
+ });
+
+ appEngine = new AngularAppEngine();
+ });
+
+ beforeEach(() => {
+ consoleErrorSpy = spyOn(console, 'error');
+ });
+
+ it('should log error and fallback to CSR when disallowed host', async () => {
+ const request = new Request('https://example.com');
+ const response = await appEngine.handle(request);
+ expect(response).not.toBeNull();
+ expect(await response?.text()).toContain('CSR page');
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
+ jasmine.stringMatching('URL with hostname "example.com" is not allowed.'),
+ );
+ });
+
+ it('should log error and fallback to CSR when host with path separator', async () => {
+ const request = new Request('https://example.com/home', {
+ headers: { 'host': 'example.com/evil' },
+ });
+ const response = await appEngine.handle(request);
+ expect(response).not.toBeNull();
+ expect(await response?.text()).toContain('CSR page');
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
+ jasmine.stringMatching('Header "host" contains characters that are not allowed.'),
+ );
+ });
+ });
+ });
});
diff --git a/packages/angular/ssr/test/utils/url_spec.ts b/packages/angular/ssr/test/utils/url_spec.ts
index 9a7a7cb3ad49..a108c7ff1df6 100644
--- a/packages/angular/ssr/test/utils/url_spec.ts
+++ b/packages/angular/ssr/test/utils/url_spec.ts
@@ -100,6 +100,18 @@ describe('URL Utils', () => {
it('should handle an all-empty URL parts', () => {
expect(joinUrlParts('', '')).toBe('/');
});
+
+ it('should normalize parts with multiple leading and trailing slashes', () => {
+ expect(joinUrlParts('//path//', '///to///', '//resource//')).toBe('/path/to/resource');
+ });
+
+ it('should handle a single part', () => {
+ expect(joinUrlParts('path')).toBe('/path');
+ });
+
+ it('should handle parts containing only slashes', () => {
+ expect(joinUrlParts('//', '///')).toBe('/');
+ });
});
describe('stripIndexHtmlFromURL', () => {
diff --git a/packages/angular/ssr/test/utils/validation_spec.ts b/packages/angular/ssr/test/utils/validation_spec.ts
new file mode 100644
index 000000000000..10ab896e36f1
--- /dev/null
+++ b/packages/angular/ssr/test/utils/validation_spec.ts
@@ -0,0 +1,307 @@
+/**
+ * @license
+ * Copyright Google LLC All Rights Reserved.
+ *
+ * Use of this source code is governed by an MIT-style license that can be
+ * found in the LICENSE file at https://angular.dev/license
+ */
+
+import {
+ cloneRequestAndPatchHeaders,
+ getFirstHeaderValue,
+ validateRequest,
+ validateUrl,
+} from '../../src/utils/validation';
+
+describe('Validation Utils', () => {
+ describe('getFirstHeaderValue', () => {
+ it('should return the first value from a comma-separated string', () => {
+ expect(getFirstHeaderValue('value1, value2')).toBe('value1');
+ });
+
+ it('should return the value if it is a single string', () => {
+ expect(getFirstHeaderValue('value1')).toBe('value1');
+ });
+
+ it('should return the first value from an array of strings', () => {
+ expect(getFirstHeaderValue(['value1', 'value2'])).toBe('value1');
+ });
+
+ it('should return undefined for null or undefined', () => {
+ expect(getFirstHeaderValue(null)).toBeUndefined();
+ expect(getFirstHeaderValue(undefined)).toBeUndefined();
+ });
+
+ it('should return empty string for empty string input', () => {
+ expect(getFirstHeaderValue('')).toBe('');
+ });
+ });
+
+ describe('validateUrl', () => {
+ const allowedHosts = new Set(['example.com', '*.google.com']);
+
+ it('should pass for allowed hostname', () => {
+ expect(() => validateUrl(new URL('http://example.com'), allowedHosts)).not.toThrow();
+ });
+
+ it('should pass for wildcard allowed hostname', () => {
+ expect(() => validateUrl(new URL('http://foo.google.com'), allowedHosts)).not.toThrow();
+ });
+
+ it('should throw for disallowed hostname', () => {
+ expect(() => validateUrl(new URL('http://evil.com'), allowedHosts)).toThrowError(
+ /URL with hostname "evil.com" is not allowed/,
+ );
+ });
+
+ it('should match subdomains for wildcard', () => {
+ expect(() => validateUrl(new URL('http://sub.foo.google.com'), allowedHosts)).not.toThrow();
+ });
+
+ it('should not match base domain for wildcard (*.google.com vs google.com)', () => {
+ // Logic: hostname.endsWith('.google.com') -> 'google.com'.endsWith('.google.com') is false
+ expect(() => validateUrl(new URL('http://google.com'), allowedHosts)).toThrowError(
+ /URL with hostname "google.com" is not allowed/,
+ );
+ });
+ });
+
+ describe('validateRequest', () => {
+ const allowedHosts = new Set(['example.com']);
+
+ it('should pass for valid request', () => {
+ const req = new Request('http://example.com', {
+ headers: {
+ 'x-forwarded-port': '443',
+ 'x-forwarded-proto': 'https',
+ },
+ });
+
+ expect(() => validateRequest(req, allowedHosts)).not.toThrow();
+ });
+
+ it('should throw if URL hostname is invalid', () => {
+ const req = new Request('http://evil.com');
+
+ expect(() => validateRequest(req, allowedHosts)).toThrowError(
+ /URL with hostname "evil.com" is not allowed/,
+ );
+ });
+
+ it('should throw if x-forwarded-port is invalid', () => {
+ const req = new Request('http://example.com', {
+ headers: { 'x-forwarded-port': 'abc' },
+ });
+
+ expect(() => validateRequest(req, allowedHosts)).toThrowError(
+ 'Header "x-forwarded-port" must be a numeric value.',
+ );
+ });
+
+ it('should throw if x-forwarded-proto is invalid', () => {
+ const req = new Request('http://example.com', {
+ headers: { 'x-forwarded-proto': 'ftp' },
+ });
+ expect(() => validateRequest(req, allowedHosts)).toThrowError(
+ 'Header "x-forwarded-proto" must be either "http" or "https".',
+ );
+ });
+
+ it('should throw if host contains path separators', () => {
+ const req = new Request('http://example.com', {
+ headers: { 'host': 'example.com/bad' },
+ });
+ expect(() => validateRequest(req, allowedHosts)).toThrowError(
+ 'Header "host" contains characters that are not allowed.',
+ );
+ });
+
+ it('should throw if x-forwarded-host contains path separators', () => {
+ const req = new Request('http://example.com', {
+ headers: { 'x-forwarded-host': 'example.com/bad' },
+ });
+ expect(() => validateRequest(req, allowedHosts)).toThrowError(
+ 'Header "x-forwarded-host" contains characters that are not allowed.',
+ );
+ });
+
+ it('should throw error if x-forwarded-prefix starts with multiple slashes or backslashes', () => {
+ const inputs = ['//evil', '\\\\evil', '/\\evil', '\\/evil'];
+
+ for (const prefix of inputs) {
+ const request = new Request('https://example.com', {
+ headers: {
+ 'x-forwarded-prefix': prefix,
+ },
+ });
+
+ expect(() => validateRequest(request, allowedHosts))
+ .withContext(`Prefix: "${prefix}"`)
+ .toThrowError(
+ 'Header "x-forwarded-prefix" must not start with multiple "/" or "\\" or contain ".", ".." path segments.',
+ );
+ }
+ });
+
+ it('should throw error if x-forwarded-prefix contains dot segments', () => {
+ const inputs = [
+ '/./',
+ '/../',
+ '/foo/./bar',
+ '/foo/../bar',
+ '/.',
+ '/..',
+ './',
+ '../',
+ '.\\',
+ '..\\',
+ '/foo/.\\bar',
+ '/foo/..\\bar',
+ '.',
+ '..',
+ ];
+
+ for (const prefix of inputs) {
+ const request = new Request('https://example.com', {
+ headers: {
+ 'x-forwarded-prefix': prefix,
+ },
+ });
+
+ expect(() => validateRequest(request, allowedHosts))
+ .withContext(`Prefix: "${prefix}"`)
+ .toThrowError(
+ 'Header "x-forwarded-prefix" must not start with multiple "/" or "\\" or contain ".", ".." path segments.',
+ );
+ }
+ });
+
+ it('should validate x-forwarded-prefix with valid dot usage', () => {
+ const inputs = ['/foo.bar', '/foo.bar/baz', '/v1.2', '/.well-known'];
+
+ for (const prefix of inputs) {
+ const request = new Request('https://example.com', {
+ headers: {
+ 'x-forwarded-prefix': prefix,
+ },
+ });
+
+ expect(() => validateRequest(request, allowedHosts))
+ .withContext(`Prefix: "${prefix}"`)
+ .not.toThrow();
+ }
+ });
+ });
+
+ describe('cloneRequestAndPatchHeaders', () => {
+ const allowedHosts = new Set(['example.com', '*.valid.com']);
+
+ it('should validate host header when accessed via get()', async () => {
+ const req = new Request('http://example.com', {
+ headers: { 'host': 'evil.com' },
+ });
+ const { request: secured, onError } = cloneRequestAndPatchHeaders(req, allowedHosts);
+
+ expect(() => secured.headers.get('host')).toThrowError(
+ 'Header "host" with value "evil.com" is not allowed.',
+ );
+ await expectAsync(onError).toBeResolvedTo(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching('Header "host" with value "evil.com" is not allowed'),
+ }),
+ );
+ });
+
+ it('should allow valid host header', () => {
+ const req = new Request('http://example.com', {
+ headers: { 'host': 'example.com' },
+ });
+ const { request: secured } = cloneRequestAndPatchHeaders(req, allowedHosts);
+ expect(secured.headers.get('host')).toBe('example.com');
+ });
+
+ it('should validate x-forwarded-host header', async () => {
+ const req = new Request('http://example.com', {
+ headers: { 'x-forwarded-host': 'evil.com' },
+ });
+ const { request: secured, onError } = cloneRequestAndPatchHeaders(req, allowedHosts);
+
+ expect(() => secured.headers.get('x-forwarded-host')).toThrowError(
+ 'Header "x-forwarded-host" with value "evil.com" is not allowed.',
+ );
+ await expectAsync(onError).toBeResolvedTo(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching(
+ 'Header "x-forwarded-host" with value "evil.com" is not allowed',
+ ),
+ }),
+ );
+ });
+
+ it('should allow accessing other headers without validation', () => {
+ const req = new Request('http://example.com', {
+ headers: { 'accept': 'application/json' },
+ });
+ const { request: secured } = cloneRequestAndPatchHeaders(req, allowedHosts);
+
+ expect(secured.headers.get('accept')).toBe('application/json');
+ });
+
+ it('should validate headers when iterating with entries()', async () => {
+ const req = new Request('http://example.com', {
+ headers: { 'host': 'evil.com' },
+ });
+ const { request: secured, onError } = cloneRequestAndPatchHeaders(req, allowedHosts);
+
+ expect(() => {
+ for (const _ of secured.headers.entries()) {
+ // access the header to trigger the validation
+ }
+ }).toThrowError('Header "host" with value "evil.com" is not allowed.');
+
+ await expectAsync(onError).toBeResolvedTo(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching('Header "host" with value "evil.com" is not allowed.'),
+ }),
+ );
+ });
+
+ it('should validate headers when iterating with values()', async () => {
+ const req = new Request('http://example.com', {
+ headers: { 'host': 'evil.com' },
+ });
+ const { request: secured, onError } = cloneRequestAndPatchHeaders(req, allowedHosts);
+
+ expect(() => {
+ for (const _ of secured.headers.values()) {
+ // access the header to trigger the validation
+ }
+ }).toThrowError('Header "host" with value "evil.com" is not allowed.');
+
+ await expectAsync(onError).toBeResolvedTo(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching('Header "host" with value "evil.com" is not allowed.'),
+ }),
+ );
+ });
+
+ it('should validate headers when iterating with for...of', async () => {
+ const req = new Request('http://example.com', {
+ headers: { 'host': 'evil.com' },
+ });
+ const { request: secured, onError } = cloneRequestAndPatchHeaders(req, allowedHosts);
+
+ expect(() => {
+ for (const _ of secured.headers) {
+ // access the header to trigger the validation
+ }
+ }).toThrowError('Header "host" with value "evil.com" is not allowed.');
+
+ await expectAsync(onError).toBeResolvedTo(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching('Header "host" with value "evil.com" is not allowed.'),
+ }),
+ );
+ });
+ });
+});
diff --git a/packages/angular_devkit/build_angular/package.json b/packages/angular_devkit/build_angular/package.json
index f557d8362da9..5ebe0bd0b01a 100644
--- a/packages/angular_devkit/build_angular/package.json
+++ b/packages/angular_devkit/build_angular/package.json
@@ -68,7 +68,7 @@
"@angular/ssr": "workspace:*",
"@web/test-runner": "0.20.2",
"browser-sync": "3.0.4",
- "ng-packagr": "21.2.0-next.0",
+ "ng-packagr": "21.2.0",
"undici": "7.22.0"
},
"peerDependencies": {
diff --git a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/index.ts b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/index.ts
index 8ec879993e4f..e919f188e0bf 100644
--- a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/index.ts
+++ b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/index.ts
@@ -115,7 +115,13 @@ export function execute(
return of([b, s]);
}
- return startNodeServer(s, nodeServerPort, context.logger, !!options.inspect).pipe(
+ return startNodeServer(
+ s,
+ nodeServerPort,
+ options.host,
+ context.logger,
+ !!options.inspect,
+ ).pipe(
map(() => [b, s]),
catchError((err) => {
context.logger.error(`A server error has occurred.\n${mapErrorToMessage(err)}`);
@@ -217,12 +223,13 @@ export function log(
function startNodeServer(
serverOutput: BuilderOutput,
port: number,
+ host: string | undefined,
logger: logging.LoggerApi,
inspectMode = false,
): Observable {
const outputPath = serverOutput.outputPath as string;
const path = join(outputPath, 'main.js');
- const env = { ...process.env, PORT: '' + port };
+ const env = { ...process.env, PORT: '' + port, NG_ALLOWED_HOSTS: host ?? 'localhost' };
const args = ['--enable-source-maps', `"${path}"`];
if (inspectMode) {
diff --git a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/proxy_spec.ts b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/proxy_spec.ts
index cbde961e59e4..425aad21dada 100644
--- a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/proxy_spec.ts
+++ b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/proxy_spec.ts
@@ -52,11 +52,12 @@ describe('Serve SSR Builder', () => {
}));
server.use((req, res, next) => {
+ const { protocol, originalUrl, baseUrl, headers } = req;
commonEngine
.render({
bootstrap: AppServerModule,
documentFilePath: indexHtml,
- url: req.originalUrl,
+ url: \`\${protocol}://\${headers.host}\${originalUrl}\`,
publicPath: distFolder,
})
.then((html) => res.send(html))
diff --git a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/ssl_spec.ts b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/ssl_spec.ts
index 7651b2387c16..e61117812bbe 100644
--- a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/ssl_spec.ts
+++ b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/ssl_spec.ts
@@ -52,11 +52,12 @@ describe('Serve SSR Builder', () => {
}));
server.use((req, res, next) => {
+ const { protocol, originalUrl, baseUrl, headers } = req;
commonEngine
.render({
bootstrap: AppServerModule,
documentFilePath: indexHtml,
- url: req.originalUrl,
+ url: \`\${protocol}://\${headers.host}\${originalUrl}\`,
publicPath: distFolder,
})
.then((html) => res.send(html))
diff --git a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/works_spec.ts b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/works_spec.ts
index 64c56024f089..c8c70d148e63 100644
--- a/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/works_spec.ts
+++ b/packages/angular_devkit/build_angular/src/builders/ssr-dev-server/specs/works_spec.ts
@@ -51,11 +51,12 @@ describe('Serve SSR Builder', () => {
}));
server.use((req, res, next) => {
+ const { protocol, originalUrl, baseUrl, headers } = req;
commonEngine
.render({
bootstrap: AppServerModule,
documentFilePath: indexHtml,
- url: req.originalUrl,
+ url: \`\${protocol}://\${headers.host}\${originalUrl}\`,
publicPath: distFolder,
})
.then((html) => res.send(html))
diff --git a/packages/ngtools/webpack/package.json b/packages/ngtools/webpack/package.json
index ef531d078cea..87b197e21d7b 100644
--- a/packages/ngtools/webpack/package.json
+++ b/packages/ngtools/webpack/package.json
@@ -27,8 +27,8 @@
},
"devDependencies": {
"@angular-devkit/core": "workspace:0.0.0-PLACEHOLDER",
- "@angular/compiler": "21.2.0-next.3",
- "@angular/compiler-cli": "21.2.0-next.3",
+ "@angular/compiler": "21.2.0",
+ "@angular/compiler-cli": "21.2.0",
"typescript": "5.9.3",
"webpack": "5.105.2"
}
diff --git a/packages/schematics/angular/application/files/common-files/src/app/app__suffix__.html.template b/packages/schematics/angular/application/files/common-files/src/app/app__suffix__.html.template
index 7d9cf8841c9d..37512bbc01e5 100644
--- a/packages/schematics/angular/application/files/common-files/src/app/app__suffix__.html.template
+++ b/packages/schematics/angular/application/files/common-files/src/app/app__suffix__.html.template
@@ -42,6 +42,8 @@
box-sizing: border-box;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
+ display: block;
+ height: 100dvh;
}
h1 {
diff --git a/packages/schematics/angular/application/index.ts b/packages/schematics/angular/application/index.ts
index e84a40530032..730debc71a8e 100644
--- a/packages/schematics/angular/application/index.ts
+++ b/packages/schematics/angular/application/index.ts
@@ -187,7 +187,7 @@ function addDependenciesToPackageJson(options: ApplicationOptions): Rule {
);
}
- if (!options.skipTests) {
+ if (!options.skipTests && !options.minimal) {
rules.push(...addTestRunnerDependencies(options.testRunner, !!options.skipInstall));
}
diff --git a/packages/schematics/angular/application/index_spec.ts b/packages/schematics/angular/application/index_spec.ts
index 215664398b8e..0fe4d142dc4a 100644
--- a/packages/schematics/angular/application/index_spec.ts
+++ b/packages/schematics/angular/application/index_spec.ts
@@ -167,6 +167,18 @@ describe('Application Schematic', () => {
);
});
+ it(`should not add test dependencies with "minimal" enabled`, async () => {
+ const tree = await schematicRunner.runSchematic(
+ 'application',
+ { ...defaultOptions, minimal: true },
+ workspaceTree,
+ );
+
+ const packageJson = JSON.parse(tree.readContent('package.json'));
+ expect(packageJson.devDependencies['vitest']).toBeUndefined();
+ expect(packageJson.devDependencies['jsdom']).toBeUndefined();
+ });
+
it('should install npm dependencies when `skipInstall` is false', async () => {
await schematicRunner.runSchematic(
'application',
diff --git a/packages/schematics/angular/ssr/files/server-builder/server.ts.template b/packages/schematics/angular/ssr/files/server-builder/server.ts.template
index 7327c26532ea..8d6a9660dfae 100644
--- a/packages/schematics/angular/ssr/files/server-builder/server.ts.template
+++ b/packages/schematics/angular/ssr/files/server-builder/server.ts.template
@@ -13,7 +13,9 @@ export function app(): express.Express {
? join(distFolder, 'index.original.html')
: join(distFolder, 'index.html');
- const commonEngine = new CommonEngine();
+ const commonEngine = new CommonEngine({
+ allowedHosts: [/* Configure your hosts here */]
+ });
server.set('view engine', 'html');
server.set('views', distFolder);
diff --git a/packages/schematics/angular/ssr/index.ts b/packages/schematics/angular/ssr/index.ts
index 6e27eab47cd5..49e57d523268 100644
--- a/packages/schematics/angular/ssr/index.ts
+++ b/packages/schematics/angular/ssr/index.ts
@@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.dev/license
*/
-import { isJsonObject } from '@angular-devkit/core';
+import { JsonObject, isJsonObject } from '@angular-devkit/core';
import {
Rule,
RuleFactory,
@@ -206,6 +206,10 @@ function updateApplicationBuilderWorkspaceConfigRule(
buildTarget.options = {
...buildTarget.options,
+ security: {
+ ...((buildTarget.options?.security as JsonObject | undefined) ?? {}),
+ allowedHosts: [],
+ },
outputPath,
outputMode: 'server',
ssr: {
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index 65a1b3421df4..4fa808440888 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -20,47 +20,47 @@ importers:
built: true
devDependencies:
'@angular/animations':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ specifier: 21.2.0
+ version: 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
'@angular/cdk':
- specifier: 21.2.0-next.4
- version: 21.2.0-next.4(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
'@angular/common':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
'@angular/compiler':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3
+ specifier: 21.2.0
+ version: 21.2.0
'@angular/compiler-cli':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3)
'@angular/core':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
'@angular/forms':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
'@angular/localize':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/compiler-cli@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3))(@angular/compiler@21.2.0-next.3)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler-cli@21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3))(@angular/compiler@21.2.0)
'@angular/material':
- specifier: 21.2.0-next.4
- version: 21.2.0-next.4(75d10325bbcfc22e53b1e47f427450d8)
+ specifier: 21.2.0
+ version: 21.2.0(16f55f8bfdd6508e9b3d27b33a1d1b40)
'@angular/ng-dev':
- specifier: https://github.com/angular/dev-infra-private-ng-dev-builds.git#2a5e8e5b5398ae13a8d9a963ac980707313a6c9e
- version: https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/2a5e8e5b5398ae13a8d9a963ac980707313a6c9e(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))
+ specifier: https://github.com/angular/dev-infra-private-ng-dev-builds.git#c78d7a03ae1ca76d2946651d0dd7f476f09b4374
+ version: https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/c78d7a03ae1ca76d2946651d0dd7f476f09b4374(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))
'@angular/platform-browser':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ specifier: 21.2.0
+ version: 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
'@angular/platform-server':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/compiler@21.2.0-next.3)(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/compiler@21.2.0)(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
'@angular/router':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
'@angular/service-worker':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
'@babel/core':
specifier: 7.29.0
version: 7.29.0
@@ -93,7 +93,7 @@ importers:
version: 16.0.3(rollup@4.57.1)
'@stylistic/eslint-plugin':
specifier: ^5.0.0
- version: 5.8.0(eslint@9.39.2(jiti@2.6.1))
+ version: 5.9.0(eslint@9.39.2(jiti@2.6.1))
'@types/babel__core':
specifier: 7.20.5
version: 7.20.5
@@ -129,7 +129,7 @@ importers:
version: 3.0.0(esbuild@0.27.3)
'@types/lodash':
specifier: ^4.17.0
- version: 4.17.23
+ version: 4.17.24
'@types/node':
specifier: ^22.12.0
version: 22.19.11
@@ -297,7 +297,7 @@ importers:
version: 10.3.1
zone.js:
specifier: ^0.16.0
- version: 0.16.0
+ version: 0.16.1
modules/testing/builder:
devDependencies:
@@ -315,7 +315,7 @@ importers:
version: link:../../../packages/angular/ssr
'@vitest/coverage-v8':
specifier: 4.0.18
- version: 4.0.18(vitest@4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.9)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))
+ version: 4.0.18(vitest@4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.13)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))
browser-sync:
specifier: 3.0.4
version: 3.0.4(bufferutil@4.1.0)(utf-8-validate@6.0.6)
@@ -327,7 +327,7 @@ importers:
version: 7.8.2
vitest:
specifier: 4.0.18
- version: 4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.9)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ version: 4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.13)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
packages/angular/build:
dependencies:
@@ -348,10 +348,10 @@ importers:
version: 7.24.7
'@inquirer/confirm':
specifier: 5.1.21
- version: 5.1.21(@types/node@24.10.9)
+ version: 5.1.21(@types/node@24.10.13)
'@vitejs/plugin-basic-ssl':
specifier: 2.1.4
- version: 2.1.4(vite@7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))
+ version: 2.1.4(vite@7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))
beasties:
specifier: 0.4.1
version: 0.4.1
@@ -408,7 +408,7 @@ importers:
version: 7.22.0
vite:
specifier: 7.3.1
- version: 7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ version: 7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
watchpack:
specifier: 2.5.1
version: 2.5.1
@@ -426,8 +426,8 @@ importers:
specifier: 4.4.2
version: 4.4.2
ng-packagr:
- specifier: 21.2.0-next.0
- version: 21.2.0-next.0(@angular/compiler-cli@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3))(tslib@2.8.1)(typescript@5.9.3)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler-cli@21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3))(tslib@2.8.1)(typescript@5.9.3)
postcss:
specifier: 8.5.6
version: 8.5.6
@@ -436,7 +436,7 @@ importers:
version: 7.8.2
vitest:
specifier: 4.0.18
- version: 4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.9)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ version: 4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.13)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
optionalDependencies:
lmdb:
specifier: 3.5.1
@@ -455,10 +455,10 @@ importers:
version: link:../../angular_devkit/schematics
'@inquirer/prompts':
specifier: 7.10.1
- version: 7.10.1(@types/node@24.10.9)
+ version: 7.10.1(@types/node@24.10.13)
'@listr2/prompt-adapter-inquirer':
specifier: 3.0.5
- version: 3.0.5(@inquirer/prompts@7.10.1(@types/node@24.10.9))(@types/node@24.10.9)(listr2@9.0.5)
+ version: 3.0.5(@inquirer/prompts@7.10.1(@types/node@24.10.13))(@types/node@24.10.13)(listr2@9.0.5)
'@modelcontextprotocol/sdk':
specifier: 1.26.0
version: 1.26.0(zod@4.3.6)
@@ -521,23 +521,23 @@ importers:
specifier: workspace:*
version: link:../../angular_devkit/schematics
'@angular/common':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
'@angular/compiler':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3
+ specifier: 21.2.0
+ version: 21.2.0
'@angular/core':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
'@angular/platform-browser':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ specifier: 21.2.0
+ version: 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
'@angular/platform-server':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/compiler@21.2.0-next.3)(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/compiler@21.2.0)(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
'@angular/router':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
'@schematics/angular':
specifier: workspace:*
version: link:../../schematics/angular
@@ -735,8 +735,8 @@ importers:
specifier: 3.0.4
version: 3.0.4(bufferutil@4.1.0)(utf-8-validate@6.0.6)
ng-packagr:
- specifier: 21.2.0-next.0
- version: 21.2.0-next.0(@angular/compiler-cli@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3))(tslib@2.8.1)(typescript@5.9.3)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler-cli@21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3))(tslib@2.8.1)(typescript@5.9.3)
undici:
specifier: 7.22.0
version: 7.22.0
@@ -820,7 +820,7 @@ importers:
version: link:../schematics
'@inquirer/prompts':
specifier: 7.10.1
- version: 7.10.1(@types/node@24.10.9)
+ version: 7.10.1(@types/node@24.10.13)
packages/ngtools/webpack:
devDependencies:
@@ -828,11 +828,11 @@ importers:
specifier: workspace:0.0.0-PLACEHOLDER
version: link:../../angular_devkit/core
'@angular/compiler':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3
+ specifier: 21.2.0
+ version: 21.2.0
'@angular/compiler-cli':
- specifier: 21.2.0-next.3
- version: 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3)
+ specifier: 21.2.0
+ version: 21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3)
typescript:
specifier: 5.9.3
version: 5.9.3
@@ -944,47 +944,47 @@ packages:
resolution: {integrity: sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==}
engines: {node: '>=6.0.0'}
- '@angular/animations@21.2.0-next.3':
- resolution: {integrity: sha512-DTiRWm2aBlX+uVS1K3PnEbJZ4SQbWBPTl898lUYJ79WArPaHoTtrBurZ5zcGA6R5OyBfAvv3MvU0qZtihFIMoA==}
+ '@angular/animations@21.2.0':
+ resolution: {integrity: sha512-SzlLoMT/r5wKqPicx5okCAiN5UD5+VE7x/F1G6gSJCcnBfbK5PqHPUmDnMW4jw9Ode06KZDT7ntstn6fG+Ld8w==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/core': 21.2.0-next.3
+ '@angular/core': 21.2.0
- '@angular/cdk@21.2.0-next.4':
- resolution: {integrity: sha512-GbcDW+qeSV4QlPQojGh5jCPATGfClCYnZhEGLXfabfEy/uhKGehILW37FK4VRhLdZKW9nMTs7w436jaovh84/A==}
+ '@angular/cdk@21.2.0':
+ resolution: {integrity: sha512-1P0TNL1F51NC7JAaXabaAHY7Y1zBloLSZXfml1POa4a116V+y/QZfPGsxM0LwD1qSSXhSb2LNl7duTtJAP39bA==}
peerDependencies:
- '@angular/common': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
- '@angular/core': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
- '@angular/platform-browser': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
+ '@angular/common': ^21.0.0 || ^22.0.0
+ '@angular/core': ^21.0.0 || ^22.0.0
+ '@angular/platform-browser': ^21.0.0 || ^22.0.0
rxjs: ^6.5.3 || ^7.4.0
- '@angular/common@21.2.0-next.3':
- resolution: {integrity: sha512-7Mfsk/ahcKqvPFW1e9IMnfyvsyGKARIORdVxI8r/05N+Bt1aeW6iZbteEqM2cCEFZ4UqWjkaqdlBZndevqfqsA==}
+ '@angular/common@21.2.0':
+ resolution: {integrity: sha512-6zJMPi0i/XDniEgv3/t2BjuDHiOG44lgIR5PYyxqGpgJ0kqB5hku/0TuentNEi1VnBYgthnfhjek7c+lakXmhw==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/core': 21.2.0-next.3
+ '@angular/core': 21.2.0
rxjs: ^6.5.3 || ^7.4.0
- '@angular/compiler-cli@21.2.0-next.3':
- resolution: {integrity: sha512-q3OUWuK/mspdxRfPQq2RPnGosq7yYzWzK+FikEYHv9I2pPTqt1cXWArWzknLouxs2zVaRhpaqXreCGtBeicYwg==}
+ '@angular/compiler-cli@21.2.0':
+ resolution: {integrity: sha512-gZd58p0/JjgdxMX3v+LjCB6e3dBIfNVr/YzXoh55TfffdBCUQY94hl1+DFQkJ72K5EX+1zbaz03dIm30kw1bGw==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
hasBin: true
peerDependencies:
- '@angular/compiler': 21.2.0-next.3
- typescript: '>=5.9 <6.0'
+ '@angular/compiler': 21.2.0
+ typescript: '>=5.9 <6.1'
peerDependenciesMeta:
typescript:
optional: true
- '@angular/compiler@21.2.0-next.3':
- resolution: {integrity: sha512-vMNEt+qFt4+k7lGoalHqmxgB6XiEWUTG7UaT9BkcU5Ezs64RLfkLVSGd0MNAuE2RL1gOa0/wvIl11yv0pD2T7g==}
+ '@angular/compiler@21.2.0':
+ resolution: {integrity: sha512-0RPkma8UVNpse/VJcXT9w6SKzTMz4J/uMGj0l9enM1frg9xrx1fwi/lLmaVV9Nr9LfqPjQdxNFFlvaBB7g/2zg==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
- '@angular/core@21.2.0-next.3':
- resolution: {integrity: sha512-9cieyRQ3DIWANEQ1DfjOuZS8EdzcPzaikO1Vi81hNmsr7an8yok0oRXfonkkudFUtAE0Xl6B1wYyD4FZ3ig47g==}
+ '@angular/core@21.2.0':
+ resolution: {integrity: sha512-VnTbmZq3g3Q+s3nCZ8VUDMLjMezOg/bqUxAJ/DrRWCrEcTP5JO3mrNPs3FHj+qlB0T+BQP7uQv6QTzPVKybwoA==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/compiler': 21.2.0-next.3
+ '@angular/compiler': 21.2.0
rxjs: ^6.5.3 || ^7.4.0
zone.js: ~0.15.0 || ~0.16.0
peerDependenciesMeta:
@@ -993,78 +993,79 @@ packages:
zone.js:
optional: true
- '@angular/forms@21.2.0-next.3':
- resolution: {integrity: sha512-YHrLsKO1fozXw0jptI3kwCc5w13DRQzf/Q8ajmtu+o2oy1NjYuoPlRyTBuIufThYKgBqC2ZfWOCWkOa8gyyKQA==}
+ '@angular/forms@21.2.0':
+ resolution: {integrity: sha512-NduUtPWLauH/FLayEDkLyaKAGqKzXbcfO7468LOWCXN3crhNVQyIWRQPOUcdpoJwDAGLpN85m3DhJhXNnA9c5w==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/common': 21.2.0-next.3
- '@angular/core': 21.2.0-next.3
- '@angular/platform-browser': 21.2.0-next.3
+ '@angular/common': 21.2.0
+ '@angular/core': 21.2.0
+ '@angular/platform-browser': 21.2.0
rxjs: ^6.5.3 || ^7.4.0
- '@angular/localize@21.2.0-next.3':
- resolution: {integrity: sha512-348iYNZ5PrnjxiSVuPE9XomIWrZNYOPPDq/3VVVTl/f9G+iwmHGfCJ1GsGxwg31DRu3X6ZFKw4IBqUyfLJaJiQ==}
+ '@angular/localize@21.2.0':
+ resolution: {integrity: sha512-blVjzwHSaKbFNCQN/RZy8rSbFgajMw3kBzGrDY08atMDOPn90L2nE4dot+9d0JlKAX2gL8Qfx44YgIWBI5MfsA==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
hasBin: true
peerDependencies:
- '@angular/compiler': 21.2.0-next.3
- '@angular/compiler-cli': 21.2.0-next.3
+ '@angular/compiler': 21.2.0
+ '@angular/compiler-cli': 21.2.0
- '@angular/material@21.2.0-next.4':
- resolution: {integrity: sha512-mUNP2z/FehBfmdDtQNznXO/jjOm7FPCc0DBqdXsTgIyeHqf9UitwC6huBzSmzN8Kq/v/ftDp04Ph+Jjbp/6VaA==}
+ '@angular/material@21.2.0':
+ resolution: {integrity: sha512-GVJMgXa/jpy06b/txKMohuMDMu72Or576BSHKehnceYvI9sf25cnGGK1woNhALNDW449KReicAILujcFJ+mO6g==}
peerDependencies:
- '@angular/cdk': 21.2.0-next.4
- '@angular/common': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
- '@angular/core': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
- '@angular/forms': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
- '@angular/platform-browser': ^21.0.0-0 || ^21.1.0-0 || ^21.2.0-0 || ^21.3.0-0 || ^22.0.0-0
+ '@angular/cdk': 21.2.0
+ '@angular/common': ^21.0.0 || ^22.0.0
+ '@angular/core': ^21.0.0 || ^22.0.0
+ '@angular/forms': ^21.0.0 || ^22.0.0
+ '@angular/platform-browser': ^21.0.0 || ^22.0.0
rxjs: ^6.5.3 || ^7.4.0
- '@angular/ng-dev@https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/2a5e8e5b5398ae13a8d9a963ac980707313a6c9e':
- resolution: {tarball: https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/2a5e8e5b5398ae13a8d9a963ac980707313a6c9e}
- version: 0.0.0-469708f109a90884ca403d150d33079a3a5a8769
+ '@angular/ng-dev@https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/c78d7a03ae1ca76d2946651d0dd7f476f09b4374':
+ resolution: {tarball: https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/c78d7a03ae1ca76d2946651d0dd7f476f09b4374}
+ version: 0.0.0-61f9fe81614db102d95bb9bcb9909a304c6e6d10
hasBin: true
- '@angular/platform-browser@21.2.0-next.3':
- resolution: {integrity: sha512-YsKkQaZ4jPs/xicQ9baJ0hlSLfKfsBsXrwNBw7XjvnyinU8tlB707KZ0eThyx9FSoLeBO2FgrWM75NTiEd2Rvg==}
+ '@angular/platform-browser@21.2.0':
+ resolution: {integrity: sha512-IUGukpvvT2B5Dl76qzk6rY7UIHUT9u4BhT2AwVz+5JqcX9KwQtYD17Gt7wj6bvIgCXKWG+CfN8Zd9DECOCYWjg==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/animations': 21.2.0-next.3
- '@angular/common': 21.2.0-next.3
- '@angular/core': 21.2.0-next.3
+ '@angular/animations': 21.2.0
+ '@angular/common': 21.2.0
+ '@angular/core': 21.2.0
peerDependenciesMeta:
'@angular/animations':
optional: true
- '@angular/platform-server@21.2.0-next.3':
- resolution: {integrity: sha512-ETjC41XcL0m19hONZebrGNKqOTFqtuz2qP6I5g4JKS6jDJ5aLx9ERtcq+ALnuEmPICcR7rbGzMigM2WTDTdZ0A==}
+ '@angular/platform-server@21.2.0':
+ resolution: {integrity: sha512-XE/0HTYlPpO37aOOc4BEH1AZMDcPOGd2FoZcwGoWDRWhXYrRqzgcX4h1q0WxmMZRe/p7hEd3g9lrf3Vdjj/4ZQ==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/common': 21.2.0-next.3
- '@angular/compiler': 21.2.0-next.3
- '@angular/core': 21.2.0-next.3
- '@angular/platform-browser': 21.2.0-next.3
+ '@angular/common': 21.2.0
+ '@angular/compiler': 21.2.0
+ '@angular/core': 21.2.0
+ '@angular/platform-browser': 21.2.0
rxjs: ^6.5.3 || ^7.4.0
- '@angular/router@21.2.0-next.3':
- resolution: {integrity: sha512-NHokfVBqk8Elei9AtWreWdrm6oThwCbJZEbi3MqKJBEKCo33BEtZuv3QrwrbpNb+Pnm5RJVYMK4Z0sXvt/qohg==}
+ '@angular/router@21.2.0':
+ resolution: {integrity: sha512-siliJ+jJRUCRZ0cdkqc7zww9Didz56Z0Z2YPIuR2n5TZLiuJY+jAf6xotXKp/v6v8XoGJwLiRNipGgNDRIAlWA==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
peerDependencies:
- '@angular/common': 21.2.0-next.3
- '@angular/core': 21.2.0-next.3
- '@angular/platform-browser': 21.2.0-next.3
+ '@angular/common': 21.2.0
+ '@angular/core': 21.2.0
+ '@angular/platform-browser': 21.2.0
rxjs: ^6.5.3 || ^7.4.0
- '@angular/service-worker@21.2.0-next.3':
- resolution: {integrity: sha512-52Z2plR7926tBS/2nCIYLLXuXcwZ1ByJeoMHXfMXNEx4uM0vbMlcALqkCB/pvrehCtqLlT/8Ec0paHk/uE0jcw==}
+ '@angular/service-worker@21.2.0':
+ resolution: {integrity: sha512-XqLUmhXm1Btjk+9JxAwOH7L/6K7jCXlD3K+B0kNiAiQS1/XQK+FOzHcpoKegLDUv83dDplS0zpXMdSuBLGgF8w==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
hasBin: true
peerDependencies:
- '@angular/core': 21.2.0-next.3
+ '@angular/core': 21.2.0
rxjs: ^6.5.3 || ^7.4.0
- '@asamuzakjp/css-color@4.1.2':
- resolution: {integrity: sha512-NfBUvBaYgKIuq6E/RBLY1m0IohzNHAYyaJGuTK79Z23uNwmz2jl1mPsC5ZxCCxylinKhT1Amn5oNTlx1wN8cQg==}
+ '@asamuzakjp/css-color@5.0.1':
+ resolution: {integrity: sha512-2SZFvqMyvboVV1d15lMf7XiI3m7SDqXUuKaTymJYLN6dSGadqp+fVojqJlVoMlbZnlTmu3S0TLwLTJpvBMO1Aw==}
+ engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
'@asamuzakjp/dom-selector@6.8.1':
resolution: {integrity: sha512-MvRz1nCqW0fsy8Qz4dnLIvhOlMzqDVBabZx6lH+YywFDdjXhMY37SmpV1XFX3JzG5GWHn63j6HX6QPr3lZXHvQ==}
@@ -1613,8 +1614,8 @@ packages:
resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==}
engines: {node: '>=12'}
- '@csstools/color-helpers@6.0.1':
- resolution: {integrity: sha512-NmXRccUJMk2AWA5A7e5a//3bCIMyOu2hAtdRYrhPPHjDxINuCwX1w6rnIZ4xjLcp0ayv6h8Pc3X0eJUGiAAXHQ==}
+ '@csstools/color-helpers@6.0.2':
+ resolution: {integrity: sha512-LMGQLS9EuADloEFkcTBR3BwV/CGHV7zyDxVRtVDTwdI2Ca4it0CCVTT9wCkxSgokjE5Ho41hEPgb8OEUwoXr6Q==}
engines: {node: '>=20.19.0'}
'@csstools/css-calc@3.1.1':
@@ -1624,8 +1625,8 @@ packages:
'@csstools/css-parser-algorithms': ^4.0.0
'@csstools/css-tokenizer': ^4.0.0
- '@csstools/css-color-parser@4.0.1':
- resolution: {integrity: sha512-vYwO15eRBEkeF6xjAno/KQ61HacNhfQuuU/eGwH67DplL0zD5ZixUa563phQvUelA07yDczIXdtmYojCphKJcw==}
+ '@csstools/css-color-parser@4.0.2':
+ resolution: {integrity: sha512-0GEfbBLmTFf0dJlpsNU7zwxRIH0/BGEMuXLTCvFYxuL1tNhqzTbtnFICyJLTNK4a+RechKP75e7w42ClXSnJQw==}
engines: {node: '>=20.19.0'}
peerDependencies:
'@csstools/css-parser-algorithms': ^4.0.0
@@ -1637,8 +1638,8 @@ packages:
peerDependencies:
'@csstools/css-tokenizer': ^4.0.0
- '@csstools/css-syntax-patches-for-csstree@1.0.27':
- resolution: {integrity: sha512-sxP33Jwg1bviSUXAV43cVYdmjt2TLnLXNqCWl9xmxHawWVjGz/kEbdkr7F9pxJNBN2Mh+dq0crgItbW6tQvyow==}
+ '@csstools/css-syntax-patches-for-csstree@1.0.28':
+ resolution: {integrity: sha512-1NRf1CUBjnr3K7hu8BLxjQrKCxEe8FP/xmPTenAxCRZWVLbmGotkFvG9mfNpjA6k7Bw1bw4BilZq9cu19RA5pg==}
'@csstools/css-tokenizer@4.0.0':
resolution: {integrity: sha512-QxULHAm7cNu72w97JUNCBFODFaXpbDg+dP8b/oWFAZ2MTRppA3U00Y2L1HqaS4J6yBqxwa/Y3nMBaxVKbB/NsA==}
@@ -1877,8 +1878,8 @@ packages:
'@noble/hashes':
optional: true
- '@firebase/ai@2.7.0':
- resolution: {integrity: sha512-PwpCz+TtAMWICM7uQNO0mkSPpUKwrMV4NSwHkbVKDvPKoaQmSlO96vIz+Suw2Ao1EaUUsxYb5LGImHWt/fSnRQ==}
+ '@firebase/ai@2.8.0':
+ resolution: {integrity: sha512-grWYGFPsSo+pt+6CYeKR0kWnUfoLLS3xgWPvNrhAS5EPxl6xWq7+HjDZqX24yLneETyl45AVgDsTbVgxeWeRfg==}
engines: {node: '>=20.0.0'}
peerDependencies:
'@firebase/app': 0.x
@@ -1915,15 +1916,15 @@ packages:
peerDependencies:
'@firebase/app': 0.x
- '@firebase/app-compat@0.5.7':
- resolution: {integrity: sha512-MO+jfap8IBZQ+K8L2QCiHObyMgpYHrxo4Hc7iJgfb9hjGRW/z1y6LWVdT9wBBK+VJ7cRP2DjAiWQP+thu53hHA==}
+ '@firebase/app-compat@0.5.8':
+ resolution: {integrity: sha512-4De6SUZ36zozl9kh5rZSxKWULpgty27rMzZ6x+xkoo7+NWyhWyFdsdvhFsWhTw/9GGj0wXIcbTjwHYCUIUuHyg==}
engines: {node: '>=20.0.0'}
'@firebase/app-types@0.9.3':
resolution: {integrity: sha512-kRVpIl4vVGJ4baogMDINbyrIOtOxqhkZQg4jTq3l8Lw6WSk0xfpEYzezFu+Kl4ve4fbPl79dvwRtaFqAC/ucCw==}
- '@firebase/app@0.14.7':
- resolution: {integrity: sha512-o3ZfnOx0AWBD5n/36p2zPoB0rDDxQP8H/A60zDLvvfRLtW8b3LfCyV97GKpJaAVV1JMMl/BC89EDzMyzxFZxTw==}
+ '@firebase/app@0.14.8':
+ resolution: {integrity: sha512-WiE9uCGRLUnShdjb9iP20sA3ToWrBbNXr14/N5mow7Nls9dmKgfGaGX5cynLvrltxq2OrDLh1VDNaUgsnS/k/g==}
engines: {node: '>=20.0.0'}
'@firebase/auth-compat@0.6.2':
@@ -1971,8 +1972,8 @@ packages:
resolution: {integrity: sha512-gM6MJFae3pTyNLoc9VcJNuaUDej0ctdjn3cVtILo3D5lpp0dmUHHLFN/pUKe7ImyeB1KAvRlEYxvIHNF04Filg==}
engines: {node: '>=20.0.0'}
- '@firebase/firestore-compat@0.4.4':
- resolution: {integrity: sha512-JvxxIgi+D5v9BecjLA1YomdyF7LA6CXhJuVK10b4GtRrB3m2O2hT1jJWbKYZYHUAjTaajkvnos+4U5VNxqkI2w==}
+ '@firebase/firestore-compat@0.4.5':
+ resolution: {integrity: sha512-yVX1CkVvqBI4qbA56uZo42xFA4TNU0ICQ+9AFDvYq9U9Xu6iAx9lFDAk/tN+NGereQQXXCSnpISwc/oxsQqPLA==}
engines: {node: '>=20.0.0'}
peerDependencies:
'@firebase/app-compat': 0.x
@@ -1983,8 +1984,8 @@ packages:
'@firebase/app-types': 0.x
'@firebase/util': 1.x
- '@firebase/firestore@4.10.0':
- resolution: {integrity: sha512-fgF6EbpoagGWh5Vwfu/7/jYgBFwUCwTlPNVF/aSjHcoEDRXpRsIqVfAFTp1LD+dWAUcAKEK3h+osk8spMJXtxA==}
+ '@firebase/firestore@4.11.0':
+ resolution: {integrity: sha512-Zb88s8rssBd0J2Tt+NUXMPt2sf+Dq7meatKiJf5t9oto1kZ8w9gK59Koe1uPVbaKfdgBp++N/z0I4G/HamyEhg==}
engines: {node: '>=20.0.0'}
peerDependencies:
'@firebase/app': 0.x
@@ -2118,8 +2119,8 @@ packages:
resolution: {integrity: sha512-IJn+8A3QZJfe7FUtWqHVNo3xJs7KFpurCWGWCiCz3oEh+BkRymKZ1QxfAbU2yGMDzTytLGQ2IV6T2r3cuo75/w==}
engines: {node: '>=18'}
- '@google/genai@1.38.0':
- resolution: {integrity: sha512-V/4CQVQGovvGHuS73lwJwHKR9x33kCij3zz/ReEQ4A7RJaV0U7m4k1mvYhFk55cGZdF5JLKu2S9BTaFuEs5xTA==}
+ '@google/genai@1.42.0':
+ resolution: {integrity: sha512-+3nlMTcrQufbQ8IumGkOphxD5Pd5kKyJOzLcnY0/1IuE8upJk5aLmoexZ2BJhBp1zAjRJMEB4a2CJwKI9e2EYw==}
engines: {node: '>=20.0.0'}
peerDependencies:
'@modelcontextprotocol/sdk': ^1.25.2
@@ -2190,8 +2191,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/checkbox@5.0.6':
- resolution: {integrity: sha512-qLZ1gOpsqsieB5k98GQ9bWYggvMsCXTc7HUwhEQpTsxFQYGthqR9UysCwqB7L9h47THYdXhJegnYb1IqURMjng==}
+ '@inquirer/checkbox@5.1.0':
+ resolution: {integrity: sha512-/HjF1LN0a1h4/OFsbGKHNDtWICFU/dqXCdym719HFTyJo9IG7Otr+ziGWc9S0iQuohRZllh+WprSgd5UW5Fw0g==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2208,8 +2209,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/confirm@6.0.6':
- resolution: {integrity: sha512-9ZkrGYiWnOKQPc3xfLIORE3lZW1qvtgRoJcoqopr5zssBn7yk4yONmzGynEOjc16FnUXzkAejj/I29BbfcoUfQ==}
+ '@inquirer/confirm@6.0.8':
+ resolution: {integrity: sha512-Di6dgmiZ9xCSUxWUReWTqDtbhXCuG2MQm2xmgSAIruzQzBqNf49b8E07/vbCYY506kDe8BiwJbegXweG8M1klw==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2226,8 +2227,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/core@11.1.3':
- resolution: {integrity: sha512-TBAGPDGvpwFSQ4nkawQzq5/X7DhElANjvKeUtcjpVnBIfuH/OEu4M+79R3+bGPtwxST4DOIGRtF933mUH2bRVw==}
+ '@inquirer/core@11.1.5':
+ resolution: {integrity: sha512-QQPAX+lka8GyLcZ7u7Nb1h6q72iZ/oy0blilC3IB2nSt1Qqxp7akt94Jqhi/DzARuN3Eo9QwJRvtl4tmVe4T5A==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2244,8 +2245,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/editor@5.0.6':
- resolution: {integrity: sha512-dxTi/TB29NaW18u0pQl3B140695izGUMzr340a4Yhxll3oa0/iwxl6C88sX9LDUPFaaM4FDASEMnLm8XVk2VVg==}
+ '@inquirer/editor@5.0.8':
+ resolution: {integrity: sha512-sLcpbb9B3XqUEGrj1N66KwhDhEckzZ4nI/W6SvLXyBX8Wic3LDLENlWRvkOGpCPoserabe+MxQkpiMoI8irvyA==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2262,8 +2263,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/expand@5.0.6':
- resolution: {integrity: sha512-HmgMzFdMk/gmPXfuFy4xgWkyIVbdH81otQkrFbhklFZcGauwDFD1EbgmZdgmYCN5pWhSEnYIadg1kysLgPIYag==}
+ '@inquirer/expand@5.0.8':
+ resolution: {integrity: sha512-QieW3F1prNw3j+hxO7/NKkG1pk3oz7pOB6+5Upwu3OIwADfPX0oZVppsqlL+Vl/uBHHDSOBY0BirLctLnXwGGg==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2306,8 +2307,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/input@5.0.6':
- resolution: {integrity: sha512-RZsJcjMJA3QNI9q9OiAi1fAom+Pb8on6alJB1Teh5jjKaiG5C79P69cG955ZRfgPdxTmI4uyhf33+94Xj7xWig==}
+ '@inquirer/input@5.0.8':
+ resolution: {integrity: sha512-p0IJslw0AmedLEkOU+yrEX3Aj2RTpQq7ZOf8nc1DIhjzaxRWrrgeuE5Kyh39fVRgtcACaMXx/9WNo8+GjgBOfw==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2324,8 +2325,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/number@4.0.6':
- resolution: {integrity: sha512-owMkAY+gR0BggomDTL+Z22x/yfE4ocFrmNyJacOiaDVA/d+iL4IWyk7Ds7JEuDMxuhHFB46Dubdxg1uiD7GlCA==}
+ '@inquirer/number@4.0.8':
+ resolution: {integrity: sha512-uGLiQah9A0F9UIvJBX52m0CnqtLaym0WpT9V4YZrjZ+YRDKZdwwoEPz06N6w8ChE2lrnsdyhY9sL+Y690Kh9gQ==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2342,8 +2343,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/password@5.0.6':
- resolution: {integrity: sha512-c4BT4SB79iYwPhtGVBSvrlTnn4oFSYnwocafmktpay8RK75T2c2+fLlR0i1Cxw0QOhdy/YULdmpHoy1sOrPzvA==}
+ '@inquirer/password@5.0.8':
+ resolution: {integrity: sha512-zt1sF4lYLdvPqvmvHdmjOzuUUjuCQ897pdUCO8RbXMUDKXJTTyOQgtn23le+jwcb+MpHl3VAFvzIdxRAf6aPlA==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2360,8 +2361,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/prompts@8.2.0':
- resolution: {integrity: sha512-rqTzOprAj55a27jctS3vhvDDJzYXsr33WXTjODgVOru21NvBo9yIgLIAf7SBdSV0WERVly3dR6TWyp7ZHkvKFA==}
+ '@inquirer/prompts@8.3.0':
+ resolution: {integrity: sha512-JAj66kjdH/F1+B7LCigjARbwstt3SNUOSzMdjpsvwJmzunK88gJeXmcm95L9nw1KynvFVuY4SzXh/3Y0lvtgSg==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2378,8 +2379,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/rawlist@5.2.2':
- resolution: {integrity: sha512-ld2EhLlf3fsBv7QfxR31NdBecGdS6eeFFZ+Nx88ApjtifeCEc9TNrw8x5tGe+gd6HG1ERczOb4B/bMojiGIp1g==}
+ '@inquirer/rawlist@5.2.4':
+ resolution: {integrity: sha512-fTuJ5Cq9W286isLxwj6GGyfTjx1Zdk4qppVEPexFuA6yioCCXS4V1zfKroQqw7QdbDPN73xs2DiIAlo55+kBqg==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2396,8 +2397,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/search@4.1.2':
- resolution: {integrity: sha512-kdGbbbWYKldWxpxodKYPmFl/ctBi3DjWlA4LX48jXtqJ7NEeoEKlyFTbE4xNEFcGDi15tvaxRLzCV4A53zqYIw==}
+ '@inquirer/search@4.1.4':
+ resolution: {integrity: sha512-9yPTxq7LPmYjrGn3DRuaPuPbmC6u3fiWcsE9ggfLcdgO/ICHYgxq7mEy1yJ39brVvgXhtOtvDVjDh9slJxE4LQ==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2414,8 +2415,8 @@ packages:
'@types/node':
optional: true
- '@inquirer/select@5.0.6':
- resolution: {integrity: sha512-9DyVbNCo4q0C3CkGd6zW0SW3NQuuk4Hy0NSbP6zErz2YNWF4EHHJCRzcV34/CDQLraeAQXbHYlMofuUrs6BBZQ==}
+ '@inquirer/select@5.1.0':
+ resolution: {integrity: sha512-OyYbKnchS1u+zRe14LpYrN8S0wH1vD0p2yKISvSsJdH2TpI87fh4eZdWnpdbrGauCRWDph3NwxRmM4Pcm/hx1Q==}
engines: {node: '>=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0'}
peerDependencies:
'@types/node': '>=18'
@@ -2441,22 +2442,10 @@ packages:
'@types/node':
optional: true
- '@isaacs/balanced-match@4.0.1':
- resolution: {integrity: sha512-yzMTt9lEb8Gv7zRioUilSglI0c0smZ9k5D65677DLWLtWJaXIS3CqcGyUFByYKlnUj6TkjLVs54fBl6+TiGQDQ==}
- engines: {node: 20 || >=22}
-
- '@isaacs/brace-expansion@5.0.1':
- resolution: {integrity: sha512-WMz71T1JS624nWj2n2fnYAuPovhv7EUhk69R6i9dsVyzxt5eM3bjwvgk9L+APE1TRscGysAVMANkB0jh0LQZrQ==}
- engines: {node: 20 || >=22}
-
'@isaacs/cliui@8.0.2':
resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==}
engines: {node: '>=12'}
- '@isaacs/cliui@9.0.0':
- resolution: {integrity: sha512-AokJm4tuBHillT+FpMtxQ60n8ObyXBatq7jD2/JA9dxbDDokKQm8KMht5ibGzLVU9IJDIKK4TPKgMHEYMn3lMg==}
- engines: {node: '>=18'}
-
'@isaacs/fs-minipass@4.0.1':
resolution: {integrity: sha512-wgm9Ehl2jpeqP3zw/7mo3kRHFp5MEDhqAdwy1fTGkHAwnkGOVsgpvQhL8B5n1qlb01jV3n/bI0ZfZp5lWA1k4w==}
engines: {node: '>=18.0.0'}
@@ -2698,8 +2687,8 @@ packages:
cpu: [x64]
os: [win32]
- '@mswjs/interceptors@0.39.8':
- resolution: {integrity: sha512-2+BzZbjRO7Ct61k8fMNHEtoKjeWI9pIlHFTqBwZ5icHpqszIgEZbjb1MW5Z0+bITTCTl3gk4PDBxs9tA/csXvA==}
+ '@mswjs/interceptors@0.41.3':
+ resolution: {integrity: sha512-cXu86tF4VQVfwz8W1SPbhoRyHJkti6mjH/XJIxp40jhO4j2k1m4KYrEykxqWPkFF3vrK4rgQppBh//AwyGSXPA==}
engines: {node: '>=18'}
'@napi-rs/nice-android-arm-eabi@1.1.1':
@@ -2855,8 +2844,8 @@ packages:
resolution: {integrity: sha512-uuG5HZFXLfyFKqg8QypsmgLQW7smiRjVc45bqD/ofZZcR/uxEjgQU8qDPv0s9TEeMUiAAU/GC5bR6++UdTirIQ==}
engines: {node: ^20.17.0 || >=22.9.0}
- '@npmcli/package-json@7.0.4':
- resolution: {integrity: sha512-0wInJG3j/K40OJt/33ax47WfWMzZTm6OQxB9cDhTt5huCP2a9g2GnlsxmfN+PulItNPIpPrZ+kfwwUil7eHcZQ==}
+ '@npmcli/package-json@7.0.5':
+ resolution: {integrity: sha512-iVuTlG3ORq2iaVa1IWUxAO/jIp77tUKBhoMjuzYW2kL4MLN1bi/ofqkZ7D7OOwh8coAx1/S2ge0rMdGv8sLSOQ==}
engines: {node: ^20.17.0 || >=22.9.0}
'@npmcli/promise-spawn@9.0.1':
@@ -2871,8 +2860,8 @@ packages:
resolution: {integrity: sha512-ER2N6itRkzWbbtVmZ9WKaWxVlKlOeBFF1/7xx+KA5J1xKa4JjUwBdb6tDpk0v1qA+d+VDwHI9qmLcXSWcmi+Rw==}
engines: {node: ^20.17.0 || >=22.9.0}
- '@octokit/auth-app@8.1.2':
- resolution: {integrity: sha512-db8VO0PqXxfzI6GdjtgEFHY9tzqUql5xMFXYA12juq8TeTgPAuiiP3zid4h50lwlIP457p5+56PnJOgd2GGBuw==}
+ '@octokit/auth-app@8.2.0':
+ resolution: {integrity: sha512-vVjdtQQwomrZ4V46B9LaCsxsySxGoHsyw6IYBov/TqJVROrlYdyNgw5q6tQbB7KZt53v1l1W53RiqTvpzL907g==}
engines: {node: '>= 20'}
'@octokit/auth-oauth-app@9.0.3':
@@ -2895,8 +2884,8 @@ packages:
resolution: {integrity: sha512-DhGl4xMVFGVIyMwswXeyzdL4uXD5OGILGX5N8Y+f6W7LhC1Ze2poSNrkF/fedpVDHEEZ+PHFW0vL14I+mm8K3Q==}
engines: {node: '>= 20'}
- '@octokit/endpoint@11.0.2':
- resolution: {integrity: sha512-4zCpzP1fWc7QlqunZ5bSEjxc6yLAlRTnDwKtgXfcI/FxxGoqedDG8V2+xJ60bV2kODqcGB+nATdtap/XYq2NZQ==}
+ '@octokit/endpoint@11.0.3':
+ resolution: {integrity: sha512-FWFlNxghg4HrXkD3ifYbS/IdL/mDHjh9QcsNyhQjN8dplUoZbejsdpmuqdA76nxj2xoWPs7p8uX2SNr9rYu0Ag==}
engines: {node: '>= 20'}
'@octokit/graphql-schema@15.26.1':
@@ -2939,8 +2928,8 @@ packages:
resolution: {integrity: sha512-KMQIfq5sOPpkQYajXHwnhjCC0slzCNScLHs9JafXc4RAJI+9f+jNDlBNaIMTvazOPLgb4BnlhGJOTbnN0wIjPw==}
engines: {node: '>= 20'}
- '@octokit/request@10.0.7':
- resolution: {integrity: sha512-v93h0i1yu4idj8qFPZwjehoJx4j3Ntn+JhXsdJrG9pYaX6j/XRz2RmasMUHtNgQD39nrv/VwTWSqK0RNXR8upA==}
+ '@octokit/request@10.0.8':
+ resolution: {integrity: sha512-SJZNwY9pur9Agf7l87ywFi14W+Hd9Jg6Ifivsd33+/bGUQIjNujdFiXII2/qSlN2ybqUHfp5xpekMEjIBTjlSw==}
engines: {node: '>= 20'}
'@octokit/rest@22.0.1':
@@ -3111,20 +3100,20 @@ packages:
resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==}
engines: {node: '>=14'}
- '@pnpm/crypto.hash@1000.2.1':
- resolution: {integrity: sha512-Kgo3bgYbdKkC5xFvvQshbHa+Nru7k50D91+yyq7enp4Ur2EMp4wg5oXleaC5xu5hC9A/1eSCRI8npCioplxG4A==}
+ '@pnpm/crypto.hash@1000.2.2':
+ resolution: {integrity: sha512-W8pLZvXWLlGG5p0Z2nCvtBhlM6uuTcbAbsS15wlGS31jBBJKJW2udLoFeM7qfWPo7E2PqRPGxca7APpVYAjJhw==}
engines: {node: '>=18.12'}
'@pnpm/crypto.polyfill@1000.1.0':
resolution: {integrity: sha512-tNe7a6U4rCpxLMBaR0SIYTdjxGdL0Vwb3G1zY8++sPtHSvy7qd54u8CIB0Z+Y6t5tc9pNYMYCMwhE/wdSY7ltg==}
engines: {node: '>=18.12'}
- '@pnpm/dependency-path@1001.1.9':
- resolution: {integrity: sha512-C1V4H54GyMfLL47q93PmdVRJkJyNVEE6Ht6cFrMSsjgsR7fxXWqjlem7OaA9MMjSTBB/d/g9mV4xZnoT/HAkDQ==}
+ '@pnpm/dependency-path@1001.1.10':
+ resolution: {integrity: sha512-PNImtV2SmNTDpLi4HdN86tJPmsOeIxm4VhmxgBVsMrJPEBfkNEWFcflR3wU6XVn/26g9qWdvlNHaawtCjeB93Q==}
engines: {node: '>=18.12'}
- '@pnpm/graceful-fs@1000.0.1':
- resolution: {integrity: sha512-JnzaAVFJIEgwTcB55eww8N3h5B6qJdZqDA2wYkSK+OcTvvMSQb9c2STMhBP6GfkWygG1fs3w8D7JRx9SPZnxJg==}
+ '@pnpm/graceful-fs@1000.1.0':
+ resolution: {integrity: sha512-EsMX4slK0qJN2AR0/AYohY5m0HQNYGMNe+jhN74O994zp22/WbX+PbkIKyw3UQn39yQm2+z6SgwklDxbeapsmQ==}
engines: {node: '>=18.12'}
'@pnpm/types@1001.3.0':
@@ -3161,8 +3150,8 @@ packages:
'@protobufjs/utf8@1.1.0':
resolution: {integrity: sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw==}
- '@puppeteer/browsers@2.12.1':
- resolution: {integrity: sha512-fXa6uXLxfslBlus3MEpW8S6S9fe5RwmAE5Gd8u3krqOwnkZJV3/lQJiY3LaFdTctLLqJtyMgEUGkbDnRNf6vbQ==}
+ '@puppeteer/browsers@2.13.0':
+ resolution: {integrity: sha512-46BZJYJjc/WwmKjsvDFykHtXrtomsCIrwYQPOP7VfMJoZY2bsDF9oROBABR3paDjDcmkUye1Pb1BqdcdiipaWA==}
engines: {node: '>=18'}
hasBin: true
@@ -3451,8 +3440,8 @@ packages:
cpu: [x64]
os: [win32]
- '@rollup/wasm-node@4.57.1':
- resolution: {integrity: sha512-b0rcJH8ykEanfgTeDtlPubhphIUOx0oaAek+3hizTaFkoC1FBSTsY0GixwB4D5HZ5r3Gt2yI9c8M13OcW/kW5A==}
+ '@rollup/wasm-node@4.59.0':
+ resolution: {integrity: sha512-cKB/Pe05aJWQYw3UFS79Id+KVXdExBxWful0+CSl24z3ukwOgBSy6l39XZNwfm3vCh/fpUrAAs+T7PsJ6dC8NA==}
engines: {node: '>=18.0.0', npm: '>=8.0.0'}
hasBin: true
@@ -3493,11 +3482,11 @@ packages:
'@standard-schema/spec@1.1.0':
resolution: {integrity: sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w==}
- '@stylistic/eslint-plugin@5.8.0':
- resolution: {integrity: sha512-WNPVF/FfBAjyi3OA7gok8swRiImNLKI4dmV3iK/GC/0xSJR7eCzBFsw9hLZVgb1+MYNLy7aDsjohxN1hA/FIfQ==}
+ '@stylistic/eslint-plugin@5.9.0':
+ resolution: {integrity: sha512-FqqSkvDMYJReydrMhlugc71M76yLLQWNfmGq+SIlLa7N3kHp8Qq8i2PyWrVNAfjOyOIY+xv9XaaYwvVW7vroMA==}
engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0}
peerDependencies:
- eslint: '>=9.0.0'
+ eslint: ^9.0.0 || ^10.0.0
'@szmarczak/http-timer@4.0.6':
resolution: {integrity: sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w==}
@@ -3692,8 +3681,8 @@ packages:
'@types/loader-utils@3.0.0':
resolution: {integrity: sha512-oOi4OGpiLUbb+Q/cN9FIkkDFgOpOGZ2cUAzb5i03wrGstnG6Syx1WDMhSiB5rcP10XX7cw7Uev8mv++/aplnNg==}
- '@types/lodash@4.17.23':
- resolution: {integrity: sha512-RDvF6wTulMPjrNdCoYRC8gNR880JNGT8uB+REUpC2Ns4pRqQJhGz90wh7rgdXDPpCczF3VGktDuFGVnz8zP7HA==}
+ '@types/lodash@4.17.24':
+ resolution: {integrity: sha512-gIW7lQLZbue7lRSWEFql49QJJWThrTFFeIMJdp3eH4tKoxm1OvEPg02rm4wCCSHS0cL3/Fizimb35b7k8atwsQ==}
'@types/micromatch@2.3.35':
resolution: {integrity: sha512-J749bHo/Zu56w0G0NI/IGHLQPiSsjx//0zJhfEVAN95K/xM5C8ZDmhkXtU3qns0sBOao7HuQzr8XV1/2o5LbXA==}
@@ -3707,8 +3696,8 @@ packages:
'@types/node@22.19.11':
resolution: {integrity: sha512-BH7YwL6rA93ReqeQS1c4bsPpcfOmJasG+Fkr6Y59q83f9M1WcBRHR2vM+P9eOisYRcN3ujQoiZY8uk5W+1WL8w==}
- '@types/node@24.10.9':
- resolution: {integrity: sha512-ne4A0IpG3+2ETuREInjPNhUGis1SFjv1d5asp8MzEAGtOZeTeHVDOYqOgqfhvseqg/iXty2hjBf1zAOb7RNiNw==}
+ '@types/node@24.10.13':
+ resolution: {integrity: sha512-oH72nZRfDv9lADUBSo104Aq7gPHpQZc4BTx38r9xf9pg5LfP6EzSyH2n7qFmmxRQXh7YlUXODcYsg6PuTDSxGg==}
'@types/npm-package-arg@6.1.4':
resolution: {integrity: sha512-vDgdbMy2QXHnAruzlv68pUtXCjmqUk3WrBAsRboRovsOmxbfn/WiYCjmecyKjGztnMps5dWp4Uq2prp+Ilo17Q==}
@@ -3752,6 +3741,9 @@ packages:
'@types/responselike@1.0.0':
resolution: {integrity: sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA==}
+ '@types/retry@0.12.0':
+ resolution: {integrity: sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA==}
+
'@types/retry@0.12.2':
resolution: {integrity: sha512-XISRgDJ2Tc5q4TRqvgJtzsRkFYNJzZrhTdtMoGVBttwzzQJkPnS3WWTFc7kuDRoPtPakl+T+OfdEUjYJj7Jbow==}
@@ -3854,6 +3846,10 @@ packages:
resolution: {integrity: sha512-ujT0Je8GI5BJWi+/mMoR0wxwVEQaxM+pi30xuMiJETlX80OPovb2p9E8ss87gnSVtYXtJoU9U1Cowcr6w2FE0w==}
engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0}
+ '@typescript-eslint/types@8.56.1':
+ resolution: {integrity: sha512-dbMkdIUkIkchgGDIv7KLUpa0Mda4IYjo4IAMJUZ+3xNoUXxMsk9YtKpTHSChRS85o+H9ftm51gsK1dZReY9CVw==}
+ engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0}
+
'@typescript-eslint/typescript-estree@8.55.0':
resolution: {integrity: sha512-EwrH67bSWdx/3aRQhCoxDaHM+CrZjotc2UCCpEDVqfCE+7OjKAGWNY2HsCSTEVvWH2clYQK8pdeLp42EVs+xQw==}
engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0}
@@ -4133,12 +4129,12 @@ packages:
peerDependencies:
acorn: ^6.0.0 || ^7.0.0 || ^8.0.0
- acorn-walk@8.3.4:
- resolution: {integrity: sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g==}
+ acorn-walk@8.3.5:
+ resolution: {integrity: sha512-HEHNfbars9v4pgpW6SO1KSPkfoS0xVOM/9UzkJltjlsHZmJasxg8aXkuZa7SMf8vKGIBhpUsPluQSqhJFCqebw==}
engines: {node: '>=0.4.0'}
- acorn@8.15.0:
- resolution: {integrity: sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==}
+ acorn@8.16.0:
+ resolution: {integrity: sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==}
engines: {node: '>=0.4.0'}
hasBin: true
@@ -4178,8 +4174,8 @@ packages:
peerDependencies:
ajv: ^8.8.2
- ajv@6.12.6:
- resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==}
+ ajv@6.14.0:
+ resolution: {integrity: sha512-IWrosm/yrn43eiKqkfkHis7QioDleaXQHdDVPKg0FSwwd/DuvyX79TZnFOnYpB7dcsFAMmtFztZuXPDvSePkFw==}
ajv@8.17.1:
resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==}
@@ -4373,8 +4369,8 @@ packages:
aws4@1.13.2:
resolution: {integrity: sha512-lHe62zvbTB5eEABUVi/AwVh0ZKY9rMMDhmm+eeyuuUQbQ3+J+fONVQOZyj+DdrvD4BY33uYniyRJ4UJIaSKAfw==}
- b4a@1.7.4:
- resolution: {integrity: sha512-u20zJLDaSWpxaZ+zaAkEIB2dZZ1o+DF4T/MRbmsvGp9nletHOyiai19OzX1fF8xUBYsO1bPXxODvcd0978pnug==}
+ b4a@1.8.0:
+ resolution: {integrity: sha512-qRuSmNSkGQaHwNbM7J78Wwy+ghLEYF1zNrSeMxj4Kgw6y33O3mXcQ6Ie9fRvfU/YnxWkOchPXbaLb73TkIsfdg==}
peerDependencies:
react-native-b4a: '*'
peerDependenciesMeta:
@@ -4411,9 +4407,9 @@ packages:
balanced-match@1.0.2:
resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==}
- balanced-match@4.0.2:
- resolution: {integrity: sha512-x0K50QvKQ97fdEz2kPehIerj+YTeptKF9hyYkKf6egnwmMWAkADiO0QCzSp0R5xN8FTZgYaBfSaue46Ej62nMg==}
- engines: {node: 20 || >=22}
+ balanced-match@4.0.4:
+ resolution: {integrity: sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==}
+ engines: {node: 18 || 20 || >=22}
bare-events@2.8.2:
resolution: {integrity: sha512-riJjyv1/mHLIPX4RwiK+oW9/4c3TEUeORHKefKAKnZ5kyslbN+HXowtbaVEqt4IMUB7OXlfixcs6gsFeo/jhiQ==}
@@ -4439,8 +4435,8 @@ packages:
bare-path@3.0.0:
resolution: {integrity: sha512-tyfW2cQcB5NN8Saijrhqn0Zh7AnFNsnczRcuWODH0eYAXBsJ5gVxAUuNr7tsHSC6IZ77cA0SitzT+s47kot8Mw==}
- bare-stream@2.7.0:
- resolution: {integrity: sha512-oyXQNicV1y8nc2aKffH+BUHFRXmx6VrPzlnaEvMhram0nPBrKcEdcyBg5r08D0i8VxngHFAiVyn1QKXpSG0B8A==}
+ bare-stream@2.8.0:
+ resolution: {integrity: sha512-reUN0M2sHRqCdG4lUK3Fw8w98eeUIZHL5c3H7Mbhk2yVBL+oofgaIp0ieLfD5QXwPCypBpmEEKU2WZKzbAk8GA==}
peerDependencies:
bare-buffer: '*'
bare-events: '*'
@@ -4460,12 +4456,13 @@ packages:
resolution: {integrity: sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog==}
engines: {node: ^4.5.0 || >= 5.9}
- baseline-browser-mapping@2.9.19:
- resolution: {integrity: sha512-ipDqC8FrAl/76p2SSWKSI+H9tFwm7vYqXQrItCuiVPt26Km0jS+NzSsBWAaBusvSbQcfJG+JitdMm+wZAgTYqg==}
+ baseline-browser-mapping@2.10.0:
+ resolution: {integrity: sha512-lIyg0szRfYbiy67j9KN8IyeD7q7hcmqnJ1ddWmNt19ItGpNN64mnllmxUNFIOdOm6by97jlL6wfpTTJrmnjWAA==}
+ engines: {node: '>=6.0.0'}
hasBin: true
- basic-ftp@5.1.0:
- resolution: {integrity: sha512-RkaJzeJKDbaDWTIPiJwubyljaEPwpVWkm9Rt5h9Nd6h7tEXTJ3VB4qxdZBioV7JO5yLUaOKwz7vDOzlncUsegw==}
+ basic-ftp@5.2.0:
+ resolution: {integrity: sha512-VoMINM2rqJwJgfdHq6RiUudKt2BV+FY5ZFezP/ypmwayk68+NzzAQy4XXLlqsGD4MCzq3DrmNFD/uUmBJuGoXw==}
engines: {node: '>=10.0.0'}
batch@0.6.1:
@@ -4528,9 +4525,9 @@ packages:
brace-expansion@2.0.2:
resolution: {integrity: sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==}
- brace-expansion@5.0.2:
- resolution: {integrity: sha512-Pdk8c9poy+YhOgVWw1JNN22/HcivgKWwpxKq04M/jTmHyCZn12WPJebZxdjSa5TmBqISrUSgNYU3eRORljfCCw==}
- engines: {node: 20 || >=22}
+ brace-expansion@5.0.3:
+ resolution: {integrity: sha512-fy6KJm2RawA5RcHkLa1z/ScpBeA762UF9KmZQxwIbDtRJrgLzM10depAiEQ+CXYcoiqW1/m96OAAoke2nE9EeA==}
+ engines: {node: 18 || 20 || >=22}
braces@3.0.3:
resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==}
@@ -4636,8 +4633,8 @@ packages:
resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==}
engines: {node: '>=10'}
- caniuse-lite@1.0.30001770:
- resolution: {integrity: sha512-x/2CLQ1jHENRbHg5PSId2sXq1CIO1CISvwWAj027ltMVG2UNgW+w9oH2+HzgEIRFembL8bUlXtfbBHR1fCg2xw==}
+ caniuse-lite@1.0.30001774:
+ resolution: {integrity: sha512-DDdwPGz99nmIEv216hKSgLD+D4ikHQHjBC/seF98N9CPqRX4M5mSxT9eTV6oyisnJcuzxtZy4n17yKKQYmYQOA==}
caseless@0.12.0:
resolution: {integrity: sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==}
@@ -4952,8 +4949,8 @@ packages:
engines: {node: '>=4'}
hasBin: true
- cssstyle@6.0.1:
- resolution: {integrity: sha512-IoJs7La+oFp/AB033wBStxNOJt4+9hHMxsXUPANcoXL2b3W4DZKghlJ2cI/eyeRZIQ9ysvYEorVhjrcYctWbog==}
+ cssstyle@6.1.0:
+ resolution: {integrity: sha512-Ml4fP2UT2K3CUBQnVlbdV/8aFDdlY69E+YnwJM+3VUWl08S3J8c8aRuJqCkD9Py8DHZ7zNNvsfKl8psocHZEFg==}
engines: {node: '>=20'}
custom-event@1.0.1:
@@ -5232,8 +5229,8 @@ packages:
engines: {node: '>=0.12.18'}
hasBin: true
- electron-to-chromium@1.5.286:
- resolution: {integrity: sha512-9tfDXhJ4RKFNerfjdCcZfufu49vg620741MNs26a9+bhLThdB+plgMeou98CAaHu/WATj2iHOOHTp1hWtABj2A==}
+ electron-to-chromium@1.5.302:
+ resolution: {integrity: sha512-sM6HAN2LyK82IyPBpznDRqlTQAtuSaO+ShzFiWTvoMJLHyZ+Y39r8VMfHzwbU8MVBzQ4Wdn85+wlZl2TLGIlwg==}
emoji-regex@10.6.0:
resolution: {integrity: sha512-toUI84YS5YmxW219erniWD0CIVOo46xGKColeNQRgOzDorgBi1v4D71/OFzgD9GO2UGKIv1C3Sp8DAn0+j5w7A==}
@@ -5629,8 +5626,9 @@ packages:
resolution: {integrity: sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==}
engines: {node: '>=16.0.0'}
- filelist@1.0.4:
- resolution: {integrity: sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==}
+ filelist@1.0.5:
+ resolution: {integrity: sha512-ct/ckWBV/9Dg3MlvCXsLcSUyoWwv9mCKqlhLNB2DAuXR/NZolSXlQqP5dyy6guWlPXBhodZyZ5lGPQcbQDxrEQ==}
+ engines: {node: 20 || >=22}
fill-range@7.1.1:
resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==}
@@ -5672,8 +5670,8 @@ packages:
resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==}
engines: {node: '>=10'}
- firebase@12.8.0:
- resolution: {integrity: sha512-S1tCIR3ENecee0tY2cfTHfMkXqkitHfbsvqpCtvsT0Zi9vDB7A4CodAjHfHCjVvu/XtGy1LHLjOasVcF10rCVw==}
+ firebase@12.9.0:
+ resolution: {integrity: sha512-CwwTYoqZg6KxygPOaaJqIc4aoLvo0RCRrXoln9GoxLE8QyAwTydBaSLGVlR4WPcuOgN3OEL0tJLT1H4IU/dv7w==}
flat-cache@4.0.1:
resolution: {integrity: sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==}
@@ -5793,8 +5791,8 @@ packages:
resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==}
engines: {node: 6.* || 8.* || >= 10.*}
- get-east-asian-width@1.4.0:
- resolution: {integrity: sha512-QZjmEOC+IT1uk6Rx0sX22V6uHWVwbdbxf1faPqJ1QhLdGgsRGCZoyaQBm/piRdJy/D2um6hM1UP7ZEeQ4EkP+Q==}
+ get-east-asian-width@1.5.0:
+ resolution: {integrity: sha512-CQ+bEO+Tva/qlmw24dCejulK5pMzVnUOFOijVogd3KQs07HnRIgp8TGipvCCRT06xeYEbpbgwaCxglFyiuIcmA==}
engines: {node: '>=18'}
get-intrinsic@1.3.0:
@@ -5858,9 +5856,9 @@ packages:
deprecated: Old versions of glob are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me
hasBin: true
- glob@13.0.3:
- resolution: {integrity: sha512-/g3B0mC+4x724v1TgtBlBtt2hPi/EWptsIAmXUx9Z2rvBYleQcsrmaOzd5LyL50jf/Soi83ZDJmw2+XqvH/EeA==}
- engines: {node: 20 || >=22}
+ glob@13.0.6:
+ resolution: {integrity: sha512-Wjlyrolmm8uDpm/ogGyXZXb1Z+Ca2B8NbJwqBVg0axK9GbBeoS7yGV6vjXnYdGm6X53iehEuxxbyiKp8QmN4Vw==}
+ engines: {node: 18 || 20 || >=22}
glob@7.2.3:
resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==}
@@ -5915,8 +5913,8 @@ packages:
peerDependencies:
graphql: ^0.9.0 || ^0.10.0 || ^0.11.0 || ^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0
- graphql@16.12.0:
- resolution: {integrity: sha512-DKKrynuQRne0PNpEbzuEdHlYOMksHSUI8Zc9Unei5gTsMNA2/vMpoMz/yKba50pejK56qj98qM0SjYxAKi13gQ==}
+ graphql@16.13.0:
+ resolution: {integrity: sha512-uSisMYERbaB9bkA9M4/4dnqyktaEkf1kMHNKq/7DHyxVeWqHQ2mBmVqm5u6/FVHwF3iCNalKcg82Zfl+tffWoA==}
engines: {node: ^12.22.0 || ^14.16.0 || ^16.0.0 || >=17.0.0}
grpc-gcp@1.0.1:
@@ -5981,8 +5979,8 @@ packages:
resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==}
engines: {node: '>= 0.4'}
- hono@4.11.9:
- resolution: {integrity: sha512-Eaw2YTGM6WOxA6CXbckaEvslr2Ne4NFsKrvc0v97JD5awbmeBLO5w9Ho9L9kmKonrwF9RJlW6BxT1PVv/agBHQ==}
+ hono@4.12.2:
+ resolution: {integrity: sha512-gJnaDHXKDayjt8ue0n8Gs0A007yKXj4Xzb8+cNjZeYsSzzwKc0Lr+OZgYwVfB0pHfUs17EPoLvrOsEaJ9mj+Tg==}
engines: {node: '>=16.9.0'}
hosted-git-info@9.0.2:
@@ -6461,10 +6459,6 @@ packages:
isexe@2.0.0:
resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==}
- isexe@3.1.5:
- resolution: {integrity: sha512-6B3tLtFqtQS4ekarvLVMZ+X+VlvQekbe4taUkf/rhVO3d/h0M2rfARm/pXLcPEsjjMsFgrFgSrhQIxcSVrBz8w==}
- engines: {node: '>=18'}
-
isexe@4.0.0:
resolution: {integrity: sha512-FFUtZMpoZ8RqHS3XeXEmHWLA4thH+ZxCv2lOiPIn1Xc7CxrqhWzNSDzD+/chS/zbYezmiwWLdQC09JdQKmthOw==}
engines: {node: '>=20'}
@@ -6503,10 +6497,6 @@ packages:
jackspeak@3.4.3:
resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==}
- jackspeak@4.2.3:
- resolution: {integrity: sha512-ykkVRwrYvFm1nb2AJfKKYPr0emF6IiXDYUaFx4Zn9ZuIH7MrzEZ3sD5RlqGXNRpHtvUHJyOnCEFxOlNDtGo7wg==}
- engines: {node: 20 || >=22}
-
jake@10.9.4:
resolution: {integrity: sha512-wpHYzhxiVQL+IV05BLE2Xn34zW1S223hvjtqk0+gsPrwd/8JNLXJgZZM/iPFsYc1xyphF+6M6EvdE5E9MBGkDA==}
engines: {node: '>=10'}
@@ -6521,8 +6511,8 @@ packages:
jasmine-core@5.13.0:
resolution: {integrity: sha512-vsYjfh7lyqvZX5QgqKc4YH8phs7g96Z8bsdIFNEU3VqXhlHaq+vov/Fgn/sr6MiUczdZkyXRC3TX369Ll4Nzbw==}
- jasmine-core@6.0.0:
- resolution: {integrity: sha512-fmBb8aruz2mEIDBUGWOGNmxhXwFJs44SSzwbMcBDqQnNChavPTq3Ra9A6WAn6WdxvxWEdakKTcEb3NzeR3yD1A==}
+ jasmine-core@6.1.0:
+ resolution: {integrity: sha512-p/tjBw58O6vxKIWMlrU+yys8lqR3+l3UrqwNTT7wpj+dQ7N4etQekFM8joI+cWzPDYqZf54kN+hLC1+s5TvZvg==}
jasmine-reporters@2.5.2:
resolution: {integrity: sha512-qdewRUuFOSiWhiyWZX8Yx3YNQ9JG51ntBEO4ekLQRpktxFTwUHy24a86zD/Oi2BRTKksEdfWQZcQFqzjqIkPig==}
@@ -6538,8 +6528,8 @@ packages:
resolution: {integrity: sha512-oLCXIhEb5e0zzjn9GyuvcuisvLBwUjmgz7a0RNGWKwQtJCDld4m+vwKUpAIJVLB5vbmQFdtKhT86/tIZlJ5gYw==}
hasBin: true
- jasmine@6.0.0:
- resolution: {integrity: sha512-eSPL6LPWT39WwvHSEEbRXuSvioXMTheNhIPaeUT1OPmSprDZwj4S29884DkTx6/tyiOWTWB1N+LdW2ZSg74aEA==}
+ jasmine@6.1.0:
+ resolution: {integrity: sha512-WPphPqEMY0uBRMjuhRHoVoxQNvJuxIMqz0yIcJ3k3oYxBedeGoH60/NXNgasxnx2FvfXrq5/r+2wssJ7WE8ABw==}
hasBin: true
jasminewd2@2.2.0:
@@ -6618,6 +6608,9 @@ packages:
json-stringify-safe@5.0.1:
resolution: {integrity: sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==}
+ json-with-bigint@3.5.3:
+ resolution: {integrity: sha512-QObKu6nxy7NsxqR0VK4rkXnsNr5L9ElJaGEg+ucJ6J7/suoKZ0n+p76cu9aCqowytxEbwYNzvrMerfMkXneF5A==}
+
json5@1.0.2:
resolution: {integrity: sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==}
hasBin: true
@@ -6722,8 +6715,8 @@ packages:
resolution: {integrity: sha512-zPPuIt+ku1iCpFBRwseMcPYQ1cJL8l60rSmKeOuGfOXyE6YnTBmf2aEFNL2HQGrD0cPcLO/t+v9RTgC+fwEh/g==}
engines: {node: ^4.8.4 || ^6.10.1 || ^7.10.1 || >= 8.1.4}
- launch-editor@2.12.0:
- resolution: {integrity: sha512-giOHXoOtifjdHqUamwKq6c49GzBdLjvxrd2D+Q4V6uOHopJv7p9VJxikDsQ/CBXZbEITgUqSVHXLTG3VhPP1Dg==}
+ launch-editor@2.13.0:
+ resolution: {integrity: sha512-u+9asUHMJ99lA15VRMXw5XKfySFR9dGXwgsgS14YTbUq3GITP58mIM32At90P5fZ+MUId5Yw+IwI/yKub7jnCQ==}
less-loader@12.3.1:
resolution: {integrity: sha512-JZZmG7gMzoDP3VGeEG8Sh6FW5wygB5jYL7Wp29FFihuRTsIBacqO3LbRPr2yStYD11riVf13selLm/CPFRDBRQ==}
@@ -7011,27 +7004,19 @@ packages:
minimalistic-assert@1.0.1:
resolution: {integrity: sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==}
- minimatch@10.1.1:
- resolution: {integrity: sha512-enIvLvRAFZYXJzkCYG5RKmPfrFArdLv+R+lbQ53BmIMLIry74bjKzX6iHAm8WYamJkhSSEabrWN5D97XnKObjQ==}
- engines: {node: 20 || >=22}
-
- minimatch@10.2.0:
- resolution: {integrity: sha512-ugkC31VaVg9cF0DFVoADH12k6061zNZkZON+aX8AWsR9GhPcErkcMBceb6znR8wLERM2AkkOxy2nWRLpT9Jq5w==}
- engines: {node: 20 || >=22}
+ minimatch@10.2.3:
+ resolution: {integrity: sha512-Rwi3pnapEqirPSbWbrZaa6N3nmqq4Xer/2XooiOKyV3q12ML06f7MOuc5DVH8ONZIFhwIYQ3yzPH4nt7iWHaTg==}
+ engines: {node: 18 || 20 || >=22}
- minimatch@3.1.2:
- resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==}
-
- minimatch@5.1.6:
- resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==}
- engines: {node: '>=10'}
+ minimatch@3.1.3:
+ resolution: {integrity: sha512-M2GCs7Vk83NxkUyQV1bkABc4yxgz9kILhHImZiBPAZ9ybuvCb0/H7lEl5XvIg3g+9d4eNotkZA5IWwYl0tibaA==}
minimatch@7.4.6:
resolution: {integrity: sha512-sBz8G/YjVniEz6lKPNpKxXwazJe4c19fEfV2GDMX6AjFz+MX9uDWIZW8XreVhkFW3fkIdTv/gxWr/Kks5FFAVw==}
engines: {node: '>=10'}
- minimatch@9.0.5:
- resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==}
+ minimatch@9.0.6:
+ resolution: {integrity: sha512-kQAVowdR33euIqeA0+VZTDqU+qo1IeVY+hrKYtZMio3Pg0P0vuh/kwRylLUddJhB6pf3q/botcOvRtx4IN1wqQ==}
engines: {node: '>=16 || 14 >=14.17'}
minimist@1.2.8:
@@ -7041,8 +7026,8 @@ packages:
resolution: {integrity: sha512-D7V8PO9oaz7PWGLbCACuI1qEOsq7UKfLotx/C0Aet43fCUB/wfQ7DYeq2oR/svFJGYDHPr38SHATeaj/ZoKHKw==}
engines: {node: '>=16 || 14 >=14.17'}
- minipass-fetch@5.0.1:
- resolution: {integrity: sha512-yHK8pb0iCGat0lDrs/D6RZmCdaBT64tULXjdxjSMAqoDi18Q3qKEUTHypHQZQd9+FYpIS+lkvpq6C/R6SbUeRw==}
+ minipass-fetch@5.0.2:
+ resolution: {integrity: sha512-2d0q2a8eCi2IRg/IGubCNRJoYbA1+YPXAzQVRFmB45gdGZafyivnZ5YSEfo3JikbjGxOdntGFvBQGqaSMXlAFQ==}
engines: {node: ^20.17.0 || >=22.9.0}
minipass-flush@1.0.5:
@@ -7061,8 +7046,8 @@ packages:
resolution: {integrity: sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==}
engines: {node: '>=8'}
- minipass@7.1.2:
- resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==}
+ minipass@7.1.3:
+ resolution: {integrity: sha512-tEBHqDnIoM/1rXME1zgka9g6Q2lcoCkxHLuc7ODJ5BxbP5d4c2Z5cGgtXAku59200Cx7diuHTOYfSBD8n6mm8A==}
engines: {node: '>=16 || 14 >=14.17'}
minizlib@3.1.0:
@@ -7111,9 +7096,9 @@ packages:
resolution: {integrity: sha512-2eznPJP8z2BFLX50tf0LuODrpINqP1RVIm/CObbTcBRITQgmC/TjcREF1NeTBzIcR5XO/ukWo+YHOjBbFwIupg==}
hasBin: true
- multimatch@7.0.0:
- resolution: {integrity: sha512-SYU3HBAdF4psHEL/+jXDKHO95/m5P2RvboHT2Y0WtTttvJLP4H/2WS9WlQPFvF6C8d6SpLw8vjCnQOnVIVOSJQ==}
- engines: {node: '>=18'}
+ multimatch@8.0.0:
+ resolution: {integrity: sha512-0D10M2/MnEyvoog7tmozlpSqL3HEU1evxUFa3v1dsKYmBDFSP1dLSX4CH2rNjpQ+4Fps8GKmUkCwiKryaKqd9A==}
+ engines: {node: '>=20'}
mute-stream@2.0.0:
resolution: {integrity: sha512-WWdIxpyjEn+FhQJQQv9aQAYlHoNVdzIzUySNV1gHUPDSdZJ3yZn7pAAbQcV7B56Mvu881q9FZV+0Vx2xC44VWA==}
@@ -7158,12 +7143,12 @@ packages:
resolution: {integrity: sha512-dBpDMdxv9Irdq66304OLfEmQ9tbNRFnFTuZiLo+bD+r332bBmMJ8GBLXklIXXgxd3+v9+KUnZaUR5PJMa75Gsg==}
engines: {node: '>= 0.4.0'}
- ng-packagr@21.2.0-next.0:
- resolution: {integrity: sha512-BkRAqx1ZljIYpBbjDi/+3y8AMo9S19vm8zx3YWpqMAaIpDb7cvsT+Une9b4oyEK/7p+XvWw+LaPVleTAQtQEMQ==}
+ ng-packagr@21.2.0:
+ resolution: {integrity: sha512-ASlXEboqt+ZgKzNPx3YCr924xqQRFA5qgm77GHf0Fm13hx7gVFYVm6WCdYZyeX/p9NJjFWAL+mIMfhsx2SHKoA==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
hasBin: true
peerDependencies:
- '@angular/compiler-cli': ^21.0.0 || ^21.1.0-next || ^21.2.0-next
+ '@angular/compiler-cli': ^21.0.0 || ^21.2.0-next
tailwindcss: ^2.0.0 || ^3.0.0 || ^4.0.0
tslib: ^2.3.0
typescript: '>=5.9 <6.0'
@@ -7171,8 +7156,8 @@ packages:
tailwindcss:
optional: true
- nock@14.0.10:
- resolution: {integrity: sha512-Q7HjkpyPeLa0ZVZC5qpxBt5EyLczFJ91MEewQiIi9taWuA0KB/MDJlUWtON+7dGouVdADTQsf9RA7TZk6D8VMw==}
+ nock@14.0.11:
+ resolution: {integrity: sha512-u5xUnYE+UOOBA6SpELJheMCtj2Laqx15Vl70QxKo43Wz/6nMHXS7PrEioXLjXAwhmawdEMNImwKCcPhBJWbKVw==}
engines: {node: '>=18.20.0 <20 || >=20.12.1'}
node-addon-api@6.1.0:
@@ -7256,8 +7241,8 @@ packages:
resolution: {integrity: sha512-IciCE3SY3uE84Ld8WZU23gAPPV9rIYod4F+rc+vJ7h7cwAJt9Vk6TVsK60ry7Uj3SRS3bqRRIGuTp9YVlk6WNA==}
engines: {node: ^20.17.0 || >=22.9.0}
- npm-packlist@10.0.3:
- resolution: {integrity: sha512-zPukTwJMOu5X5uvm0fztwS5Zxyvmk38H/LfidkOMt3gbZVCyro2cD/ETzwzVPcWZA3JOyPznfUN/nkyFiyUbxg==}
+ npm-packlist@10.0.4:
+ resolution: {integrity: sha512-uMW73iajD8hiH4ZBxEV3HC+eTnppIqwakjOYuvgddnalIw2lJguKviK1pcUJDlIWm1wSJkchpDZDSVVsZEYRng==}
engines: {node: ^20.17.0 || >=22.9.0}
npm-pick-manifest@11.0.3:
@@ -7420,6 +7405,10 @@ packages:
resolution: {integrity: sha512-RwFpb72c/BhQLEXIZ5K2e+AhgNVmIejGlTgiB9MzZ0e93GRvqZ7uSi0dvRF7/XIXDeNkra2fNHBxTyPDGySpjQ==}
engines: {node: '>=8'}
+ p-retry@4.6.2:
+ resolution: {integrity: sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ==}
+ engines: {node: '>=8'}
+
p-retry@6.2.1:
resolution: {integrity: sha512-hEt02O4hUct5wtwg4H4KcWgDdm+l1bOaEy/hWzd8xtXB9BqxTWBBhb+2ImAtH4Cv4rPjV76xN3Zumqk3k3AhhQ==}
engines: {node: '>=16.17'}
@@ -7504,9 +7493,9 @@ packages:
resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==}
engines: {node: '>=16 || 14 >=14.18'}
- path-scurry@2.0.1:
- resolution: {integrity: sha512-oWyT4gICAu+kaA7QWk/jvCHWarMKNs6pXOGWKDTr7cw4IGcUbW+PeTfbaQiLGheFRpjo6O9J0PmyMfQPjH71oA==}
- engines: {node: 20 || >=22}
+ path-scurry@2.0.2:
+ resolution: {integrity: sha512-3O/iVVsJAPsOnpwWIeD+d6z/7PmqApyQePUtCndjatj/9I5LylHvt5qluFaBT3I5h3r1ejfR056c+FCv+NnNXg==}
+ engines: {node: 18 || 20 || >=22}
path-to-regexp@0.1.12:
resolution: {integrity: sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==}
@@ -7759,8 +7748,8 @@ packages:
resolution: {integrity: sha512-MRtTAZfQTluz3U2oU/X2VqVWPcR1+94nbA2V6ZrSZRVEwLqZ8eclZ551qGFQD/vD2PYqHJwWOW/fpC721uznVw==}
engines: {node: '>=14.1.0'}
- puppeteer-core@24.37.3:
- resolution: {integrity: sha512-fokQ8gv+hNgsRWqVuP5rUjGp+wzV5aMTP3fcm8ekNabmLGlJdFHas1OdMscAH9Gzq4Qcf7cfI/Pe6wEcAqQhqg==}
+ puppeteer-core@24.37.5:
+ resolution: {integrity: sha512-ybL7iE78YPN4T6J+sPLO7r0lSByp/0NN6PvfBEql219cOnttoTFzCWKiBOjstXSqi/OKpwae623DWAsL7cn2MQ==}
engines: {node: '>=18'}
puppeteer@18.2.1:
@@ -8304,20 +8293,20 @@ packages:
resolution: {integrity: sha512-i5uvt8C3ikiWeNZSVZNWcfZPItFQOsYTUAOkcUPGd8DqDy1uOUikjt5dG+uRlwyvR108Fb9DOd4GvXfT0N2/uQ==}
engines: {node: '>= 12'}
- spdx-correct@3.2.0:
- resolution: {integrity: sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==}
-
spdx-exceptions@2.5.0:
resolution: {integrity: sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w==}
spdx-expression-parse@3.0.1:
resolution: {integrity: sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==}
+ spdx-expression-parse@4.0.0:
+ resolution: {integrity: sha512-Clya5JIij/7C6bRR22+tnGXbc4VKlibKSVj2iHvVeX5iMW7s1SIQlqu699JkODJJIhh/pUu8L0/VLh8xflD+LQ==}
+
spdx-expression-validate@2.0.0:
resolution: {integrity: sha512-b3wydZLM+Tc6CFvaRDBOF9d76oGIHNCLYFeHbftFXUWjnfZWganmDmvtM5sm1cRwJc/VDBMLyGGrsLFd1vOxbg==}
- spdx-license-ids@3.0.22:
- resolution: {integrity: sha512-4PRT4nh1EImPbt2jASOKHX7PB7I+e4IWNLvkKFDxNhJlfjbYlleYQh285Z/3mPTHSAK/AvdMmw5BNNuYH8ShgQ==}
+ spdx-license-ids@3.0.23:
+ resolution: {integrity: sha512-CWLcCCH7VLu13TgOH+r8p1O/Znwhqv/dbb6lqWy67G+pT1kHmeD/+V36AVb/vq8QMIQwVShJ6Ssl5FPh0fuSdw==}
spdy-transport@3.0.0:
resolution: {integrity: sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw==}
@@ -8418,8 +8407,8 @@ packages:
resolution: {integrity: sha512-tsaTIkKW9b4N+AEj+SVA+WhJzV7/zMhcSu78mLKWSk7cXMOSHsBKFWUs0fWwq8QyK3MgJBQRX6Gbi4kYbdvGkQ==}
engines: {node: '>=18'}
- string-width@8.1.1:
- resolution: {integrity: sha512-KpqHIdDL9KwYk22wEOg/VIqYbrnLeSApsKT/bSj6Ez7pn3CftUiLAv2Lccpq1ALcpLV9UX1Ppn92npZWu2w/aw==}
+ string-width@8.2.0:
+ resolution: {integrity: sha512-6hJPQ8N0V0P3SNmP6h2J99RLuzrWz2gvT7VnK5tKvrNqJoyS9W4/Fb8mo31UiPvy00z7DQXkP2hnKBVav76thw==}
engines: {node: '>=20'}
string.prototype.trim@1.2.10:
@@ -8519,6 +8508,9 @@ packages:
resolution: {integrity: sha512-3ZnLvgWF29jikg1sAQ1g0o+lr5JX6sVgYvfUJazn7ZjJroDBUTWp44/+cFVX0bULjv4vci+rBD+oGVAkWqhUbw==}
engines: {node: '>=18'}
+ teex@1.0.1:
+ resolution: {integrity: sha512-eYE6iEI62Ni1H8oIa7KlDU6uQBtqr4Eajni3wX7rpfXD8ysFx8z0+dri+KWEPWpBsxXfxu58x/0jvTVT1ekOSg==}
+
terser-webpack-plugin@5.3.16:
resolution: {integrity: sha512-h9oBFCWrq78NyWWVcSwZarJkZ01c2AyGrzs1crmHZO3QUg9D61Wu4NPjBy69n7JqylFF5y+CsUZYmYEIZ3mR+Q==}
engines: {node: '>= 10.13.0'}
@@ -8540,8 +8532,8 @@ packages:
engines: {node: '>=10'}
hasBin: true
- text-decoder@1.2.6:
- resolution: {integrity: sha512-27FeW5GQFDfw0FpwMQhMagB7BztOOlmjcSRi97t2oplhKVTZtp0DZbSegSaXS5IIC6mxMvBG4AR1Sgc6BX3CQg==}
+ text-decoder@1.2.7:
+ resolution: {integrity: sha512-vlLytXkeP4xvEq2otHeJfSQIRyWxo/oZGEbXrtEEF9Hnmrdly59sUbzZ/QgyWuLYHctCHxFF4tRQZNQ9k60ExQ==}
thingies@2.5.0:
resolution: {integrity: sha512-s+2Bwztg6PhWUD7XMfeYm5qliDdSiZm7M7n8KjTkIsm3l/2lgVRc2/Gx/v+ZX8lT4FMA+i8aQvhcWylldc+ZNw==}
@@ -8882,9 +8874,6 @@ packages:
resolution: {integrity: sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA==}
engines: {node: '>=10.12.0'}
- validate-npm-package-license@3.0.4:
- resolution: {integrity: sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==}
-
validate-npm-package-name@7.0.2:
resolution: {integrity: sha512-hVDIBwsRruT73PbK7uP5ebUt+ezEtCmzZz3F59BSr2F6OVFnJ/6h8liuvdLrQ88Xmnk6/+xGGuq+pG9WwTuy3A==}
engines: {node: ^20.17.0 || >=22.9.0}
@@ -9106,8 +9095,8 @@ packages:
resolution: {integrity: sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==}
engines: {node: '>=18'}
- whatwg-url@16.0.0:
- resolution: {integrity: sha512-9CcxtEKsf53UFwkSUZjG+9vydAsFO4lFHBpJUtjBcoJOCJpKnSJNwCw813zrYJHpCJ7sgfbtOe0V5Ku7Pa1XMQ==}
+ whatwg-url@16.0.1:
+ resolution: {integrity: sha512-1to4zXBxmXHV3IiSSEInrreIlu02vUOvrhxJJH5vcxYTBDAx51cqZiKdyTxlecdKNSjj8EcxGBxNf6Vg+945gw==}
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
whatwg-url@5.0.0:
@@ -9141,11 +9130,6 @@ packages:
engines: {node: '>= 8'}
hasBin: true
- which@6.0.0:
- resolution: {integrity: sha512-f+gEpIKMR9faW/JgAgPK1D7mekkFoqbmiwvNzuhsHetni20QSgzg9Vhn0g2JSJkkfehQnqdUAx7/e15qS1lPxg==}
- engines: {node: ^20.17.0 || >=22.9.0}
- hasBin: true
-
which@6.0.1:
resolution: {integrity: sha512-oGLe46MIrCRqX7ytPUf66EAYvdeMIZYn3WaocqqKZAxrBpkqHfL/qvTyJ/bTk5+AqHCjXmrv3CEWgy368zhRUg==}
engines: {node: ^20.17.0 || >=22.9.0}
@@ -9360,8 +9344,8 @@ packages:
zod@4.3.6:
resolution: {integrity: sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg==}
- zone.js@0.16.0:
- resolution: {integrity: sha512-LqLPpIQANebrlxY6jKcYKdgN5DTXyyHAKnnWWjE5pPfEQ4n7j5zn7mOEEpwNZVKGqx3kKKmvplEmoBrvpgROTA==}
+ zone.js@0.16.1:
+ resolution: {integrity: sha512-dpvY17vxYIW3+bNrP0ClUlaiY0CiIRK3tnoLaGoQsQcY9/I/NpzIWQ7tQNhbV7LacQMpCII6wVzuL3tuWOyfuA==}
snapshots:
@@ -9472,29 +9456,29 @@ snapshots:
'@jridgewell/gen-mapping': 0.3.13
'@jridgewell/trace-mapping': 0.3.31
- '@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))':
+ '@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))':
dependencies:
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
tslib: 2.8.1
- '@angular/cdk@21.2.0-next.4(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)':
+ '@angular/cdk@21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)':
dependencies:
- '@angular/common': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
- '@angular/platform-browser': 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ '@angular/common': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
+ '@angular/platform-browser': 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
parse5: 8.0.0
rxjs: 7.8.2
tslib: 2.8.1
- '@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)':
+ '@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)':
dependencies:
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
rxjs: 7.8.2
tslib: 2.8.1
- '@angular/compiler-cli@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3)':
+ '@angular/compiler-cli@21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3)':
dependencies:
- '@angular/compiler': 21.2.0-next.3
+ '@angular/compiler': 21.2.0
'@babel/core': 7.29.0
'@jridgewell/sourcemap-codec': 1.5.5
chokidar: 5.0.0
@@ -9508,31 +9492,31 @@ snapshots:
transitivePeerDependencies:
- supports-color
- '@angular/compiler@21.2.0-next.3':
+ '@angular/compiler@21.2.0':
dependencies:
tslib: 2.8.1
- '@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)':
+ '@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)':
dependencies:
rxjs: 7.8.2
tslib: 2.8.1
optionalDependencies:
- '@angular/compiler': 21.2.0-next.3
- zone.js: 0.16.0
+ '@angular/compiler': 21.2.0
+ zone.js: 0.16.1
- '@angular/forms@21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)':
+ '@angular/forms@21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)':
dependencies:
- '@angular/common': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
- '@angular/platform-browser': 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ '@angular/common': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
+ '@angular/platform-browser': 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
'@standard-schema/spec': 1.1.0
rxjs: 7.8.2
tslib: 2.8.1
- '@angular/localize@21.2.0-next.3(@angular/compiler-cli@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3))(@angular/compiler@21.2.0-next.3)':
+ '@angular/localize@21.2.0(@angular/compiler-cli@21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3))(@angular/compiler@21.2.0)':
dependencies:
- '@angular/compiler': 21.2.0-next.3
- '@angular/compiler-cli': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3)
+ '@angular/compiler': 21.2.0
+ '@angular/compiler-cli': 21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3)
'@babel/core': 7.29.0
'@types/babel__core': 7.20.5
tinyglobby: 0.2.15
@@ -9540,24 +9524,24 @@ snapshots:
transitivePeerDependencies:
- supports-color
- '@angular/material@21.2.0-next.4(75d10325bbcfc22e53b1e47f427450d8)':
+ '@angular/material@21.2.0(16f55f8bfdd6508e9b3d27b33a1d1b40)':
dependencies:
- '@angular/cdk': 21.2.0-next.4(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
- '@angular/common': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
- '@angular/forms': 21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)
- '@angular/platform-browser': 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ '@angular/cdk': 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
+ '@angular/common': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
+ '@angular/forms': 21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)
+ '@angular/platform-browser': 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
rxjs: 7.8.2
tslib: 2.8.1
- '@angular/ng-dev@https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/2a5e8e5b5398ae13a8d9a963ac980707313a6c9e(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))':
+ '@angular/ng-dev@https://codeload.github.com/angular/dev-infra-private-ng-dev-builds/tar.gz/c78d7a03ae1ca76d2946651d0dd7f476f09b4374(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))':
dependencies:
'@actions/core': 3.0.0
'@google-cloud/spanner': 8.0.0(supports-color@10.2.2)
- '@google/genai': 1.38.0(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))(bufferutil@4.1.0)(supports-color@10.2.2)(utf-8-validate@6.0.6)
- '@inquirer/prompts': 8.2.0(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
- '@octokit/auth-app': 8.1.2
+ '@google/genai': 1.42.0(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))(bufferutil@4.1.0)(supports-color@10.2.2)(utf-8-validate@6.0.6)
+ '@inquirer/prompts': 8.3.0(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
+ '@octokit/auth-app': 8.2.0
'@octokit/core': 7.0.6
'@octokit/graphql': 9.0.3
'@octokit/graphql-schema': 15.26.1
@@ -9567,14 +9551,14 @@ snapshots:
'@octokit/request-error': 7.1.0
'@octokit/rest': 22.0.1
'@octokit/types': 16.0.0
- '@pnpm/dependency-path': 1001.1.9
+ '@pnpm/dependency-path': 1001.1.10
'@types/cli-progress': 3.11.6
'@types/ejs': 3.1.5
'@types/events': 3.0.3
'@types/folder-hash': 4.0.4
'@types/git-raw-commits': 5.0.1
'@types/jasmine': 6.0.0
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
'@types/semver': 7.7.1
'@types/which': 3.0.4
'@types/yargs': 17.0.35
@@ -9587,23 +9571,23 @@ snapshots:
ejs: 4.0.1
encoding: 0.1.13
fast-glob: 3.3.3
- firebase: 12.8.0
+ firebase: 12.9.0
folder-hash: 4.1.1(supports-color@10.2.2)
git-raw-commits: 5.0.0(conventional-commits-filter@5.0.0)(conventional-commits-parser@6.2.1)
- jasmine: 6.0.0
- jasmine-core: 6.0.0
+ jasmine: 6.1.0
+ jasmine-core: 6.1.0
jasmine-reporters: 2.5.2
jsonc-parser: 3.3.1
- minimatch: 10.1.1
- multimatch: 7.0.0
- nock: 14.0.10
- semver: 7.7.3
+ minimatch: 10.2.3
+ multimatch: 8.0.0
+ nock: 14.0.11
+ semver: 7.7.4
supports-color: 10.2.2
tsx: 4.21.0
typed-graphqlify: 3.1.6
typescript: 5.9.3
utf-8-validate: 6.0.6
- which: 6.0.0
+ which: 6.0.1
yaml: 2.8.2
yargs: 18.0.0
zod: 4.3.6
@@ -9611,42 +9595,42 @@ snapshots:
- '@modelcontextprotocol/sdk'
- '@react-native-async-storage/async-storage'
- '@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))':
+ '@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))':
dependencies:
- '@angular/common': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
+ '@angular/common': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
tslib: 2.8.1
optionalDependencies:
- '@angular/animations': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ '@angular/animations': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
- '@angular/platform-server@21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/compiler@21.2.0-next.3)(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)':
+ '@angular/platform-server@21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/compiler@21.2.0)(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)':
dependencies:
- '@angular/common': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
- '@angular/compiler': 21.2.0-next.3
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
- '@angular/platform-browser': 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ '@angular/common': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
+ '@angular/compiler': 21.2.0
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
+ '@angular/platform-browser': 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
rxjs: 7.8.2
tslib: 2.8.1
xhr2: 0.2.1
- '@angular/router@21.2.0-next.3(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(@angular/platform-browser@21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(rxjs@7.8.2)':
+ '@angular/router@21.2.0(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(@angular/platform-browser@21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(rxjs@7.8.2)':
dependencies:
- '@angular/common': 21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
- '@angular/platform-browser': 21.2.0-next.3(@angular/animations@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)))(@angular/common@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2))(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))
+ '@angular/common': 21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
+ '@angular/platform-browser': 21.2.0(@angular/animations@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)))(@angular/common@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2))(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))
rxjs: 7.8.2
tslib: 2.8.1
- '@angular/service-worker@21.2.0-next.3(@angular/core@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0))(rxjs@7.8.2)':
+ '@angular/service-worker@21.2.0(@angular/core@21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1))(rxjs@7.8.2)':
dependencies:
- '@angular/core': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(rxjs@7.8.2)(zone.js@0.16.0)
+ '@angular/core': 21.2.0(@angular/compiler@21.2.0)(rxjs@7.8.2)(zone.js@0.16.1)
rxjs: 7.8.2
tslib: 2.8.1
- '@asamuzakjp/css-color@4.1.2':
+ '@asamuzakjp/css-color@5.0.1':
dependencies:
'@csstools/css-calc': 3.1.1(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)
- '@csstools/css-color-parser': 4.0.1(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)
+ '@csstools/css-color-parser': 4.0.2(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)
'@csstools/css-parser-algorithms': 4.0.0(@csstools/css-tokenizer@4.0.0)
'@csstools/css-tokenizer': 4.0.0
lru-cache: 11.2.6
@@ -10355,16 +10339,16 @@ snapshots:
dependencies:
'@jridgewell/trace-mapping': 0.3.9
- '@csstools/color-helpers@6.0.1': {}
+ '@csstools/color-helpers@6.0.2': {}
'@csstools/css-calc@3.1.1(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)':
dependencies:
'@csstools/css-parser-algorithms': 4.0.0(@csstools/css-tokenizer@4.0.0)
'@csstools/css-tokenizer': 4.0.0
- '@csstools/css-color-parser@4.0.1(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)':
+ '@csstools/css-color-parser@4.0.2(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)':
dependencies:
- '@csstools/color-helpers': 6.0.1
+ '@csstools/color-helpers': 6.0.2
'@csstools/css-calc': 3.1.1(@csstools/css-parser-algorithms@4.0.0(@csstools/css-tokenizer@4.0.0))(@csstools/css-tokenizer@4.0.0)
'@csstools/css-parser-algorithms': 4.0.0(@csstools/css-tokenizer@4.0.0)
'@csstools/css-tokenizer': 4.0.0
@@ -10373,7 +10357,7 @@ snapshots:
dependencies:
'@csstools/css-tokenizer': 4.0.0
- '@csstools/css-syntax-patches-for-csstree@1.0.27': {}
+ '@csstools/css-syntax-patches-for-csstree@1.0.28': {}
'@csstools/css-tokenizer@4.0.0': {}
@@ -10511,7 +10495,7 @@ snapshots:
dependencies:
'@eslint/object-schema': 2.1.7
debug: 4.4.3(supports-color@10.2.2)
- minimatch: 3.1.2
+ minimatch: 3.1.3
transitivePeerDependencies:
- supports-color
@@ -10529,14 +10513,14 @@ snapshots:
'@eslint/eslintrc@3.3.3':
dependencies:
- ajv: 6.12.6
+ ajv: 6.14.0
debug: 4.4.3(supports-color@10.2.2)
espree: 10.4.0
globals: 14.0.0
ignore: 5.3.2
import-fresh: 3.3.1
js-yaml: 4.1.1
- minimatch: 3.1.2
+ minimatch: 3.1.3
strip-json-comments: 3.1.1
transitivePeerDependencies:
- supports-color
@@ -10552,9 +10536,9 @@ snapshots:
'@exodus/bytes@1.14.1': {}
- '@firebase/ai@2.7.0(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)':
+ '@firebase/ai@2.8.0(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/app-check-interop-types': 0.3.3
'@firebase/app-types': 0.9.3
'@firebase/component': 0.7.0
@@ -10562,11 +10546,11 @@ snapshots:
'@firebase/util': 1.13.0
tslib: 2.8.1
- '@firebase/analytics-compat@0.2.25(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)':
+ '@firebase/analytics-compat@0.2.25(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/analytics': 0.10.19(@firebase/app@0.14.7)
+ '@firebase/analytics': 0.10.19(@firebase/app@0.14.8)
'@firebase/analytics-types': 0.8.3
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10575,20 +10559,20 @@ snapshots:
'@firebase/analytics-types@0.8.3': {}
- '@firebase/analytics@0.10.19(@firebase/app@0.14.7)':
+ '@firebase/analytics@0.10.19(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
- '@firebase/installations': 0.6.19(@firebase/app@0.14.7)
+ '@firebase/installations': 0.6.19(@firebase/app@0.14.8)
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
tslib: 2.8.1
- '@firebase/app-check-compat@0.4.0(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)':
+ '@firebase/app-check-compat@0.4.0(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-check': 0.11.0(@firebase/app@0.14.7)
+ '@firebase/app-check': 0.11.0(@firebase/app@0.14.8)
'@firebase/app-check-types': 0.5.3
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
@@ -10600,17 +10584,17 @@ snapshots:
'@firebase/app-check-types@0.5.3': {}
- '@firebase/app-check@0.11.0(@firebase/app@0.14.7)':
+ '@firebase/app-check@0.11.0(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
tslib: 2.8.1
- '@firebase/app-compat@0.5.7':
+ '@firebase/app-compat@0.5.8':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
@@ -10618,7 +10602,7 @@ snapshots:
'@firebase/app-types@0.9.3': {}
- '@firebase/app@0.14.7':
+ '@firebase/app@0.14.8':
dependencies:
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
@@ -10626,10 +10610,10 @@ snapshots:
idb: 7.1.1
tslib: 2.8.1
- '@firebase/auth-compat@0.6.2(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)':
+ '@firebase/auth-compat@0.6.2(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
- '@firebase/auth': 1.12.0(@firebase/app@0.14.7)
+ '@firebase/app-compat': 0.5.8
+ '@firebase/auth': 1.12.0(@firebase/app@0.14.8)
'@firebase/auth-types': 0.13.0(@firebase/app-types@0.9.3)(@firebase/util@1.13.0)
'@firebase/component': 0.7.0
'@firebase/util': 1.13.0
@@ -10646,9 +10630,9 @@ snapshots:
'@firebase/app-types': 0.9.3
'@firebase/util': 1.13.0
- '@firebase/auth@1.12.0(@firebase/app@0.14.7)':
+ '@firebase/auth@1.12.0(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
@@ -10659,9 +10643,9 @@ snapshots:
'@firebase/util': 1.13.0
tslib: 2.8.1
- '@firebase/data-connect@0.3.12(@firebase/app@0.14.7)':
+ '@firebase/data-connect@0.3.12(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/auth-interop-types': 0.2.4
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
@@ -10692,11 +10676,11 @@ snapshots:
faye-websocket: 0.11.4
tslib: 2.8.1
- '@firebase/firestore-compat@0.4.4(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)':
+ '@firebase/firestore-compat@0.4.5(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
- '@firebase/firestore': 4.10.0(@firebase/app@0.14.7)
+ '@firebase/firestore': 4.11.0(@firebase/app@0.14.8)
'@firebase/firestore-types': 3.0.3(@firebase/app-types@0.9.3)(@firebase/util@1.13.0)
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10709,9 +10693,9 @@ snapshots:
'@firebase/app-types': 0.9.3
'@firebase/util': 1.13.0
- '@firebase/firestore@4.10.0(@firebase/app@0.14.7)':
+ '@firebase/firestore@4.11.0(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
@@ -10720,11 +10704,11 @@ snapshots:
'@grpc/proto-loader': 0.7.15
tslib: 2.8.1
- '@firebase/functions-compat@0.4.1(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)':
+ '@firebase/functions-compat@0.4.1(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
- '@firebase/functions': 0.13.1(@firebase/app@0.14.7)
+ '@firebase/functions': 0.13.1(@firebase/app@0.14.8)
'@firebase/functions-types': 0.6.3
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10733,9 +10717,9 @@ snapshots:
'@firebase/functions-types@0.6.3': {}
- '@firebase/functions@0.13.1(@firebase/app@0.14.7)':
+ '@firebase/functions@0.13.1(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/app-check-interop-types': 0.3.3
'@firebase/auth-interop-types': 0.2.4
'@firebase/component': 0.7.0
@@ -10743,11 +10727,11 @@ snapshots:
'@firebase/util': 1.13.0
tslib: 2.8.1
- '@firebase/installations-compat@0.2.19(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)':
+ '@firebase/installations-compat@0.2.19(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
- '@firebase/installations': 0.6.19(@firebase/app@0.14.7)
+ '@firebase/installations': 0.6.19(@firebase/app@0.14.8)
'@firebase/installations-types': 0.5.3(@firebase/app-types@0.9.3)
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10759,9 +10743,9 @@ snapshots:
dependencies:
'@firebase/app-types': 0.9.3
- '@firebase/installations@0.6.19(@firebase/app@0.14.7)':
+ '@firebase/installations@0.6.19(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
'@firebase/util': 1.13.0
idb: 7.1.1
@@ -10771,11 +10755,11 @@ snapshots:
dependencies:
tslib: 2.8.1
- '@firebase/messaging-compat@0.2.23(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)':
+ '@firebase/messaging-compat@0.2.23(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
- '@firebase/messaging': 0.12.23(@firebase/app@0.14.7)
+ '@firebase/messaging': 0.12.23(@firebase/app@0.14.8)
'@firebase/util': 1.13.0
tslib: 2.8.1
transitivePeerDependencies:
@@ -10783,22 +10767,22 @@ snapshots:
'@firebase/messaging-interop-types@0.2.3': {}
- '@firebase/messaging@0.12.23(@firebase/app@0.14.7)':
+ '@firebase/messaging@0.12.23(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
- '@firebase/installations': 0.6.19(@firebase/app@0.14.7)
+ '@firebase/installations': 0.6.19(@firebase/app@0.14.8)
'@firebase/messaging-interop-types': 0.2.3
'@firebase/util': 1.13.0
idb: 7.1.1
tslib: 2.8.1
- '@firebase/performance-compat@0.2.22(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)':
+ '@firebase/performance-compat@0.2.22(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
- '@firebase/performance': 0.7.9(@firebase/app@0.14.7)
+ '@firebase/performance': 0.7.9(@firebase/app@0.14.8)
'@firebase/performance-types': 0.2.3
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10807,22 +10791,22 @@ snapshots:
'@firebase/performance-types@0.2.3': {}
- '@firebase/performance@0.7.9(@firebase/app@0.14.7)':
+ '@firebase/performance@0.7.9(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
- '@firebase/installations': 0.6.19(@firebase/app@0.14.7)
+ '@firebase/installations': 0.6.19(@firebase/app@0.14.8)
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
tslib: 2.8.1
web-vitals: 4.2.4
- '@firebase/remote-config-compat@0.2.21(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)':
+ '@firebase/remote-config-compat@0.2.21(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
'@firebase/logger': 0.5.0
- '@firebase/remote-config': 0.8.0(@firebase/app@0.14.7)
+ '@firebase/remote-config': 0.8.0(@firebase/app@0.14.8)
'@firebase/remote-config-types': 0.5.0
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10831,20 +10815,20 @@ snapshots:
'@firebase/remote-config-types@0.5.0': {}
- '@firebase/remote-config@0.8.0(@firebase/app@0.14.7)':
+ '@firebase/remote-config@0.8.0(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
- '@firebase/installations': 0.6.19(@firebase/app@0.14.7)
+ '@firebase/installations': 0.6.19(@firebase/app@0.14.8)
'@firebase/logger': 0.5.0
'@firebase/util': 1.13.0
tslib: 2.8.1
- '@firebase/storage-compat@0.4.0(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)':
+ '@firebase/storage-compat@0.4.0(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app-compat': 0.5.7
+ '@firebase/app-compat': 0.5.8
'@firebase/component': 0.7.0
- '@firebase/storage': 0.14.0(@firebase/app@0.14.7)
+ '@firebase/storage': 0.14.0(@firebase/app@0.14.8)
'@firebase/storage-types': 0.8.3(@firebase/app-types@0.9.3)(@firebase/util@1.13.0)
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10857,9 +10841,9 @@ snapshots:
'@firebase/app-types': 0.9.3
'@firebase/util': 1.13.0
- '@firebase/storage@0.14.0(@firebase/app@0.14.7)':
+ '@firebase/storage@0.14.0(@firebase/app@0.14.8)':
dependencies:
- '@firebase/app': 0.14.7
+ '@firebase/app': 0.14.8
'@firebase/component': 0.7.0
'@firebase/util': 1.13.0
tslib: 2.8.1
@@ -10931,9 +10915,10 @@ snapshots:
transitivePeerDependencies:
- supports-color
- '@google/genai@1.38.0(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))(bufferutil@4.1.0)(supports-color@10.2.2)(utf-8-validate@6.0.6)':
+ '@google/genai@1.42.0(@modelcontextprotocol/sdk@1.26.0(zod@4.3.6))(bufferutil@4.1.0)(supports-color@10.2.2)(utf-8-validate@6.0.6)':
dependencies:
google-auth-library: 10.5.0(supports-color@10.2.2)
+ p-retry: 4.6.2
protobufjs: 7.5.4
ws: 8.19.0(bufferutil@4.1.0)(utf-8-validate@6.0.6)
optionalDependencies:
@@ -10972,9 +10957,9 @@ snapshots:
'@harperfast/extended-iterable@1.0.3':
optional: true
- '@hono/node-server@1.19.9(hono@4.11.9)':
+ '@hono/node-server@1.19.9(hono@4.12.2)':
dependencies:
- hono: 4.11.9
+ hono: 4.12.2
'@humanfs/core@0.19.1': {}
@@ -10991,251 +10976,245 @@ snapshots:
'@inquirer/ansi@2.0.3': {}
- '@inquirer/checkbox@4.3.2(@types/node@24.10.9)':
+ '@inquirer/checkbox@4.3.2(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 1.0.2
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
'@inquirer/figures': 1.0.15
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
yoctocolors-cjs: 2.1.3
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/checkbox@5.0.6(@types/node@24.10.9)':
+ '@inquirer/checkbox@5.1.0(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 2.0.3
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
'@inquirer/figures': 2.0.3
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/confirm@5.1.21(@types/node@24.10.9)':
+ '@inquirer/confirm@5.1.21(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/confirm@6.0.6(@types/node@24.10.9)':
+ '@inquirer/confirm@6.0.8(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/core@10.3.2(@types/node@24.10.9)':
+ '@inquirer/core@10.3.2(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 1.0.2
'@inquirer/figures': 1.0.15
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
cli-width: 4.1.0
mute-stream: 2.0.0
signal-exit: 4.1.0
wrap-ansi: 6.2.0
yoctocolors-cjs: 2.1.3
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/core@11.1.3(@types/node@24.10.9)':
+ '@inquirer/core@11.1.5(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 2.0.3
'@inquirer/figures': 2.0.3
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
cli-width: 4.1.0
fast-wrap-ansi: 0.2.0
mute-stream: 3.0.0
signal-exit: 4.1.0
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/editor@4.2.23(@types/node@24.10.9)':
+ '@inquirer/editor@4.2.23(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/external-editor': 1.0.3(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/external-editor': 1.0.3(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/editor@5.0.6(@types/node@24.10.9)':
+ '@inquirer/editor@5.0.8(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/external-editor': 2.0.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/external-editor': 2.0.3(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/expand@4.0.23(@types/node@24.10.9)':
+ '@inquirer/expand@4.0.23(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
yoctocolors-cjs: 2.1.3
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/expand@5.0.6(@types/node@24.10.9)':
+ '@inquirer/expand@5.0.8(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/external-editor@1.0.3(@types/node@24.10.9)':
+ '@inquirer/external-editor@1.0.3(@types/node@24.10.13)':
dependencies:
chardet: 2.1.1
iconv-lite: 0.7.2
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/external-editor@2.0.3(@types/node@24.10.9)':
+ '@inquirer/external-editor@2.0.3(@types/node@24.10.13)':
dependencies:
chardet: 2.1.1
iconv-lite: 0.7.2
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
'@inquirer/figures@1.0.15': {}
'@inquirer/figures@2.0.3': {}
- '@inquirer/input@4.3.1(@types/node@24.10.9)':
+ '@inquirer/input@4.3.1(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/input@5.0.6(@types/node@24.10.9)':
+ '@inquirer/input@5.0.8(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/number@3.0.23(@types/node@24.10.9)':
+ '@inquirer/number@3.0.23(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/number@4.0.6(@types/node@24.10.9)':
+ '@inquirer/number@4.0.8(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/password@4.0.23(@types/node@24.10.9)':
+ '@inquirer/password@4.0.23(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 1.0.2
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/password@5.0.6(@types/node@24.10.9)':
+ '@inquirer/password@5.0.8(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 2.0.3
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
-
- '@inquirer/prompts@7.10.1(@types/node@24.10.9)':
- dependencies:
- '@inquirer/checkbox': 4.3.2(@types/node@24.10.9)
- '@inquirer/confirm': 5.1.21(@types/node@24.10.9)
- '@inquirer/editor': 4.2.23(@types/node@24.10.9)
- '@inquirer/expand': 4.0.23(@types/node@24.10.9)
- '@inquirer/input': 4.3.1(@types/node@24.10.9)
- '@inquirer/number': 3.0.23(@types/node@24.10.9)
- '@inquirer/password': 4.0.23(@types/node@24.10.9)
- '@inquirer/rawlist': 4.1.11(@types/node@24.10.9)
- '@inquirer/search': 3.2.2(@types/node@24.10.9)
- '@inquirer/select': 4.4.2(@types/node@24.10.9)
+ '@types/node': 24.10.13
+
+ '@inquirer/prompts@7.10.1(@types/node@24.10.13)':
+ dependencies:
+ '@inquirer/checkbox': 4.3.2(@types/node@24.10.13)
+ '@inquirer/confirm': 5.1.21(@types/node@24.10.13)
+ '@inquirer/editor': 4.2.23(@types/node@24.10.13)
+ '@inquirer/expand': 4.0.23(@types/node@24.10.13)
+ '@inquirer/input': 4.3.1(@types/node@24.10.13)
+ '@inquirer/number': 3.0.23(@types/node@24.10.13)
+ '@inquirer/password': 4.0.23(@types/node@24.10.13)
+ '@inquirer/rawlist': 4.1.11(@types/node@24.10.13)
+ '@inquirer/search': 3.2.2(@types/node@24.10.13)
+ '@inquirer/select': 4.4.2(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
-
- '@inquirer/prompts@8.2.0(@types/node@24.10.9)':
- dependencies:
- '@inquirer/checkbox': 5.0.6(@types/node@24.10.9)
- '@inquirer/confirm': 6.0.6(@types/node@24.10.9)
- '@inquirer/editor': 5.0.6(@types/node@24.10.9)
- '@inquirer/expand': 5.0.6(@types/node@24.10.9)
- '@inquirer/input': 5.0.6(@types/node@24.10.9)
- '@inquirer/number': 4.0.6(@types/node@24.10.9)
- '@inquirer/password': 5.0.6(@types/node@24.10.9)
- '@inquirer/rawlist': 5.2.2(@types/node@24.10.9)
- '@inquirer/search': 4.1.2(@types/node@24.10.9)
- '@inquirer/select': 5.0.6(@types/node@24.10.9)
+ '@types/node': 24.10.13
+
+ '@inquirer/prompts@8.3.0(@types/node@24.10.13)':
+ dependencies:
+ '@inquirer/checkbox': 5.1.0(@types/node@24.10.13)
+ '@inquirer/confirm': 6.0.8(@types/node@24.10.13)
+ '@inquirer/editor': 5.0.8(@types/node@24.10.13)
+ '@inquirer/expand': 5.0.8(@types/node@24.10.13)
+ '@inquirer/input': 5.0.8(@types/node@24.10.13)
+ '@inquirer/number': 4.0.8(@types/node@24.10.13)
+ '@inquirer/password': 5.0.8(@types/node@24.10.13)
+ '@inquirer/rawlist': 5.2.4(@types/node@24.10.13)
+ '@inquirer/search': 4.1.4(@types/node@24.10.13)
+ '@inquirer/select': 5.1.0(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/rawlist@4.1.11(@types/node@24.10.9)':
+ '@inquirer/rawlist@4.1.11(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
yoctocolors-cjs: 2.1.3
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/rawlist@5.2.2(@types/node@24.10.9)':
+ '@inquirer/rawlist@5.2.4(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/search@3.2.2(@types/node@24.10.9)':
+ '@inquirer/search@3.2.2(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
'@inquirer/figures': 1.0.15
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
yoctocolors-cjs: 2.1.3
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/search@4.1.2(@types/node@24.10.9)':
+ '@inquirer/search@4.1.4(@types/node@24.10.13)':
dependencies:
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
'@inquirer/figures': 2.0.3
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/select@4.4.2(@types/node@24.10.9)':
+ '@inquirer/select@4.4.2(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 1.0.2
- '@inquirer/core': 10.3.2(@types/node@24.10.9)
+ '@inquirer/core': 10.3.2(@types/node@24.10.13)
'@inquirer/figures': 1.0.15
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
yoctocolors-cjs: 2.1.3
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/select@5.0.6(@types/node@24.10.9)':
+ '@inquirer/select@5.1.0(@types/node@24.10.13)':
dependencies:
'@inquirer/ansi': 2.0.3
- '@inquirer/core': 11.1.3(@types/node@24.10.9)
+ '@inquirer/core': 11.1.5(@types/node@24.10.13)
'@inquirer/figures': 2.0.3
- '@inquirer/type': 4.0.3(@types/node@24.10.9)
+ '@inquirer/type': 4.0.3(@types/node@24.10.13)
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/type@3.0.10(@types/node@24.10.9)':
+ '@inquirer/type@3.0.10(@types/node@24.10.13)':
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
- '@inquirer/type@4.0.3(@types/node@24.10.9)':
+ '@inquirer/type@4.0.3(@types/node@24.10.13)':
optionalDependencies:
- '@types/node': 24.10.9
-
- '@isaacs/balanced-match@4.0.1': {}
-
- '@isaacs/brace-expansion@5.0.1':
- dependencies:
- '@isaacs/balanced-match': 4.0.1
+ '@types/node': 24.10.13
'@isaacs/cliui@8.0.2':
dependencies:
@@ -11246,11 +11225,9 @@ snapshots:
wrap-ansi: 8.1.0
wrap-ansi-cjs: wrap-ansi@7.0.0
- '@isaacs/cliui@9.0.0': {}
-
'@isaacs/fs-minipass@4.0.1':
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
'@istanbuljs/schema@0.1.3': {}
@@ -11416,10 +11393,10 @@ snapshots:
'@leichtgewicht/ip-codec@2.0.5': {}
- '@listr2/prompt-adapter-inquirer@3.0.5(@inquirer/prompts@7.10.1(@types/node@24.10.9))(@types/node@24.10.9)(listr2@9.0.5)':
+ '@listr2/prompt-adapter-inquirer@3.0.5(@inquirer/prompts@7.10.1(@types/node@24.10.13))(@types/node@24.10.13)(listr2@9.0.5)':
dependencies:
- '@inquirer/prompts': 7.10.1(@types/node@24.10.9)
- '@inquirer/type': 3.0.10(@types/node@24.10.9)
+ '@inquirer/prompts': 7.10.1(@types/node@24.10.13)
+ '@inquirer/type': 3.0.10(@types/node@24.10.13)
listr2: 9.0.5
transitivePeerDependencies:
- '@types/node'
@@ -11447,7 +11424,7 @@ snapshots:
'@modelcontextprotocol/sdk@1.26.0(zod@4.3.6)':
dependencies:
- '@hono/node-server': 1.19.9(hono@4.11.9)
+ '@hono/node-server': 1.19.9(hono@4.12.2)
ajv: 8.18.0
ajv-formats: 3.0.1(ajv@8.18.0)
content-type: 1.0.5
@@ -11457,7 +11434,7 @@ snapshots:
eventsource-parser: 3.0.6
express: 5.2.1
express-rate-limit: 8.2.1(express@5.2.1)
- hono: 4.11.9
+ hono: 4.12.2
jose: 6.1.3
json-schema-typed: 8.0.2
pkce-challenge: 5.0.1
@@ -11485,7 +11462,7 @@ snapshots:
'@msgpackr-extract/msgpackr-extract-win32-x64@3.0.3':
optional: true
- '@mswjs/interceptors@0.39.8':
+ '@mswjs/interceptors@0.41.3':
dependencies:
'@open-draft/deferred-promise': 2.2.0
'@open-draft/logger': 0.3.0
@@ -11619,15 +11596,15 @@ snapshots:
'@npmcli/node-gyp@5.0.0': {}
- '@npmcli/package-json@7.0.4':
+ '@npmcli/package-json@7.0.5':
dependencies:
'@npmcli/git': 7.0.1
- glob: 13.0.3
+ glob: 13.0.6
hosted-git-info: 9.0.2
json-parse-even-better-errors: 5.0.0
proc-log: 6.1.0
semver: 7.7.4
- validate-npm-package-license: 3.0.4
+ spdx-expression-parse: 4.0.0
'@npmcli/promise-spawn@9.0.1':
dependencies:
@@ -11638,7 +11615,7 @@ snapshots:
'@npmcli/run-script@10.0.3':
dependencies:
'@npmcli/node-gyp': 5.0.0
- '@npmcli/package-json': 7.0.4
+ '@npmcli/package-json': 7.0.5
'@npmcli/promise-spawn': 9.0.1
node-gyp: 12.2.0
proc-log: 6.1.0
@@ -11646,11 +11623,11 @@ snapshots:
transitivePeerDependencies:
- supports-color
- '@octokit/auth-app@8.1.2':
+ '@octokit/auth-app@8.2.0':
dependencies:
'@octokit/auth-oauth-app': 9.0.3
'@octokit/auth-oauth-user': 6.0.2
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/request-error': 7.1.0
'@octokit/types': 16.0.0
toad-cache: 3.7.0
@@ -11661,14 +11638,14 @@ snapshots:
dependencies:
'@octokit/auth-oauth-device': 8.0.3
'@octokit/auth-oauth-user': 6.0.2
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/types': 16.0.0
universal-user-agent: 7.0.3
'@octokit/auth-oauth-device@8.0.3':
dependencies:
'@octokit/oauth-methods': 6.0.2
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/types': 16.0.0
universal-user-agent: 7.0.3
@@ -11676,7 +11653,7 @@ snapshots:
dependencies:
'@octokit/auth-oauth-device': 8.0.3
'@octokit/oauth-methods': 6.0.2
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/types': 16.0.0
universal-user-agent: 7.0.3
@@ -11686,25 +11663,25 @@ snapshots:
dependencies:
'@octokit/auth-token': 6.0.0
'@octokit/graphql': 9.0.3
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/request-error': 7.1.0
'@octokit/types': 16.0.0
before-after-hook: 4.0.0
universal-user-agent: 7.0.3
- '@octokit/endpoint@11.0.2':
+ '@octokit/endpoint@11.0.3':
dependencies:
'@octokit/types': 16.0.0
universal-user-agent: 7.0.3
'@octokit/graphql-schema@15.26.1':
dependencies:
- graphql: 16.12.0
- graphql-tag: 2.12.6(graphql@16.12.0)
+ graphql: 16.13.0
+ graphql-tag: 2.12.6(graphql@16.13.0)
'@octokit/graphql@9.0.3':
dependencies:
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/types': 16.0.0
universal-user-agent: 7.0.3
@@ -11713,7 +11690,7 @@ snapshots:
'@octokit/oauth-methods@6.0.2':
dependencies:
'@octokit/oauth-authorization-url': 8.0.0
- '@octokit/request': 10.0.7
+ '@octokit/request': 10.0.8
'@octokit/request-error': 7.1.0
'@octokit/types': 16.0.0
@@ -11737,12 +11714,13 @@ snapshots:
dependencies:
'@octokit/types': 16.0.0
- '@octokit/request@10.0.7':
+ '@octokit/request@10.0.8':
dependencies:
- '@octokit/endpoint': 11.0.2
+ '@octokit/endpoint': 11.0.3
'@octokit/request-error': 7.1.0
'@octokit/types': 16.0.0
fast-content-type-parse: 3.0.0
+ json-with-bigint: 3.5.3
universal-user-agent: 7.0.3
'@octokit/rest@22.0.1':
@@ -11936,21 +11914,21 @@ snapshots:
'@pkgjs/parseargs@0.11.0':
optional: true
- '@pnpm/crypto.hash@1000.2.1':
+ '@pnpm/crypto.hash@1000.2.2':
dependencies:
'@pnpm/crypto.polyfill': 1000.1.0
- '@pnpm/graceful-fs': 1000.0.1
+ '@pnpm/graceful-fs': 1000.1.0
ssri: 10.0.5
'@pnpm/crypto.polyfill@1000.1.0': {}
- '@pnpm/dependency-path@1001.1.9':
+ '@pnpm/dependency-path@1001.1.10':
dependencies:
- '@pnpm/crypto.hash': 1000.2.1
+ '@pnpm/crypto.hash': 1000.2.2
'@pnpm/types': 1001.3.0
semver: 7.7.4
- '@pnpm/graceful-fs@1000.0.1':
+ '@pnpm/graceful-fs@1000.1.0':
dependencies:
graceful-fs: 4.2.11
@@ -11979,7 +11957,7 @@ snapshots:
'@protobufjs/utf8@1.1.0': {}
- '@puppeteer/browsers@2.12.1':
+ '@puppeteer/browsers@2.13.0':
dependencies:
debug: 4.4.3(supports-color@10.2.2)
extract-zip: 2.0.1
@@ -12170,7 +12148,7 @@ snapshots:
'@rollup/rollup-win32-x64-msvc@4.57.1':
optional: true
- '@rollup/wasm-node@4.57.1':
+ '@rollup/wasm-node@4.59.0':
dependencies:
'@types/estree': 1.0.8
optionalDependencies:
@@ -12216,10 +12194,10 @@ snapshots:
'@standard-schema/spec@1.1.0': {}
- '@stylistic/eslint-plugin@5.8.0(eslint@9.39.2(jiti@2.6.1))':
+ '@stylistic/eslint-plugin@5.9.0(eslint@9.39.2(jiti@2.6.1))':
dependencies:
'@eslint-community/eslint-utils': 4.9.1(eslint@9.39.2(jiti@2.6.1))
- '@typescript-eslint/types': 8.55.0
+ '@typescript-eslint/types': 8.56.1
eslint: 9.39.2(jiti@2.6.1)
eslint-visitor-keys: 4.2.1
espree: 10.4.0
@@ -12247,7 +12225,7 @@ snapshots:
'@tufjs/models@4.1.0':
dependencies:
'@tufjs/canonical-json': 2.0.0
- minimatch: 10.2.0
+ minimatch: 10.2.3
'@tybys/wasm-util@0.10.1':
dependencies:
@@ -12468,7 +12446,7 @@ snapshots:
- uglify-js
- webpack-cli
- '@types/lodash@4.17.23': {}
+ '@types/lodash@4.17.24': {}
'@types/micromatch@2.3.35':
dependencies:
@@ -12485,7 +12463,7 @@ snapshots:
dependencies:
undici-types: 7.22.0
- '@types/node@24.10.9':
+ '@types/node@24.10.13':
dependencies:
undici-types: 7.22.0
@@ -12537,6 +12515,8 @@ snapshots:
dependencies:
'@types/node': 22.19.11
+ '@types/retry@0.12.0': {}
+
'@types/retry@0.12.2': {}
'@types/selenium-webdriver@3.0.26': {}
@@ -12669,6 +12649,8 @@ snapshots:
'@typescript-eslint/types@8.55.0': {}
+ '@typescript-eslint/types@8.56.1': {}
+
'@typescript-eslint/typescript-estree@8.55.0(typescript@5.9.3)':
dependencies:
'@typescript-eslint/project-service': 8.55.0(typescript@5.9.3)
@@ -12676,7 +12658,7 @@ snapshots:
'@typescript-eslint/types': 8.55.0
'@typescript-eslint/visitor-keys': 8.55.0
debug: 4.4.3(supports-color@10.2.2)
- minimatch: 9.0.5
+ minimatch: 9.0.6
semver: 7.7.4
tinyglobby: 0.2.15
ts-api-utils: 2.4.0(typescript@5.9.3)
@@ -12857,11 +12839,11 @@ snapshots:
lodash: 4.17.21
minimatch: 7.4.6
- '@vitejs/plugin-basic-ssl@2.1.4(vite@7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))':
+ '@vitejs/plugin-basic-ssl@2.1.4(vite@7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))':
dependencies:
- vite: 7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ vite: 7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
- '@vitest/coverage-v8@4.0.18(vitest@4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.9)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))':
+ '@vitest/coverage-v8@4.0.18(vitest@4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.13)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))':
dependencies:
'@bcoe/v8-coverage': 1.0.2
'@vitest/utils': 4.0.18
@@ -12873,7 +12855,7 @@ snapshots:
obug: 2.1.1
std-env: 3.10.0
tinyrainbow: 3.0.3
- vitest: 4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.9)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ vitest: 4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.13)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
'@vitest/expect@4.0.18':
dependencies:
@@ -12884,13 +12866,13 @@ snapshots:
chai: 6.2.2
tinyrainbow: 3.0.3
- '@vitest/mocker@4.0.18(vite@7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))':
+ '@vitest/mocker@4.0.18(vite@7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))':
dependencies:
'@vitest/spy': 4.0.18
estree-walker: 3.0.3
magic-string: 0.30.21
optionalDependencies:
- vite: 7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ vite: 7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
'@vitest/pretty-format@4.0.18':
dependencies:
@@ -12989,7 +12971,7 @@ snapshots:
'@web/test-runner-core': 0.13.4(bufferutil@4.1.0)
'@web/test-runner-coverage-v8': 0.8.0(bufferutil@4.1.0)
chrome-launcher: 0.15.2
- puppeteer-core: 24.37.3(bufferutil@4.1.0)
+ puppeteer-core: 24.37.5(bufferutil@4.1.0)
transitivePeerDependencies:
- bare-abort-controller
- bare-buffer
@@ -13191,19 +13173,19 @@ snapshots:
mime-types: 3.0.2
negotiator: 1.0.0
- acorn-import-phases@1.0.4(acorn@8.15.0):
+ acorn-import-phases@1.0.4(acorn@8.16.0):
dependencies:
- acorn: 8.15.0
+ acorn: 8.16.0
- acorn-jsx@5.3.2(acorn@8.15.0):
+ acorn-jsx@5.3.2(acorn@8.16.0):
dependencies:
- acorn: 8.15.0
+ acorn: 8.16.0
- acorn-walk@8.3.4:
+ acorn-walk@8.3.5:
dependencies:
- acorn: 8.15.0
+ acorn: 8.16.0
- acorn@8.15.0: {}
+ acorn@8.16.0: {}
adjust-sourcemap-loader@4.0.0:
dependencies:
@@ -13237,7 +13219,7 @@ snapshots:
ajv: 8.18.0
fast-deep-equal: 3.1.3
- ajv@6.12.6:
+ ajv@6.14.0:
dependencies:
fast-deep-equal: 3.1.3
fast-json-stable-stringify: 2.1.0
@@ -13427,7 +13409,7 @@ snapshots:
autoprefixer@10.4.24(postcss@8.5.6):
dependencies:
browserslist: 4.28.1
- caniuse-lite: 1.0.30001770
+ caniuse-lite: 1.0.30001774
fraction.js: 5.3.4
picocolors: 1.1.1
postcss: 8.5.6
@@ -13441,7 +13423,7 @@ snapshots:
aws4@1.13.2: {}
- b4a@1.7.4: {}
+ b4a@1.8.0: {}
babel-loader@10.0.0(@babel/core@7.29.0)(webpack@5.105.2(esbuild@0.27.3)):
dependencies:
@@ -13483,9 +13465,7 @@ snapshots:
balanced-match@1.0.2: {}
- balanced-match@4.0.2:
- dependencies:
- jackspeak: 4.2.3
+ balanced-match@4.0.4: {}
bare-events@2.8.2: {}
@@ -13493,7 +13473,7 @@ snapshots:
dependencies:
bare-events: 2.8.2
bare-path: 3.0.0
- bare-stream: 2.7.0(bare-events@2.8.2)
+ bare-stream: 2.8.0(bare-events@2.8.2)
bare-url: 2.3.2
fast-fifo: 1.3.2
transitivePeerDependencies:
@@ -13509,9 +13489,10 @@ snapshots:
bare-os: 3.6.2
optional: true
- bare-stream@2.7.0(bare-events@2.8.2):
+ bare-stream@2.8.0(bare-events@2.8.2):
dependencies:
streamx: 2.23.0
+ teex: 1.0.1
optionalDependencies:
bare-events: 2.8.2
transitivePeerDependencies:
@@ -13528,9 +13509,9 @@ snapshots:
base64id@2.0.0: {}
- baseline-browser-mapping@2.9.19: {}
+ baseline-browser-mapping@2.10.0: {}
- basic-ftp@5.1.0: {}
+ basic-ftp@5.2.0: {}
batch@0.6.1: {}
@@ -13623,9 +13604,9 @@ snapshots:
dependencies:
balanced-match: 1.0.2
- brace-expansion@5.0.2:
+ brace-expansion@5.0.3:
dependencies:
- balanced-match: 4.0.2
+ balanced-match: 4.0.4
braces@3.0.3:
dependencies:
@@ -13695,9 +13676,9 @@ snapshots:
browserslist@4.28.1:
dependencies:
- baseline-browser-mapping: 2.9.19
- caniuse-lite: 1.0.30001770
- electron-to-chromium: 1.5.286
+ baseline-browser-mapping: 2.10.0
+ caniuse-lite: 1.0.30001774
+ electron-to-chromium: 1.5.302
node-releases: 2.0.27
update-browserslist-db: 1.2.3(browserslist@4.28.1)
@@ -13741,9 +13722,9 @@ snapshots:
dependencies:
'@npmcli/fs': 5.0.0
fs-minipass: 3.0.3
- glob: 13.0.3
+ glob: 13.0.6
lru-cache: 11.2.6
- minipass: 7.1.2
+ minipass: 7.1.3
minipass-collect: 2.0.1
minipass-flush: 1.0.5
minipass-pipeline: 1.2.4
@@ -13791,7 +13772,7 @@ snapshots:
camelcase@6.3.0: {}
- caniuse-lite@1.0.30001770: {}
+ caniuse-lite@1.0.30001774: {}
caseless@0.12.0: {}
@@ -13884,7 +13865,7 @@ snapshots:
cli-truncate@5.1.1:
dependencies:
slice-ansi: 7.1.2
- string-width: 8.1.1
+ string-width: 8.2.0
cli-width@4.1.0: {}
@@ -14137,10 +14118,10 @@ snapshots:
cssesc@3.0.0: {}
- cssstyle@6.0.1:
+ cssstyle@6.1.0:
dependencies:
- '@asamuzakjp/css-color': 4.1.2
- '@csstools/css-syntax-patches-for-csstree': 1.0.27
+ '@asamuzakjp/css-color': 5.0.1
+ '@csstools/css-syntax-patches-for-csstree': 1.0.28
css-tree: 3.1.0
lru-cache: 11.2.6
@@ -14157,7 +14138,7 @@ snapshots:
data-urls@7.0.0:
dependencies:
whatwg-mimetype: 5.0.0
- whatwg-url: 16.0.0
+ whatwg-url: 16.0.1
transitivePeerDependencies:
- '@noble/hashes'
@@ -14387,7 +14368,7 @@ snapshots:
dependencies:
jake: 10.9.4
- electron-to-chromium@1.5.286: {}
+ electron-to-chromium@1.5.302: {}
emoji-regex@10.6.0: {}
@@ -14656,7 +14637,7 @@ snapshots:
hasown: 2.0.2
is-core-module: 2.16.1
is-glob: 4.0.3
- minimatch: 3.1.2
+ minimatch: 3.1.3
object.fromentries: 2.0.8
object.groupby: 1.0.3
object.values: 1.2.1
@@ -14698,7 +14679,7 @@ snapshots:
'@humanwhocodes/module-importer': 1.0.1
'@humanwhocodes/retry': 0.4.3
'@types/estree': 1.0.8
- ajv: 6.12.6
+ ajv: 6.14.0
chalk: 4.1.2
cross-spawn: 7.0.6
debug: 4.4.3(supports-color@10.2.2)
@@ -14717,7 +14698,7 @@ snapshots:
is-glob: 4.0.3
json-stable-stringify-without-jsonify: 1.0.1
lodash.merge: 4.6.2
- minimatch: 3.1.2
+ minimatch: 3.1.3
natural-compare: 1.4.0
optionator: 0.9.4
optionalDependencies:
@@ -14727,8 +14708,8 @@ snapshots:
espree@10.4.0:
dependencies:
- acorn: 8.15.0
- acorn-jsx: 5.3.2(acorn@8.15.0)
+ acorn: 8.16.0
+ acorn-jsx: 5.3.2(acorn@8.16.0)
eslint-visitor-keys: 4.2.1
esprima@4.0.1: {}
@@ -14875,7 +14856,7 @@ snapshots:
extract-zip@2.0.1:
dependencies:
- debug: 4.3.4
+ debug: 4.4.3(supports-color@10.2.2)
get-stream: 5.2.0
yauzl: 2.10.0
optionalDependencies:
@@ -14940,9 +14921,9 @@ snapshots:
dependencies:
flat-cache: 4.0.1
- filelist@1.0.4:
+ filelist@1.0.5:
dependencies:
- minimatch: 5.1.6
+ minimatch: 10.2.3
fill-range@7.1.1:
dependencies:
@@ -15016,35 +14997,35 @@ snapshots:
locate-path: 6.0.0
path-exists: 4.0.0
- firebase@12.8.0:
+ firebase@12.9.0:
dependencies:
- '@firebase/ai': 2.7.0(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)
- '@firebase/analytics': 0.10.19(@firebase/app@0.14.7)
- '@firebase/analytics-compat': 0.2.25(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)
- '@firebase/app': 0.14.7
- '@firebase/app-check': 0.11.0(@firebase/app@0.14.7)
- '@firebase/app-check-compat': 0.4.0(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)
- '@firebase/app-compat': 0.5.7
+ '@firebase/ai': 2.8.0(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)
+ '@firebase/analytics': 0.10.19(@firebase/app@0.14.8)
+ '@firebase/analytics-compat': 0.2.25(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)
+ '@firebase/app': 0.14.8
+ '@firebase/app-check': 0.11.0(@firebase/app@0.14.8)
+ '@firebase/app-check-compat': 0.4.0(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)
+ '@firebase/app-compat': 0.5.8
'@firebase/app-types': 0.9.3
- '@firebase/auth': 1.12.0(@firebase/app@0.14.7)
- '@firebase/auth-compat': 0.6.2(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)
- '@firebase/data-connect': 0.3.12(@firebase/app@0.14.7)
+ '@firebase/auth': 1.12.0(@firebase/app@0.14.8)
+ '@firebase/auth-compat': 0.6.2(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)
+ '@firebase/data-connect': 0.3.12(@firebase/app@0.14.8)
'@firebase/database': 1.1.0
'@firebase/database-compat': 2.1.0
- '@firebase/firestore': 4.10.0(@firebase/app@0.14.7)
- '@firebase/firestore-compat': 0.4.4(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)
- '@firebase/functions': 0.13.1(@firebase/app@0.14.7)
- '@firebase/functions-compat': 0.4.1(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)
- '@firebase/installations': 0.6.19(@firebase/app@0.14.7)
- '@firebase/installations-compat': 0.2.19(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)
- '@firebase/messaging': 0.12.23(@firebase/app@0.14.7)
- '@firebase/messaging-compat': 0.2.23(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)
- '@firebase/performance': 0.7.9(@firebase/app@0.14.7)
- '@firebase/performance-compat': 0.2.22(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)
- '@firebase/remote-config': 0.8.0(@firebase/app@0.14.7)
- '@firebase/remote-config-compat': 0.2.21(@firebase/app-compat@0.5.7)(@firebase/app@0.14.7)
- '@firebase/storage': 0.14.0(@firebase/app@0.14.7)
- '@firebase/storage-compat': 0.4.0(@firebase/app-compat@0.5.7)(@firebase/app-types@0.9.3)(@firebase/app@0.14.7)
+ '@firebase/firestore': 4.11.0(@firebase/app@0.14.8)
+ '@firebase/firestore-compat': 0.4.5(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)
+ '@firebase/functions': 0.13.1(@firebase/app@0.14.8)
+ '@firebase/functions-compat': 0.4.1(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)
+ '@firebase/installations': 0.6.19(@firebase/app@0.14.8)
+ '@firebase/installations-compat': 0.2.19(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)
+ '@firebase/messaging': 0.12.23(@firebase/app@0.14.8)
+ '@firebase/messaging-compat': 0.2.23(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)
+ '@firebase/performance': 0.7.9(@firebase/app@0.14.8)
+ '@firebase/performance-compat': 0.2.22(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)
+ '@firebase/remote-config': 0.8.0(@firebase/app@0.14.8)
+ '@firebase/remote-config-compat': 0.2.21(@firebase/app-compat@0.5.8)(@firebase/app@0.14.8)
+ '@firebase/storage': 0.14.0(@firebase/app@0.14.8)
+ '@firebase/storage-compat': 0.4.0(@firebase/app-compat@0.5.8)(@firebase/app-types@0.9.3)(@firebase/app@0.14.8)
'@firebase/util': 1.13.0
transitivePeerDependencies:
- '@react-native-async-storage/async-storage'
@@ -15124,7 +15105,7 @@ snapshots:
fs-minipass@3.0.3:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
fs.realpath@1.0.0: {}
@@ -15167,7 +15148,7 @@ snapshots:
get-caller-file@2.0.5: {}
- get-east-asian-width@1.4.0: {}
+ get-east-asian-width@1.5.0: {}
get-intrinsic@1.3.0:
dependencies:
@@ -15207,7 +15188,7 @@ snapshots:
get-uri@6.0.5:
dependencies:
- basic-ftp: 5.1.0
+ basic-ftp: 5.2.0
data-uri-to-buffer: 6.0.2
debug: 4.4.3(supports-color@10.2.2)
transitivePeerDependencies:
@@ -15243,23 +15224,23 @@ snapshots:
dependencies:
foreground-child: 3.3.1
jackspeak: 3.4.3
- minimatch: 9.0.5
- minipass: 7.1.2
+ minimatch: 9.0.6
+ minipass: 7.1.3
package-json-from-dist: 1.0.1
path-scurry: 1.11.1
- glob@13.0.3:
+ glob@13.0.6:
dependencies:
- minimatch: 10.2.0
- minipass: 7.1.2
- path-scurry: 2.0.1
+ minimatch: 10.2.3
+ minipass: 7.1.3
+ path-scurry: 2.0.2
glob@7.2.3:
dependencies:
fs.realpath: 1.0.0
inflight: 1.0.6
inherits: 2.0.4
- minimatch: 3.1.2
+ minimatch: 3.1.3
once: 1.4.0
path-is-absolute: 1.0.1
@@ -15339,12 +15320,12 @@ snapshots:
graceful-fs@4.2.11: {}
- graphql-tag@2.12.6(graphql@16.12.0):
+ graphql-tag@2.12.6(graphql@16.13.0):
dependencies:
- graphql: 16.12.0
+ graphql: 16.13.0
tslib: 2.8.1
- graphql@16.12.0: {}
+ graphql@16.13.0: {}
grpc-gcp@1.0.1(protobufjs@7.5.4):
dependencies:
@@ -15382,7 +15363,7 @@ snapshots:
har-validator@5.1.5:
dependencies:
- ajv: 6.12.6
+ ajv: 6.14.0
har-schema: 2.0.0
has-ansi@2.0.0:
@@ -15411,7 +15392,7 @@ snapshots:
dependencies:
function-bind: 1.1.2
- hono@4.11.9: {}
+ hono@4.12.2: {}
hosted-git-info@9.0.2:
dependencies:
@@ -15597,7 +15578,7 @@ snapshots:
ignore-walk@8.0.0:
dependencies:
- minimatch: 10.2.0
+ minimatch: 10.2.3
ignore@5.3.2: {}
@@ -15723,7 +15704,7 @@ snapshots:
is-fullwidth-code-point@5.1.0:
dependencies:
- get-east-asian-width: 1.4.0
+ get-east-asian-width: 1.5.0
is-generator-function@1.1.2:
dependencies:
@@ -15873,8 +15854,6 @@ snapshots:
isexe@2.0.0: {}
- isexe@3.1.5: {}
-
isexe@4.0.0: {}
isobject@3.0.1: {}
@@ -15928,14 +15907,10 @@ snapshots:
optionalDependencies:
'@pkgjs/parseargs': 0.11.0
- jackspeak@4.2.3:
- dependencies:
- '@isaacs/cliui': 9.0.0
-
jake@10.9.4:
dependencies:
async: 3.2.6
- filelist: 1.0.4
+ filelist: 1.0.5
picocolors: 1.1.1
jasmine-core@2.8.0: {}
@@ -15944,7 +15919,7 @@ snapshots:
jasmine-core@5.13.0: {}
- jasmine-core@6.0.0: {}
+ jasmine-core@6.1.0: {}
jasmine-reporters@2.5.2:
dependencies:
@@ -15966,11 +15941,11 @@ snapshots:
glob: 10.5.0
jasmine-core: 5.13.0
- jasmine@6.0.0:
+ jasmine@6.1.0:
dependencies:
'@jasminejs/reporters': 1.0.0
- glob: 13.0.3
- jasmine-core: 6.0.0
+ glob: 13.0.6
+ jasmine-core: 6.1.0
jasminewd2@2.2.0: {}
@@ -16002,7 +15977,7 @@ snapshots:
'@asamuzakjp/dom-selector': 6.8.1
'@bramus/specificity': 2.4.2
'@exodus/bytes': 1.14.1
- cssstyle: 6.0.1
+ cssstyle: 6.1.0
data-urls: 7.0.0
decimal.js: 10.6.0
html-encoding-sniffer: 6.0.0
@@ -16017,7 +15992,7 @@ snapshots:
w3c-xmlserializer: 5.0.0
webidl-conversions: 8.0.1
whatwg-mimetype: 5.0.0
- whatwg-url: 16.0.0
+ whatwg-url: 16.0.1
xml-name-validator: 5.0.0
transitivePeerDependencies:
- '@noble/hashes'
@@ -16047,6 +16022,8 @@ snapshots:
json-stringify-safe@5.0.1: {}
+ json-with-bigint@3.5.3: {}
+
json5@1.0.2:
dependencies:
minimist: 1.2.8
@@ -16121,7 +16098,7 @@ snapshots:
istanbul-lib-report: 3.0.1
istanbul-lib-source-maps: 4.0.1
istanbul-reports: 3.2.0
- minimatch: 3.1.2
+ minimatch: 3.1.3
transitivePeerDependencies:
- supports-color
@@ -16156,7 +16133,7 @@ snapshots:
lodash: 4.17.23
log4js: 6.9.1
mime: 2.6.0
- minimatch: 3.1.2
+ minimatch: 3.1.3
mkdirp: 0.5.6
qjobs: 1.2.0
range-parser: 1.2.1
@@ -16236,7 +16213,7 @@ snapshots:
transitivePeerDependencies:
- supports-color
- launch-editor@2.12.0:
+ launch-editor@2.13.0:
dependencies:
picocolors: 1.1.1
shell-quote: 1.8.3
@@ -16445,8 +16422,8 @@ snapshots:
'@npmcli/agent': 4.0.0
cacache: 20.0.3
http-cache-semantics: 4.2.0
- minipass: 7.1.2
- minipass-fetch: 5.0.1
+ minipass: 7.1.3
+ minipass-fetch: 5.0.2
minipass-flush: 1.0.5
minipass-pipeline: 1.2.4
negotiator: 1.0.0
@@ -16534,43 +16511,35 @@ snapshots:
minimalistic-assert@1.0.1: {}
- minimatch@10.1.1:
+ minimatch@10.2.3:
dependencies:
- '@isaacs/brace-expansion': 5.0.1
+ brace-expansion: 5.0.3
- minimatch@10.2.0:
- dependencies:
- brace-expansion: 5.0.2
-
- minimatch@3.1.2:
+ minimatch@3.1.3:
dependencies:
brace-expansion: 1.1.12
- minimatch@5.1.6:
- dependencies:
- brace-expansion: 2.0.2
-
minimatch@7.4.6:
dependencies:
brace-expansion: 2.0.2
- minimatch@9.0.5:
+ minimatch@9.0.6:
dependencies:
- brace-expansion: 2.0.2
+ brace-expansion: 5.0.3
minimist@1.2.8: {}
minipass-collect@2.0.1:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
- minipass-fetch@5.0.1:
+ minipass-fetch@5.0.2:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
minipass-sized: 2.0.0
minizlib: 3.1.0
optionalDependencies:
- encoding: 0.1.13
+ iconv-lite: 0.7.2
minipass-flush@1.0.5:
dependencies:
@@ -16582,17 +16551,17 @@ snapshots:
minipass-sized@2.0.0:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
minipass@3.3.6:
dependencies:
yallist: 4.0.0
- minipass@7.1.2: {}
+ minipass@7.1.3: {}
minizlib@3.1.0:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
mitt@1.2.0: {}
@@ -16636,11 +16605,11 @@ snapshots:
dns-packet: 5.6.1
thunky: 1.1.0
- multimatch@7.0.0:
+ multimatch@8.0.0:
dependencies:
array-differ: 4.0.0
array-union: 3.0.1
- minimatch: 9.0.5
+ minimatch: 10.2.3
mute-stream@2.0.0: {}
@@ -16668,12 +16637,12 @@ snapshots:
netmask@2.0.2: {}
- ng-packagr@21.2.0-next.0(@angular/compiler-cli@21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3))(tslib@2.8.1)(typescript@5.9.3):
+ ng-packagr@21.2.0(@angular/compiler-cli@21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3))(tslib@2.8.1)(typescript@5.9.3):
dependencies:
'@ampproject/remapping': 2.3.0
- '@angular/compiler-cli': 21.2.0-next.3(@angular/compiler@21.2.0-next.3)(typescript@5.9.3)
+ '@angular/compiler-cli': 21.2.0(@angular/compiler@21.2.0)(typescript@5.9.3)
'@rollup/plugin-json': 6.1.0(rollup@4.57.1)
- '@rollup/wasm-node': 4.57.1
+ '@rollup/wasm-node': 4.59.0
ajv: 8.18.0
ansi-colors: 4.1.3
browserslist: 4.28.1
@@ -16697,9 +16666,9 @@ snapshots:
optionalDependencies:
rollup: 4.57.1
- nock@14.0.10:
+ nock@14.0.11:
dependencies:
- '@mswjs/interceptors': 0.39.8
+ '@mswjs/interceptors': 0.41.3
json-stringify-safe: 5.0.1
propagate: 2.0.1
@@ -16780,7 +16749,7 @@ snapshots:
semver: 7.7.4
validate-npm-package-name: 7.0.2
- npm-packlist@10.0.3:
+ npm-packlist@10.0.4:
dependencies:
ignore-walk: 8.0.0
proc-log: 6.1.0
@@ -16797,8 +16766,8 @@ snapshots:
'@npmcli/redact': 4.0.0
jsonparse: 1.3.1
make-fetch-happen: 15.0.3
- minipass: 7.1.2
- minipass-fetch: 5.0.1
+ minipass: 7.1.3
+ minipass-fetch: 5.0.2
minizlib: 3.1.0
npm-package-arg: 13.0.2
proc-log: 6.1.0
@@ -16926,7 +16895,7 @@ snapshots:
is-unicode-supported: 2.1.0
log-symbols: 7.0.1
stdin-discarder: 0.3.1
- string-width: 8.1.1
+ string-width: 8.2.0
ordered-binary@1.6.1:
optional: true
@@ -16972,6 +16941,11 @@ snapshots:
eventemitter3: 4.0.7
p-timeout: 3.2.0
+ p-retry@4.6.2:
+ dependencies:
+ '@types/retry': 0.12.0
+ retry: 0.13.1
+
p-retry@6.2.1:
dependencies:
'@types/retry': 0.12.2
@@ -17008,14 +16982,14 @@ snapshots:
dependencies:
'@npmcli/git': 7.0.1
'@npmcli/installed-package-contents': 4.0.0
- '@npmcli/package-json': 7.0.4
+ '@npmcli/package-json': 7.0.5
'@npmcli/promise-spawn': 9.0.1
'@npmcli/run-script': 10.0.3
cacache: 20.0.3
fs-minipass: 3.0.3
- minipass: 7.1.2
+ minipass: 7.1.3
npm-package-arg: 13.0.2
- npm-packlist: 10.0.3
+ npm-packlist: 10.0.4
npm-pick-manifest: 11.0.3
npm-registry-fetch: 19.1.1
proc-log: 6.1.0
@@ -17074,12 +17048,12 @@ snapshots:
path-scurry@1.11.1:
dependencies:
lru-cache: 10.4.3
- minipass: 7.1.2
+ minipass: 7.1.3
- path-scurry@2.0.1:
+ path-scurry@2.0.2:
dependencies:
lru-cache: 11.2.6
- minipass: 7.1.2
+ minipass: 7.1.3
path-to-regexp@0.1.12: {}
@@ -17360,9 +17334,9 @@ snapshots:
- supports-color
- utf-8-validate
- puppeteer-core@24.37.3(bufferutil@4.1.0):
+ puppeteer-core@24.37.5(bufferutil@4.1.0):
dependencies:
- '@puppeteer/browsers': 2.12.1
+ '@puppeteer/browsers': 2.13.0
chromium-bidi: 14.0.0(devtools-protocol@0.0.1566079)
debug: 4.4.3(supports-color@10.2.2)
devtools-protocol: 0.0.1566079
@@ -17604,7 +17578,7 @@ snapshots:
resp-modifier@6.0.2:
dependencies:
debug: 2.6.9
- minimatch: 3.1.2
+ minimatch: 3.1.3
transitivePeerDependencies:
- supports-color
@@ -18091,23 +18065,23 @@ snapshots:
source-map@0.7.6: {}
- spdx-correct@3.2.0:
- dependencies:
- spdx-expression-parse: 3.0.1
- spdx-license-ids: 3.0.22
-
spdx-exceptions@2.5.0: {}
spdx-expression-parse@3.0.1:
dependencies:
spdx-exceptions: 2.5.0
- spdx-license-ids: 3.0.22
+ spdx-license-ids: 3.0.23
+
+ spdx-expression-parse@4.0.0:
+ dependencies:
+ spdx-exceptions: 2.5.0
+ spdx-license-ids: 3.0.23
spdx-expression-validate@2.0.0:
dependencies:
spdx-expression-parse: 3.0.1
- spdx-license-ids@3.0.22: {}
+ spdx-license-ids@3.0.23: {}
spdy-transport@3.0.0:
dependencies:
@@ -18155,11 +18129,11 @@ snapshots:
ssri@10.0.5:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
ssri@13.0.1:
dependencies:
- minipass: 7.1.2
+ minipass: 7.1.3
stack-trace@0.0.10: {}
@@ -18209,7 +18183,7 @@ snapshots:
dependencies:
events-universal: 1.0.1
fast-fifo: 1.3.2
- text-decoder: 1.2.6
+ text-decoder: 1.2.7
transitivePeerDependencies:
- bare-abort-controller
- react-native-b4a
@@ -18231,12 +18205,12 @@ snapshots:
string-width@7.2.0:
dependencies:
emoji-regex: 10.6.0
- get-east-asian-width: 1.4.0
+ get-east-asian-width: 1.5.0
strip-ansi: 7.1.2
- string-width@8.1.1:
+ string-width@8.2.0:
dependencies:
- get-east-asian-width: 1.4.0
+ get-east-asian-width: 1.5.0
strip-ansi: 7.1.2
string.prototype.trim@1.2.10:
@@ -18342,7 +18316,7 @@ snapshots:
tar-stream@3.1.7:
dependencies:
- b4a: 1.7.4
+ b4a: 1.8.0
fast-fifo: 1.3.2
streamx: 2.23.0
transitivePeerDependencies:
@@ -18353,7 +18327,7 @@ snapshots:
dependencies:
'@isaacs/fs-minipass': 4.0.1
chownr: 3.0.0
- minipass: 7.1.2
+ minipass: 7.1.3
minizlib: 3.1.0
yallist: 5.0.0
@@ -18366,6 +18340,14 @@ snapshots:
transitivePeerDependencies:
- supports-color
+ teex@1.0.1:
+ dependencies:
+ streamx: 2.23.0
+ transitivePeerDependencies:
+ - bare-abort-controller
+ - react-native-b4a
+ optional: true
+
terser-webpack-plugin@5.3.16(esbuild@0.27.3)(webpack@5.105.2(esbuild@0.27.3)):
dependencies:
'@jridgewell/trace-mapping': 0.3.31
@@ -18380,13 +18362,13 @@ snapshots:
terser@5.46.0:
dependencies:
'@jridgewell/source-map': 0.3.11
- acorn: 8.15.0
+ acorn: 8.16.0
commander: 2.20.3
source-map-support: 0.5.21
- text-decoder@1.2.6:
+ text-decoder@1.2.7:
dependencies:
- b4a: 1.7.4
+ b4a: 1.8.0
transitivePeerDependencies:
- react-native-b4a
@@ -18491,8 +18473,8 @@ snapshots:
'@tsconfig/node14': 1.0.3
'@tsconfig/node16': 1.0.4
'@types/node': 22.19.11
- acorn: 8.15.0
- acorn-walk: 8.3.4
+ acorn: 8.16.0
+ acorn-walk: 8.3.5
arg: 4.1.3
create-require: 1.1.1
diff: 4.0.4
@@ -18709,11 +18691,6 @@ snapshots:
'@types/istanbul-lib-coverage': 2.0.6
convert-source-map: 2.0.0
- validate-npm-package-license@3.0.4:
- dependencies:
- spdx-correct: 3.2.0
- spdx-expression-parse: 3.0.1
-
validate-npm-package-name@7.0.2: {}
validator@13.15.26: {}
@@ -18794,7 +18771,7 @@ snapshots:
core-util-is: 1.0.2
extsprintf: 1.3.0
- vite@7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2):
+ vite@7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2):
dependencies:
esbuild: 0.27.3
fdir: 6.5.0(picomatch@4.0.3)
@@ -18803,7 +18780,7 @@ snapshots:
rollup: 4.57.1
tinyglobby: 0.2.15
optionalDependencies:
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
fsevents: 2.3.3
jiti: 2.6.1
less: 4.4.2
@@ -18812,10 +18789,10 @@ snapshots:
tsx: 4.21.0
yaml: 2.8.2
- vitest@4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.9)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2):
+ vitest@4.0.18(@opentelemetry/api@1.9.0)(@types/node@24.10.13)(jiti@2.6.1)(jsdom@28.1.0)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2):
dependencies:
'@vitest/expect': 4.0.18
- '@vitest/mocker': 4.0.18(vite@7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))
+ '@vitest/mocker': 4.0.18(vite@7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2))
'@vitest/pretty-format': 4.0.18
'@vitest/runner': 4.0.18
'@vitest/snapshot': 4.0.18
@@ -18832,11 +18809,11 @@ snapshots:
tinyexec: 1.0.2
tinyglobby: 0.2.15
tinyrainbow: 3.0.3
- vite: 7.3.1(@types/node@24.10.9)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
+ vite: 7.3.1(@types/node@24.10.13)(jiti@2.6.1)(less@4.4.2)(sass@1.97.3)(terser@5.46.0)(tsx@4.21.0)(yaml@2.8.2)
why-is-node-running: 2.3.0
optionalDependencies:
'@opentelemetry/api': 1.9.0
- '@types/node': 24.10.9
+ '@types/node': 24.10.13
jsdom: 28.1.0
transitivePeerDependencies:
- jiti
@@ -18933,7 +18910,7 @@ snapshots:
graceful-fs: 4.2.11
http-proxy-middleware: 2.0.9(@types/express@4.17.25)
ipaddr.js: 2.3.0
- launch-editor: 2.12.0
+ launch-editor: 2.13.0
open: 10.2.0
p-retry: 6.2.1
schema-utils: 4.3.3
@@ -18973,8 +18950,8 @@ snapshots:
'@webassemblyjs/ast': 1.14.1
'@webassemblyjs/wasm-edit': 1.14.1
'@webassemblyjs/wasm-parser': 1.14.1
- acorn: 8.15.0
- acorn-import-phases: 1.0.4(acorn@8.15.0)
+ acorn: 8.16.0
+ acorn-import-phases: 1.0.4(acorn@8.16.0)
browserslist: 4.28.1
chrome-trace-event: 1.0.4
enhanced-resolve: 5.19.0
@@ -19012,7 +18989,7 @@ snapshots:
tr46: 5.1.1
webidl-conversions: 7.0.0
- whatwg-url@16.0.0:
+ whatwg-url@16.0.1:
dependencies:
'@exodus/bytes': 1.14.1
tr46: 6.0.0
@@ -19076,10 +19053,6 @@ snapshots:
dependencies:
isexe: 2.0.0
- which@6.0.0:
- dependencies:
- isexe: 3.1.5
-
which@6.0.1:
dependencies:
isexe: 4.0.0
@@ -19256,4 +19229,4 @@ snapshots:
zod@4.3.6: {}
- zone.js@0.16.0: {}
+ zone.js@0.16.1: {}
diff --git a/tests/e2e/assets/ssr-project-webpack/package.json b/tests/e2e/assets/ssr-project-webpack/package.json
index 9d6b1f7338e8..0482ad3897b0 100644
--- a/tests/e2e/assets/ssr-project-webpack/package.json
+++ b/tests/e2e/assets/ssr-project-webpack/package.json
@@ -14,14 +14,14 @@
},
"private": true,
"dependencies": {
- "@angular/animations": "^21.0.0-next.0",
- "@angular/common": "^21.0.0-next.0",
- "@angular/compiler": "^21.0.0-next.0",
- "@angular/core": "^21.0.0-next.0",
- "@angular/forms": "^21.0.0-next.0",
- "@angular/platform-browser": "^21.0.0-next.0",
- "@angular/platform-server": "^21.0.0-next.0",
- "@angular/router": "^21.0.0-next.0",
+ "@angular/animations": "^21.0.0",
+ "@angular/common": "^21.0.0",
+ "@angular/compiler": "^21.0.0",
+ "@angular/core": "^21.0.0",
+ "@angular/forms": "^21.0.0",
+ "@angular/platform-browser": "^21.0.0",
+ "@angular/platform-server": "^21.0.0",
+ "@angular/router": "^21.0.0",
"@angular/ssr": "^21.0.0-next.0",
"express": "^4.18.2",
"rxjs": "~7.8.0",
diff --git a/tests/e2e/tests/build/server-rendering/express-engine-csp-nonce.ts b/tests/e2e/tests/build/server-rendering/express-engine-csp-nonce.ts
index 3d6335b48465..409b141bb48a 100644
--- a/tests/e2e/tests/build/server-rendering/express-engine-csp-nonce.ts
+++ b/tests/e2e/tests/build/server-rendering/express-engine-csp-nonce.ts
@@ -61,6 +61,7 @@ export default async function () {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/express-engine-ngmodule.ts b/tests/e2e/tests/build/server-rendering/express-engine-ngmodule.ts
index 92e34f7ca3e8..5eaaa9b0e58b 100644
--- a/tests/e2e/tests/build/server-rendering/express-engine-ngmodule.ts
+++ b/tests/e2e/tests/build/server-rendering/express-engine-ngmodule.ts
@@ -83,6 +83,7 @@ export default async function () {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/express-engine-standalone.ts b/tests/e2e/tests/build/server-rendering/express-engine-standalone.ts
index 18920e3a5893..4c8d29d9b770 100644
--- a/tests/e2e/tests/build/server-rendering/express-engine-standalone.ts
+++ b/tests/e2e/tests/build/server-rendering/express-engine-standalone.ts
@@ -50,6 +50,7 @@ export default async function () {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-base-href.ts b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-base-href.ts
index 6d0a45459a16..aa6f7e3426ee 100644
--- a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-base-href.ts
+++ b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-base-href.ts
@@ -111,6 +111,7 @@ async function spawnServer(): Promise {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-sub-path.ts b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-sub-path.ts
index 79fc755c4477..7bc323311b4f 100644
--- a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-sub-path.ts
+++ b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n-sub-path.ts
@@ -146,6 +146,7 @@ async function spawnServer(): Promise {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n.ts b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n.ts
index 994d77343d1e..efbff9871bbc 100644
--- a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n.ts
+++ b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-i18n.ts
@@ -122,6 +122,7 @@ async function spawnServer(): Promise {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-platform-neutral.ts b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-platform-neutral.ts
index 130ade10ba9f..6fdd1998cdd2 100644
--- a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-platform-neutral.ts
+++ b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server-platform-neutral.ts
@@ -11,7 +11,6 @@ import {
import { updateJsonFile, useSha } from '../../../utils/project';
import { getGlobalVariable } from '../../../utils/env';
import { findFreePort } from '../../../utils/network';
-import { readFile } from 'node:fs/promises';
export default async function () {
assert(
@@ -98,6 +97,8 @@ export default async function () {
const options = buildTarget['options'];
options['ssr']['experimentalPlatform'] = 'neutral';
options['outputMode'] = 'server';
+ options['security'] ??= {};
+ options['security']['allowedHosts'] = ['localhost'];
});
await noSilentNg('build');
diff --git a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server.ts b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server.ts
index 5205d20eeb0a..ff72cb8e8df2 100644
--- a/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server.ts
+++ b/tests/e2e/tests/build/server-rendering/server-routes-output-mode-server.ts
@@ -206,6 +206,7 @@ async function spawnServer(): Promise {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);
diff --git a/tests/e2e/tests/build/server-rendering/server-routes-preload-links.ts b/tests/e2e/tests/build/server-rendering/server-routes-preload-links.ts
index f1437392492d..fe316e3cd157 100644
--- a/tests/e2e/tests/build/server-rendering/server-routes-preload-links.ts
+++ b/tests/e2e/tests/build/server-rendering/server-routes-preload-links.ts
@@ -196,6 +196,7 @@ async function spawnServer(): Promise {
{
...process.env,
'PORT': String(port),
+ 'NG_ALLOWED_HOSTS': 'localhost',
},
);