claude-mococo 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (159) hide show
  1. package/LICENSE +21 -0
  2. package/README.ko.md +171 -0
  3. package/README.md +180 -0
  4. package/defaults/shared-rules.md +70 -0
  5. package/dist/bot/client.d.ts +10 -0
  6. package/dist/bot/client.d.ts.map +1 -0
  7. package/dist/bot/client.js +886 -0
  8. package/dist/bot/client.js.map +1 -0
  9. package/dist/bot/discord-commands.d.ts +36 -0
  10. package/dist/bot/discord-commands.d.ts.map +1 -0
  11. package/dist/bot/discord-commands.js +811 -0
  12. package/dist/bot/discord-commands.js.map +1 -0
  13. package/dist/bot/embeds.d.ts +5 -0
  14. package/dist/bot/embeds.d.ts.map +1 -0
  15. package/dist/bot/embeds.js +20 -0
  16. package/dist/bot/embeds.js.map +1 -0
  17. package/dist/bot/episode-writer.d.ts +5 -0
  18. package/dist/bot/episode-writer.d.ts.map +1 -0
  19. package/dist/bot/episode-writer.js +131 -0
  20. package/dist/bot/episode-writer.js.map +1 -0
  21. package/dist/bot/improvement-scanner.d.ts +5 -0
  22. package/dist/bot/improvement-scanner.d.ts.map +1 -0
  23. package/dist/bot/improvement-scanner.js +563 -0
  24. package/dist/bot/improvement-scanner.js.map +1 -0
  25. package/dist/bot/inbox-compactor.d.ts +6 -0
  26. package/dist/bot/inbox-compactor.d.ts.map +1 -0
  27. package/dist/bot/inbox-compactor.js +550 -0
  28. package/dist/bot/inbox-compactor.js.map +1 -0
  29. package/dist/bot/memory-consolidator.d.ts +4 -0
  30. package/dist/bot/memory-consolidator.d.ts.map +1 -0
  31. package/dist/bot/memory-consolidator.js +258 -0
  32. package/dist/bot/memory-consolidator.js.map +1 -0
  33. package/dist/bot/router.d.ts +4 -0
  34. package/dist/bot/router.d.ts.map +1 -0
  35. package/dist/bot/router.js +24 -0
  36. package/dist/bot/router.js.map +1 -0
  37. package/dist/cli/commands/add.d.ts +2 -0
  38. package/dist/cli/commands/add.d.ts.map +1 -0
  39. package/dist/cli/commands/add.js +203 -0
  40. package/dist/cli/commands/add.js.map +1 -0
  41. package/dist/cli/commands/dev.d.ts +2 -0
  42. package/dist/cli/commands/dev.d.ts.map +1 -0
  43. package/dist/cli/commands/dev.js +44 -0
  44. package/dist/cli/commands/dev.js.map +1 -0
  45. package/dist/cli/commands/edit.d.ts +2 -0
  46. package/dist/cli/commands/edit.d.ts.map +1 -0
  47. package/dist/cli/commands/edit.js +213 -0
  48. package/dist/cli/commands/edit.js.map +1 -0
  49. package/dist/cli/commands/init.d.ts +2 -0
  50. package/dist/cli/commands/init.d.ts.map +1 -0
  51. package/dist/cli/commands/init.js +126 -0
  52. package/dist/cli/commands/init.js.map +1 -0
  53. package/dist/cli/commands/list.d.ts +2 -0
  54. package/dist/cli/commands/list.d.ts.map +1 -0
  55. package/dist/cli/commands/list.js +21 -0
  56. package/dist/cli/commands/list.js.map +1 -0
  57. package/dist/cli/commands/remove.d.ts +2 -0
  58. package/dist/cli/commands/remove.d.ts.map +1 -0
  59. package/dist/cli/commands/remove.js +38 -0
  60. package/dist/cli/commands/remove.js.map +1 -0
  61. package/dist/cli/commands/restart.d.ts +2 -0
  62. package/dist/cli/commands/restart.d.ts.map +1 -0
  63. package/dist/cli/commands/restart.js +12 -0
  64. package/dist/cli/commands/restart.js.map +1 -0
  65. package/dist/cli/commands/start.d.ts +2 -0
  66. package/dist/cli/commands/start.d.ts.map +1 -0
  67. package/dist/cli/commands/start.js +33 -0
  68. package/dist/cli/commands/start.js.map +1 -0
  69. package/dist/cli/index.d.ts +3 -0
  70. package/dist/cli/index.d.ts.map +1 -0
  71. package/dist/cli/index.js +78 -0
  72. package/dist/cli/index.js.map +1 -0
  73. package/dist/cli/prompt-template.d.ts +18 -0
  74. package/dist/cli/prompt-template.d.ts.map +1 -0
  75. package/dist/cli/prompt-template.js +55 -0
  76. package/dist/cli/prompt-template.js.map +1 -0
  77. package/dist/cli/readline-utils.d.ts +5 -0
  78. package/dist/cli/readline-utils.d.ts.map +1 -0
  79. package/dist/cli/readline-utils.js +39 -0
  80. package/dist/cli/readline-utils.js.map +1 -0
  81. package/dist/cli/workspace.d.ts +7 -0
  82. package/dist/cli/workspace.d.ts.map +1 -0
  83. package/dist/cli/workspace.js +39 -0
  84. package/dist/cli/workspace.js.map +1 -0
  85. package/dist/config.d.ts +4 -0
  86. package/dist/config.d.ts.map +1 -0
  87. package/dist/config.js +96 -0
  88. package/dist/config.js.map +1 -0
  89. package/dist/index.d.ts +2 -0
  90. package/dist/index.d.ts.map +1 -0
  91. package/dist/index.js +34 -0
  92. package/dist/index.js.map +1 -0
  93. package/dist/orchestrator/claude-engine.d.ts +8 -0
  94. package/dist/orchestrator/claude-engine.d.ts.map +1 -0
  95. package/dist/orchestrator/claude-engine.js +94 -0
  96. package/dist/orchestrator/claude-engine.js.map +1 -0
  97. package/dist/orchestrator/codex-engine.d.ts +10 -0
  98. package/dist/orchestrator/codex-engine.d.ts.map +1 -0
  99. package/dist/orchestrator/codex-engine.js +82 -0
  100. package/dist/orchestrator/codex-engine.js.map +1 -0
  101. package/dist/orchestrator/engine-base.d.ts +21 -0
  102. package/dist/orchestrator/engine-base.d.ts.map +1 -0
  103. package/dist/orchestrator/engine-base.js +20 -0
  104. package/dist/orchestrator/engine-base.js.map +1 -0
  105. package/dist/orchestrator/engines.d.ts +5 -0
  106. package/dist/orchestrator/engines.d.ts.map +1 -0
  107. package/dist/orchestrator/engines.js +16 -0
  108. package/dist/orchestrator/engines.js.map +1 -0
  109. package/dist/orchestrator/gemini-engine.d.ts +10 -0
  110. package/dist/orchestrator/gemini-engine.d.ts.map +1 -0
  111. package/dist/orchestrator/gemini-engine.js +79 -0
  112. package/dist/orchestrator/gemini-engine.js.map +1 -0
  113. package/dist/orchestrator/mcp-config.d.ts +4 -0
  114. package/dist/orchestrator/mcp-config.d.ts.map +1 -0
  115. package/dist/orchestrator/mcp-config.js +23 -0
  116. package/dist/orchestrator/mcp-config.js.map +1 -0
  117. package/dist/orchestrator/prompt-builder.d.ts +3 -0
  118. package/dist/orchestrator/prompt-builder.d.ts.map +1 -0
  119. package/dist/orchestrator/prompt-builder.js +445 -0
  120. package/dist/orchestrator/prompt-builder.js.map +1 -0
  121. package/dist/server/hook-receiver.d.ts +5 -0
  122. package/dist/server/hook-receiver.d.ts.map +1 -0
  123. package/dist/server/hook-receiver.js +52 -0
  124. package/dist/server/hook-receiver.js.map +1 -0
  125. package/dist/teams/concurrency.d.ts +12 -0
  126. package/dist/teams/concurrency.d.ts.map +1 -0
  127. package/dist/teams/concurrency.js +43 -0
  128. package/dist/teams/concurrency.js.map +1 -0
  129. package/dist/teams/context.d.ts +5 -0
  130. package/dist/teams/context.d.ts.map +1 -0
  131. package/dist/teams/context.js +33 -0
  132. package/dist/teams/context.js.map +1 -0
  133. package/dist/teams/dispatch-ledger.d.ts +27 -0
  134. package/dist/teams/dispatch-ledger.d.ts.map +1 -0
  135. package/dist/teams/dispatch-ledger.js +90 -0
  136. package/dist/teams/dispatch-ledger.js.map +1 -0
  137. package/dist/teams/invoker.d.ts +9 -0
  138. package/dist/teams/invoker.d.ts.map +1 -0
  139. package/dist/teams/invoker.js +54 -0
  140. package/dist/teams/invoker.js.map +1 -0
  141. package/dist/types.d.ts +110 -0
  142. package/dist/types.d.ts.map +1 -0
  143. package/dist/types.js +2 -0
  144. package/dist/types.js.map +1 -0
  145. package/dist/utils/fs.d.ts +3 -0
  146. package/dist/utils/fs.d.ts.map +1 -0
  147. package/dist/utils/fs.js +17 -0
  148. package/dist/utils/fs.js.map +1 -0
  149. package/dist/utils/github-status.d.ts +25 -0
  150. package/dist/utils/github-status.d.ts.map +1 -0
  151. package/dist/utils/github-status.js +158 -0
  152. package/dist/utils/github-status.js.map +1 -0
  153. package/dist/utils/haiku.d.ts +6 -0
  154. package/dist/utils/haiku.d.ts.map +1 -0
  155. package/dist/utils/haiku.js +43 -0
  156. package/dist/utils/haiku.js.map +1 -0
  157. package/hooks/event-bridge.sh +21 -0
  158. package/hooks/permission-gate.sh +67 -0
  159. package/package.json +52 -0
