@goondocks/myco 0.4.3 → 0.5.0

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.
Files changed (96) hide show
  1. package/.claude-plugin/marketplace.json +1 -1
  2. package/.claude-plugin/plugin.json +1 -1
  3. package/README.md +5 -1
  4. package/dist/chunk-2AMAOSRF.js +105 -0
  5. package/dist/chunk-2AMAOSRF.js.map +1 -0
  6. package/dist/{chunk-I7PNZEBO.js → chunk-6LTNFMXO.js} +12 -1
  7. package/dist/{chunk-I7PNZEBO.js.map → chunk-6LTNFMXO.js.map} +1 -1
  8. package/dist/{chunk-2GJFTIWX.js → chunk-7KQB22DP.js} +2 -2
  9. package/dist/{chunk-JBD5KP5G.js → chunk-B6WVNDA5.js} +14 -2
  10. package/dist/chunk-B6WVNDA5.js.map +1 -0
  11. package/dist/chunk-FIA5NTRH.js +159 -0
  12. package/dist/chunk-FIA5NTRH.js.map +1 -0
  13. package/dist/{chunk-GFBG73P4.js → chunk-FIRMTYFH.js} +3 -3
  14. package/dist/{chunk-XCPQHC4X.js → chunk-HJG7Z6SJ.js} +2 -2
  15. package/dist/chunk-HL2S5QZG.js +385 -0
  16. package/dist/chunk-HL2S5QZG.js.map +1 -0
  17. package/dist/{chunk-WBT5DWGC.js → chunk-IURC35BF.js} +2 -2
  18. package/dist/{chunk-67R6EMYD.js → chunk-JI6M2L2W.js} +31 -52
  19. package/dist/chunk-JI6M2L2W.js.map +1 -0
  20. package/dist/{chunk-FPEDTLQ6.js → chunk-JJL6AMDA.js} +3 -101
  21. package/dist/chunk-JJL6AMDA.js.map +1 -0
  22. package/dist/chunk-KYL67SKZ.js +150 -0
  23. package/dist/chunk-KYL67SKZ.js.map +1 -0
  24. package/dist/{chunk-ZCBL5HER.js → chunk-ND4VK6C7.js} +2 -2
  25. package/dist/{chunk-V2OWD2VV.js → chunk-R6LQT3U7.js} +24 -146
  26. package/dist/chunk-R6LQT3U7.js.map +1 -0
  27. package/dist/{chunk-IYFKPSRP.js → chunk-RCV2I4AI.js} +3 -3
  28. package/dist/{chunk-BNIYWCST.js → chunk-X6TKHO22.js} +2 -2
  29. package/dist/{chunk-OUFSLZTX.js → chunk-ZWUFTOG3.js} +21 -9
  30. package/dist/chunk-ZWUFTOG3.js.map +1 -0
  31. package/dist/{cli-PMOFCZQL.js → cli-BLYNNKGJ.js} +24 -18
  32. package/dist/cli-BLYNNKGJ.js.map +1 -0
  33. package/dist/{client-5SUO2UYH.js → client-5GB4WVXE.js} +5 -5
  34. package/dist/curate-S4HOYWXA.js +231 -0
  35. package/dist/curate-S4HOYWXA.js.map +1 -0
  36. package/dist/{detect-providers-IRL2TTLK.js → detect-providers-BIHYFK5M.js} +3 -3
  37. package/dist/digest-7NKYXM6G.js +96 -0
  38. package/dist/digest-7NKYXM6G.js.map +1 -0
  39. package/dist/{init-NUF5UBUJ.js → init-HPQ77WWF.js} +5 -5
  40. package/dist/{main-2XEBVUR6.js → main-NFQ4II75.js} +253 -576
  41. package/dist/main-NFQ4II75.js.map +1 -0
  42. package/dist/{rebuild-E6YFIRYZ.js → rebuild-KQ6G2GZM.js} +8 -7
  43. package/dist/{rebuild-E6YFIRYZ.js.map → rebuild-KQ6G2GZM.js.map} +1 -1
  44. package/dist/{reprocess-7G7KQWCN.js → reprocess-ZL4HKTSC.js} +95 -24
  45. package/dist/reprocess-ZL4HKTSC.js.map +1 -0
  46. package/dist/{restart-ABW4ZK3P.js → restart-FYW662DR.js} +6 -6
  47. package/dist/{search-MPD7SFK6.js → search-E5JQMTXV.js} +6 -6
  48. package/dist/{server-NZLZRITH.js → server-TV3D35HZ.js} +38 -15
  49. package/dist/{server-NZLZRITH.js.map → server-TV3D35HZ.js.map} +1 -1
  50. package/dist/{session-start-YB4A4PZB.js → session-start-5MFEOVQ5.js} +6 -6
  51. package/dist/{setup-digest-K732MGOJ.js → setup-digest-DZAFIBEF.js} +5 -5
  52. package/dist/{setup-llm-XCCH5LYD.js → setup-llm-4BZM33YT.js} +5 -5
  53. package/dist/src/cli.js +4 -4
  54. package/dist/src/daemon/main.js +4 -4
  55. package/dist/src/hooks/post-tool-use.js +5 -5
  56. package/dist/src/hooks/session-end.js +5 -5
  57. package/dist/src/hooks/session-start.js +4 -4
  58. package/dist/src/hooks/stop.js +6 -6
  59. package/dist/src/hooks/stop.js.map +1 -1
  60. package/dist/src/hooks/user-prompt-submit.js +5 -5
  61. package/dist/src/mcp/server.js +4 -4
  62. package/dist/src/prompts/extraction.md +1 -1
  63. package/dist/src/prompts/summary.md +1 -11
  64. package/dist/src/prompts/supersession.md +32 -0
  65. package/dist/{stats-6G7SN5YZ.js → stats-ZIIJ2GB3.js} +5 -5
  66. package/dist/{verify-JFHQH55Z.js → verify-RACBFT2P.js} +4 -4
  67. package/dist/{version-5B2TWXQJ.js → version-HJTVNPOO.js} +4 -4
  68. package/package.json +1 -1
  69. package/skills/setup/SKILL.md +56 -28
  70. package/skills/setup/references/model-recommendations.md +49 -43
  71. package/dist/chunk-67R6EMYD.js.map +0 -1
  72. package/dist/chunk-FPEDTLQ6.js.map +0 -1
  73. package/dist/chunk-JBD5KP5G.js.map +0 -1
  74. package/dist/chunk-OUFSLZTX.js.map +0 -1
  75. package/dist/chunk-V2OWD2VV.js.map +0 -1
  76. package/dist/cli-PMOFCZQL.js.map +0 -1
  77. package/dist/main-2XEBVUR6.js.map +0 -1
  78. package/dist/reprocess-7G7KQWCN.js.map +0 -1
  79. /package/dist/{chunk-2GJFTIWX.js.map → chunk-7KQB22DP.js.map} +0 -0
  80. /package/dist/{chunk-GFBG73P4.js.map → chunk-FIRMTYFH.js.map} +0 -0
  81. /package/dist/{chunk-XCPQHC4X.js.map → chunk-HJG7Z6SJ.js.map} +0 -0
  82. /package/dist/{chunk-WBT5DWGC.js.map → chunk-IURC35BF.js.map} +0 -0
  83. /package/dist/{chunk-ZCBL5HER.js.map → chunk-ND4VK6C7.js.map} +0 -0
  84. /package/dist/{chunk-IYFKPSRP.js.map → chunk-RCV2I4AI.js.map} +0 -0
  85. /package/dist/{chunk-BNIYWCST.js.map → chunk-X6TKHO22.js.map} +0 -0
  86. /package/dist/{client-5SUO2UYH.js.map → client-5GB4WVXE.js.map} +0 -0
  87. /package/dist/{detect-providers-IRL2TTLK.js.map → detect-providers-BIHYFK5M.js.map} +0 -0
  88. /package/dist/{init-NUF5UBUJ.js.map → init-HPQ77WWF.js.map} +0 -0
  89. /package/dist/{restart-ABW4ZK3P.js.map → restart-FYW662DR.js.map} +0 -0
  90. /package/dist/{search-MPD7SFK6.js.map → search-E5JQMTXV.js.map} +0 -0
  91. /package/dist/{session-start-YB4A4PZB.js.map → session-start-5MFEOVQ5.js.map} +0 -0
  92. /package/dist/{setup-digest-K732MGOJ.js.map → setup-digest-DZAFIBEF.js.map} +0 -0
  93. /package/dist/{setup-llm-XCCH5LYD.js.map → setup-llm-4BZM33YT.js.map} +0 -0
  94. /package/dist/{stats-6G7SN5YZ.js.map → stats-ZIIJ2GB3.js.map} +0 -0
  95. /package/dist/{verify-JFHQH55Z.js.map → verify-RACBFT2P.js.map} +0 -0
  96. /package/dist/{version-5B2TWXQJ.js.map → version-HJTVNPOO.js.map} +0 -0
@@ -12,7 +12,7 @@
12
12
  "source": {
13
13
  "source": "npm",
14
14
  "package": "@goondocks/myco",
15
- "version": "0.4.2"
15
+ "version": "0.4.4"
16
16
  },
17
17
  "description": "Collective agent intelligence — captures session knowledge and serves it back via MCP",
18
18
  "license": "MIT",
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "myco",
3
- "version": "0.4.3",
3
+ "version": "0.5.0",
4
4
  "description": "Collective agent intelligence — captures session knowledge and serves it back to your team via MCP",
