@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.
- package/LICENSE +21 -0
- package/README.md +297 -0
- package/bin/openwar +45 -0
- package/dist/adapters/anthropic.d.ts +15 -0
- package/dist/adapters/anthropic.d.ts.map +1 -0
- package/dist/adapters/anthropic.js +179 -0
- package/dist/adapters/anthropic.js.map +1 -0
- package/dist/adapters/gemini.d.ts +15 -0
- package/dist/adapters/gemini.d.ts.map +1 -0
- package/dist/adapters/gemini.js +141 -0
- package/dist/adapters/gemini.js.map +1 -0
- package/dist/adapters/grok.d.ts +6 -0
- package/dist/adapters/grok.d.ts.map +1 -0
- package/dist/adapters/grok.js +15 -0
- package/dist/adapters/grok.js.map +1 -0
- package/dist/adapters/index.d.ts +16 -0
- package/dist/adapters/index.d.ts.map +1 -0
- package/dist/adapters/index.js +35 -0
- package/dist/adapters/index.js.map +1 -0
- package/dist/adapters/mock.d.ts +17 -0
- package/dist/adapters/mock.d.ts.map +1 -0
- package/dist/adapters/mock.js +33 -0
- package/dist/adapters/mock.js.map +1 -0
- package/dist/adapters/openai-compat.d.ts +6 -0
- package/dist/adapters/openai-compat.d.ts.map +1 -0
- package/dist/adapters/openai-compat.js +19 -0
- package/dist/adapters/openai-compat.js.map +1 -0
- package/dist/adapters/openai.d.ts +23 -0
- package/dist/adapters/openai.d.ts.map +1 -0
- package/dist/adapters/openai.js +176 -0
- package/dist/adapters/openai.js.map +1 -0
- package/dist/adapters/sse.d.ts +6 -0
- package/dist/adapters/sse.d.ts.map +1 -0
- package/dist/adapters/sse.js +45 -0
- package/dist/adapters/sse.js.map +1 -0
- package/dist/adapters/types.d.ts +7 -0
- package/dist/adapters/types.d.ts.map +1 -0
- package/dist/adapters/types.js +2 -0
- package/dist/adapters/types.js.map +1 -0
- package/dist/auth/categories.d.ts +13 -0
- package/dist/auth/categories.d.ts.map +1 -0
- package/dist/auth/categories.js +47 -0
- package/dist/auth/categories.js.map +1 -0
- package/dist/auth/check.d.ts +31 -0
- package/dist/auth/check.d.ts.map +1 -0
- package/dist/auth/check.js +59 -0
- package/dist/auth/check.js.map +1 -0
- package/dist/auth/role-scope.d.ts +18 -0
- package/dist/auth/role-scope.d.ts.map +1 -0
- package/dist/auth/role-scope.js +62 -0
- package/dist/auth/role-scope.js.map +1 -0
- package/dist/auth/wildcards.d.ts +5 -0
- package/dist/auth/wildcards.d.ts.map +1 -0
- package/dist/auth/wildcards.js +36 -0
- package/dist/auth/wildcards.js.map +1 -0
- package/dist/brief.d.ts +9 -0
- package/dist/brief.d.ts.map +1 -0
- package/dist/brief.js +514 -0
- package/dist/brief.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +585 -0
- package/dist/cli.js.map +1 -0
- package/dist/coordinator/cost-tracker.d.ts +14 -0
- package/dist/coordinator/cost-tracker.d.ts.map +1 -0
- package/dist/coordinator/cost-tracker.js +63 -0
- package/dist/coordinator/cost-tracker.js.map +1 -0
- package/dist/coordinator/driver.d.ts +37 -0
- package/dist/coordinator/driver.d.ts.map +1 -0
- package/dist/coordinator/driver.js +644 -0
- package/dist/coordinator/driver.js.map +1 -0
- package/dist/coordinator/index.d.ts +14 -0
- package/dist/coordinator/index.d.ts.map +1 -0
- package/dist/coordinator/index.js +8 -0
- package/dist/coordinator/index.js.map +1 -0
- package/dist/coordinator/plan-parser.d.ts +17 -0
- package/dist/coordinator/plan-parser.d.ts.map +1 -0
- package/dist/coordinator/plan-parser.js +44 -0
- package/dist/coordinator/plan-parser.js.map +1 -0
- package/dist/coordinator/result-aggregator.d.ts +21 -0
- package/dist/coordinator/result-aggregator.d.ts.map +1 -0
- package/dist/coordinator/result-aggregator.js +58 -0
- package/dist/coordinator/result-aggregator.js.map +1 -0
- package/dist/coordinator/retry-policy.d.ts +7 -0
- package/dist/coordinator/retry-policy.d.ts.map +1 -0
- package/dist/coordinator/retry-policy.js +17 -0
- package/dist/coordinator/retry-policy.js.map +1 -0
- package/dist/coordinator/state-machine.d.ts +63 -0
- package/dist/coordinator/state-machine.d.ts.map +1 -0
- package/dist/coordinator/state-machine.js +242 -0
- package/dist/coordinator/state-machine.js.map +1 -0
- package/dist/coordinator/types.d.ts +3 -0
- package/dist/coordinator/types.d.ts.map +1 -0
- package/dist/coordinator/types.js +5 -0
- package/dist/coordinator/types.js.map +1 -0
- package/dist/detectors/banned-phrases.d.ts +3 -0
- package/dist/detectors/banned-phrases.d.ts.map +1 -0
- package/dist/detectors/banned-phrases.js +33 -0
- package/dist/detectors/banned-phrases.js.map +1 -0
- package/dist/detectors/blocker.d.ts +3 -0
- package/dist/detectors/blocker.d.ts.map +1 -0
- package/dist/detectors/blocker.js +62 -0
- package/dist/detectors/blocker.js.map +1 -0
- package/dist/detectors/completion.d.ts +3 -0
- package/dist/detectors/completion.d.ts.map +1 -0
- package/dist/detectors/completion.js +17 -0
- package/dist/detectors/completion.js.map +1 -0
- package/dist/detectors/confirmation-summary.d.ts +3 -0
- package/dist/detectors/confirmation-summary.d.ts.map +1 -0
- package/dist/detectors/confirmation-summary.js +76 -0
- package/dist/detectors/confirmation-summary.js.map +1 -0
- package/dist/detectors/destructive.d.ts +3 -0
- package/dist/detectors/destructive.d.ts.map +1 -0
- package/dist/detectors/destructive.js +96 -0
- package/dist/detectors/destructive.js.map +1 -0
- package/dist/detectors/index.d.ts +12 -0
- package/dist/detectors/index.d.ts.map +1 -0
- package/dist/detectors/index.js +19 -0
- package/dist/detectors/index.js.map +1 -0
- package/dist/detectors/phase-marker.d.ts +3 -0
- package/dist/detectors/phase-marker.d.ts.map +1 -0
- package/dist/detectors/phase-marker.js +25 -0
- package/dist/detectors/phase-marker.js.map +1 -0
- package/dist/framework.d.ts +2 -0
- package/dist/framework.d.ts.map +1 -0
- package/dist/framework.js +34 -0
- package/dist/framework.js.map +1 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -0
- package/dist/index.js.map +1 -0
- package/dist/io.d.ts +25 -0
- package/dist/io.d.ts.map +1 -0
- package/dist/io.js +83 -0
- package/dist/io.js.map +1 -0
- package/dist/mcp/client.d.ts +22 -0
- package/dist/mcp/client.d.ts.map +1 -0
- package/dist/mcp/client.js +44 -0
- package/dist/mcp/client.js.map +1 -0
- package/dist/mcp/index.d.ts +5 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/index.js +6 -0
- package/dist/mcp/index.js.map +1 -0
- package/dist/mcp/registry.d.ts +16 -0
- package/dist/mcp/registry.d.ts.map +1 -0
- package/dist/mcp/registry.js +53 -0
- package/dist/mcp/registry.js.map +1 -0
- package/dist/mcp/transport-stdio.d.ts +26 -0
- package/dist/mcp/transport-stdio.d.ts.map +1 -0
- package/dist/mcp/transport-stdio.js +138 -0
- package/dist/mcp/transport-stdio.js.map +1 -0
- package/dist/mcp/types.d.ts +90 -0
- package/dist/mcp/types.d.ts.map +1 -0
- package/dist/mcp/types.js +23 -0
- package/dist/mcp/types.js.map +1 -0
- package/dist/orchestration/handoff.d.ts +27 -0
- package/dist/orchestration/handoff.d.ts.map +1 -0
- package/dist/orchestration/handoff.js +322 -0
- package/dist/orchestration/handoff.js.map +1 -0
- package/dist/phases/blocker.d.ts +8 -0
- package/dist/phases/blocker.d.ts.map +1 -0
- package/dist/phases/blocker.js +11 -0
- package/dist/phases/blocker.js.map +1 -0
- package/dist/phases/completion.d.ts +10 -0
- package/dist/phases/completion.d.ts.map +1 -0
- package/dist/phases/completion.js +47 -0
- package/dist/phases/completion.js.map +1 -0
- package/dist/phases/destructive.d.ts +10 -0
- package/dist/phases/destructive.d.ts.map +1 -0
- package/dist/phases/destructive.js +31 -0
- package/dist/phases/destructive.js.map +1 -0
- package/dist/phases/execute.d.ts +30 -0
- package/dist/phases/execute.d.ts.map +1 -0
- package/dist/phases/execute.js +222 -0
- package/dist/phases/execute.js.map +1 -0
- package/dist/phases/intake.d.ts +16 -0
- package/dist/phases/intake.d.ts.map +1 -0
- package/dist/phases/intake.js +105 -0
- package/dist/phases/intake.js.map +1 -0
- package/dist/roles/critic.d.ts +3 -0
- package/dist/roles/critic.d.ts.map +1 -0
- package/dist/roles/critic.js +35 -0
- package/dist/roles/critic.js.map +1 -0
- package/dist/roles/executor.d.ts +3 -0
- package/dist/roles/executor.d.ts.map +1 -0
- package/dist/roles/executor.js +46 -0
- package/dist/roles/executor.js.map +1 -0
- package/dist/roles/index.d.ts +8 -0
- package/dist/roles/index.d.ts.map +1 -0
- package/dist/roles/index.js +8 -0
- package/dist/roles/index.js.map +1 -0
- package/dist/roles/planner.d.ts +3 -0
- package/dist/roles/planner.d.ts.map +1 -0
- package/dist/roles/planner.js +50 -0
- package/dist/roles/planner.js.map +1 -0
- package/dist/roles/prompt-overlay.d.ts +9 -0
- package/dist/roles/prompt-overlay.d.ts.map +1 -0
- package/dist/roles/prompt-overlay.js +25 -0
- package/dist/roles/prompt-overlay.js.map +1 -0
- package/dist/roles/registry.d.ts +8 -0
- package/dist/roles/registry.d.ts.map +1 -0
- package/dist/roles/registry.js +45 -0
- package/dist/roles/registry.js.map +1 -0
- package/dist/roles/reviewer.d.ts +3 -0
- package/dist/roles/reviewer.d.ts.map +1 -0
- package/dist/roles/reviewer.js +46 -0
- package/dist/roles/reviewer.js.map +1 -0
- package/dist/roles/types.d.ts +2 -0
- package/dist/roles/types.d.ts.map +1 -0
- package/dist/roles/types.js +4 -0
- package/dist/roles/types.js.map +1 -0
- package/dist/runner.d.ts +3 -0
- package/dist/runner.d.ts.map +1 -0
- package/dist/runner.js +473 -0
- package/dist/runner.js.map +1 -0
- package/dist/sandbox/host-allowlist.d.ts +13 -0
- package/dist/sandbox/host-allowlist.d.ts.map +1 -0
- package/dist/sandbox/host-allowlist.js +85 -0
- package/dist/sandbox/host-allowlist.js.map +1 -0
- package/dist/sandbox/output-cap.d.ts +9 -0
- package/dist/sandbox/output-cap.d.ts.map +1 -0
- package/dist/sandbox/output-cap.js +66 -0
- package/dist/sandbox/output-cap.js.map +1 -0
- package/dist/sandbox/timeout.d.ts +7 -0
- package/dist/sandbox/timeout.d.ts.map +1 -0
- package/dist/sandbox/timeout.js +52 -0
- package/dist/sandbox/timeout.js.map +1 -0
- package/dist/sandbox/types.d.ts +18 -0
- package/dist/sandbox/types.d.ts.map +1 -0
- package/dist/sandbox/types.js +27 -0
- package/dist/sandbox/types.js.map +1 -0
- package/dist/sandbox/workdir.d.ts +9 -0
- package/dist/sandbox/workdir.d.ts.map +1 -0
- package/dist/sandbox/workdir.js +83 -0
- package/dist/sandbox/workdir.js.map +1 -0
- package/dist/state/index.d.ts +4 -0
- package/dist/state/index.d.ts.map +1 -0
- package/dist/state/index.js +4 -0
- package/dist/state/index.js.map +1 -0
- package/dist/state/paths.d.ts +5 -0
- package/dist/state/paths.d.ts.map +1 -0
- package/dist/state/paths.js +18 -0
- package/dist/state/paths.js.map +1 -0
- package/dist/state/persist.d.ts +14 -0
- package/dist/state/persist.d.ts.map +1 -0
- package/dist/state/persist.js +146 -0
- package/dist/state/persist.js.map +1 -0
- package/dist/state/transcript.d.ts +9 -0
- package/dist/state/transcript.d.ts.map +1 -0
- package/dist/state/transcript.js +34 -0
- package/dist/state/transcript.js.map +1 -0
- package/dist/tools/native/apply_patch.d.ts +18 -0
- package/dist/tools/native/apply_patch.d.ts.map +1 -0
- package/dist/tools/native/apply_patch.js +245 -0
- package/dist/tools/native/apply_patch.js.map +1 -0
- package/dist/tools/native/http_fetch.d.ts +6 -0
- package/dist/tools/native/http_fetch.d.ts.map +1 -0
- package/dist/tools/native/http_fetch.js +168 -0
- package/dist/tools/native/http_fetch.js.map +1 -0
- package/dist/tools/native/index.d.ts +9 -0
- package/dist/tools/native/index.d.ts.map +1 -0
- package/dist/tools/native/index.js +22 -0
- package/dist/tools/native/index.js.map +1 -0
- package/dist/tools/native/list_dir.d.ts +4 -0
- package/dist/tools/native/list_dir.d.ts.map +1 -0
- package/dist/tools/native/list_dir.js +175 -0
- package/dist/tools/native/list_dir.js.map +1 -0
- package/dist/tools/native/read_file.d.ts +4 -0
- package/dist/tools/native/read_file.d.ts.map +1 -0
- package/dist/tools/native/read_file.js +91 -0
- package/dist/tools/native/read_file.js.map +1 -0
- package/dist/tools/native/shell_exec.d.ts +4 -0
- package/dist/tools/native/shell_exec.d.ts.map +1 -0
- package/dist/tools/native/shell_exec.js +180 -0
- package/dist/tools/native/shell_exec.js.map +1 -0
- package/dist/tools/native/write_file.d.ts +4 -0
- package/dist/tools/native/write_file.d.ts.map +1 -0
- package/dist/tools/native/write_file.js +101 -0
- package/dist/tools/native/write_file.js.map +1 -0
- package/dist/tools/types.d.ts +48 -0
- package/dist/tools/types.d.ts.map +1 -0
- package/dist/tools/types.js +10 -0
- package/dist/tools/types.js.map +1 -0
- package/dist/types.d.ts +385 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +9 -0
- package/dist/types.js.map +1 -0
- package/examples/README.md +73 -0
- package/examples/creative-brief.md +34 -0
- package/examples/critic-disagreement-brief.md +42 -0
- package/examples/engineering-brief.md +35 -0
- package/examples/file-editing-brief.md +33 -0
- package/examples/mcp-brief.md +34 -0
- package/examples/multi-agent-brief.md +43 -0
- package/examples/research-brief.md +35 -0
- package/openwar.md +248 -0
- package/package.json +76 -0
- 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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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":"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"}
|