feat(cypher-executor): magic vars {{_today}} / {{_iso_week}} / {{_now}} 等

對應 LI SDD M2.x improvement,源自 Claude (我) 透過 arcrun_report_feedback
寫的 feedback block c47bf70b 提的「需要內建變數展開」需求。完整閉環:

  AI 用 MCP tool 報 bug → KBDB 收 feedback → AI 自己看 → 自己修補 → deploy

新增 cypher-executor/src/lib/magic-vars.ts:
  - _today / _yesterday / _now / _now_unix / _now_unix_s
  - _iso_week (2026-W20) / _iso_week_num / _iso_year
  - _yyyymm / _yyyymmdd
  - _year / _month / _day / _hour / _minute / _second(zero-padded)
  - _weekday (0-6, 0=日) / _iso_weekday (1-7, 1=一)
  全部 UTC,避免 worker 跨時區誤判

GraphExecutor.execute() 入口注入:
  ctxWithMagic = { ...initialContext, ...buildMagicVars() }
  順序確保 magic vars 永遠 win(防 user 不小心用 _ prefix)

不違反 §2.2(cypher-executor TS 禁實作 secret/JWT 業務邏輯):
  magic vars 是公開時間常數,跟既有 interpolateString 的 ctx 變數展開
  同層,純 orchestrator routing 職責。

AGENTS.md §10.5 加 magic vars 完整表 + weekly archive 範例。