5
5
  "author": {
6
6
  "name": "goondocks-co",
package/README.md CHANGED
@@ -50,9 +50,13 @@ myco_context(tier: 3000) → pre-computed project understanding,
50
50
 
51
51
  A background daemon reads your agent's conversation transcript after each turn — the full dialogue including prompts, AI responses, tool calls, and screenshots. Observations called **spores** (decisions, gotchas, discoveries, trade-offs, bug fixes) are extracted automatically via a local LLM and written as linked vault notes.
52
52
 
53
+ ### Curate
54
+
55
+ As a project evolves, older observations become stale. Myco automatically detects and supersedes outdated spores when new ones are created — using vector similarity to find candidates and an LLM to judge which are truly replaced vs. merely related. Superseded spores are preserved with lineage metadata (never deleted), but filtered from search results and digest synthesis. Run `myco curate` for vault-wide cleanup, or let it happen automatically on every spore write.
56
+
53
57
  ### Digest
54
58
 
55
- A **continuous reasoning engine** runs inside the daemon, periodically synthesizing all accumulated knowledge into tiered context extracts. These pre-computed summaries give agents an instant, rich understanding of the project at session start — no searching required. Four tiers serve different needs: executive briefing (1.5K tokens), team standup (3K), deep onboarding (5K), and institutional knowledge (10K).
59
+ A **continuous reasoning engine** runs inside the daemon, periodically synthesizing all accumulated knowledge into tiered context extracts. These pre-computed summaries give agents an instant, rich understanding of the project at session start — no searching required. Four tiers serve different needs: executive briefing (1.5K tokens), team standup (3K), deep onboarding (5K), and institutional knowledge (10K). Run `myco digest --tier 3000` to reprocess a specific tier from scratch, or `myco digest --full` for a complete rebuild.
56
60
 
57
61
  ### Index
58
62
 
@@ -0,0 +1,105 @@
1
+ import { createRequire as __cr } from 'node:module'; const require = __cr(import.meta.url);
2
+ import {
3
+ external_exports
4
+ } from "./chunk-6UJWI4IW.js";
5
+
6
+ // src/vault/types.ts
7
+ var SessionFrontmatterSchema = external_exports.object({
8
+ type: external_exports.literal("session"),
9
+ id: external_exports.string(),
10
+ agent: external_exports.string(),
11
+ user: external_exports.string(),
12
+ started: external_exports.string(),
13
+ ended: external_exports.string().optional(),
14
+ parent: external_exports.string().optional(),
15
+ parent_reason: external_exports.string().optional(),
16
+ plan: external_exports.string().optional(),
17
+ // backward compat read path
18
+ plans: external_exports.array(external_exports.string()).optional(),
19
+ // new: multiple plans
20
+ branch: external_exports.string().optional(),
21
+ tags: external_exports.array(external_exports.string()).default([]),
22
+ tools_used: external_exports.number().int().optional(),
23
+ files_changed: external_exports.number().int().optional()
24
+ });
25
+ var PLAN_STATUSES = ["active", "in_progress", "completed", "abandoned"];
26
+ var PlanFrontmatterSchema = external_exports.object({
27
+ type: external_exports.literal("plan"),
28
+ id: external_exports.string(),
29
+ status: external_exports.enum(PLAN_STATUSES).default("active"),
30
+ created: external_exports.string(),
31
+ author: external_exports.string().optional(),
32
+ tags: external_exports.array(external_exports.string()).default([])
33
+ });
34
+ var OBSERVATION_TYPES = ["gotcha", "bug_fix", "decision", "discovery", "trade_off", "cross-cutting"];
35
+ var SPORE_STATUSES = ["active", "superseded", "archived"];
36
+ var SporeFrontmatterSchema = external_exports.object({
37
+ type: external_exports.literal("spore"),
38
+ id: external_exports.string(),
39
+ observation_type: external_exports.string(),
40
+ status: external_exports.enum(SPORE_STATUSES).default("active"),
41
+ session: external_exports.string().optional(),
42
+ plan: external_exports.string().optional(),
43
+ superseded_by: external_exports.string().optional(),
44
+ consolidated_from: external_exports.array(external_exports.string()).optional(),
45
+ created: external_exports.string(),
46
+ tags: external_exports.array(external_exports.string()).default([])
47
+ });
48
+ var ARTIFACT_TYPES = ["spec", "plan", "rfc", "doc", "other"];
49
+ var ArtifactFrontmatterSchema = external_exports.object({
50
+ type: external_exports.literal("artifact"),
51
+ id: external_exports.string(),
52
+ artifact_type: external_exports.enum(ARTIFACT_TYPES).default("other"),
53
+ source_path: external_exports.string(),
54
+ title: external_exports.string(),
55
+ last_captured_by: external_exports.string(),
56
+ created: external_exports.string(),
57
+ updated: external_exports.string(),
58
+ tags: external_exports.array(external_exports.string()).default([])
59
+ });
60
+ var TeamMemberFrontmatterSchema = external_exports.object({
61
+ type: external_exports.literal("team-member"),
62
+ user: external_exports.string(),
63
+ joined: external_exports.string(),
64
+ role: external_exports.string().optional(),
65
+ tags: external_exports.array(external_exports.string()).default([])
66
+ });
67
+ var schemasByType = {
68
+ session: SessionFrontmatterSchema,
69
+ plan: PlanFrontmatterSchema,
70
+ spore: SporeFrontmatterSchema,
71
+ artifact: ArtifactFrontmatterSchema,
72
+ "team-member": TeamMemberFrontmatterSchema
73
+ };
74
+ function parseNoteFrontmatter(data) {
75
+ const type = data.type;
76
+ const schema = schemasByType[type];
77
+ if (!schema) {
78
+ throw new Error(`Unknown note type: ${type}. Known types: ${Object.keys(schemasByType).join(", ")}`);
79
+ }
80
+ const coerced = coerceDatesToStrings(data);
81
+ return schema.parse(coerced);
82
+ }
83
+ function coerceDatesToStrings(obj) {
84
+ const result = {};
85
+ for (const [key, value] of Object.entries(obj)) {
86
+ if (value instanceof Date) {
87
+ result[key] = value.toISOString();
88
+ } else if (Array.isArray(value)) {
89
+ result[key] = value.map((item) => item instanceof Date ? item.toISOString() : item);
90
+ } else if (value !== null && typeof value === "object" && !Array.isArray(value)) {
91
+ result[key] = coerceDatesToStrings(value);
92
+ } else {
93
+ result[key] = value;
94
+ }
95
+ }
96
+ return result;
97
+ }
98
+
99
+ export {
100
+ PLAN_STATUSES,
101
+ OBSERVATION_TYPES,
102
+ ARTIFACT_TYPES,
103
+ parseNoteFrontmatter
104
+ };
105
+ //# sourceMappingURL=chunk-2AMAOSRF.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/vault/types.ts"],"sourcesContent":["import { z } from 'zod';\n\nexport const SessionFrontmatterSchema = z.object({\n type: z.literal('session'),\n id: z.string(),\n agent: z.string(),\n user: z.string(),\n started: z.string(),\n ended: z.string().optional(),\n parent: z.string().optional(),\n parent_reason: z.string().optional(),\n plan: z.string().optional(), // backward compat read path\n plans: z.array(z.string()).optional(), // new: multiple plans\n branch: z.string().optional(),\n tags: z.array(z.string()).default([]),\n tools_used: z.number().int().optional(),\n files_changed: z.number().int().optional(),\n});\n\nexport const PLAN_STATUSES = ['active', 'in_progress', 'completed', 'abandoned'] as const;\n\nexport const PlanFrontmatterSchema = z.object({\n type: z.literal('plan'),\n id: z.string(),\n status: z.enum(PLAN_STATUSES).default('active'),\n created: z.string(),\n author: z.string().optional(),\n tags: z.array(z.string()).default([]),\n});\n\nexport const OBSERVATION_TYPES = ['gotcha', 'bug_fix', 'decision', 'discovery', 'trade_off', 'cross-cutting'] as const;\n\nexport const SPORE_STATUSES = ['active', 'superseded', 'archived'] as const;\nexport type SporeStatus = (typeof SPORE_STATUSES)[number];\n\nexport const SporeFrontmatterSchema = z.object({\n type: z.literal('spore'),\n id: z.string(),\n observation_type: z.string(),\n status: z.enum(SPORE_STATUSES).default('active'),\n session: z.string().optional(),\n plan: z.string().optional(),\n superseded_by: z.string().optional(),\n consolidated_from: z.array(z.string()).optional(),\n created: z.string(),\n tags: z.array(z.string()).default([]),\n});\n\nexport const ARTIFACT_TYPES = ['spec', 'plan', 'rfc', 'doc', 'other'] as const;\nexport type ArtifactType = (typeof ARTIFACT_TYPES)[number];\n\nexport const ArtifactFrontmatterSchema = z.object({\n type: z.literal('artifact'),\n id: z.string(),\n artifact_type: z.enum(ARTIFACT_TYPES).default('other'),\n source_path: z.string(),\n title: z.string(),\n last_captured_by: z.string(),\n created: z.string(),\n updated: z.string(),\n tags: z.array(z.string()).default([]),\n});\n\nexport const TeamMemberFrontmatterSchema = z.object({\n type: z.literal('team-member'),\n user: z.string(),\n joined: z.string(),\n role: z.string().optional(),\n tags: z.array(z.string()).default([]),\n});\n\nexport type SessionFrontmatter = z.infer<typeof SessionFrontmatterSchema>;\nexport type PlanFrontmatter = z.infer<typeof PlanFrontmatterSchema>;\nexport type SporeFrontmatter = z.infer<typeof SporeFrontmatterSchema>;\nexport type ObservationType = SporeFrontmatter['observation_type'];\nexport type ArtifactFrontmatter = z.infer<typeof ArtifactFrontmatterSchema>;\nexport type TeamMemberFrontmatter = z.infer<typeof TeamMemberFrontmatterSchema>;\n\nexport type NoteFrontmatter =\n | SessionFrontmatter\n | PlanFrontmatter\n | SporeFrontmatter\n | ArtifactFrontmatter\n | TeamMemberFrontmatter;\n\nexport interface VaultNote<T extends NoteFrontmatter = NoteFrontmatter> {\n path: string;\n frontmatter: T;\n content: string;\n}\n\nconst schemasByType: Record<string, z.ZodSchema> = {\n session: SessionFrontmatterSchema,\n plan: PlanFrontmatterSchema,\n spore: SporeFrontmatterSchema,\n artifact: ArtifactFrontmatterSchema,\n 'team-member': TeamMemberFrontmatterSchema,\n};\n\nexport function parseNoteFrontmatter(data: Record<string, unknown>): NoteFrontmatter {\n const type = data.type as string;\n const schema = schemasByType[type];\n if (!schema) {\n throw new Error(`Unknown note type: ${type}. Known types: ${Object.keys(schemasByType).join(', ')}`);\n }\n // gray-matter and YAML.parse return Date objects for ISO date strings.\n // Coerce them to strings before Zod validation.\n const coerced = coerceDatesToStrings(data);\n return schema.parse(coerced) as NoteFrontmatter;\n}\n\nfunction coerceDatesToStrings(obj: Record<string, unknown>): Record<string, unknown> {\n const result: Record<string, unknown> = {};\n for (const [key, value] of Object.entries(obj)) {\n if (value instanceof Date) {\n result[key] = value.toISOString();\n } else if (Array.isArray(value)) {\n result[key] = value.map((item) => item instanceof Date ? item.toISOString() : item);\n } else if (value !== null && typeof value === 'object' && !Array.isArray(value)) {\n result[key] = coerceDatesToStrings(value as Record<string, unknown>);\n } else {\n result[key] = value;\n }\n }\n return result;\n}\n"],"mappings":";;;;;;AAEO,IAAM,2BAA2B,iBAAE,OAAO;AAAA,EAC/C,MAAM,iBAAE,QAAQ,SAAS;AAAA,EACzB,IAAI,iBAAE,OAAO;AAAA,EACb,OAAO,iBAAE,OAAO;AAAA,EAChB,MAAM,iBAAE,OAAO;AAAA,EACf,SAAS,iBAAE,OAAO;AAAA,EAClB,OAAO,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC3B,QAAQ,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,eAAe,iBAAE,OAAO,EAAE,SAAS;AAAA,EACnC,MAAM,iBAAE,OAAO,EAAE,SAAS;AAAA;AAAA,EAC1B,OAAO,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,SAAS;AAAA;AAAA,EACpC,QAAQ,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,MAAM,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AAAA,EACpC,YAAY,iBAAE,OAAO,EAAE,IAAI,EAAE,SAAS;AAAA,EACtC,eAAe,iBAAE,OAAO,EAAE,IAAI,EAAE,SAAS;AAC3C,CAAC;AAEM,IAAM,gBAAgB,CAAC,UAAU,eAAe,aAAa,WAAW;AAExE,IAAM,wBAAwB,iBAAE,OAAO;AAAA,EAC5C,MAAM,iBAAE,QAAQ,MAAM;AAAA,EACtB,IAAI,iBAAE,OAAO;AAAA,EACb,QAAQ,iBAAE,KAAK,aAAa,EAAE,QAAQ,QAAQ;AAAA,EAC9C,SAAS,iBAAE,OAAO;AAAA,EAClB,QAAQ,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,MAAM,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AACtC,CAAC;AAEM,IAAM,oBAAoB,CAAC,UAAU,WAAW,YAAY,aAAa,aAAa,eAAe;AAErG,IAAM,iBAAiB,CAAC,UAAU,cAAc,UAAU;AAG1D,IAAM,yBAAyB,iBAAE,OAAO;AAAA,EAC7C,MAAM,iBAAE,QAAQ,OAAO;AAAA,EACvB,IAAI,iBAAE,OAAO;AAAA,EACb,kBAAkB,iBAAE,OAAO;AAAA,EAC3B,QAAQ,iBAAE,KAAK,cAAc,EAAE,QAAQ,QAAQ;AAAA,EAC/C,SAAS,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC7B,MAAM,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC1B,eAAe,iBAAE,OAAO,EAAE,SAAS;AAAA,EACnC,mBAAmB,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,SAAS;AAAA,EAChD,SAAS,iBAAE,OAAO;AAAA,EAClB,MAAM,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AACtC,CAAC;AAEM,IAAM,iBAAiB,CAAC,QAAQ,QAAQ,OAAO,OAAO,OAAO;AAG7D,IAAM,4BAA4B,iBAAE,OAAO;AAAA,EAChD,MAAM,iBAAE,QAAQ,UAAU;AAAA,EAC1B,IAAI,iBAAE,OAAO;AAAA,EACb,eAAe,iBAAE,KAAK,cAAc,EAAE,QAAQ,OAAO;AAAA,EACrD,aAAa,iBAAE,OAAO;AAAA,EACtB,OAAO,iBAAE,OAAO;AAAA,EAChB,kBAAkB,iBAAE,OAAO;AAAA,EAC3B,SAAS,iBAAE,OAAO;AAAA,EAClB,SAAS,iBAAE,OAAO;AAAA,EAClB,MAAM,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AACtC,CAAC;AAEM,IAAM,8BAA8B,iBAAE,OAAO;AAAA,EAClD,MAAM,iBAAE,QAAQ,aAAa;AAAA,EAC7B,MAAM,iBAAE,OAAO;AAAA,EACf,QAAQ,iBAAE,OAAO;AAAA,EACjB,MAAM,iBAAE,OAAO,EAAE,SAAS;AAAA,EAC1B,MAAM,iBAAE,MAAM,iBAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AACtC,CAAC;AAsBD,IAAM,gBAA6C;AAAA,EACjD,SAAS;AAAA,EACT,MAAM;AAAA,EACN,OAAO;AAAA,EACP,UAAU;AAAA,EACV,eAAe;AACjB;AAEO,SAAS,qBAAqB,MAAgD;AACnF,QAAM,OAAO,KAAK;AAClB,QAAM,SAAS,cAAc,IAAI;AACjC,MAAI,CAAC,QAAQ;AACX,UAAM,IAAI,MAAM,sBAAsB,IAAI,kBAAkB,OAAO,KAAK,aAAa,EAAE,KAAK,IAAI,CAAC,EAAE;AAAA,EACrG;AAGA,QAAM,UAAU,qBAAqB,IAAI;AACzC,SAAO,OAAO,MAAM,OAAO;AAC7B;AAEA,SAAS,qBAAqB,KAAuD;AACnF,QAAM,SAAkC,CAAC;AACzC,aAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,GAAG,GAAG;AAC9C,QAAI,iBAAiB,MAAM;AACzB,aAAO,GAAG,IAAI,MAAM,YAAY;AAAA,IAClC,WAAW,MAAM,QAAQ,KAAK,GAAG;AAC/B,aAAO,GAAG,IAAI,MAAM,IAAI,CAAC,SAAS,gBAAgB,OAAO,KAAK,YAAY,IAAI,IAAI;AAAA,IACpF,WAAW,UAAU,QAAQ,OAAO,UAAU,YAAY,CAAC,MAAM,QAAQ,KAAK,GAAG;AAC/E,aAAO,GAAG,IAAI,qBAAqB,KAAgC;AAAA,IACrE,OAAO;AACL,aAAO,GAAG,IAAI;AAAA,IAChB;AAAA,EACF;AACA,SAAO;AACT;","names":[]}
@@ -201,6 +201,15 @@ ${links.join("\n")}`);
201
201
  sections.push(footerTags(allTags));
202
202
  return sections.join("\n\n");
203
203
  }
204
+ var FOOTER_TAG_PREFIX = "\n#type/";
205
+ function extractSection(body, heading) {
206
+ const start = body.indexOf(heading);
207
+ if (start === -1) return "";
208
+ const section = body.slice(start + heading.length);
209
+ const footerIdx = section.lastIndexOf(FOOTER_TAG_PREFIX);
210
+ if (footerIdx !== -1) return section.slice(0, footerIdx).trim();
211
+ return section.trim();
212
+ }
204
213
  function capitalize(s) {
205
214
  return s.charAt(0).toUpperCase() + s.slice(1);
206
215
  }
@@ -418,8 +427,10 @@ export {
418
427
  bareSessionId,
419
428
  sessionWikilink,
420
429
  sessionRelativePath,
430
+ callout,
421
431
  formatSessionBody,
422
432
  formatSporeBody,
433
+ extractSection,
423
434
  VaultWriter
424
435
  };
425
- //# sourceMappingURL=chunk-I7PNZEBO.js.map
436
+ //# sourceMappingURL=chunk-6LTNFMXO.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/vault/session-id.ts","../src/obsidian/formatter.ts","../src/vault/writer.ts"],"sourcesContent":["/**\n * Session ID convention utilities.\n *\n * The runtime session ID is a bare UUID (e.g., \"abc123\").\n * The vault stores it with a \"session-\" prefix (e.g., \"session-abc123\").\n * These helpers convert between the two forms.\n */\nconst SESSION_PREFIX = 'session-';\n\n/** Convert a bare session ID to a vault note ID: \"abc123\" → \"session-abc123\" */\nexport function sessionNoteId(bareId: string): string {\n if (bareId.startsWith(SESSION_PREFIX)) return bareId;\n return `${SESSION_PREFIX}${bareId}`;\n}\n\n/** Convert a vault note ID to a bare session ID: \"session-abc123\" → \"abc123\" */\nexport function bareSessionId(noteId: string): string {\n if (noteId.startsWith(SESSION_PREFIX)) return noteId.slice(SESSION_PREFIX.length);\n return noteId;\n}\n\n/** Convert a bare session ID to an Obsidian wikilink: \"abc123\" → \"[[session-abc123]]\" */\nexport function sessionWikilink(bareId: string): string {\n return `[[${sessionNoteId(bareId)}]]`;\n}\n\n/** Build the relative vault path for a session note */\nexport function sessionRelativePath(bareId: string, date: string): string {\n return `sessions/${date}/${sessionNoteId(bareId)}.md`;\n}\n\n","/**\n * Pure formatting functions for Obsidian-native vault notes.\n * No I/O, no external dependencies — just string transforms.\n */\nimport type { ArtifactType } from '../vault/types.js';\nimport { sessionNoteId } from '../vault/session-id.js';\n\n// Callout type mapping for observation types\nconst CALLOUT_MAP: Record<string, string> = {\n gotcha: 'warning',\n bug_fix: 'bug',\n decision: 'info',\n discovery: 'tip',\n trade_off: 'question',\n};\n\nexport function observationCalloutType(observationType: string): string {\n return CALLOUT_MAP[observationType] ?? 'note';\n}\n\n/**\n * Escape angle brackets that Obsidian would interpret as HTML tags.\n * Matches `<` followed by a letter, `/`, or `!` (opening/closing tags, comments).\n */\nexport function escapeHtmlTags(text: string): string {\n return text.replace(/<(?=[a-zA-Z/!])/g, '\\\\<');\n}\n\nexport function callout(type: string, title: string, content: string): string {\n const safe = escapeHtmlTags(content);\n const indented = safe.split('\\n').map((line) => `> ${line}`).join('\\n');\n return `> [!${type}] ${title}\\n${indented}`;\n}\n\nexport function inlineField(key: string, value: string): string {\n return `${key}:: ${value}`;\n}\n\nexport function wikilink(target: string, display?: string): string {\n return display ? `[[${target}|${display}]]` : `[[${target}]]`;\n}\n\n/**\n * Normalize an observation_type to a tag-safe form.\n * Frontmatter keeps underscores; tags use hyphens per Obsidian convention.\n */\nfunction tagNormalize(s: string): string {\n return s.replace(/_/g, '-');\n}\n\n/**\n * Sanitize a user/LLM-provided tag for Obsidian compatibility.\n * Obsidian tags cannot contain spaces — replace with slash (nested tag).\n * Strips leading # if present.\n */\nfunction sanitizeTag(raw: string): string {\n const stripped = raw.startsWith('#') ? raw.slice(1) : raw;\n return stripped.replace(/\\s+/g, '/');\n}\n\nexport function buildTags(type: string, subtype: string, extraTags: string[] = []): string[] {\n const tags: string[] = [`type/${type}`];\n\n if (subtype) {\n tags.push(`${type}/${tagNormalize(subtype)}`);\n }\n\n for (const tag of extraTags) {\n const normalized = sanitizeTag(tag);\n if (normalized && !tags.includes(normalized)) {\n tags.push(normalized);\n }\n }\n\n return tags;\n}\n\nexport function footerTags(tags: string[]): string {\n return tags.map((t) => (t.startsWith('#') ? t : `#${t}`)).join(' ');\n}\n\n// --- Session formatting ---\n\nexport interface SessionBodyInput {\n title: string;\n narrative: string;\n sessionId: string;\n user?: string;\n started?: string;\n ended?: string;\n branch?: string;\n relatedMemories?: Array<{ id: string; title: string }>;\n turns: Array<{\n prompt: string;\n toolCount: number;\n aiResponse?: string;\n /** Filenames of images in the vault attachments folder */\n images?: string[];\n }>;\n tags?: string[];\n}\n\nexport function formatSessionBody(input: SessionBodyInput): string {\n const sections: string[] = [];\n\n sections.push(`# ${input.title}`);\n\n if (input.narrative) {\n sections.push(callout('abstract', 'Summary', input.narrative));\n }\n\n // Inline fields\n const fields: string[] = [];\n fields.push(inlineField('Session', wikilink(sessionNoteId(input.sessionId))));\n if (input.user) fields.push(inlineField('User', input.user));\n if (input.started && input.ended) {\n const duration = formatDuration(input.started, input.ended);\n if (duration) fields.push(inlineField('Duration', duration));\n }\n if (input.branch) fields.push(inlineField('Branch', `\\`${input.branch}\\``));\n sections.push(fields.join('\\n'));\n\n // Related spores\n if (input.relatedMemories?.length) {\n const links = input.relatedMemories.map((m) => `- ${wikilink(m.id, m.title)}`);\n sections.push(`## Related Spores\\n${links.join('\\n')}`);\n }\n\n // Conversation turns — always rebuilt from the full transcript.\n // The transcript is the source of truth for the complete conversation.\n if (input.turns.length > 0) {\n const turnLines: string[] = [];\n for (let i = 0; i < input.turns.length; i++) {\n const turn = input.turns[i];\n const turnNum = i + 1;\n turnLines.push(`### Turn ${turnNum}`);\n if (turn.prompt || turn.images?.length) {\n // Build prompt content: text + images + tool count\n const parts: string[] = [];\n if (turn.prompt) parts.push(turn.prompt);\n if (turn.images?.length) {\n parts.push(turn.images.map((f) => `![[${f}]]`).join('\\n'));\n }\n if (turn.toolCount > 0) parts.push(`*${turn.toolCount} tool calls*`);\n turnLines.push(callout('user', 'Prompt', parts.join('\\n\\n')));\n } else if (turn.toolCount > 0) {\n turnLines.push(callout('user', 'Prompt', `*${turn.toolCount} tool calls*`));\n }\n if (turn.aiResponse) {\n turnLines.push(callout('assistant', 'Response', turn.aiResponse));\n }\n }\n sections.push(`## Conversation\\n\\n${turnLines.join('\\n\\n')}`);\n }\n\n // Footer tags\n const allTags = buildTags('session', 'ended', [\n ...(input.user ? [`user/${input.user}`] : []),\n ...(input.tags ?? []),\n ]);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Spore formatting ---\n\nexport interface SporeBodyInput {\n title: string;\n observationType: string;\n content: string;\n sessionId?: string;\n root_cause?: string;\n fix?: string;\n rationale?: string;\n alternatives_rejected?: string;\n gained?: string;\n sacrificed?: string;\n tags?: string[];\n}\n\nexport function formatSporeBody(input: SporeBodyInput): string {\n const sections: string[] = [];\n const calloutType = observationCalloutType(input.observationType);\n const calloutTitle = capitalize(tagNormalize(input.observationType));\n\n sections.push(`# ${input.title}`);\n sections.push(callout(calloutType, calloutTitle, input.content));\n\n // Inline fields\n const fields: string[] = [];\n if (input.sessionId) {\n fields.push(inlineField('Session', wikilink(sessionNoteId(input.sessionId))));\n }\n fields.push(inlineField('Observation', input.observationType));\n if (fields.length > 0) sections.push(fields.join('\\n'));\n\n // Type-specific sub-sections\n if (input.root_cause) sections.push(`## Root Cause\\n${input.root_cause}`);\n if (input.fix) sections.push(`## Fix\\n${input.fix}`);\n if (input.rationale) sections.push(`## Rationale\\n${input.rationale}`);\n if (input.alternatives_rejected) sections.push(`## Alternatives Rejected\\n${input.alternatives_rejected}`);\n if (input.gained) sections.push(`## Gained\\n${input.gained}`);\n if (input.sacrificed) sections.push(`## Sacrificed\\n${input.sacrificed}`);\n\n // Footer tags\n const allTags = buildTags('spore', input.observationType, input.tags ?? []);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Plan formatting ---\n\nexport interface PlanBodyInput {\n id: string;\n status: string;\n author?: string;\n created?: string;\n sessions?: Array<{ id: string; title: string }>;\n content: string;\n tags?: string[];\n}\n\nexport function formatPlanBody(input: PlanBodyInput): string {\n const sections: string[] = [];\n\n // Inline fields block\n const fields: string[] = [];\n fields.push(inlineField('Plan', wikilink(input.id)));\n fields.push(inlineField('Status', input.status));\n if (input.author) fields.push(inlineField('Author', input.author));\n if (input.created) fields.push(inlineField('Created', input.created));\n sections.push(fields.join('\\n'));\n\n // User-provided content body (don't restructure)\n sections.push(input.content);\n\n // Sessions section\n if (input.sessions?.length) {\n const links = input.sessions.map((s) => `- ${wikilink(sessionNoteId(s.id), s.title)}`);\n sections.push(`## Sessions\\n${links.join('\\n')}`);\n }\n\n // Footer tags\n const statusTag = tagNormalize(input.status);\n const allTags = buildTags('plan', statusTag, input.tags ?? []);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Artifact formatting ---\n\nexport interface ArtifactBodyInput {\n id: string;\n title: string;\n artifact_type: ArtifactType;\n source_path: string;\n sessionId: string;\n content: string;\n tags?: string[];\n}\n\nexport function formatArtifactBody(input: ArtifactBodyInput): string {\n const sections: string[] = [];\n\n // Inline fields\n const fields: string[] = [];\n fields.push(inlineField('Artifact', wikilink(input.id)));\n fields.push(inlineField('Source', `\\`${input.source_path}\\``));\n fields.push(inlineField('Type', input.artifact_type));\n fields.push(inlineField('Session', wikilink(sessionNoteId(input.sessionId))));\n sections.push(fields.join('\\n'));\n\n // Body: full content from disk\n sections.push(input.content);\n\n // Footer tags\n const allTags = buildTags('artifact', input.artifact_type, input.tags ?? []);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Team formatting ---\n\nexport interface TeamBodyInput {\n user: string;\n role?: string;\n recentSessions?: Array<{ id: string; title: string }>;\n}\n\nexport function formatTeamBody(input: TeamBodyInput): string {\n const sections: string[] = [];\n\n sections.push(`# ${input.user}`);\n sections.push(callout('info', 'Team Member', input.role ?? 'Contributor'));\n\n // Inline fields\n const fields: string[] = [];\n fields.push(inlineField('User', input.user));\n if (input.role) fields.push(inlineField('Role', input.role));\n sections.push(fields.join('\\n'));\n\n // Recent sessions\n if (input.recentSessions?.length) {\n const links = input.recentSessions.map((s) => `- ${wikilink(sessionNoteId(s.id), s.title)}`);\n sections.push(`## Recent Sessions\\n${links.join('\\n')}`);\n }\n\n // Footer tags\n const allTags = buildTags('team', '', [`user/${input.user}`]);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Helpers ---\n\nfunction capitalize(s: string): string {\n return s.charAt(0).toUpperCase() + s.slice(1);\n}\n\nfunction formatDuration(started: string, ended: string): string {\n const ms = new Date(ended).getTime() - new Date(started).getTime();\n if (isNaN(ms) || ms < 0) return '';\n const totalMinutes = Math.floor(ms / 60000);\n if (totalMinutes < 1) return '<1m';\n const hours = Math.floor(totalMinutes / 60);\n const minutes = totalMinutes % 60;\n if (hours === 0) return `${minutes}m`;\n if (minutes === 0) return `${hours}h`;\n return `${hours}h ${minutes}m`;\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\nimport YAML from 'yaml';\nimport { buildTags, formatTeamBody, formatPlanBody, formatArtifactBody } from '../obsidian/formatter.js';\nimport type { ArtifactType } from './types.js';\nimport { sessionNoteId, sessionRelativePath } from './session-id.js';\n\ninterface WriteSessionInput {\n id: string;\n agent?: string;\n user?: string;\n started: string;\n ended?: string;\n parent?: string;\n parent_reason?: string;\n plans?: string[];\n branch?: string;\n tags?: string[];\n tools_used?: number;\n files_changed?: number;\n summary: string;\n}\n\ninterface WritePlanInput {\n id: string;\n status?: string;\n author?: string;\n tags?: string[];\n content: string;\n}\n\ninterface WriteSporeInput {\n id: string;\n observation_type: string;\n session?: string;\n plan?: string;\n tags?: string[];\n content: string;\n}\n\ninterface WriteArtifactInput {\n id: string;\n artifact_type: ArtifactType;\n source_path: string;\n title: string;\n session: string;\n tags?: string[];\n content: string;\n}\n\ninterface WriteTeamMemberInput {\n user: string;\n role?: string;\n}\n\nexport class VaultWriter {\n constructor(private vaultDir: string) {}\n\n writeSession(input: WriteSessionInput): string {\n const date = input.started.slice(0, 10);\n const relativePath = sessionRelativePath(input.id, date);\n\n const frontmatter: Record<string, unknown> = {\n type: 'session',\n id: input.id,\n agent: input.agent ?? 'claude-code',\n user: input.user ?? '',\n started: input.started,\n };\n if (input.ended) frontmatter.ended = input.ended;\n if (input.parent) frontmatter.parent = input.parent;\n if (input.parent_reason) frontmatter.parent_reason = input.parent_reason;\n if (input.plans?.length) frontmatter.plans = input.plans;\n if (input.branch) frontmatter.branch = input.branch;\n frontmatter.tags = buildTags('session', 'ended', [\n ...(input.user ? [`user/${input.user}`] : []),\n ...(input.tags ?? []),\n ]);\n if (input.tools_used != null) frontmatter.tools_used = input.tools_used;\n if (input.files_changed != null) frontmatter.files_changed = input.files_changed;\n\n this.writeMarkdown(relativePath, frontmatter, input.summary);\n return relativePath;\n }\n\n writePlan(input: WritePlanInput): string {\n const relativePath = `plans/${input.id}.md`;\n const fullPath = path.join(this.vaultDir, relativePath);\n\n const status = input.status ?? 'active';\n let created = new Date().toISOString();\n\n // Preserve created from existing file (idempotent)\n try {\n const existing = fs.readFileSync(fullPath, 'utf-8');\n const fmMatch = existing.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (fmMatch) {\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n if (typeof parsed.created === 'string') created = parsed.created;\n }\n } catch {\n // File doesn't exist yet\n }\n const frontmatter: Record<string, unknown> = {\n type: 'plan',\n id: input.id,\n status,\n created,\n };\n if (input.author) frontmatter.author = input.author;\n frontmatter.tags = buildTags('plan', status, input.tags ?? []);\n\n const body = formatPlanBody({\n id: input.id,\n status,\n author: input.author,\n created,\n content: input.content,\n tags: input.tags,\n });\n\n this.writeMarkdown(relativePath, frontmatter, body);\n return relativePath;\n }\n\n writeSpore(input: WriteSporeInput): string {\n const normalizedType = input.observation_type.replace(/_/g, '-');\n const relativePath = `spores/${normalizedType}/${input.id}.md`;\n const fullPath = path.join(this.vaultDir, relativePath);\n const now = new Date().toISOString();\n\n // Preserve created from existing file (idempotent)\n let created = now;\n try {\n const existing = fs.readFileSync(fullPath, 'utf-8');\n const fmMatch = existing.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (fmMatch) {\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n if (typeof parsed.created === 'string') created = parsed.created;\n }\n } catch {\n // File doesn't exist yet — created = now\n }\n\n const frontmatter: Record<string, unknown> = {\n type: 'spore',\n id: input.id,\n observation_type: input.observation_type,\n created,\n };\n if (input.session) frontmatter.session = input.session;\n if (input.plan) frontmatter.plan = input.plan;\n frontmatter.tags = buildTags('spore', input.observation_type, input.tags ?? []);\n\n this.writeMarkdown(relativePath, frontmatter, input.content);\n return relativePath;\n }\n\n writeArtifact(input: WriteArtifactInput): string {\n const relativePath = `artifacts/${input.id}.md`;\n const fullPath = path.join(this.vaultDir, relativePath);\n const now = new Date().toISOString();\n\n let created = now;\n\n // Preserve created from existing file (latest-wins update)\n try {\n const existing = fs.readFileSync(fullPath, 'utf-8');\n const fmMatch = existing.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (fmMatch) {\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n if (typeof parsed.created === 'string') created = parsed.created;\n }\n } catch {\n // File doesn't exist yet — created = now\n }\n\n const frontmatter: Record<string, unknown> = {\n type: 'artifact',\n id: input.id,\n artifact_type: input.artifact_type,\n source_path: input.source_path,\n title: input.title,\n last_captured_by: sessionNoteId(input.session),\n created,\n updated: now,\n tags: buildTags('artifact', input.artifact_type, input.tags ?? []),\n };\n\n const body = formatArtifactBody({\n id: input.id,\n title: input.title,\n artifact_type: input.artifact_type,\n source_path: input.source_path,\n sessionId: input.session,\n content: input.content,\n tags: input.tags,\n });\n\n this.writeMarkdown(relativePath, frontmatter, body);\n return relativePath;\n }\n\n writeTeamMember(input: WriteTeamMemberInput): string {\n const relativePath = `team/${input.user}.md`;\n\n const frontmatter: Record<string, unknown> = {\n type: 'team-member',\n user: input.user,\n joined: new Date().toISOString(),\n tags: buildTags('team', '', [`user/${input.user}`]),\n };\n if (input.role) frontmatter.role = input.role;\n\n const body = formatTeamBody({\n user: input.user,\n role: input.role,\n });\n\n this.writeMarkdown(relativePath, frontmatter, body);\n return relativePath;\n }\n\n /**\n * Update frontmatter fields on an existing note without touching the body.\n * By default only adds fields that don't exist. Set overwrite=true to replace existing values.\n * Returns true if the update was applied, false if the file doesn't exist.\n */\n updateNoteFrontmatter(relativePath: string, fields: Record<string, unknown>, overwrite = false): boolean {\n const fullPath = path.join(this.vaultDir, relativePath);\n let fileContent: string;\n try {\n fileContent = fs.readFileSync(fullPath, 'utf-8');\n } catch {\n return false;\n }\n\n const fmMatch = fileContent.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (!fmMatch) return false;\n\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n for (const [key, value] of Object.entries(fields)) {\n if (overwrite || parsed[key] === undefined) {\n parsed[key] = value;\n }\n }\n\n const body = fileContent.slice(fmMatch[0].length);\n const fmYaml = YAML.stringify(parsed, { defaultStringType: 'QUOTE_DOUBLE', defaultKeyType: 'PLAIN' }).trim();\n this.atomicWrite(fullPath, `---\\n${fmYaml}\\n---${body}`);\n return true;\n }\n\n /** @deprecated Use updateNoteFrontmatter instead */\n updateSessionFrontmatter(relativePath: string, fields: Record<string, unknown>): boolean {\n return this.updateNoteFrontmatter(relativePath, fields);\n }\n\n private writeMarkdown(relativePath: string, frontmatter: Record<string, unknown>, content: string): void {\n const fullPath = path.join(this.vaultDir, relativePath);\n fs.mkdirSync(path.dirname(fullPath), { recursive: true });\n\n const fmYaml = YAML.stringify(frontmatter, { defaultStringType: 'QUOTE_DOUBLE', defaultKeyType: 'PLAIN' }).trim();\n const file = `---\\n${fmYaml}\\n---\\n\\n${content}\\n`;\n this.atomicWrite(fullPath, file);\n }\n\n /** Write to a temp file then rename — prevents Obsidian from seeing a truncated file mid-write. */\n private atomicWrite(fullPath: string, content: string): void {\n const tmp = `${fullPath}.tmp`;\n fs.writeFileSync(tmp, content, 'utf-8');\n fs.renameSync(tmp, fullPath);\n }\n}\n"],"mappings":"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","names":["YAML"]}
1
+ {"version":3,"sources":["../src/vault/session-id.ts","../src/obsidian/formatter.ts","../src/vault/writer.ts"],"sourcesContent":["/**\n * Session ID convention utilities.\n *\n * The runtime session ID is a bare UUID (e.g., \"abc123\").\n * The vault stores it with a \"session-\" prefix (e.g., \"session-abc123\").\n * These helpers convert between the two forms.\n */\nconst SESSION_PREFIX = 'session-';\n\n/** Convert a bare session ID to a vault note ID: \"abc123\" → \"session-abc123\" */\nexport function sessionNoteId(bareId: string): string {\n if (bareId.startsWith(SESSION_PREFIX)) return bareId;\n return `${SESSION_PREFIX}${bareId}`;\n}\n\n/** Convert a vault note ID to a bare session ID: \"session-abc123\" → \"abc123\" */\nexport function bareSessionId(noteId: string): string {\n if (noteId.startsWith(SESSION_PREFIX)) return noteId.slice(SESSION_PREFIX.length);\n return noteId;\n}\n\n/** Convert a bare session ID to an Obsidian wikilink: \"abc123\" → \"[[session-abc123]]\" */\nexport function sessionWikilink(bareId: string): string {\n return `[[${sessionNoteId(bareId)}]]`;\n}\n\n/** Build the relative vault path for a session note */\nexport function sessionRelativePath(bareId: string, date: string): string {\n return `sessions/${date}/${sessionNoteId(bareId)}.md`;\n}\n\n","/**\n * Pure formatting functions for Obsidian-native vault notes.\n * No I/O, no external dependencies — just string transforms.\n */\nimport type { ArtifactType } from '../vault/types.js';\nimport { sessionNoteId } from '../vault/session-id.js';\n\n// Callout type mapping for observation types\nconst CALLOUT_MAP: Record<string, string> = {\n gotcha: 'warning',\n bug_fix: 'bug',\n decision: 'info',\n discovery: 'tip',\n trade_off: 'question',\n};\n\nexport function observationCalloutType(observationType: string): string {\n return CALLOUT_MAP[observationType] ?? 'note';\n}\n\n/**\n * Escape angle brackets that Obsidian would interpret as HTML tags.\n * Matches `<` followed by a letter, `/`, or `!` (opening/closing tags, comments).\n */\nexport function escapeHtmlTags(text: string): string {\n return text.replace(/<(?=[a-zA-Z/!])/g, '\\\\<');\n}\n\nexport function callout(type: string, title: string, content: string): string {\n const safe = escapeHtmlTags(content);\n const indented = safe.split('\\n').map((line) => `> ${line}`).join('\\n');\n return `> [!${type}] ${title}\\n${indented}`;\n}\n\nexport function inlineField(key: string, value: string): string {\n return `${key}:: ${value}`;\n}\n\nexport function wikilink(target: string, display?: string): string {\n return display ? `[[${target}|${display}]]` : `[[${target}]]`;\n}\n\n/**\n * Normalize an observation_type to a tag-safe form.\n * Frontmatter keeps underscores; tags use hyphens per Obsidian convention.\n */\nfunction tagNormalize(s: string): string {\n return s.replace(/_/g, '-');\n}\n\n/**\n * Sanitize a user/LLM-provided tag for Obsidian compatibility.\n * Obsidian tags cannot contain spaces — replace with slash (nested tag).\n * Strips leading # if present.\n */\nfunction sanitizeTag(raw: string): string {\n const stripped = raw.startsWith('#') ? raw.slice(1) : raw;\n return stripped.replace(/\\s+/g, '/');\n}\n\nexport function buildTags(type: string, subtype: string, extraTags: string[] = []): string[] {\n const tags: string[] = [`type/${type}`];\n\n if (subtype) {\n tags.push(`${type}/${tagNormalize(subtype)}`);\n }\n\n for (const tag of extraTags) {\n const normalized = sanitizeTag(tag);\n if (normalized && !tags.includes(normalized)) {\n tags.push(normalized);\n }\n }\n\n return tags;\n}\n\nexport function footerTags(tags: string[]): string {\n return tags.map((t) => (t.startsWith('#') ? t : `#${t}`)).join(' ');\n}\n\n// --- Session formatting ---\n\nexport interface SessionBodyInput {\n title: string;\n narrative: string;\n sessionId: string;\n user?: string;\n started?: string;\n ended?: string;\n branch?: string;\n relatedMemories?: Array<{ id: string; title: string }>;\n turns: Array<{\n prompt: string;\n toolCount: number;\n aiResponse?: string;\n /** Filenames of images in the vault attachments folder */\n images?: string[];\n }>;\n tags?: string[];\n}\n\nexport function formatSessionBody(input: SessionBodyInput): string {\n const sections: string[] = [];\n\n sections.push(`# ${input.title}`);\n\n if (input.narrative) {\n sections.push(callout('abstract', 'Summary', input.narrative));\n }\n\n // Inline fields\n const fields: string[] = [];\n fields.push(inlineField('Session', wikilink(sessionNoteId(input.sessionId))));\n if (input.user) fields.push(inlineField('User', input.user));\n if (input.started && input.ended) {\n const duration = formatDuration(input.started, input.ended);\n if (duration) fields.push(inlineField('Duration', duration));\n }\n if (input.branch) fields.push(inlineField('Branch', `\\`${input.branch}\\``));\n sections.push(fields.join('\\n'));\n\n // Related spores\n if (input.relatedMemories?.length) {\n const links = input.relatedMemories.map((m) => `- ${wikilink(m.id, m.title)}`);\n sections.push(`## Related Spores\\n${links.join('\\n')}`);\n }\n\n // Conversation turns — always rebuilt from the full transcript.\n // The transcript is the source of truth for the complete conversation.\n if (input.turns.length > 0) {\n const turnLines: string[] = [];\n for (let i = 0; i < input.turns.length; i++) {\n const turn = input.turns[i];\n const turnNum = i + 1;\n turnLines.push(`### Turn ${turnNum}`);\n if (turn.prompt || turn.images?.length) {\n // Build prompt content: text + images + tool count\n const parts: string[] = [];\n if (turn.prompt) parts.push(turn.prompt);\n if (turn.images?.length) {\n parts.push(turn.images.map((f) => `![[${f}]]`).join('\\n'));\n }\n if (turn.toolCount > 0) parts.push(`*${turn.toolCount} tool calls*`);\n turnLines.push(callout('user', 'Prompt', parts.join('\\n\\n')));\n } else if (turn.toolCount > 0) {\n turnLines.push(callout('user', 'Prompt', `*${turn.toolCount} tool calls*`));\n }\n if (turn.aiResponse) {\n turnLines.push(callout('assistant', 'Response', turn.aiResponse));\n }\n }\n sections.push(`## Conversation\\n\\n${turnLines.join('\\n\\n')}`);\n }\n\n // Footer tags\n const allTags = buildTags('session', 'ended', [\n ...(input.user ? [`user/${input.user}`] : []),\n ...(input.tags ?? []),\n ]);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Spore formatting ---\n\nexport interface SporeBodyInput {\n title: string;\n observationType: string;\n content: string;\n sessionId?: string;\n root_cause?: string;\n fix?: string;\n rationale?: string;\n alternatives_rejected?: string;\n gained?: string;\n sacrificed?: string;\n tags?: string[];\n}\n\nexport function formatSporeBody(input: SporeBodyInput): string {\n const sections: string[] = [];\n const calloutType = observationCalloutType(input.observationType);\n const calloutTitle = capitalize(tagNormalize(input.observationType));\n\n sections.push(`# ${input.title}`);\n sections.push(callout(calloutType, calloutTitle, input.content));\n\n // Inline fields\n const fields: string[] = [];\n if (input.sessionId) {\n fields.push(inlineField('Session', wikilink(sessionNoteId(input.sessionId))));\n }\n fields.push(inlineField('Observation', input.observationType));\n if (fields.length > 0) sections.push(fields.join('\\n'));\n\n // Type-specific sub-sections\n if (input.root_cause) sections.push(`## Root Cause\\n${input.root_cause}`);\n if (input.fix) sections.push(`## Fix\\n${input.fix}`);\n if (input.rationale) sections.push(`## Rationale\\n${input.rationale}`);\n if (input.alternatives_rejected) sections.push(`## Alternatives Rejected\\n${input.alternatives_rejected}`);\n if (input.gained) sections.push(`## Gained\\n${input.gained}`);\n if (input.sacrificed) sections.push(`## Sacrificed\\n${input.sacrificed}`);\n\n // Footer tags\n const allTags = buildTags('spore', input.observationType, input.tags ?? []);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Plan formatting ---\n\nexport interface PlanBodyInput {\n id: string;\n status: string;\n author?: string;\n created?: string;\n sessions?: Array<{ id: string; title: string }>;\n content: string;\n tags?: string[];\n}\n\nexport function formatPlanBody(input: PlanBodyInput): string {\n const sections: string[] = [];\n\n // Inline fields block\n const fields: string[] = [];\n fields.push(inlineField('Plan', wikilink(input.id)));\n fields.push(inlineField('Status', input.status));\n if (input.author) fields.push(inlineField('Author', input.author));\n if (input.created) fields.push(inlineField('Created', input.created));\n sections.push(fields.join('\\n'));\n\n // User-provided content body (don't restructure)\n sections.push(input.content);\n\n // Sessions section\n if (input.sessions?.length) {\n const links = input.sessions.map((s) => `- ${wikilink(sessionNoteId(s.id), s.title)}`);\n sections.push(`## Sessions\\n${links.join('\\n')}`);\n }\n\n // Footer tags\n const statusTag = tagNormalize(input.status);\n const allTags = buildTags('plan', statusTag, input.tags ?? []);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Artifact formatting ---\n\nexport interface ArtifactBodyInput {\n id: string;\n title: string;\n artifact_type: ArtifactType;\n source_path: string;\n sessionId: string;\n content: string;\n tags?: string[];\n}\n\nexport function formatArtifactBody(input: ArtifactBodyInput): string {\n const sections: string[] = [];\n\n // Inline fields\n const fields: string[] = [];\n fields.push(inlineField('Artifact', wikilink(input.id)));\n fields.push(inlineField('Source', `\\`${input.source_path}\\``));\n fields.push(inlineField('Type', input.artifact_type));\n fields.push(inlineField('Session', wikilink(sessionNoteId(input.sessionId))));\n sections.push(fields.join('\\n'));\n\n // Body: full content from disk\n sections.push(input.content);\n\n // Footer tags\n const allTags = buildTags('artifact', input.artifact_type, input.tags ?? []);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Team formatting ---\n\nexport interface TeamBodyInput {\n user: string;\n role?: string;\n recentSessions?: Array<{ id: string; title: string }>;\n}\n\nexport function formatTeamBody(input: TeamBodyInput): string {\n const sections: string[] = [];\n\n sections.push(`# ${input.user}`);\n sections.push(callout('info', 'Team Member', input.role ?? 'Contributor'));\n\n // Inline fields\n const fields: string[] = [];\n fields.push(inlineField('User', input.user));\n if (input.role) fields.push(inlineField('Role', input.role));\n sections.push(fields.join('\\n'));\n\n // Recent sessions\n if (input.recentSessions?.length) {\n const links = input.recentSessions.map((s) => `- ${wikilink(sessionNoteId(s.id), s.title)}`);\n sections.push(`## Recent Sessions\\n${links.join('\\n')}`);\n }\n\n // Footer tags\n const allTags = buildTags('team', '', [`user/${input.user}`]);\n sections.push(footerTags(allTags));\n\n return sections.join('\\n\\n');\n}\n\n// --- Section extraction ---\n\n/** Footer tag prefix used to terminate section extraction. */\nconst FOOTER_TAG_PREFIX = '\\n#type/';\n\n/**\n * Extract a named section from a session/spore note body.\n * Returns content between the section heading and the footer tags.\n */\nexport function extractSection(body: string, heading: string): string {\n const start = body.indexOf(heading);\n if (start === -1) return '';\n const section = body.slice(start + heading.length);\n const footerIdx = section.lastIndexOf(FOOTER_TAG_PREFIX);\n if (footerIdx !== -1) return section.slice(0, footerIdx).trim();\n return section.trim();\n}\n\n// --- Helpers ---\n\nfunction capitalize(s: string): string {\n return s.charAt(0).toUpperCase() + s.slice(1);\n}\n\nfunction formatDuration(started: string, ended: string): string {\n const ms = new Date(ended).getTime() - new Date(started).getTime();\n if (isNaN(ms) || ms < 0) return '';\n const totalMinutes = Math.floor(ms / 60000);\n if (totalMinutes < 1) return '<1m';\n const hours = Math.floor(totalMinutes / 60);\n const minutes = totalMinutes % 60;\n if (hours === 0) return `${minutes}m`;\n if (minutes === 0) return `${hours}h`;\n return `${hours}h ${minutes}m`;\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\nimport YAML from 'yaml';\nimport { buildTags, formatTeamBody, formatPlanBody, formatArtifactBody } from '../obsidian/formatter.js';\nimport type { ArtifactType } from './types.js';\nimport { sessionNoteId, sessionRelativePath } from './session-id.js';\n\ninterface WriteSessionInput {\n id: string;\n agent?: string;\n user?: string;\n started: string;\n ended?: string;\n parent?: string;\n parent_reason?: string;\n plans?: string[];\n branch?: string;\n tags?: string[];\n tools_used?: number;\n files_changed?: number;\n summary: string;\n}\n\ninterface WritePlanInput {\n id: string;\n status?: string;\n author?: string;\n tags?: string[];\n content: string;\n}\n\ninterface WriteSporeInput {\n id: string;\n observation_type: string;\n session?: string;\n plan?: string;\n tags?: string[];\n content: string;\n}\n\ninterface WriteArtifactInput {\n id: string;\n artifact_type: ArtifactType;\n source_path: string;\n title: string;\n session: string;\n tags?: string[];\n content: string;\n}\n\ninterface WriteTeamMemberInput {\n user: string;\n role?: string;\n}\n\nexport class VaultWriter {\n constructor(private vaultDir: string) {}\n\n writeSession(input: WriteSessionInput): string {\n const date = input.started.slice(0, 10);\n const relativePath = sessionRelativePath(input.id, date);\n\n const frontmatter: Record<string, unknown> = {\n type: 'session',\n id: input.id,\n agent: input.agent ?? 'claude-code',\n user: input.user ?? '',\n started: input.started,\n };\n if (input.ended) frontmatter.ended = input.ended;\n if (input.parent) frontmatter.parent = input.parent;\n if (input.parent_reason) frontmatter.parent_reason = input.parent_reason;\n if (input.plans?.length) frontmatter.plans = input.plans;\n if (input.branch) frontmatter.branch = input.branch;\n frontmatter.tags = buildTags('session', 'ended', [\n ...(input.user ? [`user/${input.user}`] : []),\n ...(input.tags ?? []),\n ]);\n if (input.tools_used != null) frontmatter.tools_used = input.tools_used;\n if (input.files_changed != null) frontmatter.files_changed = input.files_changed;\n\n this.writeMarkdown(relativePath, frontmatter, input.summary);\n return relativePath;\n }\n\n writePlan(input: WritePlanInput): string {\n const relativePath = `plans/${input.id}.md`;\n const fullPath = path.join(this.vaultDir, relativePath);\n\n const status = input.status ?? 'active';\n let created = new Date().toISOString();\n\n // Preserve created from existing file (idempotent)\n try {\n const existing = fs.readFileSync(fullPath, 'utf-8');\n const fmMatch = existing.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (fmMatch) {\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n if (typeof parsed.created === 'string') created = parsed.created;\n }\n } catch {\n // File doesn't exist yet\n }\n const frontmatter: Record<string, unknown> = {\n type: 'plan',\n id: input.id,\n status,\n created,\n };\n if (input.author) frontmatter.author = input.author;\n frontmatter.tags = buildTags('plan', status, input.tags ?? []);\n\n const body = formatPlanBody({\n id: input.id,\n status,\n author: input.author,\n created,\n content: input.content,\n tags: input.tags,\n });\n\n this.writeMarkdown(relativePath, frontmatter, body);\n return relativePath;\n }\n\n writeSpore(input: WriteSporeInput): string {\n const normalizedType = input.observation_type.replace(/_/g, '-');\n const relativePath = `spores/${normalizedType}/${input.id}.md`;\n const fullPath = path.join(this.vaultDir, relativePath);\n const now = new Date().toISOString();\n\n // Preserve created from existing file (idempotent)\n let created = now;\n try {\n const existing = fs.readFileSync(fullPath, 'utf-8');\n const fmMatch = existing.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (fmMatch) {\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n if (typeof parsed.created === 'string') created = parsed.created;\n }\n } catch {\n // File doesn't exist yet — created = now\n }\n\n const frontmatter: Record<string, unknown> = {\n type: 'spore',\n id: input.id,\n observation_type: input.observation_type,\n created,\n };\n if (input.session) frontmatter.session = input.session;\n if (input.plan) frontmatter.plan = input.plan;\n frontmatter.tags = buildTags('spore', input.observation_type, input.tags ?? []);\n\n this.writeMarkdown(relativePath, frontmatter, input.content);\n return relativePath;\n }\n\n writeArtifact(input: WriteArtifactInput): string {\n const relativePath = `artifacts/${input.id}.md`;\n const fullPath = path.join(this.vaultDir, relativePath);\n const now = new Date().toISOString();\n\n let created = now;\n\n // Preserve created from existing file (latest-wins update)\n try {\n const existing = fs.readFileSync(fullPath, 'utf-8');\n const fmMatch = existing.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (fmMatch) {\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n if (typeof parsed.created === 'string') created = parsed.created;\n }\n } catch {\n // File doesn't exist yet — created = now\n }\n\n const frontmatter: Record<string, unknown> = {\n type: 'artifact',\n id: input.id,\n artifact_type: input.artifact_type,\n source_path: input.source_path,\n title: input.title,\n last_captured_by: sessionNoteId(input.session),\n created,\n updated: now,\n tags: buildTags('artifact', input.artifact_type, input.tags ?? []),\n };\n\n const body = formatArtifactBody({\n id: input.id,\n title: input.title,\n artifact_type: input.artifact_type,\n source_path: input.source_path,\n sessionId: input.session,\n content: input.content,\n tags: input.tags,\n });\n\n this.writeMarkdown(relativePath, frontmatter, body);\n return relativePath;\n }\n\n writeTeamMember(input: WriteTeamMemberInput): string {\n const relativePath = `team/${input.user}.md`;\n\n const frontmatter: Record<string, unknown> = {\n type: 'team-member',\n user: input.user,\n joined: new Date().toISOString(),\n tags: buildTags('team', '', [`user/${input.user}`]),\n };\n if (input.role) frontmatter.role = input.role;\n\n const body = formatTeamBody({\n user: input.user,\n role: input.role,\n });\n\n this.writeMarkdown(relativePath, frontmatter, body);\n return relativePath;\n }\n\n /**\n * Update frontmatter fields on an existing note without touching the body.\n * By default only adds fields that don't exist. Set overwrite=true to replace existing values.\n * Returns true if the update was applied, false if the file doesn't exist.\n */\n updateNoteFrontmatter(relativePath: string, fields: Record<string, unknown>, overwrite = false): boolean {\n const fullPath = path.join(this.vaultDir, relativePath);\n let fileContent: string;\n try {\n fileContent = fs.readFileSync(fullPath, 'utf-8');\n } catch {\n return false;\n }\n\n const fmMatch = fileContent.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (!fmMatch) return false;\n\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n for (const [key, value] of Object.entries(fields)) {\n if (overwrite || parsed[key] === undefined) {\n parsed[key] = value;\n }\n }\n\n const body = fileContent.slice(fmMatch[0].length);\n const fmYaml = YAML.stringify(parsed, { defaultStringType: 'QUOTE_DOUBLE', defaultKeyType: 'PLAIN' }).trim();\n this.atomicWrite(fullPath, `---\\n${fmYaml}\\n---${body}`);\n return true;\n }\n\n /** @deprecated Use updateNoteFrontmatter instead */\n updateSessionFrontmatter(relativePath: string, fields: Record<string, unknown>): boolean {\n return this.updateNoteFrontmatter(relativePath, fields);\n }\n\n private writeMarkdown(relativePath: string, frontmatter: Record<string, unknown>, content: string): void {\n const fullPath = path.join(this.vaultDir, relativePath);\n fs.mkdirSync(path.dirname(fullPath), { recursive: true });\n\n const fmYaml = YAML.stringify(frontmatter, { defaultStringType: 'QUOTE_DOUBLE', defaultKeyType: 'PLAIN' }).trim();\n const file = `---\\n${fmYaml}\\n---\\n\\n${content}\\n`;\n this.atomicWrite(fullPath, file);\n }\n\n /** Write to a temp file then rename — prevents Obsidian from seeing a truncated file mid-write. */\n private atomicWrite(fullPath: string, content: string): void {\n const tmp = `${fullPath}.tmp`;\n fs.writeFileSync(tmp, content, 'utf-8');\n fs.renameSync(tmp, fullPath);\n }\n}\n"],"mappings":"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","names":["YAML"]}
@@ -1,7 +1,7 @@
1
1
  import { createRequire as __cr } from 'node:module'; const require = __cr(import.meta.url);
2
2
  import {
3
3
  AgentRegistry
4
- } from "./chunk-BNIYWCST.js";
4
+ } from "./chunk-X6TKHO22.js";
5
5
 
