3e65e22775
壓測四橫向問題修正(docs 壓測報告):
① 薄殼原則成鐵律:能力長在 API,CLI/MCP/lib 只暴露
- seed 下沉成 API 行為:cypher-executor POST /init/seed(一次灌 API+auth recipe),
種子資料移到 server src/lib/api-recipe-seeds.ts,CLI 改薄殼一次呼叫
- 解除 deployFullyOk 連坐 + init 補 seed auth recipe + update 補 seed/全 KV
- registry SUBMISSIONS_KV 補進 REQUIRED_KV_NAMESPACES(修 20/21)
② MCP 統一帳號來源(單一 remote MCP + .env 切 MCP URL)
- MCP 從 sibling repo 搬進 arcrun/mcp/(remote Worker,route 改 mcp.arcrun.dev)
- config 加 mcp_url 三層解析 + getMcpUrl + DEFAULT_MCP_URL
- 新增 acr mcp-setup:依 config 寫專案 .mcp.json(接案切資料夾自動切 MCP)
- acr --version 改動態讀 package.json(根治漂移)
③ Deploy 一致性
- tests/release.feature + scripts/check-release.sh
- local-deploy.sh:CLI npm publish + auto patch bump + CHANGELOG
- local-deploy.sh bash 3.2 相容修正(mapfile / 空陣列 set -u)
- builtins/pnpm-lock.yaml
④ README self-hosted 同步現況(移除 R2 殘留、加 flag/env、多帳號)
CLI bump → 1.3.0
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
101 lines
4.3 KiB
TypeScript
101 lines
4.3 KiB
TypeScript
/**
|
||
* acr update — 拉新 GitHub release,重新部署零件/引擎到用戶自己的 Cloudflare。
|
||
*
|
||
* 與 acr init --self-hosted 走同一條「下載 release → 注入 KV id → wrangler deploy」的路,
|
||
* 差別只在:init 是首次(建 KV/R2 + 寫 config),update 是沿用既有 config 重部署變動的 Worker。
|
||
*
|
||
* 對應 SDD:.agents/specs/arcrun/sdk-and-website/self-hosted-init.md §3「acr update」
|
||
*
|
||
* 誠實限制(mindset §7 / SDD §6):部署依賴 GitHub release(含預編譯 wasm),
|
||
* release 產製管道補上前,誠實回報未實作,不假裝更新成功。
|
||
*/
|
||
|
||
import chalk from 'chalk';
|
||
import { loadConfig } from '../lib/config.js';
|
||
import { CfAccountClient } from '../lib/cf-api.js';
|
||
import {
|
||
wranglerAvailable,
|
||
downloadAndDeploy,
|
||
REQUIRED_KV_NAMESPACES,
|
||
type DeployContext,
|
||
} from '../lib/deploy.js';
|
||
|
||
export async function cmdUpdate(): Promise<void> {
|
||
const config = loadConfig();
|
||
|
||
if (config.mode !== 'self-hosted') {
|
||
console.log(chalk.yellow('\n acr update 只用於 self-hosted 模式(部署在你自己的 Cloudflare)。'));
|
||
console.log(chalk.gray(' 目前模式:' + config.mode + '。如要 self-host,先跑 acr init --self-hosted。\n'));
|
||
process.exit(1);
|
||
}
|
||
|
||
if (!config.cloudflare_account_id || !config.cf_api_token) {
|
||
console.log(chalk.yellow('\n config 缺 cloudflare_account_id / cf_api_token,無法部署。'));
|
||
console.log(chalk.gray(' 請重新跑 acr init --self-hosted。\n'));
|
||
process.exit(1);
|
||
}
|
||
|
||
if (!wranglerAvailable()) {
|
||
console.log(chalk.yellow('\n ✗ 找不到 wrangler(Cloudflare CLI)。請先 npm i -g wrangler。\n'));
|
||
process.exit(1);
|
||
}
|
||
|
||
console.log(chalk.bold('\n acr update — 拉新 release 並重新部署\n'));
|
||
|
||
// 重新解析「全部」KV namespace id(冪等:已存在則重用),不只 config 存的兩個。
|
||
// 壓測 §4.1.3:舊版 update 只注入 WEBHOOKS+CREDENTIALS_KV,其餘 6 個注入成空字串 →
|
||
// 重部署反而可能弄壞需要 RECIPES/EXEC_CONTEXT/... 的 worker。改為與 init 同樣全建妥。
|
||
const cf = new CfAccountClient(config.cloudflare_account_id, config.cf_api_token);
|
||
const kvNamespaceIds: Record<string, string> = {};
|
||
try {
|
||
const existing = await cf.listKvNamespaces();
|
||
for (const title of REQUIRED_KV_NAMESPACES) {
|
||
kvNamespaceIds[title] = await cf.ensureKvNamespace(title, existing);
|
||
}
|
||
} catch (e) {
|
||
console.log(chalk.yellow(`\n ✗ 解析 KV namespace 失敗:${e instanceof Error ? e.message : e}\n`));
|
||
process.exit(1);
|
||
}
|
||
|
||
const ctx: DeployContext = {
|
||
accountId: config.cloudflare_account_id,
|
||
apiToken: config.cf_api_token,
|
||
workerSubdomain: extractSubdomain(config.cypher_executor_url),
|
||
kvNamespaceIds,
|
||
};
|
||
|
||
const result = await downloadAndDeploy(ctx);
|
||
|
||
if (result.implemented) {
|
||
console.log(chalk.green('\n ✓ 部署完成'));
|
||
// 重跑 seed(薄殼:呼叫 API /init/seed;冪等,覆寫既有)。
|
||
// 修壓測 §4.1.3「update 不做 seed,但 init 提示說 update 會重試 seed」的矛盾。
|
||
const cypherUrl = config.cypher_executor_url
|
||
?? result.cypherExecutorUrl
|
||
?? (ctx.workerSubdomain ? `https://arcrun-cypher-executor.${ctx.workerSubdomain}.workers.dev` : '');
|
||
if (cypherUrl) {
|
||
process.stdout.write(chalk.gray(' → 重新 seed recipe(API + auth,由 API 灌入)...'));
|
||
try {
|
||
const res = await fetch(`${cypherUrl}/init/seed`, { method: 'POST' });
|
||
const body = await res.json().catch(() => null) as { success?: boolean; message?: string } | null;
|
||
console.log(res.ok && body?.success
|
||
? chalk.green(` ✓ ${body.message ?? ''}`)
|
||
: chalk.yellow(` ⚠ ${body?.message ?? `HTTP ${res.status}`}`));
|
||
} catch (e) {
|
||
console.log(chalk.yellow(` ⚠ seed 失敗(${e instanceof Error ? e.message : e})`));
|
||
}
|
||
}
|
||
console.log('');
|
||
} else {
|
||
console.log(chalk.yellow(' ⚠ 更新尚未自動化:'));
|
||
console.log(chalk.gray(' ' + result.message.split('\n').join('\n ')) + '\n');
|
||
}
|
||
}
|
||
|
||
/** 從 cypher_executor_url(https://arcrun-cypher-executor.<sub>.workers.dev)抽 subdomain。*/
|
||
function extractSubdomain(url?: string): string {
|
||
if (!url) return '';
|
||
const m = url.match(/arcrun-cypher-executor\.([^.]+)\.workers\.dev/);
|
||
return m?.[1] ?? '';
|
||
}
|