@goondocks/myco 0.16.2 → 0.17.1

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 (172) hide show
  1. package/CONTRIBUTING.md +3 -3
  2. package/README.md +4 -3
  3. package/dist/{agent-run-MNU2QWHR.js → agent-run-BGW4TY3D.js} +7 -7
  4. package/dist/{agent-tasks-NCRKUU6E.js → agent-tasks-XUJ6FTPL.js} +7 -7
  5. package/dist/{chunk-P6C6ADBU.js → chunk-2IJ6C63F.js} +2 -2
  6. package/dist/{chunk-V2ZBYKDU.js → chunk-2WRXLYG6.js} +3 -3
  7. package/dist/{chunk-34NHDRWI.js → chunk-6SDC6V3N.js} +2 -2
  8. package/dist/{chunk-34NHDRWI.js.map → chunk-6SDC6V3N.js.map} +1 -1
  9. package/dist/{chunk-6JZEAOLG.js → chunk-CPL76CYD.js} +3 -3
  10. package/dist/{chunk-VWXDSDJU.js → chunk-DKSQMH5X.js} +2 -2
  11. package/dist/{chunk-GSKXOCFG.js → chunk-EBIYONNZ.js} +21 -1
  12. package/dist/chunk-EBIYONNZ.js.map +1 -0
  13. package/dist/{chunk-XAXQ72L3.js → chunk-FEX6ALLH.js} +2 -2
  14. package/dist/{chunk-UILSK6DK.js → chunk-GBYLHPML.js} +2 -2
  15. package/dist/{chunk-DZWSHCAC.js → chunk-KGL5QSDN.js} +2 -2
  16. package/dist/{chunk-4JVHWBZF.js → chunk-KKEMVH6D.js} +2 -2
  17. package/dist/{chunk-ZMODJWI5.js → chunk-KOTFMGL5.js} +4 -4
  18. package/dist/{chunk-4U6X35TH.js → chunk-KQOII5RU.js} +3 -3
  19. package/dist/{chunk-2E7YGLLN.js → chunk-LA7NDX3J.js} +2 -2
  20. package/dist/{chunk-RJRRHTAA.js → chunk-N6JNOSBQ.js} +3 -3
  21. package/dist/{chunk-CJ2KTRWI.js → chunk-OGNEW5CN.js} +2 -2
  22. package/dist/{chunk-C3GNF7RJ.js → chunk-QQ7CXA7Q.js} +5 -5
  23. package/dist/{chunk-I3S6L7QC.js → chunk-U3SSOSIR.js} +2 -2
  24. package/dist/{chunk-IRSNOBGD.js → chunk-UDBCAFXS.js} +2 -2
  25. package/dist/{chunk-SGYYOTNM.js → chunk-VGVRBSLC.js} +2 -2
  26. package/dist/{chunk-W7ZOOZMK.js → chunk-VQQ57UPG.js} +3 -3
  27. package/dist/{chunk-TIAYBVSI.js → chunk-VVGZL2HX.js} +171 -58
  28. package/dist/chunk-VVGZL2HX.js.map +1 -0
  29. package/dist/{chunk-BNAYBGPH.js → chunk-XQHL4GMO.js} +6 -5
  30. package/dist/{chunk-BNAYBGPH.js.map → chunk-XQHL4GMO.js.map} +1 -1
  31. package/dist/{chunk-RPILIIYT.js → chunk-Y7QCKCEJ.js} +2 -2
  32. package/dist/{chunk-D63XTGBV.js → chunk-YTOD6L6N.js} +6 -6
  33. package/dist/{chunk-D2NTFSVO.js → chunk-ZUSTCXHT.js} +3 -3
  34. package/dist/{cli-RYYABF2X.js → cli-353VNZIY.js} +40 -40
  35. package/dist/{client-5VXKGNN2.js → client-7KJ453V4.js} +4 -4
  36. package/dist/{config-VHHCGE4F.js → config-K3CJEFFO.js} +3 -3
  37. package/dist/{detect-6FNYONJF.js → detect-NJ2OREDP.js} +2 -2
  38. package/dist/{detect-providers-R7QOB3H6.js → detect-providers-OE6HWW3M.js} +4 -4
  39. package/dist/{doctor-M4Q7VCDO.js → doctor-RYFP7ABA.js} +14 -13
  40. package/dist/doctor-RYFP7ABA.js.map +1 -0
  41. package/dist/{executor-ULRFWJCH.js → executor-YOKYS7OT.js} +17 -17
  42. package/dist/{init-AEHAQFPK.js → init-XR2JZWY2.js} +17 -17
  43. package/dist/{init-wizard-SVKDS3LR.js → init-wizard-5CH2FD76.js} +7 -7
  44. package/dist/{installer-AARSFXI6.js → installer-45ZLP2RP.js} +2 -2
  45. package/dist/{llm-LS7U7BHC.js → llm-PGETQHZ2.js} +7 -7
  46. package/dist/{loader-QDWQTBX4.js → loader-AVWL7PNO.js} +3 -3
  47. package/dist/{loader-YQDG5GI5.js → loader-J56KP27U.js} +3 -3
  48. package/dist/{main-GAGOE6XB.js → main-6DGPZXRF.js} +170 -60
  49. package/dist/main-6DGPZXRF.js.map +1 -0
  50. package/dist/{open-4QMAL32X.js → open-P7YEH7UJ.js} +7 -7
  51. package/dist/{openai-embeddings-FUW6CSN2.js → openai-embeddings-LZKY6RV5.js} +4 -4
  52. package/dist/{openrouter-YSIUSUQL.js → openrouter-UTOZG6Z5.js} +4 -4
  53. package/dist/{post-compact-OAWEBEDK.js → post-compact-WPS4SONO.js} +7 -7
  54. package/dist/{post-tool-use-B3KOEOIM.js → post-tool-use-5WLLRGZ5.js} +6 -6
  55. package/dist/{post-tool-use-failure-2I5ELTTN.js → post-tool-use-failure-6C6HSBHI.js} +7 -7
  56. package/dist/{pre-compact-NOXNJ5EV.js → pre-compact-Z4E4JLAK.js} +7 -7
  57. package/dist/{provider-check-VEYONGNU.js → provider-check-CESRPIY5.js} +4 -4
  58. package/dist/{registry-5R3DLJQH.js → registry-SPKP2WLI.js} +4 -4
  59. package/dist/{remove-LX4G6KP7.js → remove-B2PFVQXK.js} +9 -9
  60. package/dist/{resolution-events-CHOKR35X.js → resolution-events-CLDXZF67.js} +4 -4
  61. package/dist/{restart-WSNBSALP.js → restart-XAJDOL3E.js} +8 -8
  62. package/dist/{search-Q6N3SHKP.js → search-ERTCTAQ3.js} +8 -8
  63. package/dist/{server-OFRKA6N7.js → server-LXUA7XUQ.js} +5 -5
  64. package/dist/{server-OFRKA6N7.js.map → server-LXUA7XUQ.js.map} +1 -1
  65. package/dist/{session-SKXJLJYH.js → session-433T6V3C.js} +9 -9
  66. package/dist/{session-end-5EIVRCPS.js → session-end-4Y5VY4OI.js} +6 -6
  67. package/dist/{session-start-OL2ICLED.js → session-start-3STH4HFL.js} +11 -11
  68. package/dist/{setup-llm-BRNQW7K2.js → setup-llm-UBBSQWX5.js} +8 -8
  69. package/dist/src/cli.js +1 -1
  70. package/dist/src/daemon/main.js +1 -1
  71. package/dist/src/hooks/post-tool-use.js +1 -1
  72. package/dist/src/hooks/session-end.js +1 -1
  73. package/dist/src/hooks/session-start.js +1 -1
  74. package/dist/src/hooks/stop.js +1 -1
  75. package/dist/src/hooks/user-prompt-submit.js +1 -1
  76. package/dist/src/mcp/server.js +1 -1
  77. package/dist/src/symbionts/manifests/claude-code.yaml +4 -0
  78. package/dist/src/symbionts/manifests/opencode.yaml +26 -0
  79. package/dist/src/symbionts/templates/claude-code/hooks.json +12 -12
  80. package/dist/src/symbionts/templates/codex/hooks.json +4 -4
  81. package/dist/src/symbionts/templates/cursor/hooks.json +9 -9
  82. package/dist/src/symbionts/templates/gemini/hooks.json +6 -6
  83. package/dist/src/symbionts/templates/opencode/mcp.json +6 -0
  84. package/dist/src/symbionts/templates/opencode/package.json +5 -0
  85. package/dist/src/symbionts/templates/opencode/plugin.ts +733 -0
  86. package/dist/src/symbionts/templates/opencode/settings.json +8 -0
  87. package/dist/src/symbionts/templates/vscode-copilot/hooks.json +7 -7
  88. package/dist/src/symbionts/templates/windsurf/hooks.json +4 -4
  89. package/dist/{stats-U5FHDIR7.js → stats-3NW7PGQK.js} +9 -9
  90. package/dist/{stop-YUZNQBRQ.js → stop-L7BLMHUD.js} +6 -6
  91. package/dist/{stop-failure-6WFAKH2U.js → stop-failure-P5MYHGAZ.js} +7 -7
  92. package/dist/{subagent-start-GWJXAAH3.js → subagent-start-AIEFG4HA.js} +7 -7
  93. package/dist/{subagent-stop-B44SMV2R.js → subagent-stop-TZ62BSNI.js} +7 -7
  94. package/dist/{task-completed-GIUFSRTP.js → task-completed-ZKVCUBCP.js} +7 -7
  95. package/dist/{team-3YI3UWB3.js → team-WHZW6IFU.js} +5 -5
  96. package/dist/ui/assets/{index-RYHXSJv1.js → index-2UyTdjlV.js} +12 -12
  97. package/dist/ui/assets/index-Cts1wLEW.css +1 -0
  98. package/dist/ui/index.html +2 -2
  99. package/dist/{update-QPRTLGYU.js → update-P7GIQLIV.js} +9 -9
  100. package/dist/{user-prompt-submit-FSYEPW7W.js → user-prompt-submit-4J7ZW6X3.js} +6 -6
  101. package/dist/{verify-ITBMLK67.js → verify-PSERIZPF.js} +8 -8
  102. package/dist/{version-VS2EDHBG.js → version-OHJ5ZLHX.js} +2 -2
  103. package/package.json +1 -1
  104. package/skills/rules/SKILL.md +1 -1
  105. package/dist/chunk-GSKXOCFG.js.map +0 -1
  106. package/dist/chunk-TIAYBVSI.js.map +0 -1
  107. package/dist/doctor-M4Q7VCDO.js.map +0 -1
  108. package/dist/main-GAGOE6XB.js.map +0 -1
  109. package/dist/ui/assets/index-Bjv_ck3c.css +0 -1
  110. /package/dist/{agent-run-MNU2QWHR.js.map → agent-run-BGW4TY3D.js.map} +0 -0
  111. /package/dist/{agent-tasks-NCRKUU6E.js.map → agent-tasks-XUJ6FTPL.js.map} +0 -0
  112. /package/dist/{chunk-P6C6ADBU.js.map → chunk-2IJ6C63F.js.map} +0 -0
  113. /package/dist/{chunk-V2ZBYKDU.js.map → chunk-2WRXLYG6.js.map} +0 -0
  114. /package/dist/{chunk-6JZEAOLG.js.map → chunk-CPL76CYD.js.map} +0 -0
  115. /package/dist/{chunk-VWXDSDJU.js.map → chunk-DKSQMH5X.js.map} +0 -0
  116. /package/dist/{chunk-XAXQ72L3.js.map → chunk-FEX6ALLH.js.map} +0 -0
  117. /package/dist/{chunk-UILSK6DK.js.map → chunk-GBYLHPML.js.map} +0 -0
  118. /package/dist/{chunk-DZWSHCAC.js.map → chunk-KGL5QSDN.js.map} +0 -0
  119. /package/dist/{chunk-4JVHWBZF.js.map → chunk-KKEMVH6D.js.map} +0 -0
  120. /package/dist/{chunk-ZMODJWI5.js.map → chunk-KOTFMGL5.js.map} +0 -0
  121. /package/dist/{chunk-4U6X35TH.js.map → chunk-KQOII5RU.js.map} +0 -0
  122. /package/dist/{chunk-2E7YGLLN.js.map → chunk-LA7NDX3J.js.map} +0 -0
  123. /package/dist/{chunk-RJRRHTAA.js.map → chunk-N6JNOSBQ.js.map} +0 -0
  124. /package/dist/{chunk-CJ2KTRWI.js.map → chunk-OGNEW5CN.js.map} +0 -0
  125. /package/dist/{chunk-C3GNF7RJ.js.map → chunk-QQ7CXA7Q.js.map} +0 -0
  126. /package/dist/{chunk-I3S6L7QC.js.map → chunk-U3SSOSIR.js.map} +0 -0
  127. /package/dist/{chunk-IRSNOBGD.js.map → chunk-UDBCAFXS.js.map} +0 -0
  128. /package/dist/{chunk-SGYYOTNM.js.map → chunk-VGVRBSLC.js.map} +0 -0
  129. /package/dist/{chunk-W7ZOOZMK.js.map → chunk-VQQ57UPG.js.map} +0 -0
  130. /package/dist/{chunk-RPILIIYT.js.map → chunk-Y7QCKCEJ.js.map} +0 -0
  131. /package/dist/{chunk-D63XTGBV.js.map → chunk-YTOD6L6N.js.map} +0 -0
  132. /package/dist/{chunk-D2NTFSVO.js.map → chunk-ZUSTCXHT.js.map} +0 -0
  133. /package/dist/{cli-RYYABF2X.js.map → cli-353VNZIY.js.map} +0 -0
  134. /package/dist/{client-5VXKGNN2.js.map → client-7KJ453V4.js.map} +0 -0
  135. /package/dist/{config-VHHCGE4F.js.map → config-K3CJEFFO.js.map} +0 -0
  136. /package/dist/{detect-6FNYONJF.js.map → detect-NJ2OREDP.js.map} +0 -0
  137. /package/dist/{detect-providers-R7QOB3H6.js.map → detect-providers-OE6HWW3M.js.map} +0 -0
  138. /package/dist/{executor-ULRFWJCH.js.map → executor-YOKYS7OT.js.map} +0 -0
  139. /package/dist/{init-AEHAQFPK.js.map → init-XR2JZWY2.js.map} +0 -0
  140. /package/dist/{init-wizard-SVKDS3LR.js.map → init-wizard-5CH2FD76.js.map} +0 -0
  141. /package/dist/{installer-AARSFXI6.js.map → installer-45ZLP2RP.js.map} +0 -0
  142. /package/dist/{llm-LS7U7BHC.js.map → llm-PGETQHZ2.js.map} +0 -0
  143. /package/dist/{loader-QDWQTBX4.js.map → loader-AVWL7PNO.js.map} +0 -0
  144. /package/dist/{loader-YQDG5GI5.js.map → loader-J56KP27U.js.map} +0 -0
  145. /package/dist/{open-4QMAL32X.js.map → open-P7YEH7UJ.js.map} +0 -0
  146. /package/dist/{openai-embeddings-FUW6CSN2.js.map → openai-embeddings-LZKY6RV5.js.map} +0 -0
  147. /package/dist/{openrouter-YSIUSUQL.js.map → openrouter-UTOZG6Z5.js.map} +0 -0
  148. /package/dist/{post-compact-OAWEBEDK.js.map → post-compact-WPS4SONO.js.map} +0 -0
  149. /package/dist/{post-tool-use-B3KOEOIM.js.map → post-tool-use-5WLLRGZ5.js.map} +0 -0
  150. /package/dist/{post-tool-use-failure-2I5ELTTN.js.map → post-tool-use-failure-6C6HSBHI.js.map} +0 -0
  151. /package/dist/{pre-compact-NOXNJ5EV.js.map → pre-compact-Z4E4JLAK.js.map} +0 -0
  152. /package/dist/{provider-check-VEYONGNU.js.map → provider-check-CESRPIY5.js.map} +0 -0
  153. /package/dist/{registry-5R3DLJQH.js.map → registry-SPKP2WLI.js.map} +0 -0
  154. /package/dist/{remove-LX4G6KP7.js.map → remove-B2PFVQXK.js.map} +0 -0
  155. /package/dist/{resolution-events-CHOKR35X.js.map → resolution-events-CLDXZF67.js.map} +0 -0
  156. /package/dist/{restart-WSNBSALP.js.map → restart-XAJDOL3E.js.map} +0 -0
  157. /package/dist/{search-Q6N3SHKP.js.map → search-ERTCTAQ3.js.map} +0 -0
  158. /package/dist/{session-SKXJLJYH.js.map → session-433T6V3C.js.map} +0 -0
  159. /package/dist/{session-end-5EIVRCPS.js.map → session-end-4Y5VY4OI.js.map} +0 -0
  160. /package/dist/{session-start-OL2ICLED.js.map → session-start-3STH4HFL.js.map} +0 -0
  161. /package/dist/{setup-llm-BRNQW7K2.js.map → setup-llm-UBBSQWX5.js.map} +0 -0
  162. /package/dist/{stats-U5FHDIR7.js.map → stats-3NW7PGQK.js.map} +0 -0
  163. /package/dist/{stop-YUZNQBRQ.js.map → stop-L7BLMHUD.js.map} +0 -0
  164. /package/dist/{stop-failure-6WFAKH2U.js.map → stop-failure-P5MYHGAZ.js.map} +0 -0
  165. /package/dist/{subagent-start-GWJXAAH3.js.map → subagent-start-AIEFG4HA.js.map} +0 -0
  166. /package/dist/{subagent-stop-B44SMV2R.js.map → subagent-stop-TZ62BSNI.js.map} +0 -0
  167. /package/dist/{task-completed-GIUFSRTP.js.map → task-completed-ZKVCUBCP.js.map} +0 -0
  168. /package/dist/{team-3YI3UWB3.js.map → team-WHZW6IFU.js.map} +0 -0
  169. /package/dist/{update-QPRTLGYU.js.map → update-P7GIQLIV.js.map} +0 -0
  170. /package/dist/{user-prompt-submit-FSYEPW7W.js.map → user-prompt-submit-4J7ZW6X3.js.map} +0 -0
  171. /package/dist/{verify-ITBMLK67.js.map → verify-PSERIZPF.js.map} +0 -0
  172. /package/dist/{version-VS2EDHBG.js.map → version-OHJ5ZLHX.js.map} +0 -0
