@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,1991 @@
|
|
|
1
|
+
// @logixjs/core subpath bundles for @logixjs/sandbox
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
ReplayLog,
|
|
5
|
+
getAtPath,
|
|
6
|
+
parseListItemFieldPath,
|
|
7
|
+
setAtPathMutating,
|
|
8
|
+
toListItemValuePath
|
|
9
|
+
} from "./chunk-C2MYMMO4.js";
|
|
10
|
+
import {
|
|
11
|
+
Snapshot,
|
|
12
|
+
internal,
|
|
13
|
+
keyHash
|
|
14
|
+
} from "./chunk-EZTMVCJF.js";
|
|
15
|
+
import {
|
|
16
|
+
create
|
|
17
|
+
} from "./chunk-KE2H62D6.js";
|
|
18
|
+
import {
|
|
19
|
+
mergeCanonical,
|
|
20
|
+
sanitize
|
|
21
|
+
} from "./chunk-X2Z5VYG3.js";
|
|
22
|
+
import {
|
|
23
|
+
compareFieldPath,
|
|
24
|
+
getFieldPathId,
|
|
25
|
+
makeFieldPathIdRegistry,
|
|
26
|
+
normalizeFieldPath
|
|
27
|
+
} from "./chunk-ASVMK364.js";
|
|
28
|
+
import {
|
|
29
|
+
getBoundInternals
|
|
30
|
+
} from "./chunk-7GPAF2UK.js";
|
|
31
|
+
import {
|
|
32
|
+
ReplayModeConfigTag,
|
|
33
|
+
forceSourceRefresh,
|
|
34
|
+
inSyncTransactionFiber,
|
|
35
|
+
isDevEnv
|
|
36
|
+
} from "./chunk-YMVMZA3U.js";
|
|
37
|
+
import {
|
|
38
|
+
RunSessionTag,
|
|
39
|
+
make,
|
|
40
|
+
run
|
|
41
|
+
} from "./chunk-W365QUZ5.js";
|
|
42
|
+
import {
|
|
43
|
+
currentDiagnosticsLevel,
|
|
44
|
+
record
|
|
45
|
+
} from "./chunk-SH7TP5CJ.js";
|
|
46
|
+
import {
|
|
47
|
+
EffectOpMiddlewareTag
|
|
48
|
+
} from "./chunk-SI5OKFFY.js";
|
|
49
|
+
import {
|
|
50
|
+
fnv1a32,
|
|
51
|
+
stableStringify
|
|
52
|
+
} from "./chunk-DT7YIIWE.js";
|
|
53
|
+
|
|
54
|
+
// ../logix-core/src/internal/state-trait/build.ts
|
|
55
|
+
import * as SchemaAST from "../effect/SchemaAST.js";
|
|
56
|
+
|
|
57
|
+
// ../logix-core/src/internal/state-trait/model.ts
|
|
58
|
+
var normalizeSpec = (spec) => {
|
|
59
|
+
const entries = [];
|
|
60
|
+
const isNode = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitNode";
|
|
61
|
+
const isList = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitList";
|
|
62
|
+
const joinPath = (prefix, suffix) => {
|
|
63
|
+
if (!prefix) return suffix;
|
|
64
|
+
if (!suffix) return prefix;
|
|
65
|
+
return `${prefix}.${suffix}`;
|
|
66
|
+
};
|
|
67
|
+
const prefixDeps = (deps, prefix) => {
|
|
68
|
+
if (!deps || deps.length === 0) return [];
|
|
69
|
+
return deps.map((d) => prefix ? joinPath(prefix, d) : d);
|
|
70
|
+
};
|
|
71
|
+
const normalizeEntry = (entry, fieldPath, depPrefix) => {
|
|
72
|
+
if (entry.kind === "computed") {
|
|
73
|
+
const meta = entry.meta;
|
|
74
|
+
const rawDeps = meta.deps;
|
|
75
|
+
const deps = rawDeps !== void 0 ? prefixDeps(rawDeps, depPrefix) : void 0;
|
|
76
|
+
return {
|
|
77
|
+
...entry,
|
|
78
|
+
fieldPath,
|
|
79
|
+
meta: { ...meta, deps }
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
if (entry.kind === "source") {
|
|
83
|
+
const meta = entry.meta;
|
|
84
|
+
const rawDeps = meta.deps;
|
|
85
|
+
const deps = rawDeps !== void 0 ? prefixDeps(rawDeps, depPrefix) : void 0;
|
|
86
|
+
return {
|
|
87
|
+
...entry,
|
|
88
|
+
fieldPath,
|
|
89
|
+
meta: { ...meta, deps, _fieldPath: fieldPath }
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
if (entry.kind === "externalStore") {
|
|
93
|
+
const meta = entry.meta;
|
|
94
|
+
return {
|
|
95
|
+
...entry,
|
|
96
|
+
fieldPath,
|
|
97
|
+
meta: { ...meta, _fieldPath: fieldPath }
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
if (entry.kind === "link") {
|
|
101
|
+
const meta = entry.meta;
|
|
102
|
+
const from = prefixDeps([meta.from], depPrefix)[0] ?? meta.from;
|
|
103
|
+
return {
|
|
104
|
+
...entry,
|
|
105
|
+
fieldPath,
|
|
106
|
+
meta: { ...meta, from }
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
return {
|
|
110
|
+
...entry,
|
|
111
|
+
fieldPath
|
|
112
|
+
};
|
|
113
|
+
};
|
|
114
|
+
const expandNode = (scopeId, joinPrefix, node, options) => {
|
|
115
|
+
const addEntry = (relativeTarget, raw) => {
|
|
116
|
+
const rel = raw.fieldPath ?? relativeTarget;
|
|
117
|
+
const fieldPath = joinPrefix ? joinPath(joinPrefix, String(rel)) : String(rel);
|
|
118
|
+
entries.push(normalizeEntry(raw, fieldPath, joinPrefix));
|
|
119
|
+
};
|
|
120
|
+
const expandMaybeRecord = (value) => {
|
|
121
|
+
if (!value) return;
|
|
122
|
+
if (typeof value.kind === "string") {
|
|
123
|
+
addEntry("", value);
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
const record2 = value;
|
|
127
|
+
for (const key in record2) {
|
|
128
|
+
if (!Object.prototype.hasOwnProperty.call(record2, key)) continue;
|
|
129
|
+
const entry = record2[key];
|
|
130
|
+
if (!entry) continue;
|
|
131
|
+
addEntry(key, entry);
|
|
132
|
+
}
|
|
133
|
+
};
|
|
134
|
+
expandMaybeRecord(node.computed);
|
|
135
|
+
expandMaybeRecord(node.source);
|
|
136
|
+
expandMaybeRecord(node.link);
|
|
137
|
+
expandMaybeRecord(node.externalStore);
|
|
138
|
+
if (node.check) {
|
|
139
|
+
const rules = {};
|
|
140
|
+
const checkDepsPrefix = options?.checkDepsPrefix ?? (joinPrefix.endsWith("[]") ? joinPrefix : "");
|
|
141
|
+
const prefixCheckDeps = (deps) => {
|
|
142
|
+
if (!deps || deps.length === 0) return [];
|
|
143
|
+
return deps.map((d) => d === "" ? scopeId : checkDepsPrefix ? joinPath(checkDepsPrefix, d) : d);
|
|
144
|
+
};
|
|
145
|
+
for (const name of Object.keys(node.check)) {
|
|
146
|
+
const rule = node.check[name];
|
|
147
|
+
if (typeof rule === "function") {
|
|
148
|
+
rules[name] = rule;
|
|
149
|
+
continue;
|
|
150
|
+
}
|
|
151
|
+
if (rule && typeof rule === "object") {
|
|
152
|
+
const meta = sanitize(rule.meta);
|
|
153
|
+
rules[name] = {
|
|
154
|
+
...rule,
|
|
155
|
+
deps: prefixCheckDeps(rule.deps),
|
|
156
|
+
meta
|
|
157
|
+
};
|
|
158
|
+
continue;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
entries.push({
|
|
162
|
+
fieldPath: scopeId,
|
|
163
|
+
kind: "check",
|
|
164
|
+
meta: {
|
|
165
|
+
rules,
|
|
166
|
+
writeback: { kind: "errors" }
|
|
167
|
+
}
|
|
168
|
+
});
|
|
169
|
+
}
|
|
170
|
+
};
|
|
171
|
+
for (const key in spec) {
|
|
172
|
+
if (!Object.prototype.hasOwnProperty.call(spec, key)) continue;
|
|
173
|
+
const raw = spec[key];
|
|
174
|
+
if (!raw) continue;
|
|
175
|
+
if (isList(raw)) {
|
|
176
|
+
const listPath = key;
|
|
177
|
+
if (raw.item) {
|
|
178
|
+
expandNode(`${listPath}[]`, `${listPath}[]`, raw.item);
|
|
179
|
+
}
|
|
180
|
+
if (raw.list) {
|
|
181
|
+
expandNode(listPath, listPath, raw.list, {
|
|
182
|
+
checkDepsPrefix: `${listPath}[]`
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
continue;
|
|
186
|
+
}
|
|
187
|
+
if (isNode(raw)) {
|
|
188
|
+
if (key === "$root") {
|
|
189
|
+
expandNode("$root", "", raw);
|
|
190
|
+
} else {
|
|
191
|
+
expandNode(key, key, raw);
|
|
192
|
+
}
|
|
193
|
+
continue;
|
|
194
|
+
}
|
|
195
|
+
const entry = raw;
|
|
196
|
+
const fieldPath = entry.fieldPath ?? key;
|
|
197
|
+
entries.push(normalizeEntry(entry, String(fieldPath), ""));
|
|
198
|
+
}
|
|
199
|
+
return entries;
|
|
200
|
+
};
|
|
201
|
+
var collectNodeMeta = (spec) => {
|
|
202
|
+
const out = /* @__PURE__ */ new Map();
|
|
203
|
+
const isNode = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitNode";
|
|
204
|
+
const isList = (value) => typeof value === "object" && value !== null && value._tag === "StateTraitList";
|
|
205
|
+
const add = (scopeId, node) => {
|
|
206
|
+
const meta = sanitize(node.meta);
|
|
207
|
+
if (meta) out.set(scopeId, meta);
|
|
208
|
+
};
|
|
209
|
+
for (const key in spec) {
|
|
210
|
+
if (!Object.prototype.hasOwnProperty.call(spec, key)) continue;
|
|
211
|
+
const raw = spec[key];
|
|
212
|
+
if (!raw) continue;
|
|
213
|
+
if (isList(raw)) {
|
|
214
|
+
const listPath = key;
|
|
215
|
+
if (raw.item) add(`${listPath}[]`, raw.item);
|
|
216
|
+
if (raw.list) add(listPath, raw.list);
|
|
217
|
+
continue;
|
|
218
|
+
}
|
|
219
|
+
if (isNode(raw)) {
|
|
220
|
+
if (key === "$root") add("$root", raw);
|
|
221
|
+
else add(key, raw);
|
|
222
|
+
continue;
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
return out;
|
|
226
|
+
};
|
|
227
|
+
|
|
228
|
+
// ../logix-core/src/internal/state-trait/build.ts
|
|
229
|
+
var nowPerf = () => typeof globalThis.performance !== "undefined" && typeof globalThis.performance.now === "function" ? globalThis.performance.now() : Date.now();
|
|
230
|
+
var collectMultipleWritersError = (entries) => {
|
|
231
|
+
const kindsByFieldPath = /* @__PURE__ */ new Map();
|
|
232
|
+
for (const entry of entries) {
|
|
233
|
+
if (entry.kind !== "computed" && entry.kind !== "link" && entry.kind !== "source" && entry.kind !== "externalStore") {
|
|
234
|
+
continue;
|
|
235
|
+
}
|
|
236
|
+
const set = kindsByFieldPath.get(entry.fieldPath) ?? /* @__PURE__ */ new Set();
|
|
237
|
+
set.add(entry.kind);
|
|
238
|
+
kindsByFieldPath.set(entry.fieldPath, set);
|
|
239
|
+
}
|
|
240
|
+
const conflicts = [];
|
|
241
|
+
for (const [fieldPath, kinds] of kindsByFieldPath.entries()) {
|
|
242
|
+
if (kinds.size <= 1) continue;
|
|
243
|
+
conflicts.push({ fieldPath, kinds: Array.from(kinds).sort() });
|
|
244
|
+
}
|
|
245
|
+
if (conflicts.length === 0) return void 0;
|
|
246
|
+
conflicts.sort((a, b) => a.fieldPath < b.fieldPath ? -1 : a.fieldPath > b.fieldPath ? 1 : 0);
|
|
247
|
+
const fields = conflicts.map((c) => c.fieldPath);
|
|
248
|
+
const primary = conflicts[0];
|
|
249
|
+
const kindSummary = primary.kinds.join(" + ");
|
|
250
|
+
return {
|
|
251
|
+
code: "MULTIPLE_WRITERS",
|
|
252
|
+
message: `[StateTrait.build] Multiple writers for field "${primary.fieldPath}" (${kindSummary}). Only one of computed/link/source/externalStore can write a fieldPath.`,
|
|
253
|
+
fields
|
|
254
|
+
};
|
|
255
|
+
};
|
|
256
|
+
var getConvergeWriterDeps = (entry) => {
|
|
257
|
+
if (entry.kind === "computed") {
|
|
258
|
+
return entry.meta?.deps ?? [];
|
|
259
|
+
}
|
|
260
|
+
return [entry.meta.from];
|
|
261
|
+
};
|
|
262
|
+
var computeConvergeTopoOrder = (writers) => {
|
|
263
|
+
const writerByPath = /* @__PURE__ */ new Map();
|
|
264
|
+
for (const entry of writers) {
|
|
265
|
+
const existing = writerByPath.get(entry.fieldPath);
|
|
266
|
+
if (existing) {
|
|
267
|
+
return {
|
|
268
|
+
order: [],
|
|
269
|
+
configError: {
|
|
270
|
+
code: "MULTIPLE_WRITERS",
|
|
271
|
+
message: `[StateTrait.converge] Multiple writers for field "${entry.fieldPath}" (${existing.kind} + ${entry.kind}).`,
|
|
272
|
+
fields: [entry.fieldPath]
|
|
273
|
+
}
|
|
274
|
+
};
|
|
275
|
+
}
|
|
276
|
+
writerByPath.set(entry.fieldPath, entry);
|
|
277
|
+
}
|
|
278
|
+
const nodes = /* @__PURE__ */ new Set();
|
|
279
|
+
for (const entry of writers) {
|
|
280
|
+
nodes.add(entry.fieldPath);
|
|
281
|
+
}
|
|
282
|
+
const indegree = /* @__PURE__ */ new Map();
|
|
283
|
+
const forward = /* @__PURE__ */ new Map();
|
|
284
|
+
for (const node of nodes) {
|
|
285
|
+
indegree.set(node, 0);
|
|
286
|
+
forward.set(node, []);
|
|
287
|
+
}
|
|
288
|
+
for (const entry of writers) {
|
|
289
|
+
const to = entry.fieldPath;
|
|
290
|
+
const deps = getConvergeWriterDeps(entry);
|
|
291
|
+
for (const dep of deps) {
|
|
292
|
+
if (!nodes.has(dep)) continue;
|
|
293
|
+
forward.get(dep).push(to);
|
|
294
|
+
indegree.set(to, (indegree.get(to) ?? 0) + 1);
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
const queue = [];
|
|
298
|
+
for (const [node, deg] of indegree.entries()) {
|
|
299
|
+
if (deg === 0) queue.push(node);
|
|
300
|
+
}
|
|
301
|
+
const order = [];
|
|
302
|
+
while (queue.length) {
|
|
303
|
+
const n = queue.shift();
|
|
304
|
+
order.push(n);
|
|
305
|
+
const outs = forward.get(n);
|
|
306
|
+
for (const to of outs) {
|
|
307
|
+
const next = (indegree.get(to) ?? 0) - 1;
|
|
308
|
+
indegree.set(to, next);
|
|
309
|
+
if (next === 0) queue.push(to);
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
if (order.length !== nodes.size) {
|
|
313
|
+
const remaining = Array.from(nodes).filter((n) => !order.includes(n));
|
|
314
|
+
return {
|
|
315
|
+
order: [],
|
|
316
|
+
configError: {
|
|
317
|
+
code: "CYCLE_DETECTED",
|
|
318
|
+
message: `[StateTrait.converge] Cycle detected in computed/link graph: ${remaining.join(", ")}`,
|
|
319
|
+
fields: remaining
|
|
320
|
+
}
|
|
321
|
+
};
|
|
322
|
+
}
|
|
323
|
+
return { order };
|
|
324
|
+
};
|
|
325
|
+
var collectSchemaFieldPaths = (schema) => {
|
|
326
|
+
const byKey = /* @__PURE__ */ new Map();
|
|
327
|
+
const add = (path) => {
|
|
328
|
+
const normalized = normalizeFieldPath(path);
|
|
329
|
+
if (!normalized) return;
|
|
330
|
+
byKey.set(JSON.stringify(normalized), normalized);
|
|
331
|
+
};
|
|
332
|
+
const visit = (ast, prefix, seen) => {
|
|
333
|
+
let current = ast;
|
|
334
|
+
while (true) {
|
|
335
|
+
if (SchemaAST.isSuspend(current)) {
|
|
336
|
+
if (seen.has(current)) return;
|
|
337
|
+
seen.add(current);
|
|
338
|
+
current = current.f();
|
|
339
|
+
continue;
|
|
340
|
+
}
|
|
341
|
+
if (SchemaAST.isRefinement(current)) {
|
|
342
|
+
current = current.from;
|
|
343
|
+
continue;
|
|
344
|
+
}
|
|
345
|
+
break;
|
|
346
|
+
}
|
|
347
|
+
if (SchemaAST.isTransformation(current)) {
|
|
348
|
+
visit(current.to, prefix, seen);
|
|
349
|
+
visit(current.from, prefix, seen);
|
|
350
|
+
return;
|
|
351
|
+
}
|
|
352
|
+
if (SchemaAST.isUnion(current)) {
|
|
353
|
+
for (const t of current.types) {
|
|
354
|
+
visit(t, prefix, seen);
|
|
355
|
+
}
|
|
356
|
+
return;
|
|
357
|
+
}
|
|
358
|
+
if (SchemaAST.isTupleType(current)) {
|
|
359
|
+
for (const e of current.elements) {
|
|
360
|
+
visit(e.type, prefix, seen);
|
|
361
|
+
}
|
|
362
|
+
for (const r of current.rest) {
|
|
363
|
+
visit(r.type, prefix, seen);
|
|
364
|
+
}
|
|
365
|
+
return;
|
|
366
|
+
}
|
|
367
|
+
if (SchemaAST.isTypeLiteral(current)) {
|
|
368
|
+
for (const ps of current.propertySignatures) {
|
|
369
|
+
const seg = String(ps.name);
|
|
370
|
+
if (!seg) continue;
|
|
371
|
+
const next = [...prefix, seg];
|
|
372
|
+
add(next);
|
|
373
|
+
visit(ps.type, next, seen);
|
|
374
|
+
}
|
|
375
|
+
return;
|
|
376
|
+
}
|
|
377
|
+
};
|
|
378
|
+
visit(schema.ast, [], /* @__PURE__ */ new Set());
|
|
379
|
+
return Array.from(byKey.values()).sort(compareFieldPath);
|
|
380
|
+
};
|
|
381
|
+
var buildConvergeIr = (stateSchema, entries) => {
|
|
382
|
+
const startedAt = nowPerf();
|
|
383
|
+
const generation = 0;
|
|
384
|
+
const multipleWritersError = collectMultipleWritersError(entries);
|
|
385
|
+
const writers = entries.filter((e) => e.kind === "computed" || e.kind === "link");
|
|
386
|
+
const writersKey = writers.map((entry) => `${entry.kind}:${entry.fieldPath}`).sort().join("|");
|
|
387
|
+
const depsKey = writers.map((entry) => {
|
|
388
|
+
const deps = getConvergeWriterDeps(entry).slice().sort().join(",");
|
|
389
|
+
const scheduling = entry.meta?.scheduling === "deferred" ? "d" : "i";
|
|
390
|
+
return `${entry.kind}:${entry.fieldPath}@${scheduling}=>${deps}`;
|
|
391
|
+
}).sort().join("|");
|
|
392
|
+
const writerByPath = /* @__PURE__ */ new Map();
|
|
393
|
+
for (const entry of writers) {
|
|
394
|
+
writerByPath.set(entry.fieldPath, entry);
|
|
395
|
+
}
|
|
396
|
+
const topo = multipleWritersError ? { order: [] } : writers.length > 0 ? computeConvergeTopoOrder(writers) : { order: [] };
|
|
397
|
+
const stepsById = topo.configError ? [] : topo.order.map((path) => writerByPath.get(path));
|
|
398
|
+
const fieldPathTable = /* @__PURE__ */ new Map();
|
|
399
|
+
const addPath = (path) => {
|
|
400
|
+
for (let i = 1; i <= path.length; i++) {
|
|
401
|
+
const prefix = path.slice(0, i);
|
|
402
|
+
const key = JSON.stringify(prefix);
|
|
403
|
+
if (!fieldPathTable.has(key)) fieldPathTable.set(key, prefix);
|
|
404
|
+
}
|
|
405
|
+
};
|
|
406
|
+
for (const schemaPath of collectSchemaFieldPaths(stateSchema)) {
|
|
407
|
+
addPath(schemaPath);
|
|
408
|
+
}
|
|
409
|
+
for (const entry of writers) {
|
|
410
|
+
const out = normalizeFieldPath(entry.fieldPath);
|
|
411
|
+
if (out) addPath(out);
|
|
412
|
+
for (const dep of getConvergeWriterDeps(entry)) {
|
|
413
|
+
const depPath = normalizeFieldPath(dep);
|
|
414
|
+
if (depPath) addPath(depPath);
|
|
415
|
+
}
|
|
416
|
+
}
|
|
417
|
+
const fieldPaths = Array.from(fieldPathTable.values()).sort(compareFieldPath);
|
|
418
|
+
const fieldPathIdRegistry = makeFieldPathIdRegistry(fieldPaths);
|
|
419
|
+
const fieldPathsKey = fnv1a32(stableStringify(fieldPaths));
|
|
420
|
+
const stepOutFieldPathIdByStepId = [];
|
|
421
|
+
const stepDepsFieldPathIdsByStepId = [];
|
|
422
|
+
const stepSchedulingByStepId = [];
|
|
423
|
+
for (const entry of stepsById) {
|
|
424
|
+
const out = normalizeFieldPath(entry.fieldPath);
|
|
425
|
+
const outId = out != null ? getFieldPathId(fieldPathIdRegistry, out) : void 0;
|
|
426
|
+
if (outId == null) {
|
|
427
|
+
throw new Error(`[StateTrait.build] Failed to map converge output fieldPath "${entry.fieldPath}" to FieldPathId.`);
|
|
428
|
+
}
|
|
429
|
+
const depIds = [];
|
|
430
|
+
for (const dep of getConvergeWriterDeps(entry)) {
|
|
431
|
+
const depPath = normalizeFieldPath(dep);
|
|
432
|
+
if (!depPath) continue;
|
|
433
|
+
const depId = getFieldPathId(fieldPathIdRegistry, depPath);
|
|
434
|
+
if (depId != null) depIds.push(depId);
|
|
435
|
+
}
|
|
436
|
+
stepOutFieldPathIdByStepId.push(outId);
|
|
437
|
+
stepDepsFieldPathIdsByStepId.push(depIds);
|
|
438
|
+
stepSchedulingByStepId.push(entry.meta?.scheduling === "deferred" ? "deferred" : "immediate");
|
|
439
|
+
}
|
|
440
|
+
const topoOrder = stepsById.map((_, i) => i);
|
|
441
|
+
const buildDurationMs = Math.max(0, nowPerf() - startedAt);
|
|
442
|
+
return {
|
|
443
|
+
generation,
|
|
444
|
+
writersKey,
|
|
445
|
+
depsKey,
|
|
446
|
+
fieldPathsKey,
|
|
447
|
+
fieldPaths,
|
|
448
|
+
fieldPathIdRegistry,
|
|
449
|
+
...multipleWritersError ? { configError: multipleWritersError } : topo.configError ? { configError: topo.configError } : null,
|
|
450
|
+
stepsById,
|
|
451
|
+
stepOutFieldPathIdByStepId,
|
|
452
|
+
stepDepsFieldPathIdsByStepId,
|
|
453
|
+
stepSchedulingByStepId,
|
|
454
|
+
topoOrder,
|
|
455
|
+
buildDurationMs
|
|
456
|
+
};
|
|
457
|
+
};
|
|
458
|
+
var toFieldTrait = (entry) => {
|
|
459
|
+
const deps = [];
|
|
460
|
+
if (entry.kind === "computed") {
|
|
461
|
+
const meta = entry.meta;
|
|
462
|
+
const list = meta.deps;
|
|
463
|
+
if (list) deps.push(...list);
|
|
464
|
+
} else if (entry.kind === "source") {
|
|
465
|
+
const meta = entry.meta;
|
|
466
|
+
const list = meta.deps;
|
|
467
|
+
if (list) deps.push(...list);
|
|
468
|
+
} else if (entry.kind === "link") {
|
|
469
|
+
deps.push(entry.meta.from);
|
|
470
|
+
} else if (entry.kind === "check") {
|
|
471
|
+
const meta = entry.meta;
|
|
472
|
+
const rules = meta?.rules ?? {};
|
|
473
|
+
for (const name of Object.keys(rules)) {
|
|
474
|
+
const rule = rules[name];
|
|
475
|
+
if (rule && typeof rule === "object") {
|
|
476
|
+
const list = rule.deps;
|
|
477
|
+
if (list) deps.push(...list);
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
}
|
|
481
|
+
return {
|
|
482
|
+
fieldId: entry.fieldPath,
|
|
483
|
+
kind: entry.kind,
|
|
484
|
+
// Keep meta identical to Entry.meta at runtime so install can reuse it directly.
|
|
485
|
+
meta: entry.meta,
|
|
486
|
+
deps
|
|
487
|
+
};
|
|
488
|
+
};
|
|
489
|
+
var buildGraph = (entries, nodeMetaByFieldPath) => {
|
|
490
|
+
const fieldMap = /* @__PURE__ */ new Map();
|
|
491
|
+
const nodes = [];
|
|
492
|
+
const edges = [];
|
|
493
|
+
const resourcesById = /* @__PURE__ */ new Map();
|
|
494
|
+
const planSteps = [];
|
|
495
|
+
const ensureField = (fieldPath) => {
|
|
496
|
+
let field = fieldMap.get(fieldPath);
|
|
497
|
+
if (!field) {
|
|
498
|
+
field = {
|
|
499
|
+
id: fieldPath,
|
|
500
|
+
path: fieldPath,
|
|
501
|
+
traits: []
|
|
502
|
+
};
|
|
503
|
+
fieldMap.set(fieldPath, field);
|
|
504
|
+
}
|
|
505
|
+
return field;
|
|
506
|
+
};
|
|
507
|
+
for (const entry of entries) {
|
|
508
|
+
const fieldPath = entry.fieldPath;
|
|
509
|
+
const field = ensureField(fieldPath);
|
|
510
|
+
const trait = toFieldTrait(entry);
|
|
511
|
+
field.traits.push(trait);
|
|
512
|
+
if (entry.kind === "computed") {
|
|
513
|
+
const stepId = `computed:${fieldPath}`;
|
|
514
|
+
planSteps.push({
|
|
515
|
+
id: stepId,
|
|
516
|
+
kind: "computed-update",
|
|
517
|
+
targetFieldPath: fieldPath
|
|
518
|
+
// Note: the current version does not statically analyze computed dependencies; sourceFieldPaths remains empty.
|
|
519
|
+
});
|
|
520
|
+
const deps = entry.meta.deps;
|
|
521
|
+
if (deps) {
|
|
522
|
+
for (const dep of deps) {
|
|
523
|
+
ensureField(dep);
|
|
524
|
+
edges.push({
|
|
525
|
+
id: `computed:${dep}->${fieldPath}`,
|
|
526
|
+
from: dep,
|
|
527
|
+
to: fieldPath,
|
|
528
|
+
kind: "computed"
|
|
529
|
+
});
|
|
530
|
+
}
|
|
531
|
+
}
|
|
532
|
+
} else if (entry.kind === "link") {
|
|
533
|
+
const from = entry.meta.from;
|
|
534
|
+
ensureField(from);
|
|
535
|
+
const edgeId = `link:${from}->${fieldPath}`;
|
|
536
|
+
edges.push({
|
|
537
|
+
id: edgeId,
|
|
538
|
+
from,
|
|
539
|
+
to: fieldPath,
|
|
540
|
+
kind: "link"
|
|
541
|
+
});
|
|
542
|
+
planSteps.push({
|
|
543
|
+
id: `link:${fieldPath}`,
|
|
544
|
+
kind: "link-propagate",
|
|
545
|
+
targetFieldPath: fieldPath,
|
|
546
|
+
sourceFieldPaths: [from],
|
|
547
|
+
debugInfo: {
|
|
548
|
+
graphEdgeId: edgeId
|
|
549
|
+
}
|
|
550
|
+
});
|
|
551
|
+
} else if (entry.kind === "source") {
|
|
552
|
+
const resourceId = entry.meta.resource;
|
|
553
|
+
const resourceMeta = sanitize(entry.meta.meta);
|
|
554
|
+
const existing = resourcesById.get(resourceId);
|
|
555
|
+
if (existing) {
|
|
556
|
+
const ownerFields = [...existing.ownerFields, fieldPath];
|
|
557
|
+
let meta = existing.meta;
|
|
558
|
+
let metaOrigin = existing.metaOrigin;
|
|
559
|
+
let metaConflicts = existing.metaConflicts;
|
|
560
|
+
if (resourceMeta) {
|
|
561
|
+
const merged = mergeCanonical(
|
|
562
|
+
{ meta, origin: metaOrigin, conflicts: metaConflicts },
|
|
563
|
+
{ origin: fieldPath, meta: resourceMeta }
|
|
564
|
+
);
|
|
565
|
+
meta = merged.meta;
|
|
566
|
+
metaOrigin = merged.origin;
|
|
567
|
+
metaConflicts = merged.conflicts;
|
|
568
|
+
}
|
|
569
|
+
resourcesById.set(resourceId, {
|
|
570
|
+
...existing,
|
|
571
|
+
ownerFields,
|
|
572
|
+
meta,
|
|
573
|
+
metaOrigin,
|
|
574
|
+
metaConflicts
|
|
575
|
+
});
|
|
576
|
+
} else {
|
|
577
|
+
resourcesById.set(resourceId, {
|
|
578
|
+
resourceId,
|
|
579
|
+
// Use a simple identifier string for now; may evolve into a structured form based on key rules.
|
|
580
|
+
keySelector: `StateTrait.source@${fieldPath}`,
|
|
581
|
+
ownerFields: [fieldPath],
|
|
582
|
+
meta: resourceMeta,
|
|
583
|
+
metaOrigin: resourceMeta ? fieldPath : void 0
|
|
584
|
+
});
|
|
585
|
+
}
|
|
586
|
+
planSteps.push({
|
|
587
|
+
id: `source:${fieldPath}`,
|
|
588
|
+
kind: "source-refresh",
|
|
589
|
+
targetFieldPath: fieldPath,
|
|
590
|
+
resourceId,
|
|
591
|
+
keySelectorId: `StateTrait.source@${fieldPath}`
|
|
592
|
+
});
|
|
593
|
+
const deps = entry.meta.deps;
|
|
594
|
+
if (deps) {
|
|
595
|
+
for (const dep of deps) {
|
|
596
|
+
ensureField(dep);
|
|
597
|
+
edges.push({
|
|
598
|
+
id: `source-dep:${dep}->${fieldPath}`,
|
|
599
|
+
from: dep,
|
|
600
|
+
to: fieldPath,
|
|
601
|
+
kind: "source-dep"
|
|
602
|
+
});
|
|
603
|
+
}
|
|
604
|
+
}
|
|
605
|
+
} else if (entry.kind === "externalStore") {
|
|
606
|
+
planSteps.push({
|
|
607
|
+
id: `external-store:${fieldPath}`,
|
|
608
|
+
kind: "external-store-sync",
|
|
609
|
+
targetFieldPath: fieldPath
|
|
610
|
+
});
|
|
611
|
+
} else if (entry.kind === "check") {
|
|
612
|
+
planSteps.push({
|
|
613
|
+
id: `check:${fieldPath}`,
|
|
614
|
+
kind: "check-validate",
|
|
615
|
+
targetFieldPath: fieldPath
|
|
616
|
+
});
|
|
617
|
+
if (trait.deps.length > 0) {
|
|
618
|
+
for (const dep of trait.deps) {
|
|
619
|
+
ensureField(dep);
|
|
620
|
+
edges.push({
|
|
621
|
+
id: `check-dep:${dep}->${fieldPath}`,
|
|
622
|
+
from: dep,
|
|
623
|
+
to: fieldPath,
|
|
624
|
+
kind: "check-dep"
|
|
625
|
+
});
|
|
626
|
+
}
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
}
|
|
630
|
+
for (const field of fieldMap.values()) {
|
|
631
|
+
nodes.push({
|
|
632
|
+
id: field.id,
|
|
633
|
+
field,
|
|
634
|
+
traits: field.traits,
|
|
635
|
+
meta: nodeMetaByFieldPath.get(field.id)
|
|
636
|
+
});
|
|
637
|
+
}
|
|
638
|
+
const graph = {
|
|
639
|
+
_tag: "StateTraitGraph",
|
|
640
|
+
nodes,
|
|
641
|
+
edges,
|
|
642
|
+
resources: Array.from(resourcesById.values())
|
|
643
|
+
};
|
|
644
|
+
const plan = {
|
|
645
|
+
_tag: "StateTraitPlan",
|
|
646
|
+
steps: planSteps
|
|
647
|
+
};
|
|
648
|
+
return { graph, plan };
|
|
649
|
+
};
|
|
650
|
+
var assertNoLinkCycles = (edges) => {
|
|
651
|
+
const adjacency = /* @__PURE__ */ new Map();
|
|
652
|
+
for (const edge of edges) {
|
|
653
|
+
if (edge.kind !== "link") continue;
|
|
654
|
+
const list = adjacency.get(edge.from) ?? [];
|
|
655
|
+
list.push(edge.to);
|
|
656
|
+
adjacency.set(edge.from, list);
|
|
657
|
+
}
|
|
658
|
+
const visited = /* @__PURE__ */ new Set();
|
|
659
|
+
const stack = /* @__PURE__ */ new Set();
|
|
660
|
+
const dfs = (node) => {
|
|
661
|
+
if (stack.has(node)) {
|
|
662
|
+
throw new Error(
|
|
663
|
+
`[StateTrait.build] link cycle detected at field "${node}". Please check link traits for circular dependencies.`
|
|
664
|
+
);
|
|
665
|
+
}
|
|
666
|
+
if (visited.has(node)) return;
|
|
667
|
+
visited.add(node);
|
|
668
|
+
stack.add(node);
|
|
669
|
+
const nexts = adjacency.get(node);
|
|
670
|
+
if (nexts) {
|
|
671
|
+
for (const to of nexts) {
|
|
672
|
+
dfs(to);
|
|
673
|
+
}
|
|
674
|
+
}
|
|
675
|
+
stack.delete(node);
|
|
676
|
+
};
|
|
677
|
+
for (const node of adjacency.keys()) {
|
|
678
|
+
if (!visited.has(node)) {
|
|
679
|
+
dfs(node);
|
|
680
|
+
}
|
|
681
|
+
}
|
|
682
|
+
};
|
|
683
|
+
var collectSchemaPaths = (entries) => {
|
|
684
|
+
const byKey = /* @__PURE__ */ new Map();
|
|
685
|
+
const add = (ref) => {
|
|
686
|
+
if (!ref.path) return;
|
|
687
|
+
const k = `${ref.kind}|${ref.entryKind}|${ref.entryFieldPath}|${ref.ruleName ?? ""}|${ref.path}`;
|
|
688
|
+
byKey.set(k, ref);
|
|
689
|
+
};
|
|
690
|
+
const getCheckWritebackPath = (entry) => {
|
|
691
|
+
const wb = entry.meta?.writeback;
|
|
692
|
+
const p = wb && typeof wb === "object" ? wb.path : void 0;
|
|
693
|
+
const writebackPath = typeof p === "string" && p.startsWith("errors.") ? p : void 0;
|
|
694
|
+
if (writebackPath) return writebackPath;
|
|
695
|
+
const fieldPath = entry.fieldPath;
|
|
696
|
+
if (fieldPath.endsWith("[]")) {
|
|
697
|
+
return `errors.${fieldPath.slice(0, -2)}`;
|
|
698
|
+
}
|
|
699
|
+
return `errors.${fieldPath}`;
|
|
700
|
+
};
|
|
701
|
+
for (const entry of entries) {
|
|
702
|
+
add({
|
|
703
|
+
kind: "fieldPath",
|
|
704
|
+
entryKind: entry.kind,
|
|
705
|
+
entryFieldPath: entry.fieldPath,
|
|
706
|
+
path: entry.fieldPath
|
|
707
|
+
});
|
|
708
|
+
if (entry.kind === "computed" || entry.kind === "source") {
|
|
709
|
+
const deps = entry.meta?.deps ?? [];
|
|
710
|
+
for (const dep of deps) {
|
|
711
|
+
add({
|
|
712
|
+
kind: "dep",
|
|
713
|
+
entryKind: entry.kind,
|
|
714
|
+
entryFieldPath: entry.fieldPath,
|
|
715
|
+
path: dep
|
|
716
|
+
});
|
|
717
|
+
}
|
|
718
|
+
}
|
|
719
|
+
if (entry.kind === "link") {
|
|
720
|
+
add({
|
|
721
|
+
kind: "link_from",
|
|
722
|
+
entryKind: "link",
|
|
723
|
+
entryFieldPath: entry.fieldPath,
|
|
724
|
+
path: entry.meta.from
|
|
725
|
+
});
|
|
726
|
+
}
|
|
727
|
+
if (entry.kind === "check") {
|
|
728
|
+
add({
|
|
729
|
+
kind: "check_writeback",
|
|
730
|
+
entryKind: "check",
|
|
731
|
+
entryFieldPath: entry.fieldPath,
|
|
732
|
+
path: getCheckWritebackPath(entry)
|
|
733
|
+
});
|
|
734
|
+
const rules = entry.meta?.rules ?? {};
|
|
735
|
+
for (const name of Object.keys(rules)) {
|
|
736
|
+
const rule = rules[name];
|
|
737
|
+
if (!rule || typeof rule !== "object") continue;
|
|
738
|
+
const deps = rule.deps ?? [];
|
|
739
|
+
for (const dep of deps) {
|
|
740
|
+
add({
|
|
741
|
+
kind: "dep",
|
|
742
|
+
entryKind: "check",
|
|
743
|
+
entryFieldPath: entry.fieldPath,
|
|
744
|
+
ruleName: name,
|
|
745
|
+
path: dep
|
|
746
|
+
});
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
}
|
|
750
|
+
}
|
|
751
|
+
return Array.from(byKey.entries()).sort((a, b) => a[0] < b[0] ? -1 : a[0] > b[0] ? 1 : 0).map(([, v]) => v);
|
|
752
|
+
};
|
|
753
|
+
var build = (stateSchema, spec) => {
|
|
754
|
+
const entries = normalizeSpec(spec);
|
|
755
|
+
const nodeMetaByFieldPath = collectNodeMeta(spec);
|
|
756
|
+
for (const entry of entries) {
|
|
757
|
+
if (entry.kind === "computed") {
|
|
758
|
+
const deps = entry.meta.deps;
|
|
759
|
+
if (deps === void 0) {
|
|
760
|
+
throw new Error(
|
|
761
|
+
`[StateTrait.build] Missing explicit deps for computed "${entry.fieldPath}". Please use StateTrait.computed({ deps: [...], get: ... }).`
|
|
762
|
+
);
|
|
763
|
+
}
|
|
764
|
+
}
|
|
765
|
+
if (entry.kind === "source") {
|
|
766
|
+
const deps = entry.meta.deps;
|
|
767
|
+
if (deps === void 0) {
|
|
768
|
+
throw new Error(
|
|
769
|
+
`[StateTrait.build] Missing explicit deps for source "${entry.fieldPath}". Please provide meta.deps for StateTrait.source({ deps: [...], ... }).`
|
|
770
|
+
);
|
|
771
|
+
}
|
|
772
|
+
}
|
|
773
|
+
if (entry.kind === "check") {
|
|
774
|
+
const rules = entry.meta?.rules ?? {};
|
|
775
|
+
for (const name of Object.keys(rules)) {
|
|
776
|
+
const rule = rules[name];
|
|
777
|
+
if (typeof rule === "function" || !rule || typeof rule !== "object") {
|
|
778
|
+
throw new Error(
|
|
779
|
+
`[StateTrait.build] Missing explicit deps for check "${entry.fieldPath}" rule "${name}". Please use { deps: [...], validate: ... } form.`
|
|
780
|
+
);
|
|
781
|
+
}
|
|
782
|
+
if (rule.deps === void 0) {
|
|
783
|
+
throw new Error(
|
|
784
|
+
`[StateTrait.build] Missing explicit deps for check "${entry.fieldPath}" rule "${name}". Please provide deps: [...].`
|
|
785
|
+
);
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
}
|
|
789
|
+
}
|
|
790
|
+
const { graph, plan } = buildGraph(entries, nodeMetaByFieldPath);
|
|
791
|
+
assertNoLinkCycles(graph.edges);
|
|
792
|
+
return {
|
|
793
|
+
stateSchema,
|
|
794
|
+
spec,
|
|
795
|
+
entries,
|
|
796
|
+
graph,
|
|
797
|
+
plan,
|
|
798
|
+
convergeIr: buildConvergeIr(stateSchema, entries),
|
|
799
|
+
schemaPaths: collectSchemaPaths(entries)
|
|
800
|
+
};
|
|
801
|
+
};
|
|
802
|
+
|
|
803
|
+
// ../logix-core/src/internal/state-trait/source.impl.ts
|
|
804
|
+
import { Effect, Fiber, FiberRef, Option } from "../effect.js";
|
|
805
|
+
|
|
806
|
+
// ../logix-core/src/internal/state-trait/deps-trace.ts
|
|
807
|
+
var isTraceableObject = (value) => {
|
|
808
|
+
if (!value || typeof value !== "object") return false;
|
|
809
|
+
if (Array.isArray(value)) return true;
|
|
810
|
+
if (value instanceof Date) return false;
|
|
811
|
+
if (value instanceof RegExp) return false;
|
|
812
|
+
if (value instanceof Error) return false;
|
|
813
|
+
if (value instanceof Map) return false;
|
|
814
|
+
if (value instanceof Set) return false;
|
|
815
|
+
if (value instanceof WeakMap) return false;
|
|
816
|
+
if (value instanceof WeakSet) return false;
|
|
817
|
+
return true;
|
|
818
|
+
};
|
|
819
|
+
var shouldIgnoreKey = (key) => key === "__proto__" || key === "prototype" || key === "constructor";
|
|
820
|
+
var normalizeReads = (reads) => {
|
|
821
|
+
const all = Array.from(reads).filter((p) => typeof p === "string" && p.length > 0);
|
|
822
|
+
all.sort();
|
|
823
|
+
const isPrefix = (prefix, full) => full !== prefix && full.startsWith(prefix + ".");
|
|
824
|
+
const pruned = [];
|
|
825
|
+
for (const p of all) {
|
|
826
|
+
let hasMoreSpecific = false;
|
|
827
|
+
for (const other of all) {
|
|
828
|
+
if (isPrefix(p, other)) {
|
|
829
|
+
hasMoreSpecific = true;
|
|
830
|
+
break;
|
|
831
|
+
}
|
|
832
|
+
}
|
|
833
|
+
if (!hasMoreSpecific) {
|
|
834
|
+
pruned.push(p);
|
|
835
|
+
}
|
|
836
|
+
}
|
|
837
|
+
pruned.sort();
|
|
838
|
+
return pruned;
|
|
839
|
+
};
|
|
840
|
+
var covers = (declared, read) => declared === read || read.startsWith(declared + ".");
|
|
841
|
+
var diffDeps = (declared, reads) => {
|
|
842
|
+
const declaredList = Array.from(new Set(declared)).filter((p) => typeof p === "string" && p.length > 0);
|
|
843
|
+
declaredList.sort();
|
|
844
|
+
const readList = Array.from(new Set(reads)).filter((p) => typeof p === "string" && p.length > 0);
|
|
845
|
+
readList.sort();
|
|
846
|
+
const missing = readList.filter((r) => declaredList.every((d) => !covers(d, r)));
|
|
847
|
+
const unused = declaredList.filter((d) => readList.every((r) => !covers(d, r)));
|
|
848
|
+
if (missing.length === 0 && unused.length === 0) return void 0;
|
|
849
|
+
return {
|
|
850
|
+
reads: readList,
|
|
851
|
+
declared: declaredList,
|
|
852
|
+
missing,
|
|
853
|
+
unused
|
|
854
|
+
};
|
|
855
|
+
};
|
|
856
|
+
var trace = (fn, state) => {
|
|
857
|
+
if (!isTraceableObject(state)) {
|
|
858
|
+
return { value: fn(state), reads: [] };
|
|
859
|
+
}
|
|
860
|
+
const reads = /* @__PURE__ */ new Set();
|
|
861
|
+
const proxyCache = /* @__PURE__ */ new WeakMap();
|
|
862
|
+
const proxyToTarget = /* @__PURE__ */ new WeakMap();
|
|
863
|
+
const wrap = (value2, path) => {
|
|
864
|
+
if (!isTraceableObject(value2)) return value2;
|
|
865
|
+
return getProxy(value2, path);
|
|
866
|
+
};
|
|
867
|
+
const unwrap = (value2) => {
|
|
868
|
+
if (value2 && (typeof value2 === "object" || typeof value2 === "function")) {
|
|
869
|
+
const target = proxyToTarget.get(value2);
|
|
870
|
+
if (target) return target;
|
|
871
|
+
}
|
|
872
|
+
return value2;
|
|
873
|
+
};
|
|
874
|
+
const getProxy = (target, basePath) => {
|
|
875
|
+
let byPath = proxyCache.get(target);
|
|
876
|
+
if (!byPath) {
|
|
877
|
+
byPath = /* @__PURE__ */ new Map();
|
|
878
|
+
proxyCache.set(target, byPath);
|
|
879
|
+
}
|
|
880
|
+
const cached = byPath.get(basePath);
|
|
881
|
+
if (cached) return cached;
|
|
882
|
+
const record2 = (path) => {
|
|
883
|
+
if (path) reads.add(path);
|
|
884
|
+
};
|
|
885
|
+
const proxy = new Proxy(target, {
|
|
886
|
+
get: (t, prop, receiver) => {
|
|
887
|
+
if (typeof prop === "symbol") {
|
|
888
|
+
return Reflect.get(t, prop, receiver);
|
|
889
|
+
}
|
|
890
|
+
const key = String(prop);
|
|
891
|
+
if (shouldIgnoreKey(key)) {
|
|
892
|
+
return Reflect.get(t, prop, receiver);
|
|
893
|
+
}
|
|
894
|
+
const nextPath = basePath ? `${basePath}.${key}` : key;
|
|
895
|
+
record2(nextPath);
|
|
896
|
+
const value2 = Reflect.get(t, prop, receiver);
|
|
897
|
+
return wrap(value2, nextPath);
|
|
898
|
+
},
|
|
899
|
+
has: (t, prop) => {
|
|
900
|
+
if (typeof prop === "symbol") return Reflect.has(t, prop);
|
|
901
|
+
const key = String(prop);
|
|
902
|
+
if (!shouldIgnoreKey(key)) {
|
|
903
|
+
const nextPath = basePath ? `${basePath}.${key}` : key;
|
|
904
|
+
record2(nextPath);
|
|
905
|
+
}
|
|
906
|
+
return Reflect.has(t, prop);
|
|
907
|
+
},
|
|
908
|
+
ownKeys: (t) => {
|
|
909
|
+
if (basePath) record2(basePath);
|
|
910
|
+
return Reflect.ownKeys(t);
|
|
911
|
+
},
|
|
912
|
+
getOwnPropertyDescriptor: (t, prop) => {
|
|
913
|
+
if (typeof prop === "symbol") {
|
|
914
|
+
return Reflect.getOwnPropertyDescriptor(t, prop);
|
|
915
|
+
}
|
|
916
|
+
const key = String(prop);
|
|
917
|
+
if (!shouldIgnoreKey(key)) {
|
|
918
|
+
const nextPath = basePath ? `${basePath}.${key}` : key;
|
|
919
|
+
record2(nextPath);
|
|
920
|
+
}
|
|
921
|
+
return Reflect.getOwnPropertyDescriptor(t, prop);
|
|
922
|
+
},
|
|
923
|
+
set: () => {
|
|
924
|
+
throw new Error(
|
|
925
|
+
"[deps-trace] Attempted to mutate state during deps tracing (state is readonly in dev-mode diagnostics)."
|
|
926
|
+
);
|
|
927
|
+
},
|
|
928
|
+
defineProperty: () => {
|
|
929
|
+
throw new Error(
|
|
930
|
+
"[deps-trace] Attempted to define property on state during deps tracing (state is readonly in dev-mode diagnostics)."
|
|
931
|
+
);
|
|
932
|
+
},
|
|
933
|
+
deleteProperty: () => {
|
|
934
|
+
throw new Error(
|
|
935
|
+
"[deps-trace] Attempted to delete property on state during deps tracing (state is readonly in dev-mode diagnostics)."
|
|
936
|
+
);
|
|
937
|
+
}
|
|
938
|
+
});
|
|
939
|
+
byPath.set(basePath, proxy);
|
|
940
|
+
proxyToTarget.set(proxy, target);
|
|
941
|
+
return proxy;
|
|
942
|
+
};
|
|
943
|
+
const root = getProxy(state, "");
|
|
944
|
+
const value = unwrap(fn(root));
|
|
945
|
+
return {
|
|
946
|
+
value,
|
|
947
|
+
reads: normalizeReads(reads)
|
|
948
|
+
};
|
|
949
|
+
};
|
|
950
|
+
|
|
951
|
+
// ../logix-core/src/internal/state-trait/source.impl.ts
|
|
952
|
+
var onceInRunSession = (key) => Effect.serviceOption(RunSessionTag).pipe(
|
|
953
|
+
Effect.map((maybe) => Option.isSome(maybe) ? maybe.value.local.once(key) : true)
|
|
954
|
+
);
|
|
955
|
+
var formatList = (items, limit = 10) => {
|
|
956
|
+
if (items.length === 0) return "";
|
|
957
|
+
if (items.length <= limit) return items.join(", ");
|
|
958
|
+
return `${items.slice(0, limit).join(", ")}, \u2026(+${items.length - limit})`;
|
|
959
|
+
};
|
|
960
|
+
var emitDepsMismatch = (params) => Effect.gen(function* () {
|
|
961
|
+
const key = `${params.instanceId ?? "unknown"}::${params.kind}::${params.fieldPath}`;
|
|
962
|
+
const shouldEmit = yield* onceInRunSession(`deps_mismatch:${key}`);
|
|
963
|
+
if (!shouldEmit) return;
|
|
964
|
+
yield* record({
|
|
965
|
+
type: "diagnostic",
|
|
966
|
+
moduleId: params.moduleId,
|
|
967
|
+
instanceId: params.instanceId,
|
|
968
|
+
code: "state_trait::deps_mismatch",
|
|
969
|
+
severity: "warning",
|
|
970
|
+
message: `[deps] ${params.kind} "${params.fieldPath}" declared=[${formatList(params.diff.declared)}] reads=[${formatList(params.diff.reads)}] missing=[${formatList(params.diff.missing)}] unused=[${formatList(params.diff.unused)}]`,
|
|
971
|
+
hint: 'deps is the single source of truth for dependencies: incremental scheduling / reverse closures / performance optimizations rely on deps only. Keep deps consistent with actual reads; if you really depend on the whole object, declare a coarser-grained dep (e.g. "profile") to cover sub-field reads.',
|
|
972
|
+
kind: `deps_mismatch:${params.kind}`
|
|
973
|
+
});
|
|
974
|
+
});
|
|
975
|
+
var getMiddlewareStack = () => Effect.serviceOption(EffectOpMiddlewareTag).pipe(
|
|
976
|
+
Effect.map((maybe) => Option.isSome(maybe) ? maybe.value.stack : [])
|
|
977
|
+
);
|
|
978
|
+
var recordTraitPatch = (bound, path, reason, from, to, traitNodeId) => {
|
|
979
|
+
const normalized = normalizeFieldPath(path) ?? [];
|
|
980
|
+
try {
|
|
981
|
+
const internals = getBoundInternals(bound);
|
|
982
|
+
internals.txn.recordStatePatch(normalized, reason, from, to, traitNodeId);
|
|
983
|
+
} catch {
|
|
984
|
+
}
|
|
985
|
+
};
|
|
986
|
+
var recordReplayEvent = (bound, event) => {
|
|
987
|
+
try {
|
|
988
|
+
const internals = getBoundInternals(bound);
|
|
989
|
+
internals.txn.recordReplayEvent(event);
|
|
990
|
+
} catch {
|
|
991
|
+
}
|
|
992
|
+
};
|
|
993
|
+
var getBoundScope = (bound) => {
|
|
994
|
+
try {
|
|
995
|
+
const internals = getBoundInternals(bound);
|
|
996
|
+
return { moduleId: internals.moduleId, instanceId: internals.instanceId };
|
|
997
|
+
} catch {
|
|
998
|
+
return { moduleId: void 0, instanceId: void 0 };
|
|
999
|
+
}
|
|
1000
|
+
};
|
|
1001
|
+
var setSnapshotInTxn = (bound, fieldPath, next, reason, stepId, traitNodeId) => Effect.gen(function* () {
|
|
1002
|
+
let wrote = false;
|
|
1003
|
+
yield* bound.state.mutate((draft) => {
|
|
1004
|
+
const prev = getAtPath(draft, fieldPath);
|
|
1005
|
+
if (Object.is(prev, next)) return;
|
|
1006
|
+
wrote = true;
|
|
1007
|
+
setAtPathMutating(draft, fieldPath, next);
|
|
1008
|
+
recordTraitPatch(bound, fieldPath, reason, prev, next, traitNodeId);
|
|
1009
|
+
});
|
|
1010
|
+
return wrote;
|
|
1011
|
+
});
|
|
1012
|
+
var writebackIfCurrentKeyHash = (bound, fieldPath, keyHash2, next, reason, stepId, traitNodeId, replayEvent) => Effect.gen(function* () {
|
|
1013
|
+
let wrote = false;
|
|
1014
|
+
yield* bound.state.mutate((draft) => {
|
|
1015
|
+
const current = getAtPath(draft, fieldPath);
|
|
1016
|
+
const currentKeyHash = current && typeof current === "object" ? current.keyHash : void 0;
|
|
1017
|
+
if (currentKeyHash !== keyHash2) return;
|
|
1018
|
+
const prev = current;
|
|
1019
|
+
if (Object.is(prev, next)) return;
|
|
1020
|
+
wrote = true;
|
|
1021
|
+
setAtPathMutating(draft, fieldPath, next);
|
|
1022
|
+
if (replayEvent) {
|
|
1023
|
+
recordReplayEvent(bound, replayEvent);
|
|
1024
|
+
}
|
|
1025
|
+
recordTraitPatch(bound, fieldPath, reason, prev, next, traitNodeId);
|
|
1026
|
+
});
|
|
1027
|
+
return wrote;
|
|
1028
|
+
});
|
|
1029
|
+
var syncIdleInTransaction = (program, ctx) => Effect.sync(() => {
|
|
1030
|
+
const draft = ctx.getDraft();
|
|
1031
|
+
const updates = [];
|
|
1032
|
+
for (const entry of program.entries) {
|
|
1033
|
+
if (entry.kind !== "source") continue;
|
|
1034
|
+
const fieldPath = entry.fieldPath;
|
|
1035
|
+
const listItem = parseListItemFieldPath(fieldPath);
|
|
1036
|
+
if (listItem) {
|
|
1037
|
+
const listValue = getAtPath(draft, listItem.listPath);
|
|
1038
|
+
const items = Array.isArray(listValue) ? listValue : [];
|
|
1039
|
+
for (let index = 0; index < items.length; index++) {
|
|
1040
|
+
const item = items[index];
|
|
1041
|
+
let key2;
|
|
1042
|
+
try {
|
|
1043
|
+
key2 = entry.meta.key(item);
|
|
1044
|
+
} catch {
|
|
1045
|
+
continue;
|
|
1046
|
+
}
|
|
1047
|
+
if (key2 !== void 0) continue;
|
|
1048
|
+
const concretePath = toListItemValuePath(listItem.listPath, index, listItem.itemPath);
|
|
1049
|
+
const prev2 = getAtPath(draft, concretePath);
|
|
1050
|
+
const prevStatus2 = prev2 && typeof prev2 === "object" ? prev2.status : void 0;
|
|
1051
|
+
if (prevStatus2 === "idle") {
|
|
1052
|
+
const data = prev2?.data;
|
|
1053
|
+
const error = prev2?.error;
|
|
1054
|
+
if (data === void 0 && error === void 0) {
|
|
1055
|
+
continue;
|
|
1056
|
+
}
|
|
1057
|
+
}
|
|
1058
|
+
updates.push({ fieldPath: concretePath, prev: prev2 });
|
|
1059
|
+
}
|
|
1060
|
+
continue;
|
|
1061
|
+
}
|
|
1062
|
+
let key;
|
|
1063
|
+
try {
|
|
1064
|
+
key = entry.meta.key(draft);
|
|
1065
|
+
} catch {
|
|
1066
|
+
continue;
|
|
1067
|
+
}
|
|
1068
|
+
if (key !== void 0) continue;
|
|
1069
|
+
const prev = getAtPath(draft, fieldPath);
|
|
1070
|
+
const prevStatus = prev && typeof prev === "object" ? prev.status : void 0;
|
|
1071
|
+
if (prevStatus === "idle") {
|
|
1072
|
+
const data = prev?.data;
|
|
1073
|
+
const error = prev?.error;
|
|
1074
|
+
if (data === void 0 && error === void 0) {
|
|
1075
|
+
continue;
|
|
1076
|
+
}
|
|
1077
|
+
}
|
|
1078
|
+
updates.push({ fieldPath, prev });
|
|
1079
|
+
}
|
|
1080
|
+
if (updates.length === 0) return;
|
|
1081
|
+
const reason = "source-refresh";
|
|
1082
|
+
const nextDraft = create(draft, (next) => {
|
|
1083
|
+
for (const u of updates) {
|
|
1084
|
+
setAtPathMutating(next, u.fieldPath, Snapshot.idle());
|
|
1085
|
+
}
|
|
1086
|
+
});
|
|
1087
|
+
ctx.setDraft(nextDraft);
|
|
1088
|
+
for (const u of updates) {
|
|
1089
|
+
const normalized = normalizeFieldPath(u.fieldPath) ?? [];
|
|
1090
|
+
ctx.recordPatch(normalized, reason, u.prev, Snapshot.idle(), `source:${u.fieldPath}:idle`);
|
|
1091
|
+
}
|
|
1092
|
+
});
|
|
1093
|
+
var installSourceRefresh = (bound, step, entry) => {
|
|
1094
|
+
if (!step.targetFieldPath) return Effect.void;
|
|
1095
|
+
const fieldPath = step.targetFieldPath;
|
|
1096
|
+
const resourceId = step.resourceId ?? entry.meta.resource;
|
|
1097
|
+
const listItem = parseListItemFieldPath(fieldPath);
|
|
1098
|
+
let internals;
|
|
1099
|
+
try {
|
|
1100
|
+
internals = getBoundInternals(bound);
|
|
1101
|
+
} catch {
|
|
1102
|
+
return Effect.void;
|
|
1103
|
+
}
|
|
1104
|
+
const register = internals.traits.registerSourceRefresh;
|
|
1105
|
+
const recordSnapshot = (replayMode, replayLog, input) => {
|
|
1106
|
+
if (!replayLog) return Effect.void;
|
|
1107
|
+
if (replayMode !== "live") return Effect.void;
|
|
1108
|
+
const event = input && typeof input === "object" && input._tag === "ResourceSnapshot" ? input : {
|
|
1109
|
+
_tag: "ResourceSnapshot",
|
|
1110
|
+
resourceId,
|
|
1111
|
+
fieldPath: input.fieldPath,
|
|
1112
|
+
keyHash: input.keyHash,
|
|
1113
|
+
concurrency: input.concurrency,
|
|
1114
|
+
phase: input.phase,
|
|
1115
|
+
snapshot: input.snapshot,
|
|
1116
|
+
timestamp: Date.now(),
|
|
1117
|
+
moduleId: input.moduleId,
|
|
1118
|
+
instanceId: input.instanceId
|
|
1119
|
+
};
|
|
1120
|
+
return replayLog.record(event);
|
|
1121
|
+
};
|
|
1122
|
+
if (listItem) {
|
|
1123
|
+
const store = internals.traits.rowIdStore;
|
|
1124
|
+
if (!store) {
|
|
1125
|
+
return Effect.void;
|
|
1126
|
+
}
|
|
1127
|
+
const listPath = listItem.listPath;
|
|
1128
|
+
const itemPath = listItem.itemPath;
|
|
1129
|
+
if (!itemPath) {
|
|
1130
|
+
return Effect.void;
|
|
1131
|
+
}
|
|
1132
|
+
const concurrency2 = entry.meta.concurrency;
|
|
1133
|
+
const mode2 = concurrency2 ?? "switch";
|
|
1134
|
+
const inFlight2 = /* @__PURE__ */ new Map();
|
|
1135
|
+
const trailing2 = /* @__PURE__ */ new Map();
|
|
1136
|
+
let gen2 = 0;
|
|
1137
|
+
store.onRemoved(listPath, (rowId) => {
|
|
1138
|
+
trailing2.delete(rowId);
|
|
1139
|
+
inFlight2.delete(rowId);
|
|
1140
|
+
});
|
|
1141
|
+
const setSnapshotForRowInTxn = (rowId, next, reason, stepId) => Effect.gen(function* () {
|
|
1142
|
+
let wrotePath;
|
|
1143
|
+
yield* bound.state.mutate((draft) => {
|
|
1144
|
+
const index = store.getIndex(listPath, rowId);
|
|
1145
|
+
if (index === void 0) return;
|
|
1146
|
+
const concretePath = toListItemValuePath(listPath, index, itemPath);
|
|
1147
|
+
const prev = getAtPath(draft, concretePath);
|
|
1148
|
+
if (Object.is(prev, next)) return;
|
|
1149
|
+
wrotePath = concretePath;
|
|
1150
|
+
setAtPathMutating(draft, concretePath, next);
|
|
1151
|
+
recordTraitPatch(bound, concretePath, reason, prev, next, step.debugInfo?.graphNodeId);
|
|
1152
|
+
});
|
|
1153
|
+
return wrotePath;
|
|
1154
|
+
});
|
|
1155
|
+
const writebackIfCurrentKeyHashForRow = (rowId, keyHash2, next, reason, stepId, phase) => Effect.gen(function* () {
|
|
1156
|
+
let wrotePath;
|
|
1157
|
+
yield* bound.state.mutate((draft) => {
|
|
1158
|
+
const index = store.getIndex(listPath, rowId);
|
|
1159
|
+
if (index === void 0) return;
|
|
1160
|
+
const concretePath = toListItemValuePath(listPath, index, itemPath);
|
|
1161
|
+
const current = getAtPath(draft, concretePath);
|
|
1162
|
+
const currentKeyHash = current && typeof current === "object" ? current.keyHash : void 0;
|
|
1163
|
+
if (currentKeyHash !== keyHash2) return;
|
|
1164
|
+
const prev = current;
|
|
1165
|
+
if (Object.is(prev, next)) return;
|
|
1166
|
+
wrotePath = concretePath;
|
|
1167
|
+
setAtPathMutating(draft, concretePath, next);
|
|
1168
|
+
if (phase) {
|
|
1169
|
+
const { moduleId, instanceId } = getBoundScope(bound);
|
|
1170
|
+
recordReplayEvent(bound, {
|
|
1171
|
+
_tag: "ResourceSnapshot",
|
|
1172
|
+
resourceId,
|
|
1173
|
+
fieldPath: concretePath,
|
|
1174
|
+
keyHash: keyHash2,
|
|
1175
|
+
concurrency: mode2,
|
|
1176
|
+
phase,
|
|
1177
|
+
snapshot: next,
|
|
1178
|
+
timestamp: Date.now(),
|
|
1179
|
+
moduleId,
|
|
1180
|
+
instanceId
|
|
1181
|
+
});
|
|
1182
|
+
}
|
|
1183
|
+
recordTraitPatch(bound, concretePath, reason, prev, next, step.debugInfo?.graphNodeId);
|
|
1184
|
+
});
|
|
1185
|
+
return wrotePath;
|
|
1186
|
+
});
|
|
1187
|
+
const startFetch2 = (rowId, key, keyHash2, replayMode, replayLog) => Effect.gen(function* () {
|
|
1188
|
+
const { moduleId, instanceId } = getBoundScope(bound);
|
|
1189
|
+
const indexForLog = store.getIndex(listPath, rowId);
|
|
1190
|
+
const logFieldPath = indexForLog === void 0 ? void 0 : toListItemValuePath(listPath, indexForLog, itemPath);
|
|
1191
|
+
let loadingSnapshot = Snapshot.loading({ keyHash: keyHash2 });
|
|
1192
|
+
if (replayMode === "replay" && replayLog && logFieldPath) {
|
|
1193
|
+
const replayLoading = yield* replayLog.consumeNextResourceSnapshot({
|
|
1194
|
+
resourceId,
|
|
1195
|
+
fieldPath: logFieldPath,
|
|
1196
|
+
keyHash: keyHash2,
|
|
1197
|
+
phase: "loading"
|
|
1198
|
+
});
|
|
1199
|
+
if (replayLoading) {
|
|
1200
|
+
loadingSnapshot = replayLoading.snapshot;
|
|
1201
|
+
}
|
|
1202
|
+
}
|
|
1203
|
+
const wroteLoadingPath = yield* setSnapshotForRowInTxn(
|
|
1204
|
+
rowId,
|
|
1205
|
+
loadingSnapshot,
|
|
1206
|
+
"source-refresh",
|
|
1207
|
+
`source:${fieldPath}:${rowId}:loading`
|
|
1208
|
+
);
|
|
1209
|
+
if (wroteLoadingPath) {
|
|
1210
|
+
const event = {
|
|
1211
|
+
_tag: "ResourceSnapshot",
|
|
1212
|
+
resourceId,
|
|
1213
|
+
fieldPath: wroteLoadingPath,
|
|
1214
|
+
keyHash: keyHash2,
|
|
1215
|
+
concurrency: mode2,
|
|
1216
|
+
phase: "loading",
|
|
1217
|
+
snapshot: loadingSnapshot,
|
|
1218
|
+
timestamp: Date.now(),
|
|
1219
|
+
moduleId,
|
|
1220
|
+
instanceId
|
|
1221
|
+
};
|
|
1222
|
+
recordReplayEvent(bound, event);
|
|
1223
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1224
|
+
}
|
|
1225
|
+
const io = Effect.gen(function* () {
|
|
1226
|
+
if (replayMode === "replay" && replayLog) {
|
|
1227
|
+
yield* Effect.yieldNow();
|
|
1228
|
+
const consumePath = wroteLoadingPath ?? logFieldPath;
|
|
1229
|
+
if (!consumePath) return yield* Effect.void;
|
|
1230
|
+
const replayed = yield* replayLog.consumeNextResourceSnapshot({
|
|
1231
|
+
resourceId,
|
|
1232
|
+
fieldPath: consumePath,
|
|
1233
|
+
keyHash: keyHash2
|
|
1234
|
+
});
|
|
1235
|
+
if (!replayed) return yield* Effect.void;
|
|
1236
|
+
if (replayed.phase === "success") {
|
|
1237
|
+
yield* writebackIfCurrentKeyHashForRow(
|
|
1238
|
+
rowId,
|
|
1239
|
+
keyHash2,
|
|
1240
|
+
replayed.snapshot,
|
|
1241
|
+
"source-refresh",
|
|
1242
|
+
`source:${fieldPath}:${rowId}:success`,
|
|
1243
|
+
"success"
|
|
1244
|
+
);
|
|
1245
|
+
} else if (replayed.phase === "error") {
|
|
1246
|
+
yield* writebackIfCurrentKeyHashForRow(
|
|
1247
|
+
rowId,
|
|
1248
|
+
keyHash2,
|
|
1249
|
+
replayed.snapshot,
|
|
1250
|
+
"source-refresh",
|
|
1251
|
+
`source:${fieldPath}:${rowId}:error`,
|
|
1252
|
+
"error"
|
|
1253
|
+
);
|
|
1254
|
+
}
|
|
1255
|
+
return yield* Effect.void;
|
|
1256
|
+
}
|
|
1257
|
+
const stack = yield* getMiddlewareStack();
|
|
1258
|
+
const registryOpt = yield* Effect.serviceOption(internal.ResourceRegistryTag);
|
|
1259
|
+
const registry = Option.isSome(registryOpt) ? registryOpt.value : void 0;
|
|
1260
|
+
const spec = registry?.specs.get(resourceId);
|
|
1261
|
+
if (!spec) {
|
|
1262
|
+
return yield* Effect.void;
|
|
1263
|
+
}
|
|
1264
|
+
const loadEffect = spec.load(key);
|
|
1265
|
+
const meta = {
|
|
1266
|
+
moduleId,
|
|
1267
|
+
instanceId,
|
|
1268
|
+
fieldPath,
|
|
1269
|
+
resourceId,
|
|
1270
|
+
key,
|
|
1271
|
+
keyHash: keyHash2,
|
|
1272
|
+
rowId,
|
|
1273
|
+
traitNodeId: step.debugInfo?.graphNodeId,
|
|
1274
|
+
stepId: step.id
|
|
1275
|
+
};
|
|
1276
|
+
if (!(typeof meta.opSeq === "number" && Number.isFinite(meta.opSeq))) {
|
|
1277
|
+
const sessionOpt = yield* Effect.serviceOption(RunSessionTag);
|
|
1278
|
+
if (Option.isSome(sessionOpt)) {
|
|
1279
|
+
const seqKey = instanceId ?? "global";
|
|
1280
|
+
meta.opSeq = sessionOpt.value.local.nextSeq("opSeq", seqKey);
|
|
1281
|
+
}
|
|
1282
|
+
}
|
|
1283
|
+
const op = make({
|
|
1284
|
+
kind: "service",
|
|
1285
|
+
name: resourceId,
|
|
1286
|
+
effect: loadEffect,
|
|
1287
|
+
meta
|
|
1288
|
+
});
|
|
1289
|
+
const exit = yield* Effect.exit(run(op, stack));
|
|
1290
|
+
if (exit._tag === "Success") {
|
|
1291
|
+
const successSnapshot = Snapshot.success({ keyHash: keyHash2, data: exit.value });
|
|
1292
|
+
const wroteSuccessPath = yield* writebackIfCurrentKeyHashForRow(
|
|
1293
|
+
rowId,
|
|
1294
|
+
keyHash2,
|
|
1295
|
+
successSnapshot,
|
|
1296
|
+
"source-refresh",
|
|
1297
|
+
`source:${fieldPath}:${rowId}:success`,
|
|
1298
|
+
"success"
|
|
1299
|
+
);
|
|
1300
|
+
if (wroteSuccessPath) {
|
|
1301
|
+
yield* recordSnapshot(replayMode, replayLog, {
|
|
1302
|
+
_tag: "ResourceSnapshot",
|
|
1303
|
+
resourceId,
|
|
1304
|
+
fieldPath: wroteSuccessPath,
|
|
1305
|
+
keyHash: keyHash2,
|
|
1306
|
+
concurrency: mode2,
|
|
1307
|
+
phase: "success",
|
|
1308
|
+
snapshot: successSnapshot,
|
|
1309
|
+
timestamp: Date.now(),
|
|
1310
|
+
moduleId,
|
|
1311
|
+
instanceId
|
|
1312
|
+
});
|
|
1313
|
+
}
|
|
1314
|
+
} else {
|
|
1315
|
+
const errorSnapshot = Snapshot.error({ keyHash: keyHash2, error: exit.cause });
|
|
1316
|
+
const wroteErrorPath = yield* writebackIfCurrentKeyHashForRow(
|
|
1317
|
+
rowId,
|
|
1318
|
+
keyHash2,
|
|
1319
|
+
errorSnapshot,
|
|
1320
|
+
"source-refresh",
|
|
1321
|
+
`source:${fieldPath}:${rowId}:error`,
|
|
1322
|
+
"error"
|
|
1323
|
+
);
|
|
1324
|
+
if (wroteErrorPath) {
|
|
1325
|
+
yield* recordSnapshot(replayMode, replayLog, {
|
|
1326
|
+
_tag: "ResourceSnapshot",
|
|
1327
|
+
resourceId,
|
|
1328
|
+
fieldPath: wroteErrorPath,
|
|
1329
|
+
keyHash: keyHash2,
|
|
1330
|
+
concurrency: mode2,
|
|
1331
|
+
phase: "error",
|
|
1332
|
+
snapshot: errorSnapshot,
|
|
1333
|
+
timestamp: Date.now(),
|
|
1334
|
+
moduleId,
|
|
1335
|
+
instanceId
|
|
1336
|
+
});
|
|
1337
|
+
}
|
|
1338
|
+
}
|
|
1339
|
+
}).pipe(Effect.catchAllCause(() => Effect.void));
|
|
1340
|
+
const fiber = yield* Effect.forkScoped(Effect.locally(inSyncTransactionFiber, false)(io));
|
|
1341
|
+
const myGen = gen2 += 1;
|
|
1342
|
+
inFlight2.set(rowId, { gen: myGen, fiber, keyHash: keyHash2 });
|
|
1343
|
+
yield* Effect.forkScoped(
|
|
1344
|
+
Effect.locally(
|
|
1345
|
+
inSyncTransactionFiber,
|
|
1346
|
+
false
|
|
1347
|
+
)(
|
|
1348
|
+
Fiber.await(fiber).pipe(
|
|
1349
|
+
Effect.zipRight(
|
|
1350
|
+
Effect.sync(() => {
|
|
1351
|
+
const current = inFlight2.get(rowId);
|
|
1352
|
+
if (current && current.gen === myGen) {
|
|
1353
|
+
inFlight2.delete(rowId);
|
|
1354
|
+
}
|
|
1355
|
+
})
|
|
1356
|
+
),
|
|
1357
|
+
Effect.zipRight(
|
|
1358
|
+
mode2 === "exhaust-trailing" ? Effect.gen(function* () {
|
|
1359
|
+
const next = trailing2.get(rowId);
|
|
1360
|
+
trailing2.delete(rowId);
|
|
1361
|
+
if (next) {
|
|
1362
|
+
yield* startFetch2(rowId, next.key, next.keyHash, replayMode, replayLog);
|
|
1363
|
+
}
|
|
1364
|
+
}) : Effect.void
|
|
1365
|
+
),
|
|
1366
|
+
Effect.catchAllCause(() => Effect.void)
|
|
1367
|
+
)
|
|
1368
|
+
)
|
|
1369
|
+
);
|
|
1370
|
+
});
|
|
1371
|
+
register(
|
|
1372
|
+
fieldPath,
|
|
1373
|
+
(state) => Effect.gen(function* () {
|
|
1374
|
+
const { moduleId, instanceId } = getBoundScope(bound);
|
|
1375
|
+
const replayModeOpt = yield* Effect.serviceOption(ReplayModeConfigTag);
|
|
1376
|
+
const replayMode = Option.isSome(replayModeOpt) ? replayModeOpt.value.mode : "live";
|
|
1377
|
+
const replayLogOpt = yield* Effect.serviceOption(ReplayLog);
|
|
1378
|
+
const replayLog = Option.isSome(replayLogOpt) ? replayLogOpt.value : void 0;
|
|
1379
|
+
const force = yield* FiberRef.get(forceSourceRefresh);
|
|
1380
|
+
const listValue = getAtPath(state, listPath);
|
|
1381
|
+
const items = Array.isArray(listValue) ? listValue : [];
|
|
1382
|
+
const ids = store.ensureList(listPath, items);
|
|
1383
|
+
const traceKey = `${instanceId ?? "unknown"}::source::${fieldPath}`;
|
|
1384
|
+
if (isDevEnv() && (yield* onceInRunSession(`deps_trace_settled:${traceKey}`))) {
|
|
1385
|
+
try {
|
|
1386
|
+
const sample = items[0];
|
|
1387
|
+
if (sample !== void 0) {
|
|
1388
|
+
const traced = trace((s) => entry.meta.key(s), sample);
|
|
1389
|
+
const prefixedReads = traced.reads.map((r) => r ? `${listPath}[].${r}` : `${listPath}[]`);
|
|
1390
|
+
const diff = diffDeps(entry.meta.deps ?? [], prefixedReads);
|
|
1391
|
+
if (diff) {
|
|
1392
|
+
yield* emitDepsMismatch({
|
|
1393
|
+
moduleId,
|
|
1394
|
+
instanceId,
|
|
1395
|
+
kind: "source",
|
|
1396
|
+
fieldPath,
|
|
1397
|
+
diff
|
|
1398
|
+
});
|
|
1399
|
+
}
|
|
1400
|
+
}
|
|
1401
|
+
} catch {
|
|
1402
|
+
}
|
|
1403
|
+
}
|
|
1404
|
+
for (let index = 0; index < items.length; index++) {
|
|
1405
|
+
const rowId = ids[index];
|
|
1406
|
+
if (!rowId) continue;
|
|
1407
|
+
const concretePath = toListItemValuePath(listPath, index, itemPath);
|
|
1408
|
+
const prevSnapshot = getAtPath(state, concretePath);
|
|
1409
|
+
let key;
|
|
1410
|
+
try {
|
|
1411
|
+
key = entry.meta.key(items[index]);
|
|
1412
|
+
} catch {
|
|
1413
|
+
key = void 0;
|
|
1414
|
+
}
|
|
1415
|
+
const current = inFlight2.get(rowId);
|
|
1416
|
+
if (key === void 0) {
|
|
1417
|
+
trailing2.delete(rowId);
|
|
1418
|
+
inFlight2.delete(rowId);
|
|
1419
|
+
if (prevSnapshot && typeof prevSnapshot === "object" && prevSnapshot.status === "idle" && prevSnapshot.data === void 0 && prevSnapshot.error === void 0) {
|
|
1420
|
+
continue;
|
|
1421
|
+
}
|
|
1422
|
+
const idleSnapshot = Snapshot.idle();
|
|
1423
|
+
const wroteIdlePath = yield* setSnapshotForRowInTxn(
|
|
1424
|
+
rowId,
|
|
1425
|
+
idleSnapshot,
|
|
1426
|
+
"source-refresh",
|
|
1427
|
+
`source:${fieldPath}:${rowId}:idle`
|
|
1428
|
+
);
|
|
1429
|
+
if (wroteIdlePath) {
|
|
1430
|
+
const event = {
|
|
1431
|
+
_tag: "ResourceSnapshot",
|
|
1432
|
+
resourceId,
|
|
1433
|
+
fieldPath: wroteIdlePath,
|
|
1434
|
+
keyHash: void 0,
|
|
1435
|
+
concurrency: mode2,
|
|
1436
|
+
phase: "idle",
|
|
1437
|
+
snapshot: idleSnapshot,
|
|
1438
|
+
timestamp: Date.now(),
|
|
1439
|
+
moduleId,
|
|
1440
|
+
instanceId
|
|
1441
|
+
};
|
|
1442
|
+
recordReplayEvent(bound, event);
|
|
1443
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1444
|
+
}
|
|
1445
|
+
continue;
|
|
1446
|
+
}
|
|
1447
|
+
const h = keyHash(key);
|
|
1448
|
+
if (!force && current && current.keyHash === h) {
|
|
1449
|
+
continue;
|
|
1450
|
+
}
|
|
1451
|
+
const prevKeyHash = prevSnapshot && typeof prevSnapshot === "object" ? prevSnapshot.keyHash : void 0;
|
|
1452
|
+
if (!force && !current && prevKeyHash === h) {
|
|
1453
|
+
continue;
|
|
1454
|
+
}
|
|
1455
|
+
if (mode2 === "exhaust-trailing" && current) {
|
|
1456
|
+
trailing2.set(rowId, { key, keyHash: h });
|
|
1457
|
+
const loadingSnapshot = Snapshot.loading({ keyHash: h });
|
|
1458
|
+
const wroteLoadingPath = yield* setSnapshotForRowInTxn(
|
|
1459
|
+
rowId,
|
|
1460
|
+
loadingSnapshot,
|
|
1461
|
+
"source-refresh",
|
|
1462
|
+
`source:${fieldPath}:${rowId}:loading`
|
|
1463
|
+
);
|
|
1464
|
+
if (wroteLoadingPath) {
|
|
1465
|
+
const event = {
|
|
1466
|
+
_tag: "ResourceSnapshot",
|
|
1467
|
+
resourceId,
|
|
1468
|
+
fieldPath: wroteLoadingPath,
|
|
1469
|
+
keyHash: h,
|
|
1470
|
+
concurrency: mode2,
|
|
1471
|
+
phase: "loading",
|
|
1472
|
+
snapshot: loadingSnapshot,
|
|
1473
|
+
timestamp: Date.now(),
|
|
1474
|
+
moduleId,
|
|
1475
|
+
instanceId
|
|
1476
|
+
};
|
|
1477
|
+
recordReplayEvent(bound, event);
|
|
1478
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1479
|
+
}
|
|
1480
|
+
continue;
|
|
1481
|
+
}
|
|
1482
|
+
if (mode2 === "switch" && current) {
|
|
1483
|
+
trailing2.delete(rowId);
|
|
1484
|
+
inFlight2.delete(rowId);
|
|
1485
|
+
}
|
|
1486
|
+
yield* startFetch2(rowId, key, h, replayMode, replayLog);
|
|
1487
|
+
}
|
|
1488
|
+
})
|
|
1489
|
+
);
|
|
1490
|
+
return Effect.void;
|
|
1491
|
+
}
|
|
1492
|
+
let inFlight;
|
|
1493
|
+
let gen = 0;
|
|
1494
|
+
let trailing;
|
|
1495
|
+
const concurrency = entry.meta.concurrency;
|
|
1496
|
+
const mode = concurrency ?? "switch";
|
|
1497
|
+
const startFetch = (key, keyHash2, replayMode, replayLog) => Effect.gen(function* () {
|
|
1498
|
+
const { moduleId, instanceId } = getBoundScope(bound);
|
|
1499
|
+
let loadingSnapshot = Snapshot.loading({ keyHash: keyHash2 });
|
|
1500
|
+
if (replayMode === "replay" && replayLog) {
|
|
1501
|
+
const replayLoading = yield* replayLog.consumeNextResourceSnapshot({
|
|
1502
|
+
resourceId,
|
|
1503
|
+
fieldPath,
|
|
1504
|
+
keyHash: keyHash2,
|
|
1505
|
+
phase: "loading"
|
|
1506
|
+
});
|
|
1507
|
+
if (replayLoading) {
|
|
1508
|
+
loadingSnapshot = replayLoading.snapshot;
|
|
1509
|
+
}
|
|
1510
|
+
}
|
|
1511
|
+
const wroteLoading = yield* setSnapshotInTxn(
|
|
1512
|
+
bound,
|
|
1513
|
+
fieldPath,
|
|
1514
|
+
loadingSnapshot,
|
|
1515
|
+
"source-refresh",
|
|
1516
|
+
`source:${fieldPath}:loading`,
|
|
1517
|
+
step.debugInfo?.graphNodeId
|
|
1518
|
+
);
|
|
1519
|
+
if (wroteLoading) {
|
|
1520
|
+
const event = {
|
|
1521
|
+
_tag: "ResourceSnapshot",
|
|
1522
|
+
resourceId,
|
|
1523
|
+
fieldPath,
|
|
1524
|
+
keyHash: keyHash2,
|
|
1525
|
+
concurrency: mode,
|
|
1526
|
+
phase: "loading",
|
|
1527
|
+
snapshot: loadingSnapshot,
|
|
1528
|
+
timestamp: Date.now(),
|
|
1529
|
+
moduleId,
|
|
1530
|
+
instanceId
|
|
1531
|
+
};
|
|
1532
|
+
recordReplayEvent(bound, event);
|
|
1533
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1534
|
+
}
|
|
1535
|
+
const io = Effect.gen(function* () {
|
|
1536
|
+
if (replayMode === "replay" && replayLog) {
|
|
1537
|
+
yield* Effect.yieldNow();
|
|
1538
|
+
const replayed = yield* replayLog.consumeNextResourceSnapshot({
|
|
1539
|
+
resourceId,
|
|
1540
|
+
fieldPath,
|
|
1541
|
+
keyHash: keyHash2
|
|
1542
|
+
});
|
|
1543
|
+
if (!replayed) return yield* Effect.void;
|
|
1544
|
+
if (replayed.phase === "success") {
|
|
1545
|
+
const event = {
|
|
1546
|
+
_tag: "ResourceSnapshot",
|
|
1547
|
+
resourceId,
|
|
1548
|
+
fieldPath,
|
|
1549
|
+
keyHash: keyHash2,
|
|
1550
|
+
concurrency: mode,
|
|
1551
|
+
phase: "success",
|
|
1552
|
+
snapshot: replayed.snapshot,
|
|
1553
|
+
timestamp: Date.now(),
|
|
1554
|
+
moduleId,
|
|
1555
|
+
instanceId
|
|
1556
|
+
};
|
|
1557
|
+
yield* writebackIfCurrentKeyHash(
|
|
1558
|
+
bound,
|
|
1559
|
+
fieldPath,
|
|
1560
|
+
keyHash2,
|
|
1561
|
+
replayed.snapshot,
|
|
1562
|
+
"source-refresh",
|
|
1563
|
+
`source:${fieldPath}:success`,
|
|
1564
|
+
step.debugInfo?.graphNodeId,
|
|
1565
|
+
event
|
|
1566
|
+
);
|
|
1567
|
+
} else if (replayed.phase === "error") {
|
|
1568
|
+
const event = {
|
|
1569
|
+
_tag: "ResourceSnapshot",
|
|
1570
|
+
resourceId,
|
|
1571
|
+
fieldPath,
|
|
1572
|
+
keyHash: keyHash2,
|
|
1573
|
+
concurrency: mode,
|
|
1574
|
+
phase: "error",
|
|
1575
|
+
snapshot: replayed.snapshot,
|
|
1576
|
+
timestamp: Date.now(),
|
|
1577
|
+
moduleId,
|
|
1578
|
+
instanceId
|
|
1579
|
+
};
|
|
1580
|
+
yield* writebackIfCurrentKeyHash(
|
|
1581
|
+
bound,
|
|
1582
|
+
fieldPath,
|
|
1583
|
+
keyHash2,
|
|
1584
|
+
replayed.snapshot,
|
|
1585
|
+
"source-refresh",
|
|
1586
|
+
`source:${fieldPath}:error`,
|
|
1587
|
+
step.debugInfo?.graphNodeId,
|
|
1588
|
+
event
|
|
1589
|
+
);
|
|
1590
|
+
}
|
|
1591
|
+
return yield* Effect.void;
|
|
1592
|
+
}
|
|
1593
|
+
const stack = yield* getMiddlewareStack();
|
|
1594
|
+
const registryOpt = yield* Effect.serviceOption(internal.ResourceRegistryTag);
|
|
1595
|
+
const registry = Option.isSome(registryOpt) ? registryOpt.value : void 0;
|
|
1596
|
+
const spec = registry?.specs.get(resourceId);
|
|
1597
|
+
if (!spec) {
|
|
1598
|
+
return yield* Effect.void;
|
|
1599
|
+
}
|
|
1600
|
+
const loadEffect = spec.load(key);
|
|
1601
|
+
const meta = {
|
|
1602
|
+
moduleId,
|
|
1603
|
+
instanceId,
|
|
1604
|
+
fieldPath,
|
|
1605
|
+
resourceId,
|
|
1606
|
+
key,
|
|
1607
|
+
keyHash: keyHash2,
|
|
1608
|
+
traitNodeId: step.debugInfo?.graphNodeId,
|
|
1609
|
+
stepId: step.id
|
|
1610
|
+
};
|
|
1611
|
+
if (!(typeof meta.opSeq === "number" && Number.isFinite(meta.opSeq))) {
|
|
1612
|
+
const sessionOpt = yield* Effect.serviceOption(RunSessionTag);
|
|
1613
|
+
if (Option.isSome(sessionOpt)) {
|
|
1614
|
+
const seqKey = instanceId ?? "global";
|
|
1615
|
+
meta.opSeq = sessionOpt.value.local.nextSeq("opSeq", seqKey);
|
|
1616
|
+
}
|
|
1617
|
+
}
|
|
1618
|
+
const op = make({
|
|
1619
|
+
kind: "trait-source",
|
|
1620
|
+
name: resourceId,
|
|
1621
|
+
effect: loadEffect,
|
|
1622
|
+
meta
|
|
1623
|
+
});
|
|
1624
|
+
const exit = yield* Effect.exit(run(op, stack));
|
|
1625
|
+
if (exit._tag === "Success") {
|
|
1626
|
+
const successSnapshot = Snapshot.success({ keyHash: keyHash2, data: exit.value });
|
|
1627
|
+
const event = {
|
|
1628
|
+
_tag: "ResourceSnapshot",
|
|
1629
|
+
resourceId,
|
|
1630
|
+
fieldPath,
|
|
1631
|
+
keyHash: keyHash2,
|
|
1632
|
+
concurrency: mode,
|
|
1633
|
+
phase: "success",
|
|
1634
|
+
snapshot: successSnapshot,
|
|
1635
|
+
timestamp: Date.now(),
|
|
1636
|
+
moduleId,
|
|
1637
|
+
instanceId
|
|
1638
|
+
};
|
|
1639
|
+
const wroteSuccess = yield* writebackIfCurrentKeyHash(
|
|
1640
|
+
bound,
|
|
1641
|
+
fieldPath,
|
|
1642
|
+
keyHash2,
|
|
1643
|
+
successSnapshot,
|
|
1644
|
+
"source-refresh",
|
|
1645
|
+
`source:${fieldPath}:success`,
|
|
1646
|
+
step.debugInfo?.graphNodeId,
|
|
1647
|
+
event
|
|
1648
|
+
);
|
|
1649
|
+
if (wroteSuccess) {
|
|
1650
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1651
|
+
}
|
|
1652
|
+
} else {
|
|
1653
|
+
const errorSnapshot = Snapshot.error({ keyHash: keyHash2, error: exit.cause });
|
|
1654
|
+
const event = {
|
|
1655
|
+
_tag: "ResourceSnapshot",
|
|
1656
|
+
resourceId,
|
|
1657
|
+
fieldPath,
|
|
1658
|
+
keyHash: keyHash2,
|
|
1659
|
+
concurrency: mode,
|
|
1660
|
+
phase: "error",
|
|
1661
|
+
snapshot: errorSnapshot,
|
|
1662
|
+
timestamp: Date.now(),
|
|
1663
|
+
moduleId,
|
|
1664
|
+
instanceId
|
|
1665
|
+
};
|
|
1666
|
+
const wroteError = yield* writebackIfCurrentKeyHash(
|
|
1667
|
+
bound,
|
|
1668
|
+
fieldPath,
|
|
1669
|
+
keyHash2,
|
|
1670
|
+
errorSnapshot,
|
|
1671
|
+
"source-refresh",
|
|
1672
|
+
`source:${fieldPath}:error`,
|
|
1673
|
+
step.debugInfo?.graphNodeId,
|
|
1674
|
+
event
|
|
1675
|
+
);
|
|
1676
|
+
if (wroteError) {
|
|
1677
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1678
|
+
}
|
|
1679
|
+
}
|
|
1680
|
+
}).pipe(Effect.catchAllCause(() => Effect.void));
|
|
1681
|
+
const fiber = yield* Effect.forkScoped(Effect.locally(inSyncTransactionFiber, false)(io));
|
|
1682
|
+
const myGen = gen += 1;
|
|
1683
|
+
inFlight = { gen: myGen, fiber, keyHash: keyHash2 };
|
|
1684
|
+
yield* Effect.forkScoped(
|
|
1685
|
+
Effect.locally(
|
|
1686
|
+
inSyncTransactionFiber,
|
|
1687
|
+
false
|
|
1688
|
+
)(
|
|
1689
|
+
Fiber.await(fiber).pipe(
|
|
1690
|
+
Effect.zipRight(
|
|
1691
|
+
Effect.sync(() => {
|
|
1692
|
+
if (inFlight && inFlight.gen === myGen) {
|
|
1693
|
+
inFlight = void 0;
|
|
1694
|
+
}
|
|
1695
|
+
})
|
|
1696
|
+
),
|
|
1697
|
+
Effect.zipRight(
|
|
1698
|
+
mode === "exhaust-trailing" ? Effect.gen(function* () {
|
|
1699
|
+
const next = trailing;
|
|
1700
|
+
trailing = void 0;
|
|
1701
|
+
if (next) {
|
|
1702
|
+
yield* startFetch(next.key, next.keyHash, replayMode, replayLog);
|
|
1703
|
+
}
|
|
1704
|
+
}) : Effect.void
|
|
1705
|
+
),
|
|
1706
|
+
Effect.catchAllCause(() => Effect.void)
|
|
1707
|
+
)
|
|
1708
|
+
)
|
|
1709
|
+
);
|
|
1710
|
+
});
|
|
1711
|
+
register(
|
|
1712
|
+
fieldPath,
|
|
1713
|
+
(state) => Effect.gen(function* () {
|
|
1714
|
+
const { moduleId, instanceId } = getBoundScope(bound);
|
|
1715
|
+
const replayModeOpt = yield* Effect.serviceOption(ReplayModeConfigTag);
|
|
1716
|
+
const replayMode = Option.isSome(replayModeOpt) ? replayModeOpt.value.mode : "live";
|
|
1717
|
+
const replayLogOpt = yield* Effect.serviceOption(ReplayLog);
|
|
1718
|
+
const replayLog = Option.isSome(replayLogOpt) ? replayLogOpt.value : void 0;
|
|
1719
|
+
const force = yield* FiberRef.get(forceSourceRefresh);
|
|
1720
|
+
let key;
|
|
1721
|
+
try {
|
|
1722
|
+
key = entry.meta.key(state);
|
|
1723
|
+
} catch {
|
|
1724
|
+
key = void 0;
|
|
1725
|
+
}
|
|
1726
|
+
const traceKey = `${instanceId ?? "unknown"}::source::${fieldPath}`;
|
|
1727
|
+
if (isDevEnv() && (yield* onceInRunSession(`deps_trace_settled:${traceKey}`))) {
|
|
1728
|
+
try {
|
|
1729
|
+
const traced = trace((s) => entry.meta.key(s), state);
|
|
1730
|
+
const diff = diffDeps(entry.meta.deps ?? [], traced.reads);
|
|
1731
|
+
if (diff) {
|
|
1732
|
+
yield* emitDepsMismatch({
|
|
1733
|
+
moduleId,
|
|
1734
|
+
instanceId,
|
|
1735
|
+
kind: "source",
|
|
1736
|
+
fieldPath,
|
|
1737
|
+
diff
|
|
1738
|
+
});
|
|
1739
|
+
}
|
|
1740
|
+
} catch {
|
|
1741
|
+
}
|
|
1742
|
+
}
|
|
1743
|
+
if (key === void 0) {
|
|
1744
|
+
if (inFlight) {
|
|
1745
|
+
yield* Fiber.interruptFork(inFlight.fiber);
|
|
1746
|
+
inFlight = void 0;
|
|
1747
|
+
}
|
|
1748
|
+
trailing = void 0;
|
|
1749
|
+
const idleSnapshot = Snapshot.idle();
|
|
1750
|
+
const wroteIdle = yield* setSnapshotInTxn(
|
|
1751
|
+
bound,
|
|
1752
|
+
fieldPath,
|
|
1753
|
+
idleSnapshot,
|
|
1754
|
+
"source-refresh",
|
|
1755
|
+
`source:${fieldPath}:idle`,
|
|
1756
|
+
step.debugInfo?.graphNodeId
|
|
1757
|
+
);
|
|
1758
|
+
if (wroteIdle) {
|
|
1759
|
+
const event = {
|
|
1760
|
+
_tag: "ResourceSnapshot",
|
|
1761
|
+
resourceId,
|
|
1762
|
+
fieldPath,
|
|
1763
|
+
keyHash: void 0,
|
|
1764
|
+
concurrency: mode,
|
|
1765
|
+
phase: "idle",
|
|
1766
|
+
snapshot: idleSnapshot,
|
|
1767
|
+
timestamp: Date.now(),
|
|
1768
|
+
moduleId,
|
|
1769
|
+
instanceId
|
|
1770
|
+
};
|
|
1771
|
+
recordReplayEvent(bound, event);
|
|
1772
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1773
|
+
}
|
|
1774
|
+
return;
|
|
1775
|
+
}
|
|
1776
|
+
const h = keyHash(key);
|
|
1777
|
+
if (!force) {
|
|
1778
|
+
if (inFlight && inFlight.keyHash === h) {
|
|
1779
|
+
return;
|
|
1780
|
+
}
|
|
1781
|
+
const currentSnapshot = getAtPath(state, fieldPath);
|
|
1782
|
+
const currentKeyHash = currentSnapshot && typeof currentSnapshot === "object" ? currentSnapshot.keyHash : void 0;
|
|
1783
|
+
const currentStatus = currentSnapshot && typeof currentSnapshot === "object" ? currentSnapshot.status : void 0;
|
|
1784
|
+
if (currentStatus && currentStatus !== "idle" && currentKeyHash === h) {
|
|
1785
|
+
return;
|
|
1786
|
+
}
|
|
1787
|
+
}
|
|
1788
|
+
if (mode === "exhaust-trailing" && inFlight) {
|
|
1789
|
+
trailing = { key, keyHash: h };
|
|
1790
|
+
const loadingSnapshot = Snapshot.loading({ keyHash: h });
|
|
1791
|
+
const wroteLoading = yield* setSnapshotInTxn(
|
|
1792
|
+
bound,
|
|
1793
|
+
fieldPath,
|
|
1794
|
+
loadingSnapshot,
|
|
1795
|
+
"source-refresh",
|
|
1796
|
+
`source:${fieldPath}:loading`,
|
|
1797
|
+
step.debugInfo?.graphNodeId
|
|
1798
|
+
);
|
|
1799
|
+
if (wroteLoading) {
|
|
1800
|
+
const event = {
|
|
1801
|
+
_tag: "ResourceSnapshot",
|
|
1802
|
+
resourceId,
|
|
1803
|
+
fieldPath,
|
|
1804
|
+
keyHash: h,
|
|
1805
|
+
concurrency: mode,
|
|
1806
|
+
phase: "loading",
|
|
1807
|
+
snapshot: loadingSnapshot,
|
|
1808
|
+
timestamp: Date.now(),
|
|
1809
|
+
moduleId,
|
|
1810
|
+
instanceId
|
|
1811
|
+
};
|
|
1812
|
+
recordReplayEvent(bound, event);
|
|
1813
|
+
yield* recordSnapshot(replayMode, replayLog, event);
|
|
1814
|
+
}
|
|
1815
|
+
return;
|
|
1816
|
+
}
|
|
1817
|
+
if (mode === "switch" && inFlight) {
|
|
1818
|
+
yield* Fiber.interruptFork(inFlight.fiber);
|
|
1819
|
+
inFlight = void 0;
|
|
1820
|
+
trailing = void 0;
|
|
1821
|
+
}
|
|
1822
|
+
yield* startFetch(key, h, replayMode, replayLog);
|
|
1823
|
+
})
|
|
1824
|
+
);
|
|
1825
|
+
return Effect.void;
|
|
1826
|
+
};
|
|
1827
|
+
|
|
1828
|
+
// ../logix-core/src/internal/state-trait/converge-diagnostics.ts
|
|
1829
|
+
import { Effect as Effect2, FiberRef as FiberRef2, Option as Option2 } from "../effect.js";
|
|
1830
|
+
import * as SchemaAST2 from "../effect/SchemaAST.js";
|
|
1831
|
+
var onceKeysFallback = /* @__PURE__ */ new Set();
|
|
1832
|
+
var onceInRunSession2 = (key) => Effect2.serviceOption(RunSessionTag).pipe(
|
|
1833
|
+
Effect2.map((maybe) => {
|
|
1834
|
+
if (Option2.isSome(maybe)) {
|
|
1835
|
+
return maybe.value.local.once(key);
|
|
1836
|
+
}
|
|
1837
|
+
if (onceKeysFallback.has(key)) return false;
|
|
1838
|
+
onceKeysFallback.add(key);
|
|
1839
|
+
return true;
|
|
1840
|
+
})
|
|
1841
|
+
);
|
|
1842
|
+
var formatList2 = (items, limit = 10) => {
|
|
1843
|
+
if (items.length === 0) return "";
|
|
1844
|
+
if (items.length <= limit) return items.join(", ");
|
|
1845
|
+
return `${items.slice(0, limit).join(", ")}, \u2026(+${items.length - limit})`;
|
|
1846
|
+
};
|
|
1847
|
+
var emitDepsMismatch2 = (params) => {
|
|
1848
|
+
return Effect2.gen(function* () {
|
|
1849
|
+
const key = `${params.moduleId ?? "unknown"}::${params.instanceId ?? "unknown"}::${params.kind}::${params.fieldPath}`;
|
|
1850
|
+
const shouldEmit = yield* onceInRunSession2(`deps_mismatch:${key}`);
|
|
1851
|
+
if (!shouldEmit) return;
|
|
1852
|
+
yield* record({
|
|
1853
|
+
type: "diagnostic",
|
|
1854
|
+
moduleId: params.moduleId,
|
|
1855
|
+
instanceId: params.instanceId,
|
|
1856
|
+
code: "state_trait::deps_mismatch",
|
|
1857
|
+
severity: "warning",
|
|
1858
|
+
message: `[deps] ${params.kind} "${params.fieldPath}" declared=[${formatList2(params.diff.declared)}] reads=[${formatList2(params.diff.reads)}] missing=[${formatList2(params.diff.missing)}] unused=[${formatList2(params.diff.unused)}]`,
|
|
1859
|
+
hint: 'deps is the single source of truth for dependencies: incremental scheduling / reverse closures / performance optimizations rely on deps only. Keep deps consistent with actual reads; if you really depend on the whole object, declare a coarser-grained dep (e.g. "profile") to cover sub-field reads.',
|
|
1860
|
+
kind: `deps_mismatch:${params.kind}`
|
|
1861
|
+
});
|
|
1862
|
+
});
|
|
1863
|
+
};
|
|
1864
|
+
var schemaHasPath = (ast, segments, seen = /* @__PURE__ */ new Set()) => {
|
|
1865
|
+
if (segments.length === 0) return true;
|
|
1866
|
+
let current = ast;
|
|
1867
|
+
while (true) {
|
|
1868
|
+
if (SchemaAST2.isSuspend(current)) {
|
|
1869
|
+
if (seen.has(current)) {
|
|
1870
|
+
return true;
|
|
1871
|
+
}
|
|
1872
|
+
seen.add(current);
|
|
1873
|
+
current = current.f();
|
|
1874
|
+
continue;
|
|
1875
|
+
}
|
|
1876
|
+
if (SchemaAST2.isRefinement(current)) {
|
|
1877
|
+
current = current.from;
|
|
1878
|
+
continue;
|
|
1879
|
+
}
|
|
1880
|
+
break;
|
|
1881
|
+
}
|
|
1882
|
+
if (SchemaAST2.isTransformation(current)) {
|
|
1883
|
+
return schemaHasPath(current.to, segments, seen) || schemaHasPath(current.from, segments, seen);
|
|
1884
|
+
}
|
|
1885
|
+
if (SchemaAST2.isUnion(current)) {
|
|
1886
|
+
return current.types.some((t) => schemaHasPath(t, segments, seen));
|
|
1887
|
+
}
|
|
1888
|
+
if (SchemaAST2.isTupleType(current)) {
|
|
1889
|
+
const candidates = [];
|
|
1890
|
+
for (const e of current.elements) candidates.push(e.type);
|
|
1891
|
+
for (const r of current.rest) candidates.push(r.type);
|
|
1892
|
+
if (candidates.length === 0) return true;
|
|
1893
|
+
return candidates.some((t) => schemaHasPath(t, segments, seen));
|
|
1894
|
+
}
|
|
1895
|
+
if (SchemaAST2.isTypeLiteral(current)) {
|
|
1896
|
+
const [head, ...tail] = segments;
|
|
1897
|
+
for (const ps of current.propertySignatures) {
|
|
1898
|
+
if (String(ps.name) !== head) continue;
|
|
1899
|
+
return schemaHasPath(ps.type, tail, seen);
|
|
1900
|
+
}
|
|
1901
|
+
for (const sig of current.indexSignatures) {
|
|
1902
|
+
let param = sig.parameter;
|
|
1903
|
+
while (SchemaAST2.isRefinement(param)) {
|
|
1904
|
+
param = param.from;
|
|
1905
|
+
}
|
|
1906
|
+
const tag2 = param?._tag;
|
|
1907
|
+
if (tag2 === "StringKeyword" || tag2 === "TemplateLiteral") {
|
|
1908
|
+
return schemaHasPath(sig.type, tail, seen);
|
|
1909
|
+
}
|
|
1910
|
+
}
|
|
1911
|
+
return false;
|
|
1912
|
+
}
|
|
1913
|
+
const tag = current?._tag;
|
|
1914
|
+
if (tag === "AnyKeyword" || tag === "UnknownKeyword" || tag === "ObjectKeyword" || tag === "Declaration") {
|
|
1915
|
+
return true;
|
|
1916
|
+
}
|
|
1917
|
+
return false;
|
|
1918
|
+
};
|
|
1919
|
+
var schemaHasFieldPath = (stateSchemaAst, path) => {
|
|
1920
|
+
if (!path) return true;
|
|
1921
|
+
if (path === "$root") return true;
|
|
1922
|
+
const normalized = normalizeFieldPath(path);
|
|
1923
|
+
if (!normalized) return false;
|
|
1924
|
+
const segs = normalized[0] === "$root" ? normalized.slice(1) : normalized;
|
|
1925
|
+
return schemaHasPath(stateSchemaAst, segs);
|
|
1926
|
+
};
|
|
1927
|
+
var formatSchemaMismatchLine = (ref) => {
|
|
1928
|
+
if (ref.kind === "fieldPath") {
|
|
1929
|
+
return `- ${ref.entryKind} "${ref.entryFieldPath}" fieldPath="${ref.path}"`;
|
|
1930
|
+
}
|
|
1931
|
+
if (ref.kind === "dep") {
|
|
1932
|
+
const rule = ref.ruleName ? ` rule="${ref.ruleName}"` : "";
|
|
1933
|
+
return `- ${ref.entryKind} "${ref.entryFieldPath}" deps="${ref.path}"${rule}`;
|
|
1934
|
+
}
|
|
1935
|
+
if (ref.kind === "link_from") {
|
|
1936
|
+
return `- link "${ref.entryFieldPath}" from="${ref.path}"`;
|
|
1937
|
+
}
|
|
1938
|
+
if (ref.kind === "check_writeback") {
|
|
1939
|
+
return `- check "${ref.entryFieldPath}" writeback="${ref.path}"`;
|
|
1940
|
+
}
|
|
1941
|
+
return `- ${ref.entryKind} "${ref.entryFieldPath}" path="${ref.path}"`;
|
|
1942
|
+
};
|
|
1943
|
+
var emitSchemaMismatch = (program, ctx) => Effect2.gen(function* () {
|
|
1944
|
+
if (!isDevEnv()) return;
|
|
1945
|
+
const level = yield* FiberRef2.get(currentDiagnosticsLevel);
|
|
1946
|
+
if (level === "off") return;
|
|
1947
|
+
const key = `${ctx.moduleId ?? "unknown"}::${ctx.instanceId}`;
|
|
1948
|
+
const shouldEmit = yield* onceInRunSession2(`schema_mismatch:${key}`);
|
|
1949
|
+
if (!shouldEmit) return;
|
|
1950
|
+
const refs = program.schemaPaths ?? [];
|
|
1951
|
+
if (refs.length === 0) return;
|
|
1952
|
+
const stateSchemaAst = program.stateSchema.ast;
|
|
1953
|
+
const mismatches = [];
|
|
1954
|
+
const seen = /* @__PURE__ */ new Set();
|
|
1955
|
+
for (const ref of refs) {
|
|
1956
|
+
if (schemaHasFieldPath(stateSchemaAst, ref.path)) continue;
|
|
1957
|
+
const k = `${ref.kind}|${ref.entryKind}|${ref.entryFieldPath}|${ref.ruleName ?? ""}|${ref.path}`;
|
|
1958
|
+
if (seen.has(k)) continue;
|
|
1959
|
+
seen.add(k);
|
|
1960
|
+
mismatches.push(ref);
|
|
1961
|
+
}
|
|
1962
|
+
if (mismatches.length === 0) return;
|
|
1963
|
+
const limit = level === "light" ? 8 : 24;
|
|
1964
|
+
const lines = mismatches.slice(0, limit).map(formatSchemaMismatchLine);
|
|
1965
|
+
if (mismatches.length > limit) {
|
|
1966
|
+
lines.push(`- \u2026(+${mismatches.length - limit})`);
|
|
1967
|
+
}
|
|
1968
|
+
yield* record({
|
|
1969
|
+
type: "diagnostic",
|
|
1970
|
+
moduleId: ctx.moduleId,
|
|
1971
|
+
instanceId: ctx.instanceId,
|
|
1972
|
+
code: "state_trait::schema_mismatch",
|
|
1973
|
+
severity: "warning",
|
|
1974
|
+
message: `[schema] The following paths are not declared in stateSchema (total ${mismatches.length}):
|
|
1975
|
+
${lines.join("\n")}`,
|
|
1976
|
+
hint: "StateTrait writeback will create missing objects/fields. Declare all fieldPath/deps/link.from and errors.* writeback paths in stateSchema, or fix typos in trait paths.",
|
|
1977
|
+
kind: "schema_mismatch"
|
|
1978
|
+
});
|
|
1979
|
+
});
|
|
1980
|
+
|
|
1981
|
+
export {
|
|
1982
|
+
onceInRunSession2 as onceInRunSession,
|
|
1983
|
+
emitDepsMismatch2 as emitDepsMismatch,
|
|
1984
|
+
emitSchemaMismatch,
|
|
1985
|
+
diffDeps,
|
|
1986
|
+
trace,
|
|
1987
|
+
syncIdleInTransaction,
|
|
1988
|
+
installSourceRefresh,
|
|
1989
|
+
build
|
|
1990
|
+
};
|
|
1991
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../logix-core/src/internal/state-trait/build.ts", "../../../../logix-core/src/internal/state-trait/model.ts", "../../../../logix-core/src/internal/state-trait/source.impl.ts", "../../../../logix-core/src/internal/state-trait/deps-trace.ts", "../../../../logix-core/src/internal/state-trait/converge-diagnostics.ts"],
  "sourcesContent": ["import type { Schema } from 'effect'\nimport * as SchemaAST from 'effect/SchemaAST'\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 StateTraitKind,\n  type StateTraitPlan,\n  type StateTraitPlanStep,\n  type StateTraitResource,\n  type StateTraitSchemaPathRef,\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 { fnv1a32, stableStringify } from '../digest.js'\nimport type { ConvergeStaticIrRegistry } from './converge-ir.js'\n\nconst nowPerf = (): number =>\n  typeof globalThis.performance !== 'undefined' && typeof globalThis.performance.now === 'function'\n    ? globalThis.performance.now()\n    : Date.now()\n\ntype ConvergeWriter = Extract<StateTraitEntry<any, string>, { readonly kind: 'computed' | 'link' }>\n\nconst collectMultipleWritersError = (\n  entries: ReadonlyArray<StateTraitEntry<any, string>>,\n): ConvergeStaticIrRegistry['configError'] | undefined => {\n  const kindsByFieldPath = new Map<string, Set<StateTraitKind>>()\n\n  for (const entry of entries) {\n    if (entry.kind !== 'computed' && entry.kind !== 'link' && entry.kind !== 'source' && entry.kind !== 'externalStore') {\n      continue\n    }\n    const set = kindsByFieldPath.get(entry.fieldPath) ?? new Set<StateTraitKind>()\n    set.add(entry.kind)\n    kindsByFieldPath.set(entry.fieldPath, set)\n  }\n\n  const conflicts: Array<{ readonly fieldPath: string; readonly kinds: ReadonlyArray<StateTraitKind> }> = []\n  for (const [fieldPath, kinds] of kindsByFieldPath.entries()) {\n    if (kinds.size <= 1) continue\n    conflicts.push({ fieldPath, kinds: Array.from(kinds).sort() })\n  }\n\n  if (conflicts.length === 0) return undefined\n\n  conflicts.sort((a, b) => (a.fieldPath < b.fieldPath ? -1 : a.fieldPath > b.fieldPath ? 1 : 0))\n  const fields = conflicts.map((c) => c.fieldPath)\n  const primary = conflicts[0]!\n  const kindSummary = primary.kinds.join(' + ')\n\n  return {\n    code: 'MULTIPLE_WRITERS',\n    message:\n      `[StateTrait.build] Multiple writers for field \"${primary.fieldPath}\" (${kindSummary}). ` +\n      'Only one of computed/link/source/externalStore can write a fieldPath.',\n    fields,\n  }\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 collectSchemaFieldPaths = (schema: Schema.Schema<any, any>): ReadonlyArray<FieldPath> => {\n  const byKey = new Map<string, FieldPath>()\n\n  const add = (path: FieldPath): void => {\n    const normalized = normalizeFieldPath(path)\n    if (!normalized) return\n    byKey.set(JSON.stringify(normalized), normalized)\n  }\n\n  const visit = (ast: SchemaAST.AST, prefix: ReadonlyArray<string>, seen: Set<SchemaAST.AST>): void => {\n    let current: SchemaAST.AST = ast\n\n    // Unwrap Suspend/Refinement (recursive schema / branded schema).\n    while (true) {\n      if (SchemaAST.isSuspend(current)) {\n        if (seen.has(current)) return\n        seen.add(current)\n        current = current.f()\n        continue\n      }\n      if (SchemaAST.isRefinement(current)) {\n        current = current.from\n        continue\n      }\n      break\n    }\n\n    if (SchemaAST.isTransformation(current)) {\n      visit(current.to, prefix, seen)\n      visit(current.from, prefix, seen)\n      return\n    }\n\n    if (SchemaAST.isUnion(current)) {\n      for (const t of current.types) {\n        visit(t, prefix, seen)\n      }\n      return\n    }\n\n    // Array / Tuple: indices do not enter the FieldPathId space; recurse into element types to support `items[0].name -> items.name`.\n    if (SchemaAST.isTupleType(current)) {\n      for (const e of current.elements) {\n        visit(e.type, prefix, seen)\n      }\n      for (const r of current.rest) {\n        visit(r.type, prefix, seen)\n      }\n      return\n    }\n\n    if (SchemaAST.isTypeLiteral(current)) {\n      for (const ps of current.propertySignatures) {\n        const seg = String(ps.name)\n        if (!seg) continue\n        const next = [...prefix, seg]\n        add(next)\n        visit(ps.type, next, seen)\n      }\n      // Index signature (Record<string, T>) can't be enumerated statically: avoid generating misaligned dynamic key paths.\n      return\n    }\n\n    // Any / Unknown / Object / Declaration (open types): cannot enumerate nested paths; stop conservatively.\n  }\n\n  visit(schema.ast as unknown as SchemaAST.AST, [], new Set())\n  return Array.from(byKey.values()).sort(compareFieldPath)\n}\n\nconst buildConvergeIr = (\n  stateSchema: Schema.Schema<any, any>,\n  entries: ReadonlyArray<StateTraitEntry<any, string>>,\n): ConvergeStaticIrRegistry => {\n  const startedAt = nowPerf()\n  const generation = 0\n\n  const multipleWritersError = collectMultipleWritersError(entries)\n\n  const writers = entries.filter((e): e is ConvergeWriter => e.kind === 'computed' || e.kind === 'link')\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      const scheduling = (entry.meta as any)?.scheduling === 'deferred' ? 'd' : 'i'\n      return `${entry.kind}:${entry.fieldPath}@${scheduling}=>${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 = multipleWritersError\n    ? { order: [] as ReadonlyArray<string> }\n    : writers.length > 0\n      ? computeConvergeTopoOrder(writers)\n      : { order: [] as ReadonlyArray<string> }\n  const stepsById: Array<ConvergeWriter> = topo.configError ? [] : 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  // 065: FieldPathId semantics must cover all enumerable field paths of stateSchema; otherwise reducer patchPaths can't map and will fall back to dirtyAll.\n  for (const schemaPath of collectSchemaFieldPaths(stateSchema)) {\n    addPath(schemaPath)\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  const fieldPathsKey = fnv1a32(stableStringify(fieldPaths))\n\n  const stepOutFieldPathIdByStepId: Array<FieldPathId> = []\n  const stepDepsFieldPathIdsByStepId: Array<ReadonlyArray<FieldPathId>> = []\n  const stepSchedulingByStepId: Array<'immediate' | 'deferred'> = []\n\n  for (const entry of stepsById) {\n    const out = normalizeFieldPath(entry.fieldPath)\n    const outId = out != null ? getFieldPathId(fieldPathIdRegistry, out) : undefined\n    if (outId == null) {\n      throw new Error(`[StateTrait.build] Failed to map converge output fieldPath \"${entry.fieldPath}\" to FieldPathId.`)\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    stepSchedulingByStepId.push((entry.meta as any)?.scheduling === 'deferred' ? 'deferred' : 'immediate')\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    fieldPathsKey,\n    fieldPaths,\n    fieldPathIdRegistry,\n    ...(multipleWritersError ? { configError: multipleWritersError } : topo.configError ? { configError: topo.configError } : null),\n    stepsById,\n    stepOutFieldPathIdByStepId,\n    stepDepsFieldPathIdsByStepId,\n    stepSchedulingByStepId,\n    topoOrder,\n    buildDurationMs,\n  }\n}\n\n/**\n * Builds a normalized FieldTrait from a StateTraitEntry.\n *\n * - Currently uses explicit deps for computed/source and link edges; deeper dependency analysis is intentionally not performed.\n * - If we evolve explicit dependency declarations further, extend here.\n */\nconst toFieldTrait = (entry: StateTraitEntry<any, string>): 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    // Keep meta identical to Entry.meta at runtime so install can reuse it directly.\n    meta: entry.meta as any,\n    deps,\n  }\n}\n\n/**\n * Builds Field / Node / Edge / Resource sets from normalized entries.\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    // Build Graph edges and Plan steps by kind.\n    if (entry.kind === 'computed') {\n      const stepId = `computed:${fieldPath}`\n      planSteps.push({\n        id: stepId,\n        kind: 'computed-update',\n        targetFieldPath: fieldPath,\n        // Note: the current version does not statically analyze computed dependencies; sourceFieldPaths remains empty.\n      })\n      // If deps is explicitly declared, add Graph edges (for diagnostics / reverse-closure computation).\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          // Use a simple identifier string for now; may evolve into a structured form based on key rules.\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 === 'externalStore') {\n      planSteps.push({\n        id: `external-store:${fieldPath}`,\n        kind: 'external-store-sync',\n        targetFieldPath: fieldPath,\n      })\n    } else if (entry.kind === 'check') {\n      planSteps.push({\n        id: `check:${fieldPath}`,\n        kind: 'check-validate',\n        targetFieldPath: fieldPath,\n      })\n\n      // If the rule explicitly declares deps, add Graph edges (for ReverseClosure scoped validate).\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 * Performs a simple cycle detection for link edges.\n *\n * - Only considers edges with kind = 'link'; computed/source do not participate in the first version.\n * - On cycle detection, throws an error with path context to avoid infinite updates at runtime.\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\nconst collectSchemaPaths = (\n  entries: ReadonlyArray<StateTraitEntry<any, string>>,\n): ReadonlyArray<StateTraitSchemaPathRef> => {\n  const byKey = new Map<string, StateTraitSchemaPathRef>()\n\n  const add = (ref: StateTraitSchemaPathRef): void => {\n    if (!ref.path) return\n    const k = `${ref.kind}|${ref.entryKind}|${ref.entryFieldPath}|${ref.ruleName ?? ''}|${ref.path}`\n    byKey.set(k, ref)\n  }\n\n  const getCheckWritebackPath = (entry: Extract<StateTraitEntry<any, string>, { readonly kind: 'check' }>): string => {\n    const wb = (entry.meta as any)?.writeback\n    const p = wb && typeof wb === 'object' ? (wb as any).path : undefined\n    const writebackPath = typeof p === 'string' && p.startsWith('errors.') ? p : undefined\n\n    if (writebackPath) return writebackPath\n\n    const fieldPath = entry.fieldPath\n    if (fieldPath.endsWith('[]')) {\n      return `errors.${fieldPath.slice(0, -2)}`\n    }\n    return `errors.${fieldPath}`\n  }\n\n  for (const entry of entries) {\n    add({\n      kind: 'fieldPath',\n      entryKind: entry.kind,\n      entryFieldPath: entry.fieldPath,\n      path: entry.fieldPath,\n    })\n\n    if (entry.kind === 'computed' || entry.kind === 'source') {\n      const deps = ((entry.meta as any)?.deps ?? []) as ReadonlyArray<string>\n      for (const dep of deps) {\n        add({\n          kind: 'dep',\n          entryKind: entry.kind,\n          entryFieldPath: entry.fieldPath,\n          path: dep,\n        })\n      }\n    }\n\n    if (entry.kind === 'link') {\n      add({\n        kind: 'link_from',\n        entryKind: 'link',\n        entryFieldPath: entry.fieldPath,\n        path: entry.meta.from as string,\n      })\n    }\n\n    if (entry.kind === 'check') {\n      add({\n        kind: 'check_writeback',\n        entryKind: 'check',\n        entryFieldPath: entry.fieldPath,\n        path: getCheckWritebackPath(entry),\n      })\n\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 (!rule || typeof rule !== 'object') continue\n        const deps = (rule.deps ?? []) as ReadonlyArray<string>\n        for (const dep of deps) {\n          add({\n            kind: 'dep',\n            entryKind: 'check',\n            entryFieldPath: entry.fieldPath,\n            ruleName: name,\n            path: dep,\n          })\n        }\n      }\n    }\n  }\n\n  return Array.from(byKey.entries())\n    .sort((a, b) => (a[0] < b[0] ? -1 : a[0] > b[0] ? 1 : 0))\n    .map(([, v]) => v)\n}\n\n/**\n * Builds a StateTraitProgram from the given stateSchema and trait spec.\n *\n * - Pure function: does not depend on external Env / global state.\n * - Current implementation focuses on:\n *   - Normalizing Spec into entries.\n *   - Building a lightweight Graph / Plan from entries.\n *   - Running basic cycle detection for link edges.\n *\n * If we later need finer-grained dependency analysis (e.g. static analysis for computed/key),\n * evolve it inside this module without changing the public API surface.\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 (US2): require explicit deps (Graph/diagnostics/replay treat deps as the single dependency source of truth).\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  // Run a cycle check for link edges to avoid obvious configuration errors.\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(stateSchema as any, entries as ReadonlyArray<StateTraitEntry<any, string>>),\n    schemaPaths: collectSchemaPaths(entries as ReadonlyArray<StateTraitEntry<any, string>>),\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'\nimport type { ConvergeExecIr } from './converge-exec-ir.js'\n\n// StateTrait core model.\n// Keep the field semantics aligned with data-model.md / references/state-trait-core.md.\n\n/**\n * StateTraitSpec<S>\uFF1A\n * - The standard shape for the `traits` slot in a Module blueprint.\n * - Keys are constrained by StateFieldPath<S>; values are declarations for the corresponding scope (Entry / Node / List).\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' | 'externalStore' | 'check'\n\nexport type TraitLane = 'urgent' | 'nonUrgent'\n\n/**\n * TraitConvergeScheduling\uFF1A\n * - Scheduling semantics for converge/dirty-check (043): decides whether a step must converge within each transaction window.\n * - Default `immediate` (keeps legacy behavior); `deferred` takes effect only with explicit declaration + time-slicing enabled.\n */\nexport type TraitConvergeScheduling = 'immediate' | 'deferred'\n\nexport interface ComputedMeta<S, P> {\n  /**\n   * Explicit dependency field paths (required):\n   *\n   * - deps is the single source of truth for dependencies: Graph/ReverseClosure/incremental scheduling/perf optimizations rely on deps only.\n   * - In the DSL, `StateTrait.computed({ deps, get })` uses deps-as-args; it does not expose `(state) => ...`.\n   * - During build, `get(...depsValues)` is lowered into `derive(state)` for runtime execution, but the dependency read-set remains deps-based.\n   * - In dev-mode, if actual reads in `derive(state)` disagree with deps, a `deps_mismatch` diagnostic warning is emitted.\n   *\n   * - For root scope: deps are StateFieldPath<State>.\n   * - For list.item scope: deps are StateFieldPath<Item> (relative paths; build will prefix them).\n   */\n  readonly deps: ReadonlyArray<StateFieldPath<S>>\n  readonly derive: (state: Readonly<S>) => StateAtPath<S, P>\n  /**\n   * Optional: equality predicate (used to skip no-op writebacks).\n   */\n  readonly equals?: (prev: StateAtPath<S, P>, next: StateAtPath<S, P>) => boolean\n  /**\n   * Converge scheduling (043): defaults to immediate.\n   */\n  readonly scheduling?: TraitConvergeScheduling\n}\n\nexport interface SourceMeta<S, P> {\n  readonly deps: ReadonlyArray<StateFieldPath<S>>\n  /**\n   * Logical resource identifier (e.g. \"user/profile\").\n   *\n   * - In the StateTrait.source DSL, `resource` is stored here during build.\n   * - data-model.md calls it resourceId; we keep the DSL naming here to avoid confusion.\n   */\n  readonly resource: string\n  /**\n   * Rule for computing the key required to access the resource.\n   *\n   * - deps is the single source of truth for dependencies: Graph/ReverseClosure/incremental scheduling/perf optimizations rely on deps only.\n   * - In the DSL, `StateTrait.source({ deps, key })` uses deps-as-args; it does not expose `(state) => ...`.\n   * - During build, `key(...depsValues)` is lowered into `key(state)` for runtime execution, but the dependency read-set remains deps-based.\n   *\n   * - Returning undefined means the resource is inactive under the current state (should be recycled to idle).\n   */\n  readonly key: (state: Readonly<S>) => unknown\n  readonly triggers?: ReadonlyArray<'onMount' | 'onKeyChange' | 'manual'>\n  readonly debounceMs?: number\n  readonly concurrency?: 'switch' | 'exhaust-trailing'\n  /**\n   * Serializable metadata for devtools/docs (whitelisted fields).\n   */\n  readonly meta?: Meta.TraitMeta\n  /**\n   * Reserved: build may populate the field path this trait is attached to, for easier debugging.\n   */\n  readonly _fieldPath?: P\n}\n\nexport interface LinkMeta<S> {\n  /**\n   * Source field path (also constrained by StateFieldPath<S>).\n   */\n  readonly from: StateFieldPath<S>\n  /**\n   * Converge scheduling (043): defaults to immediate.\n   */\n  readonly scheduling?: TraitConvergeScheduling\n}\n\nexport interface ExternalStoreLike<T> {\n  readonly getSnapshot: () => T\n  readonly getServerSnapshot?: () => T\n  readonly subscribe: (listener: () => void) => () => void\n}\n\nexport interface ExternalStoreMeta<S, P, T = unknown> {\n  readonly store: ExternalStoreLike<T>\n  readonly select?: (snapshot: T) => StateAtPath<S, P>\n  readonly equals?: (prev: StateAtPath<S, P>, next: StateAtPath<S, P>) => boolean\n  readonly coalesceWindowMs?: number\n  readonly priority?: TraitLane\n  readonly meta?: Meta.TraitMeta\n  readonly _fieldPath?: P\n}\n\nexport type CheckRule<Input = unknown, Ctx = unknown> = {\n  readonly deps: ReadonlyArray<string>\n  /**\n   * validateOn\uFF1A\n   * - Affects only auto validation phases (onChange/onBlur); submit/manual always run.\n   * - Empty array disables auto validation (only submit/manual run).\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   * Named rule set (used for deterministic merge and diagnostics display).\n   */\n  readonly rules: Readonly<Record<string, CheckRule<Input, Ctx>>>\n  /**\n   * Error-tree writeback (Phase 2: structure only; concrete writeback semantics land in later phases).\n   */\n  readonly writeback?: {\n    readonly kind: 'errors'\n    readonly path?: string\n  }\n}\n\n/**\n * StateTraitEntry<S, P>\uFF1A\n * - A single Trait configuration attached to a field path P.\n * - kind and meta always come together and are used to dispatch behavior during build/install.\n */\nexport type StateTraitEntry<S = unknown, P extends string = StateFieldPath<S>> =\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: 'externalStore'\n      readonly meta: ExternalStoreMeta<S, P>\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?: StateTraitEntry<Input, any> | Readonly<Record<string, StateTraitEntry<Input, any>>>\n  readonly source?: StateTraitEntry<Input, any> | Readonly<Record<string, StateTraitEntry<Input, any>>>\n  readonly link?: StateTraitEntry<Input, any> | Readonly<Record<string, StateTraitEntry<Input, any>>>\n  readonly externalStore?: StateTraitEntry<Input, any> | 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 * - Represents a field node in State (whether or not traits are attached).\n * - Normalized from StateTraitSpec during build.\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 * - A structured description of a field trait (computed / source / link).\n * - meta matches StateTraitEntry.meta; deps is the set of field paths this trait depends on.\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    | ExternalStoreMeta<unknown, string>\n    | CheckMeta<unknown, unknown>\n  readonly deps: ReadonlyArray<string>\n}\n\n/**\n * StateTraitResource\uFF1A\n * - Describes the logical resource metadata that a source trait depends on.\n * - Concrete ResourceSpec implementation lives in the Resource namespace; this keeps only trait-perspective dependency info.\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 * - Nodes typically correspond to fields.\n * - Edges represent dependencies between fields, or between fields and resources.\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 * - Structural view of the StateTrait engine: a graph of nodes and dependency edges.\n * - Used by Devtools / Studio / Runtime for structural analysis and visualization.\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 * - The smallest instruction unit used by StateTrait.install / Runtime execution.\n * - Derived from the graph; used at runtime to install watchers or trigger external calls.\n */\nexport interface StateTraitPlanStep {\n  readonly id: string\n  readonly kind: 'computed-update' | 'link-propagate' | 'source-refresh' | 'external-store-sync' | '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 * - Execution plan summarizing all trait behaviors for a module.\n * - install mounts behaviors onto Bound API / EffectOp pipelines according to the plan.\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\nexport type StateTraitSchemaPathKind = 'fieldPath' | 'dep' | 'link_from' | 'check_writeback'\n\nexport interface StateTraitSchemaPathRef {\n  readonly kind: StateTraitSchemaPathKind\n  readonly entryKind: StateTraitKind\n  readonly entryFieldPath: string\n  readonly path: string\n  readonly ruleName?: string\n}\n\n/**\n * StateTraitProgram<S>\uFF1A\n * - Program output of the StateTrait engine; the unified entrypoint for Runtime / Devtools.\n * - stateSchema and spec preserve original inputs; graph/plan are internal IR.\n */\nexport interface StateTraitProgram<S> {\n  readonly stateSchema: Schema.Schema<S, any>\n  readonly spec: StateTraitSpec<S>\n  /**\n   * Program.entries\uFF1A\n   * - Minimal rule set normalized from spec (including node/list/$root) during build.\n   * - Phase 2 focuses on a readable/diagnosable structure; later phases add execution and converge semantics.\n   */\n  readonly entries: ReadonlyArray<StateTraitEntry<any, string>>\n  readonly graph: StateTraitGraph\n  readonly plan: StateTraitPlan\n  readonly convergeIr?: ConvergeStaticIrRegistry\n  readonly convergeExecIr?: ConvergeExecIr\n  /**\n   * schemaPaths\uFF1A\n   * - Path references that must exist in stateSchema, collected from entries during build; dev-only diagnostics (e.g. schema_mismatch).\n   * - When diagnostics=off, validation can be skipped entirely to keep near-zero cost.\n   */\n  readonly schemaPaths?: ReadonlyArray<StateTraitSchemaPathRef>\n}\n\n/**\n * Build a normalized entry list from StateTraitSpec.\n *\n * - Phase 2: supports structural expansion of node/list/$root and fills/prefixes fieldPath/deps when necessary.\n * - Later phases may extend validation here (e.g. detecting duplicates, override rules, etc.).\n */\nexport const normalizeSpec = <S>(spec: StateTraitSpec<S>): 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 = (deps: ReadonlyArray<string> | undefined, prefix: string): 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 = rawDeps !== undefined ? prefixDeps(rawDeps, depPrefix) : 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 = rawDeps !== undefined ? prefixDeps(rawDeps, depPrefix) : undefined\n      return {\n        ...(entry as any),\n        fieldPath,\n        meta: { ...meta, deps, _fieldPath: fieldPath },\n      }\n    }\n    if (entry.kind === 'externalStore') {\n      const meta = entry.meta as any\n      return {\n        ...(entry as any),\n        fieldPath,\n        meta: { ...meta, _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: Phase 2 adjusts fieldPath only; deps are still collected from meta.rules during build.\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       * Prefix for check deps: default semantics for list-scope checks (deps are relative to a row item rather than the list root).\n       *\n       * - item scope: depsPrefix = `${listPath}[]` (matches joinPrefix).\n       * - list scope: fieldPath = `${listPath}`, but depsPrefix should be `${listPath}[]`.\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: StateTraitEntry<any, any> | Readonly<Record<string, StateTraitEntry<any, any>>> | 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    expandMaybeRecord(node.externalStore)\n\n    if (node.check) {\n      const rules: Record<string, CheckRule<any, any>> = {}\n      const checkDepsPrefix = options?.checkDepsPrefix ?? (joinPrefix.endsWith('[]') ? joinPrefix : '')\n\n      const prefixCheckDeps = (deps: ReadonlyArray<string> | undefined): ReadonlyArray<string> => {\n        if (!deps || deps.length === 0) return []\n        return deps.map((d) => (d === '' ? scopeId : checkDepsPrefix ? joinPath(checkDepsPrefix, d) : d))\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        // Invalid input: ignore (later phases may promote this into a build-time config error).\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 * - Extract StateTraitNode.meta (whitelisted fields) from StateTraitSpec for devtools structural display.\n * - meta is for diagnostics/display only and does not participate in runtime semantics.\n */\nexport const collectNodeMeta = <S>(spec: StateTraitSpec<S>): 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 = 'provider' | 'runtime_module' | 'runtime_default' | '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  | 'time_slicing_immediate'\n  | 'time_slicing_deferred'\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 TraitConvergeTimeSlicingSummary {\n  readonly scope: 'all' | 'immediate' | 'deferred'\n  readonly immediateStepCount: number\n  readonly deferredStepCount: number\n}\n\nexport interface TraitConvergeDiagnosticsSamplingSummary {\n  /**\n   * Deterministic sampling: use stable txnSeq as an anchor and sample every N transactions.\n   * - sampled=true means this invocation captured per-step timings and produced hotspots.\n   */\n  readonly strategy: 'txnSeq_interval'\n  readonly sampleEveryN: number\n  readonly topK: number\n  readonly sampled: 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 timeSlicing?: TraitConvergeTimeSlicingSummary\n  readonly thresholds?: { readonly floorRatio?: number }\n  readonly diagnosticsSampling?: TraitConvergeDiagnosticsSamplingSummary\n  readonly top3?: ReadonlyArray<TraitConvergeHotspot>\n}\n", "import { Effect, Fiber, FiberRef, Option } from 'effect'\nimport { create } from 'mutative'\nimport * as EffectOp from '../effect-op.js'\nimport { Snapshot, internal as ResourceInternal, keyHash as hashKey } from '../resource.js'\nimport * as EffectOpCore from '../runtime/core/EffectOpCore.js'\nimport * as Debug from '../runtime/core/DebugSink.js'\nimport * as TaskRunner from '../runtime/core/TaskRunner.js'\nimport { isDevEnv, ReplayModeConfigTag } from '../runtime/core/env.js'\nimport * as ReplayLog from '../runtime/core/ReplayLog.js'\nimport type { PatchReason } from '../runtime/core/StateTransaction.js'\nimport type { FieldPath, FieldPathId } from '../field-path.js'\nimport { normalizeFieldPath } from '../field-path.js'\nimport type { BoundApi } from '../runtime/core/module.js'\nimport { getBoundInternals } from '../runtime/core/runtimeInternalsAccessor.js'\nimport { RunSessionTag } from '../observability/runSession.js'\nimport * as DepsTrace from './deps-trace.js'\nimport * as RowId from './rowid.js'\nimport type { StateTraitEntry, StateTraitPlanStep, StateTraitProgram } from './model.js'\n\nexport interface SourceSyncContext<S> {\n  readonly moduleId?: string\n  readonly instanceId?: string\n  readonly getDraft: () => S\n  readonly setDraft: (next: S) => void\n  readonly recordPatch: (\n    path: string | FieldPath | FieldPathId | undefined,\n    reason: PatchReason,\n    from?: unknown,\n    to?: unknown,\n    traitNodeId?: string,\n    stepId?: number,\n  ) => void\n}\n\nconst onceInRunSession = (key: string): Effect.Effect<boolean, never, any> =>\n  Effect.serviceOption(RunSessionTag).pipe(\n    Effect.map((maybe) => (Option.isSome(maybe) ? maybe.value.local.once(key) : true)),\n  )\n\nconst formatList = (items: ReadonlyArray<string>, limit = 10): string => {\n  if (items.length === 0) return ''\n  if (items.length <= limit) return items.join(', ')\n  return `${items.slice(0, limit).join(', ')}, \u2026(+${items.length - limit})`\n}\n\nconst emitDepsMismatch = (params: {\n  readonly moduleId?: string\n  readonly instanceId?: string\n  readonly kind: 'computed' | 'source'\n  readonly fieldPath: string\n  readonly diff: DepsTrace.DepsDiff\n}): Effect.Effect<void, never, any> =>\n  Effect.gen(function* () {\n    const key = `${params.instanceId ?? 'unknown'}::${params.kind}::${params.fieldPath}`\n    const shouldEmit = yield* onceInRunSession(`deps_mismatch:${key}`)\n    if (!shouldEmit) return\n\n    yield* Debug.record({\n      type: 'diagnostic',\n      moduleId: params.moduleId,\n      instanceId: params.instanceId,\n      code: 'state_trait::deps_mismatch',\n      severity: 'warning',\n      message:\n        `[deps] ${params.kind} \"${params.fieldPath}\" declared=[${formatList(params.diff.declared)}] ` +\n        `reads=[${formatList(params.diff.reads)}] missing=[${formatList(params.diff.missing)}] ` +\n        `unused=[${formatList(params.diff.unused)}]`,\n      hint:\n        'deps is the single source of truth for dependencies: incremental scheduling / reverse closures / performance optimizations rely on deps only. ' +\n        'Keep deps consistent with actual reads; if you really depend on the whole object, declare a coarser-grained dep (e.g. \"profile\") to cover sub-field reads.',\n      kind: `deps_mismatch:${params.kind}`,\n    })\n  })\n\nconst getMiddlewareStack = (): Effect.Effect<EffectOp.MiddlewareStack, never, any> =>\n  Effect.serviceOption(EffectOpCore.EffectOpMiddlewareTag).pipe(\n    Effect.map((maybe) => (Option.isSome(maybe) ? maybe.value.stack : [])),\n  )\n\nconst recordTraitPatch = (\n  bound: BoundApi<any, any>,\n  path: string,\n  reason: PatchReason,\n  from?: unknown,\n  to?: unknown,\n  traitNodeId?: string,\n): void => {\n  const normalized = normalizeFieldPath(path) ?? []\n  try {\n    const internals = getBoundInternals(bound as any)\n    internals.txn.recordStatePatch(normalized, reason, from, to, traitNodeId)\n  } catch {\n    // no-op for legacy/mocked bound\n  }\n}\n\nconst recordReplayEvent = (bound: BoundApi<any, any>, event: ReplayLog.ReplayLogEvent): void => {\n  try {\n    const internals = getBoundInternals(bound as any)\n    internals.txn.recordReplayEvent(event)\n  } catch {\n    // no-op for legacy/mocked bound\n  }\n}\n\nconst getBoundScope = (bound: BoundApi<any, any>): { readonly moduleId?: string; readonly instanceId?: string } => {\n  try {\n    const internals = getBoundInternals(bound as any)\n    return { moduleId: internals.moduleId, instanceId: internals.instanceId }\n  } catch {\n    return { moduleId: undefined, instanceId: undefined }\n  }\n}\n\nconst setSnapshotInTxn = (\n  bound: BoundApi<any, any>,\n  fieldPath: string,\n  next: unknown,\n  reason: PatchReason,\n  stepId: string,\n  traitNodeId?: string,\n): Effect.Effect<boolean, never, any> =>\n  Effect.gen(function* () {\n    let wrote = false\n    yield* bound.state.mutate((draft) => {\n      const prev = RowId.getAtPath(draft, fieldPath)\n      if (Object.is(prev, next)) return\n      wrote = true\n      RowId.setAtPathMutating(draft, fieldPath, next)\n      recordTraitPatch(bound, fieldPath, reason, prev, next, traitNodeId)\n    })\n    return wrote\n  })\n\nconst writebackIfCurrentKeyHash = (\n  bound: BoundApi<any, any>,\n  fieldPath: string,\n  keyHash: string,\n  next: unknown,\n  reason: PatchReason,\n  stepId: string,\n  traitNodeId?: string,\n  replayEvent?: ReplayLog.ReplayLogEvent,\n): Effect.Effect<boolean, never, any> =>\n  Effect.gen(function* () {\n    let wrote = false\n    yield* bound.state.mutate((draft) => {\n      const current = RowId.getAtPath(draft, fieldPath)\n      const currentKeyHash = current && typeof current === 'object' ? (current as any).keyHash : undefined\n      if (currentKeyHash !== keyHash) return\n\n      const prev = current\n      if (Object.is(prev, next)) return\n\n      wrote = true\n      RowId.setAtPathMutating(draft, fieldPath, next)\n      if (replayEvent) {\n        recordReplayEvent(bound, replayEvent)\n      }\n      recordTraitPatch(bound, fieldPath, reason, prev, next, traitNodeId)\n    })\n    return wrote\n  })\n\n/**\n * syncIdleInTransaction\uFF1A\n * - Synchronously evaluate all source.key(state) within the transaction window.\n * - If a key becomes empty (undefined), synchronously reset the field to an idle snapshot (avoid tearing).\n */\nexport const syncIdleInTransaction = <S extends object>(\n  program: StateTraitProgram<S>,\n  ctx: SourceSyncContext<S>,\n): Effect.Effect<void> =>\n  Effect.sync(() => {\n    const draft = ctx.getDraft() as any\n    const updates: Array<{ readonly fieldPath: string; readonly prev: unknown }> = []\n\n    for (const entry of program.entries) {\n      if (entry.kind !== 'source') continue\n      const fieldPath = entry.fieldPath\n      const listItem = RowId.parseListItemFieldPath(fieldPath)\n\n      if (listItem) {\n        // list.item scope: evaluate key per row by index, and synchronously write back idle for inactive rows.\n        const listValue = RowId.getAtPath(draft, listItem.listPath)\n        const items: ReadonlyArray<unknown> = Array.isArray(listValue) ? listValue : []\n\n        for (let index = 0; index < items.length; index++) {\n          const item = items[index]\n\n          let key: unknown\n          try {\n            key = (entry.meta as any).key(item)\n          } catch {\n            continue\n          }\n\n          if (key !== undefined) continue\n\n          const concretePath = RowId.toListItemValuePath(listItem.listPath, index, listItem.itemPath)\n          const prev = RowId.getAtPath(draft, concretePath)\n          const prevStatus = prev && typeof prev === 'object' ? (prev as any).status : undefined\n          if (prevStatus === 'idle') {\n            const data = (prev as any)?.data\n            const error = (prev as any)?.error\n            if (data === undefined && error === undefined) {\n              continue\n            }\n          }\n\n          updates.push({ fieldPath: concretePath, prev })\n        }\n\n        continue\n      }\n\n      let key: unknown\n      try {\n        key = (entry.meta as any).key(draft)\n      } catch {\n        continue\n      }\n\n      if (key !== undefined) continue\n\n      const prev = RowId.getAtPath(draft, fieldPath)\n      const prevStatus = prev && typeof prev === 'object' ? (prev as any).status : undefined\n      if (prevStatus === 'idle') {\n        // Still ensure data/error are cleared.\n        const data = (prev as any)?.data\n        const error = (prev as any)?.error\n        if (data === undefined && error === undefined) {\n          continue\n        }\n      }\n\n      updates.push({ fieldPath, prev })\n    }\n\n    if (updates.length === 0) return\n\n    const reason: PatchReason = 'source-refresh'\n\n    const nextDraft = create(draft, (next) => {\n      for (const u of updates) {\n        RowId.setAtPathMutating(next, u.fieldPath, Snapshot.idle())\n      }\n    })\n\n    ctx.setDraft(nextDraft as S)\n\n    for (const u of updates) {\n      const normalized = normalizeFieldPath(u.fieldPath) ?? []\n      ctx.recordPatch(normalized, reason, u.prev, Snapshot.idle(), `source:${u.fieldPath}:idle`)\n    }\n  })\n\n/**\n * installSourceRefresh\uFF1A\n * - Register the refresh implementation for a single source field (ResourceSnapshot + keyHash gate + concurrency).\n */\nexport const installSourceRefresh = <S>(\n  bound: BoundApi<any, any>,\n  step: StateTraitPlanStep,\n  entry: Extract<StateTraitEntry<S, string>, { readonly kind: 'source' }>,\n): Effect.Effect<void, never, any> => {\n  if (!step.targetFieldPath) return Effect.void\n\n  const fieldPath = step.targetFieldPath\n  const resourceId = step.resourceId ?? entry.meta.resource\n  const listItem = RowId.parseListItemFieldPath(fieldPath)\n\n  let internals: ReturnType<typeof getBoundInternals> | undefined\n  try {\n    internals = getBoundInternals(bound as any)\n  } catch {\n    return Effect.void\n  }\n\n  const register = internals.traits.registerSourceRefresh\n\n  const recordSnapshot = (\n    replayMode: 'live' | 'replay',\n    replayLog: ReplayLog.ReplayLogService | undefined,\n    input:\n      | ReplayLog.ReplayLogEvent\n      | {\n          readonly moduleId?: string\n          readonly instanceId?: string\n          readonly fieldPath: string\n          readonly keyHash?: string\n          readonly concurrency?: string\n          readonly phase: ReplayLog.ResourceSnapshotPhase\n          readonly snapshot: unknown\n        },\n  ): Effect.Effect<void, never, any> => {\n    if (!replayLog) return Effect.void\n    if (replayMode !== 'live') return Effect.void\n    const event: ReplayLog.ReplayLogEvent =\n      input && typeof input === 'object' && (input as any)._tag === 'ResourceSnapshot'\n        ? (input as ReplayLog.ReplayLogEvent)\n        : {\n            _tag: 'ResourceSnapshot',\n            resourceId,\n            fieldPath: (input as any).fieldPath,\n            keyHash: (input as any).keyHash,\n            concurrency: (input as any).concurrency,\n            phase: (input as any).phase,\n            snapshot: (input as any).snapshot,\n            timestamp: Date.now(),\n            moduleId: (input as any).moduleId,\n            instanceId: (input as any).instanceId,\n          }\n    return replayLog.record(event)\n  }\n\n  // list.item scope: in-flight gating by RowID (avoid writing to the wrong row under insert/remove/reorder).\n  if (listItem) {\n    const store = internals.traits.rowIdStore as RowId.RowIdStore | undefined\n    if (!store) {\n      return Effect.void\n    }\n\n    const listPath = listItem.listPath\n    const itemPath = listItem.itemPath\n    if (!itemPath) {\n      // Never write the snapshot back to the whole item (it would overwrite business values).\n      return Effect.void\n    }\n\n    const concurrency = (entry.meta as any).concurrency as 'switch' | 'exhaust-trailing' | undefined\n    const mode = concurrency ?? 'switch'\n\n    const inFlight = new Map<\n      RowId.RowId,\n      {\n        readonly gen: number\n        readonly fiber: Fiber.RuntimeFiber<void, never>\n        readonly keyHash: string\n      }\n    >()\n    const trailing = new Map<RowId.RowId, { readonly key: unknown; readonly keyHash: string }>()\n    let gen = 0\n\n    // When a row is removed: clear trailing/inFlight references to avoid wrong attribution or memory leaks.\n    store.onRemoved(listPath, (rowId) => {\n      trailing.delete(rowId)\n      inFlight.delete(rowId)\n    })\n\n    const setSnapshotForRowInTxn = (\n      rowId: RowId.RowId,\n      next: unknown,\n      reason: PatchReason,\n      stepId: string,\n    ): Effect.Effect<string | undefined, never, any> =>\n      Effect.gen(function* () {\n        let wrotePath: string | undefined\n        yield* bound.state.mutate((draft) => {\n          const index = store.getIndex(listPath, rowId)\n          if (index === undefined) return\n          const concretePath = RowId.toListItemValuePath(listPath, index, itemPath)\n          const prev = RowId.getAtPath(draft, concretePath)\n          if (Object.is(prev, next)) return\n          wrotePath = concretePath\n          RowId.setAtPathMutating(draft, concretePath, next)\n          recordTraitPatch(bound, concretePath, reason, prev, next, step.debugInfo?.graphNodeId)\n        })\n        return wrotePath\n      })\n\n    const writebackIfCurrentKeyHashForRow = (\n      rowId: RowId.RowId,\n      keyHash: string,\n      next: unknown,\n      reason: PatchReason,\n      stepId: string,\n      phase?: ReplayLog.ResourceSnapshotPhase,\n    ): Effect.Effect<string | undefined, never, any> =>\n      Effect.gen(function* () {\n        let wrotePath: string | undefined\n        yield* bound.state.mutate((draft) => {\n          const index = store.getIndex(listPath, rowId)\n          if (index === undefined) return\n          const concretePath = RowId.toListItemValuePath(listPath, index, itemPath)\n\n          const current = RowId.getAtPath(draft, concretePath)\n          const currentKeyHash = current && typeof current === 'object' ? (current as any).keyHash : undefined\n          if (currentKeyHash !== keyHash) return\n\n          const prev = current\n          if (Object.is(prev, next)) return\n\n          wrotePath = concretePath\n          RowId.setAtPathMutating(draft, concretePath, next)\n          if (phase) {\n            const { moduleId, instanceId } = getBoundScope(bound)\n            recordReplayEvent(bound, {\n              _tag: 'ResourceSnapshot',\n              resourceId,\n              fieldPath: concretePath,\n              keyHash,\n              concurrency: mode,\n              phase,\n              snapshot: next,\n              timestamp: Date.now(),\n              moduleId,\n              instanceId,\n            })\n          }\n          recordTraitPatch(bound, concretePath, reason, prev, next, step.debugInfo?.graphNodeId)\n        })\n        return wrotePath\n      })\n\n    const startFetch = (\n      rowId: RowId.RowId,\n      key: unknown,\n      keyHash: string,\n      replayMode: 'live' | 'replay',\n      replayLog: ReplayLog.ReplayLogService | undefined,\n    ): Effect.Effect<void, never, any> =>\n      Effect.gen(function* () {\n        const { moduleId, instanceId } = getBoundScope(bound)\n\n        const indexForLog = store.getIndex(listPath, rowId)\n        const logFieldPath =\n          indexForLog === undefined ? undefined : RowId.toListItemValuePath(listPath, indexForLog, itemPath)\n\n        let loadingSnapshot: unknown = Snapshot.loading({ keyHash })\n        if (replayMode === 'replay' && replayLog && logFieldPath) {\n          const replayLoading = yield* replayLog.consumeNextResourceSnapshot({\n            resourceId,\n            fieldPath: logFieldPath,\n            keyHash,\n            phase: 'loading',\n          })\n          if (replayLoading) {\n            loadingSnapshot = replayLoading.snapshot\n          }\n        }\n        const wroteLoadingPath = yield* setSnapshotForRowInTxn(\n          rowId,\n          loadingSnapshot,\n          'source-refresh',\n          `source:${fieldPath}:${rowId}:loading`,\n        )\n        if (wroteLoadingPath) {\n          const event: ReplayLog.ReplayLogEvent = {\n            _tag: 'ResourceSnapshot',\n            resourceId,\n            fieldPath: wroteLoadingPath,\n            keyHash,\n            concurrency: mode,\n            phase: 'loading',\n            snapshot: loadingSnapshot,\n            timestamp: Date.now(),\n            moduleId,\n            instanceId,\n          }\n          recordReplayEvent(bound, event)\n          yield* recordSnapshot(replayMode, replayLog, event)\n        }\n\n        const io = Effect.gen(function* () {\n          if (replayMode === 'replay' && replayLog) {\n            // Let loading commit become visible first, then replay the settled phase (preserve the async-resource timeline shape).\n            yield* Effect.yieldNow()\n            const consumePath = wroteLoadingPath ?? logFieldPath\n            if (!consumePath) return yield* Effect.void\n\n            const replayed = yield* replayLog.consumeNextResourceSnapshot({\n              resourceId,\n              fieldPath: consumePath,\n              keyHash,\n            })\n            if (!replayed) return yield* Effect.void\n\n            if (replayed.phase === 'success') {\n              yield* writebackIfCurrentKeyHashForRow(\n                rowId,\n                keyHash,\n                replayed.snapshot,\n                'source-refresh',\n                `source:${fieldPath}:${rowId}:success`,\n                'success',\n              )\n            } else if (replayed.phase === 'error') {\n              yield* writebackIfCurrentKeyHashForRow(\n                rowId,\n                keyHash,\n                replayed.snapshot,\n                'source-refresh',\n                `source:${fieldPath}:${rowId}:error`,\n                'error',\n              )\n            }\n\n            return yield* Effect.void\n          }\n\n          const stack = yield* getMiddlewareStack()\n\n          const registryOpt = yield* Effect.serviceOption(ResourceInternal.ResourceRegistryTag)\n          const registry = Option.isSome(registryOpt) ? registryOpt.value : undefined\n          const spec = registry?.specs.get(resourceId)\n\n          if (!spec) {\n            return yield* Effect.void\n          }\n\n          const loadEffect = (spec.load as any)(key) as Effect.Effect<any, any, any>\n\n          const meta: any = {\n            moduleId,\n            instanceId,\n            fieldPath,\n            resourceId,\n            key,\n            keyHash,\n            rowId,\n            traitNodeId: step.debugInfo?.graphNodeId,\n            stepId: step.id,\n          }\n\n          if (!(typeof meta.opSeq === 'number' && Number.isFinite(meta.opSeq))) {\n            const sessionOpt = yield* Effect.serviceOption(RunSessionTag)\n            if (Option.isSome(sessionOpt)) {\n              const seqKey = instanceId ?? 'global'\n              meta.opSeq = sessionOpt.value.local.nextSeq('opSeq', seqKey)\n            }\n          }\n\n          const op = EffectOp.make<any, any, any>({\n            kind: 'service',\n            name: resourceId,\n            effect: loadEffect,\n            meta,\n          })\n\n          const exit = yield* Effect.exit(EffectOp.run(op, stack))\n\n          if (exit._tag === 'Success') {\n            const successSnapshot = Snapshot.success({ keyHash, data: exit.value })\n            const wroteSuccessPath = yield* writebackIfCurrentKeyHashForRow(\n              rowId,\n              keyHash,\n              successSnapshot,\n              'source-refresh',\n              `source:${fieldPath}:${rowId}:success`,\n              'success',\n            )\n            if (wroteSuccessPath) {\n              yield* recordSnapshot(replayMode, replayLog, {\n                _tag: 'ResourceSnapshot',\n                resourceId,\n                fieldPath: wroteSuccessPath,\n                keyHash,\n                concurrency: mode,\n                phase: 'success',\n                snapshot: successSnapshot,\n                timestamp: Date.now(),\n                moduleId,\n                instanceId,\n              })\n            }\n          } else {\n            const errorSnapshot = Snapshot.error({ keyHash, error: exit.cause })\n            const wroteErrorPath = yield* writebackIfCurrentKeyHashForRow(\n              rowId,\n              keyHash,\n              errorSnapshot,\n              'source-refresh',\n              `source:${fieldPath}:${rowId}:error`,\n              'error',\n            )\n            if (wroteErrorPath) {\n              yield* recordSnapshot(replayMode, replayLog, {\n                _tag: 'ResourceSnapshot',\n                resourceId,\n                fieldPath: wroteErrorPath,\n                keyHash,\n                concurrency: mode,\n                phase: 'error',\n                snapshot: errorSnapshot,\n                timestamp: Date.now(),\n                moduleId,\n                instanceId,\n              })\n            }\n          }\n        }).pipe(Effect.catchAllCause(() => Effect.void))\n\n        // list.item: IO fibers must detach from the sync-transaction FiberRef; otherwise they'd be misclassified as \"in txn window\"\n        // and block subsequent writeback entrypoints.\n        const fiber = yield* Effect.forkScoped(Effect.locally(TaskRunner.inSyncTransactionFiber, false)(io))\n        const myGen = (gen += 1)\n        inFlight.set(rowId, { gen: myGen, fiber, keyHash })\n\n        yield* Effect.forkScoped(\n          Effect.locally(\n            TaskRunner.inSyncTransactionFiber,\n            false,\n          )(\n            Fiber.await(fiber).pipe(\n              Effect.zipRight(\n                Effect.sync(() => {\n                  const current = inFlight.get(rowId)\n                  if (current && current.gen === myGen) {\n                    inFlight.delete(rowId)\n                  }\n                }),\n              ),\n              Effect.zipRight(\n                mode === 'exhaust-trailing'\n                  ? Effect.gen(function* () {\n                      const next = trailing.get(rowId)\n                      trailing.delete(rowId)\n                      if (next) {\n                        yield* startFetch(rowId, next.key, next.keyHash, replayMode, replayLog)\n                      }\n                    })\n                  : Effect.void,\n              ),\n              Effect.catchAllCause(() => Effect.void),\n            ),\n          ),\n        )\n      })\n\n    register(fieldPath, (state: any) =>\n      Effect.gen(function* () {\n        const { moduleId, instanceId } = getBoundScope(bound)\n        const replayModeOpt = yield* Effect.serviceOption(ReplayModeConfigTag)\n        const replayMode = Option.isSome(replayModeOpt) ? replayModeOpt.value.mode : 'live'\n        const replayLogOpt = yield* Effect.serviceOption(ReplayLog.ReplayLog)\n        const replayLog = Option.isSome(replayLogOpt) ? replayLogOpt.value : undefined\n        const force = yield* FiberRef.get(TaskRunner.forceSourceRefresh)\n\n        const listValue = RowId.getAtPath(state, listPath)\n        const items: ReadonlyArray<unknown> = Array.isArray(listValue) ? listValue : []\n        const ids = store.ensureList(listPath, items)\n\n        // dev-mode: trace deps once for the first row (diagnostics only; does not affect execution semantics).\n        const traceKey = `${instanceId ?? 'unknown'}::source::${fieldPath}`\n        if (isDevEnv() && (yield* onceInRunSession(`deps_trace_settled:${traceKey}`))) {\n          try {\n            const sample = items[0]\n            if (sample !== undefined) {\n              const traced = DepsTrace.trace((s) => (entry.meta as any).key(s), sample as any)\n              const prefixedReads = traced.reads.map((r) => (r ? `${listPath}[].${r}` : `${listPath}[]`))\n              const diff = DepsTrace.diffDeps(((entry.meta as any).deps ?? []) as ReadonlyArray<string>, prefixedReads)\n              if (diff) {\n                yield* emitDepsMismatch({\n                  moduleId,\n                  instanceId,\n                  kind: 'source',\n                  fieldPath,\n                  diff,\n                })\n              }\n            }\n          } catch {\n            // tracing failure should never break refresh flow\n          }\n        }\n\n        for (let index = 0; index < items.length; index++) {\n          const rowId = ids[index]\n          if (!rowId) continue\n\n          const concretePath = RowId.toListItemValuePath(listPath, index, itemPath)\n          const prevSnapshot = RowId.getAtPath(state, concretePath) as any\n\n          let key: unknown\n          try {\n            key = (entry.meta as any).key(items[index])\n          } catch {\n            key = undefined\n          }\n\n          const current = inFlight.get(rowId)\n\n          if (key === undefined) {\n            trailing.delete(rowId)\n            inFlight.delete(rowId)\n\n            // If it's already clean idle, avoid redundant writeback (prevents meaningless patches and UI jitter).\n            if (\n              prevSnapshot &&\n              typeof prevSnapshot === 'object' &&\n              prevSnapshot.status === 'idle' &&\n              prevSnapshot.data === undefined &&\n              prevSnapshot.error === undefined\n            ) {\n              continue\n            }\n\n            const idleSnapshot = Snapshot.idle()\n            const wroteIdlePath = yield* setSnapshotForRowInTxn(\n              rowId,\n              idleSnapshot,\n              'source-refresh',\n              `source:${fieldPath}:${rowId}:idle`,\n            )\n            if (wroteIdlePath) {\n              const event: ReplayLog.ReplayLogEvent = {\n                _tag: 'ResourceSnapshot',\n                resourceId,\n                fieldPath: wroteIdlePath,\n                keyHash: undefined,\n                concurrency: mode,\n                phase: 'idle',\n                snapshot: idleSnapshot,\n                timestamp: Date.now(),\n                moduleId,\n                instanceId,\n              }\n              recordReplayEvent(bound, event)\n              yield* recordSnapshot(replayMode, replayLog, event)\n            }\n            continue\n          }\n\n          const h = hashKey(key)\n\n          // keyHash unchanged: avoid redundant refresh while keeping in-flight.\n          if (!force && current && current.keyHash === h) {\n            continue\n          }\n\n          // Not in-flight: if snapshot.keyHash already matches, treat it as already up-to-date (avoid full refresh and row jitter).\n          const prevKeyHash =\n            prevSnapshot && typeof prevSnapshot === 'object' ? (prevSnapshot as any).keyHash : undefined\n          if (!force && !current && prevKeyHash === h) {\n            continue\n          }\n\n          if (mode === 'exhaust-trailing' && current) {\n            trailing.set(rowId, { key, keyHash: h })\n            const loadingSnapshot = Snapshot.loading({ keyHash: h })\n            const wroteLoadingPath = yield* setSnapshotForRowInTxn(\n              rowId,\n              loadingSnapshot,\n              'source-refresh',\n              `source:${fieldPath}:${rowId}:loading`,\n            )\n            if (wroteLoadingPath) {\n              const event: ReplayLog.ReplayLogEvent = {\n                _tag: 'ResourceSnapshot',\n                resourceId,\n                fieldPath: wroteLoadingPath,\n                keyHash: h,\n                concurrency: mode,\n                phase: 'loading',\n                snapshot: loadingSnapshot,\n                timestamp: Date.now(),\n                moduleId,\n                instanceId,\n              }\n              recordReplayEvent(bound, event)\n              yield* recordSnapshot(replayMode, replayLog, event)\n            }\n            continue\n          }\n\n          if (mode === 'switch' && current) {\n            // Do not rely on cancellation correctness: stale writebacks are dropped by the keyHash gate.\n            trailing.delete(rowId)\n            inFlight.delete(rowId)\n          }\n\n          yield* startFetch(rowId, key, h, replayMode, replayLog)\n        }\n      }),\n    )\n\n    return Effect.void\n  }\n\n  // in-flight state (per field)\n  let inFlight:\n    | {\n        readonly gen: number\n        readonly fiber: Fiber.RuntimeFiber<void, never>\n        readonly keyHash: string\n      }\n    | undefined\n  let gen = 0\n  let trailing: { readonly key: unknown; readonly keyHash: string } | undefined\n\n  const concurrency = (entry.meta as any).concurrency as 'switch' | 'exhaust-trailing' | undefined\n  const mode = concurrency ?? 'switch'\n\n  const startFetch = (\n    key: unknown,\n    keyHash: string,\n    replayMode: 'live' | 'replay',\n    replayLog: ReplayLog.ReplayLogService | undefined,\n  ): Effect.Effect<void, never, any> =>\n    Effect.gen(function* () {\n      const { moduleId, instanceId } = getBoundScope(bound)\n\n      // 1) pending: synchronously write a loading snapshot (within the current transaction window).\n      let loadingSnapshot: unknown = Snapshot.loading({ keyHash })\n      if (replayMode === 'replay' && replayLog) {\n        const replayLoading = yield* replayLog.consumeNextResourceSnapshot({\n          resourceId,\n          fieldPath,\n          keyHash,\n          phase: 'loading',\n        })\n        if (replayLoading) {\n          loadingSnapshot = replayLoading.snapshot\n        }\n      }\n      const wroteLoading = yield* setSnapshotInTxn(\n        bound,\n        fieldPath,\n        loadingSnapshot,\n        'source-refresh',\n        `source:${fieldPath}:loading`,\n        step.debugInfo?.graphNodeId,\n      )\n      if (wroteLoading) {\n        const event: ReplayLog.ReplayLogEvent = {\n          _tag: 'ResourceSnapshot',\n          resourceId,\n          fieldPath,\n          keyHash,\n          concurrency: mode,\n          phase: 'loading',\n          snapshot: loadingSnapshot,\n          timestamp: Date.now(),\n          moduleId,\n          instanceId,\n        }\n        recordReplayEvent(bound, event)\n        yield* recordSnapshot(replayMode, replayLog, event)\n      }\n\n      // 2) IO: run in a background fiber (avoid blocking the current transaction).\n      const io = Effect.gen(function* () {\n        if (replayMode === 'replay' && replayLog) {\n          // Let loading commit become visible first, then replay the settled phase (preserve the async-resource timeline shape).\n          yield* Effect.yieldNow()\n          const replayed = yield* replayLog.consumeNextResourceSnapshot({\n            resourceId,\n            fieldPath,\n            keyHash,\n          })\n          if (!replayed) return yield* Effect.void\n\n          if (replayed.phase === 'success') {\n            const event: ReplayLog.ReplayLogEvent = {\n              _tag: 'ResourceSnapshot',\n              resourceId,\n              fieldPath,\n              keyHash,\n              concurrency: mode,\n              phase: 'success',\n              snapshot: replayed.snapshot,\n              timestamp: Date.now(),\n              moduleId,\n              instanceId,\n            }\n            yield* writebackIfCurrentKeyHash(\n              bound,\n              fieldPath,\n              keyHash,\n              replayed.snapshot,\n              'source-refresh',\n              `source:${fieldPath}:success`,\n              step.debugInfo?.graphNodeId,\n              event,\n            )\n          } else if (replayed.phase === 'error') {\n            const event: ReplayLog.ReplayLogEvent = {\n              _tag: 'ResourceSnapshot',\n              resourceId,\n              fieldPath,\n              keyHash,\n              concurrency: mode,\n              phase: 'error',\n              snapshot: replayed.snapshot,\n              timestamp: Date.now(),\n              moduleId,\n              instanceId,\n            }\n            yield* writebackIfCurrentKeyHash(\n              bound,\n              fieldPath,\n              keyHash,\n              replayed.snapshot,\n              'source-refresh',\n              `source:${fieldPath}:error`,\n              step.debugInfo?.graphNodeId,\n              event,\n            )\n          }\n\n          return yield* Effect.void\n        }\n\n        const stack = yield* getMiddlewareStack()\n\n        const registryOpt = yield* Effect.serviceOption(ResourceInternal.ResourceRegistryTag)\n        const registry = Option.isSome(registryOpt) ? registryOpt.value : undefined\n        const spec = registry?.specs.get(resourceId)\n\n        if (!spec) {\n          return yield* Effect.void\n        }\n\n        const loadEffect = (spec.load as any)(key) as Effect.Effect<any, any, any>\n\n        const meta: any = {\n          moduleId,\n          instanceId,\n          fieldPath,\n          resourceId,\n          key,\n          keyHash,\n          traitNodeId: step.debugInfo?.graphNodeId,\n          stepId: step.id,\n        }\n\n        if (!(typeof meta.opSeq === 'number' && Number.isFinite(meta.opSeq))) {\n          const sessionOpt = yield* Effect.serviceOption(RunSessionTag)\n          if (Option.isSome(sessionOpt)) {\n            const seqKey = instanceId ?? 'global'\n            meta.opSeq = sessionOpt.value.local.nextSeq('opSeq', seqKey)\n          }\n        }\n\n        const op = EffectOp.make<any, any, any>({\n          kind: 'trait-source',\n          name: resourceId,\n          effect: loadEffect,\n          meta,\n        })\n\n        const exit = yield* Effect.exit(EffectOp.run(op, stack))\n\n        // 3) writeback: use a keyHash gate to prevent stale results from writing back onto a new key.\n        if (exit._tag === 'Success') {\n          const successSnapshot = Snapshot.success({ keyHash, data: exit.value })\n          const event: ReplayLog.ReplayLogEvent = {\n            _tag: 'ResourceSnapshot',\n            resourceId,\n            fieldPath,\n            keyHash,\n            concurrency: mode,\n            phase: 'success',\n            snapshot: successSnapshot,\n            timestamp: Date.now(),\n            moduleId,\n            instanceId,\n          }\n          const wroteSuccess = yield* writebackIfCurrentKeyHash(\n            bound,\n            fieldPath,\n            keyHash,\n            successSnapshot,\n            'source-refresh',\n            `source:${fieldPath}:success`,\n            step.debugInfo?.graphNodeId,\n            event,\n          )\n          if (wroteSuccess) {\n            yield* recordSnapshot(replayMode, replayLog, event)\n          }\n        } else {\n          const errorSnapshot = Snapshot.error({ keyHash, error: exit.cause })\n          const event: ReplayLog.ReplayLogEvent = {\n            _tag: 'ResourceSnapshot',\n            resourceId,\n            fieldPath,\n            keyHash,\n            concurrency: mode,\n            phase: 'error',\n            snapshot: errorSnapshot,\n            timestamp: Date.now(),\n            moduleId,\n            instanceId,\n          }\n          const wroteError = yield* writebackIfCurrentKeyHash(\n            bound,\n            fieldPath,\n            keyHash,\n            errorSnapshot,\n            'source-refresh',\n            `source:${fieldPath}:error`,\n            step.debugInfo?.graphNodeId,\n            event,\n          )\n          if (wroteError) {\n            yield* recordSnapshot(replayMode, replayLog, event)\n          }\n        }\n      }).pipe(Effect.catchAllCause(() => Effect.void))\n\n      // Do not wait for IO completion: forkScoped into the runtime scope so unmount will interrupt automatically.\n      const fiber = yield* Effect.forkScoped(Effect.locally(TaskRunner.inSyncTransactionFiber, false)(io))\n      const myGen = (gen += 1)\n      inFlight = { gen: myGen, fiber, keyHash }\n\n      // After in-flight completes, clean up; in exhaust-trailing mode, run one trailing fetch if present.\n      yield* Effect.forkScoped(\n        Effect.locally(\n          TaskRunner.inSyncTransactionFiber,\n          false,\n        )(\n          Fiber.await(fiber).pipe(\n            Effect.zipRight(\n              Effect.sync(() => {\n                if (inFlight && inFlight.gen === myGen) {\n                  inFlight = undefined\n                }\n              }),\n            ),\n            Effect.zipRight(\n              mode === 'exhaust-trailing'\n                ? Effect.gen(function* () {\n                    const next = trailing\n                    trailing = undefined\n                    if (next) {\n                      yield* startFetch(next.key, next.keyHash, replayMode, replayLog)\n                    }\n                  })\n                : Effect.void,\n            ),\n            Effect.catchAllCause(() => Effect.void),\n          ),\n        ),\n      )\n    })\n\n  register(fieldPath, (state: any) =>\n    Effect.gen(function* () {\n      const { moduleId, instanceId } = getBoundScope(bound)\n      const replayModeOpt = yield* Effect.serviceOption(ReplayModeConfigTag)\n      const replayMode = Option.isSome(replayModeOpt) ? replayModeOpt.value.mode : 'live'\n      const replayLogOpt = yield* Effect.serviceOption(ReplayLog.ReplayLog)\n      const replayLog = Option.isSome(replayLogOpt) ? replayLogOpt.value : undefined\n      const force = yield* FiberRef.get(TaskRunner.forceSourceRefresh)\n\n      let key: unknown\n      try {\n        key = (entry.meta as any).key(state)\n      } catch {\n        key = undefined\n      }\n\n      // dev-mode: detect mismatch between actual reads in keySelector and declared deps (diagnostics only; does not affect execution semantics).\n      const traceKey = `${instanceId ?? 'unknown'}::source::${fieldPath}`\n      if (isDevEnv() && (yield* onceInRunSession(`deps_trace_settled:${traceKey}`))) {\n        try {\n          const traced = DepsTrace.trace((s) => (entry.meta as any).key(s), state)\n          const diff = DepsTrace.diffDeps(((entry.meta as any).deps ?? []) as ReadonlyArray<string>, traced.reads)\n          if (diff) {\n            yield* emitDepsMismatch({\n              moduleId,\n              instanceId,\n              kind: 'source',\n              fieldPath,\n              diff,\n            })\n          }\n        } catch {\n          // tracing failure should never break refresh flow\n        }\n      }\n\n      // Key becomes empty: synchronously clear to idle (and interrupt in-flight).\n      if (key === undefined) {\n        if (inFlight) {\n          yield* Fiber.interruptFork(inFlight.fiber)\n          inFlight = undefined\n        }\n        trailing = undefined\n\n        const idleSnapshot = Snapshot.idle()\n        const wroteIdle = yield* setSnapshotInTxn(\n          bound,\n          fieldPath,\n          idleSnapshot,\n          'source-refresh',\n          `source:${fieldPath}:idle`,\n          step.debugInfo?.graphNodeId,\n        )\n        if (wroteIdle) {\n          const event: ReplayLog.ReplayLogEvent = {\n            _tag: 'ResourceSnapshot',\n            resourceId,\n            fieldPath,\n            keyHash: undefined,\n            concurrency: mode,\n            phase: 'idle',\n            snapshot: idleSnapshot,\n            timestamp: Date.now(),\n            moduleId,\n            instanceId,\n          }\n          recordReplayEvent(bound, event)\n          yield* recordSnapshot(replayMode, replayLog, event)\n        }\n        return\n      }\n\n      const h = hashKey(key)\n\n      // Default semantics: when a non-idle snapshot already exists for the same keyHash, refresh should be a no-op when possible\n      // (avoid duplicate IO/writeback). Explicit refresh/invalidate can bypass via force.\n      if (!force) {\n        if (inFlight && inFlight.keyHash === h) {\n          return\n        }\n\n        const currentSnapshot = RowId.getAtPath(state, fieldPath) as any\n        const currentKeyHash =\n          currentSnapshot && typeof currentSnapshot === 'object' ? (currentSnapshot as any).keyHash : undefined\n        const currentStatus =\n          currentSnapshot && typeof currentSnapshot === 'object' ? (currentSnapshot as any).status : undefined\n        if (currentStatus && currentStatus !== 'idle' && currentKeyHash === h) {\n          return\n        }\n      }\n\n      if (mode === 'exhaust-trailing' && inFlight) {\n        // Busy: record trailing and update loading immediately; stale in-flight writebacks will be blocked by the keyHash gate.\n        trailing = { key, keyHash: h }\n        const loadingSnapshot = Snapshot.loading({ keyHash: h })\n        const wroteLoading = yield* setSnapshotInTxn(\n          bound,\n          fieldPath,\n          loadingSnapshot,\n          'source-refresh',\n          `source:${fieldPath}:loading`,\n          step.debugInfo?.graphNodeId,\n        )\n        if (wroteLoading) {\n          const event: ReplayLog.ReplayLogEvent = {\n            _tag: 'ResourceSnapshot',\n            resourceId,\n            fieldPath,\n            keyHash: h,\n            concurrency: mode,\n            phase: 'loading',\n            snapshot: loadingSnapshot,\n            timestamp: Date.now(),\n            moduleId,\n            instanceId,\n          }\n          recordReplayEvent(bound, event)\n          yield* recordSnapshot(replayMode, replayLog, event)\n        }\n        return\n      }\n\n      if (mode === 'switch' && inFlight) {\n        yield* Fiber.interruptFork(inFlight.fiber)\n        inFlight = undefined\n        trailing = undefined\n      }\n\n      // start fetch (pending tx + fork IO)\n      yield* startFetch(key, h, replayMode, replayLog)\n    }),\n  )\n\n  return Effect.void\n}\n", "export interface DepsTraceResult<T> {\n  readonly value: T\n  readonly reads: ReadonlyArray<string>\n}\n\nexport interface DepsDiff {\n  readonly reads: ReadonlyArray<string>\n  readonly declared: ReadonlyArray<string>\n  readonly missing: ReadonlyArray<string>\n  readonly unused: ReadonlyArray<string>\n}\n\nconst isTraceableObject = (value: unknown): value is object => {\n  if (!value || typeof value !== 'object') return false\n  if (Array.isArray(value)) return true\n  if (value instanceof Date) return false\n  if (value instanceof RegExp) return false\n  if (value instanceof Error) return false\n  if (value instanceof Map) return false\n  if (value instanceof Set) return false\n  if (value instanceof WeakMap) return false\n  if (value instanceof WeakSet) return false\n  return true\n}\n\nconst shouldIgnoreKey = (key: string): boolean => key === '__proto__' || key === 'prototype' || key === 'constructor'\n\nconst normalizeReads = (reads: ReadonlySet<string>): ReadonlyArray<string> => {\n  const all = Array.from(reads).filter((p) => typeof p === 'string' && p.length > 0)\n  all.sort()\n\n  // Drop strict prefix paths (e.g. if both profile and profile.id are read, keep only the more specific profile.id).\n  const isPrefix = (prefix: string, full: string): boolean => full !== prefix && full.startsWith(prefix + '.')\n\n  const pruned: Array<string> = []\n  for (const p of all) {\n    let hasMoreSpecific = false\n    for (const other of all) {\n      if (isPrefix(p, other)) {\n        hasMoreSpecific = true\n        break\n      }\n    }\n    if (!hasMoreSpecific) {\n      pruned.push(p)\n    }\n  }\n\n  pruned.sort()\n  return pruned\n}\n\nconst covers = (declared: string, read: string): boolean => declared === read || read.startsWith(declared + '.')\n\nexport const diffDeps = (declared: ReadonlyArray<string>, reads: ReadonlyArray<string>): DepsDiff | undefined => {\n  const declaredList = Array.from(new Set(declared)).filter((p) => typeof p === 'string' && p.length > 0)\n  declaredList.sort()\n  const readList = Array.from(new Set(reads)).filter((p) => typeof p === 'string' && p.length > 0)\n  readList.sort()\n\n  const missing = readList.filter((r) => declaredList.every((d) => !covers(d, r)))\n  const unused = declaredList.filter((d) => readList.every((r) => !covers(d, r)))\n\n  if (missing.length === 0 && unused.length === 0) return undefined\n\n  return {\n    reads: readList,\n    declared: declaredList,\n    missing,\n    unused,\n  }\n}\n\nexport const trace = <T>(fn: (state: any) => T, state: unknown): DepsTraceResult<T> => {\n  if (!isTraceableObject(state)) {\n    return { value: fn(state as any), reads: [] }\n  }\n\n  const reads = new Set<string>()\n\n  // per-trace caches to preserve reference identity within the traced call.\n  const proxyCache = new WeakMap<object, Map<string, any>>()\n  const proxyToTarget = new WeakMap<object, object>()\n\n  const wrap = (value: unknown, path: string): unknown => {\n    if (!isTraceableObject(value)) return value\n    return getProxy(value as any, path)\n  }\n\n  const unwrap = <V>(value: V): V => {\n    if (value && (typeof value === 'object' || typeof value === 'function')) {\n      const target = proxyToTarget.get(value as any)\n      if (target) return target as any as V\n    }\n    return value\n  }\n\n  const getProxy = (target: object, basePath: string): any => {\n    let byPath = proxyCache.get(target)\n    if (!byPath) {\n      byPath = new Map()\n      proxyCache.set(target, byPath)\n    }\n\n    const cached = byPath.get(basePath)\n    if (cached) return cached\n\n    const record = (path: string) => {\n      if (path) reads.add(path)\n    }\n\n    const proxy = new Proxy(target as any, {\n      get: (t, prop, receiver) => {\n        if (typeof prop === 'symbol') {\n          return Reflect.get(t, prop, receiver)\n        }\n        const key = String(prop)\n        if (shouldIgnoreKey(key)) {\n          return Reflect.get(t, prop, receiver)\n        }\n\n        const nextPath = basePath ? `${basePath}.${key}` : key\n        record(nextPath)\n\n        const value = Reflect.get(t, prop, receiver) as unknown\n        return wrap(value, nextPath)\n      },\n      has: (t, prop) => {\n        if (typeof prop === 'symbol') return Reflect.has(t, prop)\n        const key = String(prop)\n        if (!shouldIgnoreKey(key)) {\n          const nextPath = basePath ? `${basePath}.${key}` : key\n          record(nextPath)\n        }\n        return Reflect.has(t, prop)\n      },\n      ownKeys: (t) => {\n        if (basePath) record(basePath)\n        return Reflect.ownKeys(t)\n      },\n      getOwnPropertyDescriptor: (t, prop) => {\n        if (typeof prop === 'symbol') {\n          return Reflect.getOwnPropertyDescriptor(t, prop)\n        }\n        const key = String(prop)\n        if (!shouldIgnoreKey(key)) {\n          const nextPath = basePath ? `${basePath}.${key}` : key\n          record(nextPath)\n        }\n        return Reflect.getOwnPropertyDescriptor(t, prop)\n      },\n      set: () => {\n        throw new Error(\n          '[deps-trace] Attempted to mutate state during deps tracing (state is readonly in dev-mode diagnostics).',\n        )\n      },\n      defineProperty: () => {\n        throw new Error(\n          '[deps-trace] Attempted to define property on state during deps tracing (state is readonly in dev-mode diagnostics).',\n        )\n      },\n      deleteProperty: () => {\n        throw new Error(\n          '[deps-trace] Attempted to delete property on state during deps tracing (state is readonly in dev-mode diagnostics).',\n        )\n      },\n    })\n\n    byPath.set(basePath, proxy)\n    proxyToTarget.set(proxy, target)\n    return proxy\n  }\n\n  const root = getProxy(state as any, '')\n  const value = unwrap(fn(root))\n\n  return {\n    value,\n    reads: normalizeReads(reads),\n  }\n}\n", "import { Effect, FiberRef, Option } from 'effect'\nimport * as SchemaAST from 'effect/SchemaAST'\nimport * as Debug from '../runtime/core/DebugSink.js'\nimport { isDevEnv } from '../runtime/core/env.js'\nimport { RunSessionTag } from '../observability/runSession.js'\nimport { normalizeFieldPath } from '../field-path.js'\nimport * as DepsTrace from './deps-trace.js'\nimport type { ConvergeContext } from './converge.types.js'\nimport type { StateTraitProgram, StateTraitSchemaPathRef } from './model.js'\n\nconst onceKeysFallback = new Set<string>()\n\nexport const onceInRunSession = (key: string): Effect.Effect<boolean> =>\n  Effect.serviceOption(RunSessionTag).pipe(\n    Effect.map((maybe) => {\n      if (Option.isSome(maybe)) {\n        return maybe.value.local.once(key)\n      }\n      if (onceKeysFallback.has(key)) return false\n      onceKeysFallback.add(key)\n      return true\n    }),\n  )\n\nconst formatList = (items: ReadonlyArray<string>, limit = 10): string => {\n  if (items.length === 0) return ''\n  if (items.length <= limit) return items.join(', ')\n  return `${items.slice(0, limit).join(', ')}, \u2026(+${items.length - limit})`\n}\n\nexport const emitDepsMismatch = (params: {\n  readonly moduleId?: string\n  readonly instanceId?: string\n  readonly kind: 'computed' | 'source'\n  readonly fieldPath: string\n  readonly diff: DepsTrace.DepsDiff\n}): Effect.Effect<void> => {\n  return Effect.gen(function* () {\n    const key = `${params.moduleId ?? 'unknown'}::${params.instanceId ?? 'unknown'}::${params.kind}::${params.fieldPath}`\n    const shouldEmit = yield* onceInRunSession(`deps_mismatch:${key}`)\n    if (!shouldEmit) return\n\n    yield* Debug.record({\n      type: 'diagnostic',\n      moduleId: params.moduleId,\n      instanceId: params.instanceId,\n      code: 'state_trait::deps_mismatch',\n      severity: 'warning',\n      message:\n        `[deps] ${params.kind} \"${params.fieldPath}\" declared=[${formatList(params.diff.declared)}] ` +\n        `reads=[${formatList(params.diff.reads)}] missing=[${formatList(params.diff.missing)}] ` +\n        `unused=[${formatList(params.diff.unused)}]`,\n      hint:\n        'deps is the single source of truth for dependencies: incremental scheduling / reverse closures / performance optimizations rely on deps only. ' +\n        'Keep deps consistent with actual reads; if you really depend on the whole object, declare a coarser-grained dep (e.g. \"profile\") to cover sub-field reads.',\n      kind: `deps_mismatch:${params.kind}`,\n    })\n  })\n}\n\nconst schemaHasPath = (\n  ast: SchemaAST.AST,\n  segments: ReadonlyArray<string>,\n  seen: Set<SchemaAST.AST> = new Set(),\n): boolean => {\n  if (segments.length === 0) return true\n\n  let current = ast\n\n  // unwrap Suspend/Refinement (common for recursive schemas and branded schemas)\n  while (true) {\n    if (SchemaAST.isSuspend(current)) {\n      if (seen.has(current)) {\n        // Recursion: if we can't statically decide further, allow conservatively to avoid false positives.\n        return true\n      }\n      seen.add(current)\n      current = current.f()\n      continue\n    }\n    if (SchemaAST.isRefinement(current)) {\n      current = current.from\n      continue\n    }\n    break\n  }\n\n  // Transformation: prefer `to` (decoded shape), but also allow `from` to reduce false positives.\n  if (SchemaAST.isTransformation(current)) {\n    return schemaHasPath(current.to, segments, seen) || schemaHasPath(current.from, segments, seen)\n  }\n\n  if (SchemaAST.isUnion(current)) {\n    return current.types.some((t) => schemaHasPath(t, segments, seen))\n  }\n\n  if (SchemaAST.isTupleType(current)) {\n    const candidates: Array<SchemaAST.AST> = []\n    for (const e of current.elements) candidates.push(e.type)\n    for (const r of current.rest) candidates.push(r.type)\n    if (candidates.length === 0) return true\n    return candidates.some((t) => schemaHasPath(t, segments, seen))\n  }\n\n  if (SchemaAST.isTypeLiteral(current)) {\n    const [head, ...tail] = segments\n\n    for (const ps of current.propertySignatures) {\n      if (String(ps.name) !== head) continue\n      return schemaHasPath(ps.type, tail, seen)\n    }\n\n    // index signature: open objects like Record<string, T> allow any key\n    for (const sig of current.indexSignatures) {\n      let param: SchemaAST.AST = sig.parameter as unknown as SchemaAST.AST\n      while (SchemaAST.isRefinement(param)) {\n        param = param.from\n      }\n      const tag = (param as any)?._tag\n      if (tag === 'StringKeyword' || tag === 'TemplateLiteral') {\n        return schemaHasPath(sig.type, tail, seen)\n      }\n    }\n\n    return false\n  }\n\n  const tag = (current as any)?._tag\n  if (tag === 'AnyKeyword' || tag === 'UnknownKeyword' || tag === 'ObjectKeyword' || tag === 'Declaration') {\n    return true\n  }\n\n  return false\n}\n\nconst schemaHasFieldPath = (stateSchemaAst: SchemaAST.AST, path: string): boolean => {\n  if (!path) return true\n  if (path === '$root') return true\n\n  const normalized = normalizeFieldPath(path)\n  if (!normalized) return false\n\n  const segs = normalized[0] === '$root' ? normalized.slice(1) : normalized\n  return schemaHasPath(stateSchemaAst, segs)\n}\n\nconst formatSchemaMismatchLine = (ref: StateTraitSchemaPathRef): string => {\n  if (ref.kind === 'fieldPath') {\n    return `- ${ref.entryKind} \"${ref.entryFieldPath}\" fieldPath=\"${ref.path}\"`\n  }\n  if (ref.kind === 'dep') {\n    const rule = ref.ruleName ? ` rule=\"${ref.ruleName}\"` : ''\n    return `- ${ref.entryKind} \"${ref.entryFieldPath}\" deps=\"${ref.path}\"${rule}`\n  }\n  if (ref.kind === 'link_from') {\n    return `- link \"${ref.entryFieldPath}\" from=\"${ref.path}\"`\n  }\n  if (ref.kind === 'check_writeback') {\n    return `- check \"${ref.entryFieldPath}\" writeback=\"${ref.path}\"`\n  }\n  return `- ${ref.entryKind} \"${ref.entryFieldPath}\" path=\"${ref.path}\"`\n}\n\nexport const emitSchemaMismatch = <S extends object>(\n  program: StateTraitProgram<S>,\n  ctx: Pick<ConvergeContext<S>, 'moduleId' | 'instanceId'>,\n): Effect.Effect<void> =>\n  Effect.gen(function* () {\n    if (!isDevEnv()) return\n\n    const level = yield* FiberRef.get(Debug.currentDiagnosticsLevel)\n    if (level === 'off') return\n\n    const key = `${ctx.moduleId ?? 'unknown'}::${ctx.instanceId}`\n    const shouldEmit = yield* onceInRunSession(`schema_mismatch:${key}`)\n    if (!shouldEmit) return\n\n    const refs = (program.schemaPaths ?? []) as ReadonlyArray<StateTraitSchemaPathRef>\n    if (refs.length === 0) return\n\n    const stateSchemaAst = program.stateSchema.ast as unknown as SchemaAST.AST\n\n    const mismatches: Array<StateTraitSchemaPathRef> = []\n    const seen = new Set<string>()\n\n    for (const ref of refs) {\n      if (schemaHasFieldPath(stateSchemaAst, ref.path)) continue\n      const k = `${ref.kind}|${ref.entryKind}|${ref.entryFieldPath}|${ref.ruleName ?? ''}|${ref.path}`\n      if (seen.has(k)) continue\n      seen.add(k)\n      mismatches.push(ref)\n    }\n\n    if (mismatches.length === 0) return\n\n    const limit = level === 'light' ? 8 : 24\n    const lines = mismatches.slice(0, limit).map(formatSchemaMismatchLine)\n    if (mismatches.length > limit) {\n      lines.push(`- \u2026(+${mismatches.length - limit})`)\n    }\n\n    yield* Debug.record({\n      type: 'diagnostic',\n      moduleId: ctx.moduleId,\n      instanceId: ctx.instanceId,\n      code: 'state_trait::schema_mismatch',\n      severity: 'warning',\n      message: `[schema] The following paths are not declared in stateSchema (total ${mismatches.length}):\\n${lines.join('\\n')}`,\n      hint: 'StateTrait writeback will create missing objects/fields. Declare all fieldPath/deps/link.from and errors.* writeback paths in stateSchema, or fix typos in trait paths.',\n      kind: 'schema_mismatch',\n    })\n  })\n"],
  "mappings": "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",
  "names": ["record", "value", "record", "keyHash", "key", "prev", "prevStatus", "concurrency", "mode", "inFlight", "trailing", "gen", "startFetch", "Effect", "FiberRef", "Option", "SchemaAST", "onceInRunSession", "Effect", "Option", "formatList", "emitDepsMismatch", "tag", "FiberRef"]
}

|