@northflare/runner 0.0.16 → 0.0.17
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/lib/codex-sdk/dist/src/codex.d.ts +27 -0
- package/lib/codex-sdk/dist/src/codex.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/codex.js +33 -0
- package/lib/codex-sdk/dist/src/codex.js.map +1 -0
- package/lib/codex-sdk/dist/src/codexOptions.d.ts +11 -0
- package/lib/codex-sdk/dist/src/codexOptions.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/codexOptions.js +2 -0
- package/lib/codex-sdk/dist/src/codexOptions.js.map +1 -0
- package/lib/codex-sdk/dist/src/events.d.ts +60 -0
- package/lib/codex-sdk/dist/src/events.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/events.js +3 -0
- package/lib/codex-sdk/dist/src/events.js.map +1 -0
- package/lib/codex-sdk/dist/src/exec.d.ts +27 -0
- package/lib/codex-sdk/dist/src/exec.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/exec.js +264 -0
- package/lib/codex-sdk/dist/src/exec.js.map +1 -0
- package/lib/codex-sdk/dist/src/index.d.ts +9 -0
- package/lib/codex-sdk/dist/src/index.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/index.js +3 -0
- package/lib/codex-sdk/dist/src/index.js.map +1 -0
- package/lib/codex-sdk/dist/src/items.d.ts +103 -0
- package/lib/codex-sdk/dist/src/items.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/items.js +3 -0
- package/lib/codex-sdk/dist/src/items.js.map +1 -0
- package/lib/codex-sdk/dist/src/outputSchemaFile.d.ts +6 -0
- package/lib/codex-sdk/dist/src/outputSchemaFile.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/outputSchemaFile.js +33 -0
- package/lib/codex-sdk/dist/src/outputSchemaFile.js.map +1 -0
- package/lib/codex-sdk/dist/src/thread.d.ts +41 -0
- package/lib/codex-sdk/dist/src/thread.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/thread.js +109 -0
- package/lib/codex-sdk/dist/src/thread.js.map +1 -0
- package/lib/codex-sdk/dist/src/threadOptions.d.ts +16 -0
- package/lib/codex-sdk/dist/src/threadOptions.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/threadOptions.js +2 -0
- package/lib/codex-sdk/dist/src/threadOptions.js.map +1 -0
- package/lib/codex-sdk/dist/src/turnOptions.d.ts +7 -0
- package/lib/codex-sdk/dist/src/turnOptions.d.ts.map +1 -0
- package/lib/codex-sdk/dist/src/turnOptions.js +2 -0
- package/lib/codex-sdk/dist/src/turnOptions.js.map +1 -0
- package/lib/codex-sdk/dist/tests/abort.test.d.ts +2 -0
- package/lib/codex-sdk/dist/tests/abort.test.d.ts.map +1 -0
- package/lib/codex-sdk/dist/tests/abort.test.js +130 -0
- package/lib/codex-sdk/dist/tests/abort.test.js.map +1 -0
- package/lib/codex-sdk/dist/tests/codexExecSpy.d.ts +6 -0
- package/lib/codex-sdk/dist/tests/codexExecSpy.d.ts.map +1 -0
- package/lib/codex-sdk/dist/tests/codexExecSpy.js +28 -0
- package/lib/codex-sdk/dist/tests/codexExecSpy.js.map +1 -0
- package/lib/codex-sdk/dist/tests/responsesProxy.d.ts +56 -0
- package/lib/codex-sdk/dist/tests/responsesProxy.d.ts.map +1 -0
- package/lib/codex-sdk/dist/tests/responsesProxy.js +155 -0
- package/lib/codex-sdk/dist/tests/responsesProxy.js.map +1 -0
- package/lib/codex-sdk/dist/tests/run.test.d.ts +2 -0
- package/lib/codex-sdk/dist/tests/run.test.d.ts.map +1 -0
- package/lib/codex-sdk/dist/tests/run.test.js +524 -0
- package/lib/codex-sdk/dist/tests/run.test.js.map +1 -0
- package/lib/codex-sdk/dist/tests/runStreamed.test.d.ts +2 -0
- package/lib/codex-sdk/dist/tests/runStreamed.test.d.ts.map +1 -0
- package/lib/codex-sdk/dist/tests/runStreamed.test.js +153 -0
- package/lib/codex-sdk/dist/tests/runStreamed.test.js.map +1 -0
- package/lib/codex-sdk/samples/basic_streaming.ts +0 -0
- package/lib/codex-sdk/samples/structured_output.ts +0 -0
- package/lib/codex-sdk/samples/structured_output_zod.ts +0 -0
- package/lib/codex-sdk/src/codex.ts +38 -0
- package/lib/codex-sdk/src/codexOptions.ts +10 -0
- package/lib/codex-sdk/src/events.ts +80 -0
- package/lib/codex-sdk/src/exec.ts +336 -0
- package/lib/codex-sdk/src/index.ts +39 -0
- package/lib/codex-sdk/src/items.ts +127 -0
- package/lib/codex-sdk/src/outputSchemaFile.ts +40 -0
- package/lib/codex-sdk/src/thread.ts +155 -0
- package/lib/codex-sdk/src/threadOptions.ts +18 -0
- package/lib/codex-sdk/src/turnOptions.ts +6 -0
- package/lib/codex-sdk/tests/abort.test.ts +165 -0
- package/lib/codex-sdk/tests/codexExecSpy.ts +37 -0
- package/lib/codex-sdk/tests/responsesProxy.ts +225 -0
- package/lib/codex-sdk/tests/run.test.ts +687 -0
- package/lib/codex-sdk/tests/runStreamed.test.ts +211 -0
- package/package.json +15 -16
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import path from "node:path";
|
|
2
|
+
import { describe, expect, it } from "@jest/globals";
|
|
3
|
+
import { Codex } from "../src/codex";
|
|
4
|
+
import { assistantMessage, responseCompleted, responseStarted, sse, startResponsesTestProxy, } from "./responsesProxy";
|
|
5
|
+
const codexExecPath = path.join(process.cwd(), "..", "..", "codex-rs", "target", "debug", "codex");
|
|
6
|
+
describe("Codex", () => {
|
|
7
|
+
it("returns thread events", async () => {
|
|
8
|
+
const { url, close } = await startResponsesTestProxy({
|
|
9
|
+
statusCode: 200,
|
|
10
|
+
responseBodies: [sse(responseStarted(), assistantMessage("Hi!"), responseCompleted())],
|
|
11
|
+
});
|
|
12
|
+
try {
|
|
13
|
+
const client = new Codex({ codexPathOverride: codexExecPath, baseUrl: url, apiKey: "test" });
|
|
14
|
+
const thread = client.startThread();
|
|
15
|
+
const result = await thread.runStreamed("Hello, world!");
|
|
16
|
+
const events = [];
|
|
17
|
+
for await (const event of result.events) {
|
|
18
|
+
events.push(event);
|
|
19
|
+
}
|
|
20
|
+
expect(events).toEqual([
|
|
21
|
+
{
|
|
22
|
+
type: "thread.started",
|
|
23
|
+
thread_id: expect.any(String),
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
type: "turn.started",
|
|
27
|
+
},
|
|
28
|
+
{
|
|
29
|
+
type: "item.completed",
|
|
30
|
+
item: {
|
|
31
|
+
id: "item_0",
|
|
32
|
+
type: "agent_message",
|
|
33
|
+
text: "Hi!",
|
|
34
|
+
},
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
type: "turn.completed",
|
|
38
|
+
usage: {
|
|
39
|
+
cached_input_tokens: 12,
|
|
40
|
+
input_tokens: 42,
|
|
41
|
+
output_tokens: 5,
|
|
42
|
+
},
|
|
43
|
+
},
|
|
44
|
+
]);
|
|
45
|
+
expect(thread.id).toEqual(expect.any(String));
|
|
46
|
+
}
|
|
47
|
+
finally {
|
|
48
|
+
await close();
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
it("sends previous items when runStreamed is called twice", async () => {
|
|
52
|
+
const { url, close, requests } = await startResponsesTestProxy({
|
|
53
|
+
statusCode: 200,
|
|
54
|
+
responseBodies: [
|
|
55
|
+
sse(responseStarted("response_1"), assistantMessage("First response", "item_1"), responseCompleted("response_1")),
|
|
56
|
+
sse(responseStarted("response_2"), assistantMessage("Second response", "item_2"), responseCompleted("response_2")),
|
|
57
|
+
],
|
|
58
|
+
});
|
|
59
|
+
try {
|
|
60
|
+
const client = new Codex({ codexPathOverride: codexExecPath, baseUrl: url, apiKey: "test" });
|
|
61
|
+
const thread = client.startThread();
|
|
62
|
+
const first = await thread.runStreamed("first input");
|
|
63
|
+
await drainEvents(first.events);
|
|
64
|
+
const second = await thread.runStreamed("second input");
|
|
65
|
+
await drainEvents(second.events);
|
|
66
|
+
// Check second request continues the same thread
|
|
67
|
+
expect(requests.length).toBeGreaterThanOrEqual(2);
|
|
68
|
+
const secondRequest = requests[1];
|
|
69
|
+
expect(secondRequest).toBeDefined();
|
|
70
|
+
const payload = secondRequest.json;
|
|
71
|
+
const assistantEntry = payload.input.find((entry) => entry.role === "assistant");
|
|
72
|
+
expect(assistantEntry).toBeDefined();
|
|
73
|
+
const assistantText = assistantEntry?.content?.find((item) => item.type === "output_text")?.text;
|
|
74
|
+
expect(assistantText).toBe("First response");
|
|
75
|
+
}
|
|
76
|
+
finally {
|
|
77
|
+
await close();
|
|
78
|
+
}
|
|
79
|
+
});
|
|
80
|
+
it("resumes thread by id when streaming", async () => {
|
|
81
|
+
const { url, close, requests } = await startResponsesTestProxy({
|
|
82
|
+
statusCode: 200,
|
|
83
|
+
responseBodies: [
|
|
84
|
+
sse(responseStarted("response_1"), assistantMessage("First response", "item_1"), responseCompleted("response_1")),
|
|
85
|
+
sse(responseStarted("response_2"), assistantMessage("Second response", "item_2"), responseCompleted("response_2")),
|
|
86
|
+
],
|
|
87
|
+
});
|
|
88
|
+
try {
|
|
89
|
+
const client = new Codex({ codexPathOverride: codexExecPath, baseUrl: url, apiKey: "test" });
|
|
90
|
+
const originalThread = client.startThread();
|
|
91
|
+
const first = await originalThread.runStreamed("first input");
|
|
92
|
+
await drainEvents(first.events);
|
|
93
|
+
const resumedThread = client.resumeThread(originalThread.id);
|
|
94
|
+
const second = await resumedThread.runStreamed("second input");
|
|
95
|
+
await drainEvents(second.events);
|
|
96
|
+
expect(resumedThread.id).toBe(originalThread.id);
|
|
97
|
+
expect(requests.length).toBeGreaterThanOrEqual(2);
|
|
98
|
+
const secondRequest = requests[1];
|
|
99
|
+
expect(secondRequest).toBeDefined();
|
|
100
|
+
const payload = secondRequest.json;
|
|
101
|
+
const assistantEntry = payload.input.find((entry) => entry.role === "assistant");
|
|
102
|
+
expect(assistantEntry).toBeDefined();
|
|
103
|
+
const assistantText = assistantEntry?.content?.find((item) => item.type === "output_text")?.text;
|
|
104
|
+
expect(assistantText).toBe("First response");
|
|
105
|
+
}
|
|
106
|
+
finally {
|
|
107
|
+
await close();
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
it("applies output schema turn options when streaming", async () => {
|
|
111
|
+
const { url, close, requests } = await startResponsesTestProxy({
|
|
112
|
+
statusCode: 200,
|
|
113
|
+
responseBodies: [
|
|
114
|
+
sse(responseStarted("response_1"), assistantMessage("Structured response", "item_1"), responseCompleted("response_1")),
|
|
115
|
+
],
|
|
116
|
+
});
|
|
117
|
+
const schema = {
|
|
118
|
+
type: "object",
|
|
119
|
+
properties: {
|
|
120
|
+
answer: { type: "string" },
|
|
121
|
+
},
|
|
122
|
+
required: ["answer"],
|
|
123
|
+
additionalProperties: false,
|
|
124
|
+
};
|
|
125
|
+
try {
|
|
126
|
+
const client = new Codex({ codexPathOverride: codexExecPath, baseUrl: url, apiKey: "test" });
|
|
127
|
+
const thread = client.startThread();
|
|
128
|
+
const streamed = await thread.runStreamed("structured", { outputSchema: schema });
|
|
129
|
+
await drainEvents(streamed.events);
|
|
130
|
+
expect(requests.length).toBeGreaterThanOrEqual(1);
|
|
131
|
+
const payload = requests[0];
|
|
132
|
+
expect(payload).toBeDefined();
|
|
133
|
+
const text = payload.json.text;
|
|
134
|
+
expect(text).toBeDefined();
|
|
135
|
+
expect(text?.format).toEqual({
|
|
136
|
+
name: "codex_output_schema",
|
|
137
|
+
type: "json_schema",
|
|
138
|
+
strict: true,
|
|
139
|
+
schema,
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
finally {
|
|
143
|
+
await close();
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
async function drainEvents(events) {
|
|
148
|
+
let done = false;
|
|
149
|
+
do {
|
|
150
|
+
done = (await events.next()).done ?? false;
|
|
151
|
+
} while (!done);
|
|
152
|
+
}
|
|
153
|
+
//# sourceMappingURL=runStreamed.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runStreamed.test.js","sourceRoot":"","sources":["../../tests/runStreamed.test.ts"],"names":[],"mappings":"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"}
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { CodexOptions } from "./codexOptions";
|
|
2
|
+
import { CodexExec } from "./exec";
|
|
3
|
+
import { Thread } from "./thread";
|
|
4
|
+
import { ThreadOptions } from "./threadOptions";
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Codex is the main class for interacting with the Codex agent.
|
|
8
|
+
*
|
|
9
|
+
* Use the `startThread()` method to start a new thread or `resumeThread()` to resume a previously started thread.
|
|
10
|
+
*/
|
|
11
|
+
export class Codex {
|
|
12
|
+
private exec: CodexExec;
|
|
13
|
+
private options: CodexOptions;
|
|
14
|
+
|
|
15
|
+
constructor(options: CodexOptions = {}) {
|
|
16
|
+
this.exec = new CodexExec(options.codexPathOverride, options.env);
|
|
17
|
+
this.options = options;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Starts a new conversation with an agent.
|
|
22
|
+
* @returns A new thread instance.
|
|
23
|
+
*/
|
|
24
|
+
startThread(options: ThreadOptions = {}): Thread {
|
|
25
|
+
return new Thread(this.exec, this.options, options);
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Resumes a conversation with an agent based on the thread id.
|
|
30
|
+
* Threads are persisted in ~/.codex/sessions.
|
|
31
|
+
*
|
|
32
|
+
* @param id The id of the thread to resume.
|
|
33
|
+
* @returns A new thread instance.
|
|
34
|
+
*/
|
|
35
|
+
resumeThread(id: string, options: ThreadOptions = {}): Thread {
|
|
36
|
+
return new Thread(this.exec, this.options, options, id);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export type CodexOptions = {
|
|
2
|
+
codexPathOverride?: string;
|
|
3
|
+
baseUrl?: string;
|
|
4
|
+
apiKey?: string;
|
|
5
|
+
/**
|
|
6
|
+
* Environment variables passed to the Codex CLI process. When provided, the SDK
|
|
7
|
+
* will not inherit variables from `process.env`.
|
|
8
|
+
*/
|
|
9
|
+
env?: Record<string, string>;
|
|
10
|
+
};
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
// based on event types from codex-rs/exec/src/exec_events.rs
|
|
2
|
+
|
|
3
|
+
import type { ThreadItem } from "./items";
|
|
4
|
+
|
|
5
|
+
/** Emitted when a new thread is started as the first event. */
|
|
6
|
+
export type ThreadStartedEvent = {
|
|
7
|
+
type: "thread.started";
|
|
8
|
+
/** The identifier of the new thread. Can be used to resume the thread later. */
|
|
9
|
+
thread_id: string;
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Emitted when a turn is started by sending a new prompt to the model.
|
|
14
|
+
* A turn encompasses all events that happen while the agent is processing the prompt.
|
|
15
|
+
*/
|
|
16
|
+
export type TurnStartedEvent = {
|
|
17
|
+
type: "turn.started";
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
/** Describes the usage of tokens during a turn. */
|
|
21
|
+
export type Usage = {
|
|
22
|
+
/** The number of input tokens used during the turn. */
|
|
23
|
+
input_tokens: number;
|
|
24
|
+
/** The number of cached input tokens used during the turn. */
|
|
25
|
+
cached_input_tokens: number;
|
|
26
|
+
/** The number of output tokens used during the turn. */
|
|
27
|
+
output_tokens: number;
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
/** Emitted when a turn is completed. Typically right after the assistant's response. */
|
|
31
|
+
export type TurnCompletedEvent = {
|
|
32
|
+
type: "turn.completed";
|
|
33
|
+
usage: Usage;
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
/** Indicates that a turn failed with an error. */
|
|
37
|
+
export type TurnFailedEvent = {
|
|
38
|
+
type: "turn.failed";
|
|
39
|
+
error: ThreadError;
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
/** Emitted when a new item is added to the thread. Typically the item is initially "in progress". */
|
|
43
|
+
export type ItemStartedEvent = {
|
|
44
|
+
type: "item.started";
|
|
45
|
+
item: ThreadItem;
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
/** Emitted when an item is updated. */
|
|
49
|
+
export type ItemUpdatedEvent = {
|
|
50
|
+
type: "item.updated";
|
|
51
|
+
item: ThreadItem;
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
/** Signals that an item has reached a terminal state—either success or failure. */
|
|
55
|
+
export type ItemCompletedEvent = {
|
|
56
|
+
type: "item.completed";
|
|
57
|
+
item: ThreadItem;
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
/** Fatal error emitted by the stream. */
|
|
61
|
+
export type ThreadError = {
|
|
62
|
+
message: string;
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
/** Represents an unrecoverable error emitted directly by the event stream. */
|
|
66
|
+
export type ThreadErrorEvent = {
|
|
67
|
+
type: "error";
|
|
68
|
+
message: string;
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
/** Top-level JSONL events emitted by codex exec. */
|
|
72
|
+
export type ThreadEvent =
|
|
73
|
+
| ThreadStartedEvent
|
|
74
|
+
| TurnStartedEvent
|
|
75
|
+
| TurnCompletedEvent
|
|
76
|
+
| TurnFailedEvent
|
|
77
|
+
| ItemStartedEvent
|
|
78
|
+
| ItemUpdatedEvent
|
|
79
|
+
| ItemCompletedEvent
|
|
80
|
+
| ThreadErrorEvent;
|
|
@@ -0,0 +1,336 @@
|
|
|
1
|
+
import { spawn } from "node:child_process";
|
|
2
|
+
import { existsSync } from "node:fs";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
import readline from "node:readline";
|
|
5
|
+
import { fileURLToPath } from "node:url";
|
|
6
|
+
import { createRequire } from "node:module";
|
|
7
|
+
|
|
8
|
+
import { SandboxMode, ModelReasoningEffort, ApprovalMode } from "./threadOptions";
|
|
9
|
+
|
|
10
|
+
export type CodexExecArgs = {
|
|
11
|
+
input: string;
|
|
12
|
+
|
|
13
|
+
baseUrl?: string;
|
|
14
|
+
apiKey?: string;
|
|
15
|
+
threadId?: string | null;
|
|
16
|
+
images?: string[];
|
|
17
|
+
// --model
|
|
18
|
+
model?: string;
|
|
19
|
+
// --sandbox
|
|
20
|
+
sandboxMode?: SandboxMode;
|
|
21
|
+
// --cd
|
|
22
|
+
workingDirectory?: string;
|
|
23
|
+
// --add-dir
|
|
24
|
+
additionalDirectories?: string[];
|
|
25
|
+
// --skip-git-repo-check
|
|
26
|
+
skipGitRepoCheck?: boolean;
|
|
27
|
+
// --output-schema
|
|
28
|
+
outputSchemaFile?: string;
|
|
29
|
+
// --config model_reasoning_effort
|
|
30
|
+
modelReasoningEffort?: ModelReasoningEffort;
|
|
31
|
+
// AbortSignal to cancel the execution
|
|
32
|
+
signal?: AbortSignal;
|
|
33
|
+
// --config sandbox_workspace_write.network_access
|
|
34
|
+
networkAccessEnabled?: boolean;
|
|
35
|
+
// --config features.web_search_request
|
|
36
|
+
webSearchEnabled?: boolean;
|
|
37
|
+
// --config approval_policy
|
|
38
|
+
approvalPolicy?: ApprovalMode;
|
|
39
|
+
// Generic config overrides for `-c key=value`
|
|
40
|
+
configOverrides?: Record<string, unknown>;
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
const INTERNAL_ORIGINATOR_ENV = "CODEX_INTERNAL_ORIGINATOR_OVERRIDE";
|
|
44
|
+
const TYPESCRIPT_SDK_ORIGINATOR = "codex_sdk_ts";
|
|
45
|
+
const isDebugEnabled = process.env["DEBUG"] === "true";
|
|
46
|
+
|
|
47
|
+
export class CodexExec {
|
|
48
|
+
private executablePath: string;
|
|
49
|
+
private envOverride?: Record<string, string>;
|
|
50
|
+
|
|
51
|
+
constructor(executablePath: string | null = null, env?: Record<string, string>) {
|
|
52
|
+
this.executablePath = executablePath || findCodexPath();
|
|
53
|
+
this.envOverride = env;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
async *run(args: CodexExecArgs): AsyncGenerator<string> {
|
|
57
|
+
const commandArgs: string[] = ["exec", "--experimental-json"];
|
|
58
|
+
|
|
59
|
+
if (args.model) {
|
|
60
|
+
commandArgs.push("--model", args.model);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
if (args.sandboxMode) {
|
|
64
|
+
commandArgs.push("--sandbox", args.sandboxMode);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
if (args.workingDirectory) {
|
|
68
|
+
commandArgs.push("--cd", args.workingDirectory);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
if (args.additionalDirectories?.length) {
|
|
72
|
+
for (const dir of args.additionalDirectories) {
|
|
73
|
+
commandArgs.push("--add-dir", dir);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if (args.skipGitRepoCheck) {
|
|
78
|
+
commandArgs.push("--skip-git-repo-check");
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
if (args.outputSchemaFile) {
|
|
82
|
+
commandArgs.push("--output-schema", args.outputSchemaFile);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
if (args.modelReasoningEffort) {
|
|
86
|
+
commandArgs.push("--config", `model_reasoning_effort="${args.modelReasoningEffort}"`);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
if (args.networkAccessEnabled !== undefined) {
|
|
90
|
+
commandArgs.push(
|
|
91
|
+
"--config",
|
|
92
|
+
`sandbox_workspace_write.network_access=${args.networkAccessEnabled}`,
|
|
93
|
+
);
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
if (args.webSearchEnabled !== undefined) {
|
|
97
|
+
commandArgs.push("--config", `features.web_search_request=${args.webSearchEnabled}`);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
if (args.approvalPolicy) {
|
|
101
|
+
commandArgs.push("--config", `approval_policy="${args.approvalPolicy}"`);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (args.configOverrides) {
|
|
105
|
+
for (const [key, value] of Object.entries(args.configOverrides)) {
|
|
106
|
+
const formatted = formatConfigValue(value);
|
|
107
|
+
commandArgs.push("--config", `${key}=${formatted}`);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
if (args.images?.length) {
|
|
112
|
+
for (const image of args.images) {
|
|
113
|
+
commandArgs.push("--image", image);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
if (args.threadId) {
|
|
118
|
+
commandArgs.push("resume", args.threadId);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
if (isDebugEnabled) {
|
|
122
|
+
console.log("[CodexExec] Executable path:", this.executablePath);
|
|
123
|
+
console.log("[CodexExec] Command arguments:", commandArgs);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
const env: Record<string, string> = {};
|
|
127
|
+
if (this.envOverride) {
|
|
128
|
+
Object.assign(env, this.envOverride);
|
|
129
|
+
} else {
|
|
130
|
+
for (const [key, value] of Object.entries(process.env)) {
|
|
131
|
+
if (value !== undefined) {
|
|
132
|
+
env[key] = value;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
if (!env[INTERNAL_ORIGINATOR_ENV]) {
|
|
137
|
+
env[INTERNAL_ORIGINATOR_ENV] = TYPESCRIPT_SDK_ORIGINATOR;
|
|
138
|
+
}
|
|
139
|
+
if (args.baseUrl) {
|
|
140
|
+
env.OPENAI_BASE_URL = args.baseUrl;
|
|
141
|
+
}
|
|
142
|
+
if (args.apiKey) {
|
|
143
|
+
env.CODEX_API_KEY = args.apiKey;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
const child = spawn(this.executablePath, commandArgs, {
|
|
147
|
+
env,
|
|
148
|
+
signal: args.signal,
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
let spawnError: unknown | null = null;
|
|
152
|
+
child.once("error", (err) => (spawnError = err));
|
|
153
|
+
|
|
154
|
+
if (!child.stdin) {
|
|
155
|
+
child.kill();
|
|
156
|
+
throw new Error("Child process has no stdin");
|
|
157
|
+
}
|
|
158
|
+
child.stdin.write(args.input);
|
|
159
|
+
child.stdin.end();
|
|
160
|
+
|
|
161
|
+
if (!child.stdout) {
|
|
162
|
+
child.kill();
|
|
163
|
+
throw new Error("Child process has no stdout");
|
|
164
|
+
}
|
|
165
|
+
const stderrChunks: Buffer[] = [];
|
|
166
|
+
|
|
167
|
+
if (child.stderr) {
|
|
168
|
+
child.stderr.on("data", (data) => {
|
|
169
|
+
stderrChunks.push(data);
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
const rl = readline.createInterface({
|
|
174
|
+
input: child.stdout,
|
|
175
|
+
crlfDelay: Infinity,
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
try {
|
|
179
|
+
for await (const line of rl) {
|
|
180
|
+
// `line` is a string (Node sets default encoding to utf8 for readline)
|
|
181
|
+
yield line as string;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
const exitCode = new Promise((resolve, reject) => {
|
|
185
|
+
child.once("exit", (code) => {
|
|
186
|
+
if (code === 0) {
|
|
187
|
+
resolve(code);
|
|
188
|
+
} else {
|
|
189
|
+
const stderrBuffer = Buffer.concat(stderrChunks);
|
|
190
|
+
reject(
|
|
191
|
+
new Error(`Codex Exec exited with code ${code}: ${stderrBuffer.toString("utf8")}`),
|
|
192
|
+
);
|
|
193
|
+
}
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
if (spawnError) throw spawnError;
|
|
198
|
+
await exitCode;
|
|
199
|
+
} finally {
|
|
200
|
+
rl.close();
|
|
201
|
+
child.removeAllListeners();
|
|
202
|
+
try {
|
|
203
|
+
if (!child.killed) child.kill();
|
|
204
|
+
} catch {
|
|
205
|
+
// ignore
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
const scriptFileName = fileURLToPath(import.meta.url);
|
|
212
|
+
const scriptDirName = path.dirname(scriptFileName);
|
|
213
|
+
const require = createRequire(import.meta.url);
|
|
214
|
+
|
|
215
|
+
function findCodexPath() {
|
|
216
|
+
const { platform, arch } = process;
|
|
217
|
+
|
|
218
|
+
let targetTriple = null;
|
|
219
|
+
switch (platform) {
|
|
220
|
+
case "linux":
|
|
221
|
+
case "android":
|
|
222
|
+
switch (arch) {
|
|
223
|
+
case "x64":
|
|
224
|
+
targetTriple = "x86_64-unknown-linux-musl";
|
|
225
|
+
break;
|
|
226
|
+
case "arm64":
|
|
227
|
+
targetTriple = "aarch64-unknown-linux-musl";
|
|
228
|
+
break;
|
|
229
|
+
default:
|
|
230
|
+
break;
|
|
231
|
+
}
|
|
232
|
+
break;
|
|
233
|
+
case "darwin":
|
|
234
|
+
switch (arch) {
|
|
235
|
+
case "x64":
|
|
236
|
+
targetTriple = "x86_64-apple-darwin";
|
|
237
|
+
break;
|
|
238
|
+
case "arm64":
|
|
239
|
+
targetTriple = "aarch64-apple-darwin";
|
|
240
|
+
break;
|
|
241
|
+
default:
|
|
242
|
+
break;
|
|
243
|
+
}
|
|
244
|
+
break;
|
|
245
|
+
case "win32":
|
|
246
|
+
switch (arch) {
|
|
247
|
+
case "x64":
|
|
248
|
+
targetTriple = "x86_64-pc-windows-msvc";
|
|
249
|
+
break;
|
|
250
|
+
case "arm64":
|
|
251
|
+
targetTriple = "aarch64-pc-windows-msvc";
|
|
252
|
+
break;
|
|
253
|
+
default:
|
|
254
|
+
break;
|
|
255
|
+
}
|
|
256
|
+
break;
|
|
257
|
+
default:
|
|
258
|
+
break;
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
if (!targetTriple) {
|
|
262
|
+
throw new Error(`Unsupported platform: ${platform} (${arch})`);
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
const vendorResolution = resolveUpstreamVendorRoot();
|
|
266
|
+
let vendorRoot: string;
|
|
267
|
+
if (vendorResolution.path) {
|
|
268
|
+
vendorRoot = vendorResolution.path;
|
|
269
|
+
} else {
|
|
270
|
+
vendorRoot = path.join(scriptDirName, "..", "vendor");
|
|
271
|
+
const reason = vendorResolution.reason ? `: ${vendorResolution.reason}` : "";
|
|
272
|
+
console.warn(
|
|
273
|
+
`[Codex SDK] Failed to resolve upstream vendor package${reason}, falling back to local vendor directory`,
|
|
274
|
+
);
|
|
275
|
+
}
|
|
276
|
+
const archRoot = path.join(vendorRoot, targetTriple);
|
|
277
|
+
const codexBinaryName = process.platform === "win32" ? "codex.exe" : "codex";
|
|
278
|
+
const binaryPath = path.join(archRoot, "codex", codexBinaryName);
|
|
279
|
+
|
|
280
|
+
return binaryPath;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
function resolveUpstreamVendorRoot(): { path: string | null; reason?: string } {
|
|
284
|
+
let searchPaths: string[] | null = null;
|
|
285
|
+
try {
|
|
286
|
+
searchPaths = require.resolve.paths("@openai/codex-sdk");
|
|
287
|
+
} catch (error) {
|
|
288
|
+
return { path: null, reason: (error as Error)?.message || String(error) };
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
if (!searchPaths?.length) {
|
|
292
|
+
return { path: null, reason: "no module search paths returned" };
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
let locatedPackageRoot: string | null = null;
|
|
296
|
+
for (const basePath of searchPaths) {
|
|
297
|
+
if (!basePath) continue;
|
|
298
|
+
const packageRoot = path.join(basePath, "@openai/codex-sdk");
|
|
299
|
+
const packageJsonPath = path.join(packageRoot, "package.json");
|
|
300
|
+
if (!existsSync(packageJsonPath)) {
|
|
301
|
+
continue;
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
locatedPackageRoot = packageRoot;
|
|
305
|
+
const vendorPath = path.join(packageRoot, "vendor");
|
|
306
|
+
if (existsSync(vendorPath)) {
|
|
307
|
+
return { path: vendorPath };
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
if (locatedPackageRoot) {
|
|
312
|
+
return { path: null, reason: `vendor directory missing in ${locatedPackageRoot}` };
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
return {
|
|
316
|
+
path: null,
|
|
317
|
+
reason: `@openai/codex-sdk not found in node_modules search paths (${searchPaths.join(", ")})`,
|
|
318
|
+
};
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
function formatConfigValue(value: unknown): string {
|
|
322
|
+
if (typeof value === "string") {
|
|
323
|
+
return JSON.stringify(value);
|
|
324
|
+
}
|
|
325
|
+
if (typeof value === "number" || typeof value === "boolean") {
|
|
326
|
+
return String(value);
|
|
327
|
+
}
|
|
328
|
+
if (value === null || value === undefined) {
|
|
329
|
+
return "null";
|
|
330
|
+
}
|
|
331
|
+
try {
|
|
332
|
+
return JSON.stringify(value);
|
|
333
|
+
} catch {
|
|
334
|
+
return JSON.stringify(String(value));
|
|
335
|
+
}
|
|
336
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
export type {
|
|
2
|
+
ThreadEvent,
|
|
3
|
+
ThreadStartedEvent,
|
|
4
|
+
TurnStartedEvent,
|
|
5
|
+
TurnCompletedEvent,
|
|
6
|
+
TurnFailedEvent,
|
|
7
|
+
ItemStartedEvent,
|
|
8
|
+
ItemUpdatedEvent,
|
|
9
|
+
ItemCompletedEvent,
|
|
10
|
+
ThreadError,
|
|
11
|
+
ThreadErrorEvent,
|
|
12
|
+
Usage,
|
|
13
|
+
} from "./events";
|
|
14
|
+
export type {
|
|
15
|
+
ThreadItem,
|
|
16
|
+
AgentMessageItem,
|
|
17
|
+
ReasoningItem,
|
|
18
|
+
CommandExecutionItem,
|
|
19
|
+
FileChangeItem,
|
|
20
|
+
McpToolCallItem,
|
|
21
|
+
WebSearchItem,
|
|
22
|
+
TodoListItem,
|
|
23
|
+
ErrorItem,
|
|
24
|
+
} from "./items";
|
|
25
|
+
|
|
26
|
+
export { Thread } from "./thread";
|
|
27
|
+
export type { RunResult, RunStreamedResult, Input, UserInput } from "./thread";
|
|
28
|
+
|
|
29
|
+
export { Codex } from "./codex";
|
|
30
|
+
|
|
31
|
+
export type { CodexOptions } from "./codexOptions";
|
|
32
|
+
|
|
33
|
+
export type {
|
|
34
|
+
ThreadOptions,
|
|
35
|
+
ApprovalMode,
|
|
36
|
+
SandboxMode,
|
|
37
|
+
ModelReasoningEffort,
|
|
38
|
+
} from "./threadOptions";
|
|
39
|
+
export type { TurnOptions } from "./turnOptions";
|