實測(commit 後 deploy + 觸發 weekly_review):
  - KBDB 新建 roadmap-2026-W20 block 正確展開 page_name
  - roadmap-latest 同步更新(雙寫 pattern)
  - 證明 weekly archive 從此真能累積歷史,不再固定 latest 覆蓋

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
2026-05-16 21:20:52 +08:00
parent ef406d714a
commit 8e985684f9
3 changed files with 131 additions and 3 deletions
+12 -3
View File
@@ -5,6 +5,7 @@ import { injectCredentials } from './actions/credential-injector';
import { tryAuthDispatch } from './actions/auth-dispatcher';
import { expandPromptRecipe } from './lib/recipe-expander';
import { persistPausedRun, isResumablePending, parseRecipeOutput } from './lib/paused-runs';
import { buildMagicVars } from './lib/magic-vars';
export type ComponentLoader = (componentId: string) => Promise<ComponentRunner>;
export type WorkflowLoader = (workflowId: string) => Promise<ExecutionGraph>;
@@ -51,12 +52,20 @@ export class GraphExecutor {
this.currentGraph = graph;
this.currentRunId = kvStore?.runId ?? `${graph.id}-${Date.now()}`;
// Magic vars:注入 _today / _now / _iso_week 等系統變數(LI SDD M2.x
// initialContext 寫前,magic vars 寫後 → magic vars 永遠 win(防 user accidentally 用 _ prefix
// 同時保留 user 既有 ctxmagic vars 不破壞既有 workflow_ prefix reserved
const ctxWithMagic: Record<string, unknown> = {
...initialContext,
...buildMagicVars(),
};
// 找出所有起點(沒有任何邊指向的節點)
const hasIncoming = new Set(graph.edges.map(e => e.to));
const startNodes = graph.nodes.filter(n => !hasIncoming.has(n.id));
if (startNodes.length === 0) {
return { data: initialContext, trace };
return { data: ctxWithMagic, trace };
}
// 建立 fan-in 狀態(入度 > 1 的節點需要等所有上游)
@@ -64,14 +73,14 @@ export class GraphExecutor {
for (const node of graph.nodes) {
const inDeg = graph.edges.filter(e => e.to === node.id).length;
if (inDeg > 1) {
fanIn.set(node.id, { ctx: { ...initialContext }, remaining: inDeg });
fanIn.set(node.id, { ctx: { ...ctxWithMagic }, remaining: inDeg });
}
}
// 並行執行所有起點
const results = await Promise.all(
startNodes.map(node =>
this.executeNode(node, graph, initialContext, new Set(), trace, fanIn, kvStore)
this.executeNode(node, graph, ctxWithMagic, new Set(), trace, fanIn, kvStore)
)
);
+89
View File
@@ -0,0 +1,89 @@
/**
* Magic vars — workflow YAML 內建變數
*
* 對應 LI SDD M2.x improvementfeedback block c47bf70b)。
*
* 任何以 `_` 開頭的變數名都是 reserved(system)。常見:時間、執行 metadata。
* 用於 page_name / file path / URL 等需要時間戳的場景。
*
* 範例 YAML
* page_name: "roadmap-week-{{_iso_week}}" # roadmap-week-2026-W20
* page_name: "log-{{_today}}" # log-2026-05-16
* filename: "snapshot-{{_now_unix}}.json" # snapshot-1778940000123.json
*
* 不違反 §2.2:這是 orchestrator routing 提供的「環境變數」(像 shell 的 $DATE),
* 不涉及 secret / credential / JWT,跟既有 ctx 變數展開同層。
*/
/**
* 算 ISO 8601 週數(W01-W53)。
* 週一為週首,W01 含當年首個週四(ISO 標準)。
* https://en.wikipedia.org/wiki/ISO_week_date
*/
function isoWeekNumber(d: Date): { year: number; week: number } {
const target = new Date(Date.UTC(d.getUTCFullYear(), d.getUTCMonth(), d.getUTCDate()));
const dayNum = (target.getUTCDay() + 6) % 7; // Mon=0
target.setUTCDate(target.getUTCDate() - dayNum + 3);
const firstThursday = new Date(Date.UTC(target.getUTCFullYear(), 0, 4));
const weekNum = 1 + Math.round(
((target.getTime() - firstThursday.getTime()) / 86400000 -
3 + ((firstThursday.getUTCDay() + 6) % 7)) / 7
);
return { year: target.getUTCFullYear(), week: weekNum };
}
function pad2(n: number): string {
return n.toString().padStart(2, '0');
}
/**
* 建立 magic vars。每次 workflow 觸發時呼叫一次,貫穿整個執行。
*
* 設計:UTC 為基準(避免 worker 跨時區誤判)。需要本地時區的場景,
* 用戶可自己組(例如 yaml 寫 `{{_today_utc}}` + 自己處理偏移)。
*/
export function buildMagicVars(now: Date = new Date()): Record<string, string | number> {
const iso = now.toISOString(); // 2026-05-16T09:30:00.123Z
const yyyy = now.getUTCFullYear();
const mm = pad2(now.getUTCMonth() + 1);
const dd = pad2(now.getUTCDate());
const hh = pad2(now.getUTCHours());
const mi = pad2(now.getUTCMinutes());
const ss = pad2(now.getUTCSeconds());
const yesterday = new Date(now.getTime() - 86400000);
const yMm = pad2(yesterday.getUTCMonth() + 1);
const yDd = pad2(yesterday.getUTCDate());
const { year: isoYear, week: isoWeek } = isoWeekNumber(now);
return {
// 日期 / 時間(UTC
_today: `${yyyy}-${mm}-${dd}`, // 2026-05-16
_yesterday: `${yesterday.getUTCFullYear()}-${yMm}-${yDd}`, // 2026-05-15
_now: iso, // ISO 8601
_now_unix: now.getTime(), // unix ms
_now_unix_s: Math.floor(now.getTime() / 1000), // unix sec
// 個別欄位(給 path / page_name 拼)
_year: yyyy,
_month: mm,
_day: dd,
_hour: hh,
_minute: mi,
_second: ss,
// ISO 週(roadmap weekly archive 必備)
_iso_week: `${isoYear}-W${pad2(isoWeek)}`, // 2026-W20
_iso_week_num: isoWeek,
_iso_year: isoYear,
// 簡單時間 slotcron-friendly
_yyyymm: `${yyyy}${mm}`, // 202605
_yyyymmdd: `${yyyy}${mm}${dd}`, // 20260516
// 週幾(0=週日,1=週一 ... 6=週六;ISO 風格在 _iso_weekday
_weekday: now.getUTCDay(),
_iso_weekday: ((now.getUTCDay() + 6) % 7) + 1, // 1=Mon...7=Sun
};
}