Compare commits
22 Commits
codex/inst
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
aaf0df956f | ||
|
|
127687134a | ||
|
|
b4bf554c43 | ||
|
|
8bbe9a4532 | ||
|
|
926afe32ab | ||
|
|
e447e9c534 | ||
|
|
4955f7ffb8 | ||
|
|
886c9ac60a | ||
|
|
f3a37af6dd | ||
|
|
0012d40688 | ||
|
|
983d42f9d6 | ||
|
|
8d8043c587 | ||
|
|
b31cbea7a0 | ||
|
|
0f6d639711 | ||
|
|
c38308dee0 | ||
|
|
7f96cf49f8 | ||
|
|
48a1c6db5a | ||
|
|
b81a86617a | ||
|
|
eaec92c010 | ||
|
|
5cd0ffdecd | ||
|
|
1c571d990a | ||
|
|
061e028153 |
11
.github/workflows/deploy.yml
vendored
11
.github/workflows/deploy.yml
vendored
@ -19,8 +19,13 @@ jobs:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
||||
|
||||
- name: Setup Node
|
||||
uses: actions/setup-node@48b55a011bda9f5d6aeb4c2d9c7362e8dae4041e # v6.4.0
|
||||
with:
|
||||
node-version: 24
|
||||
|
||||
- name: Setup Bun
|
||||
uses: oven-sh/setup-bun@ecf28ddc73e819eb6fa29df6b34ef8921c743461 # v2.1.3
|
||||
uses: oven-sh/setup-bun@0c5077e51419868618aeaa5fe8019c62421857d6 # v2.2.0
|
||||
with:
|
||||
bun-version: latest
|
||||
|
||||
@ -31,7 +36,7 @@ jobs:
|
||||
run: bun run build
|
||||
|
||||
- name: Upload artifact
|
||||
uses: actions/upload-pages-artifact@7b1f4a764d45c48632c6b24a0339c27f5614fb0b # v4.0.0
|
||||
uses: actions/upload-pages-artifact@fc324d3547104276b827a68afc52ff2a11cc49c9 # v5.0.0
|
||||
with:
|
||||
path: ./dist
|
||||
|
||||
@ -44,4 +49,4 @@ jobs:
|
||||
steps:
|
||||
- name: Deploy to GitHub Pages
|
||||
id: deployment
|
||||
uses: actions/deploy-pages@d6db90164ac5ed86f2b6aed7e0febac5b3c0c03e # v4.0.5
|
||||
uses: actions/deploy-pages@cd2ce8fcbc39b97be8ca5fce6e763baed58fa128 # v5.0.0
|
||||
|
||||
2
.github/workflows/install-smoke.yml
vendored
2
.github/workflows/install-smoke.yml
vendored
@ -135,7 +135,7 @@ jobs:
|
||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@53b83947a5a98c8d113130e565377fae1a50d02f # v6.3.0
|
||||
uses: actions/setup-node@48b55a011bda9f5d6aeb4c2d9c7362e8dae4041e # v6.4.0
|
||||
with:
|
||||
node-version: ${{ matrix.node }}
|
||||
|
||||
|
||||
10
CHANGELOG.md
10
CHANGELOG.md
@ -1,5 +1,15 @@
|
||||
# Changelog
|
||||
|
||||
## 2026-04-26
|
||||
|
||||
- Installer: normalize `HOME` when headless VM execution reports `/`, preventing npm from trying to use `/.npm` on macOS Parallels guests.
|
||||
- Installer: keep `install-cli.sh` package installs under the prefix-local Node toolchain so `openclaw update` does not create a second package root (#118, thanks @AISymbiote).
|
||||
- Integrations: add QQ Bot to the chat provider list with the canonical docs link (#119, thanks @sliverp).
|
||||
- Integrations: point Notion, Bear Notes, GitHub, Image Gen, and Camera cards to their specific ClawHub skill pages (#117, thanks @DJStompZone).
|
||||
- Dependencies: update Astro to 6.1.9, migrate blog content to Astro 6 content collections, refresh RSS/icons/analytics packages, and bump pinned GitHub Actions.
|
||||
- Windows installer: route PowerShell install failures through a top-level handler so `irm ... | iex` returns control to the current shell while direct script-file runs still exit non-zero. Fixes openclaw/openclaw#38054, thanks @PwrSrg.
|
||||
- Installer: warn when multiple npm global roots contain OpenClaw installs, showing active Node/npm/openclaw plus each install path and version so stale version-manager installs are visible. Fixes openclaw/openclaw#40839, thanks @zhixianio.
|
||||
|
||||
## 2026-03-16
|
||||
|
||||
- Integrations: correct MS Teams docs link to the canonical `/channels/msteams` path (#109, thanks @SidU).
|
||||
|
||||
@ -1,229 +0,0 @@
|
||||
# Code Optimizations Summary
|
||||
|
||||
This document summarizes the technical improvements made to the OpenClaw landing page codebase.
|
||||
|
||||
## Changes Made
|
||||
|
||||
### ✅ 1. Fixed Deprecated API Usage
|
||||
**File**: `src/pages/index.astro` (Line ~320)
|
||||
|
||||
**Before**:
|
||||
```javascript
|
||||
const isWindows = navigator.platform.toLowerCase().includes('win') ||
|
||||
navigator.userAgent.toLowerCase().includes('windows');
|
||||
```
|
||||
|
||||
**After**:
|
||||
```javascript
|
||||
const isWindows = navigator.userAgentData?.platform === 'Windows' ||
|
||||
navigator.userAgent.toLowerCase().includes('windows');
|
||||
```
|
||||
|
||||
**Impact**: `navigator.platform` is deprecated and will be removed from browsers. The new code uses the modern `navigator.userAgentData` API with a fallback.
|
||||
|
||||
---
|
||||
|
||||
### ✅ 2. Removed Dead Code
|
||||
**File**: `src/pages/index.astro` (Lines 297-309)
|
||||
|
||||
**Removed**:
|
||||
- `installCmds` object (unused, duplicated in `copyCommands`)
|
||||
- `osCmds` object (unused)
|
||||
|
||||
**Impact**: Reduced bundle size by ~150 bytes and improved code clarity.
|
||||
|
||||
---
|
||||
|
||||
### ✅ 3. Cached DOM Queries for Performance
|
||||
**File**: `src/pages/index.astro` (Lines 349-353)
|
||||
|
||||
**Added**:
|
||||
```javascript
|
||||
// Cached query selectors for frequently updated elements
|
||||
const pmCmdElements = document.querySelectorAll('.pm-cmd');
|
||||
const pmInstallElements = document.querySelectorAll('.pm-install');
|
||||
const osCmdElements = document.querySelectorAll('.os-cmd');
|
||||
const osCmdHackableElements = document.querySelectorAll('.os-cmd-hackable');
|
||||
```
|
||||
|
||||
**Before** (in `updateCommands` function):
|
||||
```javascript
|
||||
document.querySelectorAll('.pm-cmd').forEach(...); // Called on every update
|
||||
document.querySelectorAll('.pm-install').forEach(...);
|
||||
document.querySelectorAll('.os-cmd').forEach(...);
|
||||
document.querySelectorAll('.os-cmd-hackable').forEach(...);
|
||||
```
|
||||
|
||||
**After**:
|
||||
```javascript
|
||||
pmCmdElements.forEach(...); // Uses cached reference
|
||||
pmInstallElements.forEach(...);
|
||||
osCmdElements.forEach(...);
|
||||
osCmdHackableElements.forEach(...);
|
||||
```
|
||||
|
||||
**Impact**: Eliminated 4 repeated DOM queries per state update. Improved performance, especially on slower devices.
|
||||
|
||||
---
|
||||
|
||||
### ✅ 4. Added Null-Safe Operations
|
||||
**Files**: `src/pages/index.astro`
|
||||
|
||||
**Changed**: All DOM operations now include null checks using `if` statements:
|
||||
|
||||
```javascript
|
||||
// Before
|
||||
osDetected.textContent = osLabels[currentOs];
|
||||
|
||||
// After
|
||||
if (osDetected) osDetected.textContent = osLabels[currentOs];
|
||||
```
|
||||
|
||||
**Affected functions**:
|
||||
- `updateCommands()` - 8 null checks added
|
||||
- `updateVisibility()` - 11 null checks added
|
||||
- Event listeners - 2 null checks added
|
||||
- Easter egg animation - 1 null check added
|
||||
|
||||
**Impact**: Prevents runtime crashes if HTML elements are missing or renamed. More resilient code.
|
||||
|
||||
---
|
||||
|
||||
### ✅ 5. Improved Clipboard Copy Handler
|
||||
**File**: `src/pages/index.astro` (Lines 531-578)
|
||||
|
||||
**Improvements**:
|
||||
1. **Added fallback to `execCommand`** for older browsers or non-HTTPS contexts
|
||||
2. **Added visual error feedback** - red flash on copy failure
|
||||
3. **Added null checks** for icon elements
|
||||
4. **Added validation** for command key existence
|
||||
|
||||
**Before**:
|
||||
```javascript
|
||||
try {
|
||||
await navigator.clipboard.writeText(code);
|
||||
// ... success handling
|
||||
} catch (err) {
|
||||
console.error('Failed to copy:', err); // Silent failure
|
||||
}
|
||||
```
|
||||
|
||||
**After**:
|
||||
```javascript
|
||||
let success = false;
|
||||
try {
|
||||
if (navigator.clipboard && navigator.clipboard.writeText) {
|
||||
await navigator.clipboard.writeText(code);
|
||||
success = true;
|
||||
} else {
|
||||
// Fallback using textarea + execCommand
|
||||
const textArea = document.createElement('textarea');
|
||||
textArea.value = code;
|
||||
textArea.style.position = 'fixed';
|
||||
textArea.style.opacity = '0';
|
||||
document.body.appendChild(textArea);
|
||||
textArea.select();
|
||||
success = document.execCommand('copy');
|
||||
document.body.removeChild(textArea);
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('Failed to copy:', err);
|
||||
success = false;
|
||||
}
|
||||
|
||||
if (success) {
|
||||
// Success feedback (green checkmark)
|
||||
} else {
|
||||
// Visual error feedback - brief red flash
|
||||
btn.style.background = 'rgba(239, 68, 68, 0.3)';
|
||||
setTimeout(() => {
|
||||
btn.style.background = '';
|
||||
}, 1000);
|
||||
}
|
||||
```
|
||||
|
||||
**Impact**: Copy works in more environments, users get visual feedback on failure.
|
||||
|
||||
---
|
||||
|
||||
### ✅ 6. Updated Font Loading Comment
|
||||
**File**: `src/layouts/Layout.astro` (Line 40)
|
||||
|
||||
**Changed**: Updated comment to clarify that `display=swap` is already implemented for performance.
|
||||
|
||||
**Note**: The `&display=swap` parameter was already present in the URL. No functional change, just documentation improvement.
|
||||
|
||||
---
|
||||
|
||||
### ✅ 7. Fixed Easter Egg Null Safety
|
||||
**File**: `src/pages/index.astro` (Lines 582-615)
|
||||
|
||||
**Before**:
|
||||
```javascript
|
||||
const lobsterIcon = document.querySelector('.lobster-icon');
|
||||
const tagline = document.getElementById('tagline');
|
||||
const originalTagline = tagline.textContent; // Could crash if null
|
||||
|
||||
lobsterIcon.addEventListener('mouseenter', () => { ... }); // Could crash if null
|
||||
```
|
||||
|
||||
**After**:
|
||||
```javascript
|
||||
const lobsterIcon = document.querySelector('.lobster-icon');
|
||||
const tagline = document.getElementById('tagline');
|
||||
|
||||
if (lobsterIcon && tagline) {
|
||||
const originalTagline = tagline.textContent;
|
||||
lobsterIcon.addEventListener('mouseenter', () => { ... });
|
||||
}
|
||||
```
|
||||
|
||||
**Impact**: Easter egg won't crash if elements are missing.
|
||||
|
||||
---
|
||||
|
||||
## Summary Statistics
|
||||
|
||||
| Metric | Before | After | Improvement |
|
||||
|--------|--------|-------|-------------|
|
||||
| Dead code lines | 13 | 0 | -13 lines |
|
||||
| DOM queries per update | 4 | 0 | 100% cached |
|
||||
| Null checks | 0 | ~23 | +∞ |
|
||||
| Clipboard fallback | ❌ | ✅ | Works in more contexts |
|
||||
| Deprecated APIs | 1 | 0 | Future-proof |
|
||||
| Visual error feedback | ❌ | ✅ | Better UX |
|
||||
|
||||
---
|
||||
|
||||
## Remaining Recommendations
|
||||
|
||||
### Lock File Cleanup (Not Implemented)
|
||||
The project currently has **3 lock files**:
|
||||
- `bun.lock` (86KB)
|
||||
- `package-lock.json` (188KB)
|
||||
- `pnpm-lock.yaml` (107KB)
|
||||
|
||||
**Recommendation**: Choose one package manager and remove the other lock files. Based on the README using `bun install`, keep `bun.lock` and delete/gitignore the others.
|
||||
|
||||
**Why**: Different contributors using different package managers can lead to dependency version mismatches.
|
||||
|
||||
---
|
||||
|
||||
## Files Modified
|
||||
|
||||
1. ✅ `src/pages/index.astro` - Main JavaScript improvements
|
||||
2. ✅ `src/layouts/Layout.astro` - Font loading comment
|
||||
|
||||
## Testing Recommendations
|
||||
|
||||
1. Test clipboard copy on different browsers (Chrome, Firefox, Safari)
|
||||
2. Test clipboard copy in HTTP vs HTTPS contexts
|
||||
3. Test with browser DevTools - delete elements and verify no console errors
|
||||
4. Test OS detection on Windows, macOS, Linux
|
||||
5. Test mode switching (One-liner, npm, Hackable, macOS)
|
||||
6. Test beta toggle functionality
|
||||
7. Hover over lobster icon to test Easter egg
|
||||
|
||||
---
|
||||
|
||||
**All critical technical flaws have been addressed.** The code is now more robust, performant, and future-proof.
|
||||
12
README.md
12
README.md
@ -48,14 +48,24 @@ The landing page hosts installer scripts:
|
||||
|
||||
Installer UI controls (macOS/Linux `install.sh`):
|
||||
- Gum UI is auto-detected; interactive terminals get richer status output, non-interactive shells fall back to plain output automatically.
|
||||
- Windows `install.ps1` keeps `irm ... | iex` failures in the current PowerShell session while preserving non-zero exits for direct script-file automation.
|
||||
|
||||
These scripts:
|
||||
1. Install Homebrew (macOS) or detect package managers (Windows)
|
||||
2. Install Node.js 22+ if needed
|
||||
3. Install openclaw globally via npm
|
||||
3. Install openclaw globally via npm, or from a pnpm-backed git checkout with `--install-method git`
|
||||
4. Run `openclaw doctor --non-interactive` for migrations (upgrades only)
|
||||
5. Prompt to run `openclaw onboard` (new installs)
|
||||
|
||||
Switching after install:
|
||||
- npm package to git checkout: `openclaw update --channel dev`
|
||||
- git checkout to npm package: `openclaw update --channel stable`
|
||||
- installer-forced switch: rerun the installer with `--install-method git` or `--install-method npm`
|
||||
|
||||
Source checkouts use the OpenClaw pnpm workspace. Keep hackable/dev-channel copy
|
||||
pointing at `pnpm install`; root `npm install` is for packaged installs, not
|
||||
source trees.
|
||||
|
||||
Troubleshooting:
|
||||
- macOS first-run Homebrew bootstrap needs an Administrator account. If install fails with a sudo/admin error, use an admin account (or add the current user to the `admin` group) and rerun the installer.
|
||||
|
||||
|
||||
278
bun.lock
278
bun.lock
@ -3,14 +3,14 @@
|
||||
"configVersion": 0,
|
||||
"workspaces": {
|
||||
"": {
|
||||
"name": "clawd-bot-landing",
|
||||
"name": "openclaw-ai",
|
||||
"dependencies": {
|
||||
"@astrojs/rss": "^4.0.15",
|
||||
"@lucide/astro": "^0.577.0",
|
||||
"@vercel/analytics": "^1.6.1",
|
||||
"astro": "^5.18.0",
|
||||
"@astrojs/rss": "^4.0.18",
|
||||
"@lucide/astro": "^1.11.0",
|
||||
"@vercel/analytics": "^2.0.1",
|
||||
"astro": "^6.1.9",
|
||||
"js-yaml": "^4.1.1",
|
||||
"simple-icons": "^16.10.0",
|
||||
"simple-icons": "^16.18.0",
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/js-yaml": "^4.0.9",
|
||||
@ -18,28 +18,32 @@
|
||||
},
|
||||
},
|
||||
"packages": {
|
||||
"@astrojs/compiler": ["@astrojs/compiler@2.13.0", "", {}, "sha512-mqVORhUJViA28fwHYaWmsXSzLO9osbdZ5ImUfxBarqsYdMlPbqAqGJCxsNzvppp1BEzc1mJNjOVvQqeDN8Vspw=="],
|
||||
"@astrojs/compiler": ["@astrojs/compiler@3.0.1", "", {}, "sha512-z97oYbdebO5aoWzuJ/8q5hLK232+17KcLZ7cJ8BCWk6+qNzVxn/gftC0KzMBUTD8WAaBkPpNSQK6PXLnNrZ0CA=="],
|
||||
|
||||
"@astrojs/internal-helpers": ["@astrojs/internal-helpers@0.7.5", "", {}, "sha512-vreGnYSSKhAjFJCWAwe/CNhONvoc5lokxtRoZims+0wa3KbHBdPHSSthJsKxPd8d/aic6lWKpRTYGY/hsgK6EA=="],
|
||||
"@astrojs/internal-helpers": ["@astrojs/internal-helpers@0.9.0", "", { "dependencies": { "picomatch": "^4.0.4" } }, "sha512-GdYkzR26re8izmyYlBqf4z2s7zNngmWLFuxw0UKiPNqHraZGS6GKWIwSHgS22RDlu2ePFJ8bzmpBcUszut/SDg=="],
|
||||
|
||||
"@astrojs/markdown-remark": ["@astrojs/markdown-remark@6.3.10", "", { "dependencies": { "@astrojs/internal-helpers": "0.7.5", "@astrojs/prism": "3.3.0", "github-slugger": "^2.0.0", "hast-util-from-html": "^2.0.3", "hast-util-to-text": "^4.0.2", "import-meta-resolve": "^4.2.0", "js-yaml": "^4.1.1", "mdast-util-definitions": "^6.0.0", "rehype-raw": "^7.0.0", "rehype-stringify": "^10.0.1", "remark-gfm": "^4.0.1", "remark-parse": "^11.0.0", "remark-rehype": "^11.1.2", "remark-smartypants": "^3.0.2", "shiki": "^3.19.0", "smol-toml": "^1.5.2", "unified": "^11.0.5", "unist-util-remove-position": "^5.0.0", "unist-util-visit": "^5.0.0", "unist-util-visit-parents": "^6.0.2", "vfile": "^6.0.3" } }, "sha512-kk4HeYR6AcnzC4QV8iSlOfh+N8TZ3MEStxPyenyCtemqn8IpEATBFMTJcfrNW32dgpt6MY3oCkMM/Tv3/I4G3A=="],
|
||||
"@astrojs/markdown-remark": ["@astrojs/markdown-remark@7.1.1", "", { "dependencies": { "@astrojs/internal-helpers": "0.9.0", "@astrojs/prism": "4.0.1", "github-slugger": "^2.0.0", "hast-util-from-html": "^2.0.3", "hast-util-to-text": "^4.0.2", "js-yaml": "^4.1.1", "mdast-util-definitions": "^6.0.0", "rehype-raw": "^7.0.0", "rehype-stringify": "^10.0.1", "remark-gfm": "^4.0.1", "remark-parse": "^11.0.0", "remark-rehype": "^11.1.2", "remark-smartypants": "^3.0.2", "retext-smartypants": "^6.2.0", "shiki": "^4.0.0", "smol-toml": "^1.6.0", "unified": "^11.0.5", "unist-util-remove-position": "^5.0.0", "unist-util-visit": "^5.1.0", "unist-util-visit-parents": "^6.0.2", "vfile": "^6.0.3" } }, "sha512-C6e9BnLGlbdv6bV8MYGeHpHxsUHrCrB4OuRLqi5LI7oiBVcBcqfUN06zpwFQdHgV48QCCrMmLpyqBr7VqC+swA=="],
|
||||
|
||||
"@astrojs/prism": ["@astrojs/prism@3.3.0", "", { "dependencies": { "prismjs": "^1.30.0" } }, "sha512-q8VwfU/fDZNoDOf+r7jUnMC2//H2l0TuQ6FkGJL8vD8nw/q5KiL3DS1KKBI3QhI9UQhpJ5dc7AtqfbXWuOgLCQ=="],
|
||||
"@astrojs/prism": ["@astrojs/prism@4.0.1", "", { "dependencies": { "prismjs": "^1.30.0" } }, "sha512-nksZQVjlferuWzhPsBpQ1JE5XuKAf1id1/9Hj4a9KG4+ofrlzxUUwX4YGQF/SuDiuiGKEnzopGOt38F3AnVWsQ=="],
|
||||
|
||||
"@astrojs/rss": ["@astrojs/rss@4.0.15", "", { "dependencies": { "fast-xml-parser": "^5.3.3", "piccolore": "^0.1.3" } }, "sha512-uXO/k6AhRkIDXmRoc6xQpoPZrimQNUmS43X4+60yunfuMNHtSRN5e/FiSi7NApcZqmugSMc5+cJi8ovqgO+qIg=="],
|
||||
"@astrojs/rss": ["@astrojs/rss@4.0.18", "", { "dependencies": { "fast-xml-parser": "^5.5.7", "piccolore": "^0.1.3", "zod": "^4.3.6" } }, "sha512-wc5DwKlbTEdgVAWnHy8krFTeQ42t1v/DJqeq5HtulYK3FYHE4krtRGjoyhS3eXXgfdV6Raoz2RU3wrMTFAitRg=="],
|
||||
|
||||
"@astrojs/telemetry": ["@astrojs/telemetry@3.3.0", "", { "dependencies": { "ci-info": "^4.2.0", "debug": "^4.4.0", "dlv": "^1.1.3", "dset": "^3.1.4", "is-docker": "^3.0.0", "is-wsl": "^3.1.0", "which-pm-runs": "^1.1.0" } }, "sha512-UFBgfeldP06qu6khs/yY+q1cDAaArM2/7AEIqQ9Cuvf7B1hNLq0xDrZkct+QoIGyjq56y8IaE2I3CTvG99mlhQ=="],
|
||||
"@astrojs/telemetry": ["@astrojs/telemetry@3.3.1", "", { "dependencies": { "ci-info": "^4.4.0", "dlv": "^1.1.3", "dset": "^3.1.4", "is-docker": "^4.0.0", "is-wsl": "^3.1.1", "which-pm-runs": "^1.1.0" } }, "sha512-7fcIxXS9J4ls5tr8b3ww9rbAIz2+HrhNJYZdkAhhB4za/I5IZ/60g+Bs8q7zwG0tOIZfNB4JWhVJ1Qkl/OrNCw=="],
|
||||
|
||||
"@babel/helper-string-parser": ["@babel/helper-string-parser@7.27.1", "", {}, "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA=="],
|
||||
|
||||
"@babel/helper-validator-identifier": ["@babel/helper-validator-identifier@7.28.5", "", {}, "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q=="],
|
||||
|
||||
"@babel/parser": ["@babel/parser@7.28.5", "", { "dependencies": { "@babel/types": "^7.28.5" }, "bin": "./bin/babel-parser.js" }, "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ=="],
|
||||
"@babel/parser": ["@babel/parser@7.29.2", "", { "dependencies": { "@babel/types": "^7.29.0" }, "bin": "./bin/babel-parser.js" }, "sha512-4GgRzy/+fsBa72/RZVJmGKPmZu9Byn8o4MoLpmNe1m8ZfYnz5emHLQz3U4gLud6Zwl0RZIcgiLD7Uq7ySFuDLA=="],
|
||||
|
||||
"@babel/types": ["@babel/types@7.28.5", "", { "dependencies": { "@babel/helper-string-parser": "^7.27.1", "@babel/helper-validator-identifier": "^7.28.5" } }, "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA=="],
|
||||
"@babel/types": ["@babel/types@7.29.0", "", { "dependencies": { "@babel/helper-string-parser": "^7.27.1", "@babel/helper-validator-identifier": "^7.28.5" } }, "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A=="],
|
||||
|
||||
"@capsizecss/unpack": ["@capsizecss/unpack@4.0.0", "", { "dependencies": { "fontkitten": "^1.0.0" } }, "sha512-VERIM64vtTP1C4mxQ5thVT9fK0apjPFobqybMtA1UdUujWka24ERHbRHFGmpbbhp73MhV+KSsHQH9C6uOTdEQA=="],
|
||||
|
||||
"@clack/core": ["@clack/core@1.2.0", "", { "dependencies": { "fast-wrap-ansi": "^0.1.3", "sisteransi": "^1.0.5" } }, "sha512-qfxof/3T3t9DPU/Rj3OmcFyZInceqj/NVtO9rwIuJqCUgh32gwPjpFQQp/ben07qKlhpwq7GzfWpST4qdJ5Drg=="],
|
||||
|
||||
"@clack/prompts": ["@clack/prompts@1.2.0", "", { "dependencies": { "@clack/core": "1.2.0", "fast-string-width": "^1.1.0", "fast-wrap-ansi": "^0.1.3", "sisteransi": "^1.0.5" } }, "sha512-4jmztR9fMqPMjz6H/UZXj0zEmE43ha1euENwkckKKel4XpSfokExPo5AiVStdHSAlHekz4d0CA/r45Ok1E4D3w=="],
|
||||
|
||||
"@emnapi/runtime": ["@emnapi/runtime@1.8.1", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-mehfKSMWjjNol8659Z8KxEMrdSJDDot5SXMq00dM8BN4o+CLNXQ0xH2V7EchNHV4RmbZLmmPdEaXZc5H2FXmDg=="],
|
||||
|
||||
"@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.27.3", "", { "os": "aix", "cpu": "ppc64" }, "sha512-9fJMTNFTWZMh5qwrBItuziu834eOCUcEqymSH7pY+zoMVEZg3gcPuBNxH1EvfVYe9h0x/Ptw8KBzv7qxb7l8dg=="],
|
||||
@ -146,7 +150,9 @@
|
||||
|
||||
"@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.5", "", {}, "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og=="],
|
||||
|
||||
"@lucide/astro": ["@lucide/astro@0.577.0", "", { "peerDependencies": { "astro": "^4 || ^5" } }, "sha512-xZcdx+MyYXY74cBzFxr9N49SlofbGjCVD9GxJZcJA40PJvJSLWSeJpD9bgKFcZfFxf1QXkEYaR6BnCS+0lek8Q=="],
|
||||
"@lucide/astro": ["@lucide/astro@1.11.0", "", { "peerDependencies": { "astro": "^4 || ^5 || ^6" } }, "sha512-b9AA35k3L/DR9J9OoGWIHks4dtMzcDo9P0sZEEDmFxIy0qqw+rKMt3TLk+YEoXppcqXMK/DTM1tIfuaH+frX/w=="],
|
||||
|
||||
"@nodable/entities": ["@nodable/entities@2.1.0", "", {}, "sha512-nyT7T3nbMyBI/lvr6L5TyWbFJAI9FTgVRakNoBqCD+PmID8DzFrrNdLLtHMwMszOtqZa8PAOV24ZqDnQrhQINA=="],
|
||||
|
||||
"@oslojs/encoding": ["@oslojs/encoding@1.1.0", "", {}, "sha512-70wQhgYmndg4GCPxPPxPGevRKqTIJ2Nh4OkiMWmDAVYsTQ+Ta7Sq+rPevXyXGdzr30/qZBnyOalCszoMxlyldQ=="],
|
||||
|
||||
@ -202,17 +208,19 @@
|
||||
|
||||
"@rollup/rollup-win32-x64-msvc": ["@rollup/rollup-win32-x64-msvc@4.55.1", "", { "os": "win32", "cpu": "x64" }, "sha512-SPEpaL6DX4rmcXtnhdrQYgzQ5W2uW3SCJch88lB2zImhJRhIIK44fkUrgIV/Q8yUNfw5oyZ5vkeQsZLhCb06lw=="],
|
||||
|
||||
"@shikijs/core": ["@shikijs/core@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4", "hast-util-to-html": "^9.0.5" } }, "sha512-iAlTtSDDbJiRpvgL5ugKEATDtHdUVkqgHDm/gbD2ZS9c88mx7G1zSYjjOxp5Qa0eaW0MAQosFRmJSk354PRoQA=="],
|
||||
"@shikijs/core": ["@shikijs/core@4.0.2", "", { "dependencies": { "@shikijs/primitive": "4.0.2", "@shikijs/types": "4.0.2", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4", "hast-util-to-html": "^9.0.5" } }, "sha512-hxT0YF4ExEqB8G/qFdtJvpmHXBYJ2lWW7qTHDarVkIudPFE6iCIrqdgWxGn5s+ppkGXI0aEGlibI0PAyzP3zlw=="],
|
||||
|
||||
"@shikijs/engine-javascript": ["@shikijs/engine-javascript@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0", "@shikijs/vscode-textmate": "^10.0.2", "oniguruma-to-es": "^4.3.4" } }, "sha512-jdKhfgW9CRtj3Tor0L7+yPwdG3CgP7W+ZEqSsojrMzCjD1e0IxIbwUMDDpYlVBlC08TACg4puwFGkZfLS+56Tw=="],
|
||||
"@shikijs/engine-javascript": ["@shikijs/engine-javascript@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2", "@shikijs/vscode-textmate": "^10.0.2", "oniguruma-to-es": "^4.3.4" } }, "sha512-7PW0Nm49DcoUIQEXlJhNNBHyoGMjalRETTCcjMqEaMoJRLljy1Bi/EGV3/qLBgLKQejdspiiYuHGQW6dX94Nag=="],
|
||||
|
||||
"@shikijs/engine-oniguruma": ["@shikijs/engine-oniguruma@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0", "@shikijs/vscode-textmate": "^10.0.2" } }, "sha512-DyXsOG0vGtNtl7ygvabHd7Mt5EY8gCNqR9Y7Lpbbd/PbJvgWrqaKzH1JW6H6qFkuUa8aCxoiYVv8/YfFljiQxA=="],
|
||||
"@shikijs/engine-oniguruma": ["@shikijs/engine-oniguruma@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2", "@shikijs/vscode-textmate": "^10.0.2" } }, "sha512-UpCB9Y2sUKlS9z8juFSKz7ZtysmeXCgnRF0dlhXBkmQnek7lAToPte8DkxmEYGNTMii72zU/lyXiCB6StuZeJg=="],
|
||||
|
||||
"@shikijs/langs": ["@shikijs/langs@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0" } }, "sha512-x/42TfhWmp6H00T6uwVrdTJGKgNdFbrEdhaDwSR5fd5zhQ1Q46bHq9EO61SCEWJR0HY7z2HNDMaBZp8JRmKiIA=="],
|
||||
"@shikijs/langs": ["@shikijs/langs@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2" } }, "sha512-KaXby5dvoeuZzN0rYQiPMjFoUrz4hgwIE+D6Du9owcHcl6/g16/yT5BQxSW5cGt2MZBz6Hl0YuRqf12omRfUUg=="],
|
||||
|
||||
"@shikijs/themes": ["@shikijs/themes@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0" } }, "sha512-o+tlOKqsr6FE4+mYJG08tfCFDS+3CG20HbldXeVoyP+cYSUxDhrFf3GPjE60U55iOkkjbpY2uC3It/eeja35/g=="],
|
||||
"@shikijs/primitive": ["@shikijs/primitive@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-M6UMPrSa3fN5ayeJwFVl9qWofl273wtK1VG8ySDZ1mQBfhCpdd8nEx7nPZ/tk7k+TYcpqBZzj/AnwxT9lO+HJw=="],
|
||||
|
||||
"@shikijs/types": ["@shikijs/types@3.22.0", "", { "dependencies": { "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-491iAekgKDBFE67z70Ok5a8KBMsQ2IJwOWw3us/7ffQkIBCyOQfm/aNwVMBUriP02QshIfgHCBSIYAl3u2eWjg=="],
|
||||
"@shikijs/themes": ["@shikijs/themes@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2" } }, "sha512-mjCafwt8lJJaVSsQvNVrJumbnnj1RI8jbUKrPKgE6E3OvQKxnuRoBaYC51H4IGHePsGN/QtALglWBU7DoKDFnA=="],
|
||||
|
||||
"@shikijs/types": ["@shikijs/types@4.0.2", "", { "dependencies": { "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-qzbeRooUTPnLE+sHD/Z8DStmaDgnbbc/pMrU203950aRqjX/6AFHeDYT+j00y2lPdz0ywJKx7o/7qnqTivtlXg=="],
|
||||
|
||||
"@shikijs/vscode-textmate": ["@shikijs/vscode-textmate@10.0.2", "", {}, "sha512-83yeghZ2xxin3Nj8z1NMd/NCuca+gsYXswywDy5bHvwlWL8tpTQmzGeUuHd9FC3E/SBEMvzJRwWEOz5gGes9Qg=="],
|
||||
|
||||
@ -230,21 +238,11 @@
|
||||
|
||||
"@types/nlcst": ["@types/nlcst@2.0.3", "", { "dependencies": { "@types/unist": "*" } }, "sha512-vSYNSDe6Ix3q+6Z7ri9lyWqgGhJTmzRjZRqyq15N0Z/1/UnVsno9G/N40NBijoYx2seFDIl0+B2mgAb9mezUCA=="],
|
||||
|
||||
"@types/node": ["@types/node@25.0.3", "", { "dependencies": { "undici-types": "~7.16.0" } }, "sha512-W609buLVRVmeW693xKfzHeIV6nJGGz98uCPfeXI1ELMLXVeKYZ9m15fAMSaUPBHYLGFsVRcMmSCksQOrZV9BYA=="],
|
||||
|
||||
"@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="],
|
||||
|
||||
"@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="],
|
||||
|
||||
"@vercel/analytics": ["@vercel/analytics@1.6.1", "", { "peerDependencies": { "@remix-run/react": "^2", "@sveltejs/kit": "^1 || ^2", "next": ">= 13", "react": "^18 || ^19 || ^19.0.0-rc", "svelte": ">= 4", "vue": "^3", "vue-router": "^4" }, "optionalPeers": ["@remix-run/react", "@sveltejs/kit", "next", "react", "svelte", "vue", "vue-router"] }, "sha512-oH9He/bEM+6oKlv3chWuOOcp8Y6fo6/PSro8hEkgCW3pu9/OiCXiUpRUogDh3Fs3LH2sosDrx8CxeOLBEE+afg=="],
|
||||
|
||||
"acorn": ["acorn@8.15.0", "", { "bin": { "acorn": "bin/acorn" } }, "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg=="],
|
||||
|
||||
"ansi-align": ["ansi-align@3.0.1", "", { "dependencies": { "string-width": "^4.1.0" } }, "sha512-IOfwwBF5iczOjp/WeY4YxyjqAFMQoZufdQWDd19SEExbVLNXqvpzSJ/M7Za4/sCPmQ0+GRquoA7bGcINcxew6w=="],
|
||||
|
||||
"ansi-regex": ["ansi-regex@6.2.2", "", {}, "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg=="],
|
||||
|
||||
"ansi-styles": ["ansi-styles@6.2.3", "", {}, "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg=="],
|
||||
"@vercel/analytics": ["@vercel/analytics@2.0.1", "", { "peerDependencies": { "@remix-run/react": "^2", "@sveltejs/kit": "^1 || ^2", "next": ">= 13", "nuxt": ">= 3", "react": "^18 || ^19 || ^19.0.0-rc", "svelte": ">= 4", "vue": "^3", "vue-router": "^4" }, "optionalPeers": ["@remix-run/react", "@sveltejs/kit", "next", "nuxt", "react", "svelte", "vue", "vue-router"] }, "sha512-MTQG6V9qQrt1tsDeF+2Uoo5aPjqbVPys1xvnIftXSJYG2SrwXRHnqEvVoYID7BTruDz4lCd2Z7rM1BdkUehk2g=="],
|
||||
|
||||
"anymatch": ["anymatch@3.1.3", "", { "dependencies": { "normalize-path": "^3.0.0", "picomatch": "^2.0.4" } }, "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw=="],
|
||||
|
||||
@ -254,24 +252,16 @@
|
||||
|
||||
"array-iterate": ["array-iterate@2.0.1", "", {}, "sha512-I1jXZMjAgCMmxT4qxXfPXa6SthSoE8h6gkSI9BGGNv8mP8G/v0blc+qFnZu6K42vTOiuME596QaLO0TP3Lk0xg=="],
|
||||
|
||||
"astro": ["astro@5.18.0", "", { "dependencies": { "@astrojs/compiler": "^2.13.0", "@astrojs/internal-helpers": "0.7.5", "@astrojs/markdown-remark": "6.3.10", "@astrojs/telemetry": "3.3.0", "@capsizecss/unpack": "^4.0.0", "@oslojs/encoding": "^1.1.0", "@rollup/pluginutils": "^5.3.0", "acorn": "^8.15.0", "aria-query": "^5.3.2", "axobject-query": "^4.1.0", "boxen": "8.0.1", "ci-info": "^4.3.1", "clsx": "^2.1.1", "common-ancestor-path": "^1.0.1", "cookie": "^1.1.1", "cssesc": "^3.0.0", "debug": "^4.4.3", "deterministic-object-hash": "^2.0.2", "devalue": "^5.6.2", "diff": "^8.0.3", "dlv": "^1.1.3", "dset": "^3.1.4", "es-module-lexer": "^1.7.0", "esbuild": "^0.27.3", "estree-walker": "^3.0.3", "flattie": "^1.1.1", "fontace": "~0.4.0", "github-slugger": "^2.0.0", "html-escaper": "3.0.3", "http-cache-semantics": "^4.2.0", "import-meta-resolve": "^4.2.0", "js-yaml": "^4.1.1", "magic-string": "^0.30.21", "magicast": "^0.5.1", "mrmime": "^2.0.1", "neotraverse": "^0.6.18", "p-limit": "^6.2.0", "p-queue": "^8.1.1", "package-manager-detector": "^1.6.0", "piccolore": "^0.1.3", "picomatch": "^4.0.3", "prompts": "^2.4.2", "rehype": "^13.0.2", "semver": "^7.7.3", "shiki": "^3.21.0", "smol-toml": "^1.6.0", "svgo": "^4.0.0", "tinyexec": "^1.0.2", "tinyglobby": "^0.2.15", "tsconfck": "^3.1.6", "ultrahtml": "^1.6.0", "unifont": "~0.7.3", "unist-util-visit": "^5.0.0", "unstorage": "^1.17.4", "vfile": "^6.0.3", "vite": "^6.4.1", "vitefu": "^1.1.1", "xxhash-wasm": "^1.1.0", "yargs-parser": "^21.1.1", "yocto-spinner": "^0.2.3", "zod": "^3.25.76", "zod-to-json-schema": "^3.25.1", "zod-to-ts": "^1.2.0" }, "optionalDependencies": { "sharp": "^0.34.0" }, "bin": { "astro": "astro.js" } }, "sha512-CHiohwJIS4L0G6/IzE1Fx3dgWqXBCXus/od0eGUfxrZJD2um2pE7ehclMmgL/fXqbU7NfE1Ze2pq34h2QaA6iQ=="],
|
||||
"astro": ["astro@6.1.9", "", { "dependencies": { "@astrojs/compiler": "^3.0.1", "@astrojs/internal-helpers": "0.9.0", "@astrojs/markdown-remark": "7.1.1", "@astrojs/telemetry": "3.3.1", "@capsizecss/unpack": "^4.0.0", "@clack/prompts": "^1.1.0", "@oslojs/encoding": "^1.1.0", "@rollup/pluginutils": "^5.3.0", "aria-query": "^5.3.2", "axobject-query": "^4.1.0", "ci-info": "^4.4.0", "clsx": "^2.1.1", "common-ancestor-path": "^2.0.0", "cookie": "^1.1.1", "devalue": "^5.6.3", "diff": "^8.0.3", "dset": "^3.1.4", "es-module-lexer": "^2.0.0", "esbuild": "^0.27.3", "flattie": "^1.1.1", "fontace": "~0.4.1", "github-slugger": "^2.0.0", "html-escaper": "3.0.3", "http-cache-semantics": "^4.2.0", "js-yaml": "^4.1.1", "magic-string": "^0.30.21", "magicast": "^0.5.2", "mrmime": "^2.0.1", "neotraverse": "^0.6.18", "obug": "^2.1.1", "p-limit": "^7.3.0", "p-queue": "^9.1.0", "package-manager-detector": "^1.6.0", "piccolore": "^0.1.3", "picomatch": "^4.0.4", "rehype": "^13.0.2", "semver": "^7.7.4", "shiki": "^4.0.2", "smol-toml": "^1.6.0", "svgo": "^4.0.1", "tinyclip": "^0.1.12", "tinyexec": "^1.0.4", "tinyglobby": "^0.2.15", "tsconfck": "^3.1.6", "ultrahtml": "^1.6.0", "unifont": "~0.7.4", "unist-util-visit": "^5.1.0", "unstorage": "^1.17.5", "vfile": "^6.0.3", "vite": "^7.3.2", "vitefu": "^1.1.2", "xxhash-wasm": "^1.1.0", "yargs-parser": "^22.0.0", "zod": "^4.3.6" }, "optionalDependencies": { "sharp": "^0.34.0" }, "bin": { "astro": "bin/astro.mjs" } }, "sha512-NsAHzMzpznB281g2aM5qnBt2QjfH6ttKiZ3hSZw52If8JJ+62kbnBKbyKhR2glQcJLl7Jfe4GSl0DihFZ36rRQ=="],
|
||||
|
||||
"axobject-query": ["axobject-query@4.1.0", "", {}, "sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ=="],
|
||||
|
||||
"bail": ["bail@2.0.2", "", {}, "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw=="],
|
||||
|
||||
"base-64": ["base-64@1.0.0", "", {}, "sha512-kwDPIFCGx0NZHog36dj+tHiwP4QMzsZ3AgMViUBKI0+V5n4U0ufTCUMhnQ04diaRI8EX/QcPfql7zlhZ7j4zgg=="],
|
||||
|
||||
"boolbase": ["boolbase@1.0.0", "", {}, "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww=="],
|
||||
|
||||
"boxen": ["boxen@8.0.1", "", { "dependencies": { "ansi-align": "^3.0.1", "camelcase": "^8.0.0", "chalk": "^5.3.0", "cli-boxes": "^3.0.0", "string-width": "^7.2.0", "type-fest": "^4.21.0", "widest-line": "^5.0.0", "wrap-ansi": "^9.0.0" } }, "sha512-F3PH5k5juxom4xktynS7MoFY+NUWH5LC4CnH11YB8NPew+HLpmBLCybSAEyb2F+4pRXhuhWqFesoQd6DAyc2hw=="],
|
||||
|
||||
"camelcase": ["camelcase@8.0.0", "", {}, "sha512-8WB3Jcas3swSvjIeA2yvCJ+Miyz5l1ZmB6HFb9R1317dt9LCQoswg/BGrmAmkWVEszSrrg4RwmO46qIm2OEnSA=="],
|
||||
|
||||
"ccount": ["ccount@2.0.1", "", {}, "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg=="],
|
||||
|
||||
"chalk": ["chalk@5.6.2", "", {}, "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA=="],
|
||||
|
||||
"character-entities": ["character-entities@2.0.2", "", {}, "sha512-shx7oQ0Awen/BRIdkjkvz54PnEEI/EjwXDSIZp86/KKdbafHh1Df/RYGBhn4hbe2+uKC9FnT5UCEdyPz3ai9hQ=="],
|
||||
|
||||
"character-entities-html4": ["character-entities-html4@2.1.0", "", {}, "sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA=="],
|
||||
@ -280,9 +270,7 @@
|
||||
|
||||
"chokidar": ["chokidar@5.0.0", "", { "dependencies": { "readdirp": "^5.0.0" } }, "sha512-TQMmc3w+5AxjpL8iIiwebF73dRDF4fBIieAqGn9RGCWaEVwQ6Fb2cGe31Yns0RRIzii5goJ1Y7xbMwo1TxMplw=="],
|
||||
|
||||
"ci-info": ["ci-info@4.3.1", "", {}, "sha512-Wdy2Igu8OcBpI2pZePZ5oWjPC38tmDVx5WKUXKwlLYkA0ozo85sLsLvkBbBn/sZaSCMFOGZJ14fvW9t5/d7kdA=="],
|
||||
|
||||
"cli-boxes": ["cli-boxes@3.0.0", "", {}, "sha512-/lzGpEWL/8PfI0BmBOPRwp0c/wFNX1RdUML3jK/RcSBA9T8mZDdQpqYBKtCFTOfQbwPqWEOpjqW+Fnayc0969g=="],
|
||||
"ci-info": ["ci-info@4.4.0", "", {}, "sha512-77PSwercCZU2Fc4sX94eF8k8Pxte6JAwL4/ICZLFjJLqegs7kCuAsqqj/70NQF6TvDpgFjkubQB2FW2ZZddvQg=="],
|
||||
|
||||
"clsx": ["clsx@2.1.1", "", {}, "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA=="],
|
||||
|
||||
@ -290,11 +278,11 @@
|
||||
|
||||
"commander": ["commander@11.1.0", "", {}, "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ=="],
|
||||
|
||||
"common-ancestor-path": ["common-ancestor-path@1.0.1", "", {}, "sha512-L3sHRo1pXXEqX8VU28kfgUY+YGsk09hPqZiZmLacNib6XNTCM8ubYeT7ryXQw8asB1sKgcU5lkB7ONug08aB8w=="],
|
||||
"common-ancestor-path": ["common-ancestor-path@2.0.0", "", {}, "sha512-dnN3ibLeoRf2HNC+OlCiNc5d2zxbLJXOtiZUudNFSXZrNSydxcCsSpRzXwfu7BBWCIfHPw+xTayeBvJCP/D8Ng=="],
|
||||
|
||||
"cookie": ["cookie@1.1.1", "", {}, "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ=="],
|
||||
|
||||
"cookie-es": ["cookie-es@1.2.2", "", {}, "sha512-+W7VmiVINB+ywl1HGXJXmrqkOhpKrIiVZV6tQuV54ZyQC7MMuBt81Vc336GMLoHBq5hV/F9eXgt5Mnx0Rha5Fg=="],
|
||||
"cookie-es": ["cookie-es@1.2.3", "", {}, "sha512-lXVyvUvrNXblMqzIRrxHb57UUVmqsSWlxqt3XIjCkUP0wDAf6uicO6KMbEgYrMNtEvWgWHwe42CKxPu9MYAnWw=="],
|
||||
|
||||
"crossws": ["crossws@0.3.5", "", { "dependencies": { "uncrypto": "^0.1.3" } }, "sha512-ojKiDvcmByhwa8YYqbQI/hg7MEU0NC03+pSdEq4ZUnZR9xXpwk7E43SMNGkn+JxJGPFtNvQ48+vV2p+P1ml5PA=="],
|
||||
|
||||
@ -304,15 +292,13 @@
|
||||
|
||||
"css-what": ["css-what@6.2.2", "", {}, "sha512-u/O3vwbptzhMs3L1fQE82ZSLHQQfto5gyZzwteVIEyeaY5Fc7R4dapF/BvRoSYFeqfBk4m0V1Vafq5Pjv25wvA=="],
|
||||
|
||||
"cssesc": ["cssesc@3.0.0", "", { "bin": { "cssesc": "bin/cssesc" } }, "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg=="],
|
||||
|
||||
"csso": ["csso@5.0.5", "", { "dependencies": { "css-tree": "~2.2.0" } }, "sha512-0LrrStPOdJj+SPCCrGhzryycLjwcgUSHBtxNA8aIDxf0GLsRh1cKYhB00Gd1lDOS4yGH69+SNn13+TWbVHETFQ=="],
|
||||
|
||||
"debug": ["debug@4.4.3", "", { "dependencies": { "ms": "^2.1.3" } }, "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA=="],
|
||||
|
||||
"decode-named-character-reference": ["decode-named-character-reference@1.2.0", "", { "dependencies": { "character-entities": "^2.0.0" } }, "sha512-c6fcElNV6ShtZXmsgNgFFV5tVX2PaV4g+MOAkb8eXHvn6sryJBrZa9r0zV6+dtTyoCKxtDy5tyQ5ZwQuidtd+Q=="],
|
||||
|
||||
"defu": ["defu@6.1.4", "", {}, "sha512-mEQCMmwJu317oSz8CwdIOdwf3xMif1ttiM8LTufzc3g6kR+9Pe236twL8j3IYT1F7GfRgGcW6MWxzZjLIkuHIg=="],
|
||||
"defu": ["defu@6.1.7", "", {}, "sha512-7z22QmUWiQ/2d0KkdYmANbRUVABpZ9SNYyH5vx6PZ+nE5bcC0l7uFvEfHlyld/HcGBFTL536ClDt3DEcSlEJAQ=="],
|
||||
|
||||
"dequal": ["dequal@2.0.3", "", {}, "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA=="],
|
||||
|
||||
@ -320,9 +306,7 @@
|
||||
|
||||
"detect-libc": ["detect-libc@2.1.2", "", {}, "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ=="],
|
||||
|
||||
"deterministic-object-hash": ["deterministic-object-hash@2.0.2", "", { "dependencies": { "base-64": "^1.0.0" } }, "sha512-KxektNH63SrbfUyDiwXqRb1rLwKt33AmMv+5Nhsw1kqZ13SJBRTgZHtGbE+hH3a1mVW1cz+4pqSWVPAtLVXTzQ=="],
|
||||
|
||||
"devalue": ["devalue@5.6.2", "", {}, "sha512-nPRkjWzzDQlsejL1WVifk5rvcFi/y1onBRxjaFMjZeR9mFpqu2gmAZ9xUB9/IEanEP/vBtGeGganC/GO1fmufg=="],
|
||||
"devalue": ["devalue@5.7.1", "", {}, "sha512-MUbZ586EgQqdRnC4yDrlod3BEdyvE4TapGYHMW2CiaW+KkkFmWEFqBUaLltEZCGi0iFXCEjRF0OjF0DV2QHjOA=="],
|
||||
|
||||
"devlop": ["devlop@1.1.0", "", { "dependencies": { "dequal": "^2.0.0" } }, "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA=="],
|
||||
|
||||
@ -340,39 +324,43 @@
|
||||
|
||||
"dset": ["dset@3.1.4", "", {}, "sha512-2QF/g9/zTaPDc3BjNcVTGoBbXBgYfMTTceLaYcFJ/W9kggFUkhxD/hMEeuLKbugyef9SqAx8cpgwlIP/jinUTA=="],
|
||||
|
||||
"emoji-regex": ["emoji-regex@10.6.0", "", {}, "sha512-toUI84YS5YmxW219erniWD0CIVOo46xGKColeNQRgOzDorgBi1v4D71/OFzgD9GO2UGKIv1C3Sp8DAn0+j5w7A=="],
|
||||
|
||||
"entities": ["entities@6.0.1", "", {}, "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g=="],
|
||||
|
||||
"es-module-lexer": ["es-module-lexer@1.7.0", "", {}, "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA=="],
|
||||
"es-module-lexer": ["es-module-lexer@2.1.0", "", {}, "sha512-n27zTYMjYu1aj4MjCWzSP7G9r75utsaoc8m61weK+W8JMBGGQybd43GstCXZ3WNmSFtGT9wi59qQTW6mhTR5LQ=="],
|
||||
|
||||
"esbuild": ["esbuild@0.27.3", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.27.3", "@esbuild/android-arm": "0.27.3", "@esbuild/android-arm64": "0.27.3", "@esbuild/android-x64": "0.27.3", "@esbuild/darwin-arm64": "0.27.3", "@esbuild/darwin-x64": "0.27.3", "@esbuild/freebsd-arm64": "0.27.3", "@esbuild/freebsd-x64": "0.27.3", "@esbuild/linux-arm": "0.27.3", "@esbuild/linux-arm64": "0.27.3", "@esbuild/linux-ia32": "0.27.3", "@esbuild/linux-loong64": "0.27.3", "@esbuild/linux-mips64el": "0.27.3", "@esbuild/linux-ppc64": "0.27.3", "@esbuild/linux-riscv64": "0.27.3", "@esbuild/linux-s390x": "0.27.3", "@esbuild/linux-x64": "0.27.3", "@esbuild/netbsd-arm64": "0.27.3", "@esbuild/netbsd-x64": "0.27.3", "@esbuild/openbsd-arm64": "0.27.3", "@esbuild/openbsd-x64": "0.27.3", "@esbuild/openharmony-arm64": "0.27.3", "@esbuild/sunos-x64": "0.27.3", "@esbuild/win32-arm64": "0.27.3", "@esbuild/win32-ia32": "0.27.3", "@esbuild/win32-x64": "0.27.3" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-8VwMnyGCONIs6cWue2IdpHxHnAjzxnw2Zr7MkVxB2vjmQ2ivqGFb4LEG3SMnv0Gb2F/G/2yA8zUaiL1gywDCCg=="],
|
||||
|
||||
"escape-string-regexp": ["escape-string-regexp@5.0.0", "", {}, "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw=="],
|
||||
|
||||
"estree-walker": ["estree-walker@3.0.3", "", { "dependencies": { "@types/estree": "^1.0.0" } }, "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g=="],
|
||||
"estree-walker": ["estree-walker@2.0.2", "", {}, "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="],
|
||||
|
||||
"eventemitter3": ["eventemitter3@5.0.1", "", {}, "sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA=="],
|
||||
|
||||
"extend": ["extend@3.0.2", "", {}, "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g=="],
|
||||
|
||||
"fast-xml-parser": ["fast-xml-parser@5.3.5", "", { "dependencies": { "strnum": "^2.1.2" }, "bin": { "fxparser": "src/cli/cli.js" } }, "sha512-JeaA2Vm9ffQKp9VjvfzObuMCjUYAp5WDYhRYL5LrBPY/jUDlUtOvDfot0vKSkB9tuX885BDHjtw4fZadD95wnA=="],
|
||||
"fast-string-truncated-width": ["fast-string-truncated-width@1.2.1", "", {}, "sha512-Q9acT/+Uu3GwGj+5w/zsGuQjh9O1TyywhIwAxHudtWrgF09nHOPrvTLhQevPbttcxjr/SNN7mJmfOw/B1bXgow=="],
|
||||
|
||||
"fast-string-width": ["fast-string-width@1.1.0", "", { "dependencies": { "fast-string-truncated-width": "^1.2.0" } }, "sha512-O3fwIVIH5gKB38QNbdg+3760ZmGz0SZMgvwJbA1b2TGXceKE6A2cOlfogh1iw8lr049zPyd7YADHy+B7U4W9bQ=="],
|
||||
|
||||
"fast-wrap-ansi": ["fast-wrap-ansi@0.1.6", "", { "dependencies": { "fast-string-width": "^1.1.0" } }, "sha512-HlUwET7a5gqjURj70D5jl7aC3Zmy4weA1SHUfM0JFI0Ptq987NH2TwbBFLoERhfwk+E+eaq4EK3jXoT+R3yp3w=="],
|
||||
|
||||
"fast-xml-builder": ["fast-xml-builder@1.1.5", "", { "dependencies": { "path-expression-matcher": "^1.1.3" } }, "sha512-4TJn/8FKLeslLAH3dnohXqE3QSoxkhvaMzepOIZytwJXZO69Bfz0HBdDHzOTOon6G59Zrk6VQ2bEiv1t61rfkA=="],
|
||||
|
||||
"fast-xml-parser": ["fast-xml-parser@5.7.2", "", { "dependencies": { "@nodable/entities": "^2.1.0", "fast-xml-builder": "^1.1.5", "path-expression-matcher": "^1.5.0", "strnum": "^2.2.3" }, "bin": { "fxparser": "src/cli/cli.js" } }, "sha512-P7oW7tLbYnhOLQk/Gv7cZgzgMPP/XN03K02/Jy6Y/NHzyIAIpxuZIM/YqAkfiXFPxA2CTm7NtCijK9EDu09u2w=="],
|
||||
|
||||
"fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="],
|
||||
|
||||
"flattie": ["flattie@1.1.1", "", {}, "sha512-9UbaD6XdAL97+k/n+N7JwX46K/M6Zc6KcFYskrYL8wbBV/Uyk0CTAMY0VT+qiK5PM7AIc9aTWYtq65U7T+aCNQ=="],
|
||||
|
||||
"fontace": ["fontace@0.4.0", "", { "dependencies": { "fontkitten": "^1.0.0" } }, "sha512-moThBCItUe2bjZip5PF/iZClpKHGLwMvR79Kp8XpGRBrvoRSnySN4VcILdv3/MJzbhvUA5WeiUXF5o538m5fvg=="],
|
||||
"fontace": ["fontace@0.4.1", "", { "dependencies": { "fontkitten": "^1.0.2" } }, "sha512-lDMvbAzSnHmbYMTEld5qdtvNH2/pWpICOqpean9IgC7vUbUJc3k+k5Dokp85CegamqQpFbXf0rAVkbzpyTA8aw=="],
|
||||
|
||||
"fontkitten": ["fontkitten@1.0.1", "", { "dependencies": { "tiny-inflate": "^1.0.3" } }, "sha512-m+/cO+/kAU9farlejecXLgQH20+UXyH0K6oosGtogAz7BWco+KTYE60epKwMt8eVxqlOE2Fs+GoHVlGDUbKOoA=="],
|
||||
|
||||
"fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="],
|
||||
|
||||
"get-east-asian-width": ["get-east-asian-width@1.4.0", "", {}, "sha512-QZjmEOC+IT1uk6Rx0sX22V6uHWVwbdbxf1faPqJ1QhLdGgsRGCZoyaQBm/piRdJy/D2um6hM1UP7ZEeQ4EkP+Q=="],
|
||||
|
||||
"github-slugger": ["github-slugger@2.0.0", "", {}, "sha512-IaOQ9puYtjrkq7Y0Ygl9KDZnrf/aiUJYUpVf89y8kyaxbRG7Y1SrX/jaumrv81vc61+kiMempujsM3Yw7w5qcw=="],
|
||||
|
||||
"h3": ["h3@1.15.5", "", { "dependencies": { "cookie-es": "^1.2.2", "crossws": "^0.3.5", "defu": "^6.1.4", "destr": "^2.0.5", "iron-webcrypto": "^1.2.1", "node-mock-http": "^1.0.4", "radix3": "^1.1.2", "ufo": "^1.6.3", "uncrypto": "^0.1.3" } }, "sha512-xEyq3rSl+dhGX2Lm0+eFQIAzlDN6Fs0EcC4f7BNUmzaRX/PTzeuM+Tr2lHB8FoXggsQIeXLj8EDVgs5ywxyxmg=="],
|
||||
"h3": ["h3@1.15.11", "", { "dependencies": { "cookie-es": "^1.2.3", "crossws": "^0.3.5", "defu": "^6.1.6", "destr": "^2.0.5", "iron-webcrypto": "^1.2.1", "node-mock-http": "^1.0.4", "radix3": "^1.1.2", "ufo": "^1.6.3", "uncrypto": "^0.1.3" } }, "sha512-L3THSe2MPeBwgIZVSH5zLdBBU90TOxarvhK9d04IDY2AmVS8j2Jz2LIWtwsGOU3lu2I5jCN7FNvVfY2+XyF+mg=="],
|
||||
|
||||
"hast-util-from-html": ["hast-util-from-html@2.0.3", "", { "dependencies": { "@types/hast": "^3.0.0", "devlop": "^1.1.0", "hast-util-from-parse5": "^8.0.0", "parse5": "^7.0.0", "vfile": "^6.0.0", "vfile-message": "^4.0.0" } }, "sha512-CUSRHXyKjzHov8yKsQjGOElXy/3EKpyX56ELnkHH34vDVw1N1XSQ1ZcAvTyAPtGqLTuKP/uxM+aLkSPqF/EtMw=="],
|
||||
|
||||
@ -400,31 +388,25 @@
|
||||
|
||||
"http-cache-semantics": ["http-cache-semantics@4.2.0", "", {}, "sha512-dTxcvPXqPvXBQpq5dUr6mEMJX4oIEFv6bwom3FDwKRDsuIjjJGANqhBuoAn9c1RQJIdAKav33ED65E2ys+87QQ=="],
|
||||
|
||||
"import-meta-resolve": ["import-meta-resolve@4.2.0", "", {}, "sha512-Iqv2fzaTQN28s/FwZAoFq0ZSs/7hMAHJVX+w8PZl3cY19Pxk6jFFalxQoIfW2826i/fDLXv8IiEZRIT0lDuWcg=="],
|
||||
|
||||
"iron-webcrypto": ["iron-webcrypto@1.2.1", "", {}, "sha512-feOM6FaSr6rEABp/eDfVseKyTMDt+KGpeB35SkVn9Tyn0CqvVsY3EwI0v5i8nMHyJnzCIQf7nsy3p41TPkJZhg=="],
|
||||
|
||||
"is-docker": ["is-docker@3.0.0", "", { "bin": { "is-docker": "cli.js" } }, "sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ=="],
|
||||
|
||||
"is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="],
|
||||
"is-docker": ["is-docker@4.0.0", "", { "bin": { "is-docker": "cli.js" } }, "sha512-LHE+wROyG/Y/0ZnbktRCoTix2c1RhgWaZraMZ8o1Q7zCh0VSrICJQO5oqIIISrcSBtrXv0o233w1IYwsWCjTzA=="],
|
||||
|
||||
"is-inside-container": ["is-inside-container@1.0.0", "", { "dependencies": { "is-docker": "^3.0.0" }, "bin": { "is-inside-container": "cli.js" } }, "sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA=="],
|
||||
|
||||
"is-plain-obj": ["is-plain-obj@4.1.0", "", {}, "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg=="],
|
||||
|
||||
"is-wsl": ["is-wsl@3.1.0", "", { "dependencies": { "is-inside-container": "^1.0.0" } }, "sha512-UcVfVfaK4Sc4m7X3dUSoHoozQGBEFeDC+zVo06t98xe8CzHSZZBekNXH+tu0NalHolcJ/QAGqS46Hef7QXBIMw=="],
|
||||
"is-wsl": ["is-wsl@3.1.1", "", { "dependencies": { "is-inside-container": "^1.0.0" } }, "sha512-e6rvdUCiQCAuumZslxRJWR/Doq4VpPR82kqclvcS0efgt430SlGIk05vdCN58+VrzgtIcfNODjozVielycD4Sw=="],
|
||||
|
||||
"js-yaml": ["js-yaml@4.1.1", "", { "dependencies": { "argparse": "^2.0.1" }, "bin": { "js-yaml": "bin/js-yaml.js" } }, "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA=="],
|
||||
|
||||
"kleur": ["kleur@3.0.3", "", {}, "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w=="],
|
||||
|
||||
"longest-streak": ["longest-streak@3.1.0", "", {}, "sha512-9Ri+o0JYgehTaVBBDoMqIl8GXtbWg711O3srftcHhZ0dqnETqLaoIK0x17fUw9rFSlK/0NlsKe0Ahhyl5pXE2g=="],
|
||||
|
||||
"lru-cache": ["lru-cache@11.2.5", "", {}, "sha512-vFrFJkWtJvJnD5hg+hJvVE8Lh/TcMzKnTgCWmtBipwI5yLX/iX+5UB2tfuyODF5E7k9xEzMdYgGqaSb1c0c5Yw=="],
|
||||
"lru-cache": ["lru-cache@11.3.5", "", {}, "sha512-NxVFwLAnrd9i7KUBxC4DrUhmgjzOs+1Qm50D3oF1/oL+r1NpZ4gA7xvG0/zJ8evR7zIKn4vLf7qTNduWFtCrRw=="],
|
||||
|
||||
"magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="],
|
||||
|
||||
"magicast": ["magicast@0.5.1", "", { "dependencies": { "@babel/parser": "^7.28.5", "@babel/types": "^7.28.5", "source-map-js": "^1.2.1" } }, "sha512-xrHS24IxaLrvuo613F719wvOIv9xPHFWQHuvGUBmPnCA/3MQxKI3b+r7n1jAoDHmsbC5bRhTZYR77invLAxVnw=="],
|
||||
"magicast": ["magicast@0.5.2", "", { "dependencies": { "@babel/parser": "^7.29.0", "@babel/types": "^7.29.0", "source-map-js": "^1.2.1" } }, "sha512-E3ZJh4J3S9KfwdjZhe2afj6R9lGIN5Pher1pF39UGrXRqq/VDaGVIGN13BjHd2u8B61hArAGOnso7nBOouW3TQ=="],
|
||||
|
||||
"markdown-table": ["markdown-table@3.0.4", "", {}, "sha512-wiYz4+JrLyb/DqW2hkFJxP7Vd7JuTDm77fvbM8VfEQdmSMqcImWeeRbHwZjBjIFki/VaMK2BhFi7oUUZeM5bqw=="],
|
||||
|
||||
@ -530,6 +512,8 @@
|
||||
|
||||
"nth-check": ["nth-check@2.1.1", "", { "dependencies": { "boolbase": "^1.0.0" } }, "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w=="],
|
||||
|
||||
"obug": ["obug@2.1.1", "", {}, "sha512-uTqF9MuPraAQ+IsnPf366RG4cP9RtUi7MLO1N3KEc+wb0a6yKpeL0lmk2IB1jY5KHPAlTc6T/JRdC/YqxHNwkQ=="],
|
||||
|
||||
"ofetch": ["ofetch@1.5.1", "", { "dependencies": { "destr": "^2.0.5", "node-fetch-native": "^1.6.7", "ufo": "^1.6.1" } }, "sha512-2W4oUZlVaqAPAil6FUg/difl6YhqhUR7x2eZY4bQCko22UXg3hptq9KLQdqFClV+Wu85UX7hNtdGTngi/1BxcA=="],
|
||||
|
||||
"ohash": ["ohash@2.0.11", "", {}, "sha512-RdR9FQrFwNBNXAr4GixM8YaRZRJ5PUWbKYbE5eOsrwAjJW0q2REGcf79oYPsLyskQCZG1PLN+S/K1V00joZAoQ=="],
|
||||
@ -538,11 +522,11 @@
|
||||
|
||||
"oniguruma-to-es": ["oniguruma-to-es@4.3.4", "", { "dependencies": { "oniguruma-parser": "^0.12.1", "regex": "^6.0.1", "regex-recursion": "^6.0.2" } }, "sha512-3VhUGN3w2eYxnTzHn+ikMI+fp/96KoRSVK9/kMTcFqj1NRDh2IhQCKvYxDnWePKRXY/AqH+Fuiyb7VHSzBjHfA=="],
|
||||
|
||||
"p-limit": ["p-limit@6.2.0", "", { "dependencies": { "yocto-queue": "^1.1.1" } }, "sha512-kuUqqHNUqoIWp/c467RI4X6mmyuojY5jGutNU0wVTmEOOfcuwLqyMVoAi9MKi2Ak+5i9+nhmrK4ufZE8069kHA=="],
|
||||
"p-limit": ["p-limit@7.3.0", "", { "dependencies": { "yocto-queue": "^1.2.1" } }, "sha512-7cIXg/Z0M5WZRblrsOla88S4wAK+zOQQWeBYfV3qJuJXMr+LnbYjaadrFaS0JILfEDPVqHyKnZ1Z/1d6J9VVUw=="],
|
||||
|
||||
"p-queue": ["p-queue@8.1.1", "", { "dependencies": { "eventemitter3": "^5.0.1", "p-timeout": "^6.1.2" } }, "sha512-aNZ+VfjobsWryoiPnEApGGmf5WmNsCo9xu8dfaYamG5qaLP7ClhLN6NgsFe6SwJ2UbLEBK5dv9x8Mn5+RVhMWQ=="],
|
||||
"p-queue": ["p-queue@9.1.2", "", { "dependencies": { "eventemitter3": "^5.0.1", "p-timeout": "^7.0.0" } }, "sha512-ktsDOALzTYTWWF1PbkNVg2rOt+HaOaMWJMUnt7T3qf5tvZ1L8dBW3tObzprBcXNMKkwj+yFSLqHso0x+UFcJXw=="],
|
||||
|
||||
"p-timeout": ["p-timeout@6.1.4", "", {}, "sha512-MyIV3ZA/PmyBN/ud8vV9XzwTrNtR4jFrObymZYnZqMmW0zA8Z17vnT0rBgFE/TlohB+YCHqXMgZzb3Csp49vqg=="],
|
||||
"p-timeout": ["p-timeout@7.0.1", "", {}, "sha512-AxTM2wDGORHGEkPCt8yqxOTMgpfbEHqF51f/5fJCmwFC3C/zNcGT63SymH2ttOAaiIws2zVg4+izQCjrakcwHg=="],
|
||||
|
||||
"package-manager-detector": ["package-manager-detector@1.6.0", "", {}, "sha512-61A5ThoTiDG/C8s8UMZwSorAGwMJ0ERVGj2OjoW5pAalsNOg15+iQiPzrLJ4jhZ1HJzmC2PIHT2oEiH3R5fzNA=="],
|
||||
|
||||
@ -550,18 +534,18 @@
|
||||
|
||||
"parse5": ["parse5@7.3.0", "", { "dependencies": { "entities": "^6.0.0" } }, "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw=="],
|
||||
|
||||
"path-expression-matcher": ["path-expression-matcher@1.5.0", "", {}, "sha512-cbrerZV+6rvdQrrD+iGMcZFEiiSrbv9Tfdkvnusy6y0x0GKBXREFg/Y65GhIfm0tnLntThhzCnfKwp1WRjeCyQ=="],
|
||||
|
||||
"piccolore": ["piccolore@0.1.3", "", {}, "sha512-o8bTeDWjE086iwKrROaDf31K0qC/BENdm15/uH9usSC/uZjJOKb2YGiVHfLY4GhwsERiPI1jmwI2XrA7ACOxVw=="],
|
||||
|
||||
"picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="],
|
||||
|
||||
"picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||
"picomatch": ["picomatch@4.0.4", "", {}, "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A=="],
|
||||
|
||||
"postcss": ["postcss@8.5.6", "", { "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg=="],
|
||||
|
||||
"prismjs": ["prismjs@1.30.0", "", {}, "sha512-DEvV2ZF2r2/63V+tK8hQvrR2ZGn10srHbXviTlcv7Kpzw8jWiNTqbVgjO3IY8RxrrOUF8VPMQQFysYYYv0YZxw=="],
|
||||
|
||||
"prompts": ["prompts@2.4.2", "", { "dependencies": { "kleur": "^3.0.3", "sisteransi": "^1.0.5" } }, "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q=="],
|
||||
|
||||
"property-information": ["property-information@7.1.0", "", {}, "sha512-TwEZ+X+yCJmYfL7TPUOcvBZ4QfoT5YenQiJuX//0th53DE6w0xxLEtfK3iyryQFddXuvkIk51EEgrJQ0WJkOmQ=="],
|
||||
|
||||
"radix3": ["radix3@1.1.2", "", {}, "sha512-b484I/7b8rDEdSDKckSSBA8knMpcdsXudlE/LNL639wFoHKwLbEkQFZHWEYwDC0wa0FKUcCY+GAF73Z7wxNVFA=="],
|
||||
@ -602,15 +586,15 @@
|
||||
|
||||
"rollup": ["rollup@4.55.1", "", { "dependencies": { "@types/estree": "1.0.8" }, "optionalDependencies": { "@rollup/rollup-android-arm-eabi": "4.55.1", "@rollup/rollup-android-arm64": "4.55.1", "@rollup/rollup-darwin-arm64": "4.55.1", "@rollup/rollup-darwin-x64": "4.55.1", "@rollup/rollup-freebsd-arm64": "4.55.1", "@rollup/rollup-freebsd-x64": "4.55.1", "@rollup/rollup-linux-arm-gnueabihf": "4.55.1", "@rollup/rollup-linux-arm-musleabihf": "4.55.1", "@rollup/rollup-linux-arm64-gnu": "4.55.1", "@rollup/rollup-linux-arm64-musl": "4.55.1", "@rollup/rollup-linux-loong64-gnu": "4.55.1", "@rollup/rollup-linux-loong64-musl": "4.55.1", "@rollup/rollup-linux-ppc64-gnu": "4.55.1", "@rollup/rollup-linux-ppc64-musl": "4.55.1", "@rollup/rollup-linux-riscv64-gnu": "4.55.1", "@rollup/rollup-linux-riscv64-musl": "4.55.1", "@rollup/rollup-linux-s390x-gnu": "4.55.1", "@rollup/rollup-linux-x64-gnu": "4.55.1", "@rollup/rollup-linux-x64-musl": "4.55.1", "@rollup/rollup-openbsd-x64": "4.55.1", "@rollup/rollup-openharmony-arm64": "4.55.1", "@rollup/rollup-win32-arm64-msvc": "4.55.1", "@rollup/rollup-win32-ia32-msvc": "4.55.1", "@rollup/rollup-win32-x64-gnu": "4.55.1", "@rollup/rollup-win32-x64-msvc": "4.55.1", "fsevents": "~2.3.2" }, "bin": { "rollup": "dist/bin/rollup" } }, "sha512-wDv/Ht1BNHB4upNbK74s9usvl7hObDnvVzknxqY/E/O3X6rW1U1rV1aENEfJ54eFZDTNo7zv1f5N4edCluH7+A=="],
|
||||
|
||||
"sax": ["sax@1.4.3", "", {}, "sha512-yqYn1JhPczigF94DMS+shiDMjDowYO6y9+wB/4WgO0Y19jWYk0lQ4tuG5KI7kj4FTp1wxPj5IFfcrz/s1c3jjQ=="],
|
||||
"sax": ["sax@1.6.0", "", {}, "sha512-6R3J5M4AcbtLUdZmRv2SygeVaM7IhrLXu9BmnOGmmACak8fiUtOsYNWUS4uK7upbmHIBbLBeFeI//477BKLBzA=="],
|
||||
|
||||
"semver": ["semver@7.7.3", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q=="],
|
||||
"semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="],
|
||||
|
||||
"sharp": ["sharp@0.34.5", "", { "dependencies": { "@img/colour": "^1.0.0", "detect-libc": "^2.1.2", "semver": "^7.7.3" }, "optionalDependencies": { "@img/sharp-darwin-arm64": "0.34.5", "@img/sharp-darwin-x64": "0.34.5", "@img/sharp-libvips-darwin-arm64": "1.2.4", "@img/sharp-libvips-darwin-x64": "1.2.4", "@img/sharp-libvips-linux-arm": "1.2.4", "@img/sharp-libvips-linux-arm64": "1.2.4", "@img/sharp-libvips-linux-ppc64": "1.2.4", "@img/sharp-libvips-linux-riscv64": "1.2.4", "@img/sharp-libvips-linux-s390x": "1.2.4", "@img/sharp-libvips-linux-x64": "1.2.4", "@img/sharp-libvips-linuxmusl-arm64": "1.2.4", "@img/sharp-libvips-linuxmusl-x64": "1.2.4", "@img/sharp-linux-arm": "0.34.5", "@img/sharp-linux-arm64": "0.34.5", "@img/sharp-linux-ppc64": "0.34.5", "@img/sharp-linux-riscv64": "0.34.5", "@img/sharp-linux-s390x": "0.34.5", "@img/sharp-linux-x64": "0.34.5", "@img/sharp-linuxmusl-arm64": "0.34.5", "@img/sharp-linuxmusl-x64": "0.34.5", "@img/sharp-wasm32": "0.34.5", "@img/sharp-win32-arm64": "0.34.5", "@img/sharp-win32-ia32": "0.34.5", "@img/sharp-win32-x64": "0.34.5" } }, "sha512-Ou9I5Ft9WNcCbXrU9cMgPBcCK8LiwLqcbywW3t4oDV37n1pzpuNLsYiAV8eODnjbtQlSDwZ2cUEeQz4E54Hltg=="],
|
||||
|
||||
"shiki": ["shiki@3.22.0", "", { "dependencies": { "@shikijs/core": "3.22.0", "@shikijs/engine-javascript": "3.22.0", "@shikijs/engine-oniguruma": "3.22.0", "@shikijs/langs": "3.22.0", "@shikijs/themes": "3.22.0", "@shikijs/types": "3.22.0", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-LBnhsoYEe0Eou4e1VgJACes+O6S6QC0w71fCSp5Oya79inkwkm15gQ1UF6VtQ8j/taMDh79hAB49WUk8ALQW3g=="],
|
||||
"shiki": ["shiki@4.0.2", "", { "dependencies": { "@shikijs/core": "4.0.2", "@shikijs/engine-javascript": "4.0.2", "@shikijs/engine-oniguruma": "4.0.2", "@shikijs/langs": "4.0.2", "@shikijs/themes": "4.0.2", "@shikijs/types": "4.0.2", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-eAVKTMedR5ckPo4xne/PjYQYrU3qx78gtJZ+sHlXEg5IHhhoQhMfZVzetTYuaJS0L2Ef3AcCRzCHV8T0WI6nIQ=="],
|
||||
|
||||
"simple-icons": ["simple-icons@16.10.0", "", {}, "sha512-62kuxaG3pE+cFNerudUtwb9BLmudzayHrlHkvU9gf8Nxcj7VYOm9dh3WNbGaFk60aQtfnRyzViZaouFG2B45kg=="],
|
||||
"simple-icons": ["simple-icons@16.18.0", "", {}, "sha512-5KbjcP456Gm1lrk+rhDuX4zFri+3lRX39IjzXAvoMAO8Ne76WlVlM+Z3kA6jdZ7+QHadgXsf++R7g2jaYVbYig=="],
|
||||
|
||||
"sisteransi": ["sisteransi@1.0.5", "", {}, "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg=="],
|
||||
|
||||
@ -620,19 +604,17 @@
|
||||
|
||||
"space-separated-tokens": ["space-separated-tokens@2.0.2", "", {}, "sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q=="],
|
||||
|
||||
"string-width": ["string-width@7.2.0", "", { "dependencies": { "emoji-regex": "^10.3.0", "get-east-asian-width": "^1.0.0", "strip-ansi": "^7.1.0" } }, "sha512-tsaTIkKW9b4N+AEj+SVA+WhJzV7/zMhcSu78mLKWSk7cXMOSHsBKFWUs0fWwq8QyK3MgJBQRX6Gbi4kYbdvGkQ=="],
|
||||
|
||||
"stringify-entities": ["stringify-entities@4.0.4", "", { "dependencies": { "character-entities-html4": "^2.0.0", "character-entities-legacy": "^3.0.0" } }, "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg=="],
|
||||
|
||||
"strip-ansi": ["strip-ansi@7.1.2", "", { "dependencies": { "ansi-regex": "^6.0.1" } }, "sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA=="],
|
||||
"strnum": ["strnum@2.2.3", "", {}, "sha512-oKx6RUCuHfT3oyVjtnrmn19H1SiCqgJSg+54XqURKp5aCMbrXrhLjRN9TjuwMjiYstZ0MzDrHqkGZ5dFTKd+zg=="],
|
||||
|
||||
"strnum": ["strnum@2.1.2", "", {}, "sha512-l63NF9y/cLROq/yqKXSLtcMeeyOfnSQlfMSlzFt/K73oIaD8DGaQWd7Z34X9GPiKqP5rbSh84Hl4bOlLcjiSrQ=="],
|
||||
|
||||
"svgo": ["svgo@4.0.0", "", { "dependencies": { "commander": "^11.1.0", "css-select": "^5.1.0", "css-tree": "^3.0.1", "css-what": "^6.1.0", "csso": "^5.0.5", "picocolors": "^1.1.1", "sax": "^1.4.1" }, "bin": "./bin/svgo.js" }, "sha512-VvrHQ+9uniE+Mvx3+C9IEe/lWasXCU0nXMY2kZeLrHNICuRiC8uMPyM14UEaMOFA5mhyQqEkB02VoQ16n3DLaw=="],
|
||||
"svgo": ["svgo@4.0.1", "", { "dependencies": { "commander": "^11.1.0", "css-select": "^5.1.0", "css-tree": "^3.0.1", "css-what": "^6.1.0", "csso": "^5.0.5", "picocolors": "^1.1.1", "sax": "^1.5.0" }, "bin": "./bin/svgo.js" }, "sha512-XDpWUOPC6FEibaLzjfe0ucaV0YrOjYotGJO1WpF0Zd+n6ZGEQUsSugaoLq9QkEZtAfQIxT42UChcssDVPP3+/w=="],
|
||||
|
||||
"tiny-inflate": ["tiny-inflate@1.0.3", "", {}, "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw=="],
|
||||
|
||||
"tinyexec": ["tinyexec@1.0.2", "", {}, "sha512-W/KYk+NFhkmsYpuHq5JykngiOCnxeVL8v8dFnqxSD8qEEdRfXk1SDM6JzNqcERbcGYj9tMrDQBYV9cjgnunFIg=="],
|
||||
"tinyclip": ["tinyclip@0.1.12", "", {}, "sha512-Ae3OVUqifDw0wBriIBS7yVaW44Dp6eSHQcyq4Igc7eN2TJH/2YsicswaW+J/OuMvhpDPOKEgpAZCjkb4hpoyeA=="],
|
||||
|
||||
"tinyexec": ["tinyexec@1.1.1", "", {}, "sha512-VKS/ZaQhhkKFMANmAOhhXVoIfBXblQxGX1myCQ2faQrfmobMftXeJPcZGp0gS07ocvGJWDLZGyOZDadDBqYIJg=="],
|
||||
|
||||
"tinyglobby": ["tinyglobby@0.2.15", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.3" } }, "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ=="],
|
||||
|
||||
@ -644,8 +626,6 @@
|
||||
|
||||
"tslib": ["tslib@2.8.1", "", {}, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
|
||||
|
||||
"type-fest": ["type-fest@4.41.0", "", {}, "sha512-TeTSQ6H5YHvpqVwBRcnLDCBnDOHWYu7IvGbHT6N8AOymcr9PJGjc1GTtiWZTYg0NCgYwvnYWEkVChQAr9bjfwA=="],
|
||||
|
||||
"typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="],
|
||||
|
||||
"ufo": ["ufo@1.6.3", "", {}, "sha512-yDJTmhydvl5lJzBmy/hyOAA0d+aqCBuwl818haVdYCRrWV84o7YyeVm4QlVHStqNrrJSTb6jKuFAVqAFsr+K3Q=="],
|
||||
@ -654,11 +634,9 @@
|
||||
|
||||
"uncrypto": ["uncrypto@0.1.3", "", {}, "sha512-Ql87qFHB3s/De2ClA9e0gsnS6zXG27SkTiSJwjCc9MebbfapQfuPzumMIUMi38ezPZVNFcHI9sUIepeQfw8J8Q=="],
|
||||
|
||||
"undici-types": ["undici-types@7.16.0", "", {}, "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw=="],
|
||||
|
||||
"unified": ["unified@11.0.5", "", { "dependencies": { "@types/unist": "^3.0.0", "bail": "^2.0.0", "devlop": "^1.0.0", "extend": "^3.0.0", "is-plain-obj": "^4.0.0", "trough": "^2.0.0", "vfile": "^6.0.0" } }, "sha512-xKvGhPWw3k84Qjh8bI3ZeJjqnyadK+GEFtazSfZv/rKeTkTjOJho6mFqh2SM96iIcZokxiOpg78GazTSg8+KHA=="],
|
||||
|
||||
"unifont": ["unifont@0.7.3", "", { "dependencies": { "css-tree": "^3.1.0", "ofetch": "^1.5.1", "ohash": "^2.0.11" } }, "sha512-b0GtQzKCyuSHGsfj5vyN8st7muZ6VCI4XD4vFlr7Uy1rlWVYxC3npnfk8MyreHxJYrz1ooLDqDzFe9XqQTlAhA=="],
|
||||
"unifont": ["unifont@0.7.4", "", { "dependencies": { "css-tree": "^3.1.0", "ofetch": "^1.5.1", "ohash": "^2.0.11" } }, "sha512-oHeis4/xl42HUIeHuNZRGEvxj5AaIKR+bHPNegRq5LV1gdc3jundpONbjglKpihmJf+dswygdMJn3eftGIMemg=="],
|
||||
|
||||
"unist-util-find-after": ["unist-util-find-after@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0" } }, "sha512-amQa0Ep2m6hE2g72AugUItjbuM8X8cGQnFoHk0pGfrFeT9GZhzN5SW8nRsiGKK7Aif4CrACPENkA6P/Lw6fHGQ=="],
|
||||
|
||||
@ -672,13 +650,13 @@
|
||||
|
||||
"unist-util-stringify-position": ["unist-util-stringify-position@4.0.0", "", { "dependencies": { "@types/unist": "^3.0.0" } }, "sha512-0ASV06AAoKCDkS2+xw5RXJywruurpbC4JZSm7nr7MOt1ojAzvyyaO+UxZf18j8FCF6kmzCZKcAgN/yu2gm2XgQ=="],
|
||||
|
||||
"unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
"unist-util-visit": ["unist-util-visit@5.1.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-m+vIdyeCOpdr/QeQCu2EzxX/ohgS8KbnPDgFni4dQsfSCtpz8UqDyY5GjRru8PDKuYn7Fq19j1CQ+nJSsGKOzg=="],
|
||||
|
||||
"unist-util-visit-children": ["unist-util-visit-children@3.0.0", "", { "dependencies": { "@types/unist": "^3.0.0" } }, "sha512-RgmdTfSBOg04sdPcpTSD1jzoNBjt9a80/ZCzp5cI9n1qPzLZWF9YdvWGN2zmTumP1HWhXKdUWexjy/Wy/lJ7tA=="],
|
||||
|
||||
"unist-util-visit-parents": ["unist-util-visit-parents@6.0.2", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0" } }, "sha512-goh1s1TBrqSqukSc8wrjwWhL0hiJxgA8m4kFxGlQ+8FYQ3C/m11FcTs4YYem7V664AhHVvgoQLk890Ssdsr2IQ=="],
|
||||
|
||||
"unstorage": ["unstorage@1.17.4", "", { "dependencies": { "anymatch": "^3.1.3", "chokidar": "^5.0.0", "destr": "^2.0.5", "h3": "^1.15.5", "lru-cache": "^11.2.0", "node-fetch-native": "^1.6.7", "ofetch": "^1.5.1", "ufo": "^1.6.3" }, "peerDependencies": { "@azure/app-configuration": "^1.8.0", "@azure/cosmos": "^4.2.0", "@azure/data-tables": "^13.3.0", "@azure/identity": "^4.6.0", "@azure/keyvault-secrets": "^4.9.0", "@azure/storage-blob": "^12.26.0", "@capacitor/preferences": "^6 || ^7 || ^8", "@deno/kv": ">=0.9.0", "@netlify/blobs": "^6.5.0 || ^7.0.0 || ^8.1.0 || ^9.0.0 || ^10.0.0", "@planetscale/database": "^1.19.0", "@upstash/redis": "^1.34.3", "@vercel/blob": ">=0.27.1", "@vercel/functions": "^2.2.12 || ^3.0.0", "@vercel/kv": "^1 || ^2 || ^3", "aws4fetch": "^1.0.20", "db0": ">=0.2.1", "idb-keyval": "^6.2.1", "ioredis": "^5.4.2", "uploadthing": "^7.4.4" }, "optionalPeers": ["@azure/app-configuration", "@azure/cosmos", "@azure/data-tables", "@azure/identity", "@azure/keyvault-secrets", "@azure/storage-blob", "@capacitor/preferences", "@deno/kv", "@netlify/blobs", "@planetscale/database", "@upstash/redis", "@vercel/blob", "@vercel/functions", "@vercel/kv", "aws4fetch", "db0", "idb-keyval", "ioredis", "uploadthing"] }, "sha512-fHK0yNg38tBiJKp/Vgsq4j0JEsCmgqH58HAn707S7zGkArbZsVr/CwINoi+nh3h98BRCwKvx1K3Xg9u3VV83sw=="],
|
||||
"unstorage": ["unstorage@1.17.5", "", { "dependencies": { "anymatch": "^3.1.3", "chokidar": "^5.0.0", "destr": "^2.0.5", "h3": "^1.15.10", "lru-cache": "^11.2.7", "node-fetch-native": "^1.6.7", "ofetch": "^1.5.1", "ufo": "^1.6.3" }, "peerDependencies": { "@azure/app-configuration": "^1.8.0", "@azure/cosmos": "^4.2.0", "@azure/data-tables": "^13.3.0", "@azure/identity": "^4.6.0", "@azure/keyvault-secrets": "^4.9.0", "@azure/storage-blob": "^12.26.0", "@capacitor/preferences": "^6 || ^7 || ^8", "@deno/kv": ">=0.9.0", "@netlify/blobs": "^6.5.0 || ^7.0.0 || ^8.1.0 || ^9.0.0 || ^10.0.0", "@planetscale/database": "^1.19.0", "@upstash/redis": "^1.34.3", "@vercel/blob": ">=0.27.1", "@vercel/functions": "^2.2.12 || ^3.0.0", "@vercel/kv": "^1 || ^2 || ^3", "aws4fetch": "^1.0.20", "db0": ">=0.2.1", "idb-keyval": "^6.2.1", "ioredis": "^5.4.2", "uploadthing": "^7.4.4" }, "optionalPeers": ["@azure/app-configuration", "@azure/cosmos", "@azure/data-tables", "@azure/identity", "@azure/keyvault-secrets", "@azure/storage-blob", "@capacitor/preferences", "@deno/kv", "@netlify/blobs", "@planetscale/database", "@upstash/redis", "@vercel/blob", "@vercel/functions", "@vercel/kv", "aws4fetch", "db0", "idb-keyval", "ioredis", "uploadthing"] }, "sha512-0i3iqvRfx29hkNntHyQvJTpf5W9dQ9ZadSoRU8+xVlhVtT7jAX57fazYO9EHvcRCfBCyi5YRya7XCDOsbTgkPg=="],
|
||||
|
||||
"vfile": ["vfile@6.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "vfile-message": "^4.0.0" } }, "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q=="],
|
||||
|
||||
@ -686,41 +664,25 @@
|
||||
|
||||
"vfile-message": ["vfile-message@4.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-stringify-position": "^4.0.0" } }, "sha512-QTHzsGd1EhbZs4AsQ20JX1rC3cOlt/IWJruk893DfLRr57lcnOeMaWG4K0JrRta4mIJZKth2Au3mM3u03/JWKw=="],
|
||||
|
||||
"vite": ["vite@6.4.1", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g=="],
|
||||
"vite": ["vite@7.3.2", "", { "dependencies": { "esbuild": "^0.27.0", "fdir": "^6.5.0", "picomatch": "^4.0.3", "postcss": "^8.5.6", "rollup": "^4.43.0", "tinyglobby": "^0.2.15" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^20.19.0 || >=22.12.0", "jiti": ">=1.21.0", "less": "^4.0.0", "lightningcss": "^1.21.0", "sass": "^1.70.0", "sass-embedded": "^1.70.0", "stylus": ">=0.54.8", "sugarss": "^5.0.0", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-Bby3NOsna2jsjfLVOHKes8sGwgl4TT0E6vvpYgnAYDIF/tie7MRaFthmKuHx1NSXjiTueXH3do80FMQgvEktRg=="],
|
||||
|
||||
"vitefu": ["vitefu@1.1.1", "", { "peerDependencies": { "vite": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0-beta.0" }, "optionalPeers": ["vite"] }, "sha512-B/Fegf3i8zh0yFbpzZ21amWzHmuNlLlmJT6n7bu5e+pCHUKQIfXSYokrqOBGEMMe9UG2sostKQF9mml/vYaWJQ=="],
|
||||
"vitefu": ["vitefu@1.1.3", "", { "peerDependencies": { "vite": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0" }, "optionalPeers": ["vite"] }, "sha512-ub4okH7Z5KLjb6hDyjqrGXqWtWvoYdU3IGm/NorpgHncKoLTCfRIbvlhBm7r0YstIaQRYlp4yEbFqDcKSzXSSg=="],
|
||||
|
||||
"web-namespaces": ["web-namespaces@2.0.1", "", {}, "sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ=="],
|
||||
|
||||
"which-pm-runs": ["which-pm-runs@1.1.0", "", {}, "sha512-n1brCuqClxfFfq/Rb0ICg9giSZqCS+pLtccdag6C2HyufBrh3fBOiy9nb6ggRMvWOVH5GrdJskj5iGTZNxd7SA=="],
|
||||
|
||||
"widest-line": ["widest-line@5.0.0", "", { "dependencies": { "string-width": "^7.0.0" } }, "sha512-c9bZp7b5YtRj2wOe6dlj32MK+Bx/M/d+9VB2SHM1OtsUHR0aV0tdP6DWh/iMt0kWi1t5g1Iudu6hQRNd1A4PVA=="],
|
||||
|
||||
"wrap-ansi": ["wrap-ansi@9.0.2", "", { "dependencies": { "ansi-styles": "^6.2.1", "string-width": "^7.0.0", "strip-ansi": "^7.1.0" } }, "sha512-42AtmgqjV+X1VpdOfyTGOYRi0/zsoLqtXQckTmqTeybT+BDIbM/Guxo7x3pE2vtpr1ok6xRqM9OpBe+Jyoqyww=="],
|
||||
|
||||
"xxhash-wasm": ["xxhash-wasm@1.1.0", "", {}, "sha512-147y/6YNh+tlp6nd/2pWq38i9h6mz/EuQ6njIrmW8D1BS5nCqs0P6DG+m6zTGnNz5I+uhZ0SHxBs9BsPrwcKDA=="],
|
||||
|
||||
"yargs-parser": ["yargs-parser@21.1.1", "", {}, "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw=="],
|
||||
"yargs-parser": ["yargs-parser@22.0.0", "", {}, "sha512-rwu/ClNdSMpkSrUb+d6BRsSkLUq1fmfsY6TOpYzTwvwkg1/NRG85KBy3kq++A8LKQwX6lsu+aWad+2khvuXrqw=="],
|
||||
|
||||
"yocto-queue": ["yocto-queue@1.2.2", "", {}, "sha512-4LCcse/U2MHZ63HAJVE+v71o7yOdIe4cZ70Wpf8D/IyjDKYQLV5GD46B+hSTjJsvV5PztjvHoU580EftxjDZFQ=="],
|
||||
|
||||
"yocto-spinner": ["yocto-spinner@0.2.3", "", { "dependencies": { "yoctocolors": "^2.1.1" } }, "sha512-sqBChb33loEnkoXte1bLg45bEBsOP9N1kzQh5JZNKj/0rik4zAPTNSAVPj3uQAdc6slYJ0Ksc403G2XgxsJQFQ=="],
|
||||
|
||||
"yoctocolors": ["yoctocolors@2.1.2", "", {}, "sha512-CzhO+pFNo8ajLM2d2IW/R93ipy99LWjtwblvC1RsoSUMZgyLbYFr221TnSNT7GjGdYui6P459mw9JH/g/zW2ug=="],
|
||||
|
||||
"zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="],
|
||||
|
||||
"zod-to-json-schema": ["zod-to-json-schema@3.25.1", "", { "peerDependencies": { "zod": "^3.25 || ^4" } }, "sha512-pM/SU9d3YAggzi6MtR4h7ruuQlqKtad8e9S0fmxcMi+ueAK5Korys/aWcV9LIIHTVbj01NdzxcnXSN+O74ZIVA=="],
|
||||
|
||||
"zod-to-ts": ["zod-to-ts@1.2.0", "", { "peerDependencies": { "typescript": "^4.9.4 || ^5.0.2", "zod": "^3" } }, "sha512-x30XE43V+InwGpvTySRNz9kB7qFU8DlyEy7BsSTCHPH1R0QasMmHWZDCzYm6bVXtj/9NNJAZF3jW8rzFvH5OFA=="],
|
||||
"zod": ["zod@4.3.6", "", {}, "sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg=="],
|
||||
|
||||
"zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki": ["shiki@3.20.0", "", { "dependencies": { "@shikijs/core": "3.20.0", "@shikijs/engine-javascript": "3.20.0", "@shikijs/engine-oniguruma": "3.20.0", "@shikijs/langs": "3.20.0", "@shikijs/themes": "3.20.0", "@shikijs/types": "3.20.0", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-kgCOlsnyWb+p0WU+01RjkCH+eBVsjL1jOwUYWv0YDWkM2/A46+LDKVs5yZCUXjJG6bj4ndFoAg5iLIIue6dulg=="],
|
||||
|
||||
"@rollup/pluginutils/estree-walker": ["estree-walker@2.0.2", "", {}, "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="],
|
||||
|
||||
"ansi-align/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="],
|
||||
"@rollup/pluginutils/picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||
|
||||
"anymatch/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
|
||||
|
||||
@ -728,80 +690,30 @@
|
||||
|
||||
"dom-serializer/entities": ["entities@4.5.0", "", {}, "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw=="],
|
||||
|
||||
"fontace/fontkitten": ["fontkitten@1.0.3", "", { "dependencies": { "tiny-inflate": "^1.0.3" } }, "sha512-Wp1zXWPVUPBmfoa3Cqc9ctaKuzKAV6uLstRqlR56kSjplf5uAce+qeyYym7F+PHbGTk+tCEdkCW6RD7DX/gBZw=="],
|
||||
|
||||
"hast-util-raw/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"is-inside-container/is-docker": ["is-docker@3.0.0", "", { "bin": { "is-docker": "cli.js" } }, "sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ=="],
|
||||
|
||||
"mdast-util-definitions/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"mdast-util-to-hast/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"mdast-util-to-markdown/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"ofetch/ufo": ["ufo@1.6.2", "", {}, "sha512-heMioaxBcG9+Znsda5Q8sQbWnLJSl98AFDXTO80wELWEzX3hordXsTdxrIfMQoO9IY1MEnoGoPjpoKpMj+Yx0Q=="],
|
||||
|
||||
"vite/esbuild": ["esbuild@0.25.12", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.25.12", "@esbuild/android-arm": "0.25.12", "@esbuild/android-arm64": "0.25.12", "@esbuild/android-x64": "0.25.12", "@esbuild/darwin-arm64": "0.25.12", "@esbuild/darwin-x64": "0.25.12", "@esbuild/freebsd-arm64": "0.25.12", "@esbuild/freebsd-x64": "0.25.12", "@esbuild/linux-arm": "0.25.12", "@esbuild/linux-arm64": "0.25.12", "@esbuild/linux-ia32": "0.25.12", "@esbuild/linux-loong64": "0.25.12", "@esbuild/linux-mips64el": "0.25.12", "@esbuild/linux-ppc64": "0.25.12", "@esbuild/linux-riscv64": "0.25.12", "@esbuild/linux-s390x": "0.25.12", "@esbuild/linux-x64": "0.25.12", "@esbuild/netbsd-arm64": "0.25.12", "@esbuild/netbsd-x64": "0.25.12", "@esbuild/openbsd-arm64": "0.25.12", "@esbuild/openbsd-x64": "0.25.12", "@esbuild/openharmony-arm64": "0.25.12", "@esbuild/sunos-x64": "0.25.12", "@esbuild/win32-arm64": "0.25.12", "@esbuild/win32-ia32": "0.25.12", "@esbuild/win32-x64": "0.25.12" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg=="],
|
||||
"remark-smartypants/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki/@shikijs/core": ["@shikijs/core@3.20.0", "", { "dependencies": { "@shikijs/types": "3.20.0", "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4", "hast-util-to-html": "^9.0.5" } }, "sha512-f2ED7HYV4JEk827mtMDwe/yQ25pRiXZmtHjWF8uzZKuKiEsJR7Ce1nuQ+HhV9FzDcbIo4ObBCD9GPTzNuy9S1g=="],
|
||||
"retext-smartypants/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki/@shikijs/engine-javascript": ["@shikijs/engine-javascript@3.20.0", "", { "dependencies": { "@shikijs/types": "3.20.0", "@shikijs/vscode-textmate": "^10.0.2", "oniguruma-to-es": "^4.3.4" } }, "sha512-OFx8fHAZuk7I42Z9YAdZ95To6jDePQ9Rnfbw9uSRTSbBhYBp1kEOKv/3jOimcj3VRUKusDYM6DswLauwfhboLg=="],
|
||||
"sharp/semver": ["semver@7.7.3", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki/@shikijs/engine-oniguruma": ["@shikijs/engine-oniguruma@3.20.0", "", { "dependencies": { "@shikijs/types": "3.20.0", "@shikijs/vscode-textmate": "^10.0.2" } }, "sha512-Yx3gy7xLzM0ZOjqoxciHjA7dAt5tyzJE3L4uQoM83agahy+PlW244XJSrmJRSBvGYELDhYXPacD4R/cauV5bzQ=="],
|
||||
"tinyglobby/picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki/@shikijs/langs": ["@shikijs/langs@3.20.0", "", { "dependencies": { "@shikijs/types": "3.20.0" } }, "sha512-le+bssCxcSHrygCWuOrYJHvjus6zhQ2K7q/0mgjiffRbkhM4o1EWu2m+29l0yEsHDbWaWPNnDUTRVVBvBBeKaA=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki/@shikijs/themes": ["@shikijs/themes@3.20.0", "", { "dependencies": { "@shikijs/types": "3.20.0" } }, "sha512-U1NSU7Sl26Q7ErRvJUouArxfM2euWqq1xaSrbqMu2iqa+tSp0D1Yah8216sDYbdDHw4C8b75UpE65eWorm2erQ=="],
|
||||
|
||||
"@astrojs/markdown-remark/shiki/@shikijs/types": ["@shikijs/types@3.20.0", "", { "dependencies": { "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-lhYAATn10nkZcBQ0BlzSbJA3wcmL5MXUUF8d2Zzon6saZDlToKaiRX60n2+ZaHJCmXEcZRWNzn+k9vplr8Jhsw=="],
|
||||
|
||||
"ansi-align/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="],
|
||||
|
||||
"ansi-align/string-width/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="],
|
||||
"unist-util-remove-position/unist-util-visit": ["unist-util-visit@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg=="],
|
||||
|
||||
"csso/css-tree/mdn-data": ["mdn-data@2.0.28", "", {}, "sha512-aylIc7Z9y4yzHYAJNuESG3hfhC+0Ibp/MAMiaOZgNv4pmEdFyfZhhhny4MNiAfWdBQ1RQ2mfDWmM1x8SvGyp8g=="],
|
||||
|
||||
"vite/esbuild/@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.25.12", "", { "os": "aix", "cpu": "ppc64" }, "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA=="],
|
||||
|
||||
"vite/esbuild/@esbuild/android-arm": ["@esbuild/android-arm@0.25.12", "", { "os": "android", "cpu": "arm" }, "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/android-arm64": ["@esbuild/android-arm64@0.25.12", "", { "os": "android", "cpu": "arm64" }, "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/android-x64": ["@esbuild/android-x64@0.25.12", "", { "os": "android", "cpu": "x64" }, "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.25.12", "", { "os": "darwin", "cpu": "arm64" }, "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.25.12", "", { "os": "darwin", "cpu": "x64" }, "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA=="],
|
||||
|
||||
"vite/esbuild/@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.25.12", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.25.12", "", { "os": "freebsd", "cpu": "x64" }, "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-arm": ["@esbuild/linux-arm@0.25.12", "", { "os": "linux", "cpu": "arm" }, "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.25.12", "", { "os": "linux", "cpu": "arm64" }, "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.25.12", "", { "os": "linux", "cpu": "ia32" }, "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.25.12", "", { "os": "linux", "cpu": "ppc64" }, "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.25.12", "", { "os": "linux", "cpu": "s390x" }, "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/linux-x64": ["@esbuild/linux-x64@0.25.12", "", { "os": "linux", "cpu": "x64" }, "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw=="],
|
||||
|
||||
"vite/esbuild/@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.25.12", "", { "os": "none", "cpu": "x64" }, "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ=="],
|
||||
|
||||
"vite/esbuild/@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.25.12", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A=="],
|
||||
|
||||
"vite/esbuild/@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.25.12", "", { "os": "openbsd", "cpu": "x64" }, "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw=="],
|
||||
|
||||
"vite/esbuild/@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.25.12", "", { "os": "sunos", "cpu": "x64" }, "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w=="],
|
||||
|
||||
"vite/esbuild/@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.25.12", "", { "os": "win32", "cpu": "arm64" }, "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg=="],
|
||||
|
||||
"vite/esbuild/@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.25.12", "", { "os": "win32", "cpu": "ia32" }, "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ=="],
|
||||
|
||||
"vite/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.25.12", "", { "os": "win32", "cpu": "x64" }, "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA=="],
|
||||
|
||||
"ansi-align/string-width/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="],
|
||||
}
|
||||
}
|
||||
|
||||
12
package.json
12
package.json
@ -1,5 +1,5 @@
|
||||
{
|
||||
"name": "clawd-bot-landing",
|
||||
"name": "openclaw-ai",
|
||||
"type": "module",
|
||||
"version": "1.0.0",
|
||||
"scripts": {
|
||||
@ -14,12 +14,12 @@
|
||||
]
|
||||
},
|
||||
"dependencies": {
|
||||
"@astrojs/rss": "^4.0.15",
|
||||
"@lucide/astro": "^0.577.0",
|
||||
"@vercel/analytics": "^1.6.1",
|
||||
"astro": "^5.18.0",
|
||||
"@astrojs/rss": "^4.0.18",
|
||||
"@lucide/astro": "^1.11.0",
|
||||
"@vercel/analytics": "^2.0.1",
|
||||
"astro": "^6.1.9",
|
||||
"js-yaml": "^4.1.1",
|
||||
"simple-icons": "^16.10.0"
|
||||
"simple-icons": "^16.18.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/js-yaml": "^4.0.9"
|
||||
|
||||
@ -4,6 +4,31 @@ set -euo pipefail
|
||||
# OpenClaw CLI installer (non-interactive, no onboarding)
|
||||
# Usage: curl -fsSL --proto '=https' --tlsv1.2 https://openclaw.ai/install-cli.sh | bash -s -- [--json] [--prefix <path>] [--version <ver>] [--node-version <ver>] [--onboard]
|
||||
|
||||
ensure_home_env() {
|
||||
if [[ -n "${HOME:-}" && "${HOME}" != "/" && -d "${HOME}" ]]; then
|
||||
return 0
|
||||
fi
|
||||
|
||||
local user_name=""
|
||||
local home_dir=""
|
||||
user_name="$(id -un 2>/dev/null || true)"
|
||||
|
||||
if [[ -n "$user_name" ]]; then
|
||||
if command -v getent >/dev/null 2>&1; then
|
||||
home_dir="$(getent passwd "$user_name" 2>/dev/null | awk -F: '{print $6; exit}' || true)"
|
||||
fi
|
||||
if [[ -z "$home_dir" && "$(uname -s 2>/dev/null || true)" == "Darwin" ]] && command -v dscl >/dev/null 2>&1; then
|
||||
home_dir="$(dscl . -read "/Users/${user_name}" NFSHomeDirectory 2>/dev/null | awk '{print $2; exit}' || true)"
|
||||
fi
|
||||
fi
|
||||
|
||||
if [[ -n "$home_dir" && "$home_dir" != "/" && -d "$home_dir" ]]; then
|
||||
export HOME="$home_dir"
|
||||
fi
|
||||
}
|
||||
|
||||
ensure_home_env
|
||||
|
||||
PREFIX="${OPENCLAW_PREFIX:-${HOME}/.openclaw}"
|
||||
OPENCLAW_VERSION="${OPENCLAW_VERSION:-latest}"
|
||||
NODE_VERSION="${OPENCLAW_NODE_VERSION:-22.22.0}"
|
||||
@ -475,21 +500,22 @@ install_openclaw() {
|
||||
fi
|
||||
|
||||
if [[ "${requested}" == "latest" ]]; then
|
||||
if ! SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$PREFIX" "${npm_args[@]}" "openclaw@latest"; then
|
||||
if ! SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$(node_dir)" "${npm_args[@]}" "openclaw@latest"; then
|
||||
log "npm install openclaw@latest failed; retrying openclaw@next"
|
||||
emit_json "{\"event\":\"step\",\"name\":\"openclaw\",\"status\":\"retry\",\"version\":\"next\"}"
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$PREFIX" "${npm_args[@]}" "openclaw@next"
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$(node_dir)" "${npm_args[@]}" "openclaw@next"
|
||||
requested="next"
|
||||
fi
|
||||
else
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$PREFIX" "${npm_args[@]}" "openclaw@${requested}"
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$(node_dir)" "${npm_args[@]}" "openclaw@${requested}"
|
||||
fi
|
||||
|
||||
mkdir -p "${PREFIX}/bin"
|
||||
rm -f "${PREFIX}/bin/openclaw"
|
||||
cat > "${PREFIX}/bin/openclaw" <<EOF
|
||||
#!/usr/bin/env bash
|
||||
set -euo pipefail
|
||||
exec "${PREFIX}/tools/node/bin/node" "${PREFIX}/lib/node_modules/openclaw/dist/entry.js" "\$@"
|
||||
exec "${PREFIX}/tools/node/bin/node" "$(node_dir)/lib/node_modules/openclaw/dist/entry.js" "\$@"
|
||||
EOF
|
||||
chmod +x "${PREFIX}/bin/openclaw"
|
||||
emit_json "{\"event\":\"step\",\"name\":\"openclaw\",\"status\":\"ok\",\"version\":\"${requested}\"}"
|
||||
|
||||
@ -14,6 +14,29 @@ param(
|
||||
|
||||
$ErrorActionPreference = "Stop"
|
||||
|
||||
$script:InstallExitCode = 0
|
||||
|
||||
function Fail-Install {
|
||||
param([int]$Code = 1)
|
||||
|
||||
$script:InstallExitCode = $Code
|
||||
return $false
|
||||
}
|
||||
|
||||
function Complete-Install {
|
||||
param([bool]$Succeeded)
|
||||
|
||||
if ($Succeeded) {
|
||||
return
|
||||
}
|
||||
|
||||
if ($PSCommandPath) {
|
||||
exit $script:InstallExitCode
|
||||
}
|
||||
|
||||
throw "OpenClaw installation failed with exit code $($script:InstallExitCode)."
|
||||
}
|
||||
|
||||
Write-Host ""
|
||||
Write-Host " OpenClaw Installer" -ForegroundColor Cyan
|
||||
Write-Host ""
|
||||
@ -21,7 +44,8 @@ Write-Host ""
|
||||
# Check if running in PowerShell
|
||||
if ($PSVersionTable.PSVersion.Major -lt 5) {
|
||||
Write-Host "Error: PowerShell 5+ required" -ForegroundColor Red
|
||||
exit 1
|
||||
Complete-Install -Succeeded:$false
|
||||
return
|
||||
}
|
||||
|
||||
Write-Host "[OK] Windows detected" -ForegroundColor Green
|
||||
@ -91,11 +115,11 @@ function Install-Node {
|
||||
$env:Path = [System.Environment]::GetEnvironmentVariable("Path","Machine") + ";" + [System.Environment]::GetEnvironmentVariable("Path","User")
|
||||
if (Check-Node) {
|
||||
Write-Host "[OK] Node.js installed via winget" -ForegroundColor Green
|
||||
return
|
||||
return $true
|
||||
}
|
||||
Write-Host "[!] winget completed, but Node.js is still unavailable in this shell" -ForegroundColor Yellow
|
||||
Write-Host "Restart PowerShell and re-run the installer if Node.js was installed successfully." -ForegroundColor Yellow
|
||||
exit 1
|
||||
return $false
|
||||
}
|
||||
|
||||
# Try Chocolatey
|
||||
@ -106,7 +130,7 @@ function Install-Node {
|
||||
# Refresh PATH
|
||||
$env:Path = [System.Environment]::GetEnvironmentVariable("Path","Machine") + ";" + [System.Environment]::GetEnvironmentVariable("Path","User")
|
||||
Write-Host "[OK] Node.js installed via Chocolatey" -ForegroundColor Green
|
||||
return
|
||||
return $true
|
||||
}
|
||||
|
||||
# Try Scoop
|
||||
@ -114,7 +138,7 @@ function Install-Node {
|
||||
Write-Host " Using Scoop..." -ForegroundColor Gray
|
||||
scoop install nodejs-lts
|
||||
Write-Host "[OK] Node.js installed via Scoop" -ForegroundColor Green
|
||||
return
|
||||
return $true
|
||||
}
|
||||
|
||||
# Manual download fallback
|
||||
@ -125,7 +149,7 @@ function Install-Node {
|
||||
Write-Host " https://nodejs.org/en/download/" -ForegroundColor Cyan
|
||||
Write-Host ""
|
||||
Write-Host "Or install winget (App Installer) from the Microsoft Store." -ForegroundColor Gray
|
||||
exit 1
|
||||
return $false
|
||||
}
|
||||
|
||||
# Check for existing OpenClaw installation
|
||||
@ -281,12 +305,12 @@ function Install-PortableGit {
|
||||
}
|
||||
|
||||
function Ensure-Git {
|
||||
if (Check-Git) { return }
|
||||
if (Use-PortableGitIfPresent) { return }
|
||||
if (Check-Git) { return $true }
|
||||
if (Use-PortableGitIfPresent) { return $true }
|
||||
try {
|
||||
Install-PortableGit
|
||||
if (Check-Git) {
|
||||
return
|
||||
return $true
|
||||
}
|
||||
} catch {
|
||||
Write-Host "[!] Portable Git bootstrap failed: $($_.Exception.Message)" -ForegroundColor Yellow
|
||||
@ -297,7 +321,7 @@ function Ensure-Git {
|
||||
Write-Host "Auto-bootstrap of user-local Git did not succeed." -ForegroundColor Yellow
|
||||
Write-Host "Install Git for Windows manually, then re-run this installer:" -ForegroundColor Yellow
|
||||
Write-Host " https://git-scm.com/download/win" -ForegroundColor Cyan
|
||||
exit 1
|
||||
return $false
|
||||
}
|
||||
|
||||
function Get-OpenClawCommandPath {
|
||||
@ -446,18 +470,46 @@ function Ensure-Pnpm {
|
||||
}
|
||||
|
||||
# Install OpenClaw
|
||||
function Resolve-NpmOpenClawInstallSpec {
|
||||
param(
|
||||
[string]$PackageName,
|
||||
[string]$RequestedTag
|
||||
)
|
||||
|
||||
if ([string]::IsNullOrWhiteSpace($RequestedTag)) {
|
||||
return "$PackageName@latest"
|
||||
}
|
||||
|
||||
$trimmedTag = $RequestedTag.Trim()
|
||||
if (
|
||||
$trimmedTag -match '^(https?|file):' -or
|
||||
$trimmedTag -match '^(git\+|github:)' -or
|
||||
$trimmedTag -match '^[A-Za-z]:[\\/]' -or
|
||||
$trimmedTag -match '^\\\\' -or
|
||||
$trimmedTag -match '^\.\.?[\\/]' -or
|
||||
$trimmedTag -match '\.tgz($|[?#])'
|
||||
) {
|
||||
return $trimmedTag
|
||||
}
|
||||
|
||||
return "$PackageName@$trimmedTag"
|
||||
}
|
||||
|
||||
function Install-OpenClaw {
|
||||
if ([string]::IsNullOrWhiteSpace($Tag)) {
|
||||
$Tag = "latest"
|
||||
}
|
||||
Ensure-Git
|
||||
if (-not (Ensure-Git)) {
|
||||
return $false
|
||||
}
|
||||
|
||||
# Use openclaw package for beta, openclaw for stable
|
||||
$packageName = "openclaw"
|
||||
if ($Tag -eq "beta" -or $Tag -match "^beta\.") {
|
||||
$packageName = "openclaw"
|
||||
}
|
||||
Write-Host "[*] Installing OpenClaw ($packageName@$Tag)..." -ForegroundColor Yellow
|
||||
$installSpec = Resolve-NpmOpenClawInstallSpec -PackageName $packageName -RequestedTag $Tag
|
||||
Write-Host "[*] Installing OpenClaw ($installSpec)..." -ForegroundColor Yellow
|
||||
$prevLogLevel = $env:NPM_CONFIG_LOGLEVEL
|
||||
$prevUpdateNotifier = $env:NPM_CONFIG_UPDATE_NOTIFIER
|
||||
$prevFund = $env:NPM_CONFIG_FUND
|
||||
@ -471,7 +523,7 @@ function Install-OpenClaw {
|
||||
$env:NPM_CONFIG_SCRIPT_SHELL = "cmd.exe"
|
||||
$env:NODE_LLAMA_CPP_SKIP_DOWNLOAD = "1"
|
||||
try {
|
||||
$npmOutput = & (Get-NpmCommandPath) install -g "$packageName@$Tag" 2>&1
|
||||
$npmOutput = & (Get-NpmCommandPath) install -g "$installSpec" 2>&1
|
||||
if ($LASTEXITCODE -ne 0) {
|
||||
Write-Host "[!] npm install failed" -ForegroundColor Red
|
||||
if ($npmOutput -match "spawn git" -or $npmOutput -match "ENOENT.*git") {
|
||||
@ -483,7 +535,7 @@ function Install-OpenClaw {
|
||||
Write-Host ' powershell -c "irm https://openclaw.ai/install.ps1 | iex"' -ForegroundColor Cyan
|
||||
}
|
||||
$npmOutput | ForEach-Object { Write-Host $_ }
|
||||
exit 1
|
||||
return $false
|
||||
}
|
||||
} finally {
|
||||
$env:NPM_CONFIG_LOGLEVEL = $prevLogLevel
|
||||
@ -494,6 +546,7 @@ function Install-OpenClaw {
|
||||
$env:NODE_LLAMA_CPP_SKIP_DOWNLOAD = $prevNodeLlamaSkipDownload
|
||||
}
|
||||
Write-Host "[OK] OpenClaw installed" -ForegroundColor Green
|
||||
return $true
|
||||
}
|
||||
|
||||
# Install OpenClaw from GitHub
|
||||
@ -502,7 +555,9 @@ function Install-OpenClawFromGit {
|
||||
[string]$RepoDir,
|
||||
[switch]$SkipUpdate
|
||||
)
|
||||
Ensure-Git
|
||||
if (-not (Ensure-Git)) {
|
||||
return $false
|
||||
}
|
||||
Ensure-Pnpm
|
||||
|
||||
$repoUrl = "https://github.com/openclaw/openclaw.git"
|
||||
@ -557,6 +612,7 @@ function Install-OpenClawFromGit {
|
||||
|
||||
Write-Host "[OK] OpenClaw wrapper installed to $cmdPath" -ForegroundColor Green
|
||||
Write-Host "[i] This checkout uses pnpm. For deps, run: pnpm install (avoid npm install in the repo)." -ForegroundColor Gray
|
||||
return $true
|
||||
}
|
||||
|
||||
# Run doctor for migrations (safe, non-interactive)
|
||||
@ -637,7 +693,7 @@ function Remove-LegacySubmodule {
|
||||
function Main {
|
||||
if ($InstallMethod -ne "npm" -and $InstallMethod -ne "git") {
|
||||
Write-Host "Error: invalid -InstallMethod (use npm or git)." -ForegroundColor Red
|
||||
exit 2
|
||||
return (Fail-Install -Code 2)
|
||||
}
|
||||
|
||||
if ($DryRun) {
|
||||
@ -654,7 +710,7 @@ function Main {
|
||||
if ($NoOnboard) {
|
||||
Write-Host "[OK] Onboard: skipped" -ForegroundColor Green
|
||||
}
|
||||
return
|
||||
return $true
|
||||
}
|
||||
|
||||
Remove-LegacySubmodule -RepoDir $RepoDir
|
||||
@ -664,14 +720,16 @@ function Main {
|
||||
|
||||
# Step 1: Node.js
|
||||
if (-not (Check-Node)) {
|
||||
Install-Node
|
||||
if (-not (Install-Node)) {
|
||||
return (Fail-Install)
|
||||
}
|
||||
|
||||
# Verify installation
|
||||
if (-not (Check-Node)) {
|
||||
Write-Host ""
|
||||
Write-Host "Error: Node.js installation may require a terminal restart" -ForegroundColor Red
|
||||
Write-Host "Please close this terminal, open a new one, and run this installer again." -ForegroundColor Yellow
|
||||
exit 1
|
||||
return (Fail-Install)
|
||||
}
|
||||
}
|
||||
|
||||
@ -679,10 +737,26 @@ function Main {
|
||||
|
||||
# Step 2: OpenClaw
|
||||
if ($InstallMethod -eq "git") {
|
||||
try {
|
||||
$npmCommand = Get-NpmCommandPath
|
||||
if ($npmCommand) {
|
||||
& $npmCommand uninstall -g openclaw 2>$null | Out-Null
|
||||
Write-Host "[OK] Removed npm global install if present" -ForegroundColor Green
|
||||
}
|
||||
} catch { }
|
||||
$finalGitDir = $GitDir
|
||||
Install-OpenClawFromGit -RepoDir $GitDir -SkipUpdate:$NoGitUpdate
|
||||
if (-not (Install-OpenClawFromGit -RepoDir $GitDir -SkipUpdate:$NoGitUpdate)) {
|
||||
return (Fail-Install)
|
||||
}
|
||||
} else {
|
||||
Install-OpenClaw
|
||||
$gitWrapper = Join-Path (Join-Path $env:USERPROFILE ".local\\bin") "openclaw.cmd"
|
||||
if (Test-Path $gitWrapper) {
|
||||
Remove-Item -Force $gitWrapper
|
||||
Write-Host "[OK] Removed git wrapper (switching to npm)" -ForegroundColor Green
|
||||
}
|
||||
if (-not (Install-OpenClaw)) {
|
||||
return (Fail-Install)
|
||||
}
|
||||
}
|
||||
|
||||
if (-not (Ensure-OpenClawOnPath)) {
|
||||
@ -785,6 +859,10 @@ function Main {
|
||||
Invoke-OpenClawCommand onboard
|
||||
}
|
||||
}
|
||||
|
||||
return $true
|
||||
}
|
||||
|
||||
Main
|
||||
$mainResults = @(Main)
|
||||
$installSucceeded = $mainResults.Count -gt 0 -and $mainResults[-1] -eq $true
|
||||
Complete-Install -Succeeded:$installSucceeded
|
||||
|
||||
@ -19,6 +19,31 @@ DEFAULT_TAGLINE="All your chats, one OpenClaw."
|
||||
|
||||
ORIGINAL_PATH="${PATH:-}"
|
||||
|
||||
ensure_home_env() {
|
||||
if [[ -n "${HOME:-}" && "${HOME}" != "/" && -d "${HOME}" ]]; then
|
||||
return 0
|
||||
fi
|
||||
|
||||
local user_name=""
|
||||
local home_dir=""
|
||||
user_name="$(id -un 2>/dev/null || true)"
|
||||
|
||||
if [[ -n "$user_name" ]]; then
|
||||
if command -v getent >/dev/null 2>&1; then
|
||||
home_dir="$(getent passwd "$user_name" 2>/dev/null | awk -F: '{print $6; exit}' || true)"
|
||||
fi
|
||||
if [[ -z "$home_dir" && "$(uname -s 2>/dev/null || true)" == "Darwin" ]] && command -v dscl >/dev/null 2>&1; then
|
||||
home_dir="$(dscl . -read "/Users/${user_name}" NFSHomeDirectory 2>/dev/null | awk '{print $2; exit}' || true)"
|
||||
fi
|
||||
fi
|
||||
|
||||
if [[ -n "$home_dir" && "$home_dir" != "/" && -d "$home_dir" ]]; then
|
||||
export HOME="$home_dir"
|
||||
fi
|
||||
}
|
||||
|
||||
ensure_home_env
|
||||
|
||||
TMPFILES=()
|
||||
cleanup_tmpfiles() {
|
||||
local f
|
||||
@ -993,6 +1018,9 @@ SHARP_IGNORE_GLOBAL_LIBVIPS="${SHARP_IGNORE_GLOBAL_LIBVIPS:-1}"
|
||||
NPM_LOGLEVEL="${OPENCLAW_NPM_LOGLEVEL:-error}"
|
||||
NPM_SILENT_FLAG="--silent"
|
||||
VERBOSE="${OPENCLAW_VERBOSE:-0}"
|
||||
INSTALL_PROFILE="${OPENCLAW_PROFILE:-}"
|
||||
INSTALL_WORKSPACE="${OPENCLAW_WORKSPACE:-}"
|
||||
INSTALL_GATEWAY_PORT="${OPENCLAW_GATEWAY_PORT:-}"
|
||||
OPENCLAW_BIN=""
|
||||
SELECTED_NODE_BIN=""
|
||||
PNPM_CMD=()
|
||||
@ -1012,6 +1040,9 @@ Options:
|
||||
--version <version|dist-tag> npm install: version (default: latest)
|
||||
--beta Use beta if available, else latest
|
||||
--git-dir, --dir <path> Checkout directory (default: ~/openclaw)
|
||||
--profile <name> Isolated OpenClaw profile for config/state/service naming
|
||||
--workspace <dir> Workspace to use during onboarding
|
||||
--gateway-port <port> Gateway port to write during onboarding
|
||||
--no-git-update Skip git pull for existing checkout
|
||||
--no-onboard Skip onboarding (non-interactive)
|
||||
--no-prompt Disable prompts (required in CI/automation)
|
||||
@ -1025,6 +1056,9 @@ Environment variables:
|
||||
OPENCLAW_BETA=0|1
|
||||
OPENCLAW_GIT_DIR=...
|
||||
OPENCLAW_GIT_UPDATE=0|1
|
||||
OPENCLAW_PROFILE=<name>
|
||||
OPENCLAW_WORKSPACE=<dir>
|
||||
OPENCLAW_GATEWAY_PORT=<port>
|
||||
OPENCLAW_NO_PROMPT=1
|
||||
OPENCLAW_DRY_RUN=1
|
||||
OPENCLAW_NO_ONBOARD=1
|
||||
@ -1036,6 +1070,7 @@ Examples:
|
||||
curl -fsSL --proto '=https' --tlsv1.2 https://openclaw.ai/install.sh | bash
|
||||
curl -fsSL --proto '=https' --tlsv1.2 https://openclaw.ai/install.sh | bash -s -- --no-onboard
|
||||
curl -fsSL --proto '=https' --tlsv1.2 https://openclaw.ai/install.sh | bash -s -- --install-method git --no-onboard
|
||||
curl -fsSL --proto '=https' --tlsv1.2 https://openclaw.ai/install.sh | OPENCLAW_PROFILE=rescue bash -s -- --profile rescue --gateway-port 19001 --workspace ~/openclaw-rescue
|
||||
EOF
|
||||
}
|
||||
|
||||
@ -1090,6 +1125,18 @@ parse_args() {
|
||||
GIT_DIR="$2"
|
||||
shift 2
|
||||
;;
|
||||
--profile)
|
||||
INSTALL_PROFILE="$2"
|
||||
shift 2
|
||||
;;
|
||||
--workspace)
|
||||
INSTALL_WORKSPACE="$2"
|
||||
shift 2
|
||||
;;
|
||||
--gateway-port)
|
||||
INSTALL_GATEWAY_PORT="$2"
|
||||
shift 2
|
||||
;;
|
||||
--no-git-update)
|
||||
GIT_UPDATE=0
|
||||
shift
|
||||
@ -1896,6 +1943,148 @@ npm_global_bin_dir() {
|
||||
return 1
|
||||
}
|
||||
|
||||
canonicalize_dir() {
|
||||
local dir="$1"
|
||||
if [[ -z "$dir" || ! -d "$dir" ]]; then
|
||||
return 1
|
||||
fi
|
||||
(cd "$dir" 2>/dev/null && pwd -P) || return 1
|
||||
}
|
||||
|
||||
openclaw_package_version() {
|
||||
local package_json="$1"
|
||||
if [[ ! -f "$package_json" ]]; then
|
||||
echo "unknown"
|
||||
return 0
|
||||
fi
|
||||
|
||||
local version=""
|
||||
if command -v node >/dev/null 2>&1; then
|
||||
version="$(node -e 'const fs = require("fs"); const pkg = JSON.parse(fs.readFileSync(process.argv[1], "utf8")); process.stdout.write(String(pkg.version || "unknown"));' "$package_json" 2>/dev/null || true)"
|
||||
fi
|
||||
if [[ -z "$version" ]]; then
|
||||
version="$(sed -n -E 's/^[[:space:]]*"version"[[:space:]]*:[[:space:]]*"([^"]+)".*/\1/p' "$package_json" | head -n1)"
|
||||
fi
|
||||
echo "${version:-unknown}"
|
||||
}
|
||||
|
||||
emit_npm_root_candidate() {
|
||||
local root="${1%/}"
|
||||
if [[ -n "$root" && "$root" == /* ]]; then
|
||||
echo "$root"
|
||||
fi
|
||||
}
|
||||
|
||||
collect_openclaw_npm_root_candidates() {
|
||||
local root=""
|
||||
root="$(npm root -g 2>/dev/null || true)"
|
||||
emit_npm_root_candidate "$root"
|
||||
|
||||
local npm_cmd=""
|
||||
while IFS= read -r npm_cmd; do
|
||||
[[ -n "$npm_cmd" ]] || continue
|
||||
root="$("$npm_cmd" root -g 2>/dev/null || true)"
|
||||
emit_npm_root_candidate "$root"
|
||||
done < <(type -aP npm 2>/dev/null | awk '!seen[$0]++' || true)
|
||||
|
||||
local extra_root=""
|
||||
local old_ifs="$IFS"
|
||||
IFS=":"
|
||||
for extra_root in ${OPENCLAW_INSTALL_EXTRA_NPM_ROOTS:-}; do
|
||||
emit_npm_root_candidate "$extra_root"
|
||||
done
|
||||
IFS="$old_ifs"
|
||||
|
||||
emit_npm_root_candidate "/opt/homebrew/lib/node_modules"
|
||||
emit_npm_root_candidate "/usr/local/lib/node_modules"
|
||||
emit_npm_root_candidate "/usr/lib/node_modules"
|
||||
|
||||
local manager_dir=""
|
||||
local candidate=""
|
||||
for manager_dir in "${NVM_DIR:-}" "$HOME/.nvm"; do
|
||||
[[ -n "$manager_dir" && -d "$manager_dir" ]] || continue
|
||||
for candidate in "$manager_dir"/versions/node/*/lib/node_modules; do
|
||||
[[ -d "$candidate" ]] && emit_npm_root_candidate "$candidate"
|
||||
done
|
||||
done
|
||||
|
||||
for manager_dir in "${FNM_DIR:-}" "$HOME/.fnm" "$HOME/.local/share/fnm"; do
|
||||
[[ -n "$manager_dir" && -d "$manager_dir" ]] || continue
|
||||
for candidate in "$manager_dir"/node-versions/*/installation/lib/node_modules; do
|
||||
[[ -d "$candidate" ]] && emit_npm_root_candidate "$candidate"
|
||||
done
|
||||
done
|
||||
|
||||
for manager_dir in "${VOLTA_HOME:-}" "$HOME/.volta"; do
|
||||
[[ -n "$manager_dir" && -d "$manager_dir" ]] || continue
|
||||
for candidate in "$manager_dir"/tools/image/node/*/lib/node_modules; do
|
||||
[[ -d "$candidate" ]] && emit_npm_root_candidate "$candidate"
|
||||
done
|
||||
done
|
||||
}
|
||||
|
||||
find_openclaw_global_installs() {
|
||||
local seen="|"
|
||||
local npm_root=""
|
||||
while IFS= read -r npm_root; do
|
||||
[[ -n "$npm_root" ]] || continue
|
||||
local package_dir="${npm_root%/}/openclaw"
|
||||
local package_json="${package_dir}/package.json"
|
||||
[[ -f "$package_json" ]] || continue
|
||||
|
||||
local real_package_dir=""
|
||||
real_package_dir="$(canonicalize_dir "$package_dir" || true)"
|
||||
[[ -n "$real_package_dir" ]] || real_package_dir="$package_dir"
|
||||
case "$seen" in
|
||||
*"|${real_package_dir}|"*) continue ;;
|
||||
esac
|
||||
seen="${seen}${real_package_dir}|"
|
||||
|
||||
local version=""
|
||||
version="$(openclaw_package_version "$package_json")"
|
||||
printf '%s\t%s\t%s\n' "$version" "$real_package_dir" "$npm_root"
|
||||
done < <(collect_openclaw_npm_root_candidates)
|
||||
}
|
||||
|
||||
warn_duplicate_openclaw_global_installs() {
|
||||
local installs=()
|
||||
local line=""
|
||||
while IFS= read -r line; do
|
||||
[[ -n "$line" ]] && installs+=("$line")
|
||||
done < <(find_openclaw_global_installs)
|
||||
|
||||
if [[ "${#installs[@]}" -le 1 ]]; then
|
||||
return 0
|
||||
fi
|
||||
|
||||
ui_warn "Multiple OpenClaw global installs detected"
|
||||
echo " Different Node/npm environments can run different OpenClaw versions."
|
||||
|
||||
local active_node active_npm active_openclaw
|
||||
active_node="$(command -v node 2>/dev/null || true)"
|
||||
active_npm="$(command -v npm 2>/dev/null || true)"
|
||||
active_openclaw="${OPENCLAW_BIN:-}"
|
||||
if [[ -z "$active_openclaw" ]]; then
|
||||
active_openclaw="$(type -P openclaw 2>/dev/null || true)"
|
||||
fi
|
||||
echo -e " Active node: ${INFO}${active_node:-none}${NC}"
|
||||
echo -e " Active npm: ${INFO}${active_npm:-none}${NC}"
|
||||
echo -e " Active openclaw: ${INFO}${active_openclaw:-none}${NC}"
|
||||
echo ""
|
||||
echo " Found installs:"
|
||||
|
||||
local install version package_dir npm_root
|
||||
for install in "${installs[@]}"; do
|
||||
IFS=$'\t' read -r version package_dir npm_root <<< "$install"
|
||||
echo -e " - ${INFO}${version:-unknown}${NC} ${package_dir}"
|
||||
echo -e " npm root: ${MUTED}${npm_root}${NC}"
|
||||
done
|
||||
|
||||
echo ""
|
||||
echo " Keep one install source, then remove stale installs with that environment's npm:"
|
||||
echo " npm uninstall -g openclaw"
|
||||
}
|
||||
|
||||
refresh_shell_command_cache() {
|
||||
hash -r 2>/dev/null || true
|
||||
}
|
||||
@ -2206,7 +2395,12 @@ maybe_open_dashboard() {
|
||||
}
|
||||
|
||||
resolve_workspace_dir() {
|
||||
local profile="${OPENCLAW_PROFILE:-default}"
|
||||
if [[ -n "${INSTALL_WORKSPACE}" ]]; then
|
||||
echo "${INSTALL_WORKSPACE}"
|
||||
return
|
||||
fi
|
||||
local profile=""
|
||||
profile="$(resolve_install_profile)"
|
||||
if [[ "${profile}" != "default" ]]; then
|
||||
echo "${HOME}/.openclaw/workspace-${profile}"
|
||||
else
|
||||
@ -2214,13 +2408,133 @@ resolve_workspace_dir() {
|
||||
fi
|
||||
}
|
||||
|
||||
resolve_install_profile() {
|
||||
local profile="${INSTALL_PROFILE:-${OPENCLAW_PROFILE:-default}}"
|
||||
local profile_lc=""
|
||||
profile_lc="$(printf '%s' "${profile}" | tr '[:upper:]' '[:lower:]')"
|
||||
if [[ -z "${profile}" || "${profile_lc}" == "default" ]]; then
|
||||
echo "default"
|
||||
return
|
||||
fi
|
||||
echo "${profile}"
|
||||
}
|
||||
|
||||
resolve_install_state_dir() {
|
||||
if [[ -n "${OPENCLAW_STATE_DIR:-}" ]]; then
|
||||
echo "${OPENCLAW_STATE_DIR}"
|
||||
return
|
||||
fi
|
||||
local profile=""
|
||||
profile="$(resolve_install_profile)"
|
||||
if [[ "${profile}" != "default" ]]; then
|
||||
echo "${HOME}/.openclaw-${profile}"
|
||||
else
|
||||
echo "${HOME}/.openclaw"
|
||||
fi
|
||||
}
|
||||
|
||||
resolve_install_config_path() {
|
||||
if [[ -n "${OPENCLAW_CONFIG_PATH:-}" ]]; then
|
||||
echo "${OPENCLAW_CONFIG_PATH}"
|
||||
return
|
||||
fi
|
||||
local state_dir=""
|
||||
state_dir="$(resolve_install_state_dir)"
|
||||
echo "${state_dir}/openclaw.json"
|
||||
}
|
||||
|
||||
install_config_already_exists() {
|
||||
local profile=""
|
||||
local config_path=""
|
||||
profile="$(resolve_install_profile)"
|
||||
config_path="$(resolve_install_config_path)"
|
||||
if [[ -f "${config_path}" ]]; then
|
||||
return 0
|
||||
fi
|
||||
if [[ "${profile}" != "default" ]]; then
|
||||
return 1
|
||||
fi
|
||||
[[ -f "$HOME/.clawdbot/clawdbot.json" || -f "$HOME/.moltbot/moltbot.json" || -f "$HOME/.moldbot/moldbot.json" ]]
|
||||
}
|
||||
|
||||
validate_install_overrides() {
|
||||
if [[ -n "${INSTALL_PROFILE}" ]]; then
|
||||
if [[ ! "${INSTALL_PROFILE}" =~ ^[A-Za-z0-9][A-Za-z0-9_-]{0,63}$ ]]; then
|
||||
ui_error "Invalid --profile: ${INSTALL_PROFILE} (use letters, numbers, '_' and '-' only)"
|
||||
exit 2
|
||||
fi
|
||||
export OPENCLAW_PROFILE="${INSTALL_PROFILE}"
|
||||
fi
|
||||
if [[ -n "${INSTALL_GATEWAY_PORT}" && ! "${INSTALL_GATEWAY_PORT}" =~ ^[0-9]+$ ]]; then
|
||||
ui_error "Invalid --gateway-port: ${INSTALL_GATEWAY_PORT}"
|
||||
exit 2
|
||||
fi
|
||||
}
|
||||
|
||||
resolve_supplied_install_profile() {
|
||||
local raw_profile=""
|
||||
local resolved_profile=""
|
||||
raw_profile="${INSTALL_PROFILE:-${OPENCLAW_PROFILE:-}}"
|
||||
if [[ -z "${raw_profile}" ]]; then
|
||||
echo ""
|
||||
return
|
||||
fi
|
||||
resolved_profile="$(resolve_install_profile)"
|
||||
if [[ "${resolved_profile}" == "default" ]]; then
|
||||
echo ""
|
||||
return
|
||||
fi
|
||||
echo "${resolved_profile}"
|
||||
}
|
||||
|
||||
build_onboard_command() {
|
||||
local claw="$1"
|
||||
local selected_profile=""
|
||||
selected_profile="$(resolve_supplied_install_profile)"
|
||||
ONBOARD_CMD=("$claw")
|
||||
if [[ -n "${selected_profile}" ]]; then
|
||||
ONBOARD_CMD+=("--profile" "${selected_profile}")
|
||||
fi
|
||||
ONBOARD_CMD+=("onboard")
|
||||
if [[ -n "${INSTALL_WORKSPACE}" ]]; then
|
||||
ONBOARD_CMD+=("--workspace" "${INSTALL_WORKSPACE}")
|
||||
fi
|
||||
if [[ -n "${INSTALL_GATEWAY_PORT}" ]]; then
|
||||
ONBOARD_CMD+=("--gateway-port" "${INSTALL_GATEWAY_PORT}")
|
||||
fi
|
||||
}
|
||||
|
||||
build_onboard_display_command() {
|
||||
local claw_name="${1:-openclaw}"
|
||||
local selected_profile=""
|
||||
selected_profile="$(resolve_supplied_install_profile)"
|
||||
ONBOARD_DISPLAY_CMD=("$claw_name")
|
||||
if [[ -n "${selected_profile}" ]]; then
|
||||
ONBOARD_DISPLAY_CMD+=("--profile" "${selected_profile}")
|
||||
fi
|
||||
ONBOARD_DISPLAY_CMD+=("onboard")
|
||||
if [[ -n "${INSTALL_WORKSPACE}" ]]; then
|
||||
ONBOARD_DISPLAY_CMD+=("--workspace" "${INSTALL_WORKSPACE}")
|
||||
fi
|
||||
if [[ -n "${INSTALL_GATEWAY_PORT}" ]]; then
|
||||
ONBOARD_DISPLAY_CMD+=("--gateway-port" "${INSTALL_GATEWAY_PORT}")
|
||||
fi
|
||||
}
|
||||
|
||||
format_onboard_display_command() {
|
||||
local claw_name="${1:-openclaw}"
|
||||
build_onboard_display_command "$claw_name"
|
||||
local formatted=""
|
||||
printf -v formatted '%q ' "${ONBOARD_DISPLAY_CMD[@]}"
|
||||
echo "${formatted% }"
|
||||
}
|
||||
|
||||
run_bootstrap_onboarding_if_needed() {
|
||||
if [[ "${NO_ONBOARD}" == "1" ]]; then
|
||||
return
|
||||
fi
|
||||
|
||||
local config_path="${OPENCLAW_CONFIG_PATH:-$HOME/.openclaw/openclaw.json}"
|
||||
if [[ -f "${config_path}" || -f "$HOME/.clawdbot/clawdbot.json" || -f "$HOME/.moltbot/moltbot.json" || -f "$HOME/.moldbot/moldbot.json" ]]; then
|
||||
if install_config_already_exists; then
|
||||
return
|
||||
fi
|
||||
|
||||
@ -2233,7 +2547,9 @@ run_bootstrap_onboarding_if_needed() {
|
||||
fi
|
||||
|
||||
if [[ ! -r /dev/tty || ! -w /dev/tty ]]; then
|
||||
ui_info "BOOTSTRAP.md found but no TTY; run openclaw onboard to finish setup"
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command)"
|
||||
ui_info "BOOTSTRAP.md found but no TTY; run ${onboard_cmd} to finish setup"
|
||||
return
|
||||
fi
|
||||
|
||||
@ -2248,8 +2564,11 @@ run_bootstrap_onboarding_if_needed() {
|
||||
return
|
||||
fi
|
||||
|
||||
"$claw" onboard || {
|
||||
ui_error "Onboarding failed; run openclaw onboard to retry"
|
||||
build_onboard_command "$claw"
|
||||
"${ONBOARD_CMD[@]}" || {
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command "$(basename "$claw")")"
|
||||
ui_error "Onboarding failed; run ${onboard_cmd} to retry"
|
||||
ui_info "If gateway startup looks unhealthy, run: openclaw gateway status --deep"
|
||||
return
|
||||
}
|
||||
@ -2337,6 +2656,8 @@ main() {
|
||||
return 0
|
||||
fi
|
||||
|
||||
validate_install_overrides
|
||||
|
||||
bootstrap_gum_temp || true
|
||||
print_installer_banner
|
||||
print_gum_status
|
||||
@ -2449,6 +2770,7 @@ main() {
|
||||
ui_stage "Finalizing setup"
|
||||
|
||||
OPENCLAW_BIN="$(resolve_openclaw_bin || true)"
|
||||
warn_duplicate_openclaw_global_installs || true
|
||||
|
||||
# PATH warning: installs can succeed while the user's login shell still lacks npm's global bin dir.
|
||||
local npm_bin=""
|
||||
@ -2535,7 +2857,7 @@ main() {
|
||||
ui_section "Source install details"
|
||||
ui_kv "Checkout" "$final_git_dir"
|
||||
ui_kv "Wrapper" "$HOME/.local/bin/openclaw"
|
||||
ui_kv "Update command" "openclaw update --restart"
|
||||
ui_kv "Update command" "openclaw update"
|
||||
ui_kv "Switch to npm" "curl -fsSL --proto '=https' --tlsv1.2 https://openclaw.ai/install.sh | bash -s -- --install-method npm"
|
||||
elif [[ "$is_upgrade" == "true" ]]; then
|
||||
ui_info "Upgrade complete"
|
||||
@ -2571,10 +2893,11 @@ main() {
|
||||
fi
|
||||
else
|
||||
if [[ "$NO_ONBOARD" == "1" || "$skip_onboard" == "true" ]]; then
|
||||
ui_info "Skipping onboard (requested); run openclaw onboard later"
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command)"
|
||||
ui_info "Skipping onboard (requested); run ${onboard_cmd} later"
|
||||
else
|
||||
local config_path="${OPENCLAW_CONFIG_PATH:-$HOME/.openclaw/openclaw.json}"
|
||||
if [[ -f "${config_path}" || -f "$HOME/.clawdbot/clawdbot.json" || -f "$HOME/.moltbot/moltbot.json" || -f "$HOME/.moldbot/moldbot.json" ]]; then
|
||||
if install_config_already_exists; then
|
||||
ui_info "Config already present; running doctor"
|
||||
run_doctor
|
||||
should_open_dashboard=true
|
||||
@ -2594,9 +2917,12 @@ main() {
|
||||
return 0
|
||||
fi
|
||||
exec </dev/tty
|
||||
exec "$claw" onboard
|
||||
build_onboard_command "$claw"
|
||||
exec "${ONBOARD_CMD[@]}"
|
||||
fi
|
||||
ui_info "No TTY; run openclaw onboard to finish setup"
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command)"
|
||||
ui_info "No TTY; run ${onboard_cmd} to finish setup"
|
||||
return 0
|
||||
fi
|
||||
fi
|
||||
|
||||
@ -32,6 +32,42 @@ export OPENCLAW_INSTALL_CLI_SH_NO_RUN=1
|
||||
# shellcheck source=../public/install-cli.sh
|
||||
source "${ROOT_DIR}/public/install-cli.sh"
|
||||
|
||||
echo "==> case: ensure_home_env repairs root HOME"
|
||||
(
|
||||
root="${TMP_DIR}/case-home-env"
|
||||
home_dir="${root}/home"
|
||||
tool_bin="${root}/tool-bin"
|
||||
mkdir -p "${home_dir}" "${tool_bin}"
|
||||
|
||||
cat >"${tool_bin}/id" <<EOF
|
||||
#!/usr/bin/env bash
|
||||
set -euo pipefail
|
||||
if [[ "\${1:-}" == "-un" ]]; then
|
||||
echo vmroot
|
||||
exit 0
|
||||
fi
|
||||
exit 1
|
||||
EOF
|
||||
chmod +x "${tool_bin}/id"
|
||||
|
||||
cat >"${tool_bin}/getent" <<EOF
|
||||
#!/usr/bin/env bash
|
||||
set -euo pipefail
|
||||
if [[ "\${1:-}" == "passwd" && "\${2:-}" == "vmroot" ]]; then
|
||||
echo 'vmroot:x:0:0:VM Root:${home_dir}:/bin/bash'
|
||||
exit 0
|
||||
fi
|
||||
exit 1
|
||||
EOF
|
||||
chmod +x "${tool_bin}/getent"
|
||||
|
||||
export PATH="${tool_bin}:/usr/bin:/bin"
|
||||
export HOME="/"
|
||||
|
||||
ensure_home_env
|
||||
assert_eq "$HOME" "${home_dir}" "ensure_home_env root HOME"
|
||||
)
|
||||
|
||||
echo "==> case: ensure_pnpm_binary_for_scripts installs prefix wrapper"
|
||||
(
|
||||
root="${TMP_DIR}/case-pnpm-wrapper"
|
||||
@ -131,6 +167,51 @@ EOF
|
||||
assert_eq "$package_count" "1" "ensure_pnpm_git_prepare_allowlist package count"
|
||||
)
|
||||
|
||||
echo "==> case: install_openclaw keeps a single package root under toolchain"
|
||||
(
|
||||
root="${TMP_DIR}/case-install-npm"
|
||||
prefix="${root}/prefix"
|
||||
fake_bin="${root}/bin"
|
||||
fake_npm="${fake_bin}/npm"
|
||||
toolchain_dir="${prefix}/tools/node-v${NODE_VERSION}"
|
||||
entry_js="${toolchain_dir}/lib/node_modules/openclaw/dist/entry.js"
|
||||
mkdir -p "${fake_bin}" "${toolchain_dir}/bin" "$(dirname "${entry_js}")"
|
||||
printf 'console.log("ok")\n' > "${entry_js}"
|
||||
printf 'legacy-bin\n' > "${toolchain_dir}/bin/openclaw"
|
||||
|
||||
cat >"${fake_npm}" <<'EOF'
|
||||
#!/usr/bin/env bash
|
||||
set -euo pipefail
|
||||
printf '%s\n' "$*" > "${FAKE_NPM_ARGS_FILE}"
|
||||
EOF
|
||||
chmod +x "${fake_npm}"
|
||||
|
||||
export PREFIX="${prefix}"
|
||||
export FAKE_NPM_ARGS_FILE="${root}/npm-args.txt"
|
||||
export SHARP_IGNORE_GLOBAL_LIBVIPS=1
|
||||
export OPENCLAW_VERSION=latest
|
||||
export NPM_LOGLEVEL=error
|
||||
|
||||
npm_bin() { echo "${fake_npm}"; }
|
||||
log() { :; }
|
||||
emit_json() { :; }
|
||||
fix_npm_prefix_if_needed() { :; }
|
||||
|
||||
install_openclaw
|
||||
|
||||
args="$(cat "${FAKE_NPM_ARGS_FILE}")"
|
||||
assert_eq "$args" "install -g --prefix ${toolchain_dir} --loglevel error --no-fund --no-audit openclaw@latest" "install_openclaw npm prefix"
|
||||
test -x "${prefix}/bin/openclaw"
|
||||
test -e "${toolchain_dir}/bin/openclaw"
|
||||
wrapper_target="$(python3 - <<'PY' "${prefix}/bin/openclaw"
|
||||
import pathlib
|
||||
import sys
|
||||
print(pathlib.Path(sys.argv[1]).read_text().splitlines()[-1])
|
||||
PY
|
||||
)"
|
||||
assert_eq "$wrapper_target" "exec \"${prefix}/tools/node/bin/node\" \"${toolchain_dir}/lib/node_modules/openclaw/dist/entry.js\" \"\$@\"" "install_openclaw wrapper target"
|
||||
)
|
||||
|
||||
echo "==> case: install_openclaw_from_git uses run_pnpm"
|
||||
(
|
||||
root="${TMP_DIR}/case-install-git"
|
||||
@ -145,8 +226,7 @@ echo "==> case: install_openclaw_from_git uses run_pnpm"
|
||||
export GIT_UPDATE=0
|
||||
export SHARP_IGNORE_GLOBAL_LIBVIPS=1
|
||||
|
||||
deps_cmd=""
|
||||
build_cmd=""
|
||||
run_pnpm_calls=()
|
||||
|
||||
ensure_git() { :; }
|
||||
ensure_pnpm() { set_pnpm_cmd echo pnpm; }
|
||||
@ -156,17 +236,14 @@ echo "==> case: install_openclaw_from_git uses run_pnpm"
|
||||
emit_json() { :; }
|
||||
fail() { echo "FAIL: $*" >&2; exit 1; }
|
||||
run_pnpm() {
|
||||
if [[ -z "$deps_cmd" ]]; then
|
||||
deps_cmd="$1"
|
||||
else
|
||||
build_cmd="$1"
|
||||
fi
|
||||
run_pnpm_calls+=("$*")
|
||||
return 0
|
||||
}
|
||||
|
||||
install_openclaw_from_git "${repo}"
|
||||
assert_eq "$deps_cmd" "-C" "install_openclaw_from_git deps command entry"
|
||||
assert_nonempty "$build_cmd" "install_openclaw_from_git build command entry"
|
||||
assert_eq "${run_pnpm_calls[0]}" "-C ${repo} install" "install_openclaw_from_git deps command"
|
||||
assert_eq "${run_pnpm_calls[1]}" "-C ${repo} ui:build" "install_openclaw_from_git ui build command"
|
||||
assert_eq "${run_pnpm_calls[2]}" "-C ${repo} build" "install_openclaw_from_git build command"
|
||||
test -x "${prefix}/bin/openclaw"
|
||||
)
|
||||
|
||||
|
||||
63
scripts/test-install-ps1-unit.sh
Executable file
63
scripts/test-install-ps1-unit.sh
Executable file
@ -0,0 +1,63 @@
|
||||
#!/usr/bin/env bash
|
||||
set -euo pipefail
|
||||
|
||||
ROOT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)"
|
||||
SCRIPTS=("${ROOT_DIR}/public/install.ps1")
|
||||
if [[ -f "${ROOT_DIR}/dist/install.ps1" ]]; then
|
||||
SCRIPTS+=("${ROOT_DIR}/dist/install.ps1")
|
||||
fi
|
||||
|
||||
fail() {
|
||||
echo "FAIL: $*" >&2
|
||||
exit 1
|
||||
}
|
||||
|
||||
require_contains() {
|
||||
local script="$1"
|
||||
local pattern="$2"
|
||||
if ! grep -Fq "$pattern" "$script"; then
|
||||
fail "$(realpath --relative-to="$ROOT_DIR" "$script"): missing pattern: $pattern"
|
||||
fi
|
||||
}
|
||||
|
||||
for script in "${SCRIPTS[@]}"; do
|
||||
exit_lines="$(grep -nE '^[[:space:]]*exit\b' "$script" || true)"
|
||||
# shellcheck disable=SC2016
|
||||
if [[ "$exit_lines" != *'exit $script:InstallExitCode'* ]]; then
|
||||
fail "$(realpath --relative-to="$ROOT_DIR" "$script"): expected the only installer exit to live in Complete-Install"
|
||||
fi
|
||||
if [[ "$(printf '%s\n' "$exit_lines" | sed '/^$/d' | wc -l | tr -d ' ')" != "1" ]]; then
|
||||
printf '%s\n' "$exit_lines" >&2
|
||||
fail "$(realpath --relative-to="$ROOT_DIR" "$script"): unexpected extra exit usage"
|
||||
fi
|
||||
|
||||
main_body="$(awk '
|
||||
/^function Main \{/ { in_main = 1; next }
|
||||
/^\$mainResults = @\(Main\)/ { in_main = 0 }
|
||||
in_main { print }
|
||||
' "$script")"
|
||||
|
||||
if grep -E '^[[:space:]]*exit\b' <<<"$main_body" >/dev/null; then
|
||||
fail "$(realpath --relative-to="$ROOT_DIR" "$script"): Main must not call exit"
|
||||
fi
|
||||
|
||||
require_contains "$script" 'function Fail-Install {'
|
||||
require_contains "$script" 'function Complete-Install {'
|
||||
require_contains "$script" 'function Resolve-NpmOpenClawInstallSpec {'
|
||||
# shellcheck disable=SC2016
|
||||
require_contains "$script" 'install -g "$installSpec"'
|
||||
# shellcheck disable=SC2016
|
||||
require_contains "$script" 'return "$PackageName@$trimmedTag"'
|
||||
require_contains "$script" 'return (Fail-Install -Code 2)'
|
||||
require_contains "$script" 'return (Fail-Install)'
|
||||
# shellcheck disable=SC2016
|
||||
require_contains "$script" '$mainResults = @(Main)'
|
||||
# shellcheck disable=SC2016
|
||||
require_contains "$script" '$installSucceeded = $mainResults.Count -gt 0 -and $mainResults[-1] -eq $true'
|
||||
# shellcheck disable=SC2016
|
||||
require_contains "$script" 'Complete-Install -Succeeded:$installSucceeded'
|
||||
# shellcheck disable=SC2016
|
||||
require_contains "$script" 'throw "OpenClaw installation failed with exit code $($script:InstallExitCode)."'
|
||||
done
|
||||
|
||||
echo "install.ps1 unit checks passed"
|
||||
@ -66,6 +66,23 @@ export CLAWDBOT_INSTALL_SH_NO_RUN=1
|
||||
# shellcheck source=../public/install.sh
|
||||
source "${ROOT_DIR}/public/install.sh"
|
||||
|
||||
echo "==> case: ensure_home_env repairs root HOME"
|
||||
(
|
||||
root="${TMP_DIR}/case-home-env"
|
||||
home_dir="${root}/home"
|
||||
tool_bin="${root}/tool-bin"
|
||||
mkdir -p "${home_dir}" "${tool_bin}"
|
||||
|
||||
make_exe "${tool_bin}/id" "if [[ \"\${1:-}\" == \"-un\" ]]; then echo vmroot; exit 0; fi; exit 1"
|
||||
make_exe "${tool_bin}/getent" "if [[ \"\${1:-}\" == \"passwd\" && \"\${2:-}\" == \"vmroot\" ]]; then echo 'vmroot:x:0:0:VM Root:${home_dir}:/bin/bash'; exit 0; fi; exit 1"
|
||||
|
||||
export PATH="${tool_bin}:/usr/bin:/bin"
|
||||
export HOME="/"
|
||||
|
||||
ensure_home_env
|
||||
assert_eq "$HOME" "${home_dir}" "ensure_home_env root HOME"
|
||||
)
|
||||
|
||||
echo "==> case: resolve_openclaw_bin (direct PATH)"
|
||||
(
|
||||
bin="${TMP_DIR}/case-path/bin"
|
||||
@ -133,6 +150,41 @@ echo "==> case: warn_openclaw_not_found (smoke)"
|
||||
assert_nonempty "$out" "warn_openclaw_not_found output"
|
||||
)
|
||||
|
||||
echo "==> case: duplicate OpenClaw install warning (multiple npm roots)"
|
||||
(
|
||||
root="${TMP_DIR}/case-duplicate-openclaw"
|
||||
mkdir -p "${root}/brew/openclaw" "${root}/fnm/openclaw"
|
||||
printf '{"version":"2026.3.7"}\n' > "${root}/brew/openclaw/package.json"
|
||||
printf '{"version":"2026.3.1"}\n' > "${root}/fnm/openclaw/package.json"
|
||||
|
||||
collect_openclaw_npm_root_candidates() { printf '%s\n' "${root}/brew" "${root}/fnm"; }
|
||||
# shellcheck disable=SC2034
|
||||
OPENCLAW_BIN="${root}/fnm/.bin/openclaw"
|
||||
ui_warn() { echo "WARN: $*"; }
|
||||
|
||||
out="$(warn_duplicate_openclaw_global_installs 2>&1)"
|
||||
assert_contains "$out" "Multiple OpenClaw global installs detected" "duplicate OpenClaw warning header"
|
||||
assert_contains "$out" "2026.3.7" "duplicate OpenClaw warning first version"
|
||||
assert_contains "$out" "2026.3.1" "duplicate OpenClaw warning second version"
|
||||
assert_contains "$out" "${root}/brew/openclaw" "duplicate OpenClaw warning first path"
|
||||
assert_contains "$out" "${root}/fnm/openclaw" "duplicate OpenClaw warning second path"
|
||||
assert_contains "$out" "Active openclaw:" "duplicate OpenClaw warning active binary"
|
||||
assert_contains "$out" "npm uninstall -g openclaw" "duplicate OpenClaw warning cleanup hint"
|
||||
)
|
||||
|
||||
echo "==> case: duplicate OpenClaw install warning (single npm root is quiet)"
|
||||
(
|
||||
root="${TMP_DIR}/case-single-openclaw"
|
||||
mkdir -p "${root}/only/openclaw"
|
||||
printf '{"version":"2026.3.7"}\n' > "${root}/only/openclaw/package.json"
|
||||
|
||||
collect_openclaw_npm_root_candidates() { printf '%s\n' "${root}/only"; }
|
||||
ui_warn() { echo "WARN: $*"; }
|
||||
|
||||
out="$(warn_duplicate_openclaw_global_installs 2>&1 || true)"
|
||||
assert_eq "$out" "" "duplicate OpenClaw warning single root output"
|
||||
)
|
||||
|
||||
echo "==> case: ensure_pnpm (existing pnpm command)"
|
||||
(
|
||||
root="${TMP_DIR}/case-pnpm-existing"
|
||||
@ -480,8 +532,7 @@ echo "==> case: install_openclaw_from_git (deps step uses run_pnpm function)"
|
||||
# shellcheck disable=SC2034
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS=1
|
||||
|
||||
deps_called=0
|
||||
deps_cmd=""
|
||||
run_pnpm_steps=()
|
||||
|
||||
check_git() { return 0; }
|
||||
install_git() { fail "install_git should not be called"; }
|
||||
@ -498,16 +549,18 @@ echo "==> case: install_openclaw_from_git (deps step uses run_pnpm function)"
|
||||
run_quiet_step() {
|
||||
local _title="$1"
|
||||
shift
|
||||
if [[ "${_title}" == "Installing dependencies" ]]; then
|
||||
deps_called=1
|
||||
deps_cmd="${1:-}"
|
||||
if [[ "${1:-}" == "run_pnpm" ]]; then
|
||||
shift
|
||||
run_pnpm_steps+=("$*")
|
||||
return 0
|
||||
fi
|
||||
"$@" >/dev/null 2>&1 || true
|
||||
}
|
||||
|
||||
install_openclaw_from_git "${repo}"
|
||||
assert_eq "$deps_called" "1" "install_openclaw_from_git dependencies step"
|
||||
assert_eq "$deps_cmd" "run_pnpm" "install_openclaw_from_git dependencies command"
|
||||
assert_eq "${run_pnpm_steps[0]}" "-C ${repo} install" "install_openclaw_from_git dependencies command"
|
||||
assert_eq "${run_pnpm_steps[1]}" "-C ${repo} ui:build" "install_openclaw_from_git ui build command"
|
||||
assert_eq "${run_pnpm_steps[2]}" "-C ${repo} build" "install_openclaw_from_git build command"
|
||||
)
|
||||
|
||||
echo "==> case: ensure_pnpm_git_prepare_allowlist (known dep added once)"
|
||||
|
||||
@ -1,4 +1,5 @@
|
||||
import { defineCollection, z } from 'astro:content';
|
||||
import { glob } from 'astro/loaders';
|
||||
|
||||
const authorSchema = z.object({
|
||||
name: z.string(),
|
||||
@ -6,7 +7,7 @@ const authorSchema = z.object({
|
||||
});
|
||||
|
||||
const blog = defineCollection({
|
||||
type: 'content',
|
||||
loader: glob({ pattern: '**/*.md', base: './src/content/blog' }),
|
||||
schema: z.object({
|
||||
title: z.string(),
|
||||
description: z.string(),
|
||||
33
src/content/blog/openclaw-rough-week.md
Normal file
33
src/content/blog/openclaw-rough-week.md
Normal file
@ -0,0 +1,33 @@
|
||||
---
|
||||
title: "OpenClaw Had a Rough Week"
|
||||
description: "What happened around the 2026.4.24 and 2026.4.29 releases, why the direction was right, and what we are changing now."
|
||||
date: 2026-05-05
|
||||
author: "Peter Steinberger"
|
||||
authorHandle: "steipete"
|
||||
draft: false
|
||||
tags: ["release", "stability", "security", "clawhub"]
|
||||
---
|
||||
|
||||
**TL;DR:** OpenClaw had a rough week. 2026.4.29 made it obvious. Sorry. We are making core smaller, moving optional stuff to [ClawHub](https://clawhub.ai/), and announcing LTS separately later in May.
|
||||
|
||||
The trouble started around 2026.4.24. By 2026.4.29 it was obvious enough that nobody could pretend this was just a few weird installs. Gateways got slower. Some installs got stuck in plugin dependency repair loops. Discord, Telegram, WhatsApp and other channels behaved worse than they should. People downgraded. People lost time.
|
||||
|
||||
This was not one bug. Plugin dependency repair ran in startup and update paths, bundled and external plugins were half-split, ClawHub artifact metadata was still settling, and gateway cold paths did too much work.
|
||||
|
||||
That sucks. I’m sorry.
|
||||
|
||||
We’ve been pushing OpenClaw to become smaller, safer and more infrastructure-grade. That means less magic in core, fewer bundled dependencies, clearer plugin boundaries, better scanning, better release hygiene, better security posture. All the boring stuff that matters once people run this as actual infrastructure and not just as my weird lobster playground.
|
||||
|
||||
Recent npm ecosystem supply-chain incidents made this feel a lot less theoretical. OpenClaw did not directly depend on [Axios](https://www.microsoft.com/en-us/security/blog/2026/04/01/mitigating-the-axios-npm-supply-chain-compromise/); the relevant risk was the shape of the dependency graph: transitive packages, install-time behavior, postinstall scripts, packages pulling packages pulling packages.
|
||||
|
||||
So we started moving things out of core: channels, providers, heavy tools, parsers, optional integrations. The [plugin inventory](https://docs.openclaw.ai/plugins/plugin-inventory) shows what still ships in core, what installs separately, and what is source-checkout only.
|
||||
|
||||
The problem: I underestimated how difficult it would be to get this right. For a few releases we ended up in the worst middle state: too much moved toward plugins, while too many plugins were still bundled, repaired, staged, checked, or dependency-loaded in places users feel immediately.
|
||||
|
||||
This also exposed an operating problem: OpenClaw was still too founder-driven. Too much release, review, packaging and support work sat with me. Through the OpenClaw Foundation, and with help from OpenAI, we are building a real team around the project.
|
||||
|
||||
Going forward, we'll be changing how releases are done, and will soon announce an LTS release next to our faster update cycles.
|
||||
|
||||
Thank you to everyone who reported issues, pasted logs, tested betas, downgraded, upgraded again, or just waited while we dug through this.
|
||||
|
||||
OpenClaw will keep getting more secure. It will also get smaller. But it has to stay boringly reliable while we do that.
|
||||
83
src/content/blog/openclaw-security-in-public.md
Normal file
83
src/content/blog/openclaw-security-in-public.md
Normal file
@ -0,0 +1,83 @@
|
||||
---
|
||||
title: "How OpenClaw Got Safer in Public"
|
||||
description: "The work you do not see behind the world's most-watched open source personal AI agent."
|
||||
date: 2026-04-30
|
||||
author: "Peter Steinberger"
|
||||
authorHandle: "steipete"
|
||||
draft: false
|
||||
tags: ["security", "open-source", "nvidia", "clawhub"]
|
||||
---
|
||||
|
||||
OpenClaw started on my Mac in Vienna as an experiment. A lot of people screamed it was so insecure.
|
||||
|
||||
Open source is supposed to be the unsafe option because everyone can see the code. Sure.
|
||||
|
||||
People used it anyway, loved it, and now companies run it in production. Those same companies are the ones now helping us secure it. Nothing that can run tools, hold credentials and install plugins is safe by default. But being open is why we got safer quickly, in public.
|
||||
|
||||
## Why So Many Reports?
|
||||
|
||||
OpenClaw launched into a weird moment for open source security. In January, curl killed its bug bounty program after drowning in reports that sounded technical, referenced real functions and contained nothing exploitable. Daniel Stenberg called it "death by a thousand slops."
|
||||
|
||||
Plus, we are the most-watched AI agent project in the world. Every CVE against OpenClaw is a career trophy, so of course people look.
|
||||
|
||||
As of April 30, GitHub shows [1,309 security advisories](https://github.com/openclaw/openclaw/security/advisories) since January 10. 535 were published. 746 were closed as invalid. The number coming in has dropped significantly over the last few months as we hardened the whole system.
|
||||
|
||||
The closer a report sits to "critical", the more likely it is to be nonsense. GitHub currently shows 109 critical reports: 14 published, 95 closed as invalid. That is 87%.
|
||||
|
||||
The false positives are often wonderfully dumb: "the agent runs commands, therefore RCE", "plugins execute code", "this dangerous opt-in mode is dangerous", "if I already have the token I can do bad things."
|
||||
|
||||
## What Actually Changed
|
||||
|
||||
At first I was just annoyed at how the game worked. A security advisory used to be an event: stop everything, reproduce, inspect, patch, disclose, ship. Five times a year was annoying; fifteen times a day breaks the process.
|
||||
|
||||
What we needed was a triage tool, not a magical sandbox: a way to decide whether a report describes a real boundary violation or OpenClaw doing expected OpenClaw things. [SECURITY.md](https://github.com/openclaw/openclaw/blob/main/SECURITY.md) defines the trust model, documents expected behavior, and gives maintainers something concrete to point at when closing bad reports.
|
||||
|
||||
Real bugs remain. OpenClaw moves fast and does weird stuff. We fixed authentication bugs, privilege confusion, reconnect scope widening, sandbox bypasses, unsafe env handling and approval path mistakes.
|
||||
|
||||
Some of this cost regular users features. We tightened allowlists, accepted regressions where the single-machine setup (the Mac Mini on your desk, your laptop) was fine, and shipped fast even when fast hurt. Most of the hardening targets multi-user threats most users never hit. We did it anyway, because the people who do hit them are now running this in production.
|
||||
|
||||
## Built for Production
|
||||
|
||||
We shrank the core. Over the last few months we pushed more functionality out to [plugins](https://docs.openclaw.ai/plugins/sdk-overview), which means a smaller attack surface, a shorter dependency tree and a clearer trust boundary. A poisoned upstream package has fewer paths to actually reach a user.
|
||||
|
||||
Releases used to be just me. Now it's me plus another [OpenClaw Foundation](https://www.openclaw.org/) employee, with each one scripted, gated and signed off. End-to-end testing in CI got leveled up so agent flows run on every PR instead of waiting for someone's laptop.
|
||||
|
||||
We added [observability](https://docs.openclaw.ai/gateway/opentelemetry): OpenTelemetry, Prometheus metrics, higher-throughput logging and better signals. Secrets moved away from "please be careful" toward references, so credentials do not end up sitting in prompts, logs, transcripts or agent state.
|
||||
|
||||
Plugins can act as harnesses now. Wire OpenAI Codex in as [the harness](https://docs.openclaw.ai/plugins/codex-harness) for GPT models and you inherit its controls, including Guardian for per-action gating, instead of running the agent in accept-each-request or YOLO mode.
|
||||
|
||||
## The Team Behind It
|
||||
|
||||
OpenClaw is not just me anymore. It's me plus an army of maintainers who triage reports, review patches, ship releases and take calls at stupid hours when something real lands. Most have day jobs. They still show up.
|
||||
|
||||
They have help. CodeQL, Semgrep, Codex Security and maintainer-owned checks catch weak commits before they merge. [ClawSweeper](https://github.com/openclaw/clawsweeper) handles issue and PR triage so the team can keep up with the firehose.
|
||||
|
||||
[NVIDIA](https://blogs.nvidia.com/blog/what-openclaw-agents-mean-for-every-organization/) showed up early with engineering time, security thinking and work on [NemoClaw](https://docs.nvidia.com/nemoclaw/latest/) and OpenShell.
|
||||
|
||||
[Microsoft](https://www.microsoft.com/) and [GitHub](https://github.com/) helped at the platform level through the [GitHub Secure Open Source Fund](https://github.blog/open-source/maintainers/securing-the-ai-software-supply-chain-security-results-across-67-open-source-projects/). [Atlassian](https://www.atlassian.com/) and other enterprise partners pushed on deployment, auditability, identity boundaries and secret handling. [Blacksmith](https://www.blacksmith.sh/) gives us the runner capacity to test agent paths at the rate we ship.
|
||||
|
||||
[Tencent](https://www.tencent.com/) added full-time maintainers on security, stability and ClawHub, plus a direct vulnerability-sync line with their internal security team.
|
||||
|
||||
[OpenAI](https://openai.com/) continues to support the project with inference, gave us [Codex Security](https://openai.com/index/codex-security-now-in-research-preview/) to proactively find and fix security issues, and has made commitments that help keep OpenClaw open and independent as the Foundation comes together. Inside OpenAI, I run a team called Claw Labs that works on shared product improvements.
|
||||
|
||||
## ClawHub
|
||||
|
||||
[Convex](https://www.convex.dev/) helped maintain ClawHub while we rebuilt the security posture around it. You do not secure marketplaces once. You keep watching, pruning and making the weird stuff easier to spot.
|
||||
|
||||
In the last month alone the team closed more than 700 ClawHub moderation issues, around 460 of them rescan appeals from skill authors whose work the automated suspicious flag had misfired on. We will publish more of the ClawHub security findings soon.
|
||||
|
||||
## Agents of Chaos
|
||||
|
||||
The [Agents of Chaos](https://arxiv.org/abs/2602.20021) paper that made the rounds in February is the loudest example of the incentive problem. Twenty researchers attacked six OpenClaw agents for two weeks and found ugly failures.
|
||||
|
||||
The annoying part is the framing. They ran OpenClaw in sudo mode with disabled guardrails, broad shell access and no sandboxing, then wrote up the results as if this is what users get out of the box. The paper has since added a short acknowledgment that guardrails were disabled; the headlines did not.
|
||||
|
||||
The lesson is simpler. OpenClaw is built for one trusted person per agent. Share that agent with people you don't trust, and they share its tool access. That is the design, not a hidden auth bug. For groups or companies, split agents and credentials per trust boundary, and turn on sandboxing.
|
||||
|
||||
## Fixes Count
|
||||
|
||||
The security industry rewards disclosure, not repair. To researchers: I would much rather read your slightly broken report with a real reproduction than your perfectly formatted slop. "I found and fixed a vulnerability in OpenClaw" should carry more credit than "I filed the scariest GHSA title."
|
||||
|
||||
Open and safe are not opposites. Open is how we get to safe at all.
|
||||
|
||||
The claw is the law. 🦞
|
||||
@ -6,7 +6,7 @@ import { getPublishedBlogPosts } from '../../lib/blog';
|
||||
export async function getStaticPaths() {
|
||||
const posts = await getPublishedBlogPosts();
|
||||
return posts.map((post) => ({
|
||||
params: { slug: post.slug },
|
||||
params: { slug: post.id },
|
||||
props: { post },
|
||||
}));
|
||||
}
|
||||
@ -38,7 +38,7 @@ function estimateReadTime(content: string): string {
|
||||
}
|
||||
|
||||
const readTime = estimateReadTime(post.body || '');
|
||||
const postUrl = `https://openclaw.ai/blog/${post.slug}`;
|
||||
const postUrl = `https://openclaw.ai/blog/${post.id}`;
|
||||
---
|
||||
|
||||
<Layout title={`${post.data.title} — OpenClaw Blog`} description={post.data.description}>
|
||||
|
||||
@ -36,7 +36,7 @@ function estimateReadTime(content: string): string {
|
||||
|
||||
<div class="posts-grid">
|
||||
{posts.map((post) => (
|
||||
<a href={`/blog/${post.slug}`} class="post-card">
|
||||
<a href={`/blog/${post.id}`} class="post-card">
|
||||
<div class="post-meta">
|
||||
<span class="post-date">{formatDate(post.data.date)}</span>
|
||||
<span class="post-separator">·</span>
|
||||
|
||||
@ -102,7 +102,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
<!-- Latest Blog Post -->
|
||||
{latestPost && (
|
||||
<section class="latest-post">
|
||||
<a href={`/blog/${latestPost.slug}`} class="latest-post-card">
|
||||
<a href={`/blog/${latestPost.id}`} class="latest-post-card">
|
||||
<span class="latest-post-badge">New</span>
|
||||
<span class="latest-post-title">{latestPost.data.title}</span>
|
||||
<span class="latest-post-link">→</span>
|
||||
@ -232,7 +232,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
</div>
|
||||
</div>
|
||||
<div id="hackable-pnpm-content" style="display: none;">
|
||||
<div class="code-line comment"># You clearly know what you're doing</div>
|
||||
<div class="code-line comment"># Source checkouts are pnpm workspaces</div>
|
||||
<div class="code-line cmd">
|
||||
<span class="code-prompt">$</span>
|
||||
<span>git clone https://github.com/openclaw/openclaw.git</span>
|
||||
@ -243,16 +243,16 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
</div>
|
||||
<div class="code-line cmd">
|
||||
<span class="code-prompt">$</span>
|
||||
<span>cd openclaw && pnpm install && pnpm run build</span>
|
||||
<span>cd openclaw && corepack enable && pnpm install</span>
|
||||
<button class="copy-line-btn" data-cmd="build" title="Copy">
|
||||
<svg class="copy-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>
|
||||
<svg class="check-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="display:none"><polyline points="20 6 9 17 4 12"/></svg>
|
||||
</button>
|
||||
</div>
|
||||
<div class="code-line comment"># You built it, now meet it</div>
|
||||
<div class="code-line comment"># Run from source; bundled plugins use extensions/* deps</div>
|
||||
<div class="code-line cmd">
|
||||
<span class="code-prompt">$</span>
|
||||
<span>pnpm run openclaw onboard</span>
|
||||
<span>pnpm openclaw onboard</span>
|
||||
<button class="copy-line-btn" data-cmd="hackable-onboard" title="Copy">
|
||||
<svg class="copy-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>
|
||||
<svg class="check-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="display:none"><polyline points="20 6 9 17 4 12"/></svg>
|
||||
@ -279,7 +279,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
</div>
|
||||
</div>
|
||||
<p class="quickstart-note">
|
||||
Works on macOS, Linux, and Windows. The one-liner installs Node.js and everything else for you.
|
||||
Works on macOS, Linux, and Windows. The one-liner installs Node.js and everything else for you. Switch later with <code>openclaw update --channel dev</code> or <code>openclaw update --channel stable</code>.
|
||||
</p>
|
||||
</section>
|
||||
|
||||
@ -287,6 +287,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
// Windows install commands
|
||||
const windowsCmd = 'powershell -c "irm https://openclaw.ai/install.ps1 | iex"';
|
||||
const windowsBetaCmd = 'powershell -c "& ([scriptblock]::Create((irm https://openclaw.ai/install.ps1))) -Tag beta"';
|
||||
const windowsGitCmd = 'powershell -c "& ([scriptblock]::Create((irm https://openclaw.ai/install.ps1))) -InstallMethod git"';
|
||||
|
||||
// State
|
||||
let currentPm = 'npm';
|
||||
@ -360,7 +361,9 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
}
|
||||
osCmdElements.forEach(cmd => cmd.textContent = onelinerCmd);
|
||||
// Update hackable OS command for installer mode
|
||||
const hackableOsCmd = "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git";
|
||||
const hackableOsCmd = currentOs === 'windows'
|
||||
? windowsGitCmd
|
||||
: "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git";
|
||||
osCmdHackableElements.forEach(cmd => cmd.textContent = hackableOsCmd);
|
||||
// Update hackable content visibility (null-safe)
|
||||
if (currentMode === 'hackable') {
|
||||
@ -386,8 +389,8 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
if (codeHackable) codeHackable.style.display = currentMode === 'hackable' ? 'block' : 'none';
|
||||
if (codeMacos) codeMacos.style.display = currentMode === 'macos' ? 'block' : 'none';
|
||||
|
||||
// Show OS switch for one-liner, PM switch for quick, hackable switch for hackable, nothing for macos
|
||||
const showOsControls = currentMode === 'oneliner';
|
||||
// Show OS switch for one-liner and installer-backed hackable mode.
|
||||
const showOsControls = currentMode === 'oneliner' || (currentMode === 'hackable' && currentHackable === 'installer');
|
||||
const showPmControls = currentMode === 'quick';
|
||||
const showHackableControls = currentMode === 'hackable';
|
||||
// Beta only applies to oneliner and npm modes (git always gets main branch)
|
||||
@ -424,6 +427,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
hackableBtns.forEach(b => b.classList.remove('active'));
|
||||
btn.classList.add('active');
|
||||
updateCommands();
|
||||
updateVisibility();
|
||||
});
|
||||
});
|
||||
|
||||
@ -466,10 +470,12 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
return currentPm === 'npm' ? `npm i -g openclaw${betaSuffix}` : `pnpm add -g openclaw${betaSuffix}`;
|
||||
},
|
||||
'onboard': () => 'openclaw onboard',
|
||||
'hackable-installer': () => "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git",
|
||||
'hackable-installer': () => currentOs === 'windows'
|
||||
? windowsGitCmd
|
||||
: "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git",
|
||||
'clone': () => 'git clone https://github.com/openclaw/openclaw.git',
|
||||
'build': () => 'cd openclaw && pnpm install && pnpm run build',
|
||||
'hackable-onboard': () => 'pnpm run openclaw onboard',
|
||||
'build': () => 'cd openclaw && corepack enable && pnpm install',
|
||||
'hackable-onboard': () => 'pnpm openclaw onboard',
|
||||
};
|
||||
|
||||
document.querySelectorAll('.copy-line-btn').forEach(btn => {
|
||||
|
||||
@ -9,7 +9,7 @@ import {
|
||||
siPhilipshue, siHomeassistant,
|
||||
siGooglechrome, siGmail, si1password,
|
||||
siX, siVercel,
|
||||
siLinux, siAndroid, siMacos, siIos
|
||||
siLinux, siAndroid, siMacos, siIos, siQq
|
||||
} from 'simple-icons';
|
||||
|
||||
// Helper to create SVG from simple-icons
|
||||
@ -84,6 +84,7 @@ const chatProviders = [
|
||||
{ name: 'Zalo', icon: siIcon(siZalo), color: '#0068FF', desc: 'Zalo Bot API', docs: 'https://docs.openclaw.ai/channels/zalo' },
|
||||
{ name: 'Zalo Personal', icon: siIcon(siZalo), color: '#0068FF', desc: 'Personal account via QR login', docs: 'https://docs.openclaw.ai/channels/zalouser' },
|
||||
{ name: 'WebChat', icon: 'lucide:globe', color: '#00E5CC', desc: 'Browser-based UI', docs: 'https://docs.openclaw.ai/webchat' },
|
||||
{ name: 'QQ', icon: siIcon(siQq), color: '#1EBAFC', desc: 'QQ Bot Official API', docs: 'https://docs.openclaw.ai/channels/qqbot' },
|
||||
];
|
||||
|
||||
const modelProviders = [
|
||||
@ -114,11 +115,11 @@ const productivityApps = [
|
||||
{ name: 'Apple Notes', icon: 'lucide:sticky-note', color: '#FFCC00', desc: 'Native macOS/iOS notes', docs: 'https://clawhub.ai/skills/apple-notes' },
|
||||
{ name: 'Apple Reminders', icon: 'lucide:check-square', color: '#FF9500', desc: 'Task management', docs: 'https://clawhub.ai/skills/apple-reminders' },
|
||||
{ name: 'Things 3', icon: 'lucide:list-todo', color: '#4A90D9', desc: 'GTD task manager', docs: 'https://clawhub.ai/skills/things-mac' },
|
||||
{ name: 'Notion', icon: siIcon(siNotion), color: '#FFFFFF', desc: 'Workspace & databases', docs: 'https://clawhub.ai/skills' },
|
||||
{ name: 'Notion', icon: siIcon(siNotion), color: '#FFFFFF', desc: 'Workspace & databases', docs: 'https://clawhub.ai/skills/notion' },
|
||||
{ name: 'Obsidian', icon: siIcon(siObsidian), color: '#7C3AED', desc: 'Knowledge graph notes', docs: 'https://clawhub.ai/skills/obsidian' },
|
||||
{ name: 'Bear Notes', icon: 'lucide:pen-tool', color: '#DD4C4F', desc: 'Markdown notes', docs: 'https://clawhub.ai/skills' },
|
||||
{ name: 'Bear Notes', icon: 'lucide:pen-tool', color: '#DD4C4F', desc: 'Markdown notes', docs: 'https://clawhub.ai/skills/bear-notes' },
|
||||
{ name: 'Trello', icon: siIcon(siTrello), color: '#0079BF', desc: 'Kanban boards', docs: 'https://clawhub.ai/skills/trello' },
|
||||
{ name: 'GitHub', icon: siIcon(siGithub), color: '#FFFFFF', desc: 'Code, issues, PRs', docs: 'https://clawhub.ai/skills' },
|
||||
{ name: 'GitHub', icon: siIcon(siGithub), color: '#FFFFFF', desc: 'Code, issues, PRs', docs: 'https://clawhub.ai/skills/github' },
|
||||
];
|
||||
|
||||
const musicAudio = [
|
||||
@ -145,10 +146,10 @@ const tools = [
|
||||
];
|
||||
|
||||
const mediaCreative = [
|
||||
{ name: 'Image Gen', icon: 'lucide:image', color: '#E91E63', desc: 'AI image generation', docs: 'https://clawhub.ai/skills' },
|
||||
{ name: 'Image Gen', icon: 'lucide:image', color: '#E91E63', desc: 'AI image generation', docs: 'https://clawhub.ai/skills/image-gen' },
|
||||
{ name: 'GIF Search', icon: 'lucide:search', color: '#00DCDC', desc: 'Find the perfect GIF', docs: 'https://clawhub.ai/skills/gifgrep' },
|
||||
{ name: 'Peekaboo', icon: 'lucide:eye', color: '#FF6B6B', desc: 'Screen capture & control', docs: 'https://clawhub.ai/skills/peekaboo' },
|
||||
{ name: 'Camera', icon: 'lucide:camera', color: '#607D8B', desc: 'Photo/video capture', docs: 'https://clawhub.ai/skills' },
|
||||
{ name: 'Camera', icon: 'lucide:camera', color: '#607D8B', desc: 'Photo/video capture', docs: 'https://clawhub.ai/skills/camera' },
|
||||
];
|
||||
|
||||
const socialComms = [
|
||||
|
||||
@ -12,7 +12,7 @@ export async function GET(context) {
|
||||
title: post.data.title,
|
||||
description: post.data.description,
|
||||
pubDate: post.data.date,
|
||||
link: `/blog/${post.slug}`,
|
||||
link: `/blog/${post.id}`,
|
||||
})),
|
||||
});
|
||||
}
|
||||
|
||||
Loading…
Reference in New Issue
Block a user