fix(cli): deploy 注入 MULTI_TENANT=false 到 self-hosted worker(修 MCP 401 注入缺口)
根因(非 code bug):partner-auth.ts MULTI_TENANT 分支邏輯對,但部署沒注入 → worker c.env.MULTI_TENANT===undefined → 走 partner-key → self-hosted 401。 mcp/wrangler.toml 的 MULTI_TENANT 原是註解掉的,injectWranglerConfig 注了 KV/WORKER_SUBDOMAIN/D1 卻漏 MULTI_TENANT。只取消註解不夠(只修手動 fork, 沒修 acr update 自動部署這條 mira 走的路)。 修法(方案①,注 vars 非 secret,self-hosted 零填寫): - deploy.ts:DeployContext 加 selfHosted;新增 injectMultiTenant(active/註解/無行三態 → 加進 [vars]);injectWranglerConfig 在 selfHosted 時呼叫。 - init.ts:deployCtx selfHosted:true(本就是 --self-hosted 分支)。 - update.ts:ctx selfHosted = mode==='self-hosted' || multi_tenant===false(mira 走這條)。 - mcp/wrangler.toml:# [vars] 改 active [vars](官方不含 MULTI_TENANT=多租戶; 注入加行在 [vars] 下,結構正確)。 本地驗注入(真實 export 函式 dry-run):mcp/cypher 注入後各 1 行 active MULTI_TENANT="false" 在 active [vars] 下 → PASS。cli tsc exit 0。 端到端交棒 mira:leo21c 重跑 acr update → curl Bearer leo /mcp 應 200。 SDD: mcp-account-source.md §5.5.1。 Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -98,6 +98,10 @@ export interface DeployContext {
|
||||
workerSubdomain: string;
|
||||
kvNamespaceIds: Record<string, string>; // title → id
|
||||
d1DatabaseId?: string; // KBDB Base D1 (arcrun-kbdb); injected into kbdb wrangler.toml
|
||||
// self-hosted 單租戶旗標。true(self-hosted)→ 注入 MULTI_TENANT="false" 到 worker [vars],
|
||||
// 讓 MCP partner-auth 走 namespace 明碼分支(mcp-account-source §5.5)。
|
||||
// 未設 / false → 不注入(官方 SaaS 多租戶,行為不變)。
|
||||
selfHosted?: boolean;
|
||||
}
|
||||
|
||||
export interface DeployResult {
|
||||
@@ -388,11 +392,46 @@ function injectWranglerConfig(tomlPath: string, ctx: DeployContext): void {
|
||||
);
|
||||
}
|
||||
|
||||
// self-hosted:注入 MULTI_TENANT="false" 到 [vars](mcp-account-source §5.5)。
|
||||
// 修「部署沒注入 → worker c.env.MULTI_TENANT===undefined → MCP 走 partner-key → 401」。
|
||||
// 只對有 [vars] 的 worker(mcp / cypher-executor)生效;其餘無 [vars] 的不動。
|
||||
if (ctx.selfHosted) {
|
||||
toml = injectMultiTenant(toml);
|
||||
}
|
||||
|
||||
toml = stripOfficialOnlyBindings(toml);
|
||||
|
||||
writeFileSync(tomlPath, toml, 'utf8');
|
||||
}
|
||||
|
||||
/**
|
||||
* self-hosted:確保 worker [vars] 有 `MULTI_TENANT = "false"`。處理三種既有狀態:
|
||||
* 1. 已有 active `MULTI_TENANT = "..."` → 改成 "false"
|
||||
* 2. 有註解的 `# MULTI_TENANT = "false"`(mcp/cypher toml 預設這樣)→ 取消註解
|
||||
* 3. 無此行但有 `[vars]` → 在 [vars] header 下一行加進去
|
||||
* 4. 無 `[vars]`(該 worker 不吃此 var)→ 不動
|
||||
* 純文字操作,與 WORKER_SUBDOMAIN/KV 注入同層級(mcp-account-source §5.5)。
|
||||
*/
|
||||
export function injectMultiTenant(toml: string): string {
|
||||
// 1. 已有 active 行 → 設 false
|
||||
if (/^\s*MULTI_TENANT\s*=/m.test(toml)) {
|
||||
return toml.replace(/^(\s*MULTI_TENANT\s*=\s*")[^"]*(".*)$/m, `$1false$2`);
|
||||
}
|
||||
// 2. 註解掉的行 → 取消註解(保留原縮排)
|
||||
if (/^\s*#\s*MULTI_TENANT\s*=/m.test(toml)) {
|
||||
return toml.replace(/^(\s*)#\s*(MULTI_TENANT\s*=\s*)"[^"]*"(.*)$/m, `$1$2"false"$3`);
|
||||
}
|
||||
// 3. 有 [vars] → 在其後插入
|
||||
if (/^\s*\[vars\]\s*$/m.test(toml)) {
|
||||
return toml.replace(
|
||||
/^(\s*\[vars\]\s*)$/m,
|
||||
`$1\nMULTI_TENANT = "false" # self-hosted 單租戶(acr update 注入,mcp-account-source §5.5)`,
|
||||
);
|
||||
}
|
||||
// 4. 無 [vars] → 不動(該 worker 不用此 var)
|
||||
return toml;
|
||||
}
|
||||
|
||||
/**
|
||||
* 移除 self-hosted fork 帳號沒有、會導致 wrangler deploy 失敗的官方專屬 TOML 區塊:
|
||||
* - `[[routes]]`(含 pattern/zone_name):fork 沒有 arcrun.dev zone
|
||||
|
||||
Reference in New Issue
Block a user