@@ -209,9 +209,8 @@ function writeOrDeleteJsonFile(filePath, data) {
209
209
  import fs2 from "fs";
210
210
  import path2 from "path";
211
211
  var MYCO_HOOK_COMMAND_PREFIX = "myco-run";
212
- var MYCO_HOOK_GUARD_PREFIX = "node .agents/myco-hook.cjs";
213
212
  function isMycoHookCommand(command) {
214
- return command.startsWith(MYCO_HOOK_COMMAND_PREFIX) || command.startsWith(MYCO_HOOK_GUARD_PREFIX);
213
+ return command.startsWith(MYCO_HOOK_COMMAND_PREFIX) || command.includes(".agents/myco-hook.cjs");
215
214
  }
216
215
  function isMycoHookGroup(group) {
217
216
  if (Array.isArray(group.hooks) && group.hooks.some((h) => h.command && isMycoHookCommand(h.command))) return true;
@@ -257,6 +256,8 @@ var HOOK_GUARD_PROJECT_PATH = `.agents/${HOOK_GUARD_INSTALLED_FILENAME}`;
257
256
  var SKILLS_SUBDIR = "skills";
258
257
  var CANONICAL_SKILLS_DIR = ".agents/skills";
259
258
  var MYCO_MCP_SERVER_NAME = "myco";
259
+ var MYCO_PLUGIN_FILE_MARKER = "myco:plugin-marker";
260
+ var HOOKS_FORMAT_PLUGIN_FILE = "plugin-file";
260
261
  var INSTRUCTIONS_STUB_MARKER = "Edit AGENTS.md, not this file";
261
262
  var INSTRUCTIONS_REF_START = "<!-- myco:agents-ref:start -->";
262
263
  var INSTRUCTIONS_REF_END = "<!-- myco:agents-ref:end -->";
@@ -271,6 +272,39 @@ var SymbiontInstaller = class {
271
272
  this.projectRoot = projectRoot;
272
273
  this.packageRoot = packageRoot;
273
274
  }
275
+ /**
276
+ * Read a template file as raw text, checking both source and dist layouts.
277
+ * `relPath` is relative to `TEMPLATES_SUBDIR` — e.g. `'hook-guard.cjs'` for
278
+ * a shared template or `'opencode/plugin.ts'` for a per-agent template.
279
+ */
280
+ readTemplateFile(relPath) {
281
+ const candidates = [
282
+ path3.join(this.packageRoot, TEMPLATES_SUBDIR, relPath),
283
+ // tsup preserves the src/ prefix under dist/, so the same subdir works in both layouts
284
+ path3.join(this.packageRoot, "dist", TEMPLATES_SUBDIR, relPath)
285
+ ];
286
+ for (const filePath of candidates) {
287
+ try {
288
+ return fs3.readFileSync(filePath, "utf-8");
289
+ } catch {
290
+ }
291
+ }
292
+ return null;
293
+ }
294
+ /**
295
+ * Write a Myco-managed file with a content-diff gate. Creates parent dirs as
296
+ * needed. Returns `true` if the file was written (new or updated), `false` if
297
+ * the on-disk content already matches and the write was skipped.
298
+ */
299
+ writeManagedFile(absPath, content) {
300
+ try {
301
+ if (fs3.readFileSync(absPath, "utf-8") === content) return false;
302
+ } catch {
303
+ }
304
+ fs3.mkdirSync(path3.dirname(absPath), { recursive: true });
305
+ fs3.writeFileSync(absPath, content, "utf-8");
306
+ return true;
307
+ }
274
308
  /**
275
309
  * Copy the hook-guard script into .agents/myco-hook.cjs.
276
310
  * Returns true if the file was written (or updated); false if skipped or N/A.
@@ -278,16 +312,12 @@ var SymbiontInstaller = class {
278
312
  installHookGuard() {
279
313
  const reg = this.manifest.registration;
280
314
  if (!reg?.hooksTarget) return false;
281
- const guardTemplate = this.loadHookGuardTemplate();
315
+ const guardTemplate = this.readTemplateFile(HOOK_GUARD_TEMPLATE_FILENAME);
282
316
  if (!guardTemplate) return false;
283
- const targetPath = path3.join(this.projectRoot, HOOK_GUARD_PROJECT_PATH);
284
- try {
285
- if (fs3.readFileSync(targetPath, "utf-8") === guardTemplate) return false;
286
- } catch {
287
- }
288
- fs3.mkdirSync(path3.dirname(targetPath), { recursive: true });
289
- fs3.writeFileSync(targetPath, guardTemplate, "utf-8");
290
- return true;
317
+ return this.writeManagedFile(
318
+ path3.join(this.projectRoot, HOOK_GUARD_PROJECT_PATH),
319
+ guardTemplate
320
+ );
291
321
  }
292
322
  /**
293
323
  * Remove the hook-guard script from .agents/myco-hook.cjs.
@@ -304,34 +334,23 @@ var SymbiontInstaller = class {
304
334
  return false;
305
335
  }
306
336
  }
307
- /** Load the hook-guard template from package root. */
308
- loadHookGuardTemplate() {
309
- const candidates = [
310
- path3.join(this.packageRoot, TEMPLATES_SUBDIR, HOOK_GUARD_TEMPLATE_FILENAME),
311
- path3.join(this.packageRoot, "dist", TEMPLATES_SUBDIR, HOOK_GUARD_TEMPLATE_FILENAME)
312
- ];
313
- for (const p of candidates) {
314
- try {
315
- return fs3.readFileSync(p, "utf-8");
316
- } catch {
317
- }
318
- }
319
- return null;
320
- }
321
337
  /** Load a JSON template file for this symbiont. Returns null if not found. */
322
338
  loadTemplate(name) {
323
- const candidates = [
324
- path3.join(this.packageRoot, TEMPLATES_SUBDIR, this.manifest.name, `${name}.json`),
325
- // tsup preserves the src/ prefix under dist/, so the same subdir works in both layouts
326
- path3.join(this.packageRoot, "dist", TEMPLATES_SUBDIR, this.manifest.name, `${name}.json`)
327
- ];
328
- for (const filePath of candidates) {
329
- try {
330
- return JSON.parse(fs3.readFileSync(filePath, "utf-8"));
331
- } catch {
332
- }
339
+ const raw = this.readTemplateFile(path3.join(this.manifest.name, `${name}.json`));
340
+ if (raw === null) return null;
341
+ try {
342
+ return JSON.parse(raw);
343
+ } catch {
344
+ return null;
333
345
  }
334
- return null;
346
+ }
347
+ /**
348
+ * Load a template file verbatim (no JSON parsing).
349
+ * Used for plugin-file hook templates (e.g., opencode's plugin.ts) and any
350
+ * other template that is copied to the project without structural merging.
351
+ */
352
+ loadTemplateRaw(filename) {
353
+ return this.readTemplateFile(path3.join(this.manifest.name, filename));
335
354
  }
336
355
  /** Run all registration steps. */
337
356
  install() {
@@ -342,7 +361,8 @@ var SymbiontInstaller = class {
342
361
  mcp: this.installMcp(),
343
362
  skills: this.installSkills(),
344
363
  settings: this.installSettings(),
345
- instructions: this.installInstructions()
364
+ instructions: this.installInstructions(),
365
+ pluginPackage: this.installPluginPackage()
346
366
  };
347
367
  this.updateGitignore();
348
368
  return result;
@@ -351,6 +371,9 @@ var SymbiontInstaller = class {
351
371
  * Check if ALL non-null JSON targets share the same file (e.g., Gemini).
352
372
  * Only batches when every target resolves to one path — partial overlaps
353
373
  * (e.g., Claude Code: hooks+settings share but MCP is separate) use normal path.
374
+ *
375
+ * Plugin-file hooks (e.g., opencode) naturally fall out of batching because their
376
+ * hooksTarget is a distinct .ts file path, yielding a Set size ≥ 2.
354
377
  */
355
378
  shouldBatchJsonTargets(reg) {
356
379
  if (!reg) return false;
@@ -383,11 +406,12 @@ var SymbiontInstaller = class {
383
406
  }
384
407
  const mcpTemplate = reg.mcpTarget ? this.loadTemplate("mcp") : null;
385
408
  if (mcpTemplate) {
386
- const servers = data.mcpServers ?? {};
409
+ const serversKey = reg.mcpServersKey ?? "mcpServers";
410
+ const servers = data[serversKey] ?? {};
387
411
  for (const [name, def] of Object.entries(mcpTemplate)) {
388
412
  servers[name] = def;
389
413
  }
390
- data.mcpServers = servers;
414
+ data[serversKey] = servers;
391
415
  mcp = true;
392
416
  }
393
417
  const settingsTemplate = reg.settingsTarget ? this.loadTemplate("settings") : null;
@@ -401,7 +425,8 @@ var SymbiontInstaller = class {
401
425
  mcp,
402
426
  skills: this.installSkills(),
403
427
  settings,
404
- instructions: this.installInstructions()
428
+ instructions: this.installInstructions(),
429
+ pluginPackage: this.installPluginPackage()
405
430
  };
406
431
  }
407
432
  /** Remove all Myco registration from this symbiont's project files. */
@@ -412,7 +437,8 @@ var SymbiontInstaller = class {
412
437
  mcp: this.uninstallMcp(),
413
438
  skills: this.uninstallSkills(),
414
439
  settings: this.uninstallSettings(),
415
- instructions: this.uninstallInstructions()
440
+ instructions: this.uninstallInstructions(),
441
+ pluginPackage: false
416
442
  };
417
443
  this.uninstallHookGuard();
418
444
  this.cleanGitignore();
@@ -425,7 +451,14 @@ var SymbiontInstaller = class {
425
451
  const targetPath = path3.join(this.projectRoot, reg.hooksTarget ?? reg.mcpTarget ?? reg.settingsTarget);
426
452
  const data = readJsonFile(targetPath);
427
453
  if (Object.keys(data).length === 0) {
428
- return { hooks: false, mcp: false, skills: this.uninstallSkills(), settings: false, instructions: this.uninstallInstructions() };
454
+ return {
455
+ hooks: false,
456
+ mcp: false,
457
+ skills: this.uninstallSkills(),
458
+ settings: false,
459
+ instructions: this.uninstallInstructions(),
460
+ pluginPackage: false
461
+ };
429
462
  }
430
463
  let hooks = false, mcp = false, settings = false;
431
464
  if (reg.hooksTarget) {
@@ -445,11 +478,12 @@ var SymbiontInstaller = class {
445
478
  }
446
479
  }
447
480
  if (reg.mcpTarget) {
448
- const servers = data.mcpServers ?? {};
481
+ const serversKey = reg.mcpServersKey ?? "mcpServers";
482
+ const servers = data[serversKey] ?? {};
449
483
  if (servers[MYCO_MCP_SERVER_NAME]) {
450
484
  delete servers[MYCO_MCP_SERVER_NAME];
451
- if (Object.keys(servers).length === 0) delete data.mcpServers;
452
- else data.mcpServers = servers;
485
+ if (Object.keys(servers).length === 0) delete data[serversKey];
486
+ else data[serversKey] = servers;
453
487
  mcp = true;
454
488
  }
455
489
  }
@@ -458,7 +492,14 @@ var SymbiontInstaller = class {
458
492
  settings = deepRemoveSettings(data, settingsTemplate);
459
493
  }
460
494
  writeOrDeleteJsonFile(targetPath, data);
461
- return { hooks, mcp, skills: this.uninstallSkills(), settings, instructions: this.uninstallInstructions() };
495
+ return {
496
+ hooks,
497
+ mcp,
498
+ skills: this.uninstallSkills(),
499
+ settings,
500
+ instructions: this.uninstallInstructions(),
501
+ pluginPackage: false
502
+ };
462
503
  }
463
504
  /**
464
505
  * Ensure the instruction file references AGENTS.md.
@@ -599,10 +640,14 @@ ${desired.join("\n")}
599
640
  /**
600
641
  * Merge hooks template into the target settings file.
601
642
  * Replaces all Myco-owned hook groups; preserves non-Myco hooks.
643
+ *
644
+ * For plugin-file agents (e.g., opencode) this dispatches to `installPluginHookFile()`
645
+ * which writes a verbatim .ts plugin source to hooksTarget instead of merging JSON.
602
646
  */
603
647
  installHooks() {
604
648
  const reg = this.manifest.registration;
605
649
  if (!reg?.hooksTarget) return false;
650
+ if (reg.hooksFormat === HOOKS_FORMAT_PLUGIN_FILE) return this.installPluginHookFile();
606
651
  const template = this.loadTemplate("hooks");
607
652
  if (!template) return false;
608
653
  const targetPath = path3.join(this.projectRoot, reg.hooksTarget);
@@ -624,6 +669,62 @@ ${desired.join("\n")}
624
669
  writeJsonFile(targetPath, settings);
625
670
  return true;
626
671
  }
672
+ /**
673
+ * Install a plugin-file hook target by copying a verbatim template.
674
+ * Used for agents whose hook system is plugin-based rather than JSON entry-based
675
+ * (e.g., opencode's TypeScript plugin system).
676
+ */
677
+ installPluginHookFile() {
678
+ const reg = this.manifest.registration;
679
+ if (!reg?.hooksTarget) return false;
680
+ const templateContent = this.loadTemplateRaw("plugin.ts");
681
+ if (templateContent === null) return false;
682
+ return this.writeManagedFile(
683
+ path3.join(this.projectRoot, reg.hooksTarget),
684
+ templateContent
685
+ );
686
+ }
687
+ /**
688
+ * Remove a plugin-file hook target.
689
+ * Only deletes files whose content contains the Myco plugin marker — contributors
690
+ * who hand-edit the plugin file without removing the marker are protected.
691
+ */
692
+ uninstallPluginHookFile() {
693
+ const reg = this.manifest.registration;
694
+ if (!reg?.hooksTarget) return false;
695
+ const targetPath = path3.join(this.projectRoot, reg.hooksTarget);
696
+ let content;
697
+ try {
698
+ content = fs3.readFileSync(targetPath, "utf-8");
699
+ } catch {
700
+ return false;
701
+ }
702
+ if (!content.includes(MYCO_PLUGIN_FILE_MARKER)) return false;
703
+ try {
704
+ fs3.unlinkSync(targetPath);
705
+ try {
706
+ fs3.rmdirSync(path3.dirname(targetPath));
707
+ } catch {
708
+ }
709
+ return true;
710
+ } catch {
711
+ return false;
712
+ }
713
+ }
714
+ /**
715
+ * Install a plugin deps package.json for plugin-file agents (e.g., opencode).
716
+ * Writes the template verbatim so the agent's package manager can install the SDK.
717
+ */
718
+ installPluginPackage() {
719
+ const reg = this.manifest.registration;
720
+ if (!reg?.pluginPackageTarget) return false;
721
+ const templateContent = this.loadTemplateRaw("package.json");
722
+ if (templateContent === null) return false;
723
+ return this.writeManagedFile(
724
+ path3.join(this.projectRoot, reg.pluginPackageTarget),
725
+ templateContent
726
+ );
727
+ }
627
728
  /**
628
729
  * Merge MCP server template into the target config file.
629
730
  * Replaces the `myco` server entry; preserves other servers.
@@ -634,20 +735,26 @@ ${desired.join("\n")}
634
735
  const template = this.loadTemplate("mcp");
635
736
  if (!template) return false;
636
737
  const targetPath = path3.join(this.projectRoot, reg.mcpTarget);
637
- const mcpFormat = reg.mcpFormat ?? "json";
638
- if (mcpFormat === "toml") {
738
+ if (reg.mcpFormat === "toml") {
639
739
  return this.installMcpToml(targetPath, template);
640
740
  }
641
741
  return this.installMcpJson(targetPath, template);
642
742
  }
643
- /** Write MCP servers to a JSON config file. */
743
+ /**
744
+ * Write MCP servers to a JSON config file under the manifest-configured key.
745
+ * Most agents use the canonical `mcpServers` key; opencode uses `mcp`.
746
+ *
747
+ * The `?? 'mcpServers'` fallback protects against test fixtures that construct
748
+ * manifests as plain object literals and bypass the schema's default.
749
+ */
644
750
  installMcpJson(targetPath, template) {
751
+ const serversKey = this.manifest.registration.mcpServersKey ?? "mcpServers";
645
752
  const config = readJsonFile(targetPath);
646
- const servers = config.mcpServers ?? {};
753
+ const servers = config[serversKey] ?? {};
647
754
  for (const [name, def] of Object.entries(template)) {
648
755
  servers[name] = def;
649
756
  }
650
- config.mcpServers = servers;
757
+ config[serversKey] = servers;
651
758
  writeJsonFile(targetPath, config);
652
759
  return true;
653
760
  }
@@ -795,10 +902,16 @@ ${desired.join("\n")}
795
902
  }
796
903
  return true;
797
904
  }
798
- /** Remove Myco hook groups from the target settings file. */
905
+ /**
906
+ * Remove Myco hook groups from the target settings file.
907
+ *
908
+ * For plugin-file agents (e.g., opencode) this dispatches to `uninstallPluginHookFile()`
909
+ * which deletes the verbatim plugin file (guarded by the Myco plugin marker).
910
+ */
799
911
  uninstallHooks() {
800
912
  const reg = this.manifest.registration;
801
913
  if (!reg?.hooksTarget) return false;
914
+ if (reg.hooksFormat === HOOKS_FORMAT_PLUGIN_FILE) return this.uninstallPluginHookFile();
802
915
  const targetPath = path3.join(this.projectRoot, reg.hooksTarget);
803
916
  const settings = readJsonFile(targetPath);
804
917
  const existingHooks = settings.hooks ?? {};
@@ -825,21 +938,21 @@ ${desired.join("\n")}
825
938
  const reg = this.manifest.registration;
826
939
  if (!reg?.mcpTarget) return false;
827
940
  const targetPath = path3.join(this.projectRoot, reg.mcpTarget);
828
- const mcpFormat = reg.mcpFormat ?? "json";
829
- if (mcpFormat === "toml") {
941
+ if (reg.mcpFormat === "toml") {
830
942
  return this.uninstallMcpToml(targetPath);
831
943
  }
832
944
  return this.uninstallMcpJson(targetPath);
833
945
  }
834
946
  uninstallMcpJson(targetPath) {
947
+ const serversKey = this.manifest.registration.mcpServersKey ?? "mcpServers";
835
948
  const config = readJsonFile(targetPath);
836
- const servers = config.mcpServers ?? {};
949
+ const servers = config[serversKey] ?? {};
837
950
  if (!servers[MYCO_MCP_SERVER_NAME]) return false;
838
951
  delete servers[MYCO_MCP_SERVER_NAME];
839
952
  if (Object.keys(servers).length === 0) {
840
- delete config.mcpServers;
953
+ delete config[serversKey];
841
954
  } else {
842
- config.mcpServers = servers;
955
+ config[serversKey] = servers;
843
956
  }
844
957
  writeOrDeleteJsonFile(targetPath, config);
845
958
  return true;
@@ -989,4 +1102,4 @@ export {
989
1102
  SymbiontInstaller,
990
1103
  syncSkillSymlinks
991
1104
  };
992
- //# sourceMappingURL=chunk-TIAYBVSI.js.map
1105
+ //# sourceMappingURL=chunk-VVGZL2HX.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/symbionts/installer.ts","../src/symbionts/toml-helpers.ts","../src/symbionts/settings-merge.ts","../src/symbionts/json-helpers.ts","../src/symbionts/install-helpers.ts"],"sourcesContent":["import type { SymbiontManifest } from './manifest-schema.js';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport { findTomlSectionEnd, buildTomlMcpSection, upsertTomlSection, removeTomlSectionKeys } from './toml-helpers.js';\nimport { deepMergeSettings, deepRemoveSettings } from './settings-merge.js';\nimport { readJsonFile, writeJsonFile, writeOrDeleteJsonFile } from './json-helpers.js';\nimport { ensureAgentsMd, ensureSymlink, isMycoHookGroup } from './install-helpers.js';\n\n/** Current comment header for Myco-managed .gitignore block. */\nconst GITIGNORE_COMMENT = '# Myco managed (machine-specific)';\n\n/** Legacy comment header — recognized for cleanup during reconciliation. */\nconst GITIGNORE_SKILLS_COMMENT_LEGACY = '# Myco skill symlinks (machine-specific)';\n\n/** Wrangler cache directory created by team sync operations. */\nconst WRANGLER_CACHE_DIR = '.wrangler/';\n\n/** Subdirectory within the package where symbiont templates live. */\nconst TEMPLATES_SUBDIR = 'src/symbionts/templates';\n\n/** Filename of the hook guard template in the templates directory. */\nconst HOOK_GUARD_TEMPLATE_FILENAME = 'hook-guard.cjs';\n\n/** Filename when installed into the project .agents/ directory. */\nconst HOOK_GUARD_INSTALLED_FILENAME = 'myco-hook.cjs';\n\n/** Project-relative path where the hook guard is installed. */\nconst HOOK_GUARD_PROJECT_PATH = `.agents/${HOOK_GUARD_INSTALLED_FILENAME}`;\n\n/** Subdirectory within the package where skills live. */\nconst SKILLS_SUBDIR = 'skills';\n\n/** Canonical cross-agent skills directory. */\nconst CANONICAL_SKILLS_DIR = '.agents/skills';\n\n/** MCP server name used by Myco in all symbiont configurations. */\nexport const MYCO_MCP_SERVER_NAME = 'myco';\n\n/**\n * Marker substring written into plugin-file hook templates (e.g., opencode's plugin.ts).\n * Uninstall only deletes plugin files whose content contains this marker, so\n * contributors who hand-edit a plugin file without removing the marker are protected.\n */\nconst MYCO_PLUGIN_FILE_MARKER = 'myco:plugin-marker';\n\n/** `hooksFormat` value selecting verbatim plugin-file install over JSON merge. */\nconst HOOKS_FORMAT_PLUGIN_FILE = 'plugin-file';\n\n/** Marker text used to identify unmodified instruction stubs. */\nconst INSTRUCTIONS_STUB_MARKER = 'Edit AGENTS.md, not this file';\n\n/** Start/end markers for the reference block prepended to existing instruction files. */\nconst INSTRUCTIONS_REF_START = '<!-- myco:agents-ref:start -->';\nconst INSTRUCTIONS_REF_END = '<!-- myco:agents-ref:end -->';\n\n/** Reference block prepended to existing instruction files. */\nconst INSTRUCTIONS_REF_BLOCK = `${INSTRUCTIONS_REF_START}\n> **Project intelligence:** This project uses [Myco](https://myco.sh). The canonical project rules are in [\\`AGENTS.md\\`](AGENTS.md) — read and follow it alongside this file.\n${INSTRUCTIONS_REF_END}\n\n`;\n\nexport interface InstallResult {\n hooks: boolean;\n mcp: boolean;\n skills: boolean;\n settings: boolean;\n instructions: boolean;\n /**\n * Plugin deps package.json (e.g., .opencode/package.json). Only present for agents\n * with `registration.pluginPackageTarget` set. False otherwise.\n */\n pluginPackage: boolean;\n}\n\nexport class SymbiontInstaller {\n constructor(\n private manifest: SymbiontManifest,\n private projectRoot: string,\n private packageRoot: string,\n ) {}\n\n /**\n * Read a template file as raw text, checking both source and dist layouts.\n * `relPath` is relative to `TEMPLATES_SUBDIR` — e.g. `'hook-guard.cjs'` for\n * a shared template or `'opencode/plugin.ts'` for a per-agent template.\n */\n private readTemplateFile(relPath: string): string | null {\n const candidates = [\n path.join(this.packageRoot, TEMPLATES_SUBDIR, relPath),\n // tsup preserves the src/ prefix under dist/, so the same subdir works in both layouts\n path.join(this.packageRoot, 'dist', TEMPLATES_SUBDIR, relPath),\n ];\n for (const filePath of candidates) {\n try { return fs.readFileSync(filePath, 'utf-8'); } catch { /* try next */ }\n }\n return null;\n }\n\n /**\n * Write a Myco-managed file with a content-diff gate. Creates parent dirs as\n * needed. Returns `true` if the file was written (new or updated), `false` if\n * the on-disk content already matches and the write was skipped.\n */\n private writeManagedFile(absPath: string, content: string): boolean {\n try {\n if (fs.readFileSync(absPath, 'utf-8') === content) return false;\n } catch { /* doesn't exist — proceed */ }\n fs.mkdirSync(path.dirname(absPath), { recursive: true });\n fs.writeFileSync(absPath, content, 'utf-8');\n return true;\n }\n\n /**\n * Copy the hook-guard script into .agents/myco-hook.cjs.\n * Returns true if the file was written (or updated); false if skipped or N/A.\n */\n installHookGuard(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.hooksTarget) return false;\n\n const guardTemplate = this.readTemplateFile(HOOK_GUARD_TEMPLATE_FILENAME);\n if (!guardTemplate) return false;\n\n return this.writeManagedFile(\n path.join(this.projectRoot, HOOK_GUARD_PROJECT_PATH),\n guardTemplate,\n );\n }\n\n /**\n * Remove the hook-guard script from .agents/myco-hook.cjs.\n * Returns true if the file was removed; false otherwise.\n */\n uninstallHookGuard(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.hooksTarget) return false;\n\n const targetPath = path.join(this.projectRoot, HOOK_GUARD_PROJECT_PATH);\n try {\n fs.unlinkSync(targetPath);\n return true;\n } catch {\n return false;\n }\n }\n\n /** Load a JSON template file for this symbiont. Returns null if not found. */\n loadTemplate(name: string): Record<string, unknown> | null {\n const raw = this.readTemplateFile(path.join(this.manifest.name, `${name}.json`));\n if (raw === null) return null;\n try { return JSON.parse(raw); } catch { return null; }\n }\n\n /**\n * Load a template file verbatim (no JSON parsing).\n * Used for plugin-file hook templates (e.g., opencode's plugin.ts) and any\n * other template that is copied to the project without structural merging.\n */\n loadTemplateRaw(filename: string): string | null {\n return this.readTemplateFile(path.join(this.manifest.name, filename));\n }\n\n /** Run all registration steps. */\n install(): InstallResult {\n const reg = this.manifest.registration;\n // Install hook guard before hooks so the guard script is in place when hooks reference it\n this.installHookGuard();\n const result = this.shouldBatchJsonTargets(reg)\n ? this.installBatchedJson(reg!)\n : {\n hooks: this.installHooks(),\n mcp: this.installMcp(),\n skills: this.installSkills(),\n settings: this.installSettings(),\n instructions: this.installInstructions(),\n pluginPackage: this.installPluginPackage(),\n };\n this.updateGitignore();\n return result;\n }\n\n /**\n * Check if ALL non-null JSON targets share the same file (e.g., Gemini).\n * Only batches when every target resolves to one path — partial overlaps\n * (e.g., Claude Code: hooks+settings share but MCP is separate) use normal path.\n *\n * Plugin-file hooks (e.g., opencode) naturally fall out of batching because their\n * hooksTarget is a distinct .ts file path, yielding a Set size ≥ 2.\n */\n private shouldBatchJsonTargets(reg: typeof this.manifest.registration): boolean {\n if (!reg) return false;\n const mcpFormat = reg.mcpFormat ?? 'json';\n if (mcpFormat !== 'json') return false;\n const targets = [reg.hooksTarget, reg.mcpTarget, reg.settingsTarget].filter(Boolean);\n return targets.length > 1 && new Set(targets).size === 1;\n }\n\n /**\n * Batched install for agents where hooks, MCP, and settings share one JSON file.\n * Single read → apply all transforms in memory → single write.\n */\n private installBatchedJson(reg: NonNullable<typeof this.manifest.registration>): InstallResult {\n const targetPath = path.join(this.projectRoot, reg.hooksTarget ?? reg.mcpTarget ?? reg.settingsTarget!);\n let data = readJsonFile(targetPath);\n let hooks = false, mcp = false, settings = false;\n\n // Apply hooks transform\n const hooksTemplate = reg.hooksTarget ? this.loadTemplate('hooks') : null;\n if (hooksTemplate) {\n const existingHooks = (data.hooks ?? {}) as Record<string, unknown[]>;\n const mergedHooks: Record<string, unknown[]> = {};\n for (const [event, groups] of Object.entries(existingHooks)) {\n const nonMyco = (groups as Array<Record<string, unknown>>).filter((g) => !isMycoHookGroup(g));\n if (nonMyco.length > 0) mergedHooks[event] = nonMyco;\n }\n for (const [event, groups] of Object.entries(hooksTemplate)) {\n mergedHooks[event] = [...(mergedHooks[event] ?? []), ...(groups as unknown[])];\n }\n data.hooks = mergedHooks;\n hooks = true;\n }\n\n // Apply MCP transform\n const mcpTemplate = reg.mcpTarget ? this.loadTemplate('mcp') : null;\n if (mcpTemplate) {\n const serversKey = reg.mcpServersKey ?? 'mcpServers';\n const servers = (data[serversKey] ?? {}) as Record<string, unknown>;\n for (const [name, def] of Object.entries(mcpTemplate)) {\n servers[name] = def;\n }\n data[serversKey] = servers;\n mcp = true;\n }\n\n // Apply settings transform\n const settingsTemplate = reg.settingsTarget ? this.loadTemplate('settings') : null;\n if (settingsTemplate) {\n data = deepMergeSettings(data, settingsTemplate);\n settings = true;\n }\n\n writeJsonFile(targetPath, data);\n\n return {\n hooks,\n mcp,\n skills: this.installSkills(),\n settings,\n instructions: this.installInstructions(),\n pluginPackage: this.installPluginPackage(),\n };\n }\n\n /** Remove all Myco registration from this symbiont's project files. */\n uninstall(): InstallResult {\n const reg = this.manifest.registration;\n const result = this.shouldBatchJsonTargets(reg)\n ? this.uninstallBatchedJson(reg!)\n : {\n hooks: this.uninstallHooks(),\n mcp: this.uninstallMcp(),\n skills: this.uninstallSkills(),\n settings: this.uninstallSettings(),\n instructions: this.uninstallInstructions(),\n pluginPackage: false,\n };\n // Remove hook guard after hooks/settings so the file is cleaned up last\n this.uninstallHookGuard();\n this.cleanGitignore();\n return result;\n }\n\n /**\n * Batched uninstall for agents where hooks, MCP, and settings share one JSON file.\n */\n private uninstallBatchedJson(reg: NonNullable<typeof this.manifest.registration>): InstallResult {\n const targetPath = path.join(this.projectRoot, reg.hooksTarget ?? reg.mcpTarget ?? reg.settingsTarget!);\n const data = readJsonFile(targetPath);\n if (Object.keys(data).length === 0) {\n return {\n hooks: false,\n mcp: false,\n skills: this.uninstallSkills(),\n settings: false,\n instructions: this.uninstallInstructions(),\n pluginPackage: false,\n };\n }\n\n let hooks = false, mcp = false, settings = false;\n\n // Remove hooks\n if (reg.hooksTarget) {\n const existingHooks = (data.hooks ?? {}) as Record<string, unknown[]>;\n if (Object.keys(existingHooks).length > 0) {\n const cleaned: Record<string, unknown[]> = {};\n for (const [event, groups] of Object.entries(existingHooks)) {\n const nonMyco = (groups as Array<Record<string, unknown>>).filter((g) => !isMycoHookGroup(g));\n if (nonMyco.length > 0) cleaned[event] = nonMyco;\n }\n if (Object.keys(cleaned).length === 0) {\n delete data.hooks;\n } else {\n data.hooks = cleaned;\n }\n hooks = true;\n }\n }\n\n // Remove MCP\n if (reg.mcpTarget) {\n const serversKey = reg.mcpServersKey ?? 'mcpServers';\n const servers = (data[serversKey] ?? {}) as Record<string, unknown>;\n if (servers[MYCO_MCP_SERVER_NAME]) {\n delete servers[MYCO_MCP_SERVER_NAME];\n if (Object.keys(servers).length === 0) delete data[serversKey];\n else data[serversKey] = servers;\n mcp = true;\n }\n }\n\n // Remove settings\n const settingsTemplate = reg.settingsTarget ? this.loadTemplate('settings') : null;\n if (settingsTemplate) {\n settings = deepRemoveSettings(data, settingsTemplate);\n }\n\n writeOrDeleteJsonFile(targetPath, data);\n\n return {\n hooks,\n mcp,\n skills: this.uninstallSkills(),\n settings,\n instructions: this.uninstallInstructions(),\n pluginPackage: false,\n };\n }\n\n /**\n * Ensure the instruction file references AGENTS.md.\n * - File doesn't exist: write the full stub template.\n * - File exists without reference: prepend a reference block.\n * - File already has reference: skip (idempotent).\n *\n * Also ensures AGENTS.md exists — creates a starter if missing.\n */\n installInstructions(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.instructionsFile) return false;\n\n // Ensure AGENTS.md exists before creating stubs that reference it\n ensureAgentsMd(this.projectRoot, this.packageRoot);\n\n const targetPath = path.join(this.projectRoot, reg.instructionsFile);\n\n // Check if file already exists\n let existing: string | null = null;\n try { existing = fs.readFileSync(targetPath, 'utf-8'); } catch { /* doesn't exist */ }\n\n if (existing !== null) {\n // File exists — check if it already has our reference\n if (existing.includes(INSTRUCTIONS_REF_START) || existing.includes(INSTRUCTIONS_STUB_MARKER)) {\n return false; // Already has reference — idempotent\n }\n // Prepend reference block to existing content\n fs.writeFileSync(targetPath, INSTRUCTIONS_REF_BLOCK + existing, 'utf-8');\n return true;\n }\n\n // File doesn't exist — write the full stub template\n const templateCandidates = [\n path.join(this.packageRoot, 'src/symbionts/templates/instructions-stub.md'),\n path.join(this.packageRoot, 'dist/src/symbionts/templates/instructions-stub.md'),\n ];\n let stub: string | null = null;\n for (const p of templateCandidates) {\n try { stub = fs.readFileSync(p, 'utf-8'); break; } catch { /* try next */ }\n }\n if (!stub) return false;\n\n stub = stub.replace('{agentDisplayName}', this.manifest.displayName);\n fs.mkdirSync(path.dirname(targetPath), { recursive: true });\n fs.writeFileSync(targetPath, stub, 'utf-8');\n return true;\n }\n\n /**\n * Remove Myco's instruction file reference.\n * - If file is the full stub (only Myco content): delete it.\n * - If file has user content + prepended reference: remove just the reference block.\n */\n uninstallInstructions(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.instructionsFile) return false;\n\n const targetPath = path.join(this.projectRoot, reg.instructionsFile);\n let content: string;\n try { content = fs.readFileSync(targetPath, 'utf-8'); } catch { return false; }\n\n // Case 1: Full stub — delete the file entirely\n if (content.includes(INSTRUCTIONS_STUB_MARKER)) {\n fs.unlinkSync(targetPath);\n return true;\n }\n\n // Case 2: Prepended reference block — remove just the block\n if (content.includes(INSTRUCTIONS_REF_START)) {\n const startIdx = content.indexOf(INSTRUCTIONS_REF_START);\n const endIdx = content.indexOf(INSTRUCTIONS_REF_END);\n if (endIdx > startIdx) {\n // Remove from start marker through end marker + trailing whitespace\n const afterEnd = endIdx + INSTRUCTIONS_REF_END.length;\n const cleaned = (content.slice(0, startIdx) + content.slice(afterEnd)).replace(/^\\n+/, '');\n fs.writeFileSync(targetPath, cleaned, 'utf-8');\n return true;\n }\n }\n\n return false;\n }\n\n /** List skill directory names from the package root. Returns empty array if not found. */\n private listSkillDirs(): string[] {\n try {\n return fs.readdirSync(path.join(this.packageRoot, SKILLS_SUBDIR), { withFileTypes: true })\n .filter((d) => d.isDirectory())\n .map((d) => d.name);\n } catch { return []; }\n }\n\n /**\n * Reconcile Myco-owned skill entries in project .gitignore.\n * Computes the desired entry set, strips any existing Myco block\n * (and legacy entries), then writes the current block if changed.\n */\n private updateGitignore(): void {\n const reg = this.manifest.registration;\n if (!reg?.skillsTarget) return;\n\n const skillNames = this.listSkillDirs();\n\n // Desired state: canonical per-skill entries + infrastructure artifacts.\n // Agent-specific targets (e.g. .claude/skills/) use local .gitignore files\n // instead of polluting the project-level .gitignore.\n const desired = [\n ...skillNames.map((name) => `${CANONICAL_SKILLS_DIR}/${name}`),\n WRANGLER_CACHE_DIR,\n ];\n\n const gitignorePath = path.join(this.projectRoot, '.gitignore');\n let content = '';\n try { content = fs.readFileSync(gitignorePath, 'utf-8'); } catch { /* doesn't exist yet */ }\n\n // Strip existing Myco block and any legacy entries\n const stripped = this.stripMycoGitignoreBlock(content, skillNames);\n\n // Build the new block\n const desiredBlock = desired.length > 0\n ? `${GITIGNORE_COMMENT}\\n${desired.join('\\n')}\\n`\n : '';\n\n // Check if anything changed\n if (stripped === content && desiredBlock === '') return;\n const separator = stripped.length > 0 && !stripped.endsWith('\\n') ? '\\n' : '';\n const spacer = stripped.length > 0 && desiredBlock.length > 0 ? '\\n' : '';\n const result = stripped + separator + spacer + desiredBlock;\n if (result === content) return;\n\n fs.writeFileSync(gitignorePath, result, 'utf-8');\n }\n\n /**\n * Remove all Myco-owned gitignore entries: the comment header, per-skill\n * entries for both canonical and agent-specific paths, and legacy blanket\n * directory entries. Returns the cleaned content.\n */\n private stripMycoGitignoreBlock(content: string, skillNames: string[]): string {\n const reg = this.manifest.registration;\n const ownedLines = new Set<string>([\n GITIGNORE_COMMENT,\n GITIGNORE_SKILLS_COMMENT_LEGACY,\n `${CANONICAL_SKILLS_DIR}/`, // legacy blanket entry\n WRANGLER_CACHE_DIR,\n ]);\n for (const name of skillNames) {\n ownedLines.add(`${CANONICAL_SKILLS_DIR}/${name}`);\n if (reg?.skillsTarget && reg.skillsTarget !== CANONICAL_SKILLS_DIR) {\n ownedLines.add(`${reg.skillsTarget}/${name}`);\n }\n }\n\n const lines = content.split('\\n');\n const filtered = lines.filter((line) => !ownedLines.has(line));\n return filtered.join('\\n').replace(/\\n{3,}/g, '\\n\\n').trimEnd() + (filtered.length > 0 ? '\\n' : '');\n }\n\n /**\n * Merge hooks template into the target settings file.\n * Replaces all Myco-owned hook groups; preserves non-Myco hooks.\n *\n * For plugin-file agents (e.g., opencode) this dispatches to `installPluginHookFile()`\n * which writes a verbatim .ts plugin source to hooksTarget instead of merging JSON.\n */\n installHooks(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.hooksTarget) return false;\n\n if (reg.hooksFormat === HOOKS_FORMAT_PLUGIN_FILE) return this.installPluginHookFile();\n\n const template = this.loadTemplate('hooks');\n if (!template) return false;\n\n const targetPath = path.join(this.projectRoot, reg.hooksTarget);\n const settings = readJsonFile(targetPath);\n const existingHooks = (settings.hooks ?? {}) as Record<string, unknown[]>;\n\n // Build merged hooks: for each event, keep non-Myco groups + add template groups\n const mergedHooks: Record<string, unknown[]> = {};\n\n // Preserve non-Myco hooks from existing config\n for (const [event, groups] of Object.entries(existingHooks)) {\n const nonMycoGroups = (groups as Array<Record<string, unknown>>).filter(\n (group) => !isMycoHookGroup(group),\n );\n if (nonMycoGroups.length > 0) {\n mergedHooks[event] = nonMycoGroups;\n }\n }\n\n // Add template hooks\n for (const [event, groups] of Object.entries(template)) {\n mergedHooks[event] = [...(mergedHooks[event] ?? []), ...(groups as unknown[])];\n }\n\n settings.hooks = mergedHooks;\n writeJsonFile(targetPath, settings);\n return true;\n }\n\n /**\n * Install a plugin-file hook target by copying a verbatim template.\n * Used for agents whose hook system is plugin-based rather than JSON entry-based\n * (e.g., opencode's TypeScript plugin system).\n */\n private installPluginHookFile(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.hooksTarget) return false;\n\n const templateContent = this.loadTemplateRaw('plugin.ts');\n if (templateContent === null) return false;\n\n return this.writeManagedFile(\n path.join(this.projectRoot, reg.hooksTarget),\n templateContent,\n );\n }\n\n /**\n * Remove a plugin-file hook target.\n * Only deletes files whose content contains the Myco plugin marker — contributors\n * who hand-edit the plugin file without removing the marker are protected.\n */\n private uninstallPluginHookFile(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.hooksTarget) return false;\n\n const targetPath = path.join(this.projectRoot, reg.hooksTarget);\n let content: string;\n try { content = fs.readFileSync(targetPath, 'utf-8'); } catch { return false; }\n\n if (!content.includes(MYCO_PLUGIN_FILE_MARKER)) return false;\n\n try {\n fs.unlinkSync(targetPath);\n // Remove parent plugins dir if now empty\n try { fs.rmdirSync(path.dirname(targetPath)); } catch { /* not empty or missing */ }\n return true;\n } catch {\n return false;\n }\n }\n\n /**\n * Install a plugin deps package.json for plugin-file agents (e.g., opencode).\n * Writes the template verbatim so the agent's package manager can install the SDK.\n */\n private installPluginPackage(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.pluginPackageTarget) return false;\n\n const templateContent = this.loadTemplateRaw('package.json');\n if (templateContent === null) return false;\n\n return this.writeManagedFile(\n path.join(this.projectRoot, reg.pluginPackageTarget),\n templateContent,\n );\n }\n\n /**\n * Merge MCP server template into the target config file.\n * Replaces the `myco` server entry; preserves other servers.\n */\n installMcp(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.mcpTarget) return false;\n\n const template = this.loadTemplate('mcp');\n if (!template) return false;\n\n const targetPath = path.join(this.projectRoot, reg.mcpTarget);\n if (reg.mcpFormat === 'toml') {\n return this.installMcpToml(targetPath, template);\n }\n return this.installMcpJson(targetPath, template);\n }\n\n /**\n * Write MCP servers to a JSON config file under the manifest-configured key.\n * Most agents use the canonical `mcpServers` key; opencode uses `mcp`.\n *\n * The `?? 'mcpServers'` fallback protects against test fixtures that construct\n * manifests as plain object literals and bypass the schema's default.\n */\n private installMcpJson(targetPath: string, template: Record<string, unknown>): boolean {\n const serversKey = this.manifest.registration!.mcpServersKey ?? 'mcpServers';\n const config = readJsonFile(targetPath);\n const servers = (config[serversKey] ?? {}) as Record<string, unknown>;\n\n for (const [name, def] of Object.entries(template)) {\n servers[name] = def;\n }\n\n config[serversKey] = servers;\n writeJsonFile(targetPath, config);\n return true;\n }\n\n /** Write MCP servers to a TOML config file. */\n private installMcpToml(targetPath: string, template: Record<string, unknown>): boolean {\n let raw = '';\n try { raw = fs.readFileSync(targetPath, 'utf-8'); } catch { /* doesn't exist */ }\n\n for (const [name, def] of Object.entries(template)) {\n raw = buildTomlMcpSection(raw, name, def as Record<string, unknown>);\n }\n\n fs.mkdirSync(path.dirname(targetPath), { recursive: true });\n fs.writeFileSync(targetPath, raw, 'utf-8');\n return true;\n }\n\n /**\n * Create symlinks for skills through .agents/skills/ canonical layer.\n * Canonical: .agents/skills/<name> -> <packageRoot>/skills/<name>\n * Agent-specific: <skillsTarget>/<name> -> ../../.agents/skills/<name>\n */\n installSkills(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.skillsTarget) return false;\n\n const skillNames = this.listSkillDirs();\n if (skillNames.length === 0) return false;\n\n const skillsSrc = path.join(this.packageRoot, SKILLS_SUBDIR);\n\n // Create canonical symlinks: .agents/skills/<name> -> package skills\n const canonicalDir = path.join(this.projectRoot, CANONICAL_SKILLS_DIR);\n fs.mkdirSync(canonicalDir, { recursive: true });\n\n for (const name of skillNames) {\n const canonicalLink = path.join(canonicalDir, name);\n const target = path.join(skillsSrc, name);\n ensureSymlink(canonicalLink, target);\n }\n\n // Create agent-specific symlinks if skillsTarget differs from canonical\n const agentSkillsDir = path.join(this.projectRoot, reg.skillsTarget);\n const canonicalRel = path.relative(agentSkillsDir, canonicalDir);\n\n if (reg.skillsTarget !== CANONICAL_SKILLS_DIR) {\n fs.mkdirSync(agentSkillsDir, { recursive: true });\n for (const name of skillNames) {\n const agentLink = path.join(agentSkillsDir, name);\n const relTarget = path.join(canonicalRel, name);\n ensureSymlink(agentLink, relTarget);\n }\n ensureLocalSkillsGitignore(agentSkillsDir);\n }\n\n return true;\n }\n\n /**\n * Merge settings template into the target settings file.\n * JSON targets: deep-merges objects and deduplicates arrays.\n * TOML targets: emits each top-level template key as a [section] with scalar children.\n */\n installSettings(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.settingsTarget) return false;\n\n const template = this.loadTemplate('settings');\n if (!template) return false;\n\n const targetPath = path.join(this.projectRoot, reg.settingsTarget);\n const settingsFormat = reg.settingsFormat ?? 'json';\n\n if (settingsFormat === 'toml') {\n return this.installSettingsToml(targetPath, template);\n }\n\n const existing = readJsonFile(targetPath);\n const merged = deepMergeSettings(existing, template);\n writeJsonFile(targetPath, merged);\n return true;\n }\n\n /**\n * Merge a settings template into a TOML config file.\n * Each top-level key in the template becomes a [section] header, with its\n * children written as scalar key = value lines. Existing sections and keys\n * outside the template (including unrelated sections like [mcp_servers.*])\n * are preserved.\n */\n private installSettingsToml(targetPath: string, template: Record<string, unknown>): boolean {\n let raw = '';\n try { raw = fs.readFileSync(targetPath, 'utf-8'); } catch { /* doesn't exist */ }\n\n for (const [sectionName, values] of Object.entries(template)) {\n if (!values || typeof values !== 'object' || Array.isArray(values)) continue;\n raw = upsertTomlSection(raw, sectionName, values as Record<string, unknown>);\n }\n\n fs.mkdirSync(path.dirname(targetPath), { recursive: true });\n fs.writeFileSync(targetPath, raw, 'utf-8');\n return true;\n }\n\n /**\n * Remove Myco entries from the target settings file.\n * Template-driven: loads the settings template and removes matching values.\n * JSON: arrays filtered by template values, object keys deleted by name.\n * TOML: removes each template key from its section; empty sections are dropped.\n */\n uninstallSettings(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.settingsTarget) return false;\n\n const template = this.loadTemplate('settings');\n if (!template) return false;\n\n const targetPath = path.join(this.projectRoot, reg.settingsTarget);\n const settingsFormat = reg.settingsFormat ?? 'json';\n\n if (settingsFormat === 'toml') {\n return this.uninstallSettingsToml(targetPath, template);\n }\n\n const settings = readJsonFile(targetPath);\n if (Object.keys(settings).length === 0) return false;\n\n const changed = deepRemoveSettings(settings, template);\n if (!changed) return false;\n\n writeOrDeleteJsonFile(targetPath, settings);\n return true;\n }\n\n /**\n * Remove template-defined keys from TOML settings file.\n * For each section in the template, deletes only the keys the template owns;\n * other keys and unrelated sections stay intact. Empty sections are stripped.\n * Deletes the file entirely if no TOML content remains.\n */\n private uninstallSettingsToml(targetPath: string, template: Record<string, unknown>): boolean {\n let raw = '';\n try { raw = fs.readFileSync(targetPath, 'utf-8'); } catch { return false; }\n if (!raw.trim()) return false;\n\n let changed = false;\n for (const [sectionName, values] of Object.entries(template)) {\n if (!values || typeof values !== 'object' || Array.isArray(values)) continue;\n const keys = Object.keys(values as Record<string, unknown>);\n const next = removeTomlSectionKeys(raw, sectionName, keys);\n if (next !== raw) {\n raw = next;\n changed = true;\n }\n }\n\n if (!changed) return false;\n\n if (!raw.trim()) {\n try { fs.unlinkSync(targetPath); } catch { /* ignore */ }\n } else {\n fs.writeFileSync(targetPath, raw, 'utf-8');\n }\n return true;\n }\n\n /**\n * Remove Myco hook groups from the target settings file.\n *\n * For plugin-file agents (e.g., opencode) this dispatches to `uninstallPluginHookFile()`\n * which deletes the verbatim plugin file (guarded by the Myco plugin marker).\n */\n uninstallHooks(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.hooksTarget) return false;\n\n if (reg.hooksFormat === HOOKS_FORMAT_PLUGIN_FILE) return this.uninstallPluginHookFile();\n\n const targetPath = path.join(this.projectRoot, reg.hooksTarget);\n const settings = readJsonFile(targetPath);\n const existingHooks = (settings.hooks ?? {}) as Record<string, unknown[]>;\n if (Object.keys(existingHooks).length === 0) return false;\n\n const cleaned: Record<string, unknown[]> = {};\n for (const [event, groups] of Object.entries(existingHooks)) {\n const nonMyco = (groups as Array<Record<string, unknown>>).filter(\n (group) => !isMycoHookGroup(group),\n );\n if (nonMyco.length > 0) {\n cleaned[event] = nonMyco;\n }\n }\n\n if (Object.keys(cleaned).length === 0) {\n delete settings.hooks;\n } else {\n settings.hooks = cleaned;\n }\n\n writeOrDeleteJsonFile(targetPath, settings);\n return true;\n }\n\n /** Remove Myco MCP server entry from the target config file. */\n uninstallMcp(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.mcpTarget) return false;\n\n const targetPath = path.join(this.projectRoot, reg.mcpTarget);\n if (reg.mcpFormat === 'toml') {\n return this.uninstallMcpToml(targetPath);\n }\n return this.uninstallMcpJson(targetPath);\n }\n\n private uninstallMcpJson(targetPath: string): boolean {\n // Fallback matches the schema default; protects test fixtures that bypass .parse().\n const serversKey = this.manifest.registration!.mcpServersKey ?? 'mcpServers';\n const config = readJsonFile(targetPath);\n const servers = (config[serversKey] ?? {}) as Record<string, unknown>;\n if (!servers[MYCO_MCP_SERVER_NAME]) return false;\n\n delete servers[MYCO_MCP_SERVER_NAME];\n\n if (Object.keys(servers).length === 0) {\n delete config[serversKey];\n } else {\n config[serversKey] = servers;\n }\n\n writeOrDeleteJsonFile(targetPath, config);\n return true;\n }\n\n private uninstallMcpToml(targetPath: string): boolean {\n let raw = '';\n try { raw = fs.readFileSync(targetPath, 'utf-8'); } catch { return false; }\n\n const sectionHeader = `[mcp_servers.${MYCO_MCP_SERVER_NAME}]`;\n if (!raw.includes(sectionHeader)) return false;\n\n const startIdx = raw.indexOf(sectionHeader);\n const endIdx = findTomlSectionEnd(raw, startIdx + sectionHeader.length, `mcp_servers.${MYCO_MCP_SERVER_NAME}`);\n const before = raw.slice(0, startIdx).trimEnd();\n const after = raw.slice(endIdx).trimStart();\n const updated = (before + (before && after ? '\\n\\n' : '') + after).trimEnd();\n\n if (!updated.trim()) {\n try { fs.unlinkSync(targetPath); } catch { /* ignore */ }\n } else {\n fs.writeFileSync(targetPath, updated + '\\n', 'utf-8');\n }\n return true;\n }\n\n /** Remove skill symlinks (canonical + agent-specific). */\n uninstallSkills(): boolean {\n const reg = this.manifest.registration;\n if (!reg?.skillsTarget) return false;\n\n const skillNames = this.listSkillDirs();\n if (skillNames.length === 0) return false;\n\n let removed = false;\n\n // Remove agent-specific symlinks\n if (reg.skillsTarget !== CANONICAL_SKILLS_DIR) {\n for (const name of skillNames) {\n const link = path.join(this.projectRoot, reg.skillsTarget, name);\n try { fs.unlinkSync(link); removed = true; } catch { /* doesn't exist */ }\n }\n // Remove agent skills dir if now empty (rmdirSync fails atomically if non-empty)\n try { fs.rmdirSync(path.join(this.projectRoot, reg.skillsTarget)); } catch { /* not empty or missing */ }\n }\n\n // Remove canonical symlinks\n const canonicalDir = path.join(this.projectRoot, CANONICAL_SKILLS_DIR);\n for (const name of skillNames) {\n const link = path.join(canonicalDir, name);\n try { fs.unlinkSync(link); removed = true; } catch { /* doesn't exist */ }\n }\n // Remove empty dirs (rmdirSync fails atomically if non-empty)\n try { fs.rmdirSync(canonicalDir); } catch { /* not empty or missing */ }\n try { fs.rmdirSync(path.join(this.projectRoot, '.agents')); } catch { /* not empty or missing */ }\n\n return removed;\n }\n\n /** Remove Myco entries from project .gitignore. */\n private cleanGitignore(): void {\n const gitignorePath = path.join(this.projectRoot, '.gitignore');\n let content = '';\n try { content = fs.readFileSync(gitignorePath, 'utf-8'); } catch { return; }\n\n const cleaned = this.stripMycoGitignoreBlock(content, this.listSkillDirs()).trim();\n if (cleaned) {\n fs.writeFileSync(gitignorePath, cleaned + '\\n', 'utf-8');\n } else {\n try { fs.unlinkSync(gitignorePath); } catch { /* ignore */ }\n }\n }\n}\n\n/**\n * Create agent-specific symlinks for a skill in `.agents/skills/<name>`.\n *\n * Reads all symbiont manifests to find skillsTarget paths that differ\n * from the canonical `.agents/skills/` directory, then creates relative\n * symlinks from each target to the canonical location.\n *\n * Called by vault_write_skill after writing a generated skill to disk.\n * Also handles removal: when `remove` is true, deletes the symlinks.\n */\nexport function syncSkillSymlinks(\n projectRoot: string,\n skillName: string,\n opts?: { remove?: boolean },\n): void {\n // Resolve manifests dir — try sibling (source layout) then dist layout\n // (tsup bundles into dist/chunk-*.js, but manifests are at dist/src/symbionts/manifests/)\n const selfDir = path.dirname(new URL(import.meta.url).pathname);\n const candidates = [\n path.join(selfDir, 'manifests'),\n path.join(selfDir, 'src', 'symbionts', 'manifests'),\n ];\n const manifestDir = candidates.find((d) => fs.existsSync(d));\n if (!manifestDir) return;\n\n const targets = new Set<string>();\n for (const file of fs.readdirSync(manifestDir).filter((f) => f.endsWith('.yaml'))) {\n try {\n const content = fs.readFileSync(path.join(manifestDir, file), 'utf-8');\n const match = content.match(/skillsTarget:\\s*(.+)/);\n if (match) targets.add(match[1].trim());\n } catch { /* skip unreadable manifests */ }\n }\n\n for (const target of targets) {\n if (target === CANONICAL_SKILLS_DIR) continue; // canonical is the source, not a link target\n\n const agentSkillsDir = path.join(projectRoot, target);\n const linkPath = path.join(agentSkillsDir, skillName);\n\n if (opts?.remove) {\n try { fs.unlinkSync(linkPath); } catch { /* doesn't exist */ }\n try { fs.rmdirSync(agentSkillsDir); } catch { /* not empty or missing */ }\n } else {\n fs.mkdirSync(agentSkillsDir, { recursive: true });\n const canonicalDir = path.join(projectRoot, CANONICAL_SKILLS_DIR);\n const relTarget = path.join(path.relative(agentSkillsDir, canonicalDir), skillName);\n ensureSymlink(linkPath, relTarget);\n // Ensure a local .gitignore ignores all symlinks in this directory.\n // Localized to the agent's skills dir — doesn't pollute the project .gitignore.\n ensureLocalSkillsGitignore(agentSkillsDir);\n }\n }\n}\n\n/** Content for the local .gitignore that ignores Myco-created symlinks. */\nconst LOCAL_SKILLS_GITIGNORE = `# Myco-managed symlinks — generated skills are symlinked here automatically.\n# The canonical location for all skills is .agents/skills/.\n#\n# To add your own skill to this directory, un-ignore it:\n# !my-skill\n*\n!.gitignore\n`;\n\n/**\n * Write a .gitignore inside an agent's skills directory that ignores all\n * symlinks Myco creates there. Idempotent — skips if already present.\n */\nfunction ensureLocalSkillsGitignore(agentSkillsDir: string): void {\n const gitignorePath = path.join(agentSkillsDir, '.gitignore');\n try {\n if (fs.readFileSync(gitignorePath, 'utf-8') === LOCAL_SKILLS_GITIGNORE) return;\n } catch { /* doesn't exist — proceed */ }\n fs.writeFileSync(gitignorePath, LOCAL_SKILLS_GITIGNORE, 'utf-8');\n}\n","/** TOML section header pattern. */\nconst TOML_SECTION_RE = /^\\[([^\\]]+)\\]/;\n\n/**\n * Find where a named TOML section ends.\n *\n * Walks forward from `searchStart` until it hits a section header that is\n * NEITHER the named section itself NOR any of its subtables (headers that\n * start with `${sectionName}.`). Returns `raw.length` if the section extends\n * to the end of the file.\n */\nexport function findTomlSectionEnd(\n raw: string,\n searchStart: number,\n sectionName: string,\n): number {\n const subsectionPrefix = `${sectionName}.`;\n const rawLines = raw.slice(searchStart).split('\\n');\n let offset = searchStart;\n for (const line of rawLines) {\n offset += line.length + 1;\n const m = line.match(TOML_SECTION_RE);\n if (m && !m[1].startsWith(subsectionPrefix) && m[1] !== sectionName) {\n return offset - line.length - 1;\n }\n }\n return raw.length;\n}\n\n/** Format a single TOML key = value line from a JS value. Skips unsupported types. */\nfunction formatTomlScalarLine(key: string, val: unknown): string | null {\n if (typeof val === 'string') return `${key} = \"${val}\"`;\n if (typeof val === 'boolean') return `${key} = ${val}`;\n if (typeof val === 'number' && Number.isFinite(val)) return `${key} = ${val}`;\n if (Array.isArray(val)) {\n return `${key} = [${val.map((v: unknown) => `\"${v}\"`).join(', ')}]`;\n }\n return null;\n}\n\n/**\n * Insert or replace a top-level TOML section with the given scalar key/value pairs.\n *\n * - Scalar values (string, boolean, number, string[]) are written as `key = value`.\n * - Nested objects are ignored — use {@link buildTomlMcpSection} or call\n * `upsertTomlSection` once per subtable if you need nested tables.\n * - Idempotent: running twice with the same inputs produces identical output.\n * - Preserves content before and after the section.\n */\nexport function upsertTomlSection(\n raw: string,\n sectionName: string,\n values: Record<string, unknown>,\n): string {\n const sectionHeader = `[${sectionName}]`;\n\n const lines: string[] = [sectionHeader];\n for (const [key, val] of Object.entries(values)) {\n const line = formatTomlScalarLine(key, val);\n if (line !== null) lines.push(line);\n }\n const block = lines.join('\\n');\n\n if (raw.includes(sectionHeader)) {\n const startIdx = raw.indexOf(sectionHeader);\n const endIdx = findTomlSectionEnd(raw, startIdx + sectionHeader.length, sectionName);\n const before = raw.slice(0, startIdx).trimEnd();\n const after = raw.slice(endIdx);\n const separator = before ? '\\n\\n' : '';\n return (before + separator + block + after).trimEnd() + '\\n';\n }\n\n const separator = raw.trim() ? '\\n\\n' : '';\n return (raw.trimEnd() + separator + block).trimEnd() + '\\n';\n}\n\n/**\n * Remove specific keys from a top-level TOML section.\n *\n * - Only the listed keys are removed. Other keys in the section are preserved.\n * - If the section ends up with no content after removal, the whole section\n * (header and body) is stripped.\n * - Returns the updated string; equal to the input if no changes were made.\n */\nexport function removeTomlSectionKeys(\n raw: string,\n sectionName: string,\n keys: string[],\n): string {\n const sectionHeader = `[${sectionName}]`;\n if (!raw.includes(sectionHeader)) return raw;\n\n const startIdx = raw.indexOf(sectionHeader);\n const endIdx = findTomlSectionEnd(raw, startIdx + sectionHeader.length, sectionName);\n const sectionBody = raw.slice(startIdx + sectionHeader.length, endIdx);\n\n const keyRes = keys.map((k) => new RegExp(`^\\\\s*${escapeRegExp(k)}\\\\s*=`));\n const bodyLines = sectionBody.split('\\n');\n const kept: string[] = [];\n let removedAny = false;\n let subtableSeen = false;\n for (const line of bodyLines) {\n // Once we hit a subtable header, everything after stays (keys belong to the parent)\n if (subtableSeen || TOML_SECTION_RE.test(line.trim())) {\n subtableSeen = true;\n kept.push(line);\n continue;\n }\n if (keyRes.some((re) => re.test(line))) {\n removedAny = true;\n continue;\n }\n kept.push(line);\n }\n if (!removedAny) return raw;\n\n // Determine whether the parent section still has any key = value lines.\n const hasRemainingKeys = kept.some((line) => {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith('#')) return false;\n if (TOML_SECTION_RE.test(trimmed)) return false;\n return /=/.test(trimmed);\n });\n\n const before = raw.slice(0, startIdx).trimEnd();\n let rebuilt: string;\n if (hasRemainingKeys) {\n const newBody = kept.join('\\n');\n const separator = before ? '\\n\\n' : '';\n rebuilt = before + separator + sectionHeader + newBody;\n } else {\n // Drop the header entirely; keep any trailing subtables we preserved above.\n const trailing = kept.join('\\n').trimStart();\n const separator = before && trailing ? '\\n\\n' : '';\n rebuilt = before + separator + trailing;\n }\n return rebuilt.trimEnd() + (rebuilt.trim() ? '\\n' : '');\n}\n\nfunction escapeRegExp(s: string): string {\n return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n}\n\n/**\n * Build/update a specific mcp_servers entry in a TOML string.\n * Pure transformation — returns updated content without writing to disk.\n *\n * Handles the MCP-specific `env` subtable pattern; for plain sections use\n * {@link upsertTomlSection}.\n */\nexport function buildTomlMcpSection(\n raw: string,\n serverName: string,\n server: Record<string, unknown>,\n): string {\n const sectionName = `mcp_servers.${serverName}`;\n const sectionHeader = `[${sectionName}]`;\n\n // Build the TOML block for this server\n const lines: string[] = [sectionHeader];\n for (const [key, val] of Object.entries(server)) {\n if (key === 'env' && typeof val === 'object' && val !== null) continue; // Handle env as subtable\n const line = formatTomlScalarLine(key, val);\n if (line !== null) lines.push(line);\n }\n\n // Add env subtable if present\n const env = server.env as Record<string, string> | undefined;\n if (env && Object.keys(env).length > 0) {\n lines.push('');\n lines.push(`[${sectionName}.env]`);\n for (const [key, val] of Object.entries(env)) {\n lines.push(`${key} = \"${val}\"`);\n }\n }\n\n const block = lines.join('\\n');\n\n let updated: string;\n if (raw.includes(sectionHeader)) {\n const startIdx = raw.indexOf(sectionHeader);\n const endIdx = findTomlSectionEnd(raw, startIdx + sectionHeader.length, sectionName);\n const before = raw.slice(0, startIdx).trimEnd();\n const after = raw.slice(endIdx);\n const separator = before ? '\\n\\n' : '';\n updated = (before + separator + block + after).trimEnd() + '\\n';\n } else {\n // Append new section\n const separator = raw.trim() ? '\\n\\n' : '';\n updated = (raw.trimEnd() + separator + block).trimEnd() + '\\n';\n }\n\n return updated;\n}\n","/** Deep merge two settings objects. Arrays are appended + deduplicated; objects recurse. */\nexport function deepMergeSettings(\n target: Record<string, unknown>,\n source: Record<string, unknown>,\n): Record<string, unknown> {\n const result = { ...target };\n for (const [key, sourceVal] of Object.entries(source)) {\n const targetVal = result[key];\n if (Array.isArray(sourceVal) && Array.isArray(targetVal)) {\n result[key] = [...new Set([...targetVal, ...sourceVal])];\n } else if (isPlainObject(sourceVal) && isPlainObject(targetVal)) {\n result[key] = deepMergeSettings(\n targetVal as Record<string, unknown>,\n sourceVal as Record<string, unknown>,\n );\n } else {\n result[key] = sourceVal;\n }\n }\n return result;\n}\n\nexport function isPlainObject(val: unknown): val is Record<string, unknown> {\n return typeof val === 'object' && val !== null && !Array.isArray(val);\n}\n\n/**\n * Remove values from target that match the template structure.\n * Arrays: filter out values present in the template array.\n * Objects: delete keys present in the template object, recurse into nested objects.\n * Returns true if anything was removed.\n */\nexport function deepRemoveSettings(\n target: Record<string, unknown>,\n template: Record<string, unknown>,\n): boolean {\n let changed = false;\n for (const [key, templateVal] of Object.entries(template)) {\n const targetVal = target[key];\n if (targetVal === undefined) continue;\n\n if (Array.isArray(templateVal) && Array.isArray(targetVal)) {\n // Filter out values that appear in the template array\n const templateSet = new Set(templateVal.map(String));\n const filtered = targetVal.filter((v) => !templateSet.has(String(v)));\n if (filtered.length !== targetVal.length) {\n if (filtered.length > 0) {\n target[key] = filtered;\n } else {\n delete target[key];\n }\n changed = true;\n }\n } else if (isPlainObject(templateVal) && isPlainObject(targetVal)) {\n // Recurse into nested objects, then prune if empty\n if (deepRemoveSettings(targetVal, templateVal)) {\n if (Object.keys(targetVal).length === 0) {\n delete target[key];\n }\n changed = true;\n }\n } else {\n // Scalar: delete if value matches\n if (String(targetVal) === String(templateVal)) {\n delete target[key];\n changed = true;\n }\n }\n }\n return changed;\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\n\nexport function readJsonFile(filePath: string): Record<string, unknown> {\n try {\n return JSON.parse(fs.readFileSync(filePath, 'utf-8'));\n } catch {\n return {};\n }\n}\n\nexport function writeJsonFile(filePath: string, data: Record<string, unknown>): void {\n fs.mkdirSync(path.dirname(filePath), { recursive: true });\n fs.writeFileSync(filePath, JSON.stringify(data, null, 2) + '\\n', 'utf-8');\n}\n\n/** Write a JSON file, or delete it if the object is empty. */\nexport function writeOrDeleteJsonFile(filePath: string, data: Record<string, unknown>): void {\n if (Object.keys(data).length === 0) {\n try { fs.unlinkSync(filePath); } catch { /* ignore */ }\n } else {\n writeJsonFile(filePath, data);\n }\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\n\n/** Prefix used to identify Myco-owned hooks in settings files. */\nconst MYCO_HOOK_COMMAND_PREFIX = 'myco-run';\n\n/** Check if a command string belongs to Myco (old or new guard format). */\nexport function isMycoHookCommand(command: string): boolean {\n return command.startsWith(MYCO_HOOK_COMMAND_PREFIX) || command.includes('.agents/myco-hook.cjs');\n}\n\n/**\n * Check if a hook group is Myco-owned.\n * Handles both nested format (Claude Code, Codex, etc.) and flat format (Windsurf).\n *\n * Nested: { hooks: [{ command: \"cd \\\"$(git rev-parse ...)\\\" && node .agents/myco-hook.cjs ...\" }] }\n * Flat: { command: \"cd \\\"$(git rev-parse ...)\\\" && node .agents/myco-hook.cjs ...\" }\n */\nexport function isMycoHookGroup(group: Record<string, unknown>): boolean {\n // Nested format: { hooks: [{ command: \"...\" }] }\n if (Array.isArray(group.hooks) && group.hooks.some((h: { command?: string }) => h.command && isMycoHookCommand(h.command))) return true;\n // Flat format: { command: \"...\" }\n if (typeof group.command === 'string' && isMycoHookCommand(group.command)) return true;\n return false;\n}\n\n/**\n * Create a starter AGENTS.md if the project doesn't have one.\n * Idempotent — skips if AGENTS.md already exists.\n */\nexport function ensureAgentsMd(projectRoot: string, packageRoot: string): void {\n const agentsMdPath = path.join(projectRoot, 'AGENTS.md');\n if (fs.existsSync(agentsMdPath)) return;\n\n const candidates = [\n path.join(packageRoot, 'src/symbionts/templates/agents-starter.md'),\n path.join(packageRoot, 'dist/src/symbionts/templates/agents-starter.md'),\n ];\n for (const p of candidates) {\n try {\n const content = fs.readFileSync(p, 'utf-8');\n fs.writeFileSync(agentsMdPath, content, 'utf-8');\n return;\n } catch { /* try next */ }\n }\n}\n\nexport function ensureSymlink(linkPath: string, target: string): void {\n try {\n if (fs.readlinkSync(linkPath) === target) return;\n } catch { /* does not exist or is not a symlink — proceed */ }\n try { fs.rmSync(linkPath, { recursive: true, force: true }); } catch { /* ignore */ }\n fs.symlinkSync(target, linkPath);\n}\n"],"mappings":";;;AACA,OAAOA,SAAQ;AACf,OAAOC,WAAU;;;ACDjB,IAAM,kBAAkB;AAUjB,SAAS,mBACd,KACA,aACA,aACQ;AACR,QAAM,mBAAmB,GAAG,WAAW;AACvC,QAAM,WAAW,IAAI,MAAM,WAAW,EAAE,MAAM,IAAI;AAClD,MAAI,SAAS;AACb,aAAW,QAAQ,UAAU;AAC3B,cAAU,KAAK,SAAS;AACxB,UAAM,IAAI,KAAK,MAAM,eAAe;AACpC,QAAI,KAAK,CAAC,EAAE,CAAC,EAAE,WAAW,gBAAgB,KAAK,EAAE,CAAC,MAAM,aAAa;AACnE,aAAO,SAAS,KAAK,SAAS;AAAA,IAChC;AAAA,EACF;AACA,SAAO,IAAI;AACb;AAGA,SAAS,qBAAqB,KAAa,KAA6B;AACtE,MAAI,OAAO,QAAQ,SAAU,QAAO,GAAG,GAAG,OAAO,GAAG;AACpD,MAAI,OAAO,QAAQ,UAAW,QAAO,GAAG,GAAG,MAAM,GAAG;AACpD,MAAI,OAAO,QAAQ,YAAY,OAAO,SAAS,GAAG,EAAG,QAAO,GAAG,GAAG,MAAM,GAAG;AAC3E,MAAI,MAAM,QAAQ,GAAG,GAAG;AACtB,WAAO,GAAG,GAAG,OAAO,IAAI,IAAI,CAAC,MAAe,IAAI,CAAC,GAAG,EAAE,KAAK,IAAI,CAAC;AAAA,EAClE;AACA,SAAO;AACT;AAWO,SAAS,kBACd,KACA,aACA,QACQ;AACR,QAAM,gBAAgB,IAAI,WAAW;AAErC,QAAM,QAAkB,CAAC,aAAa;AACtC,aAAW,CAAC,KAAK,GAAG,KAAK,OAAO,QAAQ,MAAM,GAAG;AAC/C,UAAM,OAAO,qBAAqB,KAAK,GAAG;AAC1C,QAAI,SAAS,KAAM,OAAM,KAAK,IAAI;AAAA,EACpC;AACA,QAAM,QAAQ,MAAM,KAAK,IAAI;AAE7B,MAAI,IAAI,SAAS,aAAa,GAAG;AAC/B,UAAM,WAAW,IAAI,QAAQ,aAAa;AAC1C,UAAM,SAAS,mBAAmB,KAAK,WAAW,cAAc,QAAQ,WAAW;AACnF,UAAM,SAAS,IAAI,MAAM,GAAG,QAAQ,EAAE,QAAQ;AAC9C,UAAM,QAAQ,IAAI,MAAM,MAAM;AAC9B,UAAMC,aAAY,SAAS,SAAS;AACpC,YAAQ,SAASA,aAAY,QAAQ,OAAO,QAAQ,IAAI;AAAA,EAC1D;AAEA,QAAM,YAAY,IAAI,KAAK,IAAI,SAAS;AACxC,UAAQ,IAAI,QAAQ,IAAI,YAAY,OAAO,QAAQ,IAAI;AACzD;AAUO,SAAS,sBACd,KACA,aACA,MACQ;AACR,QAAM,gBAAgB,IAAI,WAAW;AACrC,MAAI,CAAC,IAAI,SAAS,aAAa,EAAG,QAAO;AAEzC,QAAM,WAAW,IAAI,QAAQ,aAAa;AAC1C,QAAM,SAAS,mBAAmB,KAAK,WAAW,cAAc,QAAQ,WAAW;AACnF,QAAM,cAAc,IAAI,MAAM,WAAW,cAAc,QAAQ,MAAM;AAErE,QAAM,SAAS,KAAK,IAAI,CAAC,MAAM,IAAI,OAAO,QAAQ,aAAa,CAAC,CAAC,OAAO,CAAC;AACzE,QAAM,YAAY,YAAY,MAAM,IAAI;AACxC,QAAM,OAAiB,CAAC;AACxB,MAAI,aAAa;AACjB,MAAI,eAAe;AACnB,aAAW,QAAQ,WAAW;AAE5B,QAAI,gBAAgB,gBAAgB,KAAK,KAAK,KAAK,CAAC,GAAG;AACrD,qBAAe;AACf,WAAK,KAAK,IAAI;AACd;AAAA,IACF;AACA,QAAI,OAAO,KAAK,CAAC,OAAO,GAAG,KAAK,IAAI,CAAC,GAAG;AACtC,mBAAa;AACb;AAAA,IACF;AACA,SAAK,KAAK,IAAI;AAAA,EAChB;AACA,MAAI,CAAC,WAAY,QAAO;AAGxB,QAAM,mBAAmB,KAAK,KAAK,CAAC,SAAS;AAC3C,UAAM,UAAU,KAAK,KAAK;AAC1B,QAAI,CAAC,WAAW,QAAQ,WAAW,GAAG,EAAG,QAAO;AAChD,QAAI,gBAAgB,KAAK,OAAO,EAAG,QAAO;AAC1C,WAAO,IAAI,KAAK,OAAO;AAAA,EACzB,CAAC;AAED,QAAM,SAAS,IAAI,MAAM,GAAG,QAAQ,EAAE,QAAQ;AAC9C,MAAI;AACJ,MAAI,kBAAkB;AACpB,UAAM,UAAU,KAAK,KAAK,IAAI;AAC9B,UAAM,YAAY,SAAS,SAAS;AACpC,cAAU,SAAS,YAAY,gBAAgB;AAAA,EACjD,OAAO;AAEL,UAAM,WAAW,KAAK,KAAK,IAAI,EAAE,UAAU;AAC3C,UAAM,YAAY,UAAU,WAAW,SAAS;AAChD,cAAU,SAAS,YAAY;AAAA,EACjC;AACA,SAAO,QAAQ,QAAQ,KAAK,QAAQ,KAAK,IAAI,OAAO;AACtD;AAEA,SAAS,aAAa,GAAmB;AACvC,SAAO,EAAE,QAAQ,uBAAuB,MAAM;AAChD;AASO,SAAS,oBACd,KACA,YACA,QACQ;AACR,QAAM,cAAc,eAAe,UAAU;AAC7C,QAAM,gBAAgB,IAAI,WAAW;AAGrC,QAAM,QAAkB,CAAC,aAAa;AACtC,aAAW,CAAC,KAAK,GAAG,KAAK,OAAO,QAAQ,MAAM,GAAG;AAC/C,QAAI,QAAQ,SAAS,OAAO,QAAQ,YAAY,QAAQ,KAAM;AAC9D,UAAM,OAAO,qBAAqB,KAAK,GAAG;AAC1C,QAAI,SAAS,KAAM,OAAM,KAAK,IAAI;AAAA,EACpC;AAGA,QAAM,MAAM,OAAO;AACnB,MAAI,OAAO,OAAO,KAAK,GAAG,EAAE,SAAS,GAAG;AACtC,UAAM,KAAK,EAAE;AACb,UAAM,KAAK,IAAI,WAAW,OAAO;AACjC,eAAW,CAAC,KAAK,GAAG,KAAK,OAAO,QAAQ,GAAG,GAAG;AAC5C,YAAM,KAAK,GAAG,GAAG,OAAO,GAAG,GAAG;AAAA,IAChC;AAAA,EACF;AAEA,QAAM,QAAQ,MAAM,KAAK,IAAI;AAE7B,MAAI;AACJ,MAAI,IAAI,SAAS,aAAa,GAAG;AAC/B,UAAM,WAAW,IAAI,QAAQ,aAAa;AAC1C,UAAM,SAAS,mBAAmB,KAAK,WAAW,cAAc,QAAQ,WAAW;AACnF,UAAM,SAAS,IAAI,MAAM,GAAG,QAAQ,EAAE,QAAQ;AAC9C,UAAM,QAAQ,IAAI,MAAM,MAAM;AAC9B,UAAM,YAAY,SAAS,SAAS;AACpC,eAAW,SAAS,YAAY,QAAQ,OAAO,QAAQ,IAAI;AAAA,EAC7D,OAAO;AAEL,UAAM,YAAY,IAAI,KAAK,IAAI,SAAS;AACxC,eAAW,IAAI,QAAQ,IAAI,YAAY,OAAO,QAAQ,IAAI;AAAA,EAC5D;AAEA,SAAO;AACT;;;AChMO,SAAS,kBACd,QACA,QACyB;AACzB,QAAM,SAAS,EAAE,GAAG,OAAO;AAC3B,aAAW,CAAC,KAAK,SAAS,KAAK,OAAO,QAAQ,MAAM,GAAG;AACrD,UAAM,YAAY,OAAO,GAAG;AAC5B,QAAI,MAAM,QAAQ,SAAS,KAAK,MAAM,QAAQ,SAAS,GAAG;AACxD,aAAO,GAAG,IAAI,CAAC,GAAG,oBAAI,IAAI,CAAC,GAAG,WAAW,GAAG,SAAS,CAAC,CAAC;AAAA,IACzD,WAAW,cAAc,SAAS,KAAK,cAAc,SAAS,GAAG;AAC/D,aAAO,GAAG,IAAI;AAAA,QACZ;AAAA,QACA;AAAA,MACF;AAAA,IACF,OAAO;AACL,aAAO,GAAG,IAAI;AAAA,IAChB;AAAA,EACF;AACA,SAAO;AACT;AAEO,SAAS,cAAc,KAA8C;AAC1E,SAAO,OAAO,QAAQ,YAAY,QAAQ,QAAQ,CAAC,MAAM,QAAQ,GAAG;AACtE;AAQO,SAAS,mBACd,QACA,UACS;AACT,MAAI,UAAU;AACd,aAAW,CAAC,KAAK,WAAW,KAAK,OAAO,QAAQ,QAAQ,GAAG;AACzD,UAAM,YAAY,OAAO,GAAG;AAC5B,QAAI,cAAc,OAAW;AAE7B,QAAI,MAAM,QAAQ,WAAW,KAAK,MAAM,QAAQ,SAAS,GAAG;AAE1D,YAAM,cAAc,IAAI,IAAI,YAAY,IAAI,MAAM,CAAC;AACnD,YAAM,WAAW,UAAU,OAAO,CAAC,MAAM,CAAC,YAAY,IAAI,OAAO,CAAC,CAAC,CAAC;AACpE,UAAI,SAAS,WAAW,UAAU,QAAQ;AACxC,YAAI,SAAS,SAAS,GAAG;AACvB,iBAAO,GAAG,IAAI;AAAA,QAChB,OAAO;AACL,iBAAO,OAAO,GAAG;AAAA,QACnB;AACA,kBAAU;AAAA,MACZ;AAAA,IACF,WAAW,cAAc,WAAW,KAAK,cAAc,SAAS,GAAG;AAEjE,UAAI,mBAAmB,WAAW,WAAW,GAAG;AAC9C,YAAI,OAAO,KAAK,SAAS,EAAE,WAAW,GAAG;AACvC,iBAAO,OAAO,GAAG;AAAA,QACnB;AACA,kBAAU;AAAA,MACZ;AAAA,IACF,OAAO;AAEL,UAAI,OAAO,SAAS,MAAM,OAAO,WAAW,GAAG;AAC7C,eAAO,OAAO,GAAG;AACjB,kBAAU;AAAA,MACZ;AAAA,IACF;AAAA,EACF;AACA,SAAO;AACT;;;ACtEA,OAAO,QAAQ;AACf,OAAO,UAAU;AAEV,SAAS,aAAa,UAA2C;AACtE,MAAI;AACF,WAAO,KAAK,MAAM,GAAG,aAAa,UAAU,OAAO,CAAC;AAAA,EACtD,QAAQ;AACN,WAAO,CAAC;AAAA,EACV;AACF;AAEO,SAAS,cAAc,UAAkB,MAAqC;AACnF,KAAG,UAAU,KAAK,QAAQ,QAAQ,GAAG,EAAE,WAAW,KAAK,CAAC;AACxD,KAAG,cAAc,UAAU,KAAK,UAAU,MAAM,MAAM,CAAC,IAAI,MAAM,OAAO;AAC1E;AAGO,SAAS,sBAAsB,UAAkB,MAAqC;AAC3F,MAAI,OAAO,KAAK,IAAI,EAAE,WAAW,GAAG;AAClC,QAAI;AAAE,SAAG,WAAW,QAAQ;AAAA,IAAG,QAAQ;AAAA,IAAe;AAAA,EACxD,OAAO;AACL,kBAAc,UAAU,IAAI;AAAA,EAC9B;AACF;;;ACvBA,OAAOC,SAAQ;AACf,OAAOC,WAAU;AAGjB,IAAM,2BAA2B;AAG1B,SAAS,kBAAkB,SAA0B;AAC1D,SAAO,QAAQ,WAAW,wBAAwB,KAAK,QAAQ,SAAS,uBAAuB;AACjG;AASO,SAAS,gBAAgB,OAAyC;AAEvE,MAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,MAAM,MAAM,KAAK,CAAC,MAA4B,EAAE,WAAW,kBAAkB,EAAE,OAAO,CAAC,EAAG,QAAO;AAEnI,MAAI,OAAO,MAAM,YAAY,YAAY,kBAAkB,MAAM,OAAO,EAAG,QAAO;AAClF,SAAO;AACT;AAMO,SAAS,eAAe,aAAqB,aAA2B;AAC7E,QAAM,eAAeA,MAAK,KAAK,aAAa,WAAW;AACvD,MAAID,IAAG,WAAW,YAAY,EAAG;AAEjC,QAAM,aAAa;AAAA,IACjBC,MAAK,KAAK,aAAa,2CAA2C;AAAA,IAClEA,MAAK,KAAK,aAAa,gDAAgD;AAAA,EACzE;AACA,aAAW,KAAK,YAAY;AAC1B,QAAI;AACF,YAAM,UAAUD,IAAG,aAAa,GAAG,OAAO;AAC1C,MAAAA,IAAG,cAAc,cAAc,SAAS,OAAO;AAC/C;AAAA,IACF,QAAQ;AAAA,IAAiB;AAAA,EAC3B;AACF;AAEO,SAAS,cAAc,UAAkB,QAAsB;AACpE,MAAI;AACF,QAAIA,IAAG,aAAa,QAAQ,MAAM,OAAQ;AAAA,EAC5C,QAAQ;AAAA,EAAqD;AAC7D,MAAI;AAAE,IAAAA,IAAG,OAAO,UAAU,EAAE,WAAW,MAAM,OAAO,KAAK,CAAC;AAAA,EAAG,QAAQ;AAAA,EAAe;AACpF,EAAAA,IAAG,YAAY,QAAQ,QAAQ;AACjC;;;AJ5CA,IAAM,oBAAoB;AAG1B,IAAM,kCAAkC;AAGxC,IAAM,qBAAqB;AAG3B,IAAM,mBAAmB;AAGzB,IAAM,+BAA+B;AAGrC,IAAM,gCAAgC;AAGtC,IAAM,0BAA0B,WAAW,6BAA6B;AAGxE,IAAM,gBAAgB;AAGtB,IAAM,uBAAuB;AAGtB,IAAM,uBAAuB;AAOpC,IAAM,0BAA0B;AAGhC,IAAM,2BAA2B;AAGjC,IAAM,2BAA2B;AAGjC,IAAM,yBAAyB;AAC/B,IAAM,uBAAuB;AAG7B,IAAM,yBAAyB,GAAG,sBAAsB;AAAA;AAAA,EAEtD,oBAAoB;AAAA;AAAA;AAiBf,IAAM,oBAAN,MAAwB;AAAA,EAC7B,YACU,UACA,aACA,aACR;AAHQ;AACA;AACA;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOK,iBAAiB,SAAgC;AACvD,UAAM,aAAa;AAAA,MACjBE,MAAK,KAAK,KAAK,aAAa,kBAAkB,OAAO;AAAA;AAAA,MAErDA,MAAK,KAAK,KAAK,aAAa,QAAQ,kBAAkB,OAAO;AAAA,IAC/D;AACA,eAAW,YAAY,YAAY;AACjC,UAAI;AAAE,eAAOC,IAAG,aAAa,UAAU,OAAO;AAAA,MAAG,QAAQ;AAAA,MAAiB;AAAA,IAC5E;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,iBAAiB,SAAiB,SAA0B;AAClE,QAAI;AACF,UAAIA,IAAG,aAAa,SAAS,OAAO,MAAM,QAAS,QAAO;AAAA,IAC5D,QAAQ;AAAA,IAAgC;AACxC,IAAAA,IAAG,UAAUD,MAAK,QAAQ,OAAO,GAAG,EAAE,WAAW,KAAK,CAAC;AACvD,IAAAC,IAAG,cAAc,SAAS,SAAS,OAAO;AAC1C,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,mBAA4B;AAC1B,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,YAAa,QAAO;AAE9B,UAAM,gBAAgB,KAAK,iBAAiB,4BAA4B;AACxE,QAAI,CAAC,cAAe,QAAO;AAE3B,WAAO,KAAK;AAAA,MACVD,MAAK,KAAK,KAAK,aAAa,uBAAuB;AAAA,MACnD;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,qBAA8B;AAC5B,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,YAAa,QAAO;AAE9B,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,uBAAuB;AACtE,QAAI;AACF,MAAAC,IAAG,WAAW,UAAU;AACxB,aAAO;AAAA,IACT,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA,EAGA,aAAa,MAA8C;AACzD,UAAM,MAAM,KAAK,iBAAiBD,MAAK,KAAK,KAAK,SAAS,MAAM,GAAG,IAAI,OAAO,CAAC;AAC/E,QAAI,QAAQ,KAAM,QAAO;AACzB,QAAI;AAAE,aAAO,KAAK,MAAM,GAAG;AAAA,IAAG,QAAQ;AAAE,aAAO;AAAA,IAAM;AAAA,EACvD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,gBAAgB,UAAiC;AAC/C,WAAO,KAAK,iBAAiBA,MAAK,KAAK,KAAK,SAAS,MAAM,QAAQ,CAAC;AAAA,EACtE;AAAA;AAAA,EAGA,UAAyB;AACvB,UAAM,MAAM,KAAK,SAAS;AAE1B,SAAK,iBAAiB;AACtB,UAAM,SAAS,KAAK,uBAAuB,GAAG,IAC1C,KAAK,mBAAmB,GAAI,IAC5B;AAAA,MACE,OAAO,KAAK,aAAa;AAAA,MACzB,KAAK,KAAK,WAAW;AAAA,MACrB,QAAQ,KAAK,cAAc;AAAA,MAC3B,UAAU,KAAK,gBAAgB;AAAA,MAC/B,cAAc,KAAK,oBAAoB;AAAA,MACvC,eAAe,KAAK,qBAAqB;AAAA,IAC3C;AACJ,SAAK,gBAAgB;AACrB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUQ,uBAAuB,KAAiD;AAC9E,QAAI,CAAC,IAAK,QAAO;AACjB,UAAM,YAAY,IAAI,aAAa;AACnC,QAAI,cAAc,OAAQ,QAAO;AACjC,UAAM,UAAU,CAAC,IAAI,aAAa,IAAI,WAAW,IAAI,cAAc,EAAE,OAAO,OAAO;AACnF,WAAO,QAAQ,SAAS,KAAK,IAAI,IAAI,OAAO,EAAE,SAAS;AAAA,EACzD;AAAA;AAAA;AAAA;AAAA;AAAA,EAMQ,mBAAmB,KAAoE;AAC7F,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,eAAe,IAAI,aAAa,IAAI,cAAe;AACtG,QAAI,OAAO,aAAa,UAAU;AAClC,QAAI,QAAQ,OAAO,MAAM,OAAO,WAAW;AAG3C,UAAM,gBAAgB,IAAI,cAAc,KAAK,aAAa,OAAO,IAAI;AACrE,QAAI,eAAe;AACjB,YAAM,gBAAiB,KAAK,SAAS,CAAC;AACtC,YAAM,cAAyC,CAAC;AAChD,iBAAW,CAAC,OAAO,MAAM,KAAK,OAAO,QAAQ,aAAa,GAAG;AAC3D,cAAM,UAAW,OAA0C,OAAO,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;AAC5F,YAAI,QAAQ,SAAS,EAAG,aAAY,KAAK,IAAI;AAAA,MAC/C;AACA,iBAAW,CAAC,OAAO,MAAM,KAAK,OAAO,QAAQ,aAAa,GAAG;AAC3D,oBAAY,KAAK,IAAI,CAAC,GAAI,YAAY,KAAK,KAAK,CAAC,GAAI,GAAI,MAAoB;AAAA,MAC/E;AACA,WAAK,QAAQ;AACb,cAAQ;AAAA,IACV;AAGA,UAAM,cAAc,IAAI,YAAY,KAAK,aAAa,KAAK,IAAI;AAC/D,QAAI,aAAa;AACf,YAAM,aAAa,IAAI,iBAAiB;AACxC,YAAM,UAAW,KAAK,UAAU,KAAK,CAAC;AACtC,iBAAW,CAAC,MAAM,GAAG,KAAK,OAAO,QAAQ,WAAW,GAAG;AACrD,gBAAQ,IAAI,IAAI;AAAA,MAClB;AACA,WAAK,UAAU,IAAI;AACnB,YAAM;AAAA,IACR;AAGA,UAAM,mBAAmB,IAAI,iBAAiB,KAAK,aAAa,UAAU,IAAI;AAC9E,QAAI,kBAAkB;AACpB,aAAO,kBAAkB,MAAM,gBAAgB;AAC/C,iBAAW;AAAA,IACb;AAEA,kBAAc,YAAY,IAAI;AAE9B,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA,QAAQ,KAAK,cAAc;AAAA,MAC3B;AAAA,MACA,cAAc,KAAK,oBAAoB;AAAA,MACvC,eAAe,KAAK,qBAAqB;AAAA,IAC3C;AAAA,EACF;AAAA;AAAA,EAGA,YAA2B;AACzB,UAAM,MAAM,KAAK,SAAS;AAC1B,UAAM,SAAS,KAAK,uBAAuB,GAAG,IAC1C,KAAK,qBAAqB,GAAI,IAC9B;AAAA,MACE,OAAO,KAAK,eAAe;AAAA,MAC3B,KAAK,KAAK,aAAa;AAAA,MACvB,QAAQ,KAAK,gBAAgB;AAAA,MAC7B,UAAU,KAAK,kBAAkB;AAAA,MACjC,cAAc,KAAK,sBAAsB;AAAA,MACzC,eAAe;AAAA,IACjB;AAEJ,SAAK,mBAAmB;AACxB,SAAK,eAAe;AACpB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,qBAAqB,KAAoE;AAC/F,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,eAAe,IAAI,aAAa,IAAI,cAAe;AACtG,UAAM,OAAO,aAAa,UAAU;AACpC,QAAI,OAAO,KAAK,IAAI,EAAE,WAAW,GAAG;AAClC,aAAO;AAAA,QACL,OAAO;AAAA,QACP,KAAK;AAAA,QACL,QAAQ,KAAK,gBAAgB;AAAA,QAC7B,UAAU;AAAA,QACV,cAAc,KAAK,sBAAsB;AAAA,QACzC,eAAe;AAAA,MACjB;AAAA,IACF;AAEA,QAAI,QAAQ,OAAO,MAAM,OAAO,WAAW;AAG3C,QAAI,IAAI,aAAa;AACnB,YAAM,gBAAiB,KAAK,SAAS,CAAC;AACtC,UAAI,OAAO,KAAK,aAAa,EAAE,SAAS,GAAG;AACzC,cAAM,UAAqC,CAAC;AAC5C,mBAAW,CAAC,OAAO,MAAM,KAAK,OAAO,QAAQ,aAAa,GAAG;AAC3D,gBAAM,UAAW,OAA0C,OAAO,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;AAC5F,cAAI,QAAQ,SAAS,EAAG,SAAQ,KAAK,IAAI;AAAA,QAC3C;AACA,YAAI,OAAO,KAAK,OAAO,EAAE,WAAW,GAAG;AACrC,iBAAO,KAAK;AAAA,QACd,OAAO;AACL,eAAK,QAAQ;AAAA,QACf;AACA,gBAAQ;AAAA,MACV;AAAA,IACF;AAGA,QAAI,IAAI,WAAW;AACjB,YAAM,aAAa,IAAI,iBAAiB;AACxC,YAAM,UAAW,KAAK,UAAU,KAAK,CAAC;AACtC,UAAI,QAAQ,oBAAoB,GAAG;AACjC,eAAO,QAAQ,oBAAoB;AACnC,YAAI,OAAO,KAAK,OAAO,EAAE,WAAW,EAAG,QAAO,KAAK,UAAU;AAAA,YACxD,MAAK,UAAU,IAAI;AACxB,cAAM;AAAA,MACR;AAAA,IACF;AAGA,UAAM,mBAAmB,IAAI,iBAAiB,KAAK,aAAa,UAAU,IAAI;AAC9E,QAAI,kBAAkB;AACpB,iBAAW,mBAAmB,MAAM,gBAAgB;AAAA,IACtD;AAEA,0BAAsB,YAAY,IAAI;AAEtC,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA,QAAQ,KAAK,gBAAgB;AAAA,MAC7B;AAAA,MACA,cAAc,KAAK,sBAAsB;AAAA,MACzC,eAAe;AAAA,IACjB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,sBAA+B;AAC7B,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,iBAAkB,QAAO;AAGnC,mBAAe,KAAK,aAAa,KAAK,WAAW;AAEjD,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,gBAAgB;AAGnE,QAAI,WAA0B;AAC9B,QAAI;AAAE,iBAAWC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAA,IAAsB;AAErF,QAAI,aAAa,MAAM;AAErB,UAAI,SAAS,SAAS,sBAAsB,KAAK,SAAS,SAAS,wBAAwB,GAAG;AAC5F,eAAO;AAAA,MACT;AAEA,MAAAA,IAAG,cAAc,YAAY,yBAAyB,UAAU,OAAO;AACvE,aAAO;AAAA,IACT;AAGA,UAAM,qBAAqB;AAAA,MACzBD,MAAK,KAAK,KAAK,aAAa,8CAA8C;AAAA,MAC1EA,MAAK,KAAK,KAAK,aAAa,mDAAmD;AAAA,IACjF;AACA,QAAI,OAAsB;AAC1B,eAAW,KAAK,oBAAoB;AAClC,UAAI;AAAE,eAAOC,IAAG,aAAa,GAAG,OAAO;AAAG;AAAA,MAAO,QAAQ;AAAA,MAAiB;AAAA,IAC5E;AACA,QAAI,CAAC,KAAM,QAAO;AAElB,WAAO,KAAK,QAAQ,sBAAsB,KAAK,SAAS,WAAW;AACnE,IAAAA,IAAG,UAAUD,MAAK,QAAQ,UAAU,GAAG,EAAE,WAAW,KAAK,CAAC;AAC1D,IAAAC,IAAG,cAAc,YAAY,MAAM,OAAO;AAC1C,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,wBAAiC;AAC/B,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,iBAAkB,QAAO;AAEnC,UAAM,aAAaD,MAAK,KAAK,KAAK,aAAa,IAAI,gBAAgB;AACnE,QAAI;AACJ,QAAI;AAAE,gBAAUC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAE,aAAO;AAAA,IAAO;AAG9E,QAAI,QAAQ,SAAS,wBAAwB,GAAG;AAC9C,MAAAA,IAAG,WAAW,UAAU;AACxB,aAAO;AAAA,IACT;AAGA,QAAI,QAAQ,SAAS,sBAAsB,GAAG;AAC5C,YAAM,WAAW,QAAQ,QAAQ,sBAAsB;AACvD,YAAM,SAAS,QAAQ,QAAQ,oBAAoB;AACnD,UAAI,SAAS,UAAU;AAErB,cAAM,WAAW,SAAS,qBAAqB;AAC/C,cAAM,WAAW,QAAQ,MAAM,GAAG,QAAQ,IAAI,QAAQ,MAAM,QAAQ,GAAG,QAAQ,QAAQ,EAAE;AACzF,QAAAA,IAAG,cAAc,YAAY,SAAS,OAAO;AAC7C,eAAO;AAAA,MACT;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA;AAAA,EAGQ,gBAA0B;AAChC,QAAI;AACF,aAAOA,IAAG,YAAYD,MAAK,KAAK,KAAK,aAAa,aAAa,GAAG,EAAE,eAAe,KAAK,CAAC,EACtF,OAAO,CAAC,MAAM,EAAE,YAAY,CAAC,EAC7B,IAAI,CAAC,MAAM,EAAE,IAAI;AAAA,IACtB,QAAQ;AAAE,aAAO,CAAC;AAAA,IAAG;AAAA,EACvB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,kBAAwB;AAC9B,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,aAAc;AAExB,UAAM,aAAa,KAAK,cAAc;AAKtC,UAAM,UAAU;AAAA,MACd,GAAG,WAAW,IAAI,CAAC,SAAS,GAAG,oBAAoB,IAAI,IAAI,EAAE;AAAA,MAC7D;AAAA,IACF;AAEA,UAAM,gBAAgBA,MAAK,KAAK,KAAK,aAAa,YAAY;AAC9D,QAAI,UAAU;AACd,QAAI;AAAE,gBAAUC,IAAG,aAAa,eAAe,OAAO;AAAA,IAAG,QAAQ;AAAA,IAA0B;AAG3F,UAAM,WAAW,KAAK,wBAAwB,SAAS,UAAU;AAGjE,UAAM,eAAe,QAAQ,SAAS,IAClC,GAAG,iBAAiB;AAAA,EAAK,QAAQ,KAAK,IAAI,CAAC;AAAA,IAC3C;AAGJ,QAAI,aAAa,WAAW,iBAAiB,GAAI;AACjD,UAAM,YAAY,SAAS,SAAS,KAAK,CAAC,SAAS,SAAS,IAAI,IAAI,OAAO;AAC3E,UAAM,SAAS,SAAS,SAAS,KAAK,aAAa,SAAS,IAAI,OAAO;AACvE,UAAM,SAAS,WAAW,YAAY,SAAS;AAC/C,QAAI,WAAW,QAAS;AAExB,IAAAA,IAAG,cAAc,eAAe,QAAQ,OAAO;AAAA,EACjD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,wBAAwB,SAAiB,YAA8B;AAC7E,UAAM,MAAM,KAAK,SAAS;AAC1B,UAAM,aAAa,oBAAI,IAAY;AAAA,MACjC;AAAA,MACA;AAAA,MACA,GAAG,oBAAoB;AAAA;AAAA,MACvB;AAAA,IACF,CAAC;AACD,eAAW,QAAQ,YAAY;AAC7B,iBAAW,IAAI,GAAG,oBAAoB,IAAI,IAAI,EAAE;AAChD,UAAI,KAAK,gBAAgB,IAAI,iBAAiB,sBAAsB;AAClE,mBAAW,IAAI,GAAG,IAAI,YAAY,IAAI,IAAI,EAAE;AAAA,MAC9C;AAAA,IACF;AAEA,UAAM,QAAQ,QAAQ,MAAM,IAAI;AAChC,UAAM,WAAW,MAAM,OAAO,CAAC,SAAS,CAAC,WAAW,IAAI,IAAI,CAAC;AAC7D,WAAO,SAAS,KAAK,IAAI,EAAE,QAAQ,WAAW,MAAM,EAAE,QAAQ,KAAK,SAAS,SAAS,IAAI,OAAO;AAAA,EAClG;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,eAAwB;AACtB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,YAAa,QAAO;AAE9B,QAAI,IAAI,gBAAgB,yBAA0B,QAAO,KAAK,sBAAsB;AAEpF,UAAM,WAAW,KAAK,aAAa,OAAO;AAC1C,QAAI,CAAC,SAAU,QAAO;AAEtB,UAAM,aAAaD,MAAK,KAAK,KAAK,aAAa,IAAI,WAAW;AAC9D,UAAM,WAAW,aAAa,UAAU;AACxC,UAAM,gBAAiB,SAAS,SAAS,CAAC;AAG1C,UAAM,cAAyC,CAAC;AAGhD,eAAW,CAAC,OAAO,MAAM,KAAK,OAAO,QAAQ,aAAa,GAAG;AAC3D,YAAM,gBAAiB,OAA0C;AAAA,QAC/D,CAAC,UAAU,CAAC,gBAAgB,KAAK;AAAA,MACnC;AACA,UAAI,cAAc,SAAS,GAAG;AAC5B,oBAAY,KAAK,IAAI;AAAA,MACvB;AAAA,IACF;AAGA,eAAW,CAAC,OAAO,MAAM,KAAK,OAAO,QAAQ,QAAQ,GAAG;AACtD,kBAAY,KAAK,IAAI,CAAC,GAAI,YAAY,KAAK,KAAK,CAAC,GAAI,GAAI,MAAoB;AAAA,IAC/E;AAEA,aAAS,QAAQ;AACjB,kBAAc,YAAY,QAAQ;AAClC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,wBAAiC;AACvC,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,YAAa,QAAO;AAE9B,UAAM,kBAAkB,KAAK,gBAAgB,WAAW;AACxD,QAAI,oBAAoB,KAAM,QAAO;AAErC,WAAO,KAAK;AAAA,MACVA,MAAK,KAAK,KAAK,aAAa,IAAI,WAAW;AAAA,MAC3C;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,0BAAmC;AACzC,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,YAAa,QAAO;AAE9B,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,WAAW;AAC9D,QAAI;AACJ,QAAI;AAAE,gBAAUC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAE,aAAO;AAAA,IAAO;AAE9E,QAAI,CAAC,QAAQ,SAAS,uBAAuB,EAAG,QAAO;AAEvD,QAAI;AACF,MAAAA,IAAG,WAAW,UAAU;AAExB,UAAI;AAAE,QAAAA,IAAG,UAAUD,MAAK,QAAQ,UAAU,CAAC;AAAA,MAAG,QAAQ;AAAA,MAA6B;AACnF,aAAO;AAAA,IACT,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMQ,uBAAgC;AACtC,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,oBAAqB,QAAO;AAEtC,UAAM,kBAAkB,KAAK,gBAAgB,cAAc;AAC3D,QAAI,oBAAoB,KAAM,QAAO;AAErC,WAAO,KAAK;AAAA,MACVA,MAAK,KAAK,KAAK,aAAa,IAAI,mBAAmB;AAAA,MACnD;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,aAAsB;AACpB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,UAAW,QAAO;AAE5B,UAAM,WAAW,KAAK,aAAa,KAAK;AACxC,QAAI,CAAC,SAAU,QAAO;AAEtB,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,SAAS;AAC5D,QAAI,IAAI,cAAc,QAAQ;AAC5B,aAAO,KAAK,eAAe,YAAY,QAAQ;AAAA,IACjD;AACA,WAAO,KAAK,eAAe,YAAY,QAAQ;AAAA,EACjD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASQ,eAAe,YAAoB,UAA4C;AACrF,UAAM,aAAa,KAAK,SAAS,aAAc,iBAAiB;AAChE,UAAM,SAAS,aAAa,UAAU;AACtC,UAAM,UAAW,OAAO,UAAU,KAAK,CAAC;AAExC,eAAW,CAAC,MAAM,GAAG,KAAK,OAAO,QAAQ,QAAQ,GAAG;AAClD,cAAQ,IAAI,IAAI;AAAA,IAClB;AAEA,WAAO,UAAU,IAAI;AACrB,kBAAc,YAAY,MAAM;AAChC,WAAO;AAAA,EACT;AAAA;AAAA,EAGQ,eAAe,YAAoB,UAA4C;AACrF,QAAI,MAAM;AACV,QAAI;AAAE,YAAMC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAA,IAAsB;AAEhF,eAAW,CAAC,MAAM,GAAG,KAAK,OAAO,QAAQ,QAAQ,GAAG;AAClD,YAAM,oBAAoB,KAAK,MAAM,GAA8B;AAAA,IACrE;AAEA,IAAAA,IAAG,UAAUD,MAAK,QAAQ,UAAU,GAAG,EAAE,WAAW,KAAK,CAAC;AAC1D,IAAAC,IAAG,cAAc,YAAY,KAAK,OAAO;AACzC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,gBAAyB;AACvB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,aAAc,QAAO;AAE/B,UAAM,aAAa,KAAK,cAAc;AACtC,QAAI,WAAW,WAAW,EAAG,QAAO;AAEpC,UAAM,YAAYD,MAAK,KAAK,KAAK,aAAa,aAAa;AAG3D,UAAM,eAAeA,MAAK,KAAK,KAAK,aAAa,oBAAoB;AACrE,IAAAC,IAAG,UAAU,cAAc,EAAE,WAAW,KAAK,CAAC;AAE9C,eAAW,QAAQ,YAAY;AAC7B,YAAM,gBAAgBD,MAAK,KAAK,cAAc,IAAI;AAClD,YAAM,SAASA,MAAK,KAAK,WAAW,IAAI;AACxC,oBAAc,eAAe,MAAM;AAAA,IACrC;AAGA,UAAM,iBAAiBA,MAAK,KAAK,KAAK,aAAa,IAAI,YAAY;AACnE,UAAM,eAAeA,MAAK,SAAS,gBAAgB,YAAY;AAE/D,QAAI,IAAI,iBAAiB,sBAAsB;AAC7C,MAAAC,IAAG,UAAU,gBAAgB,EAAE,WAAW,KAAK,CAAC;AAChD,iBAAW,QAAQ,YAAY;AAC7B,cAAM,YAAYD,MAAK,KAAK,gBAAgB,IAAI;AAChD,cAAM,YAAYA,MAAK,KAAK,cAAc,IAAI;AAC9C,sBAAc,WAAW,SAAS;AAAA,MACpC;AACA,iCAA2B,cAAc;AAAA,IAC3C;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,kBAA2B;AACzB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,eAAgB,QAAO;AAEjC,UAAM,WAAW,KAAK,aAAa,UAAU;AAC7C,QAAI,CAAC,SAAU,QAAO;AAEtB,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,cAAc;AACjE,UAAM,iBAAiB,IAAI,kBAAkB;AAE7C,QAAI,mBAAmB,QAAQ;AAC7B,aAAO,KAAK,oBAAoB,YAAY,QAAQ;AAAA,IACtD;AAEA,UAAM,WAAW,aAAa,UAAU;AACxC,UAAM,SAAS,kBAAkB,UAAU,QAAQ;AACnD,kBAAc,YAAY,MAAM;AAChC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASQ,oBAAoB,YAAoB,UAA4C;AAC1F,QAAI,MAAM;AACV,QAAI;AAAE,YAAMC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAA,IAAsB;AAEhF,eAAW,CAAC,aAAa,MAAM,KAAK,OAAO,QAAQ,QAAQ,GAAG;AAC5D,UAAI,CAAC,UAAU,OAAO,WAAW,YAAY,MAAM,QAAQ,MAAM,EAAG;AACpE,YAAM,kBAAkB,KAAK,aAAa,MAAiC;AAAA,IAC7E;AAEA,IAAAA,IAAG,UAAUD,MAAK,QAAQ,UAAU,GAAG,EAAE,WAAW,KAAK,CAAC;AAC1D,IAAAC,IAAG,cAAc,YAAY,KAAK,OAAO;AACzC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,oBAA6B;AAC3B,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,eAAgB,QAAO;AAEjC,UAAM,WAAW,KAAK,aAAa,UAAU;AAC7C,QAAI,CAAC,SAAU,QAAO;AAEtB,UAAM,aAAaD,MAAK,KAAK,KAAK,aAAa,IAAI,cAAc;AACjE,UAAM,iBAAiB,IAAI,kBAAkB;AAE7C,QAAI,mBAAmB,QAAQ;AAC7B,aAAO,KAAK,sBAAsB,YAAY,QAAQ;AAAA,IACxD;AAEA,UAAM,WAAW,aAAa,UAAU;AACxC,QAAI,OAAO,KAAK,QAAQ,EAAE,WAAW,EAAG,QAAO;AAE/C,UAAM,UAAU,mBAAmB,UAAU,QAAQ;AACrD,QAAI,CAAC,QAAS,QAAO;AAErB,0BAAsB,YAAY,QAAQ;AAC1C,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQQ,sBAAsB,YAAoB,UAA4C;AAC5F,QAAI,MAAM;AACV,QAAI;AAAE,YAAMC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAE,aAAO;AAAA,IAAO;AAC1E,QAAI,CAAC,IAAI,KAAK,EAAG,QAAO;AAExB,QAAI,UAAU;AACd,eAAW,CAAC,aAAa,MAAM,KAAK,OAAO,QAAQ,QAAQ,GAAG;AAC5D,UAAI,CAAC,UAAU,OAAO,WAAW,YAAY,MAAM,QAAQ,MAAM,EAAG;AACpE,YAAM,OAAO,OAAO,KAAK,MAAiC;AAC1D,YAAM,OAAO,sBAAsB,KAAK,aAAa,IAAI;AACzD,UAAI,SAAS,KAAK;AAChB,cAAM;AACN,kBAAU;AAAA,MACZ;AAAA,IACF;AAEA,QAAI,CAAC,QAAS,QAAO;AAErB,QAAI,CAAC,IAAI,KAAK,GAAG;AACf,UAAI;AAAE,QAAAA,IAAG,WAAW,UAAU;AAAA,MAAG,QAAQ;AAAA,MAAe;AAAA,IAC1D,OAAO;AACL,MAAAA,IAAG,cAAc,YAAY,KAAK,OAAO;AAAA,IAC3C;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,iBAA0B;AACxB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,YAAa,QAAO;AAE9B,QAAI,IAAI,gBAAgB,yBAA0B,QAAO,KAAK,wBAAwB;AAEtF,UAAM,aAAaD,MAAK,KAAK,KAAK,aAAa,IAAI,WAAW;AAC9D,UAAM,WAAW,aAAa,UAAU;AACxC,UAAM,gBAAiB,SAAS,SAAS,CAAC;AAC1C,QAAI,OAAO,KAAK,aAAa,EAAE,WAAW,EAAG,QAAO;AAEpD,UAAM,UAAqC,CAAC;AAC5C,eAAW,CAAC,OAAO,MAAM,KAAK,OAAO,QAAQ,aAAa,GAAG;AAC3D,YAAM,UAAW,OAA0C;AAAA,QACzD,CAAC,UAAU,CAAC,gBAAgB,KAAK;AAAA,MACnC;AACA,UAAI,QAAQ,SAAS,GAAG;AACtB,gBAAQ,KAAK,IAAI;AAAA,MACnB;AAAA,IACF;AAEA,QAAI,OAAO,KAAK,OAAO,EAAE,WAAW,GAAG;AACrC,aAAO,SAAS;AAAA,IAClB,OAAO;AACL,eAAS,QAAQ;AAAA,IACnB;AAEA,0BAAsB,YAAY,QAAQ;AAC1C,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,eAAwB;AACtB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,UAAW,QAAO;AAE5B,UAAM,aAAaA,MAAK,KAAK,KAAK,aAAa,IAAI,SAAS;AAC5D,QAAI,IAAI,cAAc,QAAQ;AAC5B,aAAO,KAAK,iBAAiB,UAAU;AAAA,IACzC;AACA,WAAO,KAAK,iBAAiB,UAAU;AAAA,EACzC;AAAA,EAEQ,iBAAiB,YAA6B;AAEpD,UAAM,aAAa,KAAK,SAAS,aAAc,iBAAiB;AAChE,UAAM,SAAS,aAAa,UAAU;AACtC,UAAM,UAAW,OAAO,UAAU,KAAK,CAAC;AACxC,QAAI,CAAC,QAAQ,oBAAoB,EAAG,QAAO;AAE3C,WAAO,QAAQ,oBAAoB;AAEnC,QAAI,OAAO,KAAK,OAAO,EAAE,WAAW,GAAG;AACrC,aAAO,OAAO,UAAU;AAAA,IAC1B,OAAO;AACL,aAAO,UAAU,IAAI;AAAA,IACvB;AAEA,0BAAsB,YAAY,MAAM;AACxC,WAAO;AAAA,EACT;AAAA,EAEQ,iBAAiB,YAA6B;AACpD,QAAI,MAAM;AACV,QAAI;AAAE,YAAMC,IAAG,aAAa,YAAY,OAAO;AAAA,IAAG,QAAQ;AAAE,aAAO;AAAA,IAAO;AAE1E,UAAM,gBAAgB,gBAAgB,oBAAoB;AAC1D,QAAI,CAAC,IAAI,SAAS,aAAa,EAAG,QAAO;AAEzC,UAAM,WAAW,IAAI,QAAQ,aAAa;AAC1C,UAAM,SAAS,mBAAmB,KAAK,WAAW,cAAc,QAAQ,eAAe,oBAAoB,EAAE;AAC7G,UAAM,SAAS,IAAI,MAAM,GAAG,QAAQ,EAAE,QAAQ;AAC9C,UAAM,QAAQ,IAAI,MAAM,MAAM,EAAE,UAAU;AAC1C,UAAM,WAAW,UAAU,UAAU,QAAQ,SAAS,MAAM,OAAO,QAAQ;AAE3E,QAAI,CAAC,QAAQ,KAAK,GAAG;AACnB,UAAI;AAAE,QAAAA,IAAG,WAAW,UAAU;AAAA,MAAG,QAAQ;AAAA,MAAe;AAAA,IAC1D,OAAO;AACL,MAAAA,IAAG,cAAc,YAAY,UAAU,MAAM,OAAO;AAAA,IACtD;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,kBAA2B;AACzB,UAAM,MAAM,KAAK,SAAS;AAC1B,QAAI,CAAC,KAAK,aAAc,QAAO;AAE/B,UAAM,aAAa,KAAK,cAAc;AACtC,QAAI,WAAW,WAAW,EAAG,QAAO;AAEpC,QAAI,UAAU;AAGd,QAAI,IAAI,iBAAiB,sBAAsB;AAC7C,iBAAW,QAAQ,YAAY;AAC7B,cAAM,OAAOD,MAAK,KAAK,KAAK,aAAa,IAAI,cAAc,IAAI;AAC/D,YAAI;AAAE,UAAAC,IAAG,WAAW,IAAI;AAAG,oBAAU;AAAA,QAAM,QAAQ;AAAA,QAAsB;AAAA,MAC3E;AAEA,UAAI;AAAE,QAAAA,IAAG,UAAUD,MAAK,KAAK,KAAK,aAAa,IAAI,YAAY,CAAC;AAAA,MAAG,QAAQ;AAAA,MAA6B;AAAA,IAC1G;AAGA,UAAM,eAAeA,MAAK,KAAK,KAAK,aAAa,oBAAoB;AACrE,eAAW,QAAQ,YAAY;AAC7B,YAAM,OAAOA,MAAK,KAAK,cAAc,IAAI;AACzC,UAAI;AAAE,QAAAC,IAAG,WAAW,IAAI;AAAG,kBAAU;AAAA,MAAM,QAAQ;AAAA,MAAsB;AAAA,IAC3E;AAEA,QAAI;AAAE,MAAAA,IAAG,UAAU,YAAY;AAAA,IAAG,QAAQ;AAAA,IAA6B;AACvE,QAAI;AAAE,MAAAA,IAAG,UAAUD,MAAK,KAAK,KAAK,aAAa,SAAS,CAAC;AAAA,IAAG,QAAQ;AAAA,IAA6B;AAEjG,WAAO;AAAA,EACT;AAAA;AAAA,EAGQ,iBAAuB;AAC7B,UAAM,gBAAgBA,MAAK,KAAK,KAAK,aAAa,YAAY;AAC9D,QAAI,UAAU;AACd,QAAI;AAAE,gBAAUC,IAAG,aAAa,eAAe,OAAO;AAAA,IAAG,QAAQ;AAAE;AAAA,IAAQ;AAE3E,UAAM,UAAU,KAAK,wBAAwB,SAAS,KAAK,cAAc,CAAC,EAAE,KAAK;AACjF,QAAI,SAAS;AACX,MAAAA,IAAG,cAAc,eAAe,UAAU,MAAM,OAAO;AAAA,IACzD,OAAO;AACL,UAAI;AAAE,QAAAA,IAAG,WAAW,aAAa;AAAA,MAAG,QAAQ;AAAA,MAAe;AAAA,IAC7D;AAAA,EACF;AACF;AAYO,SAAS,kBACd,aACA,WACA,MACM;AAGN,QAAM,UAAUD,MAAK,QAAQ,IAAI,IAAI,YAAY,GAAG,EAAE,QAAQ;AAC9D,QAAM,aAAa;AAAA,IACjBA,MAAK,KAAK,SAAS,WAAW;AAAA,IAC9BA,MAAK,KAAK,SAAS,OAAO,aAAa,WAAW;AAAA,EACpD;AACA,QAAM,cAAc,WAAW,KAAK,CAAC,MAAMC,IAAG,WAAW,CAAC,CAAC;AAC3D,MAAI,CAAC,YAAa;AAElB,QAAM,UAAU,oBAAI,IAAY;AAChC,aAAW,QAAQA,IAAG,YAAY,WAAW,EAAE,OAAO,CAAC,MAAM,EAAE,SAAS,OAAO,CAAC,GAAG;AACjF,QAAI;AACF,YAAM,UAAUA,IAAG,aAAaD,MAAK,KAAK,aAAa,IAAI,GAAG,OAAO;AACrE,YAAM,QAAQ,QAAQ,MAAM,sBAAsB;AAClD,UAAI,MAAO,SAAQ,IAAI,MAAM,CAAC,EAAE,KAAK,CAAC;AAAA,IACxC,QAAQ;AAAA,IAAkC;AAAA,EAC5C;AAEA,aAAW,UAAU,SAAS;AAC5B,QAAI,WAAW,qBAAsB;AAErC,UAAM,iBAAiBA,MAAK,KAAK,aAAa,MAAM;AACpD,UAAM,WAAWA,MAAK,KAAK,gBAAgB,SAAS;AAEpD,QAAI,MAAM,QAAQ;AAChB,UAAI;AAAE,QAAAC,IAAG,WAAW,QAAQ;AAAA,MAAG,QAAQ;AAAA,MAAsB;AAC7D,UAAI;AAAE,QAAAA,IAAG,UAAU,cAAc;AAAA,MAAG,QAAQ;AAAA,MAA6B;AAAA,IAC3E,OAAO;AACL,MAAAA,IAAG,UAAU,gBAAgB,EAAE,WAAW,KAAK,CAAC;AAChD,YAAM,eAAeD,MAAK,KAAK,aAAa,oBAAoB;AAChE,YAAM,YAAYA,MAAK,KAAKA,MAAK,SAAS,gBAAgB,YAAY,GAAG,SAAS;AAClF,oBAAc,UAAU,SAAS;AAGjC,iCAA2B,cAAc;AAAA,IAC3C;AAAA,EACF;AACF;AAGA,IAAM,yBAAyB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAa/B,SAAS,2BAA2B,gBAA8B;AAChE,QAAM,gBAAgBA,MAAK,KAAK,gBAAgB,YAAY;AAC5D,MAAI;AACF,QAAIC,IAAG,aAAa,eAAe,OAAO,MAAM,uBAAwB;AAAA,EAC1E,QAAQ;AAAA,EAAgC;AACxC,EAAAA,IAAG,cAAc,eAAe,wBAAwB,OAAO;AACjE;","names":["fs","path","separator","fs","path","path","fs"]}
@@ -1,11 +1,11 @@
1
1
  import { createRequire as __cr } from 'node:module'; const require = __cr(import.meta.url);
2
2
  import {
3
3
  SymbiontInstaller
4
- } from "./chunk-TIAYBVSI.js";
4
+ } from "./chunk-VVGZL2HX.js";
5
5
  import {
6
6
  LmStudioBackend,
7
7
  OllamaBackend
8
- } from "./chunk-VWXDSDJU.js";
8
+ } from "./chunk-DKSQMH5X.js";
9
9
  import {
10
10
  closeDatabase,
11
11
  initDatabase,
@@ -13,7 +13,7 @@ import {
13
13
  } from "./chunk-MYX5NCRH.js";
14
14
  import {
15
15
  DaemonClient
16
- } from "./chunk-RJRRHTAA.js";
16
+ } from "./chunk-N6JNOSBQ.js";
17
17
 
18
18
  // src/cli/shared.ts
19
19
  import fs from "fs";
@@ -86,7 +86,8 @@ function registerSymbionts(manifests, projectRoot, packageRoot, verb) {
86
86
  result.mcp && "MCP server",
87
87
  result.skills && "skills",
88
88
  result.settings && "settings",
89
- result.instructions && "instructions"
89
+ result.instructions && "instructions",
90
+ result.pluginPackage && "plugin deps"
90
91
  ].filter(Boolean);
91
92
  if (installed.length > 0) {
92
93
  console.log(` \u2713 ${verb} ${manifest.displayName}: ${installed.join(", ")}`);
@@ -109,4 +110,4 @@ export {
109
110
  VAULT_GITIGNORE,
110
111
  registerSymbionts
111
112
  };
112
- //# sourceMappingURL=chunk-BNAYBGPH.js.map
113
+ //# sourceMappingURL=chunk-XQHL4GMO.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/cli/shared.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport os from 'node:os';\nimport { OllamaBackend } from '../intelligence/ollama.js';\nimport { LmStudioBackend } from '../intelligence/lm-studio.js';\n\nimport { DaemonClient } from '../hooks/client.js';\nimport { initDatabase, closeDatabase, vaultDbPath } from '../db/client.js';\nimport { SymbiontInstaller } from '../symbionts/installer.js';\nimport type { SymbiontManifest } from '../symbionts/manifest-schema.js';\n\nexport { parseStringFlag, parseIntFlag } from '../logs/format.js';\n\n/**\n * Initialize the singleton database for direct CLI reads.\n * Used by CLI commands that only need reads (stats, search, session).\n * Does NOT require the daemon to be running — WAL mode allows concurrent reads.\n *\n * @returns a cleanup function that closes the database.\n */\nexport function initVaultDb(vaultDir: string): () => void {\n initDatabase(vaultDbPath(vaultDir));\n return closeDatabase;\n}\n\n/** Connect to the daemon, ensuring it's running. Exits on failure. */\nexport async function connectToDaemon(vaultDir: string): Promise<DaemonClient> {\n const client = new DaemonClient(vaultDir);\n const healthy = await client.ensureRunning();\n if (!healthy) {\n console.error('Failed to connect to daemon');\n process.exit(1);\n }\n return client;\n}\n\n/** Load .env from cwd (not script location — that's the plugin install dir). */\nexport function loadEnv(): void {\n const envPath = path.resolve(process.cwd(), '.env');\n if (!fs.existsSync(envPath)) return;\n for (const line of fs.readFileSync(envPath, 'utf-8').split('\\n')) {\n const match = line.match(/^\\s*([^#=]+?)\\s*=\\s*(.*?)\\s*$/);\n if (match && !process.env[match[1]]) {\n process.env[match[1]] = match[2];\n }\n }\n}\n\nexport function isProcessAlive(pid: number): boolean {\n try { process.kill(pid, 0); return true; } catch { return false; }\n}\n\n// --- Provider defaults (sourced from backend classes) ---\nexport const PROVIDER_DEFAULTS: Record<string, { base_url: string }> = {\n ollama: { base_url: OllamaBackend.DEFAULT_BASE_URL },\n 'lm-studio': { base_url: LmStudioBackend.DEFAULT_BASE_URL },\n};\n\n\nexport const VAULT_GITIGNORE = `# SQLite database\nmyco.db*\nvectors.db*\n\n# Daemon state — per-machine, ephemeral\ndaemon.json\nbuffer/\nlogs/\n\n# Secrets — API keys for cloud providers\nsecrets.env\n\n# Machine ID\nmachine_id\n\n# Binary attachments — screenshots captured from transcripts\nattachments/\n\n# Team worker deployment — patched wrangler.toml + source copy\n.team-worker/\n`;\n\n/** Collapse an absolute home-dir path to its `~/` form for portable config storage. */\nexport function collapseHomePath(absPath: string): string {\n const home = os.homedir();\n if (absPath.startsWith(home + path.sep) || absPath === home) {\n return '~' + absPath.slice(home.length);\n }\n return absPath;\n}\n\n/**\n * Run the SymbiontInstaller for each symbiont manifest and log results.\n * Shared between myco init and myco update.\n */\nexport function registerSymbionts(\n manifests: SymbiontManifest[],\n projectRoot: string,\n packageRoot: string,\n verb: 'Registered' | 'Updated',\n): number {\n let count = 0;\n for (const manifest of manifests) {\n try {\n const installer = new SymbiontInstaller(manifest, projectRoot, packageRoot);\n const result = installer.install();\n\n const installed = [\n result.hooks && 'hooks',\n result.mcp && 'MCP server',\n result.skills && 'skills',\n result.settings && 'settings',\n result.instructions && 'instructions',\n ].filter(Boolean);\n\n if (installed.length > 0) {\n console.log(` \\u2713 ${verb} ${manifest.displayName}: ${installed.join(', ')}`);\n count++;\n } else {\n console.log(` \\u2013 ${manifest.displayName}: no registration targets configured`);\n }\n } catch (err) {\n console.error(` \\u2717 Failed to register ${manifest.displayName}: ${(err as Error).message}`);\n }\n }\n return count;\n}\n\n"],"mappings":";;;;;;;;;;;;;;;;;;AAAA,OAAO,QAAQ;AACf,OAAO,UAAU;AACjB,OAAO,QAAQ;AAkBR,SAAS,YAAY,UAA8B;AACxD,eAAa,YAAY,QAAQ,CAAC;AAClC,SAAO;AACT;AAGA,eAAsB,gBAAgB,UAAyC;AAC7E,QAAM,SAAS,IAAI,aAAa,QAAQ;AACxC,QAAM,UAAU,MAAM,OAAO,cAAc;AAC3C,MAAI,CAAC,SAAS;AACZ,YAAQ,MAAM,6BAA6B;AAC3C,YAAQ,KAAK,CAAC;AAAA,EAChB;AACA,SAAO;AACT;AAGO,SAAS,UAAgB;AAC9B,QAAM,UAAU,KAAK,QAAQ,QAAQ,IAAI,GAAG,MAAM;AAClD,MAAI,CAAC,GAAG,WAAW,OAAO,EAAG;AAC7B,aAAW,QAAQ,GAAG,aAAa,SAAS,OAAO,EAAE,MAAM,IAAI,GAAG;AAChE,UAAM,QAAQ,KAAK,MAAM,+BAA+B;AACxD,QAAI,SAAS,CAAC,QAAQ,IAAI,MAAM,CAAC,CAAC,GAAG;AACnC,cAAQ,IAAI,MAAM,CAAC,CAAC,IAAI,MAAM,CAAC;AAAA,IACjC;AAAA,EACF;AACF;AAEO,SAAS,eAAe,KAAsB;AACnD,MAAI;AAAE,YAAQ,KAAK,KAAK,CAAC;AAAG,WAAO;AAAA,EAAM,QAAQ;AAAE,WAAO;AAAA,EAAO;AACnE;AAGO,IAAM,oBAA0D;AAAA,EACrE,QAAQ,EAAE,UAAU,cAAc,iBAAiB;AAAA,EACnD,aAAa,EAAE,UAAU,gBAAgB,iBAAiB;AAC5D;AAGO,IAAM,kBAAkB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAmCxB,SAAS,kBACd,WACA,aACA,aACA,MACQ;AACR,MAAI,QAAQ;AACZ,aAAW,YAAY,WAAW;AAChC,QAAI;AACF,YAAM,YAAY,IAAI,kBAAkB,UAAU,aAAa,WAAW;AAC1E,YAAM,SAAS,UAAU,QAAQ;AAEjC,YAAM,YAAY;AAAA,QAChB,OAAO,SAAS;AAAA,QAChB,OAAO,OAAO;AAAA,QACd,OAAO,UAAU;AAAA,QACjB,OAAO,YAAY;AAAA,QACnB,OAAO,gBAAgB;AAAA,MACzB,EAAE,OAAO,OAAO;AAEhB,UAAI,UAAU,SAAS,GAAG;AACxB,gBAAQ,IAAI,YAAY,IAAI,IAAI,SAAS,WAAW,KAAK,UAAU,KAAK,IAAI,CAAC,EAAE;AAC/E;AAAA,MACF,OAAO;AACL,gBAAQ,IAAI,YAAY,SAAS,WAAW,sCAAsC;AAAA,MACpF;AAAA,IACF,SAAS,KAAK;AACZ,cAAQ,MAAM,+BAA+B,SAAS,WAAW,KAAM,IAAc,OAAO,EAAE;AAAA,IAChG;AAAA,EACF;AACA,SAAO;AACT;","names":[]}
1
+ {"version":3,"sources":["../src/cli/shared.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport os from 'node:os';\nimport { OllamaBackend } from '../intelligence/ollama.js';\nimport { LmStudioBackend } from '../intelligence/lm-studio.js';\n\nimport { DaemonClient } from '../hooks/client.js';\nimport { initDatabase, closeDatabase, vaultDbPath } from '../db/client.js';\nimport { SymbiontInstaller } from '../symbionts/installer.js';\nimport type { SymbiontManifest } from '../symbionts/manifest-schema.js';\n\nexport { parseStringFlag, parseIntFlag } from '../logs/format.js';\n\n/**\n * Initialize the singleton database for direct CLI reads.\n * Used by CLI commands that only need reads (stats, search, session).\n * Does NOT require the daemon to be running — WAL mode allows concurrent reads.\n *\n * @returns a cleanup function that closes the database.\n */\nexport function initVaultDb(vaultDir: string): () => void {\n initDatabase(vaultDbPath(vaultDir));\n return closeDatabase;\n}\n\n/** Connect to the daemon, ensuring it's running. Exits on failure. */\nexport async function connectToDaemon(vaultDir: string): Promise<DaemonClient> {\n const client = new DaemonClient(vaultDir);\n const healthy = await client.ensureRunning();\n if (!healthy) {\n console.error('Failed to connect to daemon');\n process.exit(1);\n }\n return client;\n}\n\n/** Load .env from cwd (not script location — that's the plugin install dir). */\nexport function loadEnv(): void {\n const envPath = path.resolve(process.cwd(), '.env');\n if (!fs.existsSync(envPath)) return;\n for (const line of fs.readFileSync(envPath, 'utf-8').split('\\n')) {\n const match = line.match(/^\\s*([^#=]+?)\\s*=\\s*(.*?)\\s*$/);\n if (match && !process.env[match[1]]) {\n process.env[match[1]] = match[2];\n }\n }\n}\n\nexport function isProcessAlive(pid: number): boolean {\n try { process.kill(pid, 0); return true; } catch { return false; }\n}\n\n// --- Provider defaults (sourced from backend classes) ---\nexport const PROVIDER_DEFAULTS: Record<string, { base_url: string }> = {\n ollama: { base_url: OllamaBackend.DEFAULT_BASE_URL },\n 'lm-studio': { base_url: LmStudioBackend.DEFAULT_BASE_URL },\n};\n\n\nexport const VAULT_GITIGNORE = `# SQLite database\nmyco.db*\nvectors.db*\n\n# Daemon state — per-machine, ephemeral\ndaemon.json\nbuffer/\nlogs/\n\n# Secrets — API keys for cloud providers\nsecrets.env\n\n# Machine ID\nmachine_id\n\n# Binary attachments — screenshots captured from transcripts\nattachments/\n\n# Team worker deployment — patched wrangler.toml + source copy\n.team-worker/\n`;\n\n/** Collapse an absolute home-dir path to its `~/` form for portable config storage. */\nexport function collapseHomePath(absPath: string): string {\n const home = os.homedir();\n if (absPath.startsWith(home + path.sep) || absPath === home) {\n return '~' + absPath.slice(home.length);\n }\n return absPath;\n}\n\n/**\n * Run the SymbiontInstaller for each symbiont manifest and log results.\n * Shared between myco init and myco update.\n */\nexport function registerSymbionts(\n manifests: SymbiontManifest[],\n projectRoot: string,\n packageRoot: string,\n verb: 'Registered' | 'Updated',\n): number {\n let count = 0;\n for (const manifest of manifests) {\n try {\n const installer = new SymbiontInstaller(manifest, projectRoot, packageRoot);\n const result = installer.install();\n\n const installed = [\n result.hooks && 'hooks',\n result.mcp && 'MCP server',\n result.skills && 'skills',\n result.settings && 'settings',\n result.instructions && 'instructions',\n result.pluginPackage && 'plugin deps',\n ].filter(Boolean);\n\n if (installed.length > 0) {\n console.log(` \\u2713 ${verb} ${manifest.displayName}: ${installed.join(', ')}`);\n count++;\n } else {\n console.log(` \\u2013 ${manifest.displayName}: no registration targets configured`);\n }\n } catch (err) {\n console.error(` \\u2717 Failed to register ${manifest.displayName}: ${(err as Error).message}`);\n }\n }\n return count;\n}\n\n"],"mappings":";;;;;;;;;;;;;;;;;;AAAA,OAAO,QAAQ;AACf,OAAO,UAAU;AACjB,OAAO,QAAQ;AAkBR,SAAS,YAAY,UAA8B;AACxD,eAAa,YAAY,QAAQ,CAAC;AAClC,SAAO;AACT;AAGA,eAAsB,gBAAgB,UAAyC;AAC7E,QAAM,SAAS,IAAI,aAAa,QAAQ;AACxC,QAAM,UAAU,MAAM,OAAO,cAAc;AAC3C,MAAI,CAAC,SAAS;AACZ,YAAQ,MAAM,6BAA6B;AAC3C,YAAQ,KAAK,CAAC;AAAA,EAChB;AACA,SAAO;AACT;AAGO,SAAS,UAAgB;AAC9B,QAAM,UAAU,KAAK,QAAQ,QAAQ,IAAI,GAAG,MAAM;AAClD,MAAI,CAAC,GAAG,WAAW,OAAO,EAAG;AAC7B,aAAW,QAAQ,GAAG,aAAa,SAAS,OAAO,EAAE,MAAM,IAAI,GAAG;AAChE,UAAM,QAAQ,KAAK,MAAM,+BAA+B;AACxD,QAAI,SAAS,CAAC,QAAQ,IAAI,MAAM,CAAC,CAAC,GAAG;AACnC,cAAQ,IAAI,MAAM,CAAC,CAAC,IAAI,MAAM,CAAC;AAAA,IACjC;AAAA,EACF;AACF;AAEO,SAAS,eAAe,KAAsB;AACnD,MAAI;AAAE,YAAQ,KAAK,KAAK,CAAC;AAAG,WAAO;AAAA,EAAM,QAAQ;AAAE,WAAO;AAAA,EAAO;AACnE;AAGO,IAAM,oBAA0D;AAAA,EACrE,QAAQ,EAAE,UAAU,cAAc,iBAAiB;AAAA,EACnD,aAAa,EAAE,UAAU,gBAAgB,iBAAiB;AAC5D;AAGO,IAAM,kBAAkB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAmCxB,SAAS,kBACd,WACA,aACA,aACA,MACQ;AACR,MAAI,QAAQ;AACZ,aAAW,YAAY,WAAW;AAChC,QAAI;AACF,YAAM,YAAY,IAAI,kBAAkB,UAAU,aAAa,WAAW;AAC1E,YAAM,SAAS,UAAU,QAAQ;AAEjC,YAAM,YAAY;AAAA,QAChB,OAAO,SAAS;AAAA,QAChB,OAAO,OAAO;AAAA,QACd,OAAO,UAAU;AAAA,QACjB,OAAO,YAAY;AAAA,QACnB,OAAO,gBAAgB;AAAA,QACvB,OAAO,iBAAiB;AAAA,MAC1B,EAAE,OAAO,OAAO;AAEhB,UAAI,UAAU,SAAS,GAAG;AACxB,gBAAQ,IAAI,YAAY,IAAI,IAAI,SAAS,WAAW,KAAK,UAAU,KAAK,IAAI,CAAC,EAAE;AAC/E;AAAA,MACF,OAAO;AACL,gBAAQ,IAAI,YAAY,SAAS,WAAW,sCAAsC;AAAA,MACpF;AAAA,IACF,SAAS,KAAK;AACZ,cAAQ,MAAM,+BAA+B,SAAS,WAAW,KAAM,IAAc,OAAO,EAAE;AAAA,IAChG;AAAA,EACF;AACA,SAAO;AACT;","names":[]}