6
6
  // src/version.ts
7
7
  import fs from "fs";
@@ -30,4 +30,4 @@ function readVersionFrom(dir) {
30
30
  export {
31
31
  getPluginVersion
32
32
  };
33
- //# sourceMappingURL=chunk-2GJFTIWX.js.map
33
+ //# sourceMappingURL=chunk-7KQB22DP.js.map
@@ -26,6 +26,7 @@ var FILE_WATCH_STABILITY_MS = 1e3;
26
26
  var PROVIDER_DETECT_TIMEOUT_MS = 3e3;
27
27
  var STALE_BUFFER_MAX_AGE_MS = 24 * 60 * 60 * 1e3;
28
28
  var DAEMON_HEALTH_RETRY_DELAYS = [100, 200, 400, 800, 1500];
29
+ var DAEMON_STALE_GRACE_PERIOD_MS = 6e4;
29
30
  var MAX_SLUG_LENGTH = 100;
30
31
  var CANDIDATE_CONTENT_PREVIEW = 2e3;
31
32
  var LINEAGE_RECENT_SESSIONS_LIMIT = 5;
@@ -51,6 +52,11 @@ var DIGEST_SUBSTRATE_TYPE_WEIGHTS = {
51
52
  artifact: 1,
52
53
  team: 1
53
54
  };
55
+ var LLM_REASONING_MODE = "off";
56
+ var SUPERSESSION_CANDIDATE_LIMIT = 5;
57
+ var SUPERSESSION_VECTOR_FETCH_LIMIT = 20;
58
+ var SUPERSESSION_MAX_TOKENS = 256;
59
+ var CURATION_CLUSTER_SIMILARITY = 0.75;
54
60
 
55
61
  export {
56
62
  CHARS_PER_TOKEN,
@@ -76,6 +82,7 @@ export {
76
82
  PROVIDER_DETECT_TIMEOUT_MS,
77
83
  STALE_BUFFER_MAX_AGE_MS,
78
84
  DAEMON_HEALTH_RETRY_DELAYS,
85
+ DAEMON_STALE_GRACE_PERIOD_MS,
79
86
  MAX_SLUG_LENGTH,
80
87
  CANDIDATE_CONTENT_PREVIEW,
81
88
  LINEAGE_RECENT_SESSIONS_LIMIT,
@@ -89,6 +96,11 @@ export {
89
96
  MCP_LOGS_DEFAULT_LIMIT,
90
97
  DIGEST_TIERS,
91
98
  DIGEST_TIER_MIN_CONTEXT,
92
- DIGEST_SUBSTRATE_TYPE_WEIGHTS
99
+ DIGEST_SUBSTRATE_TYPE_WEIGHTS,
100
+ LLM_REASONING_MODE,
101
+ SUPERSESSION_CANDIDATE_LIMIT,
102
+ SUPERSESSION_VECTOR_FETCH_LIMIT,
103
+ SUPERSESSION_MAX_TOKENS,
104
+ CURATION_CLUSTER_SIMILARITY
93
105
  };
94
- //# sourceMappingURL=chunk-JBD5KP5G.js.map
106
+ //# sourceMappingURL=chunk-B6WVNDA5.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/constants.ts"],"sourcesContent":["/**\n * Shared constants for the Myco codebase.\n * Per CLAUDE.md: \"No Magic Literals — Numeric and string constants\n * MUST NOT appear inline in logic.\"\n */\n\n// --- Token estimation ---\n/** Approximate characters per token for the chars/4 heuristic. */\nexport const CHARS_PER_TOKEN = 4;\n\n/** Estimate token count from character length using the CHARS_PER_TOKEN heuristic. */\nexport function estimateTokens(text: string): number {\n return Math.ceil(text.length / CHARS_PER_TOKEN);\n}\n\n// --- Embedding ---\n/** Max characters of text sent to the embedding model. */\nexport const EMBEDDING_INPUT_LIMIT = 8000;\n\n// --- Truncation limits (display/preview) ---\n/** Max chars for a user prompt preview in event summaries. */\nexport const PROMPT_PREVIEW_CHARS = 300;\n/** Max chars for an AI response preview in event summaries. */\nexport const AI_RESPONSE_PREVIEW_CHARS = 500;\n/** Max chars for a command string preview. */\nexport const COMMAND_PREVIEW_CHARS = 80;\n/** Max chars for a content snippet in search results. */\nexport const CONTENT_SNIPPET_CHARS = 120;\n/** Max chars for a tool output preview in hooks. */\nexport const TOOL_OUTPUT_PREVIEW_CHARS = 200;\n/** Max chars for a session summary preview in MCP tools. */\nexport const SESSION_SUMMARY_PREVIEW_CHARS = 300;\n/** Max chars for a recall summary preview. */\nexport const RECALL_SUMMARY_PREVIEW_CHARS = 200;\n\n// --- Context injection layer budgets (chars, not tokens — used with .slice()) ---\nexport const CONTEXT_PLAN_PREVIEW_CHARS = 100;\nexport const CONTEXT_SESSION_PREVIEW_CHARS = 80;\nexport const CONTEXT_SPORE_PREVIEW_CHARS = 80;\n\n// --- Processor maxTokens budgets ---\n/** Response token budget for observation extraction. */\nexport const EXTRACTION_MAX_TOKENS = 2048;\n/** Response token budget for session summary. */\nexport const SUMMARY_MAX_TOKENS = 512;\n/** Response token budget for session title generation. */\nexport const TITLE_MAX_TOKENS = 32;\n/** Response token budget for artifact classification. */\nexport const CLASSIFICATION_MAX_TOKENS = 1024;\n\n// --- Timeouts ---\n/** Daemon client HTTP request timeout (ms). */\nexport const DAEMON_CLIENT_TIMEOUT_MS = 2000;\n/** Health check timeout (ms) — fail fast if daemon isn't responding. */\nexport const DAEMON_HEALTH_CHECK_TIMEOUT_MS = 500;\n/** LLM request timeout (ms). All LLM calls are background daemon work — no need to be aggressive. */\nexport const LLM_REQUEST_TIMEOUT_MS = 180_000;\n/** Embedding request timeout (ms). Embeddings run in background batch processing — generous timeout. */\nexport const EMBEDDING_REQUEST_TIMEOUT_MS = 60_000;\n/** Digest LLM request timeout (ms). Digest cycles use large context windows and may need model loading time. */\nexport const DIGEST_LLM_REQUEST_TIMEOUT_MS = 600_000;\n/** Stdin read timeout for hooks (ms). */\nexport const STDIN_TIMEOUT_MS = 100;\n/** Chokidar write stability threshold (ms). */\nexport const FILE_WATCH_STABILITY_MS = 1000;\n/** Provider detection timeout for detect-providers CLI command (ms). */\nexport const PROVIDER_DETECT_TIMEOUT_MS = 3000;\n\n// --- Buffer cleanup ---\n/** Max age for stale buffer files before cleanup (ms). */\nexport const STALE_BUFFER_MAX_AGE_MS = 24 * 60 * 60 * 1000;\n\n// --- Retry backoff ---\n/** Retry delays for daemon health check (ms). */\nexport const DAEMON_HEALTH_RETRY_DELAYS = [100, 200, 400, 800, 1500];\n\n/** Grace period after daemon.json is written before stale checks can trigger a restart (ms).\n * Prevents rapid restart loops from concurrent hooks or session reloads. */\nexport const DAEMON_STALE_GRACE_PERIOD_MS = 60_000;\n\n// --- Slug limits ---\n/** Max length for slugified artifact IDs. */\nexport const MAX_SLUG_LENGTH = 100;\n\n// --- Content preview for classification prompt ---\n/** Max chars of file content per candidate in classification prompt. */\nexport const CANDIDATE_CONTENT_PREVIEW = 2000;\n\n// --- Transcript mining ---\n/** Minimum content length to consider a transcript entry meaningful. */\nexport const MIN_TRANSCRIPT_CONTENT_LENGTH = 10;\n\n// --- Query limits ---\n/** Max recent sessions to check for lineage heuristics. */\nexport const LINEAGE_RECENT_SESSIONS_LIMIT = 5;\n/** Max related spores to query for session notes. */\nexport const RELATED_SPORES_LIMIT = 50;\n\n// --- Context injection ---\n/** Max active plans to inject at session start. */\nexport const SESSION_CONTEXT_MAX_PLANS = 3;\n/** Max spores to inject per prompt. */\nexport const PROMPT_CONTEXT_MAX_SPORES = 3;\n/** Minimum similarity score for prompt context injection (0-1). */\nexport const PROMPT_CONTEXT_MIN_SIMILARITY = 0.3;\n/** Max token budget for session-start context injection. */\nexport const SESSION_CONTEXT_MAX_TOKENS = 500;\n/** Max token budget for per-prompt context injection. */\nexport const PROMPT_CONTEXT_MAX_TOKENS = 300;\n/** Minimum prompt length to trigger context search. */\nexport const PROMPT_CONTEXT_MIN_LENGTH = 10;\n\n// --- MCP tool defaults ---\n/** Default result limit for myco_search. */\nexport const MCP_SEARCH_DEFAULT_LIMIT = 10;\n/** Default result limit for myco_sessions. */\nexport const MCP_SESSIONS_DEFAULT_LIMIT = 20;\n/** Default result limit for myco_logs. */\nexport const MCP_LOGS_DEFAULT_LIMIT = 50;\n\n// --- Digest — Tiers ---\n/** Available token-budget tiers for digest synthesis. */\nexport const DIGEST_TIERS = [1500, 3000, 5000, 10000] as const;\nexport type DigestTier = (typeof DIGEST_TIERS)[number];\n\n// --- Digest — Context window minimums per tier ---\n/** Minimum context window (tokens) required to run a digest at a given tier. */\nexport const DIGEST_TIER_MIN_CONTEXT: Record<number, number> = {\n 1500: 6500,\n 3000: 11500,\n 5000: 18500,\n 10000: 30500,\n};\n\n// --- Digest — Substrate ---\n/** Scoring weights by note type when selecting substrate for synthesis. */\nexport const DIGEST_SUBSTRATE_TYPE_WEIGHTS: Record<string, number> = {\n session: 3,\n spore: 3,\n plan: 2,\n artifact: 1,\n team: 1,\n};\n\n// --- LLM reasoning control ---\n/** Reasoning mode for all Myco LLM calls. Suppresses chain-of-thought tokens from reasoning models. */\nexport const LLM_REASONING_MODE = 'off' as const;\n\n// --- Digest — System prompt overhead estimate ---\n\n// --- Vault curation ---\n/** Max candidate spores after post-filtering for supersession check. */\nexport const SUPERSESSION_CANDIDATE_LIMIT = 5;\n\n/** Over-fetch from vector index before post-filtering by status/type. */\nexport const SUPERSESSION_VECTOR_FETCH_LIMIT = 20;\n\n/** Max output tokens for supersession LLM evaluation. */\nexport const SUPERSESSION_MAX_TOKENS = 256;\n\n/** Similarity threshold for clustering related spores in batch curation. */\nexport const CURATION_CLUSTER_SIMILARITY = 0.75;\n"],"mappings":";;;AAQO,IAAM,kBAAkB;AAGxB,SAAS,eAAe,MAAsB;AACnD,SAAO,KAAK,KAAK,KAAK,SAAS,eAAe;AAChD;AAIO,IAAM,wBAAwB;AAI9B,IAAM,uBAAuB;AAE7B,IAAM,4BAA4B;AAElC,IAAM,wBAAwB;AAE9B,IAAM,wBAAwB;AAE9B,IAAM,4BAA4B;AAElC,IAAM,gCAAgC;AAEtC,IAAM,+BAA+B;AAGrC,IAAM,6BAA6B;AACnC,IAAM,gCAAgC;AACtC,IAAM,8BAA8B;AAcpC,IAAM,2BAA2B;AAEjC,IAAM,iCAAiC;AAEvC,IAAM,yBAAyB;AAE/B,IAAM,+BAA+B;AAErC,IAAM,gCAAgC;AAEtC,IAAM,mBAAmB;AAEzB,IAAM,0BAA0B;AAEhC,IAAM,6BAA6B;AAInC,IAAM,0BAA0B,KAAK,KAAK,KAAK;AAI/C,IAAM,6BAA6B,CAAC,KAAK,KAAK,KAAK,KAAK,IAAI;AAI5D,IAAM,+BAA+B;AAIrC,IAAM,kBAAkB;AAIxB,IAAM,4BAA4B;AAQlC,IAAM,gCAAgC;AAEtC,IAAM,uBAAuB;AAI7B,IAAM,4BAA4B;AAElC,IAAM,4BAA4B;AAElC,IAAM,gCAAgC;AAMtC,IAAM,4BAA4B;AAIlC,IAAM,2BAA2B;AAEjC,IAAM,6BAA6B;AAEnC,IAAM,yBAAyB;AAI/B,IAAM,eAAe,CAAC,MAAM,KAAM,KAAM,GAAK;AAK7C,IAAM,0BAAkD;AAAA,EAC7D,MAAM;AAAA,EACN,KAAM;AAAA,EACN,KAAM;AAAA,EACN,KAAO;AACT;AAIO,IAAM,gCAAwD;AAAA,EACnE,SAAS;AAAA,EACT,OAAO;AAAA,EACP,MAAM;AAAA,EACN,UAAU;AAAA,EACV,MAAM;AACR;AAIO,IAAM,qBAAqB;AAM3B,IAAM,+BAA+B;AAGrC,IAAM,kCAAkC;AAGxC,IAAM,0BAA0B;AAGhC,IAAM,8BAA8B;","names":[]}
@@ -0,0 +1,159 @@
1
+ import { createRequire as __cr } from 'node:module'; const require = __cr(import.meta.url);
2
+ import {
3
+ indexNote
4
+ } from "./chunk-JJL6AMDA.js";
5
+ import {
6
+ generateEmbedding
7
+ } from "./chunk-RGVBGTD6.js";
8
+ import {
9
+ loadPrompt,
10
+ stripReasoningTokens
11
+ } from "./chunk-KYL67SKZ.js";
12
+ import {
13
+ external_exports,
14
+ require_dist
15
+ } from "./chunk-6UJWI4IW.js";
16
+ import {
17
+ EMBEDDING_INPUT_LIMIT,
18
+ LLM_REASONING_MODE,
19
+ SUPERSESSION_CANDIDATE_LIMIT,
20
+ SUPERSESSION_MAX_TOKENS,
21
+ SUPERSESSION_VECTOR_FETCH_LIMIT
22
+ } from "./chunk-B6WVNDA5.js";
23
+ import {
24
+ __toESM
25
+ } from "./chunk-PZUWP5VK.js";
26
+
27
+ // src/vault/curation.ts
28
+ var import_yaml = __toESM(require_dist(), 1);
29
+ import fs from "fs";
30
+ import path from "path";
31
+ var supersededIdsSchema = external_exports.array(external_exports.string());
32
+ var SUPERSESSION_NOTICE_MARKER = "Superseded by::";
33
+ function isActiveSpore(frontmatter) {
34
+ const status = frontmatter.status;
35
+ return !status || status === "active";
36
+ }
37
+ function supersedeSpore(targetId, newSporeId, targetPath, deps) {
38
+ const fullPath = path.join(deps.vaultDir, targetPath);
39
+ let fileContent;
40
+ try {
41
+ fileContent = fs.readFileSync(fullPath, "utf-8");
42
+ } catch {
43
+ return false;
44
+ }
45
+ const fmMatch = fileContent.match(/^---\n([\s\S]*?)\n---/);
46
+ if (!fmMatch) return false;
47
+ const parsed = import_yaml.default.parse(fmMatch[1]);
48
+ parsed.status = "superseded";
49
+ parsed.superseded_by = newSporeId;
50
+ const fmYaml = import_yaml.default.stringify(parsed, { defaultStringType: "QUOTE_DOUBLE", defaultKeyType: "PLAIN" }).trim();
51
+ let body = fileContent.slice(fmMatch[0].length);
52
+ if (!body.includes(SUPERSESSION_NOTICE_MARKER)) {
53
+ const notice = `
54
+
55
+ > [!warning] Superseded
56
+ > This observation has been superseded.
57
+
58
+ ${SUPERSESSION_NOTICE_MARKER} [[${newSporeId}]]`;
59
+ body = body.trimEnd() + notice + "\n";
60
+ }
61
+ const tmp = `${fullPath}.tmp`;
62
+ fs.writeFileSync(tmp, `---
63
+ ${fmYaml}
64
+ ---${body}`, "utf-8");
65
+ fs.renameSync(tmp, fullPath);
66
+ indexNote(deps.index, deps.vaultDir, targetPath);
67
+ deps.vectorIndex?.delete(targetId);
68
+ return true;
69
+ }
70
+ async function checkSupersession(newSporeId, deps) {
71
+ const { index, vectorIndex, embeddingProvider, llmProvider, vaultDir, log } = deps;
72
+ if (!vectorIndex || !llmProvider) {
73
+ log?.("debug", "checkSupersession: skipped \u2014 vectorIndex or llmProvider unavailable", { newSporeId });
74
+ return [];
75
+ }
76
+ const newSporeResults = index.queryByIds([newSporeId]);
77
+ if (newSporeResults.length === 0) {
78
+ log?.("warn", "checkSupersession: new spore not found in index", { newSporeId });
79
+ return [];
80
+ }
81
+ const newSpore = newSporeResults[0];
82
+ const observationType = newSpore.frontmatter["observation_type"];
83
+ const embeddingText = newSpore.content.slice(0, EMBEDDING_INPUT_LIMIT);
84
+ const embeddingResult = await generateEmbedding(embeddingProvider, embeddingText);
85
+ const vectorResults = vectorIndex.search(embeddingResult.embedding, {
86
+ type: "spore",
87
+ limit: SUPERSESSION_VECTOR_FETCH_LIMIT
88
+ });
89
+ if (vectorResults.length === 0) {
90
+ log?.("debug", "checkSupersession: no vector results", { newSporeId });
91
+ return [];
92
+ }
93
+ const candidateIds = vectorResults.map((r) => r.id);
94
+ const candidateNotes = index.queryByIds(candidateIds);
95
+ const filtered = candidateNotes.filter((note) => {
96
+ if (note.id === newSporeId) return false;
97
+ if (!isActiveSpore(note.frontmatter)) return false;
98
+ if (observationType && note.frontmatter["observation_type"] !== observationType) return false;
99
+ return true;
100
+ }).slice(0, SUPERSESSION_CANDIDATE_LIMIT);
101
+ if (filtered.length === 0) {
102
+ log?.("debug", "checkSupersession: no candidates after filtering", { newSporeId, observationType });
103
+ return [];
104
+ }
105
+ const template = loadPrompt("supersession");
106
+ const newSporeText = `[${newSpore.id}] ${newSpore.title}
107
+ ${newSpore.content}`;
108
+ const candidatesText = filtered.map((c) => `[${c.id}] ${c.title}
109
+ ${c.content}`).join("\n\n");
110
+ const prompt = template.replace("{{new_spore}}", newSporeText).replace("{{candidates}}", candidatesText);
111
+ let responseText;
112
+ try {
113
+ const response = await llmProvider.summarize(prompt, {
114
+ maxTokens: SUPERSESSION_MAX_TOKENS,
115
+ reasoning: LLM_REASONING_MODE
116
+ });
117
+ responseText = stripReasoningTokens(response.text);
118
+ } catch (err) {
119
+ log?.("warn", "checkSupersession: LLM call failed", { newSporeId, error: String(err) });
120
+ return [];
121
+ }
122
+ let rawIds;
123
+ try {
124
+ rawIds = JSON.parse(responseText);
125
+ } catch {
126
+ log?.("warn", "checkSupersession: failed to parse LLM response", { newSporeId, responseText });
127
+ return [];
128
+ }
129
+ const parsed = supersededIdsSchema.safeParse(rawIds);
130
+ if (!parsed.success) {
131
+ log?.("warn", "checkSupersession: LLM response failed schema validation", { newSporeId });
132
+ return [];
133
+ }
134
+ const candidateMap = new Map(filtered.map((c) => [c.id, c]));
135
+ const validIds = parsed.data.filter((id) => candidateMap.has(id));
136
+ if (validIds.length === 0) {
137
+ return [];
138
+ }
139
+ const supersededIds = [];
140
+ for (const id of validIds) {
141
+ const candidate = candidateMap.get(id);
142
+ const wrote = supersedeSpore(id, newSporeId, candidate.path, { index, vectorIndex, vaultDir });
143
+ if (!wrote) {
144
+ log?.("warn", "checkSupersession: file not found for candidate, skipping", { id, path: candidate.path });
145
+ continue;
146
+ }
147
+ supersededIds.push(id);
148
+ log?.("info", "checkSupersession: marked superseded", { supersededId: id, newSporeId });
149
+ }
150
+ return supersededIds;
151
+ }
152
+
153
+ export {
154
+ supersededIdsSchema,
155
+ isActiveSpore,
156
+ supersedeSpore,
157
+ checkSupersession
158
+ };
159
+ //# sourceMappingURL=chunk-FIA5NTRH.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/vault/curation.ts"],"sourcesContent":["/**\n * Vault curation — supersession detection pipeline.\n *\n * Given a newly written spore ID, finds older spores of the same observation_type\n * that have been rendered outdated, and marks them as superseded.\n */\n\nimport { z } from 'zod';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport YAML from 'yaml';\nimport type { MycoIndex, IndexedNote } from '../index/sqlite.js';\nimport type { VectorIndex } from '../index/vectors.js';\nimport type { LlmProvider, EmbeddingProvider } from '../intelligence/llm.js';\nimport { generateEmbedding } from '../intelligence/embeddings.js';\nimport { stripReasoningTokens } from '../intelligence/response.js';\nimport { indexNote } from '../index/rebuild.js';\nimport { loadPrompt } from '../prompts/index.js';\nimport {\n SUPERSESSION_CANDIDATE_LIMIT,\n SUPERSESSION_VECTOR_FETCH_LIMIT,\n SUPERSESSION_MAX_TOKENS,\n EMBEDDING_INPUT_LIMIT,\n LLM_REASONING_MODE,\n} from '../constants.js';\n\ntype LogLevel = 'debug' | 'info' | 'warn';\ntype LogFn = (level: LogLevel, message: string, data?: Record<string, unknown>) => void;\n\n/** Zod schema for validating LLM supersession responses. */\nexport const supersededIdsSchema = z.array(z.string());\n\n/** Marker string used to detect whether a supersession notice has already been appended. */\nexport const SUPERSESSION_NOTICE_MARKER = 'Superseded by::';\n\n/** Returns true if a spore should be considered active (including legacy spores without status). */\nexport function isActiveSpore(frontmatter: Record<string, unknown>): boolean {\n const status = frontmatter.status as string | undefined;\n return !status || status === 'active';\n}\n\n/**\n * Mark a single spore as superseded: update frontmatter + append notice in a\n * single read-modify-write, then re-index and remove from vector index.\n * Returns true if the write succeeded, false if the file was not found.\n */\nexport function supersedeSpore(\n targetId: string,\n newSporeId: string,\n targetPath: string,\n deps: { index: MycoIndex; vectorIndex: VectorIndex | null; vaultDir: string },\n): boolean {\n const fullPath = path.join(deps.vaultDir, targetPath);\n let fileContent: string;\n try {\n fileContent = fs.readFileSync(fullPath, 'utf-8');\n } catch {\n return false;\n }\n\n // Parse and update frontmatter\n const fmMatch = fileContent.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (!fmMatch) return false;\n\n const parsed = YAML.parse(fmMatch[1]) as Record<string, unknown>;\n parsed.status = 'superseded';\n parsed.superseded_by = newSporeId;\n\n const fmYaml = YAML.stringify(parsed, { defaultStringType: 'QUOTE_DOUBLE', defaultKeyType: 'PLAIN' }).trim();\n let body = fileContent.slice(fmMatch[0].length);\n\n // Append supersession notice (idempotent: skip if already present)\n if (!body.includes(SUPERSESSION_NOTICE_MARKER)) {\n const notice = `\\n\\n> [!warning] Superseded\\n> This observation has been superseded.\\n\\n${SUPERSESSION_NOTICE_MARKER} [[${newSporeId}]]`;\n body = body.trimEnd() + notice + '\\n';\n }\n\n // Atomic write (temp file + rename)\n const tmp = `${fullPath}.tmp`;\n fs.writeFileSync(tmp, `---\\n${fmYaml}\\n---${body}`, 'utf-8');\n fs.renameSync(tmp, fullPath);\n\n // Re-index the updated file and remove stale vector\n indexNote(deps.index, deps.vaultDir, targetPath);\n deps.vectorIndex?.delete(targetId);\n\n return true;\n}\n\n/**\n * Check whether the newly written spore with `newSporeId` supersedes any\n * existing active spores of the same observation_type.\n *\n * Returns the list of spore IDs that were marked superseded.\n */\nexport async function checkSupersession(\n newSporeId: string,\n deps: {\n index: MycoIndex;\n vectorIndex: VectorIndex | null;\n embeddingProvider: EmbeddingProvider;\n llmProvider: LlmProvider | null;\n vaultDir: string;\n log?: LogFn;\n },\n): Promise<string[]> {\n const { index, vectorIndex, embeddingProvider, llmProvider, vaultDir, log } = deps;\n\n // Early-exit if no vector index or LLM available\n if (!vectorIndex || !llmProvider) {\n log?.('debug', 'checkSupersession: skipped — vectorIndex or llmProvider unavailable', { newSporeId });\n return [];\n }\n\n // Look up the new spore to get its content and observation_type\n const newSporeResults = index.queryByIds([newSporeId]);\n if (newSporeResults.length === 0) {\n log?.('warn', 'checkSupersession: new spore not found in index', { newSporeId });\n return [];\n }\n const newSpore = newSporeResults[0];\n const observationType = newSpore.frontmatter['observation_type'] as string | undefined;\n\n // Embed the spore content for similarity search\n const embeddingText = newSpore.content.slice(0, EMBEDDING_INPUT_LIMIT);\n const embeddingResult = await generateEmbedding(embeddingProvider, embeddingText);\n\n // Fetch candidate spore IDs from vector index\n const vectorResults = vectorIndex.search(embeddingResult.embedding, {\n type: 'spore',\n limit: SUPERSESSION_VECTOR_FETCH_LIMIT,\n });\n\n if (vectorResults.length === 0) {\n log?.('debug', 'checkSupersession: no vector results', { newSporeId });\n return [];\n }\n\n const candidateIds = vectorResults.map((r) => r.id);\n\n // Look up candidate notes and post-filter:\n // - same observation_type as the new spore\n // - active status (including legacy spores without status field)\n // - not the new spore itself\n const candidateNotes = index.queryByIds(candidateIds);\n const filtered = candidateNotes\n .filter((note) => {\n if (note.id === newSporeId) return false;\n if (!isActiveSpore(note.frontmatter)) return false;\n if (observationType && note.frontmatter['observation_type'] !== observationType) return false;\n return true;\n })\n .slice(0, SUPERSESSION_CANDIDATE_LIMIT);\n\n if (filtered.length === 0) {\n log?.('debug', 'checkSupersession: no candidates after filtering', { newSporeId, observationType });\n return [];\n }\n\n // Build the supersession prompt\n const template = loadPrompt('supersession');\n const newSporeText = `[${newSpore.id}] ${newSpore.title}\\n${newSpore.content}`;\n const candidatesText = filtered\n .map((c) => `[${c.id}] ${c.title}\\n${c.content}`)\n .join('\\n\\n');\n\n const prompt = template\n .replace('{{new_spore}}', newSporeText)\n .replace('{{candidates}}', candidatesText);\n\n // Ask the LLM which candidates are superseded\n let responseText: string;\n try {\n const response = await llmProvider.summarize(prompt, {\n maxTokens: SUPERSESSION_MAX_TOKENS,\n reasoning: LLM_REASONING_MODE,\n });\n responseText = stripReasoningTokens(response.text);\n } catch (err) {\n log?.('warn', 'checkSupersession: LLM call failed', { newSporeId, error: String(err) });\n return [];\n }\n\n // Parse the LLM response as a JSON array of IDs\n let rawIds: unknown;\n try {\n rawIds = JSON.parse(responseText);\n } catch {\n log?.('warn', 'checkSupersession: failed to parse LLM response', { newSporeId, responseText });\n return [];\n }\n\n const parsed = supersededIdsSchema.safeParse(rawIds);\n if (!parsed.success) {\n log?.('warn', 'checkSupersession: LLM response failed schema validation', { newSporeId });\n return [];\n }\n\n // Filter to IDs that actually exist in the candidate list and are still active\n const candidateMap = new Map<string, IndexedNote>(filtered.map((c) => [c.id, c]));\n const validIds = parsed.data.filter((id) => candidateMap.has(id));\n\n if (validIds.length === 0) {\n return [];\n }\n\n // Mark each validated candidate as superseded\n const supersededIds: string[] = [];\n\n for (const id of validIds) {\n const candidate = candidateMap.get(id)!;\n const wrote = supersedeSpore(id, newSporeId, candidate.path, { index, vectorIndex, vaultDir });\n\n if (!wrote) {\n log?.('warn', 'checkSupersession: file not found for candidate, skipping', { id, path: candidate.path });\n continue;\n }\n\n supersededIds.push(id);\n log?.('info', 'checkSupersession: marked superseded', { supersededId: id, newSporeId });\n }\n\n return supersededIds;\n}\n"],"mappings":"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","names":["YAML"]}
@@ -2,10 +2,10 @@ import { createRequire as __cr } from 'node:module'; const require = __cr(import
2
2
  import {
3
3
  LmStudioBackend,
4
4
  OllamaBackend
5
- } from "./chunk-67R6EMYD.js";
5
+ } from "./chunk-JI6M2L2W.js";
6
6
  import {
7
7
  AgentRegistry
8
- } from "./chunk-BNIYWCST.js";
8
+ } from "./chunk-X6TKHO22.js";
9
9
 
10
10
  // src/cli/shared.ts
11
11
  import fs from "fs";
@@ -116,4 +116,4 @@ export {
116
116
  VAULT_GITIGNORE,
117
117
  configureVaultEnv
118
118
  };
119
- //# sourceMappingURL=chunk-GFBG73P4.js.map
119
+ //# sourceMappingURL=chunk-FIRMTYFH.js.map