@@ -0,0 +1,18 @@
1
+ export interface PromptOptions {
2
+ name: string;
3
+ mbti: string;
4
+ speechStyle: string;
5
+ traits: string[];
6
+ habits: string[];
7
+ role: string;
8
+ scope: string[];
9
+ notScope: string[];
10
+ authorityIndependent: string;
11
+ authorityNeedsApproval: string;
12
+ expertise: string[];
13
+ rules: string[];
14
+ isLeader: boolean;
15
+ humanTitle?: string;
16
+ }
17
+ export declare function generatePrompt(opts: PromptOptions): string;
18
+ //# sourceMappingURL=prompt-template.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompt-template.d.ts","sourceRoot":"","sources":["../../src/cli/prompt-template.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,MAAM,EAAE,CAAC;IACjB,MAAM,EAAE,MAAM,EAAE,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,QAAQ,EAAE,MAAM,EAAE,CAAC;IACnB,oBAAoB,EAAE,MAAM,CAAC;IAC7B,sBAAsB,EAAE,MAAM,CAAC;IAC/B,SAAS,EAAE,MAAM,EAAE,CAAC;IACpB,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,QAAQ,EAAE,OAAO,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,wBAAgB,cAAc,CAAC,IAAI,EAAE,aAAa,GAAG,MAAM,CA6D1D"}
@@ -0,0 +1,55 @@
1
+ export function generatePrompt(opts) {
2
+ const humanTitle = opts.humanTitle || 'Boss';
3
+ const traitsBlock = opts.traits.length > 0
4
+ ? opts.traits.map(t => ` - ${t}`).join('\n')
5
+ : ' - (edit in persona file)';
6
+ const habitsBlock = opts.habits.length > 0
7
+ ? opts.habits.map(h => ` - ${h}`).join('\n')
8
+ : ' - (edit in persona file)';
9
+ const scopeBlock = opts.scope.length > 0
10
+ ? opts.scope.map(s => `- ${s}`).join('\n')
11
+ : '- (edit in persona file)';
12
+ const notScopeBlock = opts.notScope.length > 0
13
+ ? opts.notScope.map(s => `- ${s}`).join('\n')
14
+ : '- (edit in persona file)';
15
+ const expertiseBlock = opts.expertise.length > 0
16
+ ? `\n## Expertise\n${opts.expertise.map(e => `- ${e}`).join('\n')}\n`
17
+ : '';
18
+ const customRules = opts.rules.length > 0
19
+ ? opts.rules.map(r => `- ${r}`).join('\n')
20
+ : '- (edit in persona file)';
21
+ const leaderExtra = opts.isLeader
22
+ ? '\n- Respond to ALL channel messages (not just @mentions)\n- Never work directly — only delegate and report'
23
+ : '';
24
+ return `# ${opts.name}
25
+
26
+ You are **${opts.name}**, an AI assistant on Discord.
27
+ When addressing the human, always call them **${humanTitle}**.
28
+
29
+ ## Character
30
+ - **MBTI:** ${opts.mbti}
31
+ - **Speech style:**
32
+ ${opts.speechStyle}
33
+ - **Personality:**
34
+ ${traitsBlock}
35
+ - **Habits:**
36
+ ${habitsBlock}
37
+
38
+ ## Role
39
+ ${opts.role}
40
+
41
+ **Scope:**
42
+ ${scopeBlock}
43
+
44
+ **Not in scope:**
45
+ ${notScopeBlock}
46
+
47
+ **Decision authority:**
48
+ - Independent: ${opts.authorityIndependent || '(edit in persona file)'}
49
+ - Needs approval: ${opts.authorityNeedsApproval || '(edit in persona file)'}
50
+ ${expertiseBlock}
51
+ ## Rules${leaderExtra}
52
+ ${customRules}
53
+ `;
54
+ }
55
+ //# sourceMappingURL=prompt-template.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompt-template.js","sourceRoot":"","sources":["../../src/cli/prompt-template.ts"],"names":[],"mappings":"AAiBA,MAAM,UAAU,cAAc,CAAC,IAAmB;IAChD,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,IAAI,MAAM,CAAC;IAE7C,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC;QACxC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAC7C,CAAC,CAAC,4BAA4B,CAAC;IAEjC,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC;QACxC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAC7C,CAAC,CAAC,4BAA4B,CAAC;IAEjC,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC;QACtC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAC1C,CAAC,CAAC,0BAA0B,CAAC;IAE/B,MAAM,aAAa,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC;QAC5C,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAC7C,CAAC,CAAC,0BAA0B,CAAC;IAE/B,MAAM,cAAc,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC;QAC9C,CAAC,CAAC,mBAAmB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI;QACrE,CAAC,CAAC,EAAE,CAAC;IAEP,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC;QACvC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAC1C,CAAC,CAAC,0BAA0B,CAAC;IAE/B,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ;QAC/B,CAAC,CAAC,4GAA4G;QAC9G,CAAC,CAAC,EAAE,CAAC;IAEP,OAAO,KAAK,IAAI,CAAC,IAAI;;YAEX,IAAI,CAAC,IAAI;gDAC2B,UAAU;;;cAG5C,IAAI,CAAC,IAAI;;EAErB,IAAI,CAAC,WAAW;;EAEhB,WAAW;;EAEX,WAAW;;;EAGX,IAAI,CAAC,IAAI;;;EAGT,UAAU;;;EAGV,aAAa;;;iBAGE,IAAI,CAAC,oBAAoB,IAAI,wBAAwB;oBAClD,IAAI,CAAC,sBAAsB,IAAI,wBAAwB;EACzE,cAAc;UACN,WAAW;EACnB,WAAW;CACZ,CAAC;AACF,CAAC"}
@@ -0,0 +1,5 @@
1
+ export declare function closeRL(): void;
2
+ export declare function ask(question: string, defaultValue?: string): Promise<string>;
3
+ export declare function confirm(question: string, defaultYes?: boolean): Promise<boolean>;
4
+ export declare function choose(question: string, options: string[], defaultIndex?: number): Promise<string>;
5
+ //# sourceMappingURL=readline-utils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"readline-utils.d.ts","sourceRoot":"","sources":["../../src/cli/readline-utils.ts"],"names":[],"mappings":"AAYA,wBAAgB,OAAO,IAAI,IAAI,CAG9B;AAED,wBAAsB,GAAG,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAIlF;AAED,wBAAsB,OAAO,CAAC,QAAQ,EAAE,MAAM,EAAE,UAAU,UAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,CAMpF;AAED,wBAAsB,MAAM,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,EAAE,YAAY,SAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAUnG"}
@@ -0,0 +1,39 @@
1
+ import readline from 'node:readline/promises';
2
+ import { stdin, stdout } from 'node:process';
3
+ let rl = null;
4
+ function getRL() {
5
+ if (!rl) {
6
+ rl = readline.createInterface({ input: stdin, output: stdout });
7
+ }
8
+ return rl;
9
+ }
10
+ export function closeRL() {
11
+ rl?.close();
12
+ rl = null;
13
+ }
14
+ export async function ask(question, defaultValue) {
15
+ const suffix = defaultValue ? ` (${defaultValue})` : '';
16
+ const answer = await getRL().question(`${question}${suffix}: `);
17
+ return answer.trim() || defaultValue || '';
18
+ }
19
+ export async function confirm(question, defaultYes = false) {
20
+ const hint = defaultYes ? 'Y/n' : 'y/N';
21
+ const answer = await getRL().question(`${question} [${hint}]: `);
22
+ const val = answer.trim().toLowerCase();
23
+ if (!val)
24
+ return defaultYes;
25
+ return val === 'y' || val === 'yes';
26
+ }
27
+ export async function choose(question, options, defaultIndex = 0) {
28
+ console.log(question);
29
+ for (let i = 0; i < options.length; i++) {
30
+ const marker = i === defaultIndex ? '>' : ' ';
31
+ console.log(` ${marker} ${i + 1}. ${options[i]}`);
32
+ }
33
+ const answer = await getRL().question(`Choice (${defaultIndex + 1}): `);
34
+ const idx = parseInt(answer.trim()) - 1;
35
+ if (idx >= 0 && idx < options.length)
36
+ return options[idx];
37
+ return options[defaultIndex];
38
+ }
39
+ //# sourceMappingURL=readline-utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"readline-utils.js","sourceRoot":"","sources":["../../src/cli/readline-utils.ts"],"names":[],"mappings":"AAAA,OAAO,QAAQ,MAAM,wBAAwB,CAAC;AAC9C,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAE7C,IAAI,EAAE,GAA8B,IAAI,CAAC;AAEzC,SAAS,KAAK;IACZ,IAAI,CAAC,EAAE,EAAE,CAAC;QACR,EAAE,GAAG,QAAQ,CAAC,eAAe,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IAClE,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAED,MAAM,UAAU,OAAO;IACrB,EAAE,EAAE,KAAK,EAAE,CAAC;IACZ,EAAE,GAAG,IAAI,CAAC;AACZ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,GAAG,CAAC,QAAgB,EAAE,YAAqB;IAC/D,MAAM,MAAM,GAAG,YAAY,CAAC,CAAC,CAAC,KAAK,YAAY,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;IACxD,MAAM,MAAM,GAAG,MAAM,KAAK,EAAE,CAAC,QAAQ,CAAC,GAAG,QAAQ,GAAG,MAAM,IAAI,CAAC,CAAC;IAChE,OAAO,MAAM,CAAC,IAAI,EAAE,IAAI,YAAY,IAAI,EAAE,CAAC;AAC7C,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,QAAgB,EAAE,UAAU,GAAG,KAAK;IAChE,MAAM,IAAI,GAAG,UAAU,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC;IACxC,MAAM,MAAM,GAAG,MAAM,KAAK,EAAE,CAAC,QAAQ,CAAC,GAAG,QAAQ,KAAK,IAAI,KAAK,CAAC,CAAC;IACjE,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;IACxC,IAAI,CAAC,GAAG;QAAE,OAAO,UAAU,CAAC;IAC5B,OAAO,GAAG,KAAK,GAAG,IAAI,GAAG,KAAK,KAAK,CAAC;AACtC,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,MAAM,CAAC,QAAgB,EAAE,OAAiB,EAAE,YAAY,GAAG,CAAC;IAChF,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACtB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACxC,MAAM,MAAM,GAAG,CAAC,KAAK,YAAY,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;QAC9C,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IACrD,CAAC;IACD,MAAM,MAAM,GAAG,MAAM,KAAK,EAAE,CAAC,QAAQ,CAAC,WAAW,YAAY,GAAG,CAAC,KAAK,CAAC,CAAC;IACxE,MAAM,GAAG,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,GAAG,CAAC,CAAC;IACxC,IAAI,GAAG,IAAI,CAAC,IAAI,GAAG,GAAG,OAAO,CAAC,MAAM;QAAE,OAAO,OAAO,CAAC,GAAG,CAAC,CAAC;IAC1D,OAAO,OAAO,CAAC,YAAY,CAAC,CAAC;AAC/B,CAAC"}
@@ -0,0 +1,7 @@
1
+ /** Walk up from startDir looking for a directory containing teams.json */
2
+ export declare function findWorkspace(startDir?: string): string | null;
3
+ /** Require a workspace or exit with an error message */
4
+ export declare function requireWorkspace(startDir?: string): string;
5
+ /** Check that a workspace has the expected structure */
6
+ export declare function validateWorkspace(workspacePath: string): string[];
7
+ //# sourceMappingURL=workspace.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workspace.d.ts","sourceRoot":"","sources":["../../src/cli/workspace.ts"],"names":[],"mappings":"AAGA,0EAA0E;AAC1E,wBAAgB,aAAa,CAAC,QAAQ,GAAE,MAAsB,GAAG,MAAM,GAAG,IAAI,CAW7E;AAED,wDAAwD;AACxD,wBAAgB,gBAAgB,CAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,CAO1D;AAED,wDAAwD;AACxD,wBAAgB,iBAAiB,CAAC,aAAa,EAAE,MAAM,GAAG,MAAM,EAAE,CAajE"}
@@ -0,0 +1,39 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ /** Walk up from startDir looking for a directory containing teams.json */
4
+ export function findWorkspace(startDir = process.cwd()) {
5
+ let dir = path.resolve(startDir);
6
+ const root = path.parse(dir).root;
7
+ while (dir !== root) {
8
+ if (fs.existsSync(path.join(dir, 'teams.json'))) {
9
+ return dir;
10
+ }
11
+ dir = path.dirname(dir);
12
+ }
13
+ return null;
14
+ }
15
+ /** Require a workspace or exit with an error message */
16
+ export function requireWorkspace(startDir) {
17
+ const ws = findWorkspace(startDir);
18
+ if (!ws) {
19
+ console.error('No mococo workspace found. Run `mococo init` first.');
20
+ process.exit(1);
21
+ }
22
+ return ws;
23
+ }
24
+ /** Check that a workspace has the expected structure */
25
+ export function validateWorkspace(workspacePath) {
26
+ const issues = [];
27
+ const check = (rel) => {
28
+ if (!fs.existsSync(path.join(workspacePath, rel))) {
29
+ issues.push(`Missing: ${rel}`);
30
+ }
31
+ };
32
+ check('teams.json');
33
+ check('.env');
34
+ check('prompts');
35
+ check('repos');
36
+ check('hooks');
37
+ return issues;
38
+ }
39
+ //# sourceMappingURL=workspace.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workspace.js","sourceRoot":"","sources":["../../src/cli/workspace.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAE7B,0EAA0E;AAC1E,MAAM,UAAU,aAAa,CAAC,WAAmB,OAAO,CAAC,GAAG,EAAE;IAC5D,IAAI,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACjC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC;IAElC,OAAO,GAAG,KAAK,IAAI,EAAE,CAAC;QACpB,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC,EAAE,CAAC;YAChD,OAAO,GAAG,CAAC;QACb,CAAC;QACD,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAC1B,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,wDAAwD;AACxD,MAAM,UAAU,gBAAgB,CAAC,QAAiB;IAChD,MAAM,EAAE,GAAG,aAAa,CAAC,QAAQ,CAAC,CAAC;IACnC,IAAI,CAAC,EAAE,EAAE,CAAC;QACR,OAAO,CAAC,KAAK,CAAC,qDAAqD,CAAC,CAAC;QACrE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAED,wDAAwD;AACxD,MAAM,UAAU,iBAAiB,CAAC,aAAqB;IACrD,MAAM,MAAM,GAAa,EAAE,CAAC;IAC5B,MAAM,KAAK,GAAG,CAAC,GAAW,EAAE,EAAE;QAC5B,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC;YAClD,MAAM,CAAC,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC,CAAC;QACjC,CAAC;IACH,CAAC,CAAC;IACF,KAAK,CAAC,YAAY,CAAC,CAAC;IACpB,KAAK,CAAC,MAAM,CAAC,CAAC;IACd,KAAK,CAAC,SAAS,CAAC,CAAC;IACjB,KAAK,CAAC,OAAO,CAAC,CAAC;IACf,KAAK,CAAC,OAAO,CAAC,CAAC;IACf,OAAO,MAAM,CAAC;AAChB,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { TeamsConfig, TeamConfig } from './types.js';
2
+ export declare function loadTeamsConfig(workspacePath?: string): TeamsConfig;
3
+ export declare function getLeaderTeam(config: TeamsConfig): TeamConfig | undefined;
4
+ //# sourceMappingURL=config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,WAAW,EAAE,UAAU,EAAmB,MAAM,YAAY,CAAC;AAa3E,wBAAgB,eAAe,CAAC,aAAa,GAAE,MAAsB,GAAG,WAAW,CAqFlF;AAED,wBAAgB,aAAa,CAAC,MAAM,EAAE,WAAW,GAAG,UAAU,GAAG,SAAS,CAEzE"}
package/dist/config.js ADDED
@@ -0,0 +1,96 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ const AVATAR_MAP = {
4
+ crown: 'https://em-content.zobj.net/source/apple/391/crown_1f451.png',
5
+ brain: 'https://em-content.zobj.net/source/apple/391/brain_1f9e0.png',
6
+ gear: 'https://em-content.zobj.net/source/apple/391/gear_2699-fe0f.png',
7
+ palette: 'https://em-content.zobj.net/source/apple/391/artist-palette_1f3a8.png',
8
+ shield: 'https://em-content.zobj.net/source/apple/391/shield_1f6e1-fe0f.png',
9
+ eye: 'https://em-content.zobj.net/source/apple/391/eye_1f441-fe0f.png',
10
+ robot: 'https://em-content.zobj.net/source/apple/391/robot_1f916.png',
11
+ test: 'https://em-content.zobj.net/source/apple/391/test-tube_1f9ea.png',
12
+ };
13
+ export function loadTeamsConfig(workspacePath = process.cwd()) {
14
+ const teamsJsonPath = path.resolve(workspacePath, 'teams.json');
15
+ let raw;
16
+ try {
17
+ raw = JSON.parse(fs.readFileSync(teamsJsonPath, 'utf-8'));
18
+ }
19
+ catch (err) {
20
+ if (err instanceof Error && 'code' in err && err.code === 'ENOENT') {
21
+ throw new Error(`[config] teams.json not found at ${teamsJsonPath}. Run "mococo init" first to create a workspace.`);
22
+ }
23
+ throw err;
24
+ }
25
+ const teams = {};
26
+ for (const [id, cfg] of Object.entries(raw.teams)) {
27
+ // Resolve Discord token: read env var name from config
28
+ const discordTokenEnv = cfg.discordTokenEnv ?? `${id.toUpperCase()}_DISCORD_TOKEN`;
29
+ const discordToken = process.env[discordTokenEnv] ?? '';
30
+ if (!discordToken) {
31
+ console.warn(`[config] Missing env var ${discordTokenEnv} for team "${id}" — bot will not start`);
32
+ }
33
+ // Resolve MCP server configs: expand $VAR references in env values
34
+ let mcpServers;
35
+ if (cfg.mcpServers) {
36
+ mcpServers = {};
37
+ for (const [name, server] of Object.entries(cfg.mcpServers)) {
38
+ const resolvedEnv = {};
39
+ if (server.env) {
40
+ for (const [key, val] of Object.entries(server.env)) {
41
+ if (val.startsWith('$')) {
42
+ const envName = val.slice(1);
43
+ const envValue = process.env[envName];
44
+ if (envValue === undefined) {
45
+ console.warn(`[config] MCP server "${name}" for team "${id}": env var ${envName} is not set (key: ${key})`);
46
+ }
47
+ resolvedEnv[key] = envValue ?? '';
48
+ }
49
+ else {
50
+ resolvedEnv[key] = val;
51
+ }
52
+ }
53
+ }
54
+ mcpServers[name] = {
55
+ command: server.command,
56
+ args: server.args,
57
+ env: Object.keys(resolvedEnv).length > 0 ? resolvedEnv : undefined,
58
+ };
59
+ }
60
+ }
61
+ teams[id] = {
62
+ id,
63
+ name: cfg.name,
64
+ color: cfg.color ? parseInt(cfg.color.replace('#', ''), 16) : 0x808080,
65
+ avatar: AVATAR_MAP[cfg.avatar] ?? cfg.avatar,
66
+ engine: cfg.engine ?? 'claude',
67
+ model: cfg.model ?? 'sonnet',
68
+ maxBudget: cfg.maxBudget ?? 10,
69
+ prompt: cfg.prompt,
70
+ isLeader: cfg.isLeader ?? false,
71
+ channels: cfg.channels,
72
+ discordUserId: cfg.discordUserId,
73
+ useTeams: cfg.useTeams ?? false,
74
+ teamRules: cfg.teamRules,
75
+ git: cfg.git ?? {
76
+ name: `${cfg.name} (mococo)`,
77
+ email: `${id}@users.noreply.github.com`,
78
+ },
79
+ discordToken,
80
+ mcpServers,
81
+ permissions: cfg.permissions ?? {},
82
+ };
83
+ }
84
+ return {
85
+ teams,
86
+ globalDeny: raw.globalDeny ?? [],
87
+ conversationWindow: raw.conversationWindow ?? 30,
88
+ workspacePath,
89
+ humanDiscordId: raw.humanDiscordId,
90
+ humanTitle: raw.humanTitle ?? 'Boss',
91
+ };
92
+ }
93
+ export function getLeaderTeam(config) {
94
+ return Object.values(config.teams).find(t => t.isLeader);
95
+ }
96
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAG7B,MAAM,UAAU,GAA2B;IACzC,KAAK,EAAE,8DAA8D;IACrE,KAAK,EAAE,8DAA8D;IACrE,IAAI,EAAE,iEAAiE;IACvE,OAAO,EAAE,uEAAuE;IAChF,MAAM,EAAE,oEAAoE;IAC5E,GAAG,EAAE,iEAAiE;IACtE,KAAK,EAAE,8DAA8D;IACrE,IAAI,EAAE,kEAAkE;CACzE,CAAC;AAEF,MAAM,UAAU,eAAe,CAAC,gBAAwB,OAAO,CAAC,GAAG,EAAE;IACnE,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC;IAEhE,IAAI,GAAQ,CAAC;IACb,IAAI,CAAC;QACH,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC,CAAC;IAC5D,CAAC;IAAC,OAAO,GAAY,EAAE,CAAC;QACtB,IAAI,GAAG,YAAY,KAAK,IAAI,MAAM,IAAI,GAAG,IAAK,GAA6B,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC9F,MAAM,IAAI,KAAK,CACb,oCAAoC,aAAa,kDAAkD,CACpG,CAAC;QACJ,CAAC;QACD,MAAM,GAAG,CAAC;IACZ,CAAC;IAED,MAAM,KAAK,GAA+B,EAAE,CAAC;IAE7C,KAAK,MAAM,CAAC,EAAE,EAAE,GAAG,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,KAA4B,CAAC,EAAE,CAAC;QACzE,uDAAuD;QACvD,MAAM,eAAe,GAAG,GAAG,CAAC,eAAe,IAAI,GAAG,EAAE,CAAC,WAAW,EAAE,gBAAgB,CAAC;QACnF,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC;QACxD,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,OAAO,CAAC,IAAI,CAAC,4BAA4B,eAAe,cAAc,EAAE,wBAAwB,CAAC,CAAC;QACpG,CAAC;QAED,mEAAmE;QACnE,IAAI,UAAuD,CAAC;QAC5D,IAAI,GAAG,CAAC,UAAU,EAAE,CAAC;YACnB,UAAU,GAAG,EAAE,CAAC;YAChB,KAAK,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,UAAiC,CAAC,EAAE,CAAC;gBACnF,MAAM,WAAW,GAA2B,EAAE,CAAC;gBAC/C,IAAI,MAAM,CAAC,GAAG,EAAE,CAAC;oBACf,KAAK,MAAM,CAAC,GAAG,EAAE,GAAG,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,GAA6B,CAAC,EAAE,CAAC;wBAC9E,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;4BACxB,MAAM,OAAO,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;4BAC7B,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;4BACtC,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;gCAC3B,OAAO,CAAC,IAAI,CAAC,wBAAwB,IAAI,eAAe,EAAE,cAAc,OAAO,qBAAqB,GAAG,GAAG,CAAC,CAAC;4BAC9G,CAAC;4BACD,WAAW,CAAC,GAAG,CAAC,GAAG,QAAQ,IAAI,EAAE,CAAC;wBACpC,CAAC;6BAAM,CAAC;4BACN,WAAW,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;wBACzB,CAAC;oBACH,CAAC;gBACH,CAAC;gBACD,UAAU,CAAC,IAAI,CAAC,GAAG;oBACjB,OAAO,EAAE,MAAM,CAAC,OAAO;oBACvB,IAAI,EAAE,MAAM,CAAC,IAAI;oBACjB,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS;iBACnE,CAAC;YACJ,CAAC;QACH,CAAC;QAED,KAAK,CAAC,EAAE,CAAC,GAAG;YACV,EAAE;YACF,IAAI,EAAE,GAAG,CAAC,IAAI;YACd,KAAK,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,QAAQ;YACtE,MAAM,EAAE,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,MAAM;YAC5C,MAAM,EAAE,GAAG,CAAC,MAAM,IAAI,QAAQ;YAC9B,KAAK,EAAE,GAAG,CAAC,KAAK,IAAI,QAAQ;YAC5B,SAAS,EAAE,GAAG,CAAC,SAAS,IAAI,EAAE;YAC9B,MAAM,EAAE,GAAG,CAAC,MAAM;YAClB,QAAQ,EAAE,GAAG,CAAC,QAAQ,IAAI,KAAK;YAC/B,QAAQ,EAAE,GAAG,CAAC,QAAQ;YACtB,aAAa,EAAE,GAAG,CAAC,aAAa;YAChC,QAAQ,EAAE,GAAG,CAAC,QAAQ,IAAI,KAAK;YAC/B,SAAS,EAAE,GAAG,CAAC,SAAS;YACxB,GAAG,EAAE,GAAG,CAAC,GAAG,IAAI;gBACd,IAAI,EAAE,GAAG,GAAG,CAAC,IAAI,WAAW;gBAC5B,KAAK,EAAE,GAAG,EAAE,2BAA2B;aACxC;YACD,YAAY;YACZ,UAAU;YACV,WAAW,EAAE,GAAG,CAAC,WAAW,IAAI,EAAE;SACnC,CAAC;IACJ,CAAC;IAED,OAAO;QACL,KAAK;QACL,UAAU,EAAE,GAAG,CAAC,UAAU,IAAI,EAAE;QAChC,kBAAkB,EAAE,GAAG,CAAC,kBAAkB,IAAI,EAAE;QAChD,aAAa;QACb,cAAc,EAAE,GAAG,CAAC,cAAc;QAClC,UAAU,EAAE,GAAG,CAAC,UAAU,IAAI,MAAM;KACrC,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,MAAmB;IAC/C,OAAO,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;AAC3D,CAAC"}
@@ -0,0 +1,2 @@
1
+ import 'dotenv/config';
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,eAAe,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,34 @@
1
+ import 'dotenv/config';
2
+ import { loadTeamsConfig } from './config.js';
3
+ import { createBots } from './bot/client.js';
4
+ import { startHookServer } from './server/hook-receiver.js';
5
+ async function main() {
6
+ const config = loadTeamsConfig();
7
+ const env = {
8
+ workChannelId: process.env.WORK_CHANNEL_ID || undefined,
9
+ hookPort: parseInt(process.env.HOOK_PORT ?? '9876'),
10
+ memberTrackingChannelId: process.env.MEMBER_TRACKING_CHANNEL_ID || undefined,
11
+ decisionLogChannelId: process.env.DECISION_LOG_CHANNEL_ID || undefined,
12
+ };
13
+ if (!env.workChannelId) {
14
+ console.log('WORK_CHANNEL_ID not set — bot will respond in all channels.');
15
+ }
16
+ // Check how many teams have Discord tokens configured
17
+ const teamsWithTokens = Object.values(config.teams).filter(t => t.discordToken);
18
+ if (teamsWithTokens.length === 0) {
19
+ console.error('No team has a Discord token configured.');
20
+ console.error('Set <TEAM_ID>_DISCORD_TOKEN in .env for each team (e.g., LEADER_DISCORD_TOKEN=xxx).');
21
+ process.exit(1);
22
+ }
23
+ startHookServer(env.hookPort);
24
+ console.log(`Starting ${teamsWithTokens.length} team bots...`);
25
+ await createBots(config, env);
26
+ const teamCount = Object.keys(config.teams).length;
27
+ const engines = [...new Set(Object.values(config.teams).map(t => t.engine))];
28
+ console.log(`claude-mococo running — ${teamsWithTokens.length}/${teamCount} teams online (engines: ${engines.join(', ')})`);
29
+ }
30
+ main().catch((err) => {
31
+ console.error('Failed to start:', err);
32
+ process.exit(1);
33
+ });
34
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,eAAe,CAAC;AACvB,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAC;AAG5D,KAAK,UAAU,IAAI;IACjB,MAAM,MAAM,GAAG,eAAe,EAAE,CAAC;IACjC,MAAM,GAAG,GAAc;QACrB,aAAa,EAAE,OAAO,CAAC,GAAG,CAAC,eAAe,IAAI,SAAS;QACvD,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,IAAI,MAAM,CAAC;QACnD,uBAAuB,EAAE,OAAO,CAAC,GAAG,CAAC,0BAA0B,IAAI,SAAS;QAC5E,oBAAoB,EAAE,OAAO,CAAC,GAAG,CAAC,uBAAuB,IAAI,SAAS;KACvE,CAAC;IAEF,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;QACvB,OAAO,CAAC,GAAG,CAAC,6DAA6D,CAAC,CAAC;IAC7E,CAAC;IAED,sDAAsD;IACtD,MAAM,eAAe,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC;IAChF,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjC,OAAO,CAAC,KAAK,CAAC,yCAAyC,CAAC,CAAC;QACzD,OAAO,CAAC,KAAK,CAAC,qFAAqF,CAAC,CAAC;QACrG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAE9B,OAAO,CAAC,GAAG,CAAC,YAAY,eAAe,CAAC,MAAM,eAAe,CAAC,CAAC;IAC/D,MAAM,UAAU,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAE9B,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC;IACnD,MAAM,OAAO,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IAC7E,OAAO,CAAC,GAAG,CAAC,2BAA2B,eAAe,CAAC,MAAM,IAAI,SAAS,2BAA2B,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAC9H,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACnB,OAAO,CAAC,KAAK,CAAC,kBAAkB,EAAE,GAAG,CAAC,CAAC;IACvC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
@@ -0,0 +1,8 @@
1
+ import { BaseEngine } from './engine-base.js';
2
+ export declare class ClaudeEngine extends BaseEngine {
3
+ private proc;
4
+ private mcpConfigPath;
5
+ start(): Promise<void>;
6
+ kill(): void;
7
+ }
8
+ //# sourceMappingURL=claude-engine.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claude-engine.d.ts","sourceRoot":"","sources":["../../src/orchestrator/claude-engine.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAG9C,qBAAa,YAAa,SAAQ,UAAU;IAC1C,OAAO,CAAC,IAAI,CAA6B;IACzC,OAAO,CAAC,aAAa,CAAuB;IAEtC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAsF5B,IAAI;CAGL"}
@@ -0,0 +1,94 @@
1
+ import { spawn } from 'node:child_process';
2
+ import readline from 'node:readline';
3
+ import { BaseEngine } from './engine-base.js';
4
+ import { writeMcpConfig, cleanupMcpConfig } from './mcp-config.js';
5
+ export class ClaudeEngine extends BaseEngine {
6
+ proc = null;
7
+ mcpConfigPath = null;
8
+ async start() {
9
+ const args = [
10
+ '-p', this.opts.prompt,
11
+ '--output-format', 'stream-json',
12
+ '--verbose',
13
+ '--model', this.opts.model,
14
+ '--dangerously-skip-permissions',
15
+ '--max-budget-usd', String(this.opts.maxBudget),
16
+ ];
17
+ if (this.opts.mcpServers && Object.keys(this.opts.mcpServers).length > 0) {
18
+ try {
19
+ this.mcpConfigPath = writeMcpConfig(this.opts.teamId, this.opts.mcpServers, this.opts.cwd);
20
+ args.push('--mcp-config', this.mcpConfigPath);
21
+ }
22
+ catch (err) {
23
+ console.error(`[claude:${this.opts.teamId}] Failed to write MCP config, proceeding without MCP:`, err);
24
+ }
25
+ }
26
+ this.proc = spawn('claude', args, {
27
+ cwd: this.opts.cwd,
28
+ env: {
29
+ ...this.getTeamEnv(),
30
+ CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS: '1',
31
+ },
32
+ stdio: ['ignore', 'pipe', 'pipe'],
33
+ });
34
+ // Track both readline close and process exit to avoid race conditions.
35
+ // The 'result' event may arrive in the last stdout line — readline must
36
+ // finish processing before we emit 'exit', otherwise the invoker rejects
37
+ // the promise before the result is received.
38
+ let exitCode = null;
39
+ let rlClosed = false;
40
+ let procExited = false;
41
+ let exitEmitted = false;
42
+ const maybeEmitExit = () => {
43
+ if (exitEmitted)
44
+ return;
45
+ if (rlClosed && procExited) {
46
+ exitEmitted = true;
47
+ if (this.mcpConfigPath)
48
+ cleanupMcpConfig(this.opts.teamId, this.opts.cwd);
49
+ this.emit('exit', exitCode);
50
+ }
51
+ };
52
+ const rl = readline.createInterface({ input: this.proc.stdout });
53
+ rl.on('line', (line) => {
54
+ try {
55
+ const event = JSON.parse(line);
56
+ this.emit('message', event);
57
+ if (event.type === 'result')
58
+ this.emit('result', event);
59
+ }
60
+ catch {
61
+ // non-JSON line, log it
62
+ console.log(`[claude:${this.opts.teamId}] stdout: ${line.slice(0, 200)}`);
63
+ }
64
+ });
65
+ rl.on('close', () => {
66
+ rlClosed = true;
67
+ maybeEmitExit();
68
+ });
69
+ // Log stderr so we can see Claude errors
70
+ const stderrRl = readline.createInterface({ input: this.proc.stderr });
71
+ stderrRl.on('line', (line) => {
72
+ console.error(`[claude:${this.opts.teamId}] stderr: ${line.slice(0, 300)}`);
73
+ });
74
+ this.proc.on('error', (err) => {
75
+ console.error(`[claude:${this.opts.teamId}] spawn error: ${err.message}`);
76
+ if (!procExited) {
77
+ exitCode = 1;
78
+ procExited = true;
79
+ rlClosed = true;
80
+ maybeEmitExit();
81
+ }
82
+ });
83
+ this.proc.on('exit', (code) => {
84
+ console.log(`[claude:${this.opts.teamId}] exited with code ${code}`);
85
+ exitCode = code;
86
+ procExited = true;
87
+ maybeEmitExit();
88
+ });
89
+ }
90
+ kill() {
91
+ this.proc?.kill('SIGTERM');
92
+ }
93
+ }
94
+ //# sourceMappingURL=claude-engine.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claude-engine.js","sourceRoot":"","sources":["../../src/orchestrator/claude-engine.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import { BaseEngine } from './engine-base.js';
2
+ export declare class CodexEngine extends BaseEngine {
3
+ private proc;
4
+ private timeoutTimer;
5
+ private killTimer;
6
+ start(): Promise<void>;
7
+ kill(): void;
8
+ private clearTimers;
9
+ }
10
+ //# sourceMappingURL=codex-engine.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"codex-engine.d.ts","sourceRoot":"","sources":["../../src/orchestrator/codex-engine.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAK9C,qBAAa,WAAY,SAAQ,UAAU;IACzC,OAAO,CAAC,IAAI,CAA6B;IACzC,OAAO,CAAC,YAAY,CAA8C;IAClE,OAAO,CAAC,SAAS,CAA8C;IAEzD,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IA+D5B,IAAI;IAKJ,OAAO,CAAC,WAAW;CAIpB"}
@@ -0,0 +1,82 @@
1
+ import { spawn } from 'node:child_process';
2
+ import readline from 'node:readline';
3
+ import { BaseEngine } from './engine-base.js';
4
+ const PROCESS_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes
5
+ const SIGKILL_GRACE_MS = 10 * 1000; // 10 seconds after SIGTERM
6
+ export class CodexEngine extends BaseEngine {
7
+ proc = null;
8
+ timeoutTimer = null;
9
+ killTimer = null;
10
+ async start() {
11
+ console.log(`[codex:${this.opts.teamId}] Starting codex (model: ${this.opts.model})`);
12
+ this.proc = spawn('codex', [
13
+ 'exec',
14
+ '-c', `model="${this.opts.model}"`,
15
+ '--json',
16
+ '--skip-git-repo-check',
17
+ this.opts.prompt,
18
+ ], {
19
+ cwd: this.opts.cwd,
20
+ env: this.getTeamEnv(),
21
+ stdio: ['ignore', 'pipe', 'pipe'],
22
+ });
23
+ // Process-level timeout: SIGTERM → SIGKILL escalation
24
+ this.timeoutTimer = setTimeout(() => {
25
+ if (this.proc && !this.proc.killed) {
26
+ console.warn(`[codex:${this.opts.teamId}] process timeout (${PROCESS_TIMEOUT_MS / 1000}s) — sending SIGTERM`);
27
+ this.proc.kill('SIGTERM');
28
+ this.killTimer = setTimeout(() => {
29
+ if (this.proc && !this.proc.killed) {
30
+ console.warn(`[codex:${this.opts.teamId}] SIGTERM grace period expired — sending SIGKILL`);
31
+ this.proc.kill('SIGKILL');
32
+ }
33
+ }, SIGKILL_GRACE_MS);
34
+ }
35
+ }, PROCESS_TIMEOUT_MS);
36
+ const messages = [];
37
+ const stdoutRl = readline.createInterface({ input: this.proc.stdout });
38
+ stdoutRl.on('line', (line) => {
39
+ try {
40
+ const event = JSON.parse(line);
41
+ // Codex outputs agent_message items with the response text
42
+ if (event.type === 'item.completed' && event.item?.type === 'agent_message' && event.item.text) {
43
+ messages.push(event.item.text);
44
+ }
45
+ }
46
+ catch {
47
+ // non-JSON line
48
+ }
49
+ });
50
+ // Log stderr
51
+ const stderrRl = readline.createInterface({ input: this.proc.stderr });
52
+ stderrRl.on('line', (line) => {
53
+ console.error(`[codex:${this.opts.teamId}] stderr: ${line.slice(0, 300)}`);
54
+ });
55
+ this.proc.on('error', (err) => {
56
+ this.clearTimers();
57
+ console.error(`[codex:${this.opts.teamId}] spawn error: ${err.message}`);
58
+ });
59
+ this.proc.on('exit', (code) => {
60
+ this.clearTimers();
61
+ const output = messages.join('\n').trim();
62
+ console.log(`[codex:${this.opts.teamId}] exited with code ${code} (output: ${output.length} chars)`);
63
+ this.emit('result', { type: 'result', result: output, total_cost_usd: 0 });
64
+ this.emit('exit', code);
65
+ });
66
+ }
67
+ kill() {
68
+ this.clearTimers();
69
+ this.proc?.kill('SIGTERM');
70
+ }
71
+ clearTimers() {
72
+ if (this.timeoutTimer) {
73
+ clearTimeout(this.timeoutTimer);
74
+ this.timeoutTimer = null;
75
+ }
76
+ if (this.killTimer) {
77
+ clearTimeout(this.killTimer);
78
+ this.killTimer = null;
79
+ }
80
+ }
81
+ }
82
+ //# sourceMappingURL=codex-engine.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"codex-engine.js","sourceRoot":"","sources":["../../src/orchestrator/codex-engine.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ import { EventEmitter } from 'node:events';
2
+ import type { McpServerConfig } from '../types.js';
3
+ export interface EngineOptions {
4
+ prompt: string;
5
+ cwd: string;
6
+ model: string;
7
+ maxBudget: number;
8
+ teamId: string;
9
+ gitName: string;
10
+ gitEmail: string;
11
+ mcpServers?: Record<string, McpServerConfig>;
12
+ }
13
+ export declare abstract class BaseEngine extends EventEmitter {
14
+ protected opts: EngineOptions;
15
+ constructor(opts: EngineOptions);
16
+ abstract start(): Promise<void>;
17
+ abstract kill(): void;
18
+ /** Common env vars for all engines: git identity */
19
+ protected getTeamEnv(): Record<string, string>;
20
+ }
21
+ //# sourceMappingURL=engine-base.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"engine-base.d.ts","sourceRoot":"","sources":["../../src/orchestrator/engine-base.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAEnD,MAAM,WAAW,aAAa;IAC5B,MAAM,EAAE,MAAM,CAAC;IACf,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,UAAU,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC;CAC9C;AAED,8BAAsB,UAAW,SAAQ,YAAY;IACvC,SAAS,CAAC,IAAI,EAAE,aAAa;gBAAnB,IAAI,EAAE,aAAa;IAGzC,QAAQ,CAAC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAC/B,QAAQ,CAAC,IAAI,IAAI,IAAI;IAErB,oDAAoD;IACpD,SAAS,CAAC,UAAU,IAAI,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC;CAU/C"}
@@ -0,0 +1,20 @@
1
+ import { EventEmitter } from 'node:events';
2
+ export class BaseEngine extends EventEmitter {
3
+ opts;
4
+ constructor(opts) {
5
+ super();
6
+ this.opts = opts;
7
+ }
8
+ /** Common env vars for all engines: git identity */
9
+ getTeamEnv() {
10
+ return {
11
+ ...process.env,
12
+ MOCOCO_TEAM: this.opts.teamId,
13
+ GIT_AUTHOR_NAME: this.opts.gitName,
14
+ GIT_AUTHOR_EMAIL: this.opts.gitEmail,
15
+ GIT_COMMITTER_NAME: this.opts.gitName,
16
+ GIT_COMMITTER_EMAIL: this.opts.gitEmail,
17
+ };
18
+ }
19
+ }
20
+ //# sourceMappingURL=engine-base.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"engine-base.js","sourceRoot":"","sources":["../../src/orchestrator/engine-base.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAc3C,MAAM,OAAgB,UAAW,SAAQ,YAAY;IAC7B;IAAtB,YAAsB,IAAmB;QACvC,KAAK,EAAE,CAAC;QADY,SAAI,GAAJ,IAAI,CAAe;IAEzC,CAAC;IAID,oDAAoD;IAC1C,UAAU;QAClB,OAAO;YACL,GAAG,OAAO,CAAC,GAA6B;YACxC,WAAW,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM;YAC7B,eAAe,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO;YAClC,gBAAgB,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ;YACpC,kBAAkB,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO;YACrC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ;SACxC,CAAC;IACJ,CAAC;CACF"}