@pythonluvr/openwar 0.4.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 (298) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +297 -0
  3. package/bin/openwar +45 -0
  4. package/dist/adapters/anthropic.d.ts +15 -0
  5. package/dist/adapters/anthropic.d.ts.map +1 -0
  6. package/dist/adapters/anthropic.js +179 -0
  7. package/dist/adapters/anthropic.js.map +1 -0
  8. package/dist/adapters/gemini.d.ts +15 -0
  9. package/dist/adapters/gemini.d.ts.map +1 -0
  10. package/dist/adapters/gemini.js +141 -0
  11. package/dist/adapters/gemini.js.map +1 -0
  12. package/dist/adapters/grok.d.ts +6 -0
  13. package/dist/adapters/grok.d.ts.map +1 -0
  14. package/dist/adapters/grok.js +15 -0
  15. package/dist/adapters/grok.js.map +1 -0
  16. package/dist/adapters/index.d.ts +16 -0
  17. package/dist/adapters/index.d.ts.map +1 -0
  18. package/dist/adapters/index.js +35 -0
  19. package/dist/adapters/index.js.map +1 -0
  20. package/dist/adapters/mock.d.ts +17 -0
  21. package/dist/adapters/mock.d.ts.map +1 -0
  22. package/dist/adapters/mock.js +33 -0
  23. package/dist/adapters/mock.js.map +1 -0
  24. package/dist/adapters/openai-compat.d.ts +6 -0
  25. package/dist/adapters/openai-compat.d.ts.map +1 -0
  26. package/dist/adapters/openai-compat.js +19 -0
  27. package/dist/adapters/openai-compat.js.map +1 -0
  28. package/dist/adapters/openai.d.ts +23 -0
  29. package/dist/adapters/openai.d.ts.map +1 -0
  30. package/dist/adapters/openai.js +176 -0
  31. package/dist/adapters/openai.js.map +1 -0
  32. package/dist/adapters/sse.d.ts +6 -0
  33. package/dist/adapters/sse.d.ts.map +1 -0
  34. package/dist/adapters/sse.js +45 -0
  35. package/dist/adapters/sse.js.map +1 -0
  36. package/dist/adapters/types.d.ts +7 -0
  37. package/dist/adapters/types.d.ts.map +1 -0
  38. package/dist/adapters/types.js +2 -0
  39. package/dist/adapters/types.js.map +1 -0
  40. package/dist/auth/categories.d.ts +13 -0
  41. package/dist/auth/categories.d.ts.map +1 -0
  42. package/dist/auth/categories.js +47 -0
  43. package/dist/auth/categories.js.map +1 -0
  44. package/dist/auth/check.d.ts +31 -0
  45. package/dist/auth/check.d.ts.map +1 -0
  46. package/dist/auth/check.js +59 -0
  47. package/dist/auth/check.js.map +1 -0
  48. package/dist/auth/role-scope.d.ts +18 -0
  49. package/dist/auth/role-scope.d.ts.map +1 -0
  50. package/dist/auth/role-scope.js +62 -0
  51. package/dist/auth/role-scope.js.map +1 -0
  52. package/dist/auth/wildcards.d.ts +5 -0
  53. package/dist/auth/wildcards.d.ts.map +1 -0
  54. package/dist/auth/wildcards.js +36 -0
  55. package/dist/auth/wildcards.js.map +1 -0
  56. package/dist/brief.d.ts +9 -0
  57. package/dist/brief.d.ts.map +1 -0
  58. package/dist/brief.js +514 -0
  59. package/dist/brief.js.map +1 -0
  60. package/dist/cli.d.ts +2 -0
  61. package/dist/cli.d.ts.map +1 -0
  62. package/dist/cli.js +585 -0
  63. package/dist/cli.js.map +1 -0
  64. package/dist/coordinator/cost-tracker.d.ts +14 -0
  65. package/dist/coordinator/cost-tracker.d.ts.map +1 -0
  66. package/dist/coordinator/cost-tracker.js +63 -0
  67. package/dist/coordinator/cost-tracker.js.map +1 -0
  68. package/dist/coordinator/driver.d.ts +37 -0
  69. package/dist/coordinator/driver.d.ts.map +1 -0
  70. package/dist/coordinator/driver.js +644 -0
  71. package/dist/coordinator/driver.js.map +1 -0
  72. package/dist/coordinator/index.d.ts +14 -0
  73. package/dist/coordinator/index.d.ts.map +1 -0
  74. package/dist/coordinator/index.js +8 -0
  75. package/dist/coordinator/index.js.map +1 -0
  76. package/dist/coordinator/plan-parser.d.ts +17 -0
  77. package/dist/coordinator/plan-parser.d.ts.map +1 -0
  78. package/dist/coordinator/plan-parser.js +44 -0
  79. package/dist/coordinator/plan-parser.js.map +1 -0
  80. package/dist/coordinator/result-aggregator.d.ts +21 -0
  81. package/dist/coordinator/result-aggregator.d.ts.map +1 -0
  82. package/dist/coordinator/result-aggregator.js +58 -0
  83. package/dist/coordinator/result-aggregator.js.map +1 -0
  84. package/dist/coordinator/retry-policy.d.ts +7 -0
  85. package/dist/coordinator/retry-policy.d.ts.map +1 -0
  86. package/dist/coordinator/retry-policy.js +17 -0
  87. package/dist/coordinator/retry-policy.js.map +1 -0
  88. package/dist/coordinator/state-machine.d.ts +63 -0
  89. package/dist/coordinator/state-machine.d.ts.map +1 -0
  90. package/dist/coordinator/state-machine.js +242 -0
  91. package/dist/coordinator/state-machine.js.map +1 -0
  92. package/dist/coordinator/types.d.ts +3 -0
  93. package/dist/coordinator/types.d.ts.map +1 -0
  94. package/dist/coordinator/types.js +5 -0
  95. package/dist/coordinator/types.js.map +1 -0
  96. package/dist/detectors/banned-phrases.d.ts +3 -0
  97. package/dist/detectors/banned-phrases.d.ts.map +1 -0
  98. package/dist/detectors/banned-phrases.js +33 -0
  99. package/dist/detectors/banned-phrases.js.map +1 -0
  100. package/dist/detectors/blocker.d.ts +3 -0
  101. package/dist/detectors/blocker.d.ts.map +1 -0
  102. package/dist/detectors/blocker.js +62 -0
  103. package/dist/detectors/blocker.js.map +1 -0
  104. package/dist/detectors/completion.d.ts +3 -0
  105. package/dist/detectors/completion.d.ts.map +1 -0
  106. package/dist/detectors/completion.js +17 -0
  107. package/dist/detectors/completion.js.map +1 -0
  108. package/dist/detectors/confirmation-summary.d.ts +3 -0
  109. package/dist/detectors/confirmation-summary.d.ts.map +1 -0
  110. package/dist/detectors/confirmation-summary.js +76 -0
  111. package/dist/detectors/confirmation-summary.js.map +1 -0
  112. package/dist/detectors/destructive.d.ts +3 -0
  113. package/dist/detectors/destructive.d.ts.map +1 -0
  114. package/dist/detectors/destructive.js +96 -0
  115. package/dist/detectors/destructive.js.map +1 -0
  116. package/dist/detectors/index.d.ts +12 -0
  117. package/dist/detectors/index.d.ts.map +1 -0
  118. package/dist/detectors/index.js +19 -0
  119. package/dist/detectors/index.js.map +1 -0
  120. package/dist/detectors/phase-marker.d.ts +3 -0
  121. package/dist/detectors/phase-marker.d.ts.map +1 -0
  122. package/dist/detectors/phase-marker.js +25 -0
  123. package/dist/detectors/phase-marker.js.map +1 -0
  124. package/dist/framework.d.ts +2 -0
  125. package/dist/framework.d.ts.map +1 -0
  126. package/dist/framework.js +34 -0
  127. package/dist/framework.js.map +1 -0
  128. package/dist/index.d.ts +10 -0
  129. package/dist/index.d.ts.map +1 -0
  130. package/dist/index.js +10 -0
  131. package/dist/index.js.map +1 -0
  132. package/dist/io.d.ts +25 -0
  133. package/dist/io.d.ts.map +1 -0
  134. package/dist/io.js +83 -0
  135. package/dist/io.js.map +1 -0
  136. package/dist/mcp/client.d.ts +22 -0
  137. package/dist/mcp/client.d.ts.map +1 -0
  138. package/dist/mcp/client.js +44 -0
  139. package/dist/mcp/client.js.map +1 -0
  140. package/dist/mcp/index.d.ts +5 -0
  141. package/dist/mcp/index.d.ts.map +1 -0
  142. package/dist/mcp/index.js +6 -0
  143. package/dist/mcp/index.js.map +1 -0
  144. package/dist/mcp/registry.d.ts +16 -0
  145. package/dist/mcp/registry.d.ts.map +1 -0
  146. package/dist/mcp/registry.js +53 -0
  147. package/dist/mcp/registry.js.map +1 -0
  148. package/dist/mcp/transport-stdio.d.ts +26 -0
  149. package/dist/mcp/transport-stdio.d.ts.map +1 -0
  150. package/dist/mcp/transport-stdio.js +138 -0
  151. package/dist/mcp/transport-stdio.js.map +1 -0
  152. package/dist/mcp/types.d.ts +90 -0
  153. package/dist/mcp/types.d.ts.map +1 -0
  154. package/dist/mcp/types.js +23 -0
  155. package/dist/mcp/types.js.map +1 -0
  156. package/dist/orchestration/handoff.d.ts +27 -0
  157. package/dist/orchestration/handoff.d.ts.map +1 -0
  158. package/dist/orchestration/handoff.js +322 -0
  159. package/dist/orchestration/handoff.js.map +1 -0
  160. package/dist/phases/blocker.d.ts +8 -0
  161. package/dist/phases/blocker.d.ts.map +1 -0
  162. package/dist/phases/blocker.js +11 -0
  163. package/dist/phases/blocker.js.map +1 -0
  164. package/dist/phases/completion.d.ts +10 -0
  165. package/dist/phases/completion.d.ts.map +1 -0
  166. package/dist/phases/completion.js +47 -0
  167. package/dist/phases/completion.js.map +1 -0
  168. package/dist/phases/destructive.d.ts +10 -0
  169. package/dist/phases/destructive.d.ts.map +1 -0
  170. package/dist/phases/destructive.js +31 -0
  171. package/dist/phases/destructive.js.map +1 -0
  172. package/dist/phases/execute.d.ts +30 -0
  173. package/dist/phases/execute.d.ts.map +1 -0
  174. package/dist/phases/execute.js +222 -0
  175. package/dist/phases/execute.js.map +1 -0
  176. package/dist/phases/intake.d.ts +16 -0
  177. package/dist/phases/intake.d.ts.map +1 -0
  178. package/dist/phases/intake.js +105 -0
  179. package/dist/phases/intake.js.map +1 -0
  180. package/dist/roles/critic.d.ts +3 -0
  181. package/dist/roles/critic.d.ts.map +1 -0
  182. package/dist/roles/critic.js +35 -0
  183. package/dist/roles/critic.js.map +1 -0
  184. package/dist/roles/executor.d.ts +3 -0
  185. package/dist/roles/executor.d.ts.map +1 -0
  186. package/dist/roles/executor.js +46 -0
  187. package/dist/roles/executor.js.map +1 -0
  188. package/dist/roles/index.d.ts +8 -0
  189. package/dist/roles/index.d.ts.map +1 -0
  190. package/dist/roles/index.js +8 -0
  191. package/dist/roles/index.js.map +1 -0
  192. package/dist/roles/planner.d.ts +3 -0
  193. package/dist/roles/planner.d.ts.map +1 -0
  194. package/dist/roles/planner.js +50 -0
  195. package/dist/roles/planner.js.map +1 -0
  196. package/dist/roles/prompt-overlay.d.ts +9 -0
  197. package/dist/roles/prompt-overlay.d.ts.map +1 -0
  198. package/dist/roles/prompt-overlay.js +25 -0
  199. package/dist/roles/prompt-overlay.js.map +1 -0
  200. package/dist/roles/registry.d.ts +8 -0
  201. package/dist/roles/registry.d.ts.map +1 -0
  202. package/dist/roles/registry.js +45 -0
  203. package/dist/roles/registry.js.map +1 -0
  204. package/dist/roles/reviewer.d.ts +3 -0
  205. package/dist/roles/reviewer.d.ts.map +1 -0
  206. package/dist/roles/reviewer.js +46 -0
  207. package/dist/roles/reviewer.js.map +1 -0
  208. package/dist/roles/types.d.ts +2 -0
  209. package/dist/roles/types.d.ts.map +1 -0
  210. package/dist/roles/types.js +4 -0
  211. package/dist/roles/types.js.map +1 -0
  212. package/dist/runner.d.ts +3 -0
  213. package/dist/runner.d.ts.map +1 -0
  214. package/dist/runner.js +473 -0
  215. package/dist/runner.js.map +1 -0
  216. package/dist/sandbox/host-allowlist.d.ts +13 -0
  217. package/dist/sandbox/host-allowlist.d.ts.map +1 -0
  218. package/dist/sandbox/host-allowlist.js +85 -0
  219. package/dist/sandbox/host-allowlist.js.map +1 -0
  220. package/dist/sandbox/output-cap.d.ts +9 -0
  221. package/dist/sandbox/output-cap.d.ts.map +1 -0
  222. package/dist/sandbox/output-cap.js +66 -0
  223. package/dist/sandbox/output-cap.js.map +1 -0
  224. package/dist/sandbox/timeout.d.ts +7 -0
  225. package/dist/sandbox/timeout.d.ts.map +1 -0
  226. package/dist/sandbox/timeout.js +52 -0
  227. package/dist/sandbox/timeout.js.map +1 -0
  228. package/dist/sandbox/types.d.ts +18 -0
  229. package/dist/sandbox/types.d.ts.map +1 -0
  230. package/dist/sandbox/types.js +27 -0
  231. package/dist/sandbox/types.js.map +1 -0
  232. package/dist/sandbox/workdir.d.ts +9 -0
  233. package/dist/sandbox/workdir.d.ts.map +1 -0
  234. package/dist/sandbox/workdir.js +83 -0
  235. package/dist/sandbox/workdir.js.map +1 -0
  236. package/dist/state/index.d.ts +4 -0
  237. package/dist/state/index.d.ts.map +1 -0
  238. package/dist/state/index.js +4 -0
  239. package/dist/state/index.js.map +1 -0
  240. package/dist/state/paths.d.ts +5 -0
  241. package/dist/state/paths.d.ts.map +1 -0
  242. package/dist/state/paths.js +18 -0
  243. package/dist/state/paths.js.map +1 -0
  244. package/dist/state/persist.d.ts +14 -0
  245. package/dist/state/persist.d.ts.map +1 -0
  246. package/dist/state/persist.js +146 -0
  247. package/dist/state/persist.js.map +1 -0
  248. package/dist/state/transcript.d.ts +9 -0
  249. package/dist/state/transcript.d.ts.map +1 -0
  250. package/dist/state/transcript.js +34 -0
  251. package/dist/state/transcript.js.map +1 -0
  252. package/dist/tools/native/apply_patch.d.ts +18 -0
  253. package/dist/tools/native/apply_patch.d.ts.map +1 -0
  254. package/dist/tools/native/apply_patch.js +245 -0
  255. package/dist/tools/native/apply_patch.js.map +1 -0
  256. package/dist/tools/native/http_fetch.d.ts +6 -0
  257. package/dist/tools/native/http_fetch.d.ts.map +1 -0
  258. package/dist/tools/native/http_fetch.js +168 -0
  259. package/dist/tools/native/http_fetch.js.map +1 -0
  260. package/dist/tools/native/index.d.ts +9 -0
  261. package/dist/tools/native/index.d.ts.map +1 -0
  262. package/dist/tools/native/index.js +22 -0
  263. package/dist/tools/native/index.js.map +1 -0
  264. package/dist/tools/native/list_dir.d.ts +4 -0
  265. package/dist/tools/native/list_dir.d.ts.map +1 -0
  266. package/dist/tools/native/list_dir.js +175 -0
  267. package/dist/tools/native/list_dir.js.map +1 -0
  268. package/dist/tools/native/read_file.d.ts +4 -0
  269. package/dist/tools/native/read_file.d.ts.map +1 -0
  270. package/dist/tools/native/read_file.js +91 -0
  271. package/dist/tools/native/read_file.js.map +1 -0
  272. package/dist/tools/native/shell_exec.d.ts +4 -0
  273. package/dist/tools/native/shell_exec.d.ts.map +1 -0
  274. package/dist/tools/native/shell_exec.js +180 -0
  275. package/dist/tools/native/shell_exec.js.map +1 -0
  276. package/dist/tools/native/write_file.d.ts +4 -0
  277. package/dist/tools/native/write_file.d.ts.map +1 -0
  278. package/dist/tools/native/write_file.js +101 -0
  279. package/dist/tools/native/write_file.js.map +1 -0
  280. package/dist/tools/types.d.ts +48 -0
  281. package/dist/tools/types.d.ts.map +1 -0
  282. package/dist/tools/types.js +10 -0
  283. package/dist/tools/types.js.map +1 -0
  284. package/dist/types.d.ts +385 -0
  285. package/dist/types.d.ts.map +1 -0
  286. package/dist/types.js +9 -0
  287. package/dist/types.js.map +1 -0
  288. package/examples/README.md +73 -0
  289. package/examples/creative-brief.md +34 -0
  290. package/examples/critic-disagreement-brief.md +42 -0
  291. package/examples/engineering-brief.md +35 -0
  292. package/examples/file-editing-brief.md +33 -0
  293. package/examples/mcp-brief.md +34 -0
  294. package/examples/multi-agent-brief.md +43 -0
  295. package/examples/research-brief.md +35 -0
  296. package/openwar.md +248 -0
  297. package/package.json +76 -0
  298. package/templates/brief.md +62 -0
@@ -0,0 +1,175 @@
1
+ // list_dir native tool. Default-allowed (filesystem_read).
2
+ // Lists entries with optional depth and glob filter. Skips noise dirs.
3
+ import { readdir, stat, readFile } from "node:fs/promises";
4
+ import { join, relative } from "node:path";
5
+ import { resolveAndRealpathInWorkdir, PathEscapeError } from "../../sandbox/workdir.js";
6
+ const DEFAULT_SKIP_DIRS = new Set([
7
+ ".git",
8
+ "node_modules",
9
+ ".next",
10
+ "dist",
11
+ ".turbo",
12
+ ".cache",
13
+ "build",
14
+ ]);
15
+ export const LIST_DIR_DEFINITION = {
16
+ name: "list_dir",
17
+ description: "List entries in a directory inside the workdir. Optional depth (default 1) and glob filter. " +
18
+ "Skips .git, node_modules, dist, and patterns in .openwarignore.",
19
+ input_schema: {
20
+ type: "object",
21
+ properties: {
22
+ path: { type: "string", description: "Directory path relative to workdir." },
23
+ depth: { type: "number", description: "Recursion depth. 1 = entries only. Pass a large number for full tree. Default 1." },
24
+ glob: { type: "string", description: "Optional simple glob (* and ?) applied to entry names." },
25
+ },
26
+ required: ["path"],
27
+ },
28
+ origin: "native",
29
+ authorization_categories: ["filesystem_read"],
30
+ };
31
+ function parseArgs(call) {
32
+ if (typeof call.arguments !== "object" || call.arguments === null) {
33
+ return { error: "arguments must be an object" };
34
+ }
35
+ const a = call.arguments;
36
+ if (typeof a.path !== "string")
37
+ return { error: "path must be a string" };
38
+ if (a.depth !== undefined && (typeof a.depth !== "number" || a.depth < 1)) {
39
+ return { error: "depth must be a positive number if provided" };
40
+ }
41
+ if (a.glob !== undefined && typeof a.glob !== "string") {
42
+ return { error: "glob must be a string if provided" };
43
+ }
44
+ return { path: a.path, depth: a.depth, glob: a.glob };
45
+ }
46
+ // Convert a simple glob (* matches any chars except separator, ? matches one) to RegExp.
47
+ function globToRegExp(glob) {
48
+ let re = "^";
49
+ for (const ch of glob) {
50
+ if (ch === "*")
51
+ re += "[^/\\\\]*";
52
+ else if (ch === "?")
53
+ re += "[^/\\\\]";
54
+ else if (/[.+^${}()|[\]\\]/.test(ch))
55
+ re += "\\" + ch;
56
+ else
57
+ re += ch;
58
+ }
59
+ re += "$";
60
+ return new RegExp(re);
61
+ }
62
+ async function loadIgnore(workdir) {
63
+ try {
64
+ const txt = await readFile(join(workdir, ".openwarignore"), "utf8");
65
+ return txt
66
+ .split(/\r?\n/)
67
+ .map(s => s.trim())
68
+ .filter(s => s.length > 0 && !s.startsWith("#"));
69
+ }
70
+ catch {
71
+ return [];
72
+ }
73
+ }
74
+ function shouldSkip(name, ignore) {
75
+ if (DEFAULT_SKIP_DIRS.has(name))
76
+ return true;
77
+ for (const pat of ignore) {
78
+ if (pat === name)
79
+ return true;
80
+ if (pat.includes("*") || pat.includes("?")) {
81
+ if (globToRegExp(pat).test(name))
82
+ return true;
83
+ }
84
+ }
85
+ return false;
86
+ }
87
+ async function walk(rootAbs, currentAbs, depth, maxDepth, glob, ignore, out) {
88
+ let entries;
89
+ try {
90
+ entries = await readdir(currentAbs, { withFileTypes: true });
91
+ }
92
+ catch {
93
+ return;
94
+ }
95
+ for (const e of entries) {
96
+ if (shouldSkip(e.name, ignore))
97
+ continue;
98
+ const abs = join(currentAbs, e.name);
99
+ const rel = relative(rootAbs, abs);
100
+ const displayName = rel === "" ? e.name : rel;
101
+ if (e.isDirectory()) {
102
+ if (!glob || glob.test(e.name)) {
103
+ out.push({ name: displayName, type: "directory" });
104
+ }
105
+ if (depth < maxDepth) {
106
+ await walk(rootAbs, abs, depth + 1, maxDepth, glob, ignore, out);
107
+ }
108
+ }
109
+ else if (e.isFile()) {
110
+ if (!glob || glob.test(e.name)) {
111
+ try {
112
+ const st = await stat(abs);
113
+ out.push({ name: displayName, type: "file", size: st.size });
114
+ }
115
+ catch {
116
+ out.push({ name: displayName, type: "file" });
117
+ }
118
+ }
119
+ }
120
+ }
121
+ }
122
+ export const listDirExecutor = async (call, ctx) => {
123
+ const parsed = parseArgs(call);
124
+ if ("error" in parsed) {
125
+ return {
126
+ call_id: call.id,
127
+ success: false,
128
+ content: parsed.error,
129
+ error: { code: "INVALID_ARGS", message: parsed.error },
130
+ };
131
+ }
132
+ const start = Date.now();
133
+ try {
134
+ const resolved = await resolveAndRealpathInWorkdir(ctx.workdir, parsed.path);
135
+ const st = await stat(resolved);
136
+ if (!st.isDirectory()) {
137
+ return {
138
+ call_id: call.id,
139
+ success: false,
140
+ content: `${parsed.path} is not a directory`,
141
+ error: { code: "ENOTDIR", message: "path is not a directory" },
142
+ };
143
+ }
144
+ const maxDepth = parsed.depth ?? 1;
145
+ const glob = parsed.glob ? globToRegExp(parsed.glob) : null;
146
+ const ignore = await loadIgnore(ctx.workdir);
147
+ const out = [];
148
+ await walk(resolved, resolved, 1, maxDepth, glob, ignore, out);
149
+ return {
150
+ call_id: call.id,
151
+ success: true,
152
+ content: JSON.stringify(out, null, 2),
153
+ meta: { duration_ms: Date.now() - start, bytes: out.length },
154
+ };
155
+ }
156
+ catch (err) {
157
+ if (err instanceof PathEscapeError) {
158
+ return {
159
+ call_id: call.id,
160
+ success: false,
161
+ content: err.message,
162
+ error: { code: err.code, message: err.message },
163
+ };
164
+ }
165
+ const code = err.code ?? "UNKNOWN";
166
+ const message = err.message;
167
+ return {
168
+ call_id: call.id,
169
+ success: false,
170
+ content: `Failed to list ${parsed.path}: ${message}`,
171
+ error: { code, message },
172
+ };
173
+ }
174
+ };
175
+ //# sourceMappingURL=list_dir.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"list_dir.js","sourceRoot":"","sources":["../../../src/tools/native/list_dir.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { ToolDefinition, ToolExecutor } from "../types.js";
2
+ export declare const READ_FILE_DEFINITION: ToolDefinition;
3
+ export declare const readFileExecutor: ToolExecutor;
4
+ //# sourceMappingURL=read_file.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"read_file.d.ts","sourceRoot":"","sources":["../../../src/tools/native/read_file.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,cAAc,EAA8C,YAAY,EAAE,MAAM,aAAa,CAAC;AAK5G,eAAO,MAAM,oBAAoB,EAAE,cAqBlC,CAAC;AAmBF,eAAO,MAAM,gBAAgB,EAAE,YAqD9B,CAAC"}
@@ -0,0 +1,91 @@
1
+ // read_file native tool. Default-allowed (filesystem_read is in DEFAULT_ALLOWED).
2
+ // Reads UTF-8 text inside the session workdir, capped at max_bytes.
3
+ import { readFile, stat } from "node:fs/promises";
4
+ import { resolveAndRealpathInWorkdir, PathEscapeError } from "../../sandbox/workdir.js";
5
+ const DEFAULT_MAX_BYTES = 1_000_000;
6
+ export const READ_FILE_DEFINITION = {
7
+ name: "read_file",
8
+ description: "Read a UTF-8 text file. Path is relative to the session workdir (or absolute inside it). " +
9
+ "Returns the file contents and a truncated flag when the file exceeded max_bytes.",
10
+ input_schema: {
11
+ type: "object",
12
+ properties: {
13
+ path: {
14
+ type: "string",
15
+ description: "File path relative to workdir, or absolute inside workdir.",
16
+ },
17
+ max_bytes: {
18
+ type: "number",
19
+ description: "Maximum bytes to read. Defaults to 1000000 (1 MB).",
20
+ },
21
+ },
22
+ required: ["path"],
23
+ },
24
+ origin: "native",
25
+ authorization_categories: ["filesystem_read"],
26
+ };
27
+ function parseArgs(call) {
28
+ if (typeof call.arguments !== "object" || call.arguments === null) {
29
+ return { error: "arguments must be an object" };
30
+ }
31
+ const a = call.arguments;
32
+ if (typeof a.path !== "string")
33
+ return { error: "path must be a string" };
34
+ if (a.max_bytes !== undefined && (typeof a.max_bytes !== "number" || a.max_bytes < 0)) {
35
+ return { error: "max_bytes must be a non-negative number" };
36
+ }
37
+ return { path: a.path, max_bytes: a.max_bytes };
38
+ }
39
+ export const readFileExecutor = async (call, ctx) => {
40
+ const parsed = parseArgs(call);
41
+ if ("error" in parsed) {
42
+ return {
43
+ call_id: call.id,
44
+ success: false,
45
+ content: parsed.error,
46
+ error: { code: "INVALID_ARGS", message: parsed.error },
47
+ };
48
+ }
49
+ const maxBytes = parsed.max_bytes ?? DEFAULT_MAX_BYTES;
50
+ const start = Date.now();
51
+ try {
52
+ const resolved = await resolveAndRealpathInWorkdir(ctx.workdir, parsed.path);
53
+ const st = await stat(resolved);
54
+ if (st.isDirectory()) {
55
+ return {
56
+ call_id: call.id,
57
+ success: false,
58
+ content: `${parsed.path} is a directory; use list_dir.`,
59
+ error: { code: "EISDIR", message: "path is a directory" },
60
+ };
61
+ }
62
+ const buf = await readFile(resolved);
63
+ const truncated = buf.length > maxBytes;
64
+ const content = truncated ? buf.subarray(0, maxBytes).toString("utf8") : buf.toString("utf8");
65
+ return {
66
+ call_id: call.id,
67
+ success: true,
68
+ content,
69
+ meta: { duration_ms: Date.now() - start, bytes: content.length, truncated },
70
+ };
71
+ }
72
+ catch (err) {
73
+ if (err instanceof PathEscapeError) {
74
+ return {
75
+ call_id: call.id,
76
+ success: false,
77
+ content: err.message,
78
+ error: { code: err.code, message: err.message },
79
+ };
80
+ }
81
+ const code = err.code ?? "UNKNOWN";
82
+ const message = err.message;
83
+ return {
84
+ call_id: call.id,
85
+ success: false,
86
+ content: `Failed to read ${parsed.path}: ${message}`,
87
+ error: { code, message },
88
+ };
89
+ }
90
+ };
91
+ //# sourceMappingURL=read_file.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"read_file.js","sourceRoot":"","sources":["../../../src/tools/native/read_file.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { ToolDefinition, ToolExecutor } from "../types.js";
2
+ export declare const SHELL_EXEC_DEFINITION: ToolDefinition;
3
+ export declare const shellExecExecutor: ToolExecutor;
4
+ //# sourceMappingURL=shell_exec.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shell_exec.d.ts","sourceRoot":"","sources":["../../../src/tools/native/shell_exec.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,cAAc,EAA8C,YAAY,EAAE,MAAM,aAAa,CAAC;AAQ5G,eAAO,MAAM,qBAAqB,EAAE,cAiBnC,CAAC;AAqDF,eAAO,MAAM,iBAAiB,EAAE,YAiH/B,CAAC"}
@@ -0,0 +1,180 @@
1
+ // shell_exec native tool. Requires shell_exec category.
2
+ // SIGTERM on timeout, SIGKILL after 2s if still alive. stdout/stderr capped.
3
+ import { spawn } from "node:child_process";
4
+ import { platform } from "node:os";
5
+ import { resolvePathInWorkdir, PathEscapeError } from "../../sandbox/workdir.js";
6
+ import { capStream } from "../../sandbox/output-cap.js";
7
+ const DEFAULT_TIMEOUT_MS = 30_000;
8
+ const MAX_TIMEOUT_MS = 300_000;
9
+ const KILL_GRACE_MS = 2000;
10
+ export const SHELL_EXEC_DEFINITION = {
11
+ name: "shell_exec",
12
+ description: "Execute a shell command in the workdir. Captures stdout, stderr, exit code. Killed on timeout via SIGTERM, then SIGKILL. " +
13
+ "Default timeout 30s, hard cap 300s. stdout and stderr capped at the session's output limit; truncated flag set when hit.",
14
+ input_schema: {
15
+ type: "object",
16
+ properties: {
17
+ cmd: { type: "string", description: "Command to run. Passed to the configured shell." },
18
+ cwd: { type: "string", description: "Working directory relative to workdir. Default: workdir root." },
19
+ timeout_ms: { type: "number", description: "Override timeout. Default 30000, max 300000." },
20
+ shell: { type: "string", description: "Override shell binary. Default: bash on unix, cmd.exe on Windows." },
21
+ },
22
+ required: ["cmd"],
23
+ },
24
+ origin: "native",
25
+ authorization_categories: ["shell_exec"],
26
+ };
27
+ function parseArgs(call) {
28
+ if (typeof call.arguments !== "object" || call.arguments === null) {
29
+ return { error: "arguments must be an object" };
30
+ }
31
+ const a = call.arguments;
32
+ if (typeof a.cmd !== "string" || a.cmd.length === 0)
33
+ return { error: "cmd must be a non-empty string" };
34
+ if (a.cwd !== undefined && typeof a.cwd !== "string")
35
+ return { error: "cwd must be a string if provided" };
36
+ if (a.timeout_ms !== undefined && (typeof a.timeout_ms !== "number" || a.timeout_ms <= 0)) {
37
+ return { error: "timeout_ms must be a positive number if provided" };
38
+ }
39
+ if (a.shell !== undefined && typeof a.shell !== "string")
40
+ return { error: "shell must be a string if provided" };
41
+ return {
42
+ cmd: a.cmd,
43
+ cwd: a.cwd,
44
+ timeout_ms: a.timeout_ms,
45
+ shell: a.shell,
46
+ };
47
+ }
48
+ function defaultShell() {
49
+ return platform() === "win32" ? "cmd.exe" : "bash";
50
+ }
51
+ function shellArgs(shell, cmd) {
52
+ // Windows cmd uses /c; bash/sh use -c.
53
+ if (shell.endsWith("cmd.exe") || shell === "cmd")
54
+ return ["/c", cmd];
55
+ return ["-c", cmd];
56
+ }
57
+ // Cross-platform kill. On Windows, child.kill targets only the shell process,
58
+ // not grandchildren. Use taskkill /T /F to terminate the whole tree.
59
+ function killChild(child, severity) {
60
+ if (platform() === "win32" && child.pid !== undefined) {
61
+ try {
62
+ const args = severity === "hard" ? ["/T", "/F", "/PID", String(child.pid)] : ["/T", "/PID", String(child.pid)];
63
+ spawn("taskkill", args, { stdio: "ignore" }).on("error", () => { });
64
+ return;
65
+ }
66
+ catch { /* fall through */ }
67
+ }
68
+ try {
69
+ child.kill(severity === "hard" ? "SIGKILL" : "SIGTERM");
70
+ }
71
+ catch { /* swallow */ }
72
+ }
73
+ export const shellExecExecutor = async (call, ctx) => {
74
+ if (!ctx.shellEnabled) {
75
+ return {
76
+ call_id: call.id,
77
+ success: false,
78
+ content: "Shell execution is disabled for this session (--no-shell).",
79
+ error: { code: "SHELL_DISABLED", message: "shell disabled" },
80
+ };
81
+ }
82
+ const parsed = parseArgs(call);
83
+ if ("error" in parsed) {
84
+ return {
85
+ call_id: call.id,
86
+ success: false,
87
+ content: parsed.error,
88
+ error: { code: "INVALID_ARGS", message: parsed.error },
89
+ };
90
+ }
91
+ const start = Date.now();
92
+ const timeoutMs = Math.min(parsed.timeout_ms ?? DEFAULT_TIMEOUT_MS, MAX_TIMEOUT_MS);
93
+ let cwd;
94
+ try {
95
+ cwd = parsed.cwd ? resolvePathInWorkdir(ctx.workdir, parsed.cwd) : ctx.workdir;
96
+ }
97
+ catch (err) {
98
+ if (err instanceof PathEscapeError) {
99
+ return {
100
+ call_id: call.id,
101
+ success: false,
102
+ content: err.message,
103
+ error: { code: err.code, message: err.message },
104
+ };
105
+ }
106
+ throw err;
107
+ }
108
+ const shell = parsed.shell ?? defaultShell();
109
+ const args = shellArgs(shell, parsed.cmd);
110
+ const maxOut = ctx.defaultMaxOutputBytes;
111
+ return new Promise((resolve) => {
112
+ let child;
113
+ try {
114
+ child = spawn(shell, args, { cwd, stdio: ["ignore", "pipe", "pipe"] });
115
+ }
116
+ catch (err) {
117
+ resolve({
118
+ call_id: call.id,
119
+ success: false,
120
+ content: `Failed to spawn shell: ${err.message}`,
121
+ error: { code: "SPAWN_FAILED", message: err.message },
122
+ });
123
+ return;
124
+ }
125
+ let killed = false;
126
+ let killSignal;
127
+ const sigtermTimer = setTimeout(() => {
128
+ killed = true;
129
+ killSignal = "SIGTERM";
130
+ killChild(child, "soft");
131
+ setTimeout(() => {
132
+ if (child.exitCode === null && child.signalCode === null) {
133
+ killSignal = "SIGKILL";
134
+ killChild(child, "hard");
135
+ }
136
+ }, KILL_GRACE_MS);
137
+ }, timeoutMs);
138
+ const stdoutPromise = capStream(child.stdout, maxOut);
139
+ const stderrPromise = capStream(child.stderr, maxOut);
140
+ child.on("close", async (code, signal) => {
141
+ clearTimeout(sigtermTimer);
142
+ const [outRes, errRes] = await Promise.all([stdoutPromise, stderrPromise]);
143
+ const truncated = outRes.truncated || errRes.truncated;
144
+ const exitCode = code ?? -1;
145
+ const result = {
146
+ stdout: outRes.content.toString("utf8"),
147
+ stderr: errRes.content.toString("utf8"),
148
+ exit_code: exitCode,
149
+ signal: signal ?? killSignal,
150
+ truncated,
151
+ };
152
+ const summary = killed
153
+ ? `Killed after ${timeoutMs}ms timeout (${killSignal ?? "SIGTERM"}).`
154
+ : `Exited with code ${exitCode}${truncated ? " (output truncated)" : ""}.`;
155
+ resolve({
156
+ call_id: call.id,
157
+ success: !killed && exitCode === 0,
158
+ content: `${summary}\n\nstdout:\n${result.stdout}\n\nstderr:\n${result.stderr}`,
159
+ meta: {
160
+ duration_ms: Date.now() - start,
161
+ exit_code: result.exit_code,
162
+ signal: result.signal ?? undefined,
163
+ truncated,
164
+ bytes: result.stdout.length + result.stderr.length,
165
+ },
166
+ ...(killed && { error: { code: "TIMEOUT", message: summary } }),
167
+ });
168
+ });
169
+ child.on("error", (err) => {
170
+ clearTimeout(sigtermTimer);
171
+ resolve({
172
+ call_id: call.id,
173
+ success: false,
174
+ content: `Shell error: ${err.message}`,
175
+ error: { code: "SHELL_ERROR", message: err.message },
176
+ });
177
+ });
178
+ });
179
+ };
180
+ //# sourceMappingURL=shell_exec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shell_exec.js","sourceRoot":"","sources":["../../../src/tools/native/shell_exec.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { ToolDefinition, ToolExecutor } from "../types.js";
2
+ export declare const WRITE_FILE_DEFINITION: ToolDefinition;
3
+ export declare const writeFileExecutor: ToolExecutor;
4
+ //# sourceMappingURL=write_file.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"write_file.d.ts","sourceRoot":"","sources":["../../../src/tools/native/write_file.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,cAAc,EAA8C,YAAY,EAAE,MAAM,aAAa,CAAC;AAG5G,eAAO,MAAM,qBAAqB,EAAE,cAgBnC,CAAC;AAsCF,eAAO,MAAM,iBAAiB,EAAE,YA+C/B,CAAC"}
@@ -0,0 +1,101 @@
1
+ // write_file native tool. Requires filesystem_write.
2
+ // Atomic via tmp + rename for overwrite; append uses appendFile.
3
+ // Creates parent directories if missing.
4
+ import { writeFile, appendFile, rename, mkdir, stat } from "node:fs/promises";
5
+ import { dirname, basename } from "node:path";
6
+ import { randomBytes } from "node:crypto";
7
+ import { resolvePathInWorkdir, PathEscapeError } from "../../sandbox/workdir.js";
8
+ export const WRITE_FILE_DEFINITION = {
9
+ name: "write_file",
10
+ description: "Write UTF-8 content to a file inside the workdir. Overwrites by default; pass append=true to append. " +
11
+ "Atomic via tmp+rename for overwrite. Creates parent directories if missing.",
12
+ input_schema: {
13
+ type: "object",
14
+ properties: {
15
+ path: { type: "string", description: "File path relative to workdir." },
16
+ content: { type: "string", description: "UTF-8 content to write." },
17
+ append: { type: "boolean", description: "Append instead of overwrite. Default false." },
18
+ },
19
+ required: ["path", "content"],
20
+ },
21
+ origin: "native",
22
+ authorization_categories: ["filesystem_write"],
23
+ };
24
+ function parseArgs(call) {
25
+ if (typeof call.arguments !== "object" || call.arguments === null) {
26
+ return { error: "arguments must be an object" };
27
+ }
28
+ const a = call.arguments;
29
+ if (typeof a.path !== "string")
30
+ return { error: "path must be a string" };
31
+ if (typeof a.content !== "string")
32
+ return { error: "content must be a string" };
33
+ if (a.append !== undefined && typeof a.append !== "boolean") {
34
+ return { error: "append must be a boolean if provided" };
35
+ }
36
+ return { path: a.path, content: a.content, append: a.append };
37
+ }
38
+ async function ensureParents(path) {
39
+ const parent = dirname(path);
40
+ try {
41
+ await stat(parent);
42
+ return false; // already exists
43
+ }
44
+ catch {
45
+ await mkdir(parent, { recursive: true });
46
+ return true;
47
+ }
48
+ }
49
+ async function atomicWrite(path, content) {
50
+ const tmp = `${path}.openwar-tmp-${randomBytes(6).toString("hex")}`;
51
+ await writeFile(tmp, content, "utf8");
52
+ await rename(tmp, path);
53
+ }
54
+ export const writeFileExecutor = async (call, ctx) => {
55
+ const parsed = parseArgs(call);
56
+ if ("error" in parsed) {
57
+ return {
58
+ call_id: call.id,
59
+ success: false,
60
+ content: parsed.error,
61
+ error: { code: "INVALID_ARGS", message: parsed.error },
62
+ };
63
+ }
64
+ const start = Date.now();
65
+ try {
66
+ const resolved = resolvePathInWorkdir(ctx.workdir, parsed.path);
67
+ const createdParents = await ensureParents(resolved);
68
+ if (parsed.append) {
69
+ await appendFile(resolved, parsed.content, "utf8");
70
+ }
71
+ else {
72
+ await atomicWrite(resolved, parsed.content);
73
+ }
74
+ const bytes = Buffer.byteLength(parsed.content, "utf8");
75
+ return {
76
+ call_id: call.id,
77
+ success: true,
78
+ content: `Wrote ${bytes} bytes to ${basename(resolved)}${createdParents ? " (created parent dirs)" : ""}.`,
79
+ meta: { duration_ms: Date.now() - start, bytes },
80
+ };
81
+ }
82
+ catch (err) {
83
+ if (err instanceof PathEscapeError) {
84
+ return {
85
+ call_id: call.id,
86
+ success: false,
87
+ content: err.message,
88
+ error: { code: err.code, message: err.message },
89
+ };
90
+ }
91
+ const code = err.code ?? "UNKNOWN";
92
+ const message = err.message;
93
+ return {
94
+ call_id: call.id,
95
+ success: false,
96
+ content: `Failed to write ${parsed.path}: ${message}`,
97
+ error: { code, message },
98
+ };
99
+ }
100
+ };
101
+ //# sourceMappingURL=write_file.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"write_file.js","sourceRoot":"","sources":["../../../src/tools/native/write_file.ts"],"names":[],"mappings":"AAAA,qDAAqD;AACrD,iEAAiE;AACjE,yCAAyC;AAEzC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AAC9E,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAE1C,OAAO,EAAE,oBAAoB,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAEjF,MAAM,CAAC,MAAM,qBAAqB,GAAmB;IACnD,IAAI,EAAE,YAAY;IAClB,WAAW,EACT,uGAAuG;QACvG,6EAA6E;IAC/E,YAAY,EAAE;QACZ,IAAI,EAAE,QAAQ;QACd,UAAU,EAAE;YACV,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,WAAW,EAAE,gCAAgC,EAAE;YACvE,OAAO,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,WAAW,EAAE,yBAAyB,EAAE;YACnE,MAAM,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,WAAW,EAAE,6CAA6C,EAAE;SACxF;QACD,QAAQ,EAAE,CAAC,MAAM,EAAE,SAAS,CAAC;KAC9B;IACD,MAAM,EAAE,QAAQ;IAChB,wBAAwB,EAAE,CAAC,kBAAkB,CAAC;CAC/C,CAAC;AAQF,SAAS,SAAS,CAAC,IAAc;IAC/B,IAAI,OAAO,IAAI,CAAC,SAAS,KAAK,QAAQ,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;QAClE,OAAO,EAAE,KAAK,EAAE,6BAA6B,EAAE,CAAC;IAClD,CAAC;IACD,MAAM,CAAC,GAAG,IAAI,CAAC,SAAoC,CAAC;IACpD,IAAI,OAAO,CAAC,CAAC,IAAI,KAAK,QAAQ;QAAE,OAAO,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC;IAC1E,IAAI,OAAO,CAAC,CAAC,OAAO,KAAK,QAAQ;QAAE,OAAO,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC;IAChF,IAAI,CAAC,CAAC,MAAM,KAAK,SAAS,IAAI,OAAO,CAAC,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;QAC5D,OAAO,EAAE,KAAK,EAAE,sCAAsC,EAAE,CAAC;IAC3D,CAAC;IACD,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC,MAA6B,EAAE,CAAC;AACvF,CAAC;AAED,KAAK,UAAU,aAAa,CAAC,IAAY;IACvC,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7B,IAAI,CAAC;QACH,MAAM,IAAI,CAAC,MAAM,CAAC,CAAC;QACnB,OAAO,KAAK,CAAC,CAAC,iBAAiB;IACjC,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,KAAK,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACzC,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,KAAK,UAAU,WAAW,CAAC,IAAY,EAAE,OAAe;IACtD,MAAM,GAAG,GAAG,GAAG,IAAI,gBAAgB,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;IACpE,MAAM,SAAS,CAAC,GAAG,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;IACtC,MAAM,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,MAAM,CAAC,MAAM,iBAAiB,GAAiB,KAAK,EAClD,IAAc,EACd,GAAyB,EACJ,EAAE;IACvB,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAC/B,IAAI,OAAO,IAAI,MAAM,EAAE,CAAC;QACtB,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,EAAE;YAChB,OAAO,EAAE,KAAK;YACd,OAAO,EAAE,MAAM,CAAC,KAAK;YACrB,KAAK,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,OAAO,EAAE,MAAM,CAAC,KAAK,EAAE;SACvD,CAAC;IACJ,CAAC;IACD,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACzB,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,oBAAoB,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;QAChE,MAAM,cAAc,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAC;QACrD,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;YAClB,MAAM,UAAU,CAAC,QAAQ,EAAE,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACrD,CAAC;aAAM,CAAC;YACN,MAAM,WAAW,CAAC,QAAQ,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC;QAC9C,CAAC;QACD,MAAM,KAAK,GAAG,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACxD,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,EAAE;YAChB,OAAO,EAAE,IAAI;YACb,OAAO,EAAE,SAAS,KAAK,aAAa,QAAQ,CAAC,QAAQ,CAAC,GAAG,cAAc,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,EAAE,GAAG;YAC1G,IAAI,EAAE,EAAE,WAAW,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,EAAE,KAAK,EAAE;SACjD,CAAC;IACJ,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,IAAI,GAAG,YAAY,eAAe,EAAE,CAAC;YACnC,OAAO;gBACL,OAAO,EAAE,IAAI,CAAC,EAAE;gBAChB,OAAO,EAAE,KAAK;gBACd,OAAO,EAAE,GAAG,CAAC,OAAO;gBACpB,KAAK,EAAE,EAAE,IAAI,EAAE,GAAG,CAAC,IAAI,EAAE,OAAO,EAAE,GAAG,CAAC,OAAO,EAAE;aAChD,CAAC;QACJ,CAAC;QACD,MAAM,IAAI,GAAI,GAA6B,CAAC,IAAI,IAAI,SAAS,CAAC;QAC9D,MAAM,OAAO,GAAI,GAAa,CAAC,OAAO,CAAC;QACvC,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,EAAE;YAChB,OAAO,EAAE,KAAK;YACd,OAAO,EAAE,mBAAmB,MAAM,CAAC,IAAI,KAAK,OAAO,EAAE;YACrD,KAAK,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE;SACzB,CAAC;IACJ,CAAC;AACH,CAAC,CAAC"}