Compare commits
1 Commits
main
...
dependabot
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c93a3dde8e |
7
.github/workflows/deploy.yml
vendored
7
.github/workflows/deploy.yml
vendored
@ -19,11 +19,6 @@ 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@0c5077e51419868618aeaa5fe8019c62421857d6 # v2.2.0
|
||||
with:
|
||||
@ -36,7 +31,7 @@ jobs:
|
||||
run: bun run build
|
||||
|
||||
- name: Upload artifact
|
||||
uses: actions/upload-pages-artifact@fc324d3547104276b827a68afc52ff2a11cc49c9 # v5.0.0
|
||||
uses: actions/upload-pages-artifact@7b1f4a764d45c48632c6b24a0339c27f5614fb0b # v4.0.0
|
||||
with:
|
||||
path: ./dist
|
||||
|
||||
|
||||
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@48b55a011bda9f5d6aeb4c2d9c7362e8dae4041e # v6.4.0
|
||||
uses: actions/setup-node@53b83947a5a98c8d113130e565377fae1a50d02f # v6.3.0
|
||||
with:
|
||||
node-version: ${{ matrix.node }}
|
||||
|
||||
|
||||
11
CHANGELOG.md
11
CHANGELOG.md
@ -1,19 +1,8 @@
|
||||
# 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).
|
||||
- Showcase/press: add curated community builds and a press page, and surface standout examples on the homepage (#108, thanks @jchopard69).
|
||||
|
||||
## 2026-03-07
|
||||
|
||||
|
||||
229
OPTIMIZATIONS_SUMMARY.md
Normal file
229
OPTIMIZATIONS_SUMMARY.md
Normal file
@ -0,0 +1,229 @@
|
||||
# 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,24 +48,14 @@ 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, or from a pnpm-backed git checkout with `--install-method git`
|
||||
3. Install openclaw globally via npm
|
||||
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": "openclaw-ai",
|
||||
"name": "clawd-bot-landing",
|
||||
"dependencies": {
|
||||
"@astrojs/rss": "^4.0.18",
|
||||
"@lucide/astro": "^1.11.0",
|
||||
"@vercel/analytics": "^2.0.1",
|
||||
"astro": "^6.1.9",
|
||||
"@astrojs/rss": "^4.0.15",
|
||||
"@lucide/astro": "^0.577.0",
|
||||
"@vercel/analytics": "^1.6.1",
|
||||
"astro": "^5.18.0",
|
||||
"js-yaml": "^4.1.1",
|
||||
"simple-icons": "^16.18.0",
|
||||
"simple-icons": "^16.10.0",
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/js-yaml": "^4.0.9",
|
||||
@ -18,32 +18,28 @@
|
||||
},
|
||||
},
|
||||
"packages": {
|
||||
"@astrojs/compiler": ["@astrojs/compiler@3.0.1", "", {}, "sha512-z97oYbdebO5aoWzuJ/8q5hLK232+17KcLZ7cJ8BCWk6+qNzVxn/gftC0KzMBUTD8WAaBkPpNSQK6PXLnNrZ0CA=="],
|
||||
"@astrojs/compiler": ["@astrojs/compiler@2.13.0", "", {}, "sha512-mqVORhUJViA28fwHYaWmsXSzLO9osbdZ5ImUfxBarqsYdMlPbqAqGJCxsNzvppp1BEzc1mJNjOVvQqeDN8Vspw=="],
|
||||
|
||||
"@astrojs/internal-helpers": ["@astrojs/internal-helpers@0.9.0", "", { "dependencies": { "picomatch": "^4.0.4" } }, "sha512-GdYkzR26re8izmyYlBqf4z2s7zNngmWLFuxw0UKiPNqHraZGS6GKWIwSHgS22RDlu2ePFJ8bzmpBcUszut/SDg=="],
|
||||
"@astrojs/internal-helpers": ["@astrojs/internal-helpers@0.7.5", "", {}, "sha512-vreGnYSSKhAjFJCWAwe/CNhONvoc5lokxtRoZims+0wa3KbHBdPHSSthJsKxPd8d/aic6lWKpRTYGY/hsgK6EA=="],
|
||||
|
||||
"@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/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/prism": ["@astrojs/prism@4.0.1", "", { "dependencies": { "prismjs": "^1.30.0" } }, "sha512-nksZQVjlferuWzhPsBpQ1JE5XuKAf1id1/9Hj4a9KG4+ofrlzxUUwX4YGQF/SuDiuiGKEnzopGOt38F3AnVWsQ=="],
|
||||
"@astrojs/prism": ["@astrojs/prism@3.3.0", "", { "dependencies": { "prismjs": "^1.30.0" } }, "sha512-q8VwfU/fDZNoDOf+r7jUnMC2//H2l0TuQ6FkGJL8vD8nw/q5KiL3DS1KKBI3QhI9UQhpJ5dc7AtqfbXWuOgLCQ=="],
|
||||
|
||||
"@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/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/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=="],
|
||||
"@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=="],
|
||||
|
||||
"@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.29.2", "", { "dependencies": { "@babel/types": "^7.29.0" }, "bin": "./bin/babel-parser.js" }, "sha512-4GgRzy/+fsBa72/RZVJmGKPmZu9Byn8o4MoLpmNe1m8ZfYnz5emHLQz3U4gLud6Zwl0RZIcgiLD7Uq7ySFuDLA=="],
|
||||
"@babel/parser": ["@babel/parser@7.28.5", "", { "dependencies": { "@babel/types": "^7.28.5" }, "bin": "./bin/babel-parser.js" }, "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ=="],
|
||||
|
||||
"@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=="],
|
||||
"@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=="],
|
||||
|
||||
"@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=="],
|
||||
@ -150,9 +146,7 @@
|
||||
|
||||
"@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.5", "", {}, "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og=="],
|
||||
|
||||
"@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=="],
|
||||
"@lucide/astro": ["@lucide/astro@0.577.0", "", { "peerDependencies": { "astro": "^4 || ^5" } }, "sha512-xZcdx+MyYXY74cBzFxr9N49SlofbGjCVD9GxJZcJA40PJvJSLWSeJpD9bgKFcZfFxf1QXkEYaR6BnCS+0lek8Q=="],
|
||||
|
||||
"@oslojs/encoding": ["@oslojs/encoding@1.1.0", "", {}, "sha512-70wQhgYmndg4GCPxPPxPGevRKqTIJ2Nh4OkiMWmDAVYsTQ+Ta7Sq+rPevXyXGdzr30/qZBnyOalCszoMxlyldQ=="],
|
||||
|
||||
@ -208,19 +202,17 @@
|
||||
|
||||
"@rollup/rollup-win32-x64-msvc": ["@rollup/rollup-win32-x64-msvc@4.55.1", "", { "os": "win32", "cpu": "x64" }, "sha512-SPEpaL6DX4rmcXtnhdrQYgzQ5W2uW3SCJch88lB2zImhJRhIIK44fkUrgIV/Q8yUNfw5oyZ5vkeQsZLhCb06lw=="],
|
||||
|
||||
"@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/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/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-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-oniguruma": ["@shikijs/engine-oniguruma@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2", "@shikijs/vscode-textmate": "^10.0.2" } }, "sha512-UpCB9Y2sUKlS9z8juFSKz7ZtysmeXCgnRF0dlhXBkmQnek7lAToPte8DkxmEYGNTMii72zU/lyXiCB6StuZeJg=="],
|
||||
"@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/langs": ["@shikijs/langs@4.0.2", "", { "dependencies": { "@shikijs/types": "4.0.2" } }, "sha512-KaXby5dvoeuZzN0rYQiPMjFoUrz4hgwIE+D6Du9owcHcl6/g16/yT5BQxSW5cGt2MZBz6Hl0YuRqf12omRfUUg=="],
|
||||
"@shikijs/langs": ["@shikijs/langs@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0" } }, "sha512-x/42TfhWmp6H00T6uwVrdTJGKgNdFbrEdhaDwSR5fd5zhQ1Q46bHq9EO61SCEWJR0HY7z2HNDMaBZp8JRmKiIA=="],
|
||||
|
||||
"@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/themes": ["@shikijs/themes@3.22.0", "", { "dependencies": { "@shikijs/types": "3.22.0" } }, "sha512-o+tlOKqsr6FE4+mYJG08tfCFDS+3CG20HbldXeVoyP+cYSUxDhrFf3GPjE60U55iOkkjbpY2uC3It/eeja35/g=="],
|
||||
|
||||
"@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/types": ["@shikijs/types@3.22.0", "", { "dependencies": { "@shikijs/vscode-textmate": "^10.0.2", "@types/hast": "^3.0.4" } }, "sha512-491iAekgKDBFE67z70Ok5a8KBMsQ2IJwOWw3us/7ffQkIBCyOQfm/aNwVMBUriP02QshIfgHCBSIYAl3u2eWjg=="],
|
||||
|
||||
"@shikijs/vscode-textmate": ["@shikijs/vscode-textmate@10.0.2", "", {}, "sha512-83yeghZ2xxin3Nj8z1NMd/NCuca+gsYXswywDy5bHvwlWL8tpTQmzGeUuHd9FC3E/SBEMvzJRwWEOz5gGes9Qg=="],
|
||||
|
||||
@ -238,11 +230,21 @@
|
||||
|
||||
"@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@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=="],
|
||||
"@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=="],
|
||||
|
||||
"anymatch": ["anymatch@3.1.3", "", { "dependencies": { "normalize-path": "^3.0.0", "picomatch": "^2.0.4" } }, "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw=="],
|
||||
|
||||
@ -252,16 +254,24 @@
|
||||
|
||||
"array-iterate": ["array-iterate@2.0.1", "", {}, "sha512-I1jXZMjAgCMmxT4qxXfPXa6SthSoE8h6gkSI9BGGNv8mP8G/v0blc+qFnZu6K42vTOiuME596QaLO0TP3Lk0xg=="],
|
||||
|
||||
"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=="],
|
||||
"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=="],
|
||||
|
||||
"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=="],
|
||||
@ -270,7 +280,9 @@
|
||||
|
||||
"chokidar": ["chokidar@5.0.0", "", { "dependencies": { "readdirp": "^5.0.0" } }, "sha512-TQMmc3w+5AxjpL8iIiwebF73dRDF4fBIieAqGn9RGCWaEVwQ6Fb2cGe31Yns0RRIzii5goJ1Y7xbMwo1TxMplw=="],
|
||||
|
||||
"ci-info": ["ci-info@4.4.0", "", {}, "sha512-77PSwercCZU2Fc4sX94eF8k8Pxte6JAwL4/ICZLFjJLqegs7kCuAsqqj/70NQF6TvDpgFjkubQB2FW2ZZddvQg=="],
|
||||
"ci-info": ["ci-info@4.3.1", "", {}, "sha512-Wdy2Igu8OcBpI2pZePZ5oWjPC38tmDVx5WKUXKwlLYkA0ozo85sLsLvkBbBn/sZaSCMFOGZJ14fvW9t5/d7kdA=="],
|
||||
|
||||
"cli-boxes": ["cli-boxes@3.0.0", "", {}, "sha512-/lzGpEWL/8PfI0BmBOPRwp0c/wFNX1RdUML3jK/RcSBA9T8mZDdQpqYBKtCFTOfQbwPqWEOpjqW+Fnayc0969g=="],
|
||||
|
||||
"clsx": ["clsx@2.1.1", "", {}, "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA=="],
|
||||
|
||||
@ -278,11 +290,11 @@
|
||||
|
||||
"commander": ["commander@11.1.0", "", {}, "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ=="],
|
||||
|
||||
"common-ancestor-path": ["common-ancestor-path@2.0.0", "", {}, "sha512-dnN3ibLeoRf2HNC+OlCiNc5d2zxbLJXOtiZUudNFSXZrNSydxcCsSpRzXwfu7BBWCIfHPw+xTayeBvJCP/D8Ng=="],
|
||||
"common-ancestor-path": ["common-ancestor-path@1.0.1", "", {}, "sha512-L3sHRo1pXXEqX8VU28kfgUY+YGsk09hPqZiZmLacNib6XNTCM8ubYeT7ryXQw8asB1sKgcU5lkB7ONug08aB8w=="],
|
||||
|
||||
"cookie": ["cookie@1.1.1", "", {}, "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ=="],
|
||||
|
||||
"cookie-es": ["cookie-es@1.2.3", "", {}, "sha512-lXVyvUvrNXblMqzIRrxHb57UUVmqsSWlxqt3XIjCkUP0wDAf6uicO6KMbEgYrMNtEvWgWHwe42CKxPu9MYAnWw=="],
|
||||
"cookie-es": ["cookie-es@1.2.2", "", {}, "sha512-+W7VmiVINB+ywl1HGXJXmrqkOhpKrIiVZV6tQuV54ZyQC7MMuBt81Vc336GMLoHBq5hV/F9eXgt5Mnx0Rha5Fg=="],
|
||||
|
||||
"crossws": ["crossws@0.3.5", "", { "dependencies": { "uncrypto": "^0.1.3" } }, "sha512-ojKiDvcmByhwa8YYqbQI/hg7MEU0NC03+pSdEq4ZUnZR9xXpwk7E43SMNGkn+JxJGPFtNvQ48+vV2p+P1ml5PA=="],
|
||||
|
||||
@ -292,13 +304,15 @@
|
||||
|
||||
"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.7", "", {}, "sha512-7z22QmUWiQ/2d0KkdYmANbRUVABpZ9SNYyH5vx6PZ+nE5bcC0l7uFvEfHlyld/HcGBFTL536ClDt3DEcSlEJAQ=="],
|
||||
"defu": ["defu@6.1.4", "", {}, "sha512-mEQCMmwJu317oSz8CwdIOdwf3xMif1ttiM8LTufzc3g6kR+9Pe236twL8j3IYT1F7GfRgGcW6MWxzZjLIkuHIg=="],
|
||||
|
||||
"dequal": ["dequal@2.0.3", "", {}, "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA=="],
|
||||
|
||||
@ -306,7 +320,9 @@
|
||||
|
||||
"detect-libc": ["detect-libc@2.1.2", "", {}, "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ=="],
|
||||
|
||||
"devalue": ["devalue@5.7.1", "", {}, "sha512-MUbZ586EgQqdRnC4yDrlod3BEdyvE4TapGYHMW2CiaW+KkkFmWEFqBUaLltEZCGi0iFXCEjRF0OjF0DV2QHjOA=="],
|
||||
"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=="],
|
||||
|
||||
"devlop": ["devlop@1.1.0", "", { "dependencies": { "dequal": "^2.0.0" } }, "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA=="],
|
||||
|
||||
@ -324,43 +340,39 @@
|
||||
|
||||
"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@2.1.0", "", {}, "sha512-n27zTYMjYu1aj4MjCWzSP7G9r75utsaoc8m61weK+W8JMBGGQybd43GstCXZ3WNmSFtGT9wi59qQTW6mhTR5LQ=="],
|
||||
"es-module-lexer": ["es-module-lexer@1.7.0", "", {}, "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA=="],
|
||||
|
||||
"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@2.0.2", "", {}, "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="],
|
||||
"estree-walker": ["estree-walker@3.0.3", "", { "dependencies": { "@types/estree": "^1.0.0" } }, "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g=="],
|
||||
|
||||
"eventemitter3": ["eventemitter3@5.0.1", "", {}, "sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA=="],
|
||||
|
||||
"extend": ["extend@3.0.2", "", {}, "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g=="],
|
||||
|
||||
"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=="],
|
||||
"fast-xml-parser": ["fast-xml-parser@5.3.5", "", { "dependencies": { "strnum": "^2.1.2" }, "bin": { "fxparser": "src/cli/cli.js" } }, "sha512-JeaA2Vm9ffQKp9VjvfzObuMCjUYAp5WDYhRYL5LrBPY/jUDlUtOvDfot0vKSkB9tuX885BDHjtw4fZadD95wnA=="],
|
||||
|
||||
"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.1", "", { "dependencies": { "fontkitten": "^1.0.2" } }, "sha512-lDMvbAzSnHmbYMTEld5qdtvNH2/pWpICOqpean9IgC7vUbUJc3k+k5Dokp85CegamqQpFbXf0rAVkbzpyTA8aw=="],
|
||||
"fontace": ["fontace@0.4.0", "", { "dependencies": { "fontkitten": "^1.0.0" } }, "sha512-moThBCItUe2bjZip5PF/iZClpKHGLwMvR79Kp8XpGRBrvoRSnySN4VcILdv3/MJzbhvUA5WeiUXF5o538m5fvg=="],
|
||||
|
||||
"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.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=="],
|
||||
"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=="],
|
||||
|
||||
"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=="],
|
||||
|
||||
@ -388,25 +400,31 @@
|
||||
|
||||
"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@4.0.0", "", { "bin": { "is-docker": "cli.js" } }, "sha512-LHE+wROyG/Y/0ZnbktRCoTix2c1RhgWaZraMZ8o1Q7zCh0VSrICJQO5oqIIISrcSBtrXv0o233w1IYwsWCjTzA=="],
|
||||
"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-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.1", "", { "dependencies": { "is-inside-container": "^1.0.0" } }, "sha512-e6rvdUCiQCAuumZslxRJWR/Doq4VpPR82kqclvcS0efgt430SlGIk05vdCN58+VrzgtIcfNODjozVielycD4Sw=="],
|
||||
"is-wsl": ["is-wsl@3.1.0", "", { "dependencies": { "is-inside-container": "^1.0.0" } }, "sha512-UcVfVfaK4Sc4m7X3dUSoHoozQGBEFeDC+zVo06t98xe8CzHSZZBekNXH+tu0NalHolcJ/QAGqS46Hef7QXBIMw=="],
|
||||
|
||||
"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.3.5", "", {}, "sha512-NxVFwLAnrd9i7KUBxC4DrUhmgjzOs+1Qm50D3oF1/oL+r1NpZ4gA7xvG0/zJ8evR7zIKn4vLf7qTNduWFtCrRw=="],
|
||||
"lru-cache": ["lru-cache@11.2.5", "", {}, "sha512-vFrFJkWtJvJnD5hg+hJvVE8Lh/TcMzKnTgCWmtBipwI5yLX/iX+5UB2tfuyODF5E7k9xEzMdYgGqaSb1c0c5Yw=="],
|
||||
|
||||
"magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="],
|
||||
|
||||
"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=="],
|
||||
"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=="],
|
||||
|
||||
"markdown-table": ["markdown-table@3.0.4", "", {}, "sha512-wiYz4+JrLyb/DqW2hkFJxP7Vd7JuTDm77fvbM8VfEQdmSMqcImWeeRbHwZjBjIFki/VaMK2BhFi7oUUZeM5bqw=="],
|
||||
|
||||
@ -512,8 +530,6 @@
|
||||
|
||||
"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=="],
|
||||
@ -522,11 +538,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@7.3.0", "", { "dependencies": { "yocto-queue": "^1.2.1" } }, "sha512-7cIXg/Z0M5WZRblrsOla88S4wAK+zOQQWeBYfV3qJuJXMr+LnbYjaadrFaS0JILfEDPVqHyKnZ1Z/1d6J9VVUw=="],
|
||||
"p-limit": ["p-limit@6.2.0", "", { "dependencies": { "yocto-queue": "^1.1.1" } }, "sha512-kuUqqHNUqoIWp/c467RI4X6mmyuojY5jGutNU0wVTmEOOfcuwLqyMVoAi9MKi2Ak+5i9+nhmrK4ufZE8069kHA=="],
|
||||
|
||||
"p-queue": ["p-queue@9.1.2", "", { "dependencies": { "eventemitter3": "^5.0.1", "p-timeout": "^7.0.0" } }, "sha512-ktsDOALzTYTWWF1PbkNVg2rOt+HaOaMWJMUnt7T3qf5tvZ1L8dBW3tObzprBcXNMKkwj+yFSLqHso0x+UFcJXw=="],
|
||||
"p-queue": ["p-queue@8.1.1", "", { "dependencies": { "eventemitter3": "^5.0.1", "p-timeout": "^6.1.2" } }, "sha512-aNZ+VfjobsWryoiPnEApGGmf5WmNsCo9xu8dfaYamG5qaLP7ClhLN6NgsFe6SwJ2UbLEBK5dv9x8Mn5+RVhMWQ=="],
|
||||
|
||||
"p-timeout": ["p-timeout@7.0.1", "", {}, "sha512-AxTM2wDGORHGEkPCt8yqxOTMgpfbEHqF51f/5fJCmwFC3C/zNcGT63SymH2ttOAaiIws2zVg4+izQCjrakcwHg=="],
|
||||
"p-timeout": ["p-timeout@6.1.4", "", {}, "sha512-MyIV3ZA/PmyBN/ud8vV9XzwTrNtR4jFrObymZYnZqMmW0zA8Z17vnT0rBgFE/TlohB+YCHqXMgZzb3Csp49vqg=="],
|
||||
|
||||
"package-manager-detector": ["package-manager-detector@1.6.0", "", {}, "sha512-61A5ThoTiDG/C8s8UMZwSorAGwMJ0ERVGj2OjoW5pAalsNOg15+iQiPzrLJ4jhZ1HJzmC2PIHT2oEiH3R5fzNA=="],
|
||||
|
||||
@ -534,18 +550,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.4", "", {}, "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A=="],
|
||||
"picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||
|
||||
"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=="],
|
||||
@ -586,15 +602,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.6.0", "", {}, "sha512-6R3J5M4AcbtLUdZmRv2SygeVaM7IhrLXu9BmnOGmmACak8fiUtOsYNWUS4uK7upbmHIBbLBeFeI//477BKLBzA=="],
|
||||
"sax": ["sax@1.4.3", "", {}, "sha512-yqYn1JhPczigF94DMS+shiDMjDowYO6y9+wB/4WgO0Y19jWYk0lQ4tuG5KI7kj4FTp1wxPj5IFfcrz/s1c3jjQ=="],
|
||||
|
||||
"semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="],
|
||||
"semver": ["semver@7.7.3", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q=="],
|
||||
|
||||
"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@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=="],
|
||||
"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=="],
|
||||
|
||||
"simple-icons": ["simple-icons@16.18.0", "", {}, "sha512-5KbjcP456Gm1lrk+rhDuX4zFri+3lRX39IjzXAvoMAO8Ne76WlVlM+Z3kA6jdZ7+QHadgXsf++R7g2jaYVbYig=="],
|
||||
"simple-icons": ["simple-icons@16.10.0", "", {}, "sha512-62kuxaG3pE+cFNerudUtwb9BLmudzayHrlHkvU9gf8Nxcj7VYOm9dh3WNbGaFk60aQtfnRyzViZaouFG2B45kg=="],
|
||||
|
||||
"sisteransi": ["sisteransi@1.0.5", "", {}, "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg=="],
|
||||
|
||||
@ -604,17 +620,19 @@
|
||||
|
||||
"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=="],
|
||||
|
||||
"strnum": ["strnum@2.2.3", "", {}, "sha512-oKx6RUCuHfT3oyVjtnrmn19H1SiCqgJSg+54XqURKp5aCMbrXrhLjRN9TjuwMjiYstZ0MzDrHqkGZ5dFTKd+zg=="],
|
||||
"strip-ansi": ["strip-ansi@7.1.2", "", { "dependencies": { "ansi-regex": "^6.0.1" } }, "sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA=="],
|
||||
|
||||
"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=="],
|
||||
"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=="],
|
||||
|
||||
"tiny-inflate": ["tiny-inflate@1.0.3", "", {}, "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw=="],
|
||||
|
||||
"tinyclip": ["tinyclip@0.1.12", "", {}, "sha512-Ae3OVUqifDw0wBriIBS7yVaW44Dp6eSHQcyq4Igc7eN2TJH/2YsicswaW+J/OuMvhpDPOKEgpAZCjkb4hpoyeA=="],
|
||||
|
||||
"tinyexec": ["tinyexec@1.1.1", "", {}, "sha512-VKS/ZaQhhkKFMANmAOhhXVoIfBXblQxGX1myCQ2faQrfmobMftXeJPcZGp0gS07ocvGJWDLZGyOZDadDBqYIJg=="],
|
||||
"tinyexec": ["tinyexec@1.0.2", "", {}, "sha512-W/KYk+NFhkmsYpuHq5JykngiOCnxeVL8v8dFnqxSD8qEEdRfXk1SDM6JzNqcERbcGYj9tMrDQBYV9cjgnunFIg=="],
|
||||
|
||||
"tinyglobby": ["tinyglobby@0.2.15", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.3" } }, "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ=="],
|
||||
|
||||
@ -626,6 +644,8 @@
|
||||
|
||||
"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=="],
|
||||
@ -634,9 +654,11 @@
|
||||
|
||||
"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.4", "", { "dependencies": { "css-tree": "^3.1.0", "ofetch": "^1.5.1", "ohash": "^2.0.11" } }, "sha512-oHeis4/xl42HUIeHuNZRGEvxj5AaIKR+bHPNegRq5LV1gdc3jundpONbjglKpihmJf+dswygdMJn3eftGIMemg=="],
|
||||
"unifont": ["unifont@0.7.3", "", { "dependencies": { "css-tree": "^3.1.0", "ofetch": "^1.5.1", "ohash": "^2.0.11" } }, "sha512-b0GtQzKCyuSHGsfj5vyN8st7muZ6VCI4XD4vFlr7Uy1rlWVYxC3npnfk8MyreHxJYrz1ooLDqDzFe9XqQTlAhA=="],
|
||||
|
||||
"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=="],
|
||||
|
||||
@ -650,13 +672,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.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": ["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-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.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=="],
|
||||
"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=="],
|
||||
|
||||
"vfile": ["vfile@6.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "vfile-message": "^4.0.0" } }, "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q=="],
|
||||
|
||||
@ -664,25 +686,41 @@
|
||||
|
||||
"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@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=="],
|
||||
"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=="],
|
||||
|
||||
"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=="],
|
||||
"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=="],
|
||||
|
||||
"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@22.0.0", "", {}, "sha512-rwu/ClNdSMpkSrUb+d6BRsSkLUq1fmfsY6TOpYzTwvwkg1/NRG85KBy3kq++A8LKQwX6lsu+aWad+2khvuXrqw=="],
|
||||
"yargs-parser": ["yargs-parser@21.1.1", "", {}, "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw=="],
|
||||
|
||||
"yocto-queue": ["yocto-queue@1.2.2", "", {}, "sha512-4LCcse/U2MHZ63HAJVE+v71o7yOdIe4cZ70Wpf8D/IyjDKYQLV5GD46B+hSTjJsvV5PztjvHoU580EftxjDZFQ=="],
|
||||
|
||||
"zod": ["zod@4.3.6", "", {}, "sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg=="],
|
||||
"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=="],
|
||||
|
||||
"zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="],
|
||||
|
||||
"@rollup/pluginutils/picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||
"@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=="],
|
||||
|
||||
"anymatch/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
|
||||
|
||||
@ -690,30 +728,80 @@
|
||||
|
||||
"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=="],
|
||||
|
||||
"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=="],
|
||||
"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=="],
|
||||
|
||||
"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/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=="],
|
||||
|
||||
"sharp/semver": ["semver@7.7.3", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q=="],
|
||||
"@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=="],
|
||||
|
||||
"tinyglobby/picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||
"@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=="],
|
||||
|
||||
"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=="],
|
||||
"@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=="],
|
||||
|
||||
"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": "openclaw-ai",
|
||||
"name": "clawd-bot-landing",
|
||||
"type": "module",
|
||||
"version": "1.0.0",
|
||||
"scripts": {
|
||||
@ -14,12 +14,12 @@
|
||||
]
|
||||
},
|
||||
"dependencies": {
|
||||
"@astrojs/rss": "^4.0.18",
|
||||
"@lucide/astro": "^1.11.0",
|
||||
"@vercel/analytics": "^2.0.1",
|
||||
"astro": "^6.1.9",
|
||||
"@astrojs/rss": "^4.0.15",
|
||||
"@lucide/astro": "^0.577.0",
|
||||
"@vercel/analytics": "^1.6.1",
|
||||
"astro": "^5.18.0",
|
||||
"js-yaml": "^4.1.1",
|
||||
"simple-icons": "^16.18.0"
|
||||
"simple-icons": "^16.10.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/js-yaml": "^4.0.9"
|
||||
|
||||
@ -4,31 +4,6 @@ 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}"
|
||||
@ -500,22 +475,21 @@ install_openclaw() {
|
||||
fi
|
||||
|
||||
if [[ "${requested}" == "latest" ]]; then
|
||||
if ! SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$(node_dir)" "${npm_args[@]}" "openclaw@latest"; then
|
||||
if ! SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$PREFIX" "${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 "$(node_dir)" "${npm_args[@]}" "openclaw@next"
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$PREFIX" "${npm_args[@]}" "openclaw@next"
|
||||
requested="next"
|
||||
fi
|
||||
else
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$(node_dir)" "${npm_args[@]}" "openclaw@${requested}"
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS="$SHARP_IGNORE_GLOBAL_LIBVIPS" "$(npm_bin)" install -g --prefix "$PREFIX" "${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" "$(node_dir)/lib/node_modules/openclaw/dist/entry.js" "\$@"
|
||||
exec "${PREFIX}/tools/node/bin/node" "${PREFIX}/lib/node_modules/openclaw/dist/entry.js" "\$@"
|
||||
EOF
|
||||
chmod +x "${PREFIX}/bin/openclaw"
|
||||
emit_json "{\"event\":\"step\",\"name\":\"openclaw\",\"status\":\"ok\",\"version\":\"${requested}\"}"
|
||||
|
||||
@ -14,29 +14,6 @@ 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 ""
|
||||
@ -44,8 +21,7 @@ Write-Host ""
|
||||
# Check if running in PowerShell
|
||||
if ($PSVersionTable.PSVersion.Major -lt 5) {
|
||||
Write-Host "Error: PowerShell 5+ required" -ForegroundColor Red
|
||||
Complete-Install -Succeeded:$false
|
||||
return
|
||||
exit 1
|
||||
}
|
||||
|
||||
Write-Host "[OK] Windows detected" -ForegroundColor Green
|
||||
@ -115,11 +91,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 $true
|
||||
return
|
||||
}
|
||||
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
|
||||
return $false
|
||||
exit 1
|
||||
}
|
||||
|
||||
# Try Chocolatey
|
||||
@ -130,7 +106,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 $true
|
||||
return
|
||||
}
|
||||
|
||||
# Try Scoop
|
||||
@ -138,7 +114,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 $true
|
||||
return
|
||||
}
|
||||
|
||||
# Manual download fallback
|
||||
@ -149,7 +125,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
|
||||
return $false
|
||||
exit 1
|
||||
}
|
||||
|
||||
# Check for existing OpenClaw installation
|
||||
@ -305,12 +281,12 @@ function Install-PortableGit {
|
||||
}
|
||||
|
||||
function Ensure-Git {
|
||||
if (Check-Git) { return $true }
|
||||
if (Use-PortableGitIfPresent) { return $true }
|
||||
if (Check-Git) { return }
|
||||
if (Use-PortableGitIfPresent) { return }
|
||||
try {
|
||||
Install-PortableGit
|
||||
if (Check-Git) {
|
||||
return $true
|
||||
return
|
||||
}
|
||||
} catch {
|
||||
Write-Host "[!] Portable Git bootstrap failed: $($_.Exception.Message)" -ForegroundColor Yellow
|
||||
@ -321,7 +297,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
|
||||
return $false
|
||||
exit 1
|
||||
}
|
||||
|
||||
function Get-OpenClawCommandPath {
|
||||
@ -470,46 +446,18 @@ 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"
|
||||
}
|
||||
if (-not (Ensure-Git)) {
|
||||
return $false
|
||||
}
|
||||
Ensure-Git
|
||||
|
||||
# Use openclaw package for beta, openclaw for stable
|
||||
$packageName = "openclaw"
|
||||
if ($Tag -eq "beta" -or $Tag -match "^beta\.") {
|
||||
$packageName = "openclaw"
|
||||
}
|
||||
$installSpec = Resolve-NpmOpenClawInstallSpec -PackageName $packageName -RequestedTag $Tag
|
||||
Write-Host "[*] Installing OpenClaw ($installSpec)..." -ForegroundColor Yellow
|
||||
Write-Host "[*] Installing OpenClaw ($packageName@$Tag)..." -ForegroundColor Yellow
|
||||
$prevLogLevel = $env:NPM_CONFIG_LOGLEVEL
|
||||
$prevUpdateNotifier = $env:NPM_CONFIG_UPDATE_NOTIFIER
|
||||
$prevFund = $env:NPM_CONFIG_FUND
|
||||
@ -523,7 +471,7 @@ function Install-OpenClaw {
|
||||
$env:NPM_CONFIG_SCRIPT_SHELL = "cmd.exe"
|
||||
$env:NODE_LLAMA_CPP_SKIP_DOWNLOAD = "1"
|
||||
try {
|
||||
$npmOutput = & (Get-NpmCommandPath) install -g "$installSpec" 2>&1
|
||||
$npmOutput = & (Get-NpmCommandPath) install -g "$packageName@$Tag" 2>&1
|
||||
if ($LASTEXITCODE -ne 0) {
|
||||
Write-Host "[!] npm install failed" -ForegroundColor Red
|
||||
if ($npmOutput -match "spawn git" -or $npmOutput -match "ENOENT.*git") {
|
||||
@ -535,7 +483,7 @@ function Install-OpenClaw {
|
||||
Write-Host ' powershell -c "irm https://openclaw.ai/install.ps1 | iex"' -ForegroundColor Cyan
|
||||
}
|
||||
$npmOutput | ForEach-Object { Write-Host $_ }
|
||||
return $false
|
||||
exit 1
|
||||
}
|
||||
} finally {
|
||||
$env:NPM_CONFIG_LOGLEVEL = $prevLogLevel
|
||||
@ -546,7 +494,6 @@ function Install-OpenClaw {
|
||||
$env:NODE_LLAMA_CPP_SKIP_DOWNLOAD = $prevNodeLlamaSkipDownload
|
||||
}
|
||||
Write-Host "[OK] OpenClaw installed" -ForegroundColor Green
|
||||
return $true
|
||||
}
|
||||
|
||||
# Install OpenClaw from GitHub
|
||||
@ -555,9 +502,7 @@ function Install-OpenClawFromGit {
|
||||
[string]$RepoDir,
|
||||
[switch]$SkipUpdate
|
||||
)
|
||||
if (-not (Ensure-Git)) {
|
||||
return $false
|
||||
}
|
||||
Ensure-Git
|
||||
Ensure-Pnpm
|
||||
|
||||
$repoUrl = "https://github.com/openclaw/openclaw.git"
|
||||
@ -612,7 +557,6 @@ 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)
|
||||
@ -693,7 +637,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
|
||||
return (Fail-Install -Code 2)
|
||||
exit 2
|
||||
}
|
||||
|
||||
if ($DryRun) {
|
||||
@ -710,7 +654,7 @@ function Main {
|
||||
if ($NoOnboard) {
|
||||
Write-Host "[OK] Onboard: skipped" -ForegroundColor Green
|
||||
}
|
||||
return $true
|
||||
return
|
||||
}
|
||||
|
||||
Remove-LegacySubmodule -RepoDir $RepoDir
|
||||
@ -720,16 +664,14 @@ function Main {
|
||||
|
||||
# Step 1: Node.js
|
||||
if (-not (Check-Node)) {
|
||||
if (-not (Install-Node)) {
|
||||
return (Fail-Install)
|
||||
}
|
||||
Install-Node
|
||||
|
||||
# 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
|
||||
return (Fail-Install)
|
||||
exit 1
|
||||
}
|
||||
}
|
||||
|
||||
@ -737,26 +679,10 @@ 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
|
||||
if (-not (Install-OpenClawFromGit -RepoDir $GitDir -SkipUpdate:$NoGitUpdate)) {
|
||||
return (Fail-Install)
|
||||
}
|
||||
Install-OpenClawFromGit -RepoDir $GitDir -SkipUpdate:$NoGitUpdate
|
||||
} else {
|
||||
$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)
|
||||
}
|
||||
Install-OpenClaw
|
||||
}
|
||||
|
||||
if (-not (Ensure-OpenClawOnPath)) {
|
||||
@ -859,10 +785,6 @@ function Main {
|
||||
Invoke-OpenClawCommand onboard
|
||||
}
|
||||
}
|
||||
|
||||
return $true
|
||||
}
|
||||
|
||||
$mainResults = @(Main)
|
||||
$installSucceeded = $mainResults.Count -gt 0 -and $mainResults[-1] -eq $true
|
||||
Complete-Install -Succeeded:$installSucceeded
|
||||
Main
|
||||
|
||||
@ -19,31 +19,6 @@ 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
|
||||
@ -1018,9 +993,6 @@ 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=()
|
||||
@ -1040,9 +1012,6 @@ 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)
|
||||
@ -1056,9 +1025,6 @@ 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
|
||||
@ -1070,7 +1036,6 @@ 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
|
||||
}
|
||||
|
||||
@ -1125,18 +1090,6 @@ 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
|
||||
@ -1943,148 +1896,6 @@ 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
|
||||
}
|
||||
@ -2395,12 +2206,7 @@ maybe_open_dashboard() {
|
||||
}
|
||||
|
||||
resolve_workspace_dir() {
|
||||
if [[ -n "${INSTALL_WORKSPACE}" ]]; then
|
||||
echo "${INSTALL_WORKSPACE}"
|
||||
return
|
||||
fi
|
||||
local profile=""
|
||||
profile="$(resolve_install_profile)"
|
||||
local profile="${OPENCLAW_PROFILE:-default}"
|
||||
if [[ "${profile}" != "default" ]]; then
|
||||
echo "${HOME}/.openclaw/workspace-${profile}"
|
||||
else
|
||||
@ -2408,133 +2214,13 @@ 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
|
||||
|
||||
if install_config_already_exists; then
|
||||
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
|
||||
return
|
||||
fi
|
||||
|
||||
@ -2547,9 +2233,7 @@ run_bootstrap_onboarding_if_needed() {
|
||||
fi
|
||||
|
||||
if [[ ! -r /dev/tty || ! -w /dev/tty ]]; then
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command)"
|
||||
ui_info "BOOTSTRAP.md found but no TTY; run ${onboard_cmd} to finish setup"
|
||||
ui_info "BOOTSTRAP.md found but no TTY; run openclaw onboard to finish setup"
|
||||
return
|
||||
fi
|
||||
|
||||
@ -2564,11 +2248,8 @@ run_bootstrap_onboarding_if_needed() {
|
||||
return
|
||||
fi
|
||||
|
||||
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"
|
||||
"$claw" onboard || {
|
||||
ui_error "Onboarding failed; run openclaw onboard to retry"
|
||||
ui_info "If gateway startup looks unhealthy, run: openclaw gateway status --deep"
|
||||
return
|
||||
}
|
||||
@ -2656,8 +2337,6 @@ main() {
|
||||
return 0
|
||||
fi
|
||||
|
||||
validate_install_overrides
|
||||
|
||||
bootstrap_gum_temp || true
|
||||
print_installer_banner
|
||||
print_gum_status
|
||||
@ -2770,7 +2449,6 @@ 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=""
|
||||
@ -2857,7 +2535,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"
|
||||
ui_kv "Update command" "openclaw update --restart"
|
||||
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"
|
||||
@ -2893,11 +2571,10 @@ main() {
|
||||
fi
|
||||
else
|
||||
if [[ "$NO_ONBOARD" == "1" || "$skip_onboard" == "true" ]]; then
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command)"
|
||||
ui_info "Skipping onboard (requested); run ${onboard_cmd} later"
|
||||
ui_info "Skipping onboard (requested); run openclaw onboard later"
|
||||
else
|
||||
if install_config_already_exists; then
|
||||
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
|
||||
ui_info "Config already present; running doctor"
|
||||
run_doctor
|
||||
should_open_dashboard=true
|
||||
@ -2917,12 +2594,9 @@ main() {
|
||||
return 0
|
||||
fi
|
||||
exec </dev/tty
|
||||
build_onboard_command "$claw"
|
||||
exec "${ONBOARD_CMD[@]}"
|
||||
exec "$claw" onboard
|
||||
fi
|
||||
local onboard_cmd=""
|
||||
onboard_cmd="$(format_onboard_display_command)"
|
||||
ui_info "No TTY; run ${onboard_cmd} to finish setup"
|
||||
ui_info "No TTY; run openclaw onboard to finish setup"
|
||||
return 0
|
||||
fi
|
||||
fi
|
||||
|
||||
@ -32,42 +32,6 @@ 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"
|
||||
@ -167,51 +131,6 @@ 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"
|
||||
@ -226,7 +145,8 @@ echo "==> case: install_openclaw_from_git uses run_pnpm"
|
||||
export GIT_UPDATE=0
|
||||
export SHARP_IGNORE_GLOBAL_LIBVIPS=1
|
||||
|
||||
run_pnpm_calls=()
|
||||
deps_cmd=""
|
||||
build_cmd=""
|
||||
|
||||
ensure_git() { :; }
|
||||
ensure_pnpm() { set_pnpm_cmd echo pnpm; }
|
||||
@ -236,14 +156,17 @@ echo "==> case: install_openclaw_from_git uses run_pnpm"
|
||||
emit_json() { :; }
|
||||
fail() { echo "FAIL: $*" >&2; exit 1; }
|
||||
run_pnpm() {
|
||||
run_pnpm_calls+=("$*")
|
||||
if [[ -z "$deps_cmd" ]]; then
|
||||
deps_cmd="$1"
|
||||
else
|
||||
build_cmd="$1"
|
||||
fi
|
||||
return 0
|
||||
}
|
||||
|
||||
install_openclaw_from_git "${repo}"
|
||||
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"
|
||||
assert_eq "$deps_cmd" "-C" "install_openclaw_from_git deps command entry"
|
||||
assert_nonempty "$build_cmd" "install_openclaw_from_git build command entry"
|
||||
test -x "${prefix}/bin/openclaw"
|
||||
)
|
||||
|
||||
|
||||
@ -1,63 +0,0 @@
|
||||
#!/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,23 +66,6 @@ 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"
|
||||
@ -150,41 +133,6 @@ 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"
|
||||
@ -532,7 +480,8 @@ echo "==> case: install_openclaw_from_git (deps step uses run_pnpm function)"
|
||||
# shellcheck disable=SC2034
|
||||
SHARP_IGNORE_GLOBAL_LIBVIPS=1
|
||||
|
||||
run_pnpm_steps=()
|
||||
deps_called=0
|
||||
deps_cmd=""
|
||||
|
||||
check_git() { return 0; }
|
||||
install_git() { fail "install_git should not be called"; }
|
||||
@ -549,18 +498,16 @@ echo "==> case: install_openclaw_from_git (deps step uses run_pnpm function)"
|
||||
run_quiet_step() {
|
||||
local _title="$1"
|
||||
shift
|
||||
if [[ "${1:-}" == "run_pnpm" ]]; then
|
||||
shift
|
||||
run_pnpm_steps+=("$*")
|
||||
return 0
|
||||
if [[ "${_title}" == "Installing dependencies" ]]; then
|
||||
deps_called=1
|
||||
deps_cmd="${1:-}"
|
||||
fi
|
||||
"$@" >/dev/null 2>&1 || true
|
||||
}
|
||||
|
||||
install_openclaw_from_git "${repo}"
|
||||
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"
|
||||
assert_eq "$deps_called" "1" "install_openclaw_from_git dependencies step"
|
||||
assert_eq "$deps_cmd" "run_pnpm" "install_openclaw_from_git dependencies command"
|
||||
)
|
||||
|
||||
echo "==> case: ensure_pnpm_git_prepare_allowlist (known dep added once)"
|
||||
|
||||
@ -1,33 +0,0 @@
|
||||
---
|
||||
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.
|
||||
@ -1,83 +0,0 @@
|
||||
---
|
||||
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. 🦞
|
||||
@ -1,5 +1,4 @@
|
||||
import { defineCollection, z } from 'astro:content';
|
||||
import { glob } from 'astro/loaders';
|
||||
|
||||
const authorSchema = z.object({
|
||||
name: z.string(),
|
||||
@ -7,7 +6,7 @@ const authorSchema = z.object({
|
||||
});
|
||||
|
||||
const blog = defineCollection({
|
||||
loader: glob({ pattern: '**/*.md', base: './src/content/blog' }),
|
||||
type: 'content',
|
||||
schema: z.object({
|
||||
title: z.string(),
|
||||
description: z.string(),
|
||||
@ -1,62 +0,0 @@
|
||||
[
|
||||
{
|
||||
"id": "2012778049406742632",
|
||||
"author": "jdrhyne",
|
||||
"title": "15+ Agent Army Running Across 3 Machines",
|
||||
"href": "https://x.com/jdrhyne/status/2012778049406742632",
|
||||
"source": "X thread",
|
||||
"summary": "A full OpenClaw setup clearing 10,000 emails, reviewing decks, building CLI tools, optimizing Google Ads, drafting posts, and orchestrating Codex workers across a Discord-driven agent fleet.",
|
||||
"tags": ["multi-agent", "discord", "automation"],
|
||||
"likes": 119
|
||||
},
|
||||
{
|
||||
"id": "2012565160586625345",
|
||||
"author": "danpeguine",
|
||||
"title": "Personal Operating System With Daily Briefings",
|
||||
"href": "https://x.com/danpeguine/status/2012565160586625345",
|
||||
"source": "X thread",
|
||||
"summary": "OpenClaw timeblocks tasks, runs weekly reviews from meeting notes, briefs before meetings, watches family school deadlines, researches projects, resolves calendar conflicts, and creates invoices.",
|
||||
"tags": ["calendar", "briefings", "productivity"],
|
||||
"likes": 257
|
||||
},
|
||||
{
|
||||
"id": "2012119327147798753",
|
||||
"author": "georgedagg_",
|
||||
"title": "Voice-Guided Production Fix While Walking the Dog",
|
||||
"href": "https://x.com/georgedagg_/status/2012119327147798753",
|
||||
"source": "X thread",
|
||||
"summary": "A live OpenClaw workflow that inspected failed Railway builds, diagnosed the root cause, changed deployment configs, redeployed, fixed a design issue, and submitted a PR over voice.",
|
||||
"tags": ["voice", "deploy", "coding"],
|
||||
"likes": 143
|
||||
},
|
||||
{
|
||||
"id": "2012535486401671588",
|
||||
"author": "dreetje",
|
||||
"title": "One Chat Controlling Mail, Messages, Orders, Calls, and a Vault",
|
||||
"href": "https://x.com/dreetje/status/2012535486401671588",
|
||||
"source": "X thread",
|
||||
"summary": "A single OpenClaw assistant checking mail, reading Beeper messages, ordering things, sending reminders, creating GitHub issues, discussing bookmarks, handling voice calls, and reading and writing a dedicated 1Password vault.",
|
||||
"tags": ["mail", "beeper", "personal-ai"],
|
||||
"likes": 271
|
||||
},
|
||||
{
|
||||
"id": "2008994096736817624",
|
||||
"author": "davekiss",
|
||||
"title": "Entire Website Rebuilt From Telegram",
|
||||
"href": "https://x.com/davekiss/status/2008994096736817624",
|
||||
"source": "X thread",
|
||||
"summary": "A full personal-site rebuild from bed: Notion to Astro, 18 posts migrated, DNS moved to Cloudflare, and no laptop involved.",
|
||||
"tags": ["telegram", "website", "astro"],
|
||||
"likes": 142
|
||||
},
|
||||
{
|
||||
"id": "2007616854689280196",
|
||||
"author": "stevecaldwell",
|
||||
"title": "Family Meal Planning System Built Overnight",
|
||||
"href": "https://x.com/stevecaldwell/status/2007616854689280196",
|
||||
"source": "X thread",
|
||||
"summary": "A weekly family planning system with a year-long meal template, aisle-sorted shopping lists, weather-aware dinner planning, and automated reminders.",
|
||||
"tags": ["family", "notion", "planning"],
|
||||
"likes": 133
|
||||
}
|
||||
]
|
||||
@ -1,65 +0,0 @@
|
||||
[
|
||||
{
|
||||
"outlet": "TechCrunch",
|
||||
"title": "OpenClaw’s AI assistants are now building their own social network",
|
||||
"url": "https://techcrunch.com/2026/01/30/openclaws-ai-assistants-are-now-building-their-own-social-network/",
|
||||
"displayDate": "Jan 30, 2026",
|
||||
"author": "Anna Heim",
|
||||
"summary": "A strong first-read on the OpenClaw takeoff: rebrand, viral growth, Moltbook, and why the project captured the open-source AI crowd.",
|
||||
"featured": true
|
||||
},
|
||||
{
|
||||
"outlet": "TechCrunch",
|
||||
"title": "OpenClaw creator Peter Steinberger joins OpenAI",
|
||||
"url": "https://techcrunch.com/2026/02/15/openclaw-creator-peter-steinberger-joins-openai/",
|
||||
"displayDate": "Feb 15, 2026",
|
||||
"author": "Anthony Ha",
|
||||
"summary": "A clean, positive headline around the next chapter for OpenClaw: founder momentum, OpenAI backing, and the long-term foundation path.",
|
||||
"featured": true
|
||||
},
|
||||
{
|
||||
"outlet": "The Verge",
|
||||
"title": "The OpenClaw superfan meetup serves optimism and lobster",
|
||||
"url": "https://www.theverge.com/ai-artificial-intelligence/890517/openclaw-clawcon-meetup-nyc-open-source-ai",
|
||||
"displayDate": "Mar 7, 2026",
|
||||
"author": "The Verge",
|
||||
"summary": "A strong community story about ClawCon, the people building around OpenClaw, and the feeling that personal agents have escaped the lab.",
|
||||
"featured": false
|
||||
},
|
||||
{
|
||||
"outlet": "The Verge",
|
||||
"title": "OpenClaw creator Peter Steinberger joins OpenAI",
|
||||
"url": "https://www.theverge.com/openai/630450/peter-steinberger-openclaw-openai",
|
||||
"displayDate": "Feb 15, 2026",
|
||||
"author": "The Verge",
|
||||
"summary": "Another positive institutional signal: OpenClaw’s creator moving into a major platform role while the project continues as open source.",
|
||||
"featured": false
|
||||
},
|
||||
{
|
||||
"outlet": "Business Insider",
|
||||
"title": "OpenClaw's creator is heading to OpenAI. He says it could've been a 'huge company,' but building one didn't excite him.",
|
||||
"url": "https://www.businessinsider.com/sam-altman-hires-openclaw-creator-peter-steinberger-personal-ai-agents-2026-2",
|
||||
"displayDate": "Feb 15, 2026",
|
||||
"author": "Business Insider",
|
||||
"summary": "A positive business-profile angle on OpenClaw’s breakout moment and why Steinberger chose leverage and reach over starting another company.",
|
||||
"featured": false
|
||||
},
|
||||
{
|
||||
"outlet": "Business Insider",
|
||||
"title": "'Eccentric but brilliant': OpenClaw's creator got feedback from Mark Zuckerberg",
|
||||
"url": "https://www.businessinsider.com/openclaw-creator-peter-steinberger-gets-feedback-from-mark-zuckerberg",
|
||||
"displayDate": "Feb 19, 2026",
|
||||
"author": "Business Insider",
|
||||
"summary": "Good social proof for the site: OpenClaw getting attention and detailed product feedback from the very top of the industry.",
|
||||
"featured": false
|
||||
},
|
||||
{
|
||||
"outlet": "TechCrunch",
|
||||
"title": "OpenClaw creator’s advice to AI builders is to be more playful and allow yourself time to improve",
|
||||
"url": "https://techcrunch.com/2026/02/25/openclaw-creators-advice-to-ai-builders-is-to-be-more-playful-and-allow-yourself-time-to-improve/",
|
||||
"displayDate": "Feb 25, 2026",
|
||||
"author": "Sarah Perez",
|
||||
"summary": "A builder-focused piece with the right tone for the homepage: experimentation, agency, and why OpenClaw resonated with makers so quickly.",
|
||||
"featured": false
|
||||
}
|
||||
]
|
||||
@ -6,7 +6,7 @@ import { getPublishedBlogPosts } from '../../lib/blog';
|
||||
export async function getStaticPaths() {
|
||||
const posts = await getPublishedBlogPosts();
|
||||
return posts.map((post) => ({
|
||||
params: { slug: post.id },
|
||||
params: { slug: post.slug },
|
||||
props: { post },
|
||||
}));
|
||||
}
|
||||
@ -38,7 +38,7 @@ function estimateReadTime(content: string): string {
|
||||
}
|
||||
|
||||
const readTime = estimateReadTime(post.body || '');
|
||||
const postUrl = `https://openclaw.ai/blog/${post.id}`;
|
||||
const postUrl = `https://openclaw.ai/blog/${post.slug}`;
|
||||
---
|
||||
|
||||
<Layout title={`${post.data.title} — OpenClaw Blog`} description={post.data.description}>
|
||||
@ -117,8 +117,6 @@ const postUrl = `https://openclaw.ai/blog/${post.id}`;
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/showcase">Showcase</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/press">Press</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/shoutouts">Shoutouts</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="https://trust.openclaw.ai/">Trust</a>
|
||||
|
||||
@ -36,7 +36,7 @@ function estimateReadTime(content: string): string {
|
||||
|
||||
<div class="posts-grid">
|
||||
{posts.map((post) => (
|
||||
<a href={`/blog/${post.id}`} class="post-card">
|
||||
<a href={`/blog/${post.slug}`} class="post-card">
|
||||
<div class="post-meta">
|
||||
<span class="post-date">{formatDate(post.data.date)}</span>
|
||||
<span class="post-separator">·</span>
|
||||
@ -70,8 +70,6 @@ function estimateReadTime(content: string): string {
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/showcase">Showcase</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/press">Press</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/shoutouts">Shoutouts</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/integrations">Integrations</a>
|
||||
|
||||
@ -3,8 +3,6 @@ import Layout from '../layouts/Layout.astro';
|
||||
import Icon from '../components/Icon.astro';
|
||||
import SectionHeader from '../components/SectionHeader.astro';
|
||||
import testimonials from '../data/testimonials.json';
|
||||
import communityBuilds from '../data/community-builds.json';
|
||||
import pressArticles from '../data/press.json';
|
||||
import { getPublishedBlogPosts } from '../lib/blog';
|
||||
|
||||
// Get latest blog post
|
||||
@ -51,8 +49,6 @@ const pixelsPerItem = 336;
|
||||
const pixelsPerSecond = 50;
|
||||
const duration1 = (row1.length / 2 * pixelsPerItem) / pixelsPerSecond;
|
||||
const duration2 = (row2.length / 2 * pixelsPerItem) / pixelsPerSecond;
|
||||
const featuredBuilds = communityBuilds.slice(0, 4);
|
||||
const featuredPress = pressArticles.slice(0, 4);
|
||||
---
|
||||
|
||||
<Layout title="OpenClaw — Personal AI Assistant">
|
||||
@ -102,7 +98,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
<!-- Latest Blog Post -->
|
||||
{latestPost && (
|
||||
<section class="latest-post">
|
||||
<a href={`/blog/${latestPost.id}`} class="latest-post-card">
|
||||
<a href={`/blog/${latestPost.slug}`} 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 +228,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
</div>
|
||||
</div>
|
||||
<div id="hackable-pnpm-content" style="display: none;">
|
||||
<div class="code-line comment"># Source checkouts are pnpm workspaces</div>
|
||||
<div class="code-line comment"># You clearly know what you're doing</div>
|
||||
<div class="code-line cmd">
|
||||
<span class="code-prompt">$</span>
|
||||
<span>git clone https://github.com/openclaw/openclaw.git</span>
|
||||
@ -243,16 +239,16 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
</div>
|
||||
<div class="code-line cmd">
|
||||
<span class="code-prompt">$</span>
|
||||
<span>cd openclaw && corepack enable && pnpm install</span>
|
||||
<span>cd openclaw && pnpm install && pnpm run build</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"># Run from source; bundled plugins use extensions/* deps</div>
|
||||
<div class="code-line comment"># You built it, now meet it</div>
|
||||
<div class="code-line cmd">
|
||||
<span class="code-prompt">$</span>
|
||||
<span>pnpm openclaw onboard</span>
|
||||
<span>pnpm run 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 +275,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. Switch later with <code>openclaw update --channel dev</code> or <code>openclaw update --channel stable</code>.
|
||||
Works on macOS, Linux, and Windows. The one-liner installs Node.js and everything else for you.
|
||||
</p>
|
||||
</section>
|
||||
|
||||
@ -287,7 +283,6 @@ 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';
|
||||
@ -361,9 +356,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
}
|
||||
osCmdElements.forEach(cmd => cmd.textContent = onelinerCmd);
|
||||
// Update hackable OS command for installer mode
|
||||
const hackableOsCmd = currentOs === 'windows'
|
||||
? windowsGitCmd
|
||||
: "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git";
|
||||
const hackableOsCmd = "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') {
|
||||
@ -389,8 +382,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 and installer-backed hackable mode.
|
||||
const showOsControls = currentMode === 'oneliner' || (currentMode === 'hackable' && currentHackable === 'installer');
|
||||
// Show OS switch for one-liner, PM switch for quick, hackable switch for hackable, nothing for macos
|
||||
const showOsControls = currentMode === 'oneliner';
|
||||
const showPmControls = currentMode === 'quick';
|
||||
const showHackableControls = currentMode === 'hackable';
|
||||
// Beta only applies to oneliner and npm modes (git always gets main branch)
|
||||
@ -427,7 +420,6 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
hackableBtns.forEach(b => b.classList.remove('active'));
|
||||
btn.classList.add('active');
|
||||
updateCommands();
|
||||
updateVisibility();
|
||||
});
|
||||
});
|
||||
|
||||
@ -470,12 +462,10 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
return currentPm === 'npm' ? `npm i -g openclaw${betaSuffix}` : `pnpm add -g openclaw${betaSuffix}`;
|
||||
},
|
||||
'onboard': () => 'openclaw onboard',
|
||||
'hackable-installer': () => currentOs === 'windows'
|
||||
? windowsGitCmd
|
||||
: "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git",
|
||||
'hackable-installer': () => "curl -fsSL https://openclaw.ai/install.sh | bash -s -- --install-method git",
|
||||
'clone': () => 'git clone https://github.com/openclaw/openclaw.git',
|
||||
'build': () => 'cd openclaw && corepack enable && pnpm install',
|
||||
'hackable-onboard': () => 'pnpm openclaw onboard',
|
||||
'build': () => 'cd openclaw && pnpm install && pnpm run build',
|
||||
'hackable-onboard': () => 'pnpm run openclaw onboard',
|
||||
};
|
||||
|
||||
document.querySelectorAll('.copy-line-btn').forEach(btn => {
|
||||
@ -623,56 +613,34 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- Community Builds -->
|
||||
<section class="builds-section">
|
||||
<SectionHeader
|
||||
title="What People Are Building"
|
||||
link={{ href: "https://docs.openclaw.ai/start/showcase", text: "View docs showcase" }}
|
||||
/>
|
||||
<div class="builds-grid">
|
||||
{featuredBuilds.map((build) => (
|
||||
<a href={build.href} target="_blank" rel="noopener" class="build-card">
|
||||
<div class="build-card-header">
|
||||
<span class="build-source">{build.source}</span>
|
||||
<span class="build-open">Open →</span>
|
||||
</div>
|
||||
<h3 class="build-title">{build.title}</h3>
|
||||
<p class="build-summary">{build.summary}</p>
|
||||
<div class="build-tags">
|
||||
{build.tags.map((tag: string) => (
|
||||
<span class="build-tag">{tag}</span>
|
||||
))}
|
||||
</div>
|
||||
</a>
|
||||
))}
|
||||
</div>
|
||||
<div class="build-links">
|
||||
<a href="/showcase" class="build-link">Community chatter →</a>
|
||||
<span class="link-separator">·</span>
|
||||
<a href="https://docs.openclaw.ai/start/showcase" class="build-link" target="_blank" rel="noopener">More project examples →</a>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- Press Section -->
|
||||
<section class="press-section">
|
||||
<SectionHeader title="Featured In" link={{ href: "/press", text: "View all" }} />
|
||||
<SectionHeader title="Featured In" />
|
||||
<div class="press-grid">
|
||||
{featuredPress.map((article) => (
|
||||
<a
|
||||
href={article.url}
|
||||
target="_blank"
|
||||
rel="noopener"
|
||||
class:list={['press-card', article.featured && 'press-featured']}
|
||||
>
|
||||
<div class="press-meta">
|
||||
<span class="press-name">{article.outlet}</span>
|
||||
<span class="press-date">{article.displayDate}</span>
|
||||
</div>
|
||||
<h3 class="press-title">{article.title}</h3>
|
||||
<p class="press-summary">{article.summary}</p>
|
||||
<span class="press-author">{article.author}</span>
|
||||
</a>
|
||||
))}
|
||||
<a href="https://www.macstories.net/stories/clawdbot-showed-me-what-the-future-of-personal-ai-assistants-looks-like/" target="_blank" rel="noopener" class="press-card press-featured">
|
||||
<div class="press-logo">
|
||||
<svg viewBox="0 0 24 24" fill="currentColor" class="press-icon">
|
||||
<path d="M18.71 19.5c-.83 1.24-1.71 2.45-3.05 2.47-1.34.03-1.77-.79-3.29-.79-1.53 0-2 .77-3.27.82-1.31.05-2.3-1.32-3.14-2.53C4.25 17 2.94 12.45 4.7 9.39c.87-1.52 2.43-2.48 4.12-2.51 1.28-.02 2.5.87 3.29.87.78 0 2.26-1.07 3.81-.91.65.03 2.47.26 3.64 1.98-.09.06-2.17 1.28-2.15 3.81.03 3.02 2.65 4.03 2.68 4.04-.03.07-.42 1.44-1.38 2.83M13 3.5c.73-.83 1.94-1.46 2.94-1.5.13 1.17-.34 2.35-1.04 3.19-.69.85-1.83 1.51-2.95 1.42-.15-1.15.41-2.35 1.05-3.11z"/>
|
||||
</svg>
|
||||
<span class="press-name">MacStories</span>
|
||||
</div>
|
||||
<blockquote class="press-quote">
|
||||
"OpenClaw Showed Me What the Future of Personal AI Assistants Looks Like"
|
||||
</blockquote>
|
||||
<span class="press-author">Federico Viticci</span>
|
||||
</a>
|
||||
<a href="https://www.starryhope.com/minipcs/clawdbot-mac-mini-ai-agent-trend/" target="_blank" rel="noopener" class="press-card">
|
||||
<div class="press-logo">
|
||||
<svg viewBox="0 0 24 24" fill="currentColor" class="press-icon">
|
||||
<polygon points="12 2 15.09 8.26 22 9.27 17 14.14 18.18 21.02 12 17.77 5.82 21.02 7 14.14 2 9.27 8.91 8.26 12 2"/>
|
||||
</svg>
|
||||
<span class="press-name">StarryHope</span>
|
||||
</div>
|
||||
<blockquote class="press-quote">
|
||||
"The Lobster Takeover: Why Developers Are Buying Mac Minis to Run Their Own AI Agents"
|
||||
</blockquote>
|
||||
<span class="press-author">Jim Mendenhall</span>
|
||||
</a>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
@ -779,8 +747,6 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/showcase">Showcase</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/press">Press</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/shoutouts">Shoutouts</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/integrations">Integrations</a>
|
||||
@ -788,7 +754,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
<a href="https://trust.openclaw.ai/">Trust</a>
|
||||
</nav>
|
||||
<p>Built by <a href="https://molty.me" target="_blank" rel="noopener">Molty</a> 🦞, a space lobster AI with a <a href="https://soul.md" target="_blank" rel="noopener">soul</a>, by <a href="https://steipete.me" target="_blank" rel="noopener">Peter Steinberger</a> & <a href="https://github.com/openclaw/openclaw#community" target="_blank" rel="noopener">community</a>.</p>
|
||||
<p class="disclaimer">Independent project, not affiliated with Anthropic.</p>
|
||||
<p class="disclaimer">Formerly known as Clawdbot and Moltbot. Independent project, not affiliated with Anthropic.</p>
|
||||
</footer>
|
||||
</main>
|
||||
</Layout>
|
||||
@ -1308,119 +1274,13 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
/* Community Builds */
|
||||
.builds-section {
|
||||
margin-bottom: 56px;
|
||||
animation: fadeInUp 0.8s ease-out 0.62s both;
|
||||
}
|
||||
|
||||
.builds-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(2, 1fr);
|
||||
gap: 20px;
|
||||
margin-bottom: 18px;
|
||||
}
|
||||
|
||||
@media (max-width: 640px) {
|
||||
.builds-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
|
||||
.build-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 14px;
|
||||
padding: 22px;
|
||||
border-radius: 18px;
|
||||
border: 1px solid var(--border-subtle);
|
||||
background: var(--surface-card-strong);
|
||||
backdrop-filter: blur(12px);
|
||||
text-decoration: none;
|
||||
color: var(--text-primary);
|
||||
transition: all 0.25s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
.build-card:hover {
|
||||
transform: translateY(-4px);
|
||||
border-color: var(--cyan-bright);
|
||||
box-shadow: 0 12px 40px var(--shadow-cyan-soft);
|
||||
}
|
||||
|
||||
.build-card-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 12px;
|
||||
}
|
||||
|
||||
.build-source,
|
||||
.build-tag {
|
||||
border: 1px solid var(--border-subtle);
|
||||
background: var(--surface-overlay);
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.build-source {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 28px;
|
||||
padding: 0 10px;
|
||||
border-radius: 999px;
|
||||
font-size: 0.75rem;
|
||||
font-weight: 600;
|
||||
letter-spacing: 0.04em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.build-open,
|
||||
.build-link {
|
||||
color: var(--coral-bright);
|
||||
font-size: 0.9rem;
|
||||
font-weight: 500;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.build-title {
|
||||
font-family: var(--font-display);
|
||||
font-size: 1.15rem;
|
||||
line-height: 1.3;
|
||||
}
|
||||
|
||||
.build-summary {
|
||||
color: var(--text-secondary);
|
||||
font-size: 0.96rem;
|
||||
line-height: 1.65;
|
||||
}
|
||||
|
||||
.build-tags {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.build-tag {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 28px;
|
||||
padding: 0 10px;
|
||||
border-radius: 999px;
|
||||
font-size: 0.78rem;
|
||||
}
|
||||
|
||||
.build-links {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
/* Press Section */
|
||||
.press-section {
|
||||
margin-bottom: 56px;
|
||||
animation: fadeInUp 0.8s ease-out 0.68s both;
|
||||
animation: fadeInUp 0.8s ease-out 0.65s both;
|
||||
}
|
||||
|
||||
|
||||
.press-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(2, 1fr);
|
||||
@ -1436,8 +1296,9 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
.press-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 14px;
|
||||
padding: 24px;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
padding: 24px 28px;
|
||||
border-radius: 16px;
|
||||
border: 1px solid var(--border-subtle);
|
||||
background: var(--surface-card-strong);
|
||||
@ -1445,6 +1306,7 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
text-decoration: none;
|
||||
color: var(--text-primary);
|
||||
transition: all 0.25s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.press-card:hover {
|
||||
@ -1453,44 +1315,39 @@ const featuredPress = pressArticles.slice(0, 4);
|
||||
box-shadow: 0 12px 40px var(--shadow-coral-soft);
|
||||
}
|
||||
|
||||
.press-meta {
|
||||
.press-logo {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 12px;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.press-icon {
|
||||
width: 28px;
|
||||
height: 28px;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.press-name {
|
||||
font-size: 0.82rem;
|
||||
font-weight: 600;
|
||||
color: var(--coral-bright);
|
||||
letter-spacing: 0.08em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.press-date {
|
||||
color: var(--text-muted);
|
||||
font-size: 0.82rem;
|
||||
}
|
||||
|
||||
.press-title {
|
||||
font-family: var(--font-display);
|
||||
font-size: 1.15rem;
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
color: var(--text-secondary);
|
||||
letter-spacing: 0.02em;
|
||||
}
|
||||
|
||||
.press-quote {
|
||||
font-family: var(--font-display);
|
||||
font-size: 1rem;
|
||||
font-weight: 500;
|
||||
line-height: 1.35;
|
||||
line-height: 1.5;
|
||||
color: var(--text-primary);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.press-summary {
|
||||
color: var(--text-secondary);
|
||||
line-height: 1.65;
|
||||
font-size: 0.95rem;
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
.press-author {
|
||||
font-size: 0.88rem;
|
||||
color: var(--text-muted);
|
||||
font-size: 0.9rem;
|
||||
color: var(--coral-bright);
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
|
||||
@ -9,7 +9,7 @@ import {
|
||||
siPhilipshue, siHomeassistant,
|
||||
siGooglechrome, siGmail, si1password,
|
||||
siX, siVercel,
|
||||
siLinux, siAndroid, siMacos, siIos, siQq
|
||||
siLinux, siAndroid, siMacos, siIos
|
||||
} from 'simple-icons';
|
||||
|
||||
// Helper to create SVG from simple-icons
|
||||
@ -84,7 +84,6 @@ 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 = [
|
||||
@ -115,11 +114,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/notion' },
|
||||
{ name: 'Notion', icon: siIcon(siNotion), color: '#FFFFFF', desc: 'Workspace & databases', docs: 'https://clawhub.ai/skills' },
|
||||
{ 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/bear-notes' },
|
||||
{ name: 'Bear Notes', icon: 'lucide:pen-tool', color: '#DD4C4F', desc: 'Markdown notes', docs: 'https://clawhub.ai/skills' },
|
||||
{ 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/github' },
|
||||
{ name: 'GitHub', icon: siIcon(siGithub), color: '#FFFFFF', desc: 'Code, issues, PRs', docs: 'https://clawhub.ai/skills' },
|
||||
];
|
||||
|
||||
const musicAudio = [
|
||||
@ -146,10 +145,10 @@ const tools = [
|
||||
];
|
||||
|
||||
const mediaCreative = [
|
||||
{ name: 'Image Gen', icon: 'lucide:image', color: '#E91E63', desc: 'AI image generation', docs: 'https://clawhub.ai/skills/image-gen' },
|
||||
{ name: 'Image Gen', icon: 'lucide:image', color: '#E91E63', desc: 'AI image generation', docs: 'https://clawhub.ai/skills' },
|
||||
{ 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/camera' },
|
||||
{ name: 'Camera', icon: 'lucide:camera', color: '#607D8B', desc: 'Photo/video capture', docs: 'https://clawhub.ai/skills' },
|
||||
];
|
||||
|
||||
const socialComms = [
|
||||
@ -379,8 +378,6 @@ const showcase = [
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/showcase">Showcase</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/press">Press</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/shoutouts">Shoutouts</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="https://trust.openclaw.ai/">Trust</a>
|
||||
|
||||
@ -1,278 +0,0 @@
|
||||
---
|
||||
import Layout from '../layouts/Layout.astro';
|
||||
import pressArticles from '../data/press.json';
|
||||
---
|
||||
|
||||
<Layout
|
||||
title="Press — OpenClaw"
|
||||
description="Coverage of OpenClaw across technology, business, and culture."
|
||||
canonicalUrl="https://openclaw.ai/press"
|
||||
>
|
||||
<div class="stars"></div>
|
||||
<div class="nebula"></div>
|
||||
|
||||
<main class="container">
|
||||
<header class="header">
|
||||
<a href="/" class="back-link">← Back to home</a>
|
||||
<h1 class="title">
|
||||
<span class="claw-accent">⟩</span> Press
|
||||
</h1>
|
||||
<p class="subtitle">
|
||||
Coverage of OpenClaw across product, business, culture, and the broader personal-agent wave.
|
||||
</p>
|
||||
<p class="note">Coverage selected for signal and relevance across product, business, and community momentum.</p>
|
||||
</header>
|
||||
|
||||
<div class="press-grid">
|
||||
{pressArticles.map((article) => (
|
||||
<a
|
||||
href={article.url}
|
||||
target="_blank"
|
||||
rel="noopener"
|
||||
class:list={['press-card', article.featured && 'press-featured']}
|
||||
>
|
||||
<div class="press-meta">
|
||||
<span class="press-outlet">{article.outlet}</span>
|
||||
<span class="press-date">{article.displayDate}</span>
|
||||
</div>
|
||||
<h2 class="press-title">{article.title}</h2>
|
||||
<p class="press-summary">{article.summary}</p>
|
||||
<span class="press-author">{article.author}</span>
|
||||
</a>
|
||||
))}
|
||||
</div>
|
||||
|
||||
<footer class="footer">
|
||||
<nav class="footer-nav">
|
||||
<a href="/">Home</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/blog">Blog</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/showcase">Showcase</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/shoutouts">Shoutouts</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/integrations">Integrations</a>
|
||||
</nav>
|
||||
<p>Built by <a href="https://molty.me" target="_blank" rel="noopener">Molty</a> 🦞</p>
|
||||
</footer>
|
||||
</main>
|
||||
</Layout>
|
||||
|
||||
<style>
|
||||
.stars {
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
background-image:
|
||||
radial-gradient(2px 2px at 20px 30px, rgba(255,255,255,0.8), transparent),
|
||||
radial-gradient(2px 2px at 40px 70px, rgba(255,255,255,0.5), transparent),
|
||||
radial-gradient(1px 1px at 90px 40px, rgba(255,255,255,0.6), transparent),
|
||||
radial-gradient(2px 2px at 130px 80px, rgba(255,255,255,0.4), transparent),
|
||||
radial-gradient(1px 1px at 160px 120px, rgba(255,255,255,0.7), transparent),
|
||||
radial-gradient(2px 2px at 200px 60px, rgba(0,229,204,0.6), transparent),
|
||||
radial-gradient(1px 1px at 250px 150px, rgba(255,255,255,0.5), transparent),
|
||||
radial-gradient(2px 2px at 300px 40px, rgba(255,77,77,0.4), transparent);
|
||||
background-size: 350px 200px;
|
||||
animation: twinkle 8s ease-in-out infinite alternate;
|
||||
pointer-events: none;
|
||||
z-index: 0;
|
||||
}
|
||||
|
||||
@keyframes twinkle {
|
||||
0% { opacity: 0.4; }
|
||||
100% { opacity: 0.7; }
|
||||
}
|
||||
|
||||
.nebula {
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
background:
|
||||
radial-gradient(ellipse 80% 50% at 20% 20%, rgba(255, 77, 77, 0.12), transparent 50%),
|
||||
radial-gradient(ellipse 60% 60% at 80% 30%, rgba(0, 229, 204, 0.08), transparent 50%),
|
||||
radial-gradient(ellipse 90% 70% at 50% 90%, rgba(255, 77, 77, 0.06), transparent 50%);
|
||||
pointer-events: none;
|
||||
z-index: 0;
|
||||
}
|
||||
|
||||
.container {
|
||||
position: relative;
|
||||
z-index: 1;
|
||||
max-width: 1100px;
|
||||
margin: 0 auto;
|
||||
padding: 40px 24px;
|
||||
min-height: 100vh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.header {
|
||||
text-align: center;
|
||||
margin-bottom: 48px;
|
||||
animation: fadeInUp 0.6s ease-out;
|
||||
}
|
||||
|
||||
@keyframes fadeInUp {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(20px);
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
}
|
||||
|
||||
.back-link {
|
||||
display: inline-block;
|
||||
margin-bottom: 24px;
|
||||
color: var(--text-muted);
|
||||
text-decoration: none;
|
||||
font-size: 0.9rem;
|
||||
transition: color 0.2s ease;
|
||||
}
|
||||
|
||||
.back-link:hover {
|
||||
color: var(--coral-bright);
|
||||
}
|
||||
|
||||
.title {
|
||||
font-family: var(--font-display);
|
||||
font-size: clamp(2rem, 6vw, 3rem);
|
||||
font-weight: 700;
|
||||
margin-bottom: 12px;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.claw-accent {
|
||||
color: var(--coral-bright);
|
||||
}
|
||||
|
||||
.subtitle,
|
||||
.note {
|
||||
max-width: 700px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.subtitle {
|
||||
color: var(--text-secondary);
|
||||
font-size: 1.08rem;
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.note {
|
||||
color: var(--text-muted);
|
||||
font-size: 0.95rem;
|
||||
}
|
||||
|
||||
.press-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.press-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 14px;
|
||||
padding: 24px;
|
||||
border-radius: 18px;
|
||||
border: 1px solid var(--border-subtle);
|
||||
background: var(--surface-card-strong);
|
||||
backdrop-filter: blur(12px);
|
||||
text-decoration: none;
|
||||
color: var(--text-primary);
|
||||
transition: all 0.25s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
.press-card:hover {
|
||||
transform: translateY(-4px);
|
||||
border-color: var(--coral-bright);
|
||||
box-shadow: 0 12px 40px var(--shadow-coral-soft);
|
||||
}
|
||||
|
||||
.press-featured {
|
||||
border-color: var(--border-accent);
|
||||
background: var(--press-featured-gradient);
|
||||
}
|
||||
|
||||
.press-meta {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
}
|
||||
|
||||
.press-outlet {
|
||||
color: var(--coral-bright);
|
||||
font-size: 0.82rem;
|
||||
font-weight: 700;
|
||||
letter-spacing: 0.08em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.press-date {
|
||||
color: var(--text-muted);
|
||||
font-size: 0.82rem;
|
||||
}
|
||||
|
||||
.press-title {
|
||||
font-family: var(--font-display);
|
||||
font-size: 1.2rem;
|
||||
line-height: 1.35;
|
||||
}
|
||||
|
||||
.press-summary {
|
||||
color: var(--text-secondary);
|
||||
line-height: 1.65;
|
||||
font-size: 0.96rem;
|
||||
}
|
||||
|
||||
.press-author {
|
||||
color: var(--text-muted);
|
||||
font-size: 0.88rem;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.footer {
|
||||
margin-top: auto;
|
||||
padding-top: 48px;
|
||||
text-align: center;
|
||||
font-size: 0.9rem;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.footer-nav {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
column-gap: 8px;
|
||||
row-gap: 8px;
|
||||
margin-bottom: 16px;
|
||||
font-size: 0.95rem;
|
||||
}
|
||||
|
||||
.footer-separator {
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.footer a {
|
||||
color: var(--coral-bright);
|
||||
text-decoration: none;
|
||||
transition: color 0.2s ease;
|
||||
}
|
||||
|
||||
.footer a:hover {
|
||||
color: var(--cyan-bright);
|
||||
}
|
||||
|
||||
@media (max-width: 720px) {
|
||||
.container {
|
||||
padding: 24px 16px;
|
||||
}
|
||||
|
||||
.press-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@ -12,7 +12,7 @@ export async function GET(context) {
|
||||
title: post.data.title,
|
||||
description: post.data.description,
|
||||
pubDate: post.data.date,
|
||||
link: `/blog/${post.id}`,
|
||||
link: `/blog/${post.slug}`,
|
||||
})),
|
||||
});
|
||||
}
|
||||
|
||||
@ -46,8 +46,6 @@ const allTestimonials = [...testimonials, ...extraTestimonials];
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/showcase">Showcase</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/press">Press</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/integrations">Integrations</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="https://trust.openclaw.ai/">Trust</a>
|
||||
|
||||
@ -1,10 +1,9 @@
|
||||
---
|
||||
import Layout from '../layouts/Layout.astro';
|
||||
import showcaseData from '../data/showcase.json';
|
||||
import featuredBuilds from '../data/community-builds.json';
|
||||
|
||||
const featuredIds = new Set(featuredBuilds.map((item) => item.id));
|
||||
const sortedShowcase = showcaseData.filter((item) => !featuredIds.has(item.id));
|
||||
// Use JSON order directly.
|
||||
const sortedShowcase = showcaseData;
|
||||
|
||||
const categoryLabels: Record<string, string> = {
|
||||
'power-user': '🚀 Power User',
|
||||
@ -28,43 +27,9 @@ const categoryLabels: Record<string, string> = {
|
||||
<h1 class="title">
|
||||
<span class="claw-accent">⟩</span> What People Are Building
|
||||
</h1>
|
||||
<p class="subtitle">The strongest OpenClaw builds we could verify in public: X threads, shipped workflows, and real automation.</p>
|
||||
<p class="subtitle">Real projects, real automation, real magic.</p>
|
||||
</header>
|
||||
|
||||
<section class="featured-section">
|
||||
<div class="featured-header">
|
||||
<h2 class="featured-title">Featured Builds</h2>
|
||||
<p class="featured-copy">
|
||||
Handpicked examples that show range: personal ops, coding, family workflows, and “I cannot believe this is running from chat.”
|
||||
</p>
|
||||
</div>
|
||||
<div class="featured-grid">
|
||||
{featuredBuilds.map((build) => (
|
||||
<a href={build.href} target="_blank" rel="noopener" class="featured-card">
|
||||
<div class="featured-topline">
|
||||
<span class="featured-source">{build.source}</span>
|
||||
<span class="featured-likes">❤️ {build.likes}</span>
|
||||
</div>
|
||||
<h3 class="featured-card-title">{build.title}</h3>
|
||||
<p class="featured-summary">{build.summary}</p>
|
||||
<div class="featured-meta">
|
||||
<span class="featured-author">@{build.author}</span>
|
||||
<div class="featured-tags">
|
||||
{build.tags.map((tag: string) => (
|
||||
<span class="featured-tag">{tag}</span>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
</a>
|
||||
))}
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<div class="grid-header">
|
||||
<h2 class="grid-title">More From the Community</h2>
|
||||
<p class="grid-copy">A broader stream of OpenClaw projects, experiments, and production workflows.</p>
|
||||
</div>
|
||||
|
||||
<div class="showcase-grid">
|
||||
{sortedShowcase.map((item) => (
|
||||
<a href={`https://x.com/${item.author}/status/${item.id}`} target="_blank" rel="noopener" class="showcase-card">
|
||||
@ -114,8 +79,6 @@ const categoryLabels: Record<string, string> = {
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/blog">Blog</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/press">Press</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/shoutouts">Shoutouts</a>
|
||||
<span class="footer-separator">·</span>
|
||||
<a href="/integrations">Integrations</a>
|
||||
@ -212,122 +175,6 @@ const categoryLabels: Record<string, string> = {
|
||||
.subtitle {
|
||||
color: var(--text-secondary);
|
||||
font-size: 1.1rem;
|
||||
max-width: 780px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.featured-section {
|
||||
margin-bottom: 48px;
|
||||
}
|
||||
|
||||
.featured-header {
|
||||
display: grid;
|
||||
gap: 10px;
|
||||
margin-bottom: 22px;
|
||||
}
|
||||
|
||||
.featured-title,
|
||||
.grid-title {
|
||||
font-family: var(--font-display);
|
||||
font-size: 1.45rem;
|
||||
font-weight: 600;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.featured-copy,
|
||||
.grid-copy {
|
||||
color: var(--text-secondary);
|
||||
max-width: 760px;
|
||||
}
|
||||
|
||||
.featured-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||
gap: 18px;
|
||||
}
|
||||
|
||||
.featured-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 14px;
|
||||
padding: 22px;
|
||||
border-radius: 18px;
|
||||
border: 1px solid var(--border-subtle);
|
||||
background:
|
||||
linear-gradient(135deg, rgba(255, 77, 77, 0.06), transparent 40%),
|
||||
var(--surface-card-strong);
|
||||
backdrop-filter: blur(10px);
|
||||
text-decoration: none;
|
||||
color: var(--text-primary);
|
||||
transition: all 0.25s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
.featured-card:hover {
|
||||
transform: translateY(-4px);
|
||||
border-color: var(--coral-bright);
|
||||
box-shadow: 0 16px 48px var(--shadow-coral-soft);
|
||||
}
|
||||
|
||||
.featured-topline,
|
||||
.featured-meta {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 12px;
|
||||
}
|
||||
|
||||
.featured-source,
|
||||
.featured-tag {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 28px;
|
||||
padding: 0 10px;
|
||||
border-radius: 999px;
|
||||
border: 1px solid var(--border-subtle);
|
||||
background: var(--surface-overlay);
|
||||
color: var(--text-muted);
|
||||
font-size: 0.76rem;
|
||||
}
|
||||
|
||||
.featured-source {
|
||||
font-weight: 600;
|
||||
letter-spacing: 0.04em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.featured-likes {
|
||||
font-size: 0.84rem;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.featured-card-title {
|
||||
font-family: var(--font-display);
|
||||
font-size: 1.2rem;
|
||||
line-height: 1.3;
|
||||
}
|
||||
|
||||
.featured-summary {
|
||||
color: var(--text-secondary);
|
||||
line-height: 1.65;
|
||||
}
|
||||
|
||||
.featured-author {
|
||||
color: var(--coral-bright);
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
}
|
||||
|
||||
.featured-tags {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
justify-content: flex-end;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.grid-header {
|
||||
display: grid;
|
||||
gap: 8px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.showcase-grid {
|
||||
@ -562,19 +409,6 @@ const categoryLabels: Record<string, string> = {
|
||||
padding: 24px 16px;
|
||||
}
|
||||
|
||||
.featured-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.featured-meta {
|
||||
align-items: flex-start;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.featured-tags {
|
||||
justify-content: flex-start;
|
||||
}
|
||||
|
||||
.showcase-grid {
|
||||
column-width: auto;
|
||||
column-count: 1;
|
||||
|
||||
Loading…
Reference in New Issue
Block a user