@logixjs/sandbox 0.0.1 → 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +201 -0
- package/package.json +3 -1
- package/public/sandbox/chunks/chunk-2AUQRESB.js +1114 -0
- package/public/sandbox/chunks/chunk-2F72LQKW.js +194 -0
- package/public/sandbox/chunks/chunk-2PYDGDD3.js +4071 -0
- package/public/sandbox/chunks/chunk-2UGKSXDZ.js +16 -0
- package/public/sandbox/chunks/chunk-2XVE7F26.js +153 -0
- package/public/sandbox/chunks/chunk-3A7TFNZN.js +219 -0
- package/public/sandbox/chunks/chunk-3BHC6JAV.js +1070 -0
- package/public/sandbox/chunks/chunk-3U4CAKCX.js +61 -0
- package/public/sandbox/chunks/chunk-3VU3JLZO.js +272 -0
- package/public/sandbox/chunks/chunk-4CPONKWY.js +1170 -0
- package/public/sandbox/chunks/chunk-4GMUQTWH.js +850 -0
- package/public/sandbox/chunks/chunk-4KIYG5E4.js +749 -0
- package/public/sandbox/chunks/chunk-4MY35XR3.js +175 -0
- package/public/sandbox/chunks/chunk-5DNHJMHA.js +69 -0
- package/public/sandbox/chunks/chunk-5EC4S7X6.js +42 -0
- package/public/sandbox/chunks/chunk-5FB7QFSH.js +227 -0
- package/public/sandbox/chunks/chunk-5GEYCJW2.js +206 -0
- package/public/sandbox/chunks/chunk-5PT7Q5SS.js +373 -0
- package/public/sandbox/chunks/chunk-5WN5IGKA.js +166 -0
- package/public/sandbox/chunks/chunk-5XOBKSOT.js +213 -0
- package/public/sandbox/chunks/chunk-67RVISUC.js +313 -0
- package/public/sandbox/chunks/chunk-6ELJEKZS.js +12 -0
- package/public/sandbox/chunks/chunk-6FKWRVP4.js +85 -0
- package/public/sandbox/chunks/chunk-6JRP52YT.js +34 -0
- package/public/sandbox/chunks/chunk-6JYAC3G6.js +42 -0
- package/public/sandbox/chunks/chunk-6TUOP63Q.js +86 -0
- package/public/sandbox/chunks/{chunk-A6JFUNQQ.js → chunk-74C5VW7Z.js} +2 -2
- package/public/sandbox/chunks/chunk-75DQFC5M.js +16 -0
- package/public/sandbox/chunks/{chunk-UCFEP3BH.js → chunk-7GPAF2UK.js} +1 -1
- package/public/sandbox/chunks/chunk-7KQFTS2Y.js +135 -0
- package/public/sandbox/chunks/chunk-7POKWYE3.js +112 -0
- package/public/sandbox/chunks/chunk-7VUEF5F3.js +395 -0
- package/public/sandbox/chunks/chunk-7W5APZS2.js +77 -0
- package/public/sandbox/chunks/chunk-A6OFCXDQ.js +397 -0
- package/public/sandbox/chunks/chunk-A7HLOSO6.js +216 -0
- package/public/sandbox/chunks/chunk-AGF4RRCB.js +408 -0
- package/public/sandbox/chunks/chunk-ASJSJAAL.js +122 -0
- package/public/sandbox/chunks/chunk-ASVMK364.js +283 -0
- package/public/sandbox/chunks/chunk-AUDTI2BQ.js +895 -0
- package/public/sandbox/chunks/chunk-B2SZTA3M.js +16 -0
- package/public/sandbox/chunks/chunk-B6YS2UMV.js +372 -0
- package/public/sandbox/chunks/chunk-BCEO3MGD.js +244 -0
- package/public/sandbox/chunks/chunk-BCOO65D4.js +373 -0
- package/public/sandbox/chunks/chunk-BJOWSSEL.js +38 -0
- package/public/sandbox/chunks/chunk-BS2WBFW4.js +373 -0
- package/public/sandbox/chunks/chunk-BUVHG3NJ.js +439 -0
- package/public/sandbox/chunks/chunk-BVE33AN6.js +42 -0
- package/public/sandbox/chunks/chunk-BWTGJZVP.js +19 -0
- package/public/sandbox/chunks/chunk-BXD7GZC3.js +1025 -0
- package/public/sandbox/chunks/chunk-BZH4EUUD.js +164 -0
- package/public/sandbox/chunks/{chunk-HFV5GJPK.js → chunk-C2MYMMO4.js} +3 -3
- package/public/sandbox/chunks/chunk-CRE4NJJ6.js +271 -0
- package/public/sandbox/chunks/chunk-CVZPRIDB.js +706 -0
- package/public/sandbox/chunks/chunk-CWOXD5NG.js +991 -0
- package/public/sandbox/chunks/chunk-D6DOC3BJ.js +373 -0
- package/public/sandbox/chunks/chunk-DEPSGFSY.js +1698 -0
- package/public/sandbox/chunks/chunk-DM7C5NP5.js +819 -0
- package/public/sandbox/chunks/chunk-DN33LMOF.js +1112 -0
- package/public/sandbox/chunks/{chunk-S46P7MC6.js → chunk-DT7YIIWE.js} +2 -2
- package/public/sandbox/chunks/chunk-DVB5WRCI.js +260 -0
- package/public/sandbox/chunks/{chunk-TOHCFMQS.js → chunk-E3EVW7NR.js} +11 -18
- package/public/sandbox/chunks/chunk-E5BV567Q.js +417 -0
- package/public/sandbox/chunks/chunk-E7BIWMQQ.js +203 -0
- package/public/sandbox/chunks/chunk-EAKGPSOQ.js +83 -0
- package/public/sandbox/chunks/chunk-EGUYL2ER.js +75 -0
- package/public/sandbox/chunks/chunk-EP5GQYZ5.js +85 -0
- package/public/sandbox/chunks/chunk-ET6CJEHC.js +221 -0
- package/public/sandbox/chunks/chunk-EU67YZHZ.js +1070 -0
- package/public/sandbox/chunks/{chunk-PTSJJANP.js → chunk-EYNDQLSB.js} +1 -1
- package/public/sandbox/chunks/{chunk-CEGSWZ5S.js → chunk-EZTMVCJF.js} +1 -1
- package/public/sandbox/chunks/chunk-F5CLVIS6.js +907 -0
- package/public/sandbox/chunks/chunk-F743SKYD.js +85 -0
- package/public/sandbox/chunks/chunk-FEGFZ5OX.js +86 -0
- package/public/sandbox/chunks/chunk-FG5B7ZX3.js +373 -0
- package/public/sandbox/chunks/chunk-FKZPYMDG.js +373 -0
- package/public/sandbox/chunks/chunk-FNLXHWIU.js +220 -0
- package/public/sandbox/chunks/chunk-FQRUTYAD.js +1991 -0
- package/public/sandbox/chunks/{chunk-VLYP4WUS.js → chunk-FY5A73NK.js} +5 -5
- package/public/sandbox/chunks/chunk-G7FAOG3O.js +221 -0
- package/public/sandbox/chunks/chunk-GFXHUC7Y.js +73 -0
- package/public/sandbox/chunks/chunk-GIPXUIZQ.js +534 -0
- package/public/sandbox/chunks/chunk-GP37MEGJ.js +52 -0
- package/public/sandbox/chunks/chunk-GXL5QCMF.js +4511 -0
- package/public/sandbox/chunks/chunk-GZ2NSE6A.js +177 -0
- package/public/sandbox/chunks/chunk-GZATW4YD.js +69 -0
- package/public/sandbox/chunks/chunk-H4EBXROQ.js +166 -0
- package/public/sandbox/chunks/chunk-HETCPK6Z.js +153 -0
- package/public/sandbox/chunks/chunk-HGUFQ2QC.js +4431 -0
- package/public/sandbox/chunks/chunk-HGV3VNH3.js +255 -0
- package/public/sandbox/chunks/chunk-HIJ67QWR.js +5397 -0
- package/public/sandbox/chunks/chunk-IDHERBKW.js +1114 -0
- package/public/sandbox/chunks/chunk-IDTK4FIY.js +102 -0
- package/public/sandbox/chunks/chunk-ILFVY7H5.js +164 -0
- package/public/sandbox/chunks/chunk-ISGSRYGO.js +57 -0
- package/public/sandbox/chunks/chunk-J45VPU24.js +408 -0
- package/public/sandbox/chunks/chunk-JAUIM3O5.js +42 -0
- package/public/sandbox/chunks/chunk-JB7OQEM6.js +373 -0
- package/public/sandbox/chunks/chunk-JCKW6NHD.js +15 -0
- package/public/sandbox/chunks/chunk-JMA7GY7Z.js +221 -0
- package/public/sandbox/chunks/chunk-JY46J6IL.js +836 -0
- package/public/sandbox/chunks/{chunk-X72PZOFA.js → chunk-K2MQ66O7.js} +1 -1
- package/public/sandbox/chunks/chunk-KA7PZA6F.js +464 -0
- package/public/sandbox/chunks/chunk-KDEQTUID.js +102 -0
- package/public/sandbox/chunks/chunk-KDJR27JV.js +1696 -0
- package/public/sandbox/chunks/chunk-KFE3JO6N.js +77 -0
- package/public/sandbox/chunks/chunk-KLDSYWT5.js +536 -0
- package/public/sandbox/chunks/chunk-KQWWFYBF.js +203 -0
- package/public/sandbox/chunks/chunk-KTFZSXOW.js +6473 -0
- package/public/sandbox/chunks/chunk-KXVIC55U.js +108 -0
- package/public/sandbox/chunks/{chunk-HNEDTEGQ.js → chunk-L4PNO2H6.js} +1 -1
- package/public/sandbox/chunks/{chunk-WE4STGO2.js → chunk-L4ZNW5DI.js} +2 -2
- package/public/sandbox/chunks/chunk-LBUINGZ4.js +836 -0
- package/public/sandbox/chunks/chunk-LIYPPU6I.js +112 -0
- package/public/sandbox/chunks/chunk-LJHTOJXW.js +221 -0
- package/public/sandbox/chunks/{chunk-BWSNPKO5.js → chunk-LQL23SWJ.js} +2 -2
- package/public/sandbox/chunks/chunk-LTFCYZQ6.js +46 -0
- package/public/sandbox/chunks/chunk-LTWLVCKQ.js +166 -0
- package/public/sandbox/chunks/chunk-LWI7BXFN.js +395 -0
- package/public/sandbox/chunks/chunk-LYBBX6WM.js +225 -0
- package/public/sandbox/chunks/chunk-MXE7HIBM.js +474 -0
- package/public/sandbox/chunks/chunk-MYBDYNK4.js +6004 -0
- package/public/sandbox/chunks/chunk-NC2MIWZX.js +732 -0
- package/public/sandbox/chunks/chunk-NE447XAF.js +597 -0
- package/public/sandbox/chunks/chunk-NEAYV5C2.js +255 -0
- package/public/sandbox/chunks/chunk-NMGFJZ4I.js +26 -0
- package/public/sandbox/chunks/chunk-NWCJJZNQ.js +71 -0
- package/public/sandbox/chunks/chunk-O57OZBDR.js +214 -0
- package/public/sandbox/chunks/chunk-OJJT664Q.js +731 -0
- package/public/sandbox/chunks/chunk-OKTBUTCH.js +1175 -0
- package/public/sandbox/chunks/chunk-OQ4SYV5D.js +534 -0
- package/public/sandbox/chunks/chunk-OTVVIHJD.js +219 -0
- package/public/sandbox/chunks/chunk-OWTMSCF7.js +1052 -0
- package/public/sandbox/chunks/chunk-P5ZQNFC3.js +122 -0
- package/public/sandbox/chunks/chunk-PBPT2U2N.js +1698 -0
- package/public/sandbox/chunks/chunk-PHBJLCP3.js +836 -0
- package/public/sandbox/chunks/chunk-PHV4HYHB.js +397 -0
- package/public/sandbox/chunks/chunk-PIXQ6SOX.js +4071 -0
- package/public/sandbox/chunks/chunk-PKJL6YLK.js +73 -0
- package/public/sandbox/chunks/chunk-PKPOITIU.js +221 -0
- package/public/sandbox/chunks/{chunk-NQW73ACJ.js → chunk-PMXBOYQI.js} +1 -1
- package/public/sandbox/chunks/chunk-PQLZD7NR.js +836 -0
- package/public/sandbox/chunks/chunk-QEESVOZU.js +16 -0
- package/public/sandbox/chunks/chunk-QGIM3FQK.js +159 -0
- package/public/sandbox/chunks/chunk-QL5JNEDO.js +69 -0
- package/public/sandbox/chunks/chunk-QQOZ3RMH.js +867 -0
- package/public/sandbox/chunks/chunk-QUMEVT2F.js +194 -0
- package/public/sandbox/chunks/chunk-QVZBXYVY.js +102 -0
- package/public/sandbox/chunks/chunk-R5HQS6RB.js +135 -0
- package/public/sandbox/chunks/chunk-R75Q2CTT.js +900 -0
- package/public/sandbox/chunks/chunk-RBB6SHW2.js +253 -0
- package/public/sandbox/chunks/chunk-RDHNRWQP.js +383 -0
- package/public/sandbox/chunks/chunk-RIWOEI3T.js +1101 -0
- package/public/sandbox/chunks/chunk-RS5FFPUV.js +103 -0
- package/public/sandbox/chunks/chunk-RYSSJBGX.js +203 -0
- package/public/sandbox/chunks/chunk-S3KBAE5Z.js +1114 -0
- package/public/sandbox/chunks/chunk-S6ZZHV6G.js +244 -0
- package/public/sandbox/chunks/chunk-SFXJ24HY.js +499 -0
- package/public/sandbox/chunks/chunk-SH7TP5CJ.js +1109 -0
- package/public/sandbox/chunks/chunk-SI5OKFFY.js +317 -0
- package/public/sandbox/chunks/chunk-SIXNQNIR.js +86 -0
- package/public/sandbox/chunks/chunk-ST353BOX.js +42 -0
- package/public/sandbox/chunks/chunk-SYEHKMLP.js +1140 -0
- package/public/sandbox/chunks/chunk-SZ4KQUJU.js +69 -0
- package/public/sandbox/chunks/{chunk-5SOP6EKV.js → chunk-SZI65PSP.js} +5 -10
- package/public/sandbox/chunks/chunk-T7RLTSMX.js +78 -0
- package/public/sandbox/chunks/chunk-TJ2RLUTW.js +57 -0
- package/public/sandbox/chunks/chunk-TJB73XPN.js +4071 -0
- package/public/sandbox/chunks/chunk-TXJCYO56.js +4046 -0
- package/public/sandbox/chunks/chunk-U5J3LCYI.js +731 -0
- package/public/sandbox/chunks/chunk-UABFN3NH.js +208 -0
- package/public/sandbox/chunks/chunk-UBMUY7WO.js +13 -0
- package/public/sandbox/chunks/chunk-UC27CE32.js +69 -0
- package/public/sandbox/chunks/chunk-UJGVDFZN.js +147 -0
- package/public/sandbox/chunks/chunk-UNVLTK4N.js +122 -0
- package/public/sandbox/chunks/chunk-UQKQ6V3V.js +208 -0
- package/public/sandbox/chunks/chunk-UZZBHWIO.js +703 -0
- package/public/sandbox/chunks/chunk-VKZIJERL.js +1114 -0
- package/public/sandbox/chunks/chunk-VQQEAGID.js +372 -0
- package/public/sandbox/chunks/{chunk-3LX3KIQQ.js → chunk-W365QUZ5.js} +3 -5
- package/public/sandbox/chunks/chunk-WJY2LSWG.js +1052 -0
- package/public/sandbox/chunks/chunk-WNOBN3DX.js +391 -0
- package/public/sandbox/chunks/chunk-X2Z5VYG3.js +245 -0
- package/public/sandbox/chunks/chunk-X74KU74D.js +959 -0
- package/public/sandbox/chunks/chunk-XE7ULSGQ.js +46 -0
- package/public/sandbox/chunks/chunk-XRXUJZ3X.js +431 -0
- package/public/sandbox/chunks/chunk-XWHIN7CC.js +85 -0
- package/public/sandbox/chunks/chunk-Y2TVGGVL.js +527 -0
- package/public/sandbox/chunks/chunk-YMVMZA3U.js +1395 -0
- package/public/sandbox/chunks/chunk-YOBLFWDV.js +431 -0
- package/public/sandbox/chunks/chunk-YR2RFEZA.js +11 -0
- package/public/sandbox/chunks/chunk-YTD6A35P.js +836 -0
- package/public/sandbox/chunks/chunk-Z5L7EZXS.js +81 -0
- package/public/sandbox/chunks/chunk-Z65I5ET2.js +108 -0
- package/public/sandbox/chunks/chunk-Z6LV34RO.js +390 -0
- package/public/sandbox/chunks/chunk-ZAKUP6QG.js +1634 -0
- package/public/sandbox/chunks/chunk-ZHNYE2YH.js +731 -0
- package/public/sandbox/chunks/chunk-ZHQX3B3Y.js +867 -0
- package/public/sandbox/chunks/chunk-ZIXDFRHJ.js +731 -0
- package/public/sandbox/chunks/chunk-ZPZ2MZ4D.js +528 -0
- package/public/sandbox/chunks/chunk-ZQD6NZXB.js +373 -0
- package/public/sandbox/chunks/chunk-ZVVBP7ON.js +16 -0
- package/public/sandbox/chunks/chunk-ZWE3P6X2.js +11 -0
- package/public/sandbox/logix-core/Bound.js +14 -14
- package/public/sandbox/logix-core/Debug.js +12 -10
- package/public/sandbox/logix-core/EffectOp.js +3 -4
- package/public/sandbox/logix-core/Env.js +5 -2
- package/public/sandbox/logix-core/ExternalStore.js +239 -0
- package/public/sandbox/logix-core/Flow.js +7 -8
- package/public/sandbox/logix-core/Kernel.js +7 -4
- package/public/sandbox/logix-core/Link.js +33 -25
- package/public/sandbox/logix-core/Middleware.js +13 -11
- package/public/sandbox/logix-core/Module.js +93 -30
- package/public/sandbox/logix-core/ModuleTag.js +28 -25
- package/public/sandbox/logix-core/Observability.js +21 -19
- package/public/sandbox/logix-core/Process.js +27 -22
- package/public/sandbox/logix-core/ReadQuery.js +7 -11
- package/public/sandbox/logix-core/Reflection.js +173 -53
- package/public/sandbox/logix-core/Resource.js +6 -3
- package/public/sandbox/logix-core/Root.js +7 -3
- package/public/sandbox/logix-core/Runtime.js +54 -40
- package/public/sandbox/logix-core/ScopeRegistry.js +6 -3
- package/public/sandbox/logix-core/StateTrait.js +17 -14
- package/public/sandbox/logix-core/TraitLifecycle.js +8 -8
- package/public/sandbox/logix-core/Workflow.js +332 -0
- package/public/sandbox/logix-core.js +11133 -6622
- package/public/sandbox/logix-core.manifest.json +4 -2
- package/public/sandbox/worker.js +3 -3
- package/public/sandbox/chunks/chunk-4LRLOTMA.js +0 -1698
- package/public/sandbox/chunks/chunk-CDLXJHXY.js +0 -120
- package/public/sandbox/chunks/chunk-CZRI7MHA.js +0 -46
- package/public/sandbox/chunks/chunk-EJDFUZ4B.js +0 -408
- package/public/sandbox/chunks/chunk-IP63FME6.js +0 -282
- package/public/sandbox/chunks/chunk-JIXTOQXJ.js +0 -1103
- package/public/sandbox/chunks/chunk-MPUSOFJ3.js +0 -1928
- package/public/sandbox/chunks/chunk-ON2LY6HJ.js +0 -135
- package/public/sandbox/chunks/chunk-POIBXAH5.js +0 -995
- package/public/sandbox/chunks/chunk-PQ652ULI.js +0 -6359
- package/public/sandbox/chunks/chunk-RJQ4PG2F.js +0 -77
- package/public/sandbox/chunks/chunk-S4EXG3OS.js +0 -533
- package/public/sandbox/chunks/chunk-SUXDGXVE.js +0 -390
- package/public/sandbox/chunks/chunk-TOQ6SJ6O.js +0 -244
- package/public/sandbox/chunks/chunk-UKYAF3NL.js +0 -1036
- package/public/sandbox/chunks/chunk-UMVN3AWO.js +0 -395
- package/public/sandbox/chunks/chunk-WFV5PPWJ.js +0 -194
- package/public/sandbox/chunks/chunk-YF6IO732.js +0 -206
- package/public/sandbox/chunks/chunk-ZUBABAAO.js +0 -86
|
@@ -0,0 +1,836 @@
|
|
|
1
|
+
// @logix/core subpath bundles for @logix/sandbox
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
exportStaticIr
|
|
5
|
+
} from "./chunk-QGIM3FQK.js";
|
|
6
|
+
import {
|
|
7
|
+
installSourceRefresh
|
|
8
|
+
} from "./chunk-S3KBAE5Z.js";
|
|
9
|
+
import {
|
|
10
|
+
getAtPath
|
|
11
|
+
} from "./chunk-67RVISUC.js";
|
|
12
|
+
import {
|
|
13
|
+
compareFieldPath,
|
|
14
|
+
getFieldPathId,
|
|
15
|
+
makeFieldPathIdRegistry,
|
|
16
|
+
normalizeFieldPath
|
|
17
|
+
} from "./chunk-5XOBKSOT.js";
|
|
18
|
+
import {
|
|
19
|
+
getBoundInternals
|
|
20
|
+
} from "./chunk-6FKWRVP4.js";
|
|
21
|
+
|
|
22
|
+
// ../logix-core/src/internal/state-trait/meta.ts
|
|
23
|
+
var uniqSortedStrings = (input) => {
|
|
24
|
+
const set = /* @__PURE__ */ new Set();
|
|
25
|
+
for (const item of input) {
|
|
26
|
+
const v = item.trim();
|
|
27
|
+
if (!v) continue;
|
|
28
|
+
set.add(v);
|
|
29
|
+
}
|
|
30
|
+
return Array.from(set).sort((a, b) => a.localeCompare(b));
|
|
31
|
+
};
|
|
32
|
+
var isPlainRecord = (value) => typeof value === "object" && value !== null && !Array.isArray(value);
|
|
33
|
+
var sanitizeJsonValue = (input, depth) => {
|
|
34
|
+
if (input === null) return null;
|
|
35
|
+
if (typeof input === "string") return input;
|
|
36
|
+
if (typeof input === "boolean") return input;
|
|
37
|
+
if (typeof input === "number") {
|
|
38
|
+
return Number.isFinite(input) ? input : void 0;
|
|
39
|
+
}
|
|
40
|
+
if (depth >= 6) return void 0;
|
|
41
|
+
if (Array.isArray(input)) {
|
|
42
|
+
const out = [];
|
|
43
|
+
for (const item of input) {
|
|
44
|
+
const v = sanitizeJsonValue(item, depth + 1);
|
|
45
|
+
if (v !== void 0) out.push(v);
|
|
46
|
+
}
|
|
47
|
+
return out;
|
|
48
|
+
}
|
|
49
|
+
if (isPlainRecord(input)) {
|
|
50
|
+
const keys = Object.keys(input).sort();
|
|
51
|
+
const out = {};
|
|
52
|
+
for (const key of keys) {
|
|
53
|
+
const v = sanitizeJsonValue(input[key], depth + 1);
|
|
54
|
+
if (v !== void 0) out[key] = v;
|
|
55
|
+
}
|
|
56
|
+
return out;
|
|
57
|
+
}
|
|
58
|
+
return void 0;
|
|
59
|
+
};
|
|
60
|
+
var sanitize = (input) => {
|
|
61
|
+
if (input === null || input === void 0) return void 0;
|
|
62
|
+
if (typeof input !== "object" || Array.isArray(input)) return void 0;
|
|
63
|
+
const record = input;
|
|
64
|
+
const out = {};
|
|
65
|
+
const pickString = (key) => {
|
|
66
|
+
const value = record[key];
|
|
67
|
+
if (typeof value !== "string") return;
|
|
68
|
+
const trimmed = value.trim();
|
|
69
|
+
if (!trimmed) return;
|
|
70
|
+
out[key] = trimmed;
|
|
71
|
+
};
|
|
72
|
+
pickString("label");
|
|
73
|
+
pickString("description");
|
|
74
|
+
pickString("group");
|
|
75
|
+
pickString("docsUrl");
|
|
76
|
+
pickString("cacheGroup");
|
|
77
|
+
const canonical = record.canonical;
|
|
78
|
+
if (typeof canonical === "boolean") {
|
|
79
|
+
out.canonical = canonical;
|
|
80
|
+
}
|
|
81
|
+
const tagsRaw = record.tags;
|
|
82
|
+
if (typeof tagsRaw === "string") {
|
|
83
|
+
const tags = uniqSortedStrings([tagsRaw]);
|
|
84
|
+
if (tags.length > 0) out.tags = tags;
|
|
85
|
+
} else if (Array.isArray(tagsRaw)) {
|
|
86
|
+
const tags = uniqSortedStrings(tagsRaw.filter((x) => typeof x === "string"));
|
|
87
|
+
if (tags.length > 0) out.tags = tags;
|
|
88
|
+
}
|
|
89
|
+
const annotations = {};
|
|
90
|
+
const annotationKeys = Object.keys(record).filter((k) => k.startsWith("x-")).sort((a, b) => a.localeCompare(b));
|
|
91
|
+
for (const key of annotationKeys) {
|
|
92
|
+
const v = sanitizeJsonValue(record[key], 0);
|
|
93
|
+
if (v !== void 0) annotations[key] = v;
|
|
94
|
+
}
|
|
95
|
+
const annotationsRaw = record.annotations;
|
|
96
|
+
if (isPlainRecord(annotationsRaw)) {
|
|
97
|
+
const keys = Object.keys(annotationsRaw).filter((k) => k.startsWith("x-")).sort((a, b) => a.localeCompare(b));
|
|
98
|
+
for (const key of keys) {
|
|
99
|
+
const v = sanitizeJsonValue(annotationsRaw[key], 0);
|
|
100
|
+
if (v !== void 0) annotations[key] = v;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
if (Object.keys(annotations).length > 0) out.annotations = annotations;
|
|
104
|
+
return Object.keys(out).length > 0 ? out : void 0;
|
|
105
|
+
};
|
|
106
|
+
var stableStringify = (meta) => {
|
|
107
|
+
const out = {};
|
|
108
|
+
const keys = Object.keys(meta).sort();
|
|
109
|
+
for (const k of keys) {
|
|
110
|
+
out[k] = meta[k];
|
|
111
|
+
}
|
|
112
|
+
return JSON.stringify(out);
|
|
113
|
+
};
|
|
114
|
+
var equals = (a, b) => {
|
|
115
|
+
if (a === b) return true;
|
|
116
|
+
if (!a || !b) return false;
|
|
117
|
+
return stableStringify(a) === stableStringify(b);
|
|
118
|
+
};
|
|
119
|
+
var mergeCanonical = (current, incoming) => {
|
|
120
|
+
const existing = current.meta;
|
|
121
|
+
if (!existing) {
|
|
122
|
+
return {
|
|
123
|
+
meta: incoming.meta,
|
|
124
|
+
origin: incoming.origin,
|
|
125
|
+
conflicts: current.conflicts
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
if (equals(existing, incoming.meta)) {
|
|
129
|
+
return current;
|
|
130
|
+
}
|
|
131
|
+
const conflicts = [
|
|
132
|
+
...current.conflicts ?? []
|
|
133
|
+
];
|
|
134
|
+
const push = (conflict) => {
|
|
135
|
+
const exists = conflicts.some(
|
|
136
|
+
(c) => c.origin === conflict.origin && equals(c.meta, conflict.meta)
|
|
137
|
+
);
|
|
138
|
+
if (!exists) conflicts.push(conflict);
|
|
139
|
+
};
|
|
140
|
+
const existingCanonical = existing.canonical === true;
|
|
141
|
+
const incomingCanonical = incoming.meta.canonical === true;
|
|
142
|
+
if (!existingCanonical && incomingCanonical) {
|
|
143
|
+
push({
|
|
144
|
+
origin: current.origin ?? "unknown",
|
|
145
|
+
meta: existing
|
|
146
|
+
});
|
|
147
|
+
return {
|
|
148
|
+
meta: incoming.meta,
|
|
149
|
+
origin: incoming.origin,
|
|
150
|
+
conflicts
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
push({
|
|
154
|
+
origin: incoming.origin,
|
|
155
|
+
meta: incoming.meta
|
|
156
|
+
});
|
|
157
|
+
return {
|
|
158
|
+
meta: existing,
|
|
159
|
+
origin: current.origin ?? incoming.origin,
|
|
160
|
+
conflicts
|
|
161
|
+
};
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
// ../logix-core/src/internal/state-trait/model.ts
|
|
165
|
+
var normalizeSpec = (spec) => {
|
|
166
|
+
const entries = [];
|
|
167
|
+
const isNode = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitNode";
|
|
168
|
+
const isList = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitList";
|
|
169
|
+
const joinPath = (prefix, suffix) => {
|
|
170
|
+
if (!prefix) return suffix;
|
|
171
|
+
if (!suffix) return prefix;
|
|
172
|
+
return `${prefix}.${suffix}`;
|
|
173
|
+
};
|
|
174
|
+
const prefixDeps = (deps, prefix) => {
|
|
175
|
+
if (!deps || deps.length === 0) return [];
|
|
176
|
+
return deps.map((d) => prefix ? joinPath(prefix, d) : d);
|
|
177
|
+
};
|
|
178
|
+
const normalizeEntry = (entry, fieldPath, depPrefix) => {
|
|
179
|
+
if (entry.kind === "computed") {
|
|
180
|
+
const meta = entry.meta;
|
|
181
|
+
const rawDeps = meta.deps;
|
|
182
|
+
const deps = rawDeps !== void 0 ? prefixDeps(rawDeps, depPrefix) : void 0;
|
|
183
|
+
return {
|
|
184
|
+
...entry,
|
|
185
|
+
fieldPath,
|
|
186
|
+
meta: { ...meta, deps }
|
|
187
|
+
};
|
|
188
|
+
}
|
|
189
|
+
if (entry.kind === "source") {
|
|
190
|
+
const meta = entry.meta;
|
|
191
|
+
const rawDeps = meta.deps;
|
|
192
|
+
const deps = rawDeps !== void 0 ? prefixDeps(rawDeps, depPrefix) : void 0;
|
|
193
|
+
return {
|
|
194
|
+
...entry,
|
|
195
|
+
fieldPath,
|
|
196
|
+
meta: { ...meta, deps, _fieldPath: fieldPath }
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
if (entry.kind === "link") {
|
|
200
|
+
const meta = entry.meta;
|
|
201
|
+
const from2 = prefixDeps([meta.from], depPrefix)[0] ?? meta.from;
|
|
202
|
+
return {
|
|
203
|
+
...entry,
|
|
204
|
+
fieldPath,
|
|
205
|
+
meta: { ...meta, from: from2 }
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
return {
|
|
209
|
+
...entry,
|
|
210
|
+
fieldPath
|
|
211
|
+
};
|
|
212
|
+
};
|
|
213
|
+
const expandNode = (scopeId, joinPrefix, node2, options) => {
|
|
214
|
+
const addEntry = (relativeTarget, raw) => {
|
|
215
|
+
const rel = raw.fieldPath ?? relativeTarget;
|
|
216
|
+
const fieldPath = joinPrefix ? joinPath(joinPrefix, String(rel)) : String(rel);
|
|
217
|
+
entries.push(normalizeEntry(raw, fieldPath, joinPrefix));
|
|
218
|
+
};
|
|
219
|
+
const expandMaybeRecord = (value) => {
|
|
220
|
+
if (!value) return;
|
|
221
|
+
if (typeof value.kind === "string") {
|
|
222
|
+
addEntry("", value);
|
|
223
|
+
return;
|
|
224
|
+
}
|
|
225
|
+
const record = value;
|
|
226
|
+
for (const key in record) {
|
|
227
|
+
if (!Object.prototype.hasOwnProperty.call(record, key)) continue;
|
|
228
|
+
const entry = record[key];
|
|
229
|
+
if (!entry) continue;
|
|
230
|
+
addEntry(key, entry);
|
|
231
|
+
}
|
|
232
|
+
};
|
|
233
|
+
expandMaybeRecord(node2.computed);
|
|
234
|
+
expandMaybeRecord(node2.source);
|
|
235
|
+
expandMaybeRecord(node2.link);
|
|
236
|
+
if (node2.check) {
|
|
237
|
+
const rules = {};
|
|
238
|
+
const checkDepsPrefix = options?.checkDepsPrefix ?? (joinPrefix.endsWith("[]") ? joinPrefix : "");
|
|
239
|
+
const prefixCheckDeps = (deps) => {
|
|
240
|
+
if (!deps || deps.length === 0) return [];
|
|
241
|
+
return deps.map(
|
|
242
|
+
(d) => d === "" ? scopeId : checkDepsPrefix ? joinPath(checkDepsPrefix, d) : d
|
|
243
|
+
);
|
|
244
|
+
};
|
|
245
|
+
for (const name of Object.keys(node2.check)) {
|
|
246
|
+
const rule = node2.check[name];
|
|
247
|
+
if (typeof rule === "function") {
|
|
248
|
+
rules[name] = rule;
|
|
249
|
+
continue;
|
|
250
|
+
}
|
|
251
|
+
if (rule && typeof rule === "object") {
|
|
252
|
+
const meta = sanitize(rule.meta);
|
|
253
|
+
rules[name] = {
|
|
254
|
+
...rule,
|
|
255
|
+
deps: prefixCheckDeps(rule.deps),
|
|
256
|
+
meta
|
|
257
|
+
};
|
|
258
|
+
continue;
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
entries.push({
|
|
262
|
+
fieldPath: scopeId,
|
|
263
|
+
kind: "check",
|
|
264
|
+
meta: {
|
|
265
|
+
rules,
|
|
266
|
+
writeback: { kind: "errors" }
|
|
267
|
+
}
|
|
268
|
+
});
|
|
269
|
+
}
|
|
270
|
+
};
|
|
271
|
+
for (const key in spec) {
|
|
272
|
+
if (!Object.prototype.hasOwnProperty.call(spec, key)) continue;
|
|
273
|
+
const raw = spec[key];
|
|
274
|
+
if (!raw) continue;
|
|
275
|
+
if (isList(raw)) {
|
|
276
|
+
const listPath = key;
|
|
277
|
+
if (raw.item) {
|
|
278
|
+
expandNode(`${listPath}[]`, `${listPath}[]`, raw.item);
|
|
279
|
+
}
|
|
280
|
+
if (raw.list) {
|
|
281
|
+
expandNode(listPath, listPath, raw.list, {
|
|
282
|
+
checkDepsPrefix: `${listPath}[]`
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
continue;
|
|
286
|
+
}
|
|
287
|
+
if (isNode(raw)) {
|
|
288
|
+
if (key === "$root") {
|
|
289
|
+
expandNode("$root", "", raw);
|
|
290
|
+
} else {
|
|
291
|
+
expandNode(key, key, raw);
|
|
292
|
+
}
|
|
293
|
+
continue;
|
|
294
|
+
}
|
|
295
|
+
const entry = raw;
|
|
296
|
+
const fieldPath = entry.fieldPath ?? key;
|
|
297
|
+
entries.push(normalizeEntry(entry, String(fieldPath), ""));
|
|
298
|
+
}
|
|
299
|
+
return entries;
|
|
300
|
+
};
|
|
301
|
+
var collectNodeMeta = (spec) => {
|
|
302
|
+
const out = /* @__PURE__ */ new Map();
|
|
303
|
+
const isNode = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitNode";
|
|
304
|
+
const isList = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitList";
|
|
305
|
+
const add = (scopeId, node2) => {
|
|
306
|
+
const meta = sanitize(node2.meta);
|
|
307
|
+
if (meta) out.set(scopeId, meta);
|
|
308
|
+
};
|
|
309
|
+
for (const key in spec) {
|
|
310
|
+
if (!Object.prototype.hasOwnProperty.call(spec, key)) continue;
|
|
311
|
+
const raw = spec[key];
|
|
312
|
+
if (!raw) continue;
|
|
313
|
+
if (isList(raw)) {
|
|
314
|
+
const listPath = key;
|
|
315
|
+
if (raw.item) add(`${listPath}[]`, raw.item);
|
|
316
|
+
if (raw.list) add(listPath, raw.list);
|
|
317
|
+
continue;
|
|
318
|
+
}
|
|
319
|
+
if (isNode(raw)) {
|
|
320
|
+
if (key === "$root") add("$root", raw);
|
|
321
|
+
else add(key, raw);
|
|
322
|
+
continue;
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
return out;
|
|
326
|
+
};
|
|
327
|
+
|
|
328
|
+
// ../logix-core/src/internal/state-trait/build.ts
|
|
329
|
+
var nowPerf = () => typeof globalThis.performance !== "undefined" && typeof globalThis.performance.now === "function" ? globalThis.performance.now() : Date.now();
|
|
330
|
+
var getConvergeWriterDeps = (entry) => {
|
|
331
|
+
if (entry.kind === "computed") {
|
|
332
|
+
return entry.meta?.deps ?? [];
|
|
333
|
+
}
|
|
334
|
+
return [entry.meta.from];
|
|
335
|
+
};
|
|
336
|
+
var computeConvergeTopoOrder = (writers) => {
|
|
337
|
+
const writerByPath = /* @__PURE__ */ new Map();
|
|
338
|
+
for (const entry of writers) {
|
|
339
|
+
const existing = writerByPath.get(entry.fieldPath);
|
|
340
|
+
if (existing) {
|
|
341
|
+
return {
|
|
342
|
+
order: [],
|
|
343
|
+
configError: {
|
|
344
|
+
code: "MULTIPLE_WRITERS",
|
|
345
|
+
message: `[StateTrait.converge] Multiple writers for field "${entry.fieldPath}" (${existing.kind} + ${entry.kind}).`,
|
|
346
|
+
fields: [entry.fieldPath]
|
|
347
|
+
}
|
|
348
|
+
};
|
|
349
|
+
}
|
|
350
|
+
writerByPath.set(entry.fieldPath, entry);
|
|
351
|
+
}
|
|
352
|
+
const nodes = /* @__PURE__ */ new Set();
|
|
353
|
+
for (const entry of writers) {
|
|
354
|
+
nodes.add(entry.fieldPath);
|
|
355
|
+
}
|
|
356
|
+
const indegree = /* @__PURE__ */ new Map();
|
|
357
|
+
const forward = /* @__PURE__ */ new Map();
|
|
358
|
+
for (const node2 of nodes) {
|
|
359
|
+
indegree.set(node2, 0);
|
|
360
|
+
forward.set(node2, []);
|
|
361
|
+
}
|
|
362
|
+
for (const entry of writers) {
|
|
363
|
+
const to = entry.fieldPath;
|
|
364
|
+
const deps = getConvergeWriterDeps(entry);
|
|
365
|
+
for (const dep of deps) {
|
|
366
|
+
if (!nodes.has(dep)) continue;
|
|
367
|
+
forward.get(dep).push(to);
|
|
368
|
+
indegree.set(to, (indegree.get(to) ?? 0) + 1);
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
const queue = [];
|
|
372
|
+
for (const [node2, deg] of indegree.entries()) {
|
|
373
|
+
if (deg === 0) queue.push(node2);
|
|
374
|
+
}
|
|
375
|
+
const order = [];
|
|
376
|
+
while (queue.length) {
|
|
377
|
+
const n = queue.shift();
|
|
378
|
+
order.push(n);
|
|
379
|
+
const outs = forward.get(n);
|
|
380
|
+
for (const to of outs) {
|
|
381
|
+
const next = (indegree.get(to) ?? 0) - 1;
|
|
382
|
+
indegree.set(to, next);
|
|
383
|
+
if (next === 0) queue.push(to);
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
if (order.length !== nodes.size) {
|
|
387
|
+
const remaining = Array.from(nodes).filter((n) => !order.includes(n));
|
|
388
|
+
return {
|
|
389
|
+
order: [],
|
|
390
|
+
configError: {
|
|
391
|
+
code: "CYCLE_DETECTED",
|
|
392
|
+
message: `[StateTrait.converge] Cycle detected in computed/link graph: ${remaining.join(", ")}`,
|
|
393
|
+
fields: remaining
|
|
394
|
+
}
|
|
395
|
+
};
|
|
396
|
+
}
|
|
397
|
+
return { order };
|
|
398
|
+
};
|
|
399
|
+
var buildConvergeIr = (entries) => {
|
|
400
|
+
const startedAt = nowPerf();
|
|
401
|
+
const generation = 0;
|
|
402
|
+
const writers = entries.filter(
|
|
403
|
+
(e) => e.kind === "computed" || e.kind === "link"
|
|
404
|
+
);
|
|
405
|
+
const writersKey = writers.map((entry) => `${entry.kind}:${entry.fieldPath}`).sort().join("|");
|
|
406
|
+
const depsKey = writers.map((entry) => {
|
|
407
|
+
const deps = getConvergeWriterDeps(entry).slice().sort().join(",");
|
|
408
|
+
return `${entry.kind}:${entry.fieldPath}=>${deps}`;
|
|
409
|
+
}).sort().join("|");
|
|
410
|
+
const writerByPath = /* @__PURE__ */ new Map();
|
|
411
|
+
for (const entry of writers) {
|
|
412
|
+
writerByPath.set(entry.fieldPath, entry);
|
|
413
|
+
}
|
|
414
|
+
const topo = writers.length > 0 ? computeConvergeTopoOrder(writers) : { order: [] };
|
|
415
|
+
const stepsById = topo.configError ? [] : topo.order.map((path) => writerByPath.get(path));
|
|
416
|
+
const fieldPathTable = /* @__PURE__ */ new Map();
|
|
417
|
+
const addPath = (path) => {
|
|
418
|
+
for (let i = 1; i <= path.length; i++) {
|
|
419
|
+
const prefix = path.slice(0, i);
|
|
420
|
+
const key = JSON.stringify(prefix);
|
|
421
|
+
if (!fieldPathTable.has(key)) fieldPathTable.set(key, prefix);
|
|
422
|
+
}
|
|
423
|
+
};
|
|
424
|
+
for (const entry of writers) {
|
|
425
|
+
const out = normalizeFieldPath(entry.fieldPath);
|
|
426
|
+
if (out) addPath(out);
|
|
427
|
+
for (const dep of getConvergeWriterDeps(entry)) {
|
|
428
|
+
const depPath = normalizeFieldPath(dep);
|
|
429
|
+
if (depPath) addPath(depPath);
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
const fieldPaths = Array.from(fieldPathTable.values()).sort(compareFieldPath);
|
|
433
|
+
const fieldPathIdRegistry = makeFieldPathIdRegistry(fieldPaths);
|
|
434
|
+
const stepOutFieldPathIdByStepId = [];
|
|
435
|
+
const stepDepsFieldPathIdsByStepId = [];
|
|
436
|
+
for (const entry of stepsById) {
|
|
437
|
+
const out = normalizeFieldPath(entry.fieldPath);
|
|
438
|
+
const outId = out != null ? getFieldPathId(fieldPathIdRegistry, out) : void 0;
|
|
439
|
+
if (outId == null) {
|
|
440
|
+
throw new Error(
|
|
441
|
+
`[StateTrait.build] Failed to map converge output fieldPath "${entry.fieldPath}" to FieldPathId.`
|
|
442
|
+
);
|
|
443
|
+
}
|
|
444
|
+
const depIds = [];
|
|
445
|
+
for (const dep of getConvergeWriterDeps(entry)) {
|
|
446
|
+
const depPath = normalizeFieldPath(dep);
|
|
447
|
+
if (!depPath) continue;
|
|
448
|
+
const depId = getFieldPathId(fieldPathIdRegistry, depPath);
|
|
449
|
+
if (depId != null) depIds.push(depId);
|
|
450
|
+
}
|
|
451
|
+
stepOutFieldPathIdByStepId.push(outId);
|
|
452
|
+
stepDepsFieldPathIdsByStepId.push(depIds);
|
|
453
|
+
}
|
|
454
|
+
const topoOrder = stepsById.map((_, i) => i);
|
|
455
|
+
const buildDurationMs = Math.max(0, nowPerf() - startedAt);
|
|
456
|
+
return {
|
|
457
|
+
generation,
|
|
458
|
+
writersKey,
|
|
459
|
+
depsKey,
|
|
460
|
+
fieldPaths,
|
|
461
|
+
fieldPathIdRegistry,
|
|
462
|
+
...topo.configError ? { configError: topo.configError } : null,
|
|
463
|
+
stepsById,
|
|
464
|
+
stepOutFieldPathIdByStepId,
|
|
465
|
+
stepDepsFieldPathIdsByStepId,
|
|
466
|
+
topoOrder,
|
|
467
|
+
buildDurationMs
|
|
468
|
+
};
|
|
469
|
+
};
|
|
470
|
+
var toFieldTrait = (entry) => {
|
|
471
|
+
const deps = [];
|
|
472
|
+
if (entry.kind === "computed") {
|
|
473
|
+
const meta = entry.meta;
|
|
474
|
+
const list2 = meta.deps;
|
|
475
|
+
if (list2) deps.push(...list2);
|
|
476
|
+
} else if (entry.kind === "source") {
|
|
477
|
+
const meta = entry.meta;
|
|
478
|
+
const list2 = meta.deps;
|
|
479
|
+
if (list2) deps.push(...list2);
|
|
480
|
+
} else if (entry.kind === "link") {
|
|
481
|
+
deps.push(entry.meta.from);
|
|
482
|
+
} else if (entry.kind === "check") {
|
|
483
|
+
const meta = entry.meta;
|
|
484
|
+
const rules = meta?.rules ?? {};
|
|
485
|
+
for (const name of Object.keys(rules)) {
|
|
486
|
+
const rule = rules[name];
|
|
487
|
+
if (rule && typeof rule === "object") {
|
|
488
|
+
const list2 = rule.deps;
|
|
489
|
+
if (list2) deps.push(...list2);
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
}
|
|
493
|
+
return {
|
|
494
|
+
fieldId: entry.fieldPath,
|
|
495
|
+
kind: entry.kind,
|
|
496
|
+
// meta 在运行时保持与 Entry.meta 一致,便于 install 直接复用。
|
|
497
|
+
meta: entry.meta,
|
|
498
|
+
deps
|
|
499
|
+
};
|
|
500
|
+
};
|
|
501
|
+
var buildGraph = (entries, nodeMetaByFieldPath) => {
|
|
502
|
+
const fieldMap = /* @__PURE__ */ new Map();
|
|
503
|
+
const nodes = [];
|
|
504
|
+
const edges = [];
|
|
505
|
+
const resourcesById = /* @__PURE__ */ new Map();
|
|
506
|
+
const planSteps = [];
|
|
507
|
+
const ensureField = (fieldPath) => {
|
|
508
|
+
let field = fieldMap.get(fieldPath);
|
|
509
|
+
if (!field) {
|
|
510
|
+
field = {
|
|
511
|
+
id: fieldPath,
|
|
512
|
+
path: fieldPath,
|
|
513
|
+
traits: []
|
|
514
|
+
};
|
|
515
|
+
fieldMap.set(fieldPath, field);
|
|
516
|
+
}
|
|
517
|
+
return field;
|
|
518
|
+
};
|
|
519
|
+
for (const entry of entries) {
|
|
520
|
+
const fieldPath = entry.fieldPath;
|
|
521
|
+
const field = ensureField(fieldPath);
|
|
522
|
+
const trait = toFieldTrait(entry);
|
|
523
|
+
field.traits.push(trait);
|
|
524
|
+
if (entry.kind === "computed") {
|
|
525
|
+
const stepId = `computed:${fieldPath}`;
|
|
526
|
+
planSteps.push({
|
|
527
|
+
id: stepId,
|
|
528
|
+
kind: "computed-update",
|
|
529
|
+
targetFieldPath: fieldPath
|
|
530
|
+
// 说明:当前版本不静态分析 computed 依赖字段,sourceFieldPaths 留空。
|
|
531
|
+
});
|
|
532
|
+
const deps = entry.meta.deps;
|
|
533
|
+
if (deps) {
|
|
534
|
+
for (const dep of deps) {
|
|
535
|
+
ensureField(dep);
|
|
536
|
+
edges.push({
|
|
537
|
+
id: `computed:${dep}->${fieldPath}`,
|
|
538
|
+
from: dep,
|
|
539
|
+
to: fieldPath,
|
|
540
|
+
kind: "computed"
|
|
541
|
+
});
|
|
542
|
+
}
|
|
543
|
+
}
|
|
544
|
+
} else if (entry.kind === "link") {
|
|
545
|
+
const from2 = entry.meta.from;
|
|
546
|
+
ensureField(from2);
|
|
547
|
+
const edgeId = `link:${from2}->${fieldPath}`;
|
|
548
|
+
edges.push({
|
|
549
|
+
id: edgeId,
|
|
550
|
+
from: from2,
|
|
551
|
+
to: fieldPath,
|
|
552
|
+
kind: "link"
|
|
553
|
+
});
|
|
554
|
+
planSteps.push({
|
|
555
|
+
id: `link:${fieldPath}`,
|
|
556
|
+
kind: "link-propagate",
|
|
557
|
+
targetFieldPath: fieldPath,
|
|
558
|
+
sourceFieldPaths: [from2],
|
|
559
|
+
debugInfo: {
|
|
560
|
+
graphEdgeId: edgeId
|
|
561
|
+
}
|
|
562
|
+
});
|
|
563
|
+
} else if (entry.kind === "source") {
|
|
564
|
+
const resourceId = entry.meta.resource;
|
|
565
|
+
const resourceMeta = sanitize(entry.meta.meta);
|
|
566
|
+
const existing = resourcesById.get(resourceId);
|
|
567
|
+
if (existing) {
|
|
568
|
+
const ownerFields = [...existing.ownerFields, fieldPath];
|
|
569
|
+
let meta = existing.meta;
|
|
570
|
+
let metaOrigin = existing.metaOrigin;
|
|
571
|
+
let metaConflicts = existing.metaConflicts;
|
|
572
|
+
if (resourceMeta) {
|
|
573
|
+
const merged = mergeCanonical(
|
|
574
|
+
{ meta, origin: metaOrigin, conflicts: metaConflicts },
|
|
575
|
+
{ origin: fieldPath, meta: resourceMeta }
|
|
576
|
+
);
|
|
577
|
+
meta = merged.meta;
|
|
578
|
+
metaOrigin = merged.origin;
|
|
579
|
+
metaConflicts = merged.conflicts;
|
|
580
|
+
}
|
|
581
|
+
resourcesById.set(resourceId, {
|
|
582
|
+
...existing,
|
|
583
|
+
ownerFields,
|
|
584
|
+
meta,
|
|
585
|
+
metaOrigin,
|
|
586
|
+
metaConflicts
|
|
587
|
+
});
|
|
588
|
+
} else {
|
|
589
|
+
resourcesById.set(resourceId, {
|
|
590
|
+
resourceId,
|
|
591
|
+
// 暂时使用简单的标识字符串,后续可根据 key 规则进一步结构化。
|
|
592
|
+
keySelector: `StateTrait.source@${fieldPath}`,
|
|
593
|
+
ownerFields: [fieldPath],
|
|
594
|
+
meta: resourceMeta,
|
|
595
|
+
metaOrigin: resourceMeta ? fieldPath : void 0
|
|
596
|
+
});
|
|
597
|
+
}
|
|
598
|
+
planSteps.push({
|
|
599
|
+
id: `source:${fieldPath}`,
|
|
600
|
+
kind: "source-refresh",
|
|
601
|
+
targetFieldPath: fieldPath,
|
|
602
|
+
resourceId,
|
|
603
|
+
keySelectorId: `StateTrait.source@${fieldPath}`
|
|
604
|
+
});
|
|
605
|
+
const deps = entry.meta.deps;
|
|
606
|
+
if (deps) {
|
|
607
|
+
for (const dep of deps) {
|
|
608
|
+
ensureField(dep);
|
|
609
|
+
edges.push({
|
|
610
|
+
id: `source-dep:${dep}->${fieldPath}`,
|
|
611
|
+
from: dep,
|
|
612
|
+
to: fieldPath,
|
|
613
|
+
kind: "source-dep"
|
|
614
|
+
});
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
} else if (entry.kind === "check") {
|
|
618
|
+
planSteps.push({
|
|
619
|
+
id: `check:${fieldPath}`,
|
|
620
|
+
kind: "check-validate",
|
|
621
|
+
targetFieldPath: fieldPath
|
|
622
|
+
});
|
|
623
|
+
if (trait.deps.length > 0) {
|
|
624
|
+
for (const dep of trait.deps) {
|
|
625
|
+
ensureField(dep);
|
|
626
|
+
edges.push({
|
|
627
|
+
id: `check-dep:${dep}->${fieldPath}`,
|
|
628
|
+
from: dep,
|
|
629
|
+
to: fieldPath,
|
|
630
|
+
kind: "check-dep"
|
|
631
|
+
});
|
|
632
|
+
}
|
|
633
|
+
}
|
|
634
|
+
}
|
|
635
|
+
}
|
|
636
|
+
for (const field of fieldMap.values()) {
|
|
637
|
+
nodes.push({
|
|
638
|
+
id: field.id,
|
|
639
|
+
field,
|
|
640
|
+
traits: field.traits,
|
|
641
|
+
meta: nodeMetaByFieldPath.get(field.id)
|
|
642
|
+
});
|
|
643
|
+
}
|
|
644
|
+
const graph = {
|
|
645
|
+
_tag: "StateTraitGraph",
|
|
646
|
+
nodes,
|
|
647
|
+
edges,
|
|
648
|
+
resources: Array.from(resourcesById.values())
|
|
649
|
+
};
|
|
650
|
+
const plan = {
|
|
651
|
+
_tag: "StateTraitPlan",
|
|
652
|
+
steps: planSteps
|
|
653
|
+
};
|
|
654
|
+
return { graph, plan };
|
|
655
|
+
};
|
|
656
|
+
var assertNoLinkCycles = (edges) => {
|
|
657
|
+
const adjacency = /* @__PURE__ */ new Map();
|
|
658
|
+
for (const edge of edges) {
|
|
659
|
+
if (edge.kind !== "link") continue;
|
|
660
|
+
const list2 = adjacency.get(edge.from) ?? [];
|
|
661
|
+
list2.push(edge.to);
|
|
662
|
+
adjacency.set(edge.from, list2);
|
|
663
|
+
}
|
|
664
|
+
const visited = /* @__PURE__ */ new Set();
|
|
665
|
+
const stack = /* @__PURE__ */ new Set();
|
|
666
|
+
const dfs = (node2) => {
|
|
667
|
+
if (stack.has(node2)) {
|
|
668
|
+
throw new Error(
|
|
669
|
+
`[StateTrait.build] link cycle detected at field "${node2}". Please check link traits for circular dependencies.`
|
|
670
|
+
);
|
|
671
|
+
}
|
|
672
|
+
if (visited.has(node2)) return;
|
|
673
|
+
visited.add(node2);
|
|
674
|
+
stack.add(node2);
|
|
675
|
+
const nexts = adjacency.get(node2);
|
|
676
|
+
if (nexts) {
|
|
677
|
+
for (const to of nexts) {
|
|
678
|
+
dfs(to);
|
|
679
|
+
}
|
|
680
|
+
}
|
|
681
|
+
stack.delete(node2);
|
|
682
|
+
};
|
|
683
|
+
for (const node2 of adjacency.keys()) {
|
|
684
|
+
if (!visited.has(node2)) {
|
|
685
|
+
dfs(node2);
|
|
686
|
+
}
|
|
687
|
+
}
|
|
688
|
+
};
|
|
689
|
+
var build = (stateSchema, spec) => {
|
|
690
|
+
const entries = normalizeSpec(spec);
|
|
691
|
+
const nodeMetaByFieldPath = collectNodeMeta(spec);
|
|
692
|
+
for (const entry of entries) {
|
|
693
|
+
if (entry.kind === "computed") {
|
|
694
|
+
const deps = entry.meta.deps;
|
|
695
|
+
if (deps === void 0) {
|
|
696
|
+
throw new Error(
|
|
697
|
+
`[StateTrait.build] Missing explicit deps for computed "${entry.fieldPath}". Please use StateTrait.computed({ deps: [...], get: ... }).`
|
|
698
|
+
);
|
|
699
|
+
}
|
|
700
|
+
}
|
|
701
|
+
if (entry.kind === "source") {
|
|
702
|
+
const deps = entry.meta.deps;
|
|
703
|
+
if (deps === void 0) {
|
|
704
|
+
throw new Error(
|
|
705
|
+
`[StateTrait.build] Missing explicit deps for source "${entry.fieldPath}". Please provide meta.deps for StateTrait.source({ deps: [...], ... }).`
|
|
706
|
+
);
|
|
707
|
+
}
|
|
708
|
+
}
|
|
709
|
+
if (entry.kind === "check") {
|
|
710
|
+
const rules = entry.meta?.rules ?? {};
|
|
711
|
+
for (const name of Object.keys(rules)) {
|
|
712
|
+
const rule = rules[name];
|
|
713
|
+
if (typeof rule === "function" || !rule || typeof rule !== "object") {
|
|
714
|
+
throw new Error(
|
|
715
|
+
`[StateTrait.build] Missing explicit deps for check "${entry.fieldPath}" rule "${name}". Please use { deps: [...], validate: ... } form.`
|
|
716
|
+
);
|
|
717
|
+
}
|
|
718
|
+
if (rule.deps === void 0) {
|
|
719
|
+
throw new Error(
|
|
720
|
+
`[StateTrait.build] Missing explicit deps for check "${entry.fieldPath}" rule "${name}". Please provide deps: [...].`
|
|
721
|
+
);
|
|
722
|
+
}
|
|
723
|
+
}
|
|
724
|
+
}
|
|
725
|
+
}
|
|
726
|
+
const { graph, plan } = buildGraph(entries, nodeMetaByFieldPath);
|
|
727
|
+
assertNoLinkCycles(graph.edges);
|
|
728
|
+
return {
|
|
729
|
+
stateSchema,
|
|
730
|
+
spec,
|
|
731
|
+
entries,
|
|
732
|
+
graph,
|
|
733
|
+
plan,
|
|
734
|
+
convergeIr: buildConvergeIr(entries)
|
|
735
|
+
};
|
|
736
|
+
};
|
|
737
|
+
|
|
738
|
+
// ../logix-core/src/internal/state-trait/install.ts
|
|
739
|
+
import { Effect } from "../effect.js";
|
|
740
|
+
var buildEntryIndex = (entries) => {
|
|
741
|
+
const index = /* @__PURE__ */ new Map();
|
|
742
|
+
for (const entry of entries) {
|
|
743
|
+
const list2 = index.get(entry.fieldPath) ?? [];
|
|
744
|
+
list2.push(entry);
|
|
745
|
+
index.set(entry.fieldPath, list2);
|
|
746
|
+
}
|
|
747
|
+
return index;
|
|
748
|
+
};
|
|
749
|
+
var install = (bound, program) => {
|
|
750
|
+
try {
|
|
751
|
+
const internals = getBoundInternals(bound);
|
|
752
|
+
internals.traits.registerStateTraitProgram(program, {
|
|
753
|
+
bumpReason: "logic_installed"
|
|
754
|
+
});
|
|
755
|
+
} catch {
|
|
756
|
+
}
|
|
757
|
+
const entryIndex = buildEntryIndex(program.entries);
|
|
758
|
+
const installStep = (step) => {
|
|
759
|
+
if (!step.targetFieldPath) {
|
|
760
|
+
return Effect.void;
|
|
761
|
+
}
|
|
762
|
+
if (step.kind !== "source-refresh") {
|
|
763
|
+
return Effect.void;
|
|
764
|
+
}
|
|
765
|
+
const candidates = entryIndex.get(step.targetFieldPath);
|
|
766
|
+
if (!candidates || candidates.length === 0) {
|
|
767
|
+
return Effect.void;
|
|
768
|
+
}
|
|
769
|
+
const entry = candidates.find((e) => e.kind === "source");
|
|
770
|
+
if (!entry) {
|
|
771
|
+
return Effect.void;
|
|
772
|
+
}
|
|
773
|
+
return installSourceRefresh(
|
|
774
|
+
bound,
|
|
775
|
+
step,
|
|
776
|
+
entry
|
|
777
|
+
);
|
|
778
|
+
};
|
|
779
|
+
return Effect.forEach(program.plan.steps, (step) => installStep(step)).pipe(Effect.asVoid);
|
|
780
|
+
};
|
|
781
|
+
|
|
782
|
+
// ../logix-core/src/StateTrait.ts
|
|
783
|
+
var $root = "$root";
|
|
784
|
+
var from = (_schema) => (spec) => spec;
|
|
785
|
+
var node = (spec) => ({
|
|
786
|
+
_tag: "StateTraitNode",
|
|
787
|
+
...spec
|
|
788
|
+
});
|
|
789
|
+
var list = (spec) => ({
|
|
790
|
+
_tag: "StateTraitList",
|
|
791
|
+
...spec
|
|
792
|
+
});
|
|
793
|
+
var computed = (input) => {
|
|
794
|
+
const derive = (state) => {
|
|
795
|
+
const args = input.deps.map(
|
|
796
|
+
(dep) => getAtPath(state, dep)
|
|
797
|
+
);
|
|
798
|
+
return input.get(...args);
|
|
799
|
+
};
|
|
800
|
+
return {
|
|
801
|
+
fieldPath: void 0,
|
|
802
|
+
kind: "computed",
|
|
803
|
+
meta: { deps: input.deps, derive, equals: input.equals }
|
|
804
|
+
};
|
|
805
|
+
};
|
|
806
|
+
var source = (meta) => ({
|
|
807
|
+
fieldPath: void 0,
|
|
808
|
+
kind: "source",
|
|
809
|
+
meta
|
|
810
|
+
});
|
|
811
|
+
var link = (meta) => ({
|
|
812
|
+
fieldPath: void 0,
|
|
813
|
+
kind: "link",
|
|
814
|
+
meta
|
|
815
|
+
});
|
|
816
|
+
var build2 = (stateSchema, spec) => build(stateSchema, spec);
|
|
817
|
+
var install2 = (bound, program) => install(bound, program);
|
|
818
|
+
var exportStaticIr2 = (program, moduleId, options) => exportStaticIr({
|
|
819
|
+
program,
|
|
820
|
+
moduleId,
|
|
821
|
+
version: options?.version
|
|
822
|
+
});
|
|
823
|
+
|
|
824
|
+
export {
|
|
825
|
+
$root,
|
|
826
|
+
from,
|
|
827
|
+
node,
|
|
828
|
+
list,
|
|
829
|
+
computed,
|
|
830
|
+
source,
|
|
831
|
+
link,
|
|
832
|
+
build2 as build,
|
|
833
|
+
install2 as install,
|
|
834
|
+
exportStaticIr2 as exportStaticIr
|
|
835
|
+
};
|
|
836
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../logix-core/src/internal/state-trait/meta.ts", "../../../../logix-core/src/internal/state-trait/model.ts", "../../../../logix-core/src/internal/state-trait/build.ts", "../../../../logix-core/src/internal/state-trait/install.ts", "../../../../logix-core/src/StateTrait.ts"],
  "sourcesContent": ["export type TraitMeta = Readonly<{\n  readonly label?: string\n  readonly description?: string\n  readonly tags?: ReadonlyArray<string>\n  readonly group?: string\n  readonly docsUrl?: string\n  readonly cacheGroup?: string\n  /**\n   * \u6269\u5C55\u6CE8\u89E3\u4F4D\uFF1A\u4EC5\u63A5\u53D7 `x-*` key\uFF08\u7528\u4E8E Phantom Source / Drift Detection \u7B49\u80FD\u529B\u7684\u951A\u70B9\uFF09\u3002\n   *\n   * \u7EA6\u675F\uFF1A\n   * - \u5FC5\u987B\u53EF\u5E8F\u5217\u5316\uFF08JSON \u503C\uFF09\uFF1B\n   * - \u5FC5\u987B Slim\uFF08\u5EFA\u8BAE\u53EA\u653E\u5C11\u91CF\u5173\u952E\u63D0\u793A\uFF0C\u4E0D\u8981\u585E\u5927\u5BF9\u8C61\uFF09\u3002\n   */\n  readonly annotations?: Readonly<Record<string, JsonValue>>\n  /**\n   * \u8BED\u4E49\u63D0\u793A\uFF1A\u5F53\u540C\u4E00\u5B9E\u4F53\uFF08\u4F8B\u5982\u540C\u4E00 resourceId\uFF09\u51FA\u73B0\u591A\u5904 meta \u58F0\u660E\u4E14\u4E0D\u4E00\u81F4\u65F6\uFF0C\n   * canonical=true \u7684\u58F0\u660E\u4F18\u5148\u6210\u4E3A\u201C\u5C55\u793A\u7528 canonical\u201D\uFF1B\u5176\u4ED6\u58F0\u660E\u4F1A\u8BB0\u5F55\u4E3A conflicts\u3002\n   */\n  readonly canonical?: boolean\n}>\n\nexport type TraitMetaConflict = Readonly<{\n  readonly origin: string\n  readonly meta: TraitMeta\n}>\n\nexport type JsonPrimitive = string | number | boolean | null\nexport type JsonValue =\n  | JsonPrimitive\n  | ReadonlyArray<JsonValue>\n  | { readonly [key: string]: JsonValue }\n\nconst uniqSortedStrings = (input: ReadonlyArray<string>): ReadonlyArray<string> => {\n  const set = new Set<string>()\n  for (const item of input) {\n    const v = item.trim()\n    if (!v) continue\n    set.add(v)\n  }\n  return Array.from(set).sort((a, b) => a.localeCompare(b))\n}\n\nconst isPlainRecord = (value: unknown): value is Record<string, unknown> =>\n  typeof value === \"object\" && value !== null && !Array.isArray(value)\n\nconst sanitizeJsonValue = (\n  input: unknown,\n  depth: number,\n): JsonValue | undefined => {\n  if (input === null) return null\n\n  if (typeof input === \"string\") return input\n  if (typeof input === \"boolean\") return input\n  if (typeof input === \"number\") {\n    return Number.isFinite(input) ? input : undefined\n  }\n\n  if (depth >= 6) return undefined\n\n  if (Array.isArray(input)) {\n    const out: Array<JsonValue> = []\n    for (const item of input) {\n      const v = sanitizeJsonValue(item, depth + 1)\n      if (v !== undefined) out.push(v)\n    }\n    return out\n  }\n\n  if (isPlainRecord(input)) {\n    const keys = Object.keys(input).sort()\n    const out: Record<string, JsonValue> = {}\n    for (const key of keys) {\n      const v = sanitizeJsonValue(input[key], depth + 1)\n      if (v !== undefined) out[key] = v\n    }\n    return out\n  }\n\n  return undefined\n}\n\nexport const sanitize = (input: unknown): TraitMeta | undefined => {\n  if (input === null || input === undefined) return undefined\n  if (typeof input !== \"object\" || Array.isArray(input)) return undefined\n\n  const record = input as Record<string, unknown>\n  const out: Record<string, unknown> = {}\n\n  const pickString = (key: keyof TraitMeta): void => {\n    const value = record[key as string]\n    if (typeof value !== \"string\") return\n    const trimmed = value.trim()\n    if (!trimmed) return\n    out[key] = trimmed\n  }\n\n  pickString(\"label\")\n  pickString(\"description\")\n  pickString(\"group\")\n  pickString(\"docsUrl\")\n  pickString(\"cacheGroup\")\n\n  const canonical = record.canonical\n  if (typeof canonical === \"boolean\") {\n    out.canonical = canonical\n  }\n\n  const tagsRaw = record.tags\n  if (typeof tagsRaw === \"string\") {\n    const tags = uniqSortedStrings([tagsRaw])\n    if (tags.length > 0) out.tags = tags\n  } else if (Array.isArray(tagsRaw)) {\n    const tags = uniqSortedStrings(tagsRaw.filter((x): x is string => typeof x === \"string\"))\n    if (tags.length > 0) out.tags = tags\n  }\n\n  const annotations: Record<string, JsonValue> = {}\n\n  // legacy: allow passing `x-*` keys at the top-level (untyped input), then normalize into `annotations`.\n  const annotationKeys = Object.keys(record)\n    .filter((k) => k.startsWith(\"x-\"))\n    .sort((a, b) => a.localeCompare(b))\n  for (const key of annotationKeys) {\n    const v = sanitizeJsonValue(record[key], 0)\n    if (v !== undefined) annotations[key] = v\n  }\n\n  // preferred: typed `annotations` field (only accepts `x-*` keys).\n  const annotationsRaw = record.annotations\n  if (isPlainRecord(annotationsRaw)) {\n    const keys = Object.keys(annotationsRaw)\n      .filter((k) => k.startsWith(\"x-\"))\n      .sort((a, b) => a.localeCompare(b))\n    for (const key of keys) {\n      const v = sanitizeJsonValue(annotationsRaw[key], 0)\n      if (v !== undefined) annotations[key] = v\n    }\n  }\n\n  if (Object.keys(annotations).length > 0) out.annotations = annotations\n\n  return Object.keys(out).length > 0 ? (out as TraitMeta) : undefined\n}\n\nconst stableStringify = (meta: TraitMeta): string => {\n  const out: Record<string, unknown> = {}\n  const keys = Object.keys(meta).sort()\n  for (const k of keys) {\n    out[k] = (meta as any)[k]\n  }\n  return JSON.stringify(out)\n}\n\nexport const equals = (a: TraitMeta | undefined, b: TraitMeta | undefined): boolean => {\n  if (a === b) return true\n  if (!a || !b) return false\n  return stableStringify(a) === stableStringify(b)\n}\n\nexport type CanonicalMergeInput = Readonly<{\n  readonly meta?: TraitMeta\n  readonly origin?: string\n  readonly conflicts?: ReadonlyArray<TraitMetaConflict>\n}>\n\nexport type CanonicalMergeResult = Readonly<{\n  readonly meta?: TraitMeta\n  readonly origin?: string\n  readonly conflicts?: ReadonlyArray<TraitMetaConflict>\n}>\n\nexport const mergeCanonical = (\n  current: CanonicalMergeInput,\n  incoming: Readonly<{ readonly origin: string; readonly meta: TraitMeta }>,\n): CanonicalMergeResult => {\n  const existing = current.meta\n  if (!existing) {\n    return {\n      meta: incoming.meta,\n      origin: incoming.origin,\n      conflicts: current.conflicts,\n    }\n  }\n\n  if (equals(existing, incoming.meta)) {\n    return current\n  }\n\n  const conflicts: Array<TraitMetaConflict> = [\n    ...((current.conflicts ?? []) as ReadonlyArray<TraitMetaConflict>),\n  ]\n\n  const push = (conflict: TraitMetaConflict): void => {\n    const exists = conflicts.some(\n      (c) => c.origin === conflict.origin && equals(c.meta, conflict.meta),\n    )\n    if (!exists) conflicts.push(conflict)\n  }\n\n  const existingCanonical = existing.canonical === true\n  const incomingCanonical = incoming.meta.canonical === true\n\n  if (!existingCanonical && incomingCanonical) {\n    push({\n      origin: current.origin ?? \"unknown\",\n      meta: existing,\n    })\n    return {\n      meta: incoming.meta,\n      origin: incoming.origin,\n      conflicts,\n    }\n  }\n\n  push({\n    origin: incoming.origin,\n    meta: incoming.meta,\n  })\n\n  return {\n    meta: existing,\n    origin: current.origin ?? incoming.origin,\n    conflicts,\n  }\n}\n", "import type { Schema } from \"effect\"\nimport type { StateAtPath, StateFieldPath } from \"./field-path.js\"\nimport type { DirtyAllReason } from \"../field-path.js\"\nimport * as Meta from \"./meta.js\"\nimport type { ConvergeStaticIrRegistry } from \"./converge-ir.js\"\n\n// StateTrait core model\u3002\n// \u5177\u4F53\u5B57\u6BB5\u8BED\u4E49\u4E0E data-model.md / references/state-trait-core.md \u4FDD\u6301\u5BF9\u9F50\u3002\n\n/**\n * StateTraitSpec<S>\uFF1A\n * - Module \u56FE\u7EB8\u4E2D traits \u69FD\u4F4D\u7684\u6807\u51C6\u5F62\u6001\uFF1B\n * - key \u53D7 StateFieldPath<S> \u7EA6\u675F\uFF0Cvalue \u4E3A\u5BF9\u5E94 scope \u7684\u58F0\u660E\uFF08Entry / Node / List\uFF09\u3002\n */\nexport type StateTraitSpec<S> = S extends object\n  ? {\n      [Path in StateFieldPath<S> | \"$root\"]?: StateTraitSpecValue<S, Path>\n    }\n  : never\n\nexport type StateTraitKind = \"computed\" | \"source\" | \"link\" | \"check\"\n\nexport interface ComputedMeta<S, P> {\n  /**\n   * \u663E\u5F0F\u4F9D\u8D56\u5B57\u6BB5\u8DEF\u5F84\u96C6\u5408\uFF08\u5FC5\u987B\uFF09\uFF1A\n   *\n   * - deps \u662F\u552F\u4E00\u4F9D\u8D56\u4E8B\u5B9E\u6E90\uFF1AGraph/ReverseClosure/\u589E\u91CF\u8C03\u5EA6/\u6027\u80FD\u4F18\u5316\u90FD\u53EA\u8BA4 deps\uFF1B\n   * - DSL \u4FA7 `StateTrait.computed({ deps, get })` \u7684 `get` \u91C7\u7528 deps-as-args\uFF0C\u4E0D\u66B4\u9732 `(state) => ...`\uFF1B\n   * - build \u9636\u6BB5\u4F1A\u5C06 `get(...depsValues)` \u964D\u89E3\u4E3A `derive(state)` \u4F9B\u8FD0\u884C\u65F6\u6267\u884C\uFF0C\u4F46\u4F9D\u8D56\u8BFB\u96C6\u4ECD\u4EE5 deps \u4E3A\u51C6\uFF1B\n   * - dev-mode \u4E0B\u82E5 `derive(state)` \u7684\u5B9E\u9645\u8BFB\u53D6\u4E0E deps \u4E0D\u4E00\u81F4\uFF0C\u4F1A\u4EA7\u751F `deps_mismatch` \u8BCA\u65AD\u544A\u8B66\u3002\n   *\n   * - \u5BF9\u4E8E Root \u89C4\u5219\uFF1Adeps \u4E3A StateFieldPath<State>\uFF1B\n   * - \u5BF9\u4E8E list.item scope\uFF1Adeps \u4E3A StateFieldPath<Item>\uFF08\u76F8\u5BF9\u8DEF\u5F84\uFF0Cbuild \u9636\u6BB5\u4F1A\u524D\u7F00\u5316\uFF09\u3002\n   */\n  readonly deps: ReadonlyArray<StateFieldPath<S>>\n  readonly derive: (state: Readonly<S>) => StateAtPath<S, P>\n  /**\n   * \u53EF\u9009\uFF1A\u7B49\u4EF7\u5224\u5B9A\uFF08\u7528\u4E8E\u8DF3\u8FC7\u65E0\u53D8\u5316\u5199\u56DE\uFF09\u3002\n   */\n  readonly equals?: (prev: StateAtPath<S, P>, next: StateAtPath<S, P>) => boolean\n}\n\nexport interface SourceMeta<S, P> {\n  readonly deps: ReadonlyArray<StateFieldPath<S>>\n  /**\n   * \u903B\u8F91\u8D44\u6E90\u6807\u8BC6\uFF08\u5982 \"user/profile\"\uFF09\u3002\n   *\n   * - StateTrait.source DSL \u4E2D\u7684 `resource` \u5B57\u6BB5\u5728 build \u9636\u6BB5\u4F1A\u843D\u5728\u6B64\u5904\uFF1B\n   * - data-model.md \u4E2D\u8BB0\u4E3A resourceId\uFF0C\u8FD9\u91CC\u6CBF\u7528 DSL \u547D\u540D\u907F\u514D\u6DF7\u6DC6\u3002\n   */\n  readonly resource: string\n  /**\n   * \u4ECE\u5B8C\u6574 State \u8BA1\u7B97\u8BBF\u95EE\u8BE5\u8D44\u6E90\u6240\u9700 key \u7684\u89C4\u5219\u3002\n   *\n   * - \u5141\u8BB8\u8FD4\u56DE undefined\uFF1A\u8868\u793A\u8BE5\u8D44\u6E90\u5728\u5F53\u524D state \u4E0B\u672A\u6FC0\u6D3B\uFF08\u5E94\u56DE\u6536\u4E3A idle\uFF09\u3002\n   */\n  readonly key: (state: Readonly<S>) => unknown\n  readonly triggers?: ReadonlyArray<\"onMount\" | \"onValueChange\" | \"manual\">\n  readonly debounceMs?: number\n  readonly concurrency?: \"switch\" | \"exhaust-trailing\"\n  /**\n   * \u7528\u4E8E Devtools/\u6587\u6863\u7684\u53EF\u5E8F\u5217\u5316\u5143\u4FE1\u606F\uFF08\u767D\u540D\u5355\u5B57\u6BB5\uFF09\u3002\n   */\n  readonly meta?: Meta.TraitMeta\n  /**\n   * \u9884\u7559\uFF1A\u5728 build \u9636\u6BB5\u53EF\u586B\u5145\u8BE5 Trait \u6240\u5728\u5B57\u6BB5\u8DEF\u5F84\uFF0C\u4FBF\u4E8E\u8C03\u8BD5\u3002\n   */\n  readonly _fieldPath?: P\n}\n\nexport interface LinkMeta<S> {\n  /**\n   * \u6E90\u5B57\u6BB5\u8DEF\u5F84\uFF0C\u540C\u6837\u53D7 StateFieldPath<S> \u7EA6\u675F\u3002\n   */\n  readonly from: StateFieldPath<S>\n}\n\nexport type CheckRule<Input = unknown, Ctx = unknown> =\n  {\n    readonly deps: ReadonlyArray<string>\n    /**\n     * validateOn\uFF1A\n     * - \u4EC5\u5F71\u54CD\u81EA\u52A8\u6821\u9A8C\u9636\u6BB5\uFF08onChange/onBlur\uFF09\uFF1Bsubmit/manual \u59CB\u7EC8\u6267\u884C\uFF1B\n     * - \u4E3A\u7A7A\u6570\u7EC4\u8868\u793A\u7981\u7528\u81EA\u52A8\u6821\u9A8C\uFF08\u4EC5 submit/manual \u6267\u884C\uFF09\u3002\n     */\n    readonly validateOn?: ReadonlyArray<\"onChange\" | \"onBlur\">\n    readonly validate: (input: Input, ctx: Ctx) => unknown\n    readonly meta?: Meta.TraitMeta\n  }\n\nexport interface CheckMeta<Input = unknown, Ctx = unknown> {\n  /**\n   * \u547D\u540D\u89C4\u5219\u96C6\u5408\uFF08\u8BB0\u5F55\u7528\u4E8E\u786E\u5B9A\u6027\u5408\u5E76\u4E0E\u8BCA\u65AD\u5C55\u793A\uFF09\u3002\n   */\n  readonly rules: Readonly<Record<string, CheckRule<Input, Ctx>>>\n  /**\n   * \u9519\u8BEF\u6811\u5199\u56DE\uFF08Phase 2\uFF1A\u4EC5\u56FA\u5316\u7ED3\u6784\uFF1B\u5177\u4F53\u5199\u56DE\u8BED\u4E49\u5728\u540E\u7EED Phase \u5B9E\u73B0\uFF09\u3002\n   */\n  readonly writeback?: {\n    readonly kind: \"errors\"\n    readonly path?: string\n  }\n}\n\n/**\n * StateTraitEntry<S, P>\uFF1A\n * - \u8868\u793A\u67D0\u4E2A\u5B57\u6BB5\u8DEF\u5F84 P \u4E0A\u7684\u4E00\u9879 Trait \u914D\u7F6E\uFF1B\n * - kind \u4E0E meta \u6210\u5BF9\u51FA\u73B0\uFF0C\u7528\u4E8E\u5728 build/install \u9636\u6BB5\u5206\u6D3E\u884C\u4E3A\u3002\n */\nexport type StateTraitEntry<\n  S = unknown,\n  P extends string = StateFieldPath<S>\n> =\n  | {\n      readonly fieldPath: P\n      readonly kind: \"computed\"\n      readonly meta: ComputedMeta<S, P>\n    }\n  | {\n      readonly fieldPath: P\n      readonly kind: \"source\"\n      readonly meta: SourceMeta<S, P>\n    }\n  | {\n      readonly fieldPath: P\n      readonly kind: \"link\"\n      readonly meta: LinkMeta<S>\n    }\n  | {\n      readonly fieldPath: P\n      readonly kind: \"check\"\n      readonly meta: CheckMeta<unknown, unknown>\n    }\n\nexport interface StateTraitNode<Input = unknown, Ctx = unknown> {\n  readonly _tag: \"StateTraitNode\"\n  readonly computed?:\n    | StateTraitEntry<Input, any>\n    | Readonly<Record<string, StateTraitEntry<Input, any>>>\n  readonly source?:\n    | StateTraitEntry<Input, any>\n    | Readonly<Record<string, StateTraitEntry<Input, any>>>\n  readonly link?:\n    | StateTraitEntry<Input, any>\n    | Readonly<Record<string, StateTraitEntry<Input, any>>>\n  readonly check?: Readonly<Record<string, CheckRule<Input, Ctx>>>\n  readonly meta?: Meta.TraitMeta\n}\n\nexport interface StateTraitList<Item = unknown> {\n  readonly _tag: \"StateTraitList\"\n  readonly item?: StateTraitNode<Item, any>\n  readonly list?: StateTraitNode<ReadonlyArray<Item>, any>\n  readonly identityHint?: {\n    readonly trackBy?: string\n  }\n}\n\nexport type StateTraitSpecValue<S, P extends string> =\n  | StateTraitEntry<S, P>\n  | StateTraitNode<any, any>\n  | StateTraitList<any>\n\n/**\n * StateTraitField\uFF1A\n * - \u8868\u793A State \u4E2D\u7684\u4E00\u4E2A\u5B57\u6BB5\u8282\u70B9\uFF08\u4E0D\u8BBA\u662F\u5426\u6302 Trait\uFF09\uFF1B\n * - \u7531 build \u9636\u6BB5\u4ECE StateTraitSpec \u5F52\u4E00\u5316\u800C\u6765\u3002\n */\nexport interface StateTraitField {\n  readonly id: string\n  readonly path: string\n  readonly displayName?: string\n  readonly valueType?: string\n  readonly traits: ReadonlyArray<StateTraitFieldTrait>\n}\n\n/**\n * StateTraitFieldTrait\uFF1A\n * - \u67D0\u4E2A\u5B57\u6BB5\u4E0A\u7684\u4E00\u9879 Trait\uFF08computed / source / link\uFF09\u7684\u7ED3\u6784\u5316\u63CF\u8FF0\uFF1B\n * - meta \u4E0E StateTraitEntry.meta \u5BF9\u9F50\uFF0Cdeps \u8868\u793A\u8BE5 Trait \u6240\u4F9D\u8D56\u7684\u5B57\u6BB5\u8DEF\u5F84\u96C6\u5408\u3002\n */\nexport interface StateTraitFieldTrait {\n  readonly fieldId: string\n  readonly kind: StateTraitKind\n  readonly meta:\n    | ComputedMeta<unknown, string>\n    | SourceMeta<unknown, string>\n    | LinkMeta<unknown>\n    | CheckMeta<unknown, unknown>\n  readonly deps: ReadonlyArray<string>\n}\n\n/**\n * StateTraitResource\uFF1A\n * - \u63CF\u8FF0 source \u578B Trait \u6240\u4F9D\u8D56\u7684\u903B\u8F91\u8D44\u6E90\u5143\u4FE1\u606F\uFF1B\n * - ResourceSpec \u7684\u5177\u4F53\u5B9E\u73B0\u7531 Resource \u547D\u540D\u7A7A\u95F4\u8D1F\u8D23\uFF0C\u8FD9\u91CC\u53EA\u4FDD\u7559 Trait \u89C6\u89D2\u7684\u4F9D\u8D56\u4FE1\u606F\u3002\n */\nexport interface StateTraitResource {\n  readonly resourceId: string\n  readonly keySelector: string\n  readonly keyExample?: unknown\n  readonly ownerFields: ReadonlyArray<string>\n  readonly meta?: Meta.TraitMeta\n  readonly metaOrigin?: string\n  readonly metaConflicts?: ReadonlyArray<Meta.TraitMetaConflict>\n}\n\n/**\n * Graph Node / Edge\uFF1A\n * - \u8282\u70B9\u901A\u5E38\u5BF9\u5E94\u4E00\u4E2A\u5B57\u6BB5\uFF1B\n * - \u8FB9\u8868\u793A\u5B57\u6BB5\u4E4B\u95F4\u6216\u5B57\u6BB5\u4E0E\u8D44\u6E90\u4E4B\u95F4\u7684\u4F9D\u8D56\u5173\u7CFB\u3002\n */\nexport interface StateTraitGraphNode {\n  readonly id: string\n  readonly field: StateTraitField\n  readonly traits: ReadonlyArray<StateTraitFieldTrait>\n  readonly meta?: Meta.TraitMeta\n}\n\nexport interface StateTraitGraphEdge {\n  readonly id: string\n  readonly from: string\n  readonly to: string\n  readonly kind: \"computed\" | \"link\" | \"source-dep\" | \"check-dep\"\n}\n\n/**\n * StateTraitGraph\uFF1A\n * - StateTrait \u5F15\u64CE\u7684\u7ED3\u6784\u89C6\u56FE\uFF0C\u4E00\u5F20\u5305\u542B\u8282\u70B9\u4E0E\u4F9D\u8D56\u8FB9\u7684\u56FE\uFF1B\n * - \u4F9B Devtools / Studio \u4E0E Runtime \u505A\u7ED3\u6784\u5206\u6790\u4E0E\u53EF\u89C6\u5316\u4F7F\u7528\u3002\n */\nexport interface StateTraitGraph {\n  readonly _tag: \"StateTraitGraph\"\n  readonly nodes: ReadonlyArray<StateTraitGraphNode>\n  readonly edges: ReadonlyArray<StateTraitGraphEdge>\n  readonly resources: ReadonlyArray<StateTraitResource>\n  readonly meta?: {\n    readonly moduleId?: string\n    readonly version?: string\n  }\n}\n\n/**\n * StateTraitPlanStep\uFF1A\n * - StateTrait.install / Runtime \u6267\u884C\u65F6\u7684\u6700\u5C0F\u6307\u4EE4\u5355\u5143\uFF1B\n * - \u7531 Graph \u63A8\u5BFC\u800C\u6765\uFF0C\u7528\u4E8E\u5728\u8FD0\u884C\u65F6\u5B89\u88C5 watcher \u6216\u89E6\u53D1\u5916\u90E8\u8C03\u7528\u3002\n */\nexport interface StateTraitPlanStep {\n  readonly id: string\n  readonly kind: \"computed-update\" | \"link-propagate\" | \"source-refresh\" | \"check-validate\"\n  readonly targetFieldPath?: string\n  readonly sourceFieldPaths?: ReadonlyArray<string>\n  readonly resourceId?: string\n  readonly keySelectorId?: string\n  readonly debugInfo?: {\n    readonly graphNodeId?: string\n    readonly graphEdgeId?: string\n  }\n}\n\n/**\n * StateTraitPlan\uFF1A\n * - \u6C47\u603B\u67D0\u6A21\u5757\u6240\u6709 Trait \u884C\u4E3A\u7684\u6267\u884C\u8BA1\u5212\uFF1B\n * - install \u9636\u6BB5\u6309 Plan \u5C06\u884C\u4E3A\u6302\u8F7D\u5230 Bound API / EffectOp \u7BA1\u9053\u4E0A\u3002\n */\nexport interface StateTraitPlan {\n  readonly _tag: \"StateTraitPlan\"\n  readonly moduleId?: string\n  readonly steps: ReadonlyArray<StateTraitPlanStep>\n  readonly meta?: Record<string, unknown>\n}\n\n/**\n * StateTraitProgram<S>\uFF1A\n * - StateTrait \u5F15\u64CE\u7684 Program \u8F93\u51FA\uFF0C\u662F Runtime / Devtools \u7684\u7EDF\u4E00\u5165\u53E3\uFF1B\n * - stateSchema \u4E0E spec \u4FDD\u7559\u539F\u59CB\u8F93\u5165\uFF0Cgraph/plan \u4E3A\u5185\u90E8 IR\u3002\n */\nexport interface StateTraitProgram<S> {\n  readonly stateSchema: Schema.Schema<S, any>\n  readonly spec: StateTraitSpec<S>\n  /**\n   * Program.entries\uFF1A\n   * - build \u9636\u6BB5\u4ECE spec\uFF08\u542B node/list/$root\uFF09\u5F52\u4E00\u5316\u5F97\u5230\u7684\u6700\u5C0F\u89C4\u5219\u96C6\u5408\uFF1B\n   * - Phase 2 \u5148\u4FDD\u8BC1\u201C\u53EF\u8BFB\u53EF\u8BCA\u65AD\u7684\u7ED3\u6784\u201D\uFF0C\u540E\u7EED Phase \u624D\u63A5\u5165\u6267\u884C\u4E0E\u6536\u655B\u8BED\u4E49\u3002\n   */\n  readonly entries: ReadonlyArray<StateTraitEntry<any, string>>\n  readonly graph: StateTraitGraph\n  readonly plan: StateTraitPlan\n  readonly convergeIr?: ConvergeStaticIrRegistry\n}\n\n/**\n * \u4ECE StateTraitSpec \u6784\u5EFA\u5F52\u4E00\u5316\u7684 entry \u5217\u8868\u3002\n *\n * - Phase 2\uFF1A\u652F\u6301 node/list/$root \u7684\u7ED3\u6784\u5C55\u5F00\uFF0C\u5E76\u5728\u5FC5\u8981\u65F6\u8865\u5168/\u524D\u7F00\u5316 fieldPath \u4E0E deps\uFF1B\n * - \u540E\u7EED Phase \u53EF\u5728\u6B64\u5904\u6269\u5C55\u6821\u9A8C\u903B\u8F91\uFF08\u4F8B\u5982\u68C0\u6D4B\u91CD\u590D\u5B9A\u4E49\u3001\u8986\u76D6\u673A\u5236\u7B49\uFF09\u3002\n */\nexport const normalizeSpec = <S>(\n  spec: StateTraitSpec<S>,\n): ReadonlyArray<StateTraitEntry<any, string>> => {\n  const entries: Array<StateTraitEntry<any, string>> = []\n\n  const isNode = (value: unknown): value is StateTraitNode<any, any> =>\n    typeof value === \"object\" && value !== null && (value as any)._tag === \"StateTraitNode\"\n\n  const isList = (value: unknown): value is StateTraitList<any> =>\n    typeof value === \"object\" && value !== null && (value as any)._tag === \"StateTraitList\"\n\n  const joinPath = (prefix: string, suffix: string): string => {\n    if (!prefix) return suffix\n    if (!suffix) return prefix\n    return `${prefix}.${suffix}`\n  }\n\n  const prefixDeps = (\n    deps: ReadonlyArray<string> | undefined,\n    prefix: string,\n  ): ReadonlyArray<string> => {\n    if (!deps || deps.length === 0) return []\n    return deps.map((d) => (prefix ? joinPath(prefix, d) : d))\n  }\n\n  const normalizeEntry = (\n    entry: StateTraitEntry<any, string>,\n    fieldPath: string,\n    depPrefix: string,\n  ): StateTraitEntry<any, string> => {\n    if (entry.kind === \"computed\") {\n      const meta = entry.meta as any\n      const rawDeps = meta.deps as ReadonlyArray<string> | undefined\n      const deps =\n        rawDeps !== undefined\n          ? prefixDeps(rawDeps, depPrefix)\n          : undefined\n      return {\n        ...(entry as any),\n        fieldPath,\n        meta: { ...meta, deps },\n      }\n    }\n    if (entry.kind === \"source\") {\n      const meta = entry.meta as any\n      const rawDeps = meta.deps as ReadonlyArray<string> | undefined\n      const deps =\n        rawDeps !== undefined\n          ? prefixDeps(rawDeps, depPrefix)\n          : undefined\n      return {\n        ...(entry as any),\n        fieldPath,\n        meta: { ...meta, deps, _fieldPath: fieldPath },\n      }\n    }\n    if (entry.kind === \"link\") {\n      const meta = entry.meta as any\n      const from = prefixDeps([meta.from as string], depPrefix)[0] ?? meta.from\n      return {\n        ...(entry as any),\n        fieldPath,\n        meta: { ...meta, from },\n      }\n    }\n    // check\uFF1APhase 2 \u4EC5\u8C03\u6574 fieldPath\uFF0Cdeps \u4ECD\u7531 meta.rules \u5728 build \u9636\u6BB5\u6C47\u603B\u3002\n    return {\n      ...(entry as any),\n      fieldPath,\n    }\n  }\n\n  const expandNode = (\n    scopeId: string,\n    joinPrefix: string,\n    node: StateTraitNode<any, any>,\n    options?: {\n      /**\n       * check deps \u524D\u7F00\uFF1A\u7528\u4E8E list-scope check \u7684\u9ED8\u8BA4\u8BED\u4E49\uFF08deps \u76F8\u5BF9\u201C\u884C\u5143\u7D20\u201D\u800C\u975E list \u6839\uFF09\u3002\n       *\n       * - item scope\uFF1AdepsPrefix = `${listPath}[]`\uFF08\u4E0E joinPrefix \u4E00\u81F4\uFF09\uFF1B\n       * - list scope\uFF1AfieldPath = `${listPath}`\uFF0C\u4F46 depsPrefix \u5E94\u4E3A `${listPath}[]`\u3002\n       */\n      readonly checkDepsPrefix?: string\n    },\n  ): void => {\n    const addEntry = (relativeTarget: string, raw: StateTraitEntry<any, string>): void => {\n      const rel = (raw as any).fieldPath ?? relativeTarget\n      const fieldPath = joinPrefix ? joinPath(joinPrefix, String(rel)) : String(rel)\n      entries.push(normalizeEntry(raw, fieldPath, joinPrefix))\n    }\n\n    const expandMaybeRecord = (\n      value:\n        | StateTraitEntry<any, any>\n        | Readonly<Record<string, StateTraitEntry<any, any>>>\n        | undefined,\n    ): void => {\n      if (!value) return\n      if (typeof (value as any).kind === \"string\") {\n        addEntry(\"\", value as any)\n        return\n      }\n      const record = value as Readonly<Record<string, StateTraitEntry<any, any>>>\n      for (const key in record) {\n        if (!Object.prototype.hasOwnProperty.call(record, key)) continue\n        const entry = record[key]\n        if (!entry) continue\n        addEntry(key, entry as any)\n      }\n    }\n\n    expandMaybeRecord(node.computed)\n    expandMaybeRecord(node.source)\n    expandMaybeRecord(node.link)\n\n    if (node.check) {\n      const rules: Record<string, CheckRule<any, any>> = {}\n      const checkDepsPrefix =\n        options?.checkDepsPrefix ?? (joinPrefix.endsWith(\"[]\") ? joinPrefix : \"\")\n\n      const prefixCheckDeps = (\n        deps: ReadonlyArray<string> | undefined,\n      ): ReadonlyArray<string> => {\n        if (!deps || deps.length === 0) return []\n        return deps.map((d) =>\n          d === \"\" ? scopeId : (checkDepsPrefix ? joinPath(checkDepsPrefix, d) : d),\n        )\n      }\n      for (const name of Object.keys(node.check)) {\n        const rule = (node.check as any)[name] as CheckRule<any, any>\n        if (typeof rule === \"function\") {\n          rules[name] = rule\n          continue\n        }\n        if (rule && typeof rule === \"object\") {\n          const meta = Meta.sanitize((rule as any).meta)\n          rules[name] = {\n            ...rule,\n            deps: prefixCheckDeps(rule.deps),\n            meta,\n          }\n          continue\n        }\n        // \u975E\u6CD5\u8F93\u5165\uFF1A\u5FFD\u7565\uFF08\u540E\u7EED Phase \u5C06\u5728 build \u9636\u6BB5\u63D0\u5347\u4E3A\u914D\u7F6E\u9519\u8BEF\uFF09\u3002\n      }\n\n      entries.push({\n        fieldPath: scopeId,\n        kind: \"check\",\n        meta: {\n          rules,\n          writeback: { kind: \"errors\" },\n        },\n      } as StateTraitEntry<any, any>)\n    }\n  }\n\n  for (const key in spec) {\n    if (!Object.prototype.hasOwnProperty.call(spec, key)) continue\n    const raw = spec[key as keyof typeof spec] as StateTraitSpecValue<S, any> | undefined\n    if (!raw) continue\n\n    if (isList(raw)) {\n      const listPath = key\n      if (raw.item) {\n        expandNode(`${listPath}[]`, `${listPath}[]`, raw.item)\n      }\n      if (raw.list) {\n        expandNode(listPath, listPath, raw.list, {\n          checkDepsPrefix: `${listPath}[]`,\n        })\n      }\n      continue\n    }\n\n    if (isNode(raw)) {\n      if (key === \"$root\") {\n        expandNode(\"$root\", \"\", raw)\n      } else {\n        expandNode(key, key, raw)\n      }\n      continue\n    }\n\n    const entry = raw as any as StateTraitEntry<any, string>\n    const fieldPath = (entry as any).fieldPath ?? key\n    entries.push(normalizeEntry(entry, String(fieldPath), \"\"))\n  }\n\n  return entries\n}\n\n/**\n * collectNodeMeta\uFF1A\n * - \u4ECE StateTraitSpec \u4E2D\u63D0\u53D6 StateTraitNode.meta\uFF08\u767D\u540D\u5355\u5B57\u6BB5\uFF09\uFF0C\u4F9B Devtools \u8FDB\u884C\u7ED3\u6784\u5C55\u793A\uFF1B\n * - meta \u4EC5\u7528\u4E8E\u8BCA\u65AD\u4E0E\u5C55\u793A\uFF0C\u4E0D\u53C2\u4E0E\u8FD0\u884C\u65F6\u8BED\u4E49\u3002\n */\nexport const collectNodeMeta = <S>(\n  spec: StateTraitSpec<S>,\n): ReadonlyMap<string, Meta.TraitMeta> => {\n  const out = new Map<string, Meta.TraitMeta>()\n\n  const isNode = (value: unknown): value is StateTraitNode<any, any> =>\n    typeof value === \"object\" && value !== null && (value as any)._tag === \"StateTraitNode\"\n\n  const isList = (value: unknown): value is StateTraitList<any> =>\n    typeof value === \"object\" && value !== null && (value as any)._tag === \"StateTraitList\"\n\n  const add = (scopeId: string, node: StateTraitNode<any, any>): void => {\n    const meta = Meta.sanitize(node.meta)\n    if (meta) out.set(scopeId, meta)\n  }\n\n  for (const key in spec) {\n    if (!Object.prototype.hasOwnProperty.call(spec, key)) continue\n    const raw = spec[key as keyof typeof spec] as StateTraitSpecValue<S, any> | undefined\n    if (!raw) continue\n\n    if (isList(raw)) {\n      const listPath = key\n      if (raw.item) add(`${listPath}[]`, raw.item)\n      if (raw.list) add(listPath, raw.list)\n      continue\n    }\n\n    if (isNode(raw)) {\n      if (key === \"$root\") add(\"$root\", raw)\n      else add(key, raw)\n      continue\n    }\n  }\n\n  return out\n}\n\n// ---- Converge (013) evidence model ----\n\nexport type TraitConvergeRequestedMode = \"auto\" | \"full\" | \"dirty\"\nexport type TraitConvergeExecutedMode = \"full\" | \"dirty\"\nexport type TraitConvergeOutcome = \"Converged\" | \"Noop\" | \"Degraded\"\n\nexport type TraitConvergeConfigScope =\n  | \"provider\"\n  | \"runtime_module\"\n  | \"runtime_default\"\n  | \"builtin\"\n\nexport type TraitConvergeReason =\n  | \"cold_start\"\n  | \"cache_hit\"\n  | \"cache_miss\"\n  | \"budget_cutoff\"\n  | \"near_full\"\n  | \"unknown_write\"\n  | \"dirty_all\"\n  | \"generation_bumped\"\n  | \"low_hit_rate_protection\"\n  | \"module_override\"\n\nexport interface TraitConvergeStepStats {\n  readonly totalSteps: number\n  readonly executedSteps: number\n  readonly skippedSteps: number\n  readonly changedSteps: number\n  readonly affectedSteps?: number\n}\n\nexport interface TraitConvergeDirtySummary {\n  readonly dirtyAll: boolean\n  readonly reason?: DirtyAllReason\n  readonly rootCount?: number\n  readonly rootIds?: ReadonlyArray<number>\n  readonly rootIdsTruncated?: boolean\n}\n\nexport interface TraitConvergePlanCacheEvidence {\n  readonly capacity: number\n  readonly size: number\n  readonly hits: number\n  readonly misses: number\n  readonly evicts: number\n  readonly hit: boolean\n  readonly keySize?: number\n  readonly missReason?: \"cold_start\" | \"generation_bumped\" | \"not_cached\" | \"unknown\"\n  readonly disabled?: boolean\n  readonly disableReason?: \"low_hit_rate\" | \"generation_thrash\" | \"manual_override\" | \"unknown\"\n}\n\nexport type TraitConvergeGenerationBumpReason =\n  | \"writers_changed\"\n  | \"deps_changed\"\n  | \"logic_installed\"\n  | \"logic_uninstalled\"\n  | \"imports_changed\"\n  | \"unknown\"\n\nexport interface TraitConvergeGenerationEvidence {\n  readonly generation: number\n  readonly generationBumpCount?: number\n  readonly lastBumpReason?: TraitConvergeGenerationBumpReason\n}\n\nexport interface TraitConvergeStaticIrEvidence {\n  readonly fieldPathCount: number\n  readonly stepCount: number\n  readonly buildDurationMs?: number\n}\n\nexport interface TraitConvergeHotspot {\n  readonly kind?: string\n  readonly stepId: number\n  readonly outFieldPathId?: number\n  readonly durationMs: number\n  readonly changed: boolean\n}\n\nexport interface TraitConvergeDecisionSummary {\n  readonly requestedMode: TraitConvergeRequestedMode\n  readonly executedMode: TraitConvergeExecutedMode\n  readonly outcome: TraitConvergeOutcome\n  readonly configScope: TraitConvergeConfigScope\n  readonly staticIrDigest: string\n  readonly executionBudgetMs: number\n  readonly executionDurationMs: number\n  readonly decisionBudgetMs?: number\n  readonly decisionDurationMs?: number\n  readonly reasons: ReadonlyArray<TraitConvergeReason>\n  readonly stepStats: TraitConvergeStepStats\n  readonly dirty?: TraitConvergeDirtySummary\n  readonly cache?: TraitConvergePlanCacheEvidence\n  readonly generation?: TraitConvergeGenerationEvidence\n  readonly staticIr?: TraitConvergeStaticIrEvidence\n  readonly thresholds?: { readonly floorRatio?: number }\n  readonly top3?: ReadonlyArray<TraitConvergeHotspot>\n}\n", "import type { Schema } from \"effect\"\nimport {\n  type StateTraitProgram,\n  type StateTraitSpec,\n  type StateTraitEntry,\n  type StateTraitGraph,\n  type StateTraitGraphEdge,\n  type StateTraitGraphNode,\n  type StateTraitField,\n  type StateTraitFieldTrait,\n  type StateTraitPlan,\n  type StateTraitPlanStep,\n  type StateTraitResource,\n  collectNodeMeta,\n  normalizeSpec,\n} from \"./model.js\"\nimport * as Meta from \"./meta.js\"\nimport {\n  compareFieldPath,\n  getFieldPathId,\n  makeFieldPathIdRegistry,\n  normalizeFieldPath,\n  type FieldPath,\n  type FieldPathId,\n} from \"../field-path.js\"\nimport type { ConvergeStaticIrRegistry } from \"./converge-ir.js\"\n\nconst nowPerf = (): number =>\n  typeof globalThis.performance !== \"undefined\" &&\n  typeof globalThis.performance.now === \"function\"\n    ? globalThis.performance.now()\n    : Date.now()\n\ntype ConvergeWriter = Extract<\n  StateTraitEntry<any, string>,\n  { readonly kind: \"computed\" | \"link\" }\n>\n\nconst getConvergeWriterDeps = (entry: ConvergeWriter): ReadonlyArray<string> => {\n  if (entry.kind === \"computed\") {\n    return ((entry.meta as any)?.deps ?? []) as ReadonlyArray<string>\n  }\n  return [entry.meta.from as string]\n}\n\nconst computeConvergeTopoOrder = (\n  writers: ReadonlyArray<ConvergeWriter>,\n): { readonly order: ReadonlyArray<string>; readonly configError?: ConvergeStaticIrRegistry[\"configError\"] } => {\n  const writerByPath = new Map<string, ConvergeWriter>()\n  for (const entry of writers) {\n    const existing = writerByPath.get(entry.fieldPath)\n    if (existing) {\n      return {\n        order: [],\n        configError: {\n          code: \"MULTIPLE_WRITERS\",\n          message: `[StateTrait.converge] Multiple writers for field \"${entry.fieldPath}\" (${existing.kind} + ${entry.kind}).`,\n          fields: [entry.fieldPath],\n        },\n      }\n    }\n    writerByPath.set(entry.fieldPath, entry)\n  }\n\n  const nodes = new Set<string>()\n  for (const entry of writers) {\n    nodes.add(entry.fieldPath)\n  }\n\n  const indegree = new Map<string, number>()\n  const forward = new Map<string, Array<string>>()\n\n  for (const node of nodes) {\n    indegree.set(node, 0)\n    forward.set(node, [])\n  }\n\n  for (const entry of writers) {\n    const to = entry.fieldPath\n    const deps = getConvergeWriterDeps(entry)\n    for (const dep of deps) {\n      if (!nodes.has(dep)) continue\n      forward.get(dep)!.push(to)\n      indegree.set(to, (indegree.get(to) ?? 0) + 1)\n    }\n  }\n\n  const queue: Array<string> = []\n  for (const [node, deg] of indegree.entries()) {\n    if (deg === 0) queue.push(node)\n  }\n\n  const order: Array<string> = []\n  while (queue.length) {\n    const n = queue.shift()!\n    order.push(n)\n    const outs = forward.get(n)!\n    for (const to of outs) {\n      const next = (indegree.get(to) ?? 0) - 1\n      indegree.set(to, next)\n      if (next === 0) queue.push(to)\n    }\n  }\n\n  if (order.length !== nodes.size) {\n    const remaining = Array.from(nodes).filter((n) => !order.includes(n))\n    return {\n      order: [],\n      configError: {\n        code: \"CYCLE_DETECTED\",\n        message: `[StateTrait.converge] Cycle detected in computed/link graph: ${remaining.join(\", \")}`,\n        fields: remaining,\n      },\n    }\n  }\n\n  return { order }\n}\n\nconst buildConvergeIr = (\n  entries: ReadonlyArray<StateTraitEntry<any, string>>,\n): ConvergeStaticIrRegistry => {\n  const startedAt = nowPerf()\n  const generation = 0\n\n  const writers = entries.filter(\n    (e): e is ConvergeWriter => e.kind === \"computed\" || e.kind === \"link\",\n  )\n\n  const writersKey = writers\n    .map((entry) => `${entry.kind}:${entry.fieldPath}`)\n    .sort()\n    .join(\"|\")\n\n  const depsKey = writers\n    .map((entry) => {\n      const deps = getConvergeWriterDeps(entry).slice().sort().join(\",\")\n      return `${entry.kind}:${entry.fieldPath}=>${deps}`\n    })\n    .sort()\n    .join(\"|\")\n\n  const writerByPath = new Map<string, ConvergeWriter>()\n  for (const entry of writers) {\n    writerByPath.set(entry.fieldPath, entry)\n  }\n\n  const topo = writers.length > 0 ? computeConvergeTopoOrder(writers) : { order: [] as ReadonlyArray<string> }\n  const stepsById: Array<ConvergeWriter> = topo.configError\n    ? []\n    : topo.order.map((path) => writerByPath.get(path)!)\n\n  const fieldPathTable = new Map<string, FieldPath>()\n  const addPath = (path: FieldPath): void => {\n    for (let i = 1; i <= path.length; i++) {\n      const prefix = path.slice(0, i)\n      const key = JSON.stringify(prefix)\n      if (!fieldPathTable.has(key)) fieldPathTable.set(key, prefix)\n    }\n  }\n\n  for (const entry of writers) {\n    const out = normalizeFieldPath(entry.fieldPath)\n    if (out) addPath(out)\n    for (const dep of getConvergeWriterDeps(entry)) {\n      const depPath = normalizeFieldPath(dep)\n      if (depPath) addPath(depPath)\n    }\n  }\n\n  const fieldPaths = Array.from(fieldPathTable.values()).sort(compareFieldPath)\n  const fieldPathIdRegistry = makeFieldPathIdRegistry(fieldPaths)\n\n  const stepOutFieldPathIdByStepId: Array<FieldPathId> = []\n  const stepDepsFieldPathIdsByStepId: Array<ReadonlyArray<FieldPathId>> = []\n\n  for (const entry of stepsById) {\n    const out = normalizeFieldPath(entry.fieldPath)\n    const outId =\n      out != null ? getFieldPathId(fieldPathIdRegistry, out) : undefined\n    if (outId == null) {\n      throw new Error(\n        `[StateTrait.build] Failed to map converge output fieldPath \"${entry.fieldPath}\" to FieldPathId.`,\n      )\n    }\n\n    const depIds: Array<FieldPathId> = []\n    for (const dep of getConvergeWriterDeps(entry)) {\n      const depPath = normalizeFieldPath(dep)\n      if (!depPath) continue\n      const depId = getFieldPathId(fieldPathIdRegistry, depPath)\n      if (depId != null) depIds.push(depId)\n    }\n\n    stepOutFieldPathIdByStepId.push(outId)\n    stepDepsFieldPathIdsByStepId.push(depIds)\n  }\n\n  const topoOrder = stepsById.map((_, i) => i)\n  const buildDurationMs = Math.max(0, nowPerf() - startedAt)\n\n  return {\n    generation,\n    writersKey,\n    depsKey,\n    fieldPaths,\n    fieldPathIdRegistry,\n    ...(topo.configError ? { configError: topo.configError } : null),\n    stepsById,\n    stepOutFieldPathIdByStepId,\n    stepDepsFieldPathIdsByStepId,\n    topoOrder,\n    buildDurationMs,\n  }\n}\n\n/**\n * \u6839\u636E StateTraitEntry \u6784\u9020\u6807\u51C6\u5316\u7684 FieldTrait\u3002\n *\n * - \u76EE\u524D\u4EC5\u5BF9 link/source \u586B\u5145 deps \u4FE1\u606F\uFF0Ccomputed \u7684\u4F9D\u8D56\u6682\u4E0D\u9759\u6001\u5206\u6790\uFF1B\n * - \u540E\u7EED\u82E5\u5F15\u5165\u663E\u5F0F\u4F9D\u8D56\u58F0\u660E\uFF0C\u53EF\u4EE5\u5728\u6B64\u5904\u6269\u5C55\u3002\n */\nconst toFieldTrait = (\n  entry: StateTraitEntry<any, string>,\n): StateTraitFieldTrait => {\n  const deps: Array<string> = []\n\n  if (entry.kind === \"computed\") {\n    const meta = entry.meta as any\n    const list = meta.deps as ReadonlyArray<string> | undefined\n    if (list) deps.push(...list)\n  } else if (entry.kind === \"source\") {\n    const meta = entry.meta as any\n    const list = meta.deps as ReadonlyArray<string> | undefined\n    if (list) deps.push(...list)\n  } else if (entry.kind === \"link\") {\n    deps.push(entry.meta.from as string)\n  } else if (entry.kind === \"check\") {\n    const meta = entry.meta as any\n    const rules = (meta?.rules ?? {}) as Record<string, any>\n    for (const name of Object.keys(rules)) {\n      const rule = rules[name]\n      if (rule && typeof rule === \"object\") {\n        const list = rule.deps as ReadonlyArray<string> | undefined\n        if (list) deps.push(...list)\n      }\n    }\n  }\n\n  return {\n    fieldId: entry.fieldPath,\n    kind: entry.kind,\n    // meta \u5728\u8FD0\u884C\u65F6\u4FDD\u6301\u4E0E Entry.meta \u4E00\u81F4\uFF0C\u4FBF\u4E8E install \u76F4\u63A5\u590D\u7528\u3002\n    meta: entry.meta as any,\n    deps,\n  }\n}\n\n/**\n * \u4ECE\u5F52\u4E00\u5316\u7684 Entry \u5217\u8868\u6784\u5EFA Field / Node / Edge / Resource \u96C6\u5408\u3002\n */\nconst buildGraph = (\n  entries: ReadonlyArray<StateTraitEntry<any, string>>,\n  nodeMetaByFieldPath: ReadonlyMap<string, Meta.TraitMeta>,\n): {\n  readonly graph: StateTraitGraph\n  readonly plan: StateTraitPlan\n} => {\n  const fieldMap = new Map<string, StateTraitField>()\n  const nodes: Array<StateTraitGraphNode> = []\n  const edges: Array<StateTraitGraphEdge> = []\n  const resourcesById = new Map<string, StateTraitResource>()\n  const planSteps: Array<StateTraitPlanStep> = []\n\n  const ensureField = (fieldPath: string): StateTraitField => {\n    let field = fieldMap.get(fieldPath)\n    if (!field) {\n      field = {\n        id: fieldPath,\n        path: fieldPath,\n        traits: [],\n      }\n      fieldMap.set(fieldPath, field)\n    }\n    return field\n  }\n\n  for (const entry of entries) {\n    const fieldPath = entry.fieldPath\n    const field = ensureField(fieldPath)\n    const trait = toFieldTrait(entry)\n\n    ;(field.traits as Array<StateTraitFieldTrait>).push(trait)\n\n    // \u9488\u5BF9\u4E0D\u540C kind \u6784\u5EFA Graph \u8FB9\u4E0E Plan \u6B65\u9AA4\u3002\n    if (entry.kind === \"computed\") {\n      const stepId = `computed:${fieldPath}`\n      planSteps.push({\n        id: stepId,\n        kind: \"computed-update\",\n        targetFieldPath: fieldPath,\n        // \u8BF4\u660E\uFF1A\u5F53\u524D\u7248\u672C\u4E0D\u9759\u6001\u5206\u6790 computed \u4F9D\u8D56\u5B57\u6BB5\uFF0CsourceFieldPaths \u7559\u7A7A\u3002\n      })\n      // \u82E5\u663E\u5F0F\u58F0\u660E deps\uFF0C\u5219\u8865\u5145 Graph \u8FB9\uFF08\u7528\u4E8E\u8BCA\u65AD/\u53CD\u5411\u95ED\u5305\u8BA1\u7B97\uFF09\u3002\n      const deps = (entry.meta as any).deps as ReadonlyArray<string> | undefined\n      if (deps) {\n        for (const dep of deps) {\n          ensureField(dep)\n          edges.push({\n            id: `computed:${dep}->${fieldPath}`,\n            from: dep,\n            to: fieldPath,\n            kind: \"computed\",\n          })\n        }\n      }\n    } else if (entry.kind === \"link\") {\n      const from = entry.meta.from as string\n      ensureField(from)\n\n      const edgeId = `link:${from}->${fieldPath}`\n      edges.push({\n        id: edgeId,\n        from,\n        to: fieldPath,\n        kind: \"link\",\n      })\n\n      planSteps.push({\n        id: `link:${fieldPath}`,\n        kind: \"link-propagate\",\n        targetFieldPath: fieldPath,\n        sourceFieldPaths: [from],\n        debugInfo: {\n          graphEdgeId: edgeId,\n        },\n      })\n    } else if (entry.kind === \"source\") {\n      const resourceId = entry.meta.resource\n      const resourceMeta = Meta.sanitize((entry.meta as any).meta)\n\n      const existing = resourcesById.get(resourceId)\n      if (existing) {\n        const ownerFields = [...existing.ownerFields, fieldPath]\n        let meta = existing.meta\n        let metaOrigin = existing.metaOrigin\n        let metaConflicts = existing.metaConflicts\n\n        if (resourceMeta) {\n          const merged = Meta.mergeCanonical(\n            { meta, origin: metaOrigin, conflicts: metaConflicts },\n            { origin: fieldPath, meta: resourceMeta },\n          )\n          meta = merged.meta\n          metaOrigin = merged.origin\n          metaConflicts = merged.conflicts\n        }\n\n        resourcesById.set(resourceId, {\n          ...existing,\n          ownerFields,\n          meta,\n          metaOrigin,\n          metaConflicts,\n        })\n      } else {\n        resourcesById.set(resourceId, {\n          resourceId,\n          // \u6682\u65F6\u4F7F\u7528\u7B80\u5355\u7684\u6807\u8BC6\u5B57\u7B26\u4E32\uFF0C\u540E\u7EED\u53EF\u6839\u636E key \u89C4\u5219\u8FDB\u4E00\u6B65\u7ED3\u6784\u5316\u3002\n          keySelector: `StateTrait.source@${fieldPath}`,\n          ownerFields: [fieldPath],\n          meta: resourceMeta,\n          metaOrigin: resourceMeta ? fieldPath : undefined,\n        })\n      }\n\n      planSteps.push({\n        id: `source:${fieldPath}`,\n        kind: \"source-refresh\",\n        targetFieldPath: fieldPath,\n        resourceId,\n        keySelectorId: `StateTrait.source@${fieldPath}`,\n      })\n\n      const deps = (entry.meta as any).deps as ReadonlyArray<string> | undefined\n      if (deps) {\n        for (const dep of deps) {\n          ensureField(dep)\n          edges.push({\n            id: `source-dep:${dep}->${fieldPath}`,\n            from: dep,\n            to: fieldPath,\n            kind: \"source-dep\",\n          })\n        }\n      }\n    } else if (entry.kind === \"check\") {\n      planSteps.push({\n        id: `check:${fieldPath}`,\n        kind: \"check-validate\",\n        targetFieldPath: fieldPath,\n      })\n\n      // \u82E5\u89C4\u5219\u663E\u5F0F\u58F0\u660E deps\uFF0C\u5219\u8865\u5145 Graph \u8FB9\uFF08\u7528\u4E8E ReverseClosure scoped validate\uFF09\u3002\n      if (trait.deps.length > 0) {\n        for (const dep of trait.deps) {\n          ensureField(dep)\n          edges.push({\n            id: `check-dep:${dep}->${fieldPath}`,\n            from: dep,\n            to: fieldPath,\n            kind: \"check-dep\",\n          })\n        }\n      }\n    }\n  }\n\n  for (const field of fieldMap.values()) {\n    nodes.push({\n      id: field.id,\n      field,\n      traits: field.traits,\n      meta: nodeMetaByFieldPath.get(field.id),\n    })\n  }\n\n  const graph: StateTraitGraph = {\n    _tag: \"StateTraitGraph\",\n    nodes,\n    edges,\n    resources: Array.from(resourcesById.values()),\n  }\n\n  const plan: StateTraitPlan = {\n    _tag: \"StateTraitPlan\",\n    steps: planSteps,\n  }\n\n  return { graph, plan }\n}\n\n/**\n * \u9488\u5BF9 link \u8FB9\u8FDB\u884C\u7B80\u5355\u7684\u73AF\u8DEF\u68C0\u6D4B\u3002\n *\n * - \u4EC5\u8003\u8651 kind = \"link\" \u7684\u8FB9\uFF0Ccomputed/source \u9996\u7248\u4E0D\u53C2\u4E0E\u73AF\u68C0\u6D4B\uFF1B\n * - \u82E5\u53D1\u73B0\u73AF\u8DEF\uFF0C\u629B\u51FA\u5E26\u6709\u8DEF\u5F84\u4FE1\u606F\u7684\u9519\u8BEF\uFF0C\u907F\u514D\u5728\u8FD0\u884C\u65F6\u8FDB\u5165\u65E0\u9650\u66F4\u65B0\u3002\n */\nconst assertNoLinkCycles = (edges: ReadonlyArray<StateTraitGraphEdge>): void => {\n  const adjacency = new Map<string, string[]>()\n\n  for (const edge of edges) {\n    if (edge.kind !== \"link\") continue\n    const list = adjacency.get(edge.from) ?? []\n    list.push(edge.to)\n    adjacency.set(edge.from, list)\n  }\n\n  const visited = new Set<string>()\n  const stack = new Set<string>()\n\n  const dfs = (node: string): void => {\n    if (stack.has(node)) {\n      throw new Error(\n        `[StateTrait.build] link cycle detected at field \"${node}\". ` +\n          \"Please check link traits for circular dependencies.\",\n      )\n    }\n    if (visited.has(node)) return\n    visited.add(node)\n    stack.add(node)\n\n    const nexts = adjacency.get(node)\n    if (nexts) {\n      for (const to of nexts) {\n        dfs(to)\n      }\n    }\n\n    stack.delete(node)\n  }\n\n  for (const node of adjacency.keys()) {\n    if (!visited.has(node)) {\n      dfs(node)\n    }\n  }\n}\n\n/**\n * \u6839\u636E\u7ED9\u5B9A\u7684 stateSchema \u4E0E traits spec \u6784\u9020 StateTraitProgram\u3002\n *\n * - \u7EAF\u51FD\u6570\uFF1A\u4E0D\u4F9D\u8D56\u5916\u90E8 Env / \u5168\u5C40\u72B6\u6001\uFF1B\n * - \u5F53\u524D\u5B9E\u73B0\u805A\u7126\u4E8E\uFF1A\n *   - \u4ECE Spec \u5F52\u4E00\u5316\u51FA Entry \u5217\u8868\uFF1B\n *   - \u57FA\u4E8E Entry \u6784\u5EFA\u8F7B\u91CF\u7248 Graph / Plan\uFF1B\n *   - \u5BF9 link \u8FB9\u505A\u57FA\u672C\u73AF\u8DEF\u68C0\u6D4B\u3002\n *\n * \u540E\u7EED\u82E5\u9700\u8981\u66F4\u7CBE\u7EC6\u7684\u4F9D\u8D56\u5206\u6790\uFF08\u4F8B\u5982\u5BF9 computed/key \u505A\u9759\u6001\u89E3\u6790\uFF09\uFF0C\n * \u53EF\u4EE5\u5728\u672C\u6A21\u5757\u5185\u6F14\u8FDB\u5B9E\u73B0\uFF0C\u800C\u4E0D\u5F71\u54CD\u5BF9\u5916 API\u3002\n */\nexport const build = <S extends object>(\n  stateSchema: Schema.Schema<S, any>,\n  spec: StateTraitSpec<S>,\n): StateTraitProgram<S> => {\n  const entries = normalizeSpec(spec) as ReadonlyArray<StateTraitEntry<S, string>>\n  const nodeMetaByFieldPath = collectNodeMeta(spec)\n\n  // Phase 4\uFF08US2\uFF09\uFF1A\u5F3A\u5236\u663E\u5F0F deps\uFF08Graph/diagnostics/replay \u53EA\u8BA4 deps \u4F5C\u4E3A\u4F9D\u8D56\u4E8B\u5B9E\u6E90\uFF09\u3002\n  for (const entry of entries) {\n    if (entry.kind === \"computed\") {\n      const deps = (entry.meta as any).deps as ReadonlyArray<string> | undefined\n      if (deps === undefined) {\n        throw new Error(\n          `[StateTrait.build] Missing explicit deps for computed \"${entry.fieldPath}\". ` +\n            \"Please use StateTrait.computed({ deps: [...], get: ... }).\",\n        )\n      }\n    }\n    if (entry.kind === \"source\") {\n      const deps = (entry.meta as any).deps as ReadonlyArray<string> | undefined\n      if (deps === undefined) {\n        throw new Error(\n          `[StateTrait.build] Missing explicit deps for source \"${entry.fieldPath}\". ` +\n            \"Please provide meta.deps for StateTrait.source({ deps: [...], ... }).\",\n        )\n      }\n    }\n    if (entry.kind === \"check\") {\n      const rules = ((entry.meta as any)?.rules ?? {}) as Record<string, any>\n      for (const name of Object.keys(rules)) {\n        const rule = rules[name]\n        if (typeof rule === \"function\" || !rule || typeof rule !== \"object\") {\n          throw new Error(\n            `[StateTrait.build] Missing explicit deps for check \"${entry.fieldPath}\" rule \"${name}\". ` +\n              \"Please use { deps: [...], validate: ... } form.\",\n          )\n        }\n        if ((rule as any).deps === undefined) {\n          throw new Error(\n            `[StateTrait.build] Missing explicit deps for check \"${entry.fieldPath}\" rule \"${name}\". ` +\n              \"Please provide deps: [...].\",\n          )\n        }\n      }\n    }\n  }\n\n  const { graph, plan } = buildGraph(entries, nodeMetaByFieldPath)\n\n  // \u9488\u5BF9 link \u8FB9\u8FDB\u884C\u4E00\u6B21\u73AF\u8DEF\u68C0\u6D4B\uFF0C\u907F\u514D\u660E\u663E\u7684\u914D\u7F6E\u9519\u8BEF\u3002\n  assertNoLinkCycles(graph.edges)\n\n  return {\n    stateSchema,\n    spec,\n    entries: entries as ReadonlyArray<StateTraitEntry<any, string>>,\n    graph,\n    plan,\n    convergeIr: buildConvergeIr(entries as ReadonlyArray<StateTraitEntry<any, string>>),\n  }\n}\n", "import { Effect } from \"effect\"\nimport type { BoundApi } from \"../runtime/core/module.js\"\nimport { getBoundInternals } from \"../runtime/core/runtimeInternalsAccessor.js\"\nimport * as SourceRuntime from \"./source.js\"\nimport type {\n  StateTraitProgram,\n  StateTraitPlanStep,\n  StateTraitEntry,\n  TraitConvergeGenerationBumpReason,\n} from \"./model.js\"\n\nconst buildEntryIndex = <S>(\n  entries: ReadonlyArray<StateTraitEntry<S, string>>,\n): Map<string, Array<StateTraitEntry<S, string>>> => {\n  const index = new Map<string, Array<StateTraitEntry<S, string>>>()\n  for (const entry of entries) {\n    const list = index.get(entry.fieldPath) ?? []\n    list.push(entry)\n    index.set(entry.fieldPath, list)\n  }\n  return index\n}\n\n/**\n * \u5728\u7ED9\u5B9A Bound API \u4E0A\u5B89\u88C5 StateTraitProgram \u63CF\u8FF0\u7684\u884C\u4E3A\u3002\n *\n * - Phase 2 \u6700\u5C0F\u5B9E\u73B0\uFF1A\n *   - \u4E3A computed \u5B57\u6BB5\u6CE8\u518C watcher\uFF1A\u5F53 State \u53D8\u5316\u65F6\u91CD\u7B97\u76EE\u6807\u5B57\u6BB5\uFF1B\n *   - \u4E3A link \u5B57\u6BB5\u6CE8\u518C watcher\uFF1A\u5F53\u6E90\u5B57\u6BB5\u53D8\u5316\u65F6\u540C\u6B65\u76EE\u6807\u5B57\u6BB5\uFF1B\n *   - \u4E3A source \u5B57\u6BB5\u9884\u7559\u5237\u65B0\u5165\u53E3\uFF08\u6682\u4E0D\u89E6\u53D1\u5916\u90E8\u8C03\u7528\uFF09\u3002\n *\n * \u8BF4\u660E\uFF1A\n * - \u6240\u6709 watcher \u90FD\u901A\u8FC7 Bound API `$` \u5B89\u88C5\uFF0C\u4E0D\u76F4\u63A5\u4F9D\u8D56 ModuleRuntime\uFF1B\n * - \u6BCF\u4E2A PlanStep \u5BF9\u5E94\u4E00\u4E2A\u957F\u751F\u547D\u5468\u671F Effect\uFF0C\u901A\u8FC7 forkScoped \u6302\u8F7D\u5230 Runtime Scope\u3002\n */\nexport const install = <S>(\n  bound: BoundApi<any, any>,\n  program: StateTraitProgram<S>,\n): Effect.Effect<void, never, any> => {\n  // \u4F18\u5148\u5C06 Program \u6CE8\u518C\u5230 Runtime\uFF08\u6536\u655B\u5F15\u64CE\u5728\u4E8B\u52A1\u63D0\u4EA4\u524D\u7EDF\u4E00\u6267\u884C computed/link/check\uFF09\u3002\n  try {\n    const internals = getBoundInternals(bound as any)\n    ;(internals.traits.registerStateTraitProgram as any)(program, {\n      bumpReason: \"logic_installed\" as TraitConvergeGenerationBumpReason,\n    })\n  } catch {\n    // no-op for legacy/mocked bound\n  }\n\n  const entryIndex = buildEntryIndex(program.entries as any)\n\n  const installStep = (\n    step: StateTraitPlanStep,\n  ): Effect.Effect<void, never, any> => {\n    if (!step.targetFieldPath) {\n      return Effect.void\n    }\n\n    if (step.kind !== \"source-refresh\") {\n      // computed/link/check \u5DF2\u7531 Runtime \u5185\u6838\u5728\u4E8B\u52A1\u7A97\u53E3\u5185\u5904\u7406\uFF1Binstall \u9636\u6BB5\u4EC5\u4FDD\u7559 source.refresh \u5165\u53E3\u3002\n      return Effect.void\n    }\n\n    const candidates = entryIndex.get(step.targetFieldPath)\n    if (!candidates || candidates.length === 0) {\n      // Plan \u4E2D\u5F15\u7528\u4E86\u4E0D\u5B58\u5728\u7684 Spec\uFF1A\u89C6\u4E3A\u6784\u5EFA\u9636\u6BB5\u7684 bug\uFF0C\u8FD9\u91CC\u76F4\u63A5\u5FFD\u7565\u8BE5\u6B65\u9AA4\u3002\n      return Effect.void\n    }\n\n    const entry = candidates.find((e) => e.kind === \"source\")\n\n    if (!entry) {\n      return Effect.void\n    }\n\n    return SourceRuntime.installSourceRefresh(\n      bound,\n      step,\n      entry as any,\n    )\n  }\n\n  return Effect.forEach(program.plan.steps, (step) => installStep(step)).pipe(Effect.asVoid)\n}\n", "// StateTrait \u547D\u540D\u7A7A\u95F4\u6A21\u5757\uFF08@logix/core/StateTrait\uFF09\uFF1A\n// - \u63D0\u4F9B traits spec \u7684 DSL\uFF08from/node/list/computed/source/link\uFF09\uFF1B\n// - build\uFF1A\u5F52\u4E00\u5316 spec \u5E76\u751F\u6210 Program/Graph/Plan\uFF1B\n// - install\uFF1A\u5C06 Program \u5B89\u88C5\u5230\u8FD0\u884C\u65F6\uFF08\u6CE8\u518C source.refresh \u7B49\u5165\u53E3\uFF0C\u5E76\u5728\u4E8B\u52A1\u7A97\u53E3\u6267\u884C converge/validate\uFF09\u3002\n\nimport type { Schema } from \"effect\"\nimport { Effect } from \"effect\"\nimport type { BoundApi } from \"./Bound.js\"\nimport * as Model from \"./internal/state-trait/model.js\"\nimport * as FieldPath from \"./internal/state-trait/field-path.js\"\nimport * as RowId from \"./internal/state-trait/rowid.js\"\nimport * as InternalBuild from \"./internal/state-trait/build.js\"\nimport * as InternalInstall from \"./internal/state-trait/install.js\"\nimport * as InternalIr from \"./internal/state-trait/ir.js\"\n\n// \u5BF9\u5916\u66B4\u9732\u7684\u6838\u5FC3\u7C7B\u578B\u522B\u540D\u3002\nexport type StateTraitSpec<S> = Model.StateTraitSpec<S>\nexport type StateTraitEntry<S = unknown, P extends string = string> =\n  Model.StateTraitEntry<S, P>\nexport type StateTraitProgram<S> = Model.StateTraitProgram<S>\nexport type StateTraitGraph = Model.StateTraitGraph\nexport type StateTraitPlan = Model.StateTraitPlan\nexport type StateTraitNode<Input = unknown, Ctx = unknown> = Model.StateTraitNode<Input, Ctx>\nexport type StateTraitList<Item = unknown> = Model.StateTraitList<Item>\nexport type CheckRule<Input = unknown, Ctx = unknown> = Model.CheckRule<Input, Ctx>\n\nexport type StateFieldPath<S> = FieldPath.StateFieldPath<S>\nexport type StateAtPath<S, P> = FieldPath.StateAtPath<S, P>\n\nexport const $root = \"$root\" as const\n\ntype DepsArgs<S extends object, Deps extends ReadonlyArray<StateFieldPath<S>>> = {\n  readonly [K in keyof Deps]: StateAtPath<S, Deps[K]>\n}\n\n/**\n * StateTrait.from\uFF1A\n * - \u6839\u636E State Schema \u6536\u7A84\u53EF\u7528\u5B57\u6BB5\u8DEF\u5F84\uFF0C\u5E76\u4E3A traits spec \u63D0\u4F9B\u7C7B\u578B\u7EA6\u675F\uFF1B\n * - \u8FD0\u884C\u65F6\u4EC5\u8FD4\u56DE\u539F\u59CB spec \u5BF9\u8C61\uFF0C\u5177\u4F53\u5F52\u4E00\u5316\u7531 internal \u5C42\u5904\u7406\u3002\n */\nexport const from = <S extends object, I>(\n  _schema: Schema.Schema<S, I>,\n) =>\n(spec: StateTraitSpec<S>): StateTraitSpec<S> =>\n  spec\n\nexport const node = <Input = unknown, Ctx = unknown>(\n  spec: Omit<Model.StateTraitNode<Input, Ctx>, \"_tag\">,\n): Model.StateTraitNode<Input, Ctx> => ({\n  _tag: \"StateTraitNode\",\n  ...spec,\n})\n\nexport const list = <Item = unknown>(\n  spec: Omit<Model.StateTraitList<Item>, \"_tag\">,\n): Model.StateTraitList<Item> => ({\n  _tag: \"StateTraitList\",\n  ...spec,\n})\n\n/**\n * StateTrait.computed\uFF1A\n * - \u4E3A\u67D0\u4E2A\u5B57\u6BB5\u58F0\u660E computed \u80FD\u529B\uFF1B\n * - \u901A\u8FC7\u663E\u5F0F deps \u4F5C\u4E3A\u552F\u4E00\u4F9D\u8D56\u4E8B\u5B9E\u6E90\uFF08\u7528\u4E8E\u8BCA\u65AD/\u53CD\u5411\u95ED\u5305/\u540E\u7EED\u589E\u91CF\u8C03\u5EA6\uFF09\u3002\n */\nexport const computed = <\n  S extends object,\n  P extends StateFieldPath<S>,\n  const Deps extends ReadonlyArray<StateFieldPath<S>>,\n>(\n  input: {\n    readonly deps: Deps\n    readonly get: (...depsValues: DepsArgs<S, Deps>) => StateAtPath<S, P>\n    readonly equals?: (prev: StateAtPath<S, P>, next: StateAtPath<S, P>) => boolean\n  },\n): StateTraitEntry<S, P> => {\n  const derive = (state: Readonly<S>): StateAtPath<S, P> => {\n    const args = (input.deps as ReadonlyArray<string>).map((dep) =>\n      RowId.getAtPath(state as any, dep),\n    )\n    return (input.get as any)(...args)\n  }\n  return {\n    fieldPath: undefined as unknown as P,\n    kind: \"computed\",\n    meta: { deps: input.deps, derive, equals: input.equals },\n  } as StateTraitEntry<S, P>\n}\n\n/**\n * StateTrait.source\uFF1A\n * - \u4E3A\u67D0\u4E2A\u5B57\u6BB5\u58F0\u660E\u5916\u90E8\u8D44\u6E90\u6765\u6E90\uFF08Resource / Query \u96C6\u6210\u5728\u540E\u7EED Phase \u4E2D\u5B9E\u73B0\uFF09\uFF1B\n * - \u5185\u6838\u8D1F\u8D23 keyHash gate / concurrency / ReplayMode \u884C\u4E3A\u3002\n */\nexport const source = <S extends object, P extends StateFieldPath<S>>(meta: {\n  resource: string\n  deps: ReadonlyArray<StateFieldPath<S>>\n  key: (state: Readonly<S>) => unknown\n  triggers?: ReadonlyArray<\"onMount\" | \"onValueChange\" | \"manual\">\n  debounceMs?: number\n  concurrency?: \"switch\" | \"exhaust-trailing\"\n  /**\n   * \u7528\u4E8E Devtools/\u6587\u6863\u7684\u53EF\u5E8F\u5217\u5316\u5143\u4FE1\u606F\uFF08\u767D\u540D\u5355\u5B57\u6BB5\u4F1A\u5728 build \u9636\u6BB5\u88AB\u63D0\u53D6\uFF09\u3002\n   */\n  meta?: Record<string, unknown>\n}): StateTraitEntry<S, P> =>\n  ({\n    fieldPath: undefined as unknown as P,\n    kind: \"source\",\n    meta,\n  } as StateTraitEntry<S, P>)\n\n/**\n * StateTrait.link\uFF1A\n * - \u4E3A\u76EE\u6807\u5B57\u6BB5\u58F0\u660E\u4ECE\u5176\u4ED6\u5B57\u6BB5\u8054\u52A8\u7684\u80FD\u529B\uFF1B\n * - meta.from \u540C\u6837\u53D7 StateFieldPath<S> \u7EA6\u675F\u3002\n */\nexport const link = <S extends object, P extends StateFieldPath<S>>(meta: {\n  from: StateFieldPath<S>\n}): StateTraitEntry<S, P> =>\n  ({\n    fieldPath: undefined as unknown as P,\n    kind: \"link\",\n    meta,\n  } as StateTraitEntry<S, P>)\n\n/**\n * StateTrait.build\uFF1A\n * - \u5C06 Spec \u5F52\u4E00\u5316\u4E3A\u53EF\u6267\u884C Program\uFF08\u542B Graph/Plan\uFF09\uFF1B\n * - DSL \u5C42\u4E0E build \u5C42\u7684\u7EA6\u675F\u4EE5 build \u4E3A\u6700\u7EC8\u88C1\u51B3\uFF08\u4F8B\u5982\u5F3A\u5236\u663E\u5F0F deps\uFF09\u3002\n */\nexport const build = <S extends object>(\n  stateSchema: Schema.Schema<S, any>,\n  spec: StateTraitSpec<S>,\n): StateTraitProgram<S> => InternalBuild.build(stateSchema, spec)\n\n/**\n * StateTrait.install\uFF1A\n * - \u5728\u7ED9\u5B9A Bound API \u4E0A\u5B89\u88C5 Program \u63CF\u8FF0\u7684\u884C\u4E3A\uFF08computed/link/source/check \u7B49\uFF09\uFF1B\n * - \u6BCF\u4E2A PlanStep \u5BF9\u5E94\u4E00\u4E2A\u957F\u751F\u547D\u5468\u671F Effect\uFF0C\u5E76\u6302\u8F7D\u5230 Runtime Scope\u3002\n */\nexport const install = <S extends object>(\n  bound: BoundApi<any, any>,\n  program: StateTraitProgram<S>,\n): Effect.Effect<void, never, any> => InternalInstall.install(bound, program)\n\n/**\n * StateTrait.exportStaticIr\uFF1A\n * - \u5BFC\u51FA\u6700\u5C0F Static IR\uFF08\u5BF9\u9F50 009 \u7684 canonical FieldPath \u7EA6\u675F\uFF09\uFF0C\u4F9B Devtools/\u5BF9\u9F50\u5B9E\u9A8C\u5BA4\u6D88\u8D39\uFF1B\n * - \u8FD4\u56DE\u503C\u5FC5\u987B\u53EF JSON.stringify\uFF08\u7528\u4E8E\u8BC1\u636E\u5305/\u56DE\u653E\u94FE\u8DEF\uFF09\u3002\n */\nexport type StaticIr = InternalIr.StaticIr\n\nexport const exportStaticIr = (\n  program: StateTraitProgram<any>,\n  moduleId: string,\n  options?: {\n    readonly version?: string\n  },\n): StaticIr =>\n  InternalIr.exportStaticIr({\n    program,\n    moduleId,\n    version: options?.version,\n  })\n"],
  "mappings": "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",
  "names": ["from", "node", "node", "list", "from", "list", "build", "install", "exportStaticIr"]
}

|