duocode 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 (220) hide show
  1. package/.env.example +36 -0
  2. package/LICENSE +21 -0
  3. package/README.md +52 -0
  4. package/dist/ast/context.d.ts +16 -0
  5. package/dist/ast/context.js +37 -0
  6. package/dist/ast/context.js.map +1 -0
  7. package/dist/ast/diff.d.ts +27 -0
  8. package/dist/ast/diff.js +44 -0
  9. package/dist/ast/diff.js.map +1 -0
  10. package/dist/ast/locks.d.ts +47 -0
  11. package/dist/ast/locks.js +88 -0
  12. package/dist/ast/locks.js.map +1 -0
  13. package/dist/ast/merge.d.ts +22 -0
  14. package/dist/ast/merge.js +120 -0
  15. package/dist/ast/merge.js.map +1 -0
  16. package/dist/ast/ownership.d.ts +31 -0
  17. package/dist/ast/ownership.js +111 -0
  18. package/dist/ast/ownership.js.map +1 -0
  19. package/dist/ast/parser.d.ts +44 -0
  20. package/dist/ast/parser.js +134 -0
  21. package/dist/ast/parser.js.map +1 -0
  22. package/dist/cli.d.ts +2 -0
  23. package/dist/cli.js +423 -0
  24. package/dist/cli.js.map +1 -0
  25. package/dist/commands/doctor.d.ts +5 -0
  26. package/dist/commands/doctor.js +63 -0
  27. package/dist/commands/doctor.js.map +1 -0
  28. package/dist/commands/duo.d.ts +9 -0
  29. package/dist/commands/duo.js +285 -0
  30. package/dist/commands/duo.js.map +1 -0
  31. package/dist/commands/github.d.ts +2 -0
  32. package/dist/commands/github.js +85 -0
  33. package/dist/commands/github.js.map +1 -0
  34. package/dist/commands/init.d.ts +2 -0
  35. package/dist/commands/init.js +33 -0
  36. package/dist/commands/init.js.map +1 -0
  37. package/dist/commands/negotiation.d.ts +2 -0
  38. package/dist/commands/negotiation.js +160 -0
  39. package/dist/commands/negotiation.js.map +1 -0
  40. package/dist/commands/repl_commands.d.ts +26 -0
  41. package/dist/commands/repl_commands.js +226 -0
  42. package/dist/commands/repl_commands.js.map +1 -0
  43. package/dist/commands/shell.d.ts +1 -0
  44. package/dist/commands/shell.js +110 -0
  45. package/dist/commands/shell.js.map +1 -0
  46. package/dist/commands/start.d.ts +2 -0
  47. package/dist/commands/start.js +231 -0
  48. package/dist/commands/start.js.map +1 -0
  49. package/dist/commands/task.d.ts +2 -0
  50. package/dist/commands/task.js +215 -0
  51. package/dist/commands/task.js.map +1 -0
  52. package/dist/config/loader.d.ts +193 -0
  53. package/dist/config/loader.js +106 -0
  54. package/dist/config/loader.js.map +1 -0
  55. package/dist/context/project_context.d.ts +79 -0
  56. package/dist/context/project_context.js +292 -0
  57. package/dist/context/project_context.js.map +1 -0
  58. package/dist/context/token_budget.d.ts +35 -0
  59. package/dist/context/token_budget.js +81 -0
  60. package/dist/context/token_budget.js.map +1 -0
  61. package/dist/db/queries.d.ts +121 -0
  62. package/dist/db/queries.js +109 -0
  63. package/dist/db/queries.js.map +1 -0
  64. package/dist/db/schema.d.ts +110 -0
  65. package/dist/db/schema.js +346 -0
  66. package/dist/db/schema.js.map +1 -0
  67. package/dist/duo/duo_orchestrator.d.ts +50 -0
  68. package/dist/duo/duo_orchestrator.js +510 -0
  69. package/dist/duo/duo_orchestrator.js.map +1 -0
  70. package/dist/duo/duo_session.d.ts +47 -0
  71. package/dist/duo/duo_session.js +127 -0
  72. package/dist/duo/duo_session.js.map +1 -0
  73. package/dist/duo/duo_types.d.ts +168 -0
  74. package/dist/duo/duo_types.js +53 -0
  75. package/dist/duo/duo_types.js.map +1 -0
  76. package/dist/duo/session_store.d.ts +71 -0
  77. package/dist/duo/session_store.js +177 -0
  78. package/dist/duo/session_store.js.map +1 -0
  79. package/dist/git/worktree.d.ts +21 -0
  80. package/dist/git/worktree.js +86 -0
  81. package/dist/git/worktree.js.map +1 -0
  82. package/dist/github/cache.d.ts +23 -0
  83. package/dist/github/cache.js +67 -0
  84. package/dist/github/cache.js.map +1 -0
  85. package/dist/github/issues.d.ts +17 -0
  86. package/dist/github/issues.js +93 -0
  87. package/dist/github/issues.js.map +1 -0
  88. package/dist/github/mcp_client.d.ts +57 -0
  89. package/dist/github/mcp_client.js +214 -0
  90. package/dist/github/mcp_client.js.map +1 -0
  91. package/dist/github/sync.d.ts +11 -0
  92. package/dist/github/sync.js +65 -0
  93. package/dist/github/sync.js.map +1 -0
  94. package/dist/github/webhook.d.ts +25 -0
  95. package/dist/github/webhook.js +197 -0
  96. package/dist/github/webhook.js.map +1 -0
  97. package/dist/negotiation/index.d.ts +1 -0
  98. package/dist/negotiation/index.js +2 -0
  99. package/dist/negotiation/index.js.map +1 -0
  100. package/dist/negotiation/protocol.d.ts +62 -0
  101. package/dist/negotiation/protocol.js +188 -0
  102. package/dist/negotiation/protocol.js.map +1 -0
  103. package/dist/orchestrator/complexity_scorer.d.ts +2 -0
  104. package/dist/orchestrator/complexity_scorer.js +79 -0
  105. package/dist/orchestrator/complexity_scorer.js.map +1 -0
  106. package/dist/orchestrator/dependency_graph.d.ts +7 -0
  107. package/dist/orchestrator/dependency_graph.js +73 -0
  108. package/dist/orchestrator/dependency_graph.js.map +1 -0
  109. package/dist/orchestrator/intent_parser.d.ts +11 -0
  110. package/dist/orchestrator/intent_parser.js +116 -0
  111. package/dist/orchestrator/intent_parser.js.map +1 -0
  112. package/dist/orchestrator/task_runner.d.ts +56 -0
  113. package/dist/orchestrator/task_runner.js +181 -0
  114. package/dist/orchestrator/task_runner.js.map +1 -0
  115. package/dist/orchestrator/types.d.ts +44 -0
  116. package/dist/orchestrator/types.js +21 -0
  117. package/dist/orchestrator/types.js.map +1 -0
  118. package/dist/providers/anthropic.d.ts +12 -0
  119. package/dist/providers/anthropic.js +258 -0
  120. package/dist/providers/anthropic.js.map +1 -0
  121. package/dist/providers/auction.d.ts +42 -0
  122. package/dist/providers/auction.js +190 -0
  123. package/dist/providers/auction.js.map +1 -0
  124. package/dist/providers/base.d.ts +103 -0
  125. package/dist/providers/base.js +2 -0
  126. package/dist/providers/base.js.map +1 -0
  127. package/dist/providers/cost_tracker.d.ts +45 -0
  128. package/dist/providers/cost_tracker.js +111 -0
  129. package/dist/providers/cost_tracker.js.map +1 -0
  130. package/dist/providers/duo_pair_router.d.ts +11 -0
  131. package/dist/providers/duo_pair_router.js +67 -0
  132. package/dist/providers/duo_pair_router.js.map +1 -0
  133. package/dist/providers/factory.d.ts +7 -0
  134. package/dist/providers/factory.js +130 -0
  135. package/dist/providers/factory.js.map +1 -0
  136. package/dist/providers/grading_rubric.d.ts +37 -0
  137. package/dist/providers/grading_rubric.js +238 -0
  138. package/dist/providers/grading_rubric.js.map +1 -0
  139. package/dist/providers/openai.d.ts +12 -0
  140. package/dist/providers/openai.js +229 -0
  141. package/dist/providers/openai.js.map +1 -0
  142. package/dist/providers/openrouter.d.ts +14 -0
  143. package/dist/providers/openrouter.js +178 -0
  144. package/dist/providers/openrouter.js.map +1 -0
  145. package/dist/providers/performance_tracker.d.ts +21 -0
  146. package/dist/providers/performance_tracker.js +63 -0
  147. package/dist/providers/performance_tracker.js.map +1 -0
  148. package/dist/providers/registry_loader.d.ts +6 -0
  149. package/dist/providers/registry_loader.js +54 -0
  150. package/dist/providers/registry_loader.js.map +1 -0
  151. package/dist/providers/retry.d.ts +66 -0
  152. package/dist/providers/retry.js +203 -0
  153. package/dist/providers/retry.js.map +1 -0
  154. package/dist/providers/role_scorer.d.ts +16 -0
  155. package/dist/providers/role_scorer.js +16 -0
  156. package/dist/providers/role_scorer.js.map +1 -0
  157. package/dist/providers/router.d.ts +84 -0
  158. package/dist/providers/router.js +542 -0
  159. package/dist/providers/router.js.map +1 -0
  160. package/dist/security/credentials.d.ts +6 -0
  161. package/dist/security/credentials.js +16 -0
  162. package/dist/security/credentials.js.map +1 -0
  163. package/dist/setup/browser.d.ts +1 -0
  164. package/dist/setup/browser.js +12 -0
  165. package/dist/setup/browser.js.map +1 -0
  166. package/dist/setup/global_config.d.ts +14 -0
  167. package/dist/setup/global_config.js +54 -0
  168. package/dist/setup/global_config.js.map +1 -0
  169. package/dist/setup/wizard.d.ts +2 -0
  170. package/dist/setup/wizard.js +206 -0
  171. package/dist/setup/wizard.js.map +1 -0
  172. package/dist/tools/agent_loop.d.ts +38 -0
  173. package/dist/tools/agent_loop.js +72 -0
  174. package/dist/tools/agent_loop.js.map +1 -0
  175. package/dist/tools/approval.d.ts +64 -0
  176. package/dist/tools/approval.js +172 -0
  177. package/dist/tools/approval.js.map +1 -0
  178. package/dist/tools/checkpoint.d.ts +65 -0
  179. package/dist/tools/checkpoint.js +342 -0
  180. package/dist/tools/checkpoint.js.map +1 -0
  181. package/dist/tools/definitions.d.ts +13 -0
  182. package/dist/tools/definitions.js +103 -0
  183. package/dist/tools/definitions.js.map +1 -0
  184. package/dist/tools/diff_display.d.ts +46 -0
  185. package/dist/tools/diff_display.js +298 -0
  186. package/dist/tools/diff_display.js.map +1 -0
  187. package/dist/tools/executor.d.ts +12 -0
  188. package/dist/tools/executor.js +340 -0
  189. package/dist/tools/executor.js.map +1 -0
  190. package/dist/tools/permissions.d.ts +17 -0
  191. package/dist/tools/permissions.js +139 -0
  192. package/dist/tools/permissions.js.map +1 -0
  193. package/dist/tools/tool_types.d.ts +48 -0
  194. package/dist/tools/tool_types.js +7 -0
  195. package/dist/tools/tool_types.js.map +1 -0
  196. package/dist/ui/banner.d.ts +4 -0
  197. package/dist/ui/banner.js +104 -0
  198. package/dist/ui/banner.js.map +1 -0
  199. package/dist/ui/callbacks.d.ts +30 -0
  200. package/dist/ui/callbacks.js +132 -0
  201. package/dist/ui/callbacks.js.map +1 -0
  202. package/dist/ui/colors.d.ts +14 -0
  203. package/dist/ui/colors.js +28 -0
  204. package/dist/ui/colors.js.map +1 -0
  205. package/dist/ui/dashboard.d.ts +51 -0
  206. package/dist/ui/dashboard.js +181 -0
  207. package/dist/ui/dashboard.js.map +1 -0
  208. package/dist/ui/leaderboard.d.ts +16 -0
  209. package/dist/ui/leaderboard.js +43 -0
  210. package/dist/ui/leaderboard.js.map +1 -0
  211. package/dist/ui/logger.d.ts +28 -0
  212. package/dist/ui/logger.js +117 -0
  213. package/dist/ui/logger.js.map +1 -0
  214. package/dist/ui/progress.d.ts +16 -0
  215. package/dist/ui/progress.js +62 -0
  216. package/dist/ui/progress.js.map +1 -0
  217. package/dist/ui/tokenizer.d.ts +5 -0
  218. package/dist/ui/tokenizer.js +54 -0
  219. package/dist/ui/tokenizer.js.map +1 -0
  220. package/package.json +63 -0
@@ -0,0 +1,215 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { randomUUID } from "node:crypto";
4
+ import { loadConfig } from "../config/loader.js";
5
+ import { createTask, createWorktreeRoute, initDatabase, } from "../db/schema.js";
6
+ import { getNegotiationLearningBySession, getNegotiationSessionById, getTaskById, listNegotiationConflicts, listNegotiationRounds, listTasksByStatus, } from "../db/queries.js";
7
+ import { createWorktree, ensureGitRepository, findWorktreeByBranch, sanitizeBranchName, } from "../git/worktree.js";
8
+ import { TaskExecutionService } from "../orchestrator/task_runner.js";
9
+ import { printDashboardNegotiation } from "../ui/dashboard.js";
10
+ import { createLogger } from "../ui/logger.js";
11
+ function readRequiredFile(filePath) {
12
+ const resolved = path.resolve(filePath);
13
+ if (!fs.existsSync(resolved)) {
14
+ throw new Error(`File not found: ${resolved}`);
15
+ }
16
+ return fs.readFileSync(resolved, "utf8");
17
+ }
18
+ function parseCapabilities(raw) {
19
+ if (!raw) {
20
+ return [];
21
+ }
22
+ return raw
23
+ .split(",")
24
+ .map((value) => value.trim())
25
+ .filter(Boolean);
26
+ }
27
+ const log = createLogger("task");
28
+ export function registerTaskCommand(program) {
29
+ const task = program
30
+ .command("task")
31
+ .description("Create and route DuoCode tasks");
32
+ task
33
+ .command("create")
34
+ .description("Create a new task")
35
+ .argument("<title>", "Task title")
36
+ .option("--cwd <path>", "Repository path", process.cwd())
37
+ .option("--description <text>", "Task description", "")
38
+ .option("--priority <level>", "low|medium|high|critical", "medium")
39
+ .option("--capabilities <list>", "Comma-separated required capability tags")
40
+ .option("--route <mode>", "auto|none|worktree", "auto")
41
+ .option("--branch <name>", "Override branch name for worktree routing")
42
+ .action((title, opts) => {
43
+ const repoPath = path.resolve(opts.cwd);
44
+ ensureGitRepository(repoPath);
45
+ const config = loadConfig(repoPath);
46
+ const dbPath = path.resolve(repoPath, config.database.path);
47
+ const db = initDatabase(dbPath);
48
+ const taskId = randomUUID();
49
+ const capabilities = parseCapabilities(opts.capabilities);
50
+ const routeMode = opts.route === "auto" ? config.routing.defaultStrategy : opts.route;
51
+ const baseBranch = config.git.baseBranch;
52
+ createTask(db, {
53
+ id: taskId,
54
+ title,
55
+ description: opts.description,
56
+ priority: opts.priority,
57
+ requiredCapabilities: capabilities,
58
+ status: routeMode === "worktree" ? "routed" : "pending",
59
+ worktreePath: null,
60
+ });
61
+ if (routeMode === "worktree") {
62
+ const branch = opts.branch
63
+ ? sanitizeBranchName(opts.branch)
64
+ : sanitizeBranchName(`task/${taskId.slice(0, 8)}-${title}`);
65
+ if (!branch) {
66
+ throw new Error("Unable to determine branch name for routed task.");
67
+ }
68
+ const existing = findWorktreeByBranch(repoPath, branch);
69
+ if (existing) {
70
+ throw new Error(`Branch already routed to worktree: ${existing.path}`);
71
+ }
72
+ const worktreeRoot = path.resolve(repoPath, config.worktrees.root);
73
+ const worktreePath = path.join(worktreeRoot, taskId);
74
+ createWorktree({
75
+ repoPath,
76
+ branchName: branch,
77
+ worktreePath,
78
+ baseRef: baseBranch,
79
+ });
80
+ createWorktreeRoute(db, taskId, branch, worktreePath);
81
+ log.info("Created task", { taskId, worktreePath, branch });
82
+ }
83
+ else {
84
+ log.info("Created task", { taskId });
85
+ }
86
+ db.close();
87
+ });
88
+ task
89
+ .command("execute")
90
+ .description("Execute Layer 4 negotiated merge for a task")
91
+ .requiredOption("--task-id <id>", "Task ID")
92
+ .requiredOption("--file-path <path>", "Logical file path in repository")
93
+ .requiredOption("--language <lang>", "typescript|python|go")
94
+ .requiredOption("--base <path>", "Absolute path to base file")
95
+ .requiredOption("--left <path>", "Absolute path to left variant")
96
+ .requiredOption("--right <path>", "Absolute path to right variant")
97
+ .option("--left-agent <id>", "Left agent id", "agent-a")
98
+ .option("--right-agent <id>", "Right agent id", "agent-b")
99
+ .option("--left-rationale <text>", "Left-side rationale")
100
+ .option("--right-rationale <text>", "Right-side rationale")
101
+ .option("--consensus-threshold <n>", "Consensus threshold (0..1)", "0.9")
102
+ .option("--max-rounds <n>", "Maximum debate rounds", "3")
103
+ .option("--json", "Emit JSON instead of dashboard", false)
104
+ .option("--cwd <path>", "Repository path", process.cwd())
105
+ .action(async (opts) => {
106
+ const repoPath = path.resolve(opts.cwd);
107
+ const config = loadConfig(repoPath);
108
+ const dbPath = path.resolve(repoPath, config.database.path);
109
+ const db = initDatabase(dbPath);
110
+ const service = new TaskExecutionService(db);
111
+ const consensusThreshold = Number(opts.consensusThreshold);
112
+ const maxRounds = Number(opts.maxRounds);
113
+ if (!Number.isFinite(consensusThreshold) || consensusThreshold < 0 || consensusThreshold > 1) {
114
+ db.close();
115
+ throw new Error("consensus-threshold must be between 0 and 1.");
116
+ }
117
+ if (!Number.isInteger(maxRounds) || maxRounds < 1) {
118
+ db.close();
119
+ throw new Error("max-rounds must be an integer >= 1.");
120
+ }
121
+ const result = await service.executeNegotiation({
122
+ taskId: opts.taskId,
123
+ filePath: opts.filePath,
124
+ language: opts.language,
125
+ baseSource: readRequiredFile(opts.base),
126
+ left: {
127
+ agentId: opts.leftAgent,
128
+ source: readRequiredFile(opts.left),
129
+ rationale: opts.leftRationale,
130
+ },
131
+ right: {
132
+ agentId: opts.rightAgent,
133
+ source: readRequiredFile(opts.right),
134
+ rationale: opts.rightRationale,
135
+ },
136
+ negotiationConfig: {
137
+ consensusThreshold,
138
+ maxRounds,
139
+ },
140
+ });
141
+ const session = getNegotiationSessionById(db, result.negotiationId);
142
+ const conflicts = listNegotiationConflicts(db, result.negotiationId);
143
+ const rounds = listNegotiationRounds(db, result.negotiationId);
144
+ const learning = getNegotiationLearningBySession(db, result.negotiationId);
145
+ if (!session) {
146
+ db.close();
147
+ throw new Error(`Negotiation session not found after execution: ${result.negotiationId}`);
148
+ }
149
+ if (opts.json) {
150
+ process.stdout.write(`${JSON.stringify({ result, session, conflicts, rounds, learning }, null, 2)}\n`);
151
+ }
152
+ else {
153
+ printDashboardNegotiation({
154
+ negotiationId: result.negotiationId,
155
+ taskId: result.taskId,
156
+ filePath: session.file_path,
157
+ language: session.language,
158
+ phase: session.phase,
159
+ conflictCount: session.conflict_count,
160
+ resolutionMethod: result.resolutionMethod,
161
+ winnerAgentId: result.winnerAgentId,
162
+ confidence: result.confidence,
163
+ explanation: result.explanation,
164
+ rounds,
165
+ conflicts,
166
+ learning: learning
167
+ ? {
168
+ consensusTrend: learning.consensus_trend,
169
+ recommendedAction: learning.recommended_action,
170
+ }
171
+ : undefined,
172
+ });
173
+ }
174
+ db.close();
175
+ });
176
+ task
177
+ .command("list")
178
+ .description("List tasks")
179
+ .option("--cwd <path>", "Repository path", process.cwd())
180
+ .option("--status <status>", "pending|routed|executing|reviewing|completed|failed|cancelled")
181
+ .action((opts) => {
182
+ const repoPath = path.resolve(opts.cwd);
183
+ const config = loadConfig(repoPath);
184
+ const dbPath = path.resolve(repoPath, config.database.path);
185
+ const db = initDatabase(dbPath);
186
+ const tasks = listTasksByStatus(db, opts.status);
187
+ for (const row of tasks) {
188
+ const capabilities = JSON.parse(row.required_capabilities);
189
+ log.raw(`${row.id} [${row.status}] ${row.priority} ${row.title} caps=${capabilities.join(",")}\n`);
190
+ }
191
+ if (tasks.length === 0) {
192
+ log.raw("No tasks found.\n");
193
+ }
194
+ db.close();
195
+ });
196
+ task
197
+ .command("show")
198
+ .description("Show one task")
199
+ .argument("<taskId>", "Task ID")
200
+ .option("--cwd <path>", "Repository path", process.cwd())
201
+ .action((taskId, opts) => {
202
+ const repoPath = path.resolve(opts.cwd);
203
+ const config = loadConfig(repoPath);
204
+ const dbPath = path.resolve(repoPath, config.database.path);
205
+ const db = initDatabase(dbPath);
206
+ const row = getTaskById(db, taskId);
207
+ if (!row) {
208
+ db.close();
209
+ throw new Error(`Task not found: ${taskId}`);
210
+ }
211
+ log.raw(`${JSON.stringify(row, null, 2)}\n`);
212
+ db.close();
213
+ });
214
+ }
215
+ //# sourceMappingURL=task.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task.js","sourceRoot":"","sources":["../../src/commands/task.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,193 @@
1
+ import { z } from "zod";
2
+ declare const duoCodeConfigSchema: z.ZodObject<{
3
+ version: z.ZodDefault<z.ZodString>;
4
+ git: z.ZodDefault<z.ZodObject<{
5
+ baseBranch: z.ZodDefault<z.ZodString>;
6
+ }, "strip", z.ZodTypeAny, {
7
+ baseBranch: string;
8
+ }, {
9
+ baseBranch?: string | undefined;
10
+ }>>;
11
+ database: z.ZodDefault<z.ZodObject<{
12
+ path: z.ZodDefault<z.ZodString>;
13
+ }, "strip", z.ZodTypeAny, {
14
+ path: string;
15
+ }, {
16
+ path?: string | undefined;
17
+ }>>;
18
+ worktrees: z.ZodDefault<z.ZodObject<{
19
+ root: z.ZodDefault<z.ZodString>;
20
+ }, "strip", z.ZodTypeAny, {
21
+ root: string;
22
+ }, {
23
+ root?: string | undefined;
24
+ }>>;
25
+ routing: z.ZodDefault<z.ZodObject<{
26
+ defaultStrategy: z.ZodDefault<z.ZodEnum<["none", "worktree"]>>;
27
+ }, "strip", z.ZodTypeAny, {
28
+ defaultStrategy: "none" | "worktree";
29
+ }, {
30
+ defaultStrategy?: "none" | "worktree" | undefined;
31
+ }>>;
32
+ providers: z.ZodDefault<z.ZodObject<{
33
+ registryPath: z.ZodDefault<z.ZodString>;
34
+ minHealthScore: z.ZodDefault<z.ZodNumber>;
35
+ }, "strip", z.ZodTypeAny, {
36
+ registryPath: string;
37
+ minHealthScore: number;
38
+ }, {
39
+ registryPath?: string | undefined;
40
+ minHealthScore?: number | undefined;
41
+ }>>;
42
+ budget: z.ZodDefault<z.ZodObject<{
43
+ maxTotalUsd: z.ZodOptional<z.ZodNumber>;
44
+ }, "strip", z.ZodTypeAny, {
45
+ maxTotalUsd?: number | undefined;
46
+ }, {
47
+ maxTotalUsd?: number | undefined;
48
+ }>>;
49
+ tools: z.ZodDefault<z.ZodObject<{
50
+ autoApprove: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
51
+ blockedPaths: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
52
+ allowedCommands: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
53
+ }, "strip", z.ZodTypeAny, {
54
+ autoApprove: string[];
55
+ blockedPaths: string[];
56
+ allowedCommands: string[];
57
+ }, {
58
+ autoApprove?: string[] | undefined;
59
+ blockedPaths?: string[] | undefined;
60
+ allowedCommands?: string[] | undefined;
61
+ }>>;
62
+ models: z.ZodDefault<z.ZodObject<{
63
+ preferredArchitect: z.ZodOptional<z.ZodString>;
64
+ preferredExecutor: z.ZodOptional<z.ZodString>;
65
+ judge: z.ZodOptional<z.ZodString>;
66
+ }, "strip", z.ZodTypeAny, {
67
+ preferredArchitect?: string | undefined;
68
+ preferredExecutor?: string | undefined;
69
+ judge?: string | undefined;
70
+ }, {
71
+ preferredArchitect?: string | undefined;
72
+ preferredExecutor?: string | undefined;
73
+ judge?: string | undefined;
74
+ }>>;
75
+ context: z.ZodDefault<z.ZodObject<{
76
+ maxDepth: z.ZodDefault<z.ZodNumber>;
77
+ maxFiles: z.ZodDefault<z.ZodNumber>;
78
+ maxFileSizeKb: z.ZodDefault<z.ZodNumber>;
79
+ maxContextChars: z.ZodDefault<z.ZodNumber>;
80
+ ignoreDirs: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
81
+ }, "strip", z.ZodTypeAny, {
82
+ maxDepth: number;
83
+ maxFiles: number;
84
+ maxFileSizeKb: number;
85
+ maxContextChars: number;
86
+ ignoreDirs: string[];
87
+ }, {
88
+ maxDepth?: number | undefined;
89
+ maxFiles?: number | undefined;
90
+ maxFileSizeKb?: number | undefined;
91
+ maxContextChars?: number | undefined;
92
+ ignoreDirs?: string[] | undefined;
93
+ }>>;
94
+ logging: z.ZodDefault<z.ZodObject<{
95
+ level: z.ZodDefault<z.ZodEnum<["error", "warn", "info", "debug", "trace"]>>;
96
+ file: z.ZodOptional<z.ZodString>;
97
+ }, "strip", z.ZodTypeAny, {
98
+ level: "error" | "warn" | "info" | "debug" | "trace";
99
+ file?: string | undefined;
100
+ }, {
101
+ level?: "error" | "warn" | "info" | "debug" | "trace" | undefined;
102
+ file?: string | undefined;
103
+ }>>;
104
+ }, "strict", z.ZodTypeAny, {
105
+ version: string;
106
+ git: {
107
+ baseBranch: string;
108
+ };
109
+ database: {
110
+ path: string;
111
+ };
112
+ worktrees: {
113
+ root: string;
114
+ };
115
+ routing: {
116
+ defaultStrategy: "none" | "worktree";
117
+ };
118
+ providers: {
119
+ registryPath: string;
120
+ minHealthScore: number;
121
+ };
122
+ budget: {
123
+ maxTotalUsd?: number | undefined;
124
+ };
125
+ tools: {
126
+ autoApprove: string[];
127
+ blockedPaths: string[];
128
+ allowedCommands: string[];
129
+ };
130
+ models: {
131
+ preferredArchitect?: string | undefined;
132
+ preferredExecutor?: string | undefined;
133
+ judge?: string | undefined;
134
+ };
135
+ context: {
136
+ maxDepth: number;
137
+ maxFiles: number;
138
+ maxFileSizeKb: number;
139
+ maxContextChars: number;
140
+ ignoreDirs: string[];
141
+ };
142
+ logging: {
143
+ level: "error" | "warn" | "info" | "debug" | "trace";
144
+ file?: string | undefined;
145
+ };
146
+ }, {
147
+ version?: string | undefined;
148
+ git?: {
149
+ baseBranch?: string | undefined;
150
+ } | undefined;
151
+ database?: {
152
+ path?: string | undefined;
153
+ } | undefined;
154
+ worktrees?: {
155
+ root?: string | undefined;
156
+ } | undefined;
157
+ routing?: {
158
+ defaultStrategy?: "none" | "worktree" | undefined;
159
+ } | undefined;
160
+ providers?: {
161
+ registryPath?: string | undefined;
162
+ minHealthScore?: number | undefined;
163
+ } | undefined;
164
+ budget?: {
165
+ maxTotalUsd?: number | undefined;
166
+ } | undefined;
167
+ tools?: {
168
+ autoApprove?: string[] | undefined;
169
+ blockedPaths?: string[] | undefined;
170
+ allowedCommands?: string[] | undefined;
171
+ } | undefined;
172
+ models?: {
173
+ preferredArchitect?: string | undefined;
174
+ preferredExecutor?: string | undefined;
175
+ judge?: string | undefined;
176
+ } | undefined;
177
+ context?: {
178
+ maxDepth?: number | undefined;
179
+ maxFiles?: number | undefined;
180
+ maxFileSizeKb?: number | undefined;
181
+ maxContextChars?: number | undefined;
182
+ ignoreDirs?: string[] | undefined;
183
+ } | undefined;
184
+ logging?: {
185
+ level?: "error" | "warn" | "info" | "debug" | "trace" | undefined;
186
+ file?: string | undefined;
187
+ } | undefined;
188
+ }>;
189
+ export type DuoCodeConfig = z.infer<typeof duoCodeConfigSchema>;
190
+ export declare function findConfigPath(cwd?: string): string | null;
191
+ export declare function loadConfig(cwd?: string, explicitPath?: string): DuoCodeConfig;
192
+ export declare function defaultConfigYaml(): string;
193
+ export {};
@@ -0,0 +1,106 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import YAML from "yaml";
4
+ import { z } from "zod";
5
+ const duoCodeConfigSchema = z.object({
6
+ version: z.string().default("1"),
7
+ git: z.object({
8
+ baseBranch: z.string().default("main"),
9
+ }).default({ baseBranch: "main" }),
10
+ database: z.object({
11
+ path: z.string().default(".duocode/duocode.db"),
12
+ }).default({ path: ".duocode/duocode.db" }),
13
+ worktrees: z.object({
14
+ root: z.string().default(".duocode/worktrees"),
15
+ }).default({ root: ".duocode/worktrees" }),
16
+ routing: z.object({
17
+ defaultStrategy: z.enum(["none", "worktree"]).default("worktree"),
18
+ }).default({ defaultStrategy: "worktree" }),
19
+ providers: z.object({
20
+ registryPath: z.string().default("registry.example.yaml"),
21
+ minHealthScore: z.number().min(0).max(1).default(0.2),
22
+ }).default({
23
+ registryPath: "registry.example.yaml",
24
+ minHealthScore: 0.2,
25
+ }),
26
+ budget: z.object({
27
+ maxTotalUsd: z.number().positive().optional(),
28
+ }).default({}),
29
+ tools: z.object({
30
+ autoApprove: z.array(z.string()).default([]),
31
+ blockedPaths: z.array(z.string()).default([]),
32
+ allowedCommands: z.array(z.string()).default([]),
33
+ }).default({ autoApprove: [], blockedPaths: [], allowedCommands: [] }),
34
+ models: z.object({
35
+ preferredArchitect: z.string().optional(),
36
+ preferredExecutor: z.string().optional(),
37
+ judge: z.string().optional(),
38
+ }).default({}),
39
+ context: z.object({
40
+ maxDepth: z.number().int().positive().default(4),
41
+ maxFiles: z.number().int().positive().default(200),
42
+ maxFileSizeKb: z.number().positive().default(100),
43
+ maxContextChars: z.number().int().positive().default(60000),
44
+ ignoreDirs: z.array(z.string()).default([]),
45
+ }).default({ maxDepth: 4, maxFiles: 200, maxFileSizeKb: 100, maxContextChars: 60000, ignoreDirs: [] }),
46
+ logging: z.object({
47
+ level: z.enum(["error", "warn", "info", "debug", "trace"]).default("info"),
48
+ file: z.string().optional(),
49
+ }).default({ level: "info" }),
50
+ }).strict();
51
+ const DEFAULT_CONFIG_BASENAME_CANDIDATES = ["duocode.yaml", "duocode.yml"];
52
+ export function findConfigPath(cwd = process.cwd()) {
53
+ for (const basename of DEFAULT_CONFIG_BASENAME_CANDIDATES) {
54
+ const candidate = path.join(cwd, basename);
55
+ if (fs.existsSync(candidate)) {
56
+ return candidate;
57
+ }
58
+ }
59
+ return null;
60
+ }
61
+ export function loadConfig(cwd = process.cwd(), explicitPath) {
62
+ const configPath = explicitPath ?? findConfigPath(cwd);
63
+ if (!configPath) {
64
+ throw new Error("No DuoCode config found. Expected duocode.yaml or duocode.yml.");
65
+ }
66
+ const raw = fs.readFileSync(configPath, "utf8");
67
+ const parsedYaml = YAML.parse(raw) ?? {};
68
+ const result = duoCodeConfigSchema.safeParse(parsedYaml);
69
+ if (!result.success) {
70
+ throw new Error(`Invalid DuoCode config (${configPath}): ${result.error.message}`);
71
+ }
72
+ return result.data;
73
+ }
74
+ export function defaultConfigYaml() {
75
+ return [
76
+ "version: \"1\"",
77
+ "git:",
78
+ " baseBranch: main",
79
+ "database:",
80
+ " path: .duocode/duocode.db",
81
+ "worktrees:",
82
+ " root: .duocode/worktrees",
83
+ "routing:",
84
+ " defaultStrategy: worktree",
85
+ "providers:",
86
+ " registryPath: registry.example.yaml",
87
+ " minHealthScore: 0.2",
88
+ "budget:",
89
+ " maxTotalUsd: 100",
90
+ "tools:",
91
+ " autoApprove: []",
92
+ " blockedPaths: []",
93
+ " allowedCommands: []",
94
+ "models: {}",
95
+ "context:",
96
+ " maxDepth: 4",
97
+ " maxFiles: 200",
98
+ " maxFileSizeKb: 100",
99
+ " maxContextChars: 60000",
100
+ " ignoreDirs: []",
101
+ "logging:",
102
+ " level: info",
103
+ "",
104
+ ].join("\n");
105
+ }
106
+ //# sourceMappingURL=loader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loader.js","sourceRoot":"","sources":["../../src/config/loader.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,79 @@
1
+ export interface ContextConfig {
2
+ /** Max depth for directory tree traversal (default 4) */
3
+ maxDepth?: number;
4
+ /** Max files to include in tree listing (default 200) */
5
+ maxTreeFiles?: number;
6
+ /** Max file size in bytes to include content (default 100KB) */
7
+ maxFileSizeBytes?: number;
8
+ /** Max total context characters (default 60_000) */
9
+ maxContextChars?: number;
10
+ /** Directories to always ignore */
11
+ ignoreDirs?: string[];
12
+ /** File patterns to always ignore */
13
+ ignorePatterns?: string[];
14
+ }
15
+ export interface FileEntry {
16
+ /** Relative path from project root */
17
+ relativePath: string;
18
+ /** Absolute path */
19
+ absolutePath: string;
20
+ /** File size in bytes */
21
+ sizeBytes: number;
22
+ /** File extension */
23
+ ext: string;
24
+ /** Last modified time */
25
+ mtimeMs: number;
26
+ }
27
+ export interface DirectoryTree {
28
+ root: string;
29
+ files: FileEntry[];
30
+ totalFiles: number;
31
+ truncated: boolean;
32
+ }
33
+ /**
34
+ * Scan a project directory and return a flat list of files.
35
+ * Respects .gitignore-style patterns and depth limits.
36
+ */
37
+ export declare function scanProjectTree(rootDir: string, config?: ContextConfig): DirectoryTree;
38
+ /**
39
+ * Render a directory tree as a concise indented string.
40
+ */
41
+ export declare function renderTree(tree: DirectoryTree): string;
42
+ /**
43
+ * Score files by relevance to a prompt using keyword matching.
44
+ * Returns files sorted by relevance score (highest first).
45
+ */
46
+ export declare function rankFilesByRelevance(files: FileEntry[], prompt: string): Array<FileEntry & {
47
+ relevanceScore: number;
48
+ }>;
49
+ export interface ContextResult {
50
+ /** The system-level context string to prepend to prompts */
51
+ systemContext: string;
52
+ /** File contents included in context */
53
+ includedFiles: Array<{
54
+ path: string;
55
+ content: string;
56
+ reason: string;
57
+ }>;
58
+ /** Total character count */
59
+ totalChars: number;
60
+ /** Files that were relevant but excluded due to budget */
61
+ droppedFiles: string[];
62
+ }
63
+ /**
64
+ * Build smart context from a project directory and task prompt.
65
+ * Includes:
66
+ * 1. Project tree summary
67
+ * 2. Contents of relevant files (within token budget)
68
+ * 3. Explicit @ mentions
69
+ */
70
+ export declare function buildProjectContext(rootDir: string, prompt: string, config?: ContextConfig): ContextResult;
71
+ export declare function extractMentions(prompt: string, rootDir: string): Array<{
72
+ relativePath: string;
73
+ absolutePath: string;
74
+ }>;
75
+ /**
76
+ * Strip @mentions from prompt text so the model sees clean input.
77
+ * Returns the prompt with @path references removed.
78
+ */
79
+ export declare function stripMentions(prompt: string): string;