@sena-ai/runtime-codex 0.0.1

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.
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=mapper.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mapper.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/mapper.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,126 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { mapCodexNotification } from '../mapper.js';
3
+ describe('mapCodexNotification', () => {
4
+ it('maps agentMessage delta to progress.delta', () => {
5
+ const event = mapCodexNotification('item/agentMessage/delta', {
6
+ threadId: 'thr_1', turnId: 'turn_1', itemId: 'item_1',
7
+ delta: 'Hello ',
8
+ });
9
+ expect(event).toEqual({ type: 'progress.delta', text: 'Hello ' });
10
+ });
11
+ it('maps commandExecution item/started to tool.start', () => {
12
+ const event = mapCodexNotification('item/started', {
13
+ item: { type: 'commandExecution', command: 'ls -la' },
14
+ });
15
+ expect(event).toEqual({ type: 'tool.start', toolName: 'shell:ls -la' });
16
+ });
17
+ it('maps commandExecution item/completed to tool.end', () => {
18
+ const event = mapCodexNotification('item/completed', {
19
+ item: { type: 'commandExecution', command: 'npm test', exitCode: 0 },
20
+ });
21
+ expect(event).toEqual({ type: 'tool.end', toolName: 'shell:npm test', isError: false });
22
+ });
23
+ it('maps failed command to tool.end with isError', () => {
24
+ const event = mapCodexNotification('item/completed', {
25
+ item: { type: 'commandExecution', command: 'bad-cmd', exitCode: 1 },
26
+ });
27
+ expect(event).toEqual({ type: 'tool.end', toolName: 'shell:bad-cmd', isError: true });
28
+ });
29
+ it('maps turn/completed success to result', () => {
30
+ const event = mapCodexNotification('turn/completed', {
31
+ turn: {
32
+ status: 'completed',
33
+ items: [
34
+ { type: 'agentMessage', content: [{ type: 'text', text: 'Done!' }] },
35
+ ],
36
+ },
37
+ });
38
+ expect(event).toEqual({ type: 'result', text: 'Done!' });
39
+ });
40
+ it('maps turn/completed failed to error', () => {
41
+ const event = mapCodexNotification('turn/completed', {
42
+ turn: { status: 'failed', error: 'Context window exceeded' },
43
+ });
44
+ expect(event).toEqual({ type: 'error', message: 'Context window exceeded' });
45
+ });
46
+ it('maps "error" notification (ErrorNotification format)', () => {
47
+ const event = mapCodexNotification('error', {
48
+ error: { message: 'Rate limit exceeded' },
49
+ willRetry: false,
50
+ threadId: 'thr_1',
51
+ turnId: 'turn_1',
52
+ });
53
+ expect(event).toEqual({ type: 'error', message: 'Rate limit exceeded' });
54
+ });
55
+ it('returns null for unrelated notifications', () => {
56
+ expect(mapCodexNotification('thread/name/updated', {})).toBeNull();
57
+ expect(mapCodexNotification('account/updated', {})).toBeNull();
58
+ });
59
+ it('maps fileChange item/started to tool.start', () => {
60
+ const event = mapCodexNotification('item/started', {
61
+ item: { type: 'fileChange', changes: [{ path: '/tmp/test.ts' }] }
62
+ });
63
+ expect(event).toEqual({ type: 'tool.start', toolName: 'file:/tmp/test.ts' });
64
+ });
65
+ it('maps fileChange item/completed to tool.end', () => {
66
+ const event = mapCodexNotification('item/completed', {
67
+ item: { type: 'fileChange', changes: [{ path: '/tmp/test.ts' }] }
68
+ });
69
+ expect(event).toEqual({ type: 'tool.end', toolName: 'file:/tmp/test.ts', isError: false });
70
+ });
71
+ it('maps mcpToolCall item/started to tool.start', () => {
72
+ const event = mapCodexNotification('item/started', {
73
+ item: { type: 'mcpToolCall', server: 'slack', tool: 'post_message' }
74
+ });
75
+ expect(event).toEqual({ type: 'tool.start', toolName: 'mcp:slack/post_message' });
76
+ });
77
+ it('maps mcpToolCall item/completed to tool.end', () => {
78
+ const event = mapCodexNotification('item/completed', {
79
+ item: { type: 'mcpToolCall', server: 'slack', tool: 'post_message', error: null }
80
+ });
81
+ expect(event).toEqual({ type: 'tool.end', toolName: 'mcp:slack/post_message', isError: false });
82
+ });
83
+ it('maps mcpToolCall item/completed with error', () => {
84
+ const event = mapCodexNotification('item/completed', {
85
+ item: { type: 'mcpToolCall', server: 'slack', tool: 'post_message', error: { message: 'timeout' } }
86
+ });
87
+ expect(event).toEqual({ type: 'tool.end', toolName: 'mcp:slack/post_message', isError: true });
88
+ });
89
+ it('maps dynamicToolCall item/started to tool.start', () => {
90
+ const event = mapCodexNotification('item/started', {
91
+ item: { type: 'dynamicToolCall', tool: 'custom_search' }
92
+ });
93
+ expect(event).toEqual({ type: 'tool.start', toolName: 'tool:custom_search' });
94
+ });
95
+ it('maps dynamicToolCall item/completed to tool.end', () => {
96
+ const event = mapCodexNotification('item/completed', {
97
+ item: { type: 'dynamicToolCall', tool: 'custom_search', success: true }
98
+ });
99
+ expect(event).toEqual({ type: 'tool.end', toolName: 'tool:custom_search', isError: false });
100
+ });
101
+ it('maps dynamicToolCall item/completed with failure', () => {
102
+ const event = mapCodexNotification('item/completed', {
103
+ item: { type: 'dynamicToolCall', tool: 'custom_search', success: false }
104
+ });
105
+ expect(event).toEqual({ type: 'tool.end', toolName: 'tool:custom_search', isError: true });
106
+ });
107
+ it('returns null for item/started with unrecognized type', () => {
108
+ expect(mapCodexNotification('item/started', { item: { type: 'userMessage' } })).toBeNull();
109
+ });
110
+ it('returns null for item/completed with null item', () => {
111
+ expect(mapCodexNotification('item/completed', {})).toBeNull();
112
+ });
113
+ it('handles turn/completed with empty items array', () => {
114
+ const event = mapCodexNotification('turn/completed', {
115
+ turn: { status: 'completed', items: [] }
116
+ });
117
+ expect(event).toEqual({ type: 'result', text: '' });
118
+ });
119
+ it('extracts text from agentMessage in item/completed', () => {
120
+ const event = mapCodexNotification('item/completed', {
121
+ item: { type: 'agentMessage', content: [{ type: 'text', text: 'hello' }, { type: 'text', text: ' world' }] }
122
+ });
123
+ expect(event).toEqual({ type: 'progress', text: 'hello world' });
124
+ });
125
+ });
126
+ //# sourceMappingURL=mapper.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mapper.test.js","sourceRoot":"","sources":["../../src/__tests__/mapper.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,42 @@
1
+ import { EventEmitter } from 'node:events';
2
+ export type JsonRpcMessage = {
3
+ id?: number;
4
+ method?: string;
5
+ params?: unknown;
6
+ result?: unknown;
7
+ error?: {
8
+ code: number;
9
+ message: string;
10
+ data?: unknown;
11
+ };
12
+ };
13
+ export declare class CodexAppServerClient extends EventEmitter {
14
+ private child;
15
+ private rl;
16
+ private nextId;
17
+ private pending;
18
+ private codexBin;
19
+ constructor(codexBin?: string);
20
+ spawn(): void;
21
+ private onLine;
22
+ private send;
23
+ request(method: string, params: object): Promise<unknown>;
24
+ notify(method: string, params?: object): void;
25
+ respond(id: number, result: unknown): void;
26
+ initialize(clientName?: string, version?: string): Promise<unknown>;
27
+ threadStart(params: {
28
+ model?: string;
29
+ cwd?: string;
30
+ approvalPolicy?: string;
31
+ sandbox?: Record<string, unknown>;
32
+ baseInstructions?: string;
33
+ }): Promise<{
34
+ threadId: string;
35
+ }>;
36
+ threadResume(threadId: string, params?: object): Promise<unknown>;
37
+ turnStart(threadId: string, text: string, params?: object): Promise<{
38
+ turnId: string;
39
+ }>;
40
+ close(): void;
41
+ }
42
+ //# sourceMappingURL=client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAA;AAE1C,MAAM,MAAM,cAAc,GAAG;IAC3B,EAAE,CAAC,EAAE,MAAM,CAAA;IACX,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,MAAM,CAAC,EAAE,OAAO,CAAA;IAChB,MAAM,CAAC,EAAE,OAAO,CAAA;IAChB,KAAK,CAAC,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,OAAO,CAAA;KAAE,CAAA;CAC1D,CAAA;AAOD,qBAAa,oBAAqB,SAAQ,YAAY;IACpD,OAAO,CAAC,KAAK,CAA4B;IACzC,OAAO,CAAC,EAAE,CAAyB;IACnC,OAAO,CAAC,MAAM,CAAI;IAClB,OAAO,CAAC,OAAO,CAAoC;IACnD,OAAO,CAAC,QAAQ,CAAQ;gBAEZ,QAAQ,SAAU;IAK9B,KAAK,IAAI,IAAI;IAYb,OAAO,CAAC,MAAM;IAmCd,OAAO,CAAC,IAAI;IAKZ,OAAO,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAQzD,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAIjD,OAAO,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,GAAG,IAAI;IAIpC,UAAU,CAAC,UAAU,SAAiB,EAAE,OAAO,SAAU,GAAG,OAAO,CAAC,OAAO,CAAC;IAS5E,WAAW,CAAC,MAAM,EAAE;QACxB,KAAK,CAAC,EAAE,MAAM,CAAA;QACd,GAAG,CAAC,EAAE,MAAM,CAAA;QACZ,cAAc,CAAC,EAAE,MAAM,CAAA;QACvB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;QACjC,gBAAgB,CAAC,EAAE,MAAM,CAAA;KAC1B,GAAG,OAAO,CAAC;QAAE,QAAQ,EAAE,MAAM,CAAA;KAAE,CAAC;IAQ3B,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,OAAO,CAAC,OAAO,CAAC;IASrE,SAAS,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,OAAO,CAAC;QAAE,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC;IAUjG,KAAK,IAAI,IAAI;CAUd"}
package/dist/client.js ADDED
@@ -0,0 +1,118 @@
1
+ import { spawn } from 'node:child_process';
2
+ import { createInterface } from 'node:readline';
3
+ import { EventEmitter } from 'node:events';
4
+ export class CodexAppServerClient extends EventEmitter {
5
+ child = null;
6
+ rl = null;
7
+ nextId = 0;
8
+ pending = new Map();
9
+ codexBin;
10
+ constructor(codexBin = 'codex') {
11
+ super();
12
+ this.codexBin = codexBin;
13
+ }
14
+ spawn() {
15
+ this.child = spawn(this.codexBin, ['app-server'], {
16
+ stdio: ['pipe', 'pipe', 'inherit'],
17
+ });
18
+ this.rl = createInterface({ input: this.child.stdout });
19
+ this.rl.on('line', (line) => this.onLine(line));
20
+ this.child.on('error', (err) => this.emit('spawn-error', err));
21
+ this.child.on('exit', (code) => this.emit('exit', code));
22
+ }
23
+ onLine(line) {
24
+ if (!line.trim())
25
+ return;
26
+ let msg;
27
+ try {
28
+ msg = JSON.parse(line);
29
+ }
30
+ catch {
31
+ return;
32
+ }
33
+ // Response to a client request
34
+ if (msg.id !== undefined && this.pending.has(msg.id)) {
35
+ const { resolve, reject } = this.pending.get(msg.id);
36
+ this.pending.delete(msg.id);
37
+ if (msg.error) {
38
+ reject(new Error(msg.error.message));
39
+ }
40
+ else {
41
+ resolve(msg.result);
42
+ }
43
+ return;
44
+ }
45
+ // Server request (has id but no pending — requires client response)
46
+ if (msg.id !== undefined && msg.method) {
47
+ this.emit('server-request', msg);
48
+ return;
49
+ }
50
+ // Server notification (no id)
51
+ if (msg.method) {
52
+ // Avoid emitting raw method names that start with 'error' — Node.js
53
+ // treats unhandled 'error' events as fatal. Use 'notification' only.
54
+ this.emit('notification', msg);
55
+ }
56
+ }
57
+ send(msg) {
58
+ if (!this.child?.stdin?.writable)
59
+ throw new Error('Client not connected');
60
+ this.child.stdin.write(JSON.stringify(msg) + '\n');
61
+ }
62
+ request(method, params) {
63
+ const id = ++this.nextId;
64
+ return new Promise((resolve, reject) => {
65
+ this.pending.set(id, { resolve, reject });
66
+ this.send({ id, method, params });
67
+ });
68
+ }
69
+ notify(method, params = {}) {
70
+ this.send({ method, params });
71
+ }
72
+ respond(id, result) {
73
+ this.send({ id, result });
74
+ }
75
+ async initialize(clientName = 'sena-runtime', version = '0.1.0') {
76
+ const result = await this.request('initialize', {
77
+ clientInfo: { name: clientName, version },
78
+ capabilities: { experimentalApi: true },
79
+ });
80
+ this.notify('initialized');
81
+ return result;
82
+ }
83
+ async threadStart(params) {
84
+ const result = await this.request('thread/start', {
85
+ ...params,
86
+ persistExtendedHistory: true,
87
+ });
88
+ return { threadId: result.thread.id };
89
+ }
90
+ async threadResume(threadId, params = {}) {
91
+ return this.request('thread/resume', {
92
+ threadId,
93
+ ...params,
94
+ experimentalRawEvents: false,
95
+ persistExtendedHistory: true,
96
+ });
97
+ }
98
+ async turnStart(threadId, text, params = {}) {
99
+ const result = await this.request('turn/start', {
100
+ threadId,
101
+ input: [{ type: 'text', text }],
102
+ persistExtendedHistory: true,
103
+ ...params,
104
+ });
105
+ return { turnId: result.turn.id };
106
+ }
107
+ close() {
108
+ this.rl?.close();
109
+ this.child?.kill();
110
+ this.child = null;
111
+ this.rl = null;
112
+ for (const [, { reject }] of this.pending) {
113
+ reject(new Error('Client closed'));
114
+ }
115
+ this.pending.clear();
116
+ }
117
+ }
118
+ //# sourceMappingURL=client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.js","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export { codexRuntime } from './runtime.js';
2
+ export type { CodexRuntimeOptions } from './runtime.js';
3
+ export { CodexAppServerClient } from './client.js';
4
+ export { mapCodexNotification } from './mapper.js';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAA;AAC3C,YAAY,EAAE,mBAAmB,EAAE,MAAM,cAAc,CAAA;AACvD,OAAO,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAA;AAClD,OAAO,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAA"}
package/dist/index.js ADDED
@@ -0,0 +1,4 @@
1
+ export { codexRuntime } from './runtime.js';
2
+ export { CodexAppServerClient } from './client.js';
3
+ export { mapCodexNotification } from './mapper.js';
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAA;AAE3C,OAAO,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAA;AAClD,OAAO,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAA"}
@@ -0,0 +1,3 @@
1
+ import type { RuntimeEvent } from '@sena-ai/core';
2
+ export declare function mapCodexNotification(method: string, params: any): RuntimeEvent | null;
3
+ //# sourceMappingURL=mapper.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mapper.d.ts","sourceRoot":"","sources":["../src/mapper.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,eAAe,CAAA;AAEjD,wBAAgB,oBAAoB,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,GAAG,YAAY,GAAG,IAAI,CAuFrF"}
package/dist/mapper.js ADDED
@@ -0,0 +1,87 @@
1
+ export function mapCodexNotification(method, params) {
2
+ switch (method) {
3
+ case 'item/agentMessage/delta':
4
+ return { type: 'progress.delta', text: params.delta ?? '' };
5
+ case 'item/started': {
6
+ const item = params.item;
7
+ const itemType = item?.type ?? params.type;
8
+ switch (itemType) {
9
+ case 'commandExecution':
10
+ return { type: 'tool.start', toolName: `shell:${item?.command ?? 'unknown'}` };
11
+ case 'fileChange':
12
+ return { type: 'tool.start', toolName: `file:${item?.changes?.[0]?.path ?? 'unknown'}` };
13
+ case 'mcpToolCall':
14
+ return { type: 'tool.start', toolName: `mcp:${item?.server ?? 'unknown'}/${item?.tool ?? 'unknown'}` };
15
+ case 'dynamicToolCall':
16
+ return { type: 'tool.start', toolName: `tool:${item?.tool ?? 'unknown'}` };
17
+ default:
18
+ return null;
19
+ }
20
+ }
21
+ case 'item/completed': {
22
+ const item = params.item;
23
+ if (!item)
24
+ return null;
25
+ switch (item.type) {
26
+ case 'commandExecution':
27
+ return {
28
+ type: 'tool.end',
29
+ toolName: `shell:${item.command ?? 'unknown'}`,
30
+ isError: item.exitCode != null ? item.exitCode !== 0 : false,
31
+ };
32
+ case 'fileChange':
33
+ return {
34
+ type: 'tool.end',
35
+ toolName: `file:${item.changes?.[0]?.path ?? 'unknown'}`,
36
+ isError: false,
37
+ };
38
+ case 'mcpToolCall':
39
+ return {
40
+ type: 'tool.end',
41
+ toolName: `mcp:${item.server ?? 'unknown'}/${item.tool ?? 'unknown'}`,
42
+ isError: item.error != null,
43
+ };
44
+ case 'dynamicToolCall':
45
+ return {
46
+ type: 'tool.end',
47
+ toolName: `tool:${item.tool ?? 'unknown'}`,
48
+ isError: item.success === false,
49
+ };
50
+ case 'agentMessage': {
51
+ const text = item.content
52
+ ?.filter((b) => b.type === 'text')
53
+ ?.map((b) => b.text)
54
+ ?.join('') ?? '';
55
+ if (text)
56
+ return { type: 'progress', text };
57
+ return null;
58
+ }
59
+ default:
60
+ return null;
61
+ }
62
+ }
63
+ case 'turn/completed': {
64
+ const turn = params.turn;
65
+ if (!turn)
66
+ return null;
67
+ if (turn.status === 'completed') {
68
+ const agentItems = (turn.items ?? []).filter((i) => i.type === 'agentMessage');
69
+ const lastMsg = agentItems[agentItems.length - 1];
70
+ const text = lastMsg?.content
71
+ ?.filter((b) => b.type === 'text')
72
+ ?.map((b) => b.text)
73
+ ?.join('') ?? '';
74
+ return { type: 'result', text };
75
+ }
76
+ if (turn.status === 'failed') {
77
+ return { type: 'error', message: turn.error ?? 'Turn failed' };
78
+ }
79
+ return { type: 'error', message: 'Turn interrupted' };
80
+ }
81
+ case 'error':
82
+ return { type: 'error', message: params.error?.message ?? 'Unknown error' };
83
+ default:
84
+ return null;
85
+ }
86
+ }
87
+ //# sourceMappingURL=mapper.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mapper.js","sourceRoot":"","sources":["../src/mapper.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { Runtime } from '@sena-ai/core';
2
+ export type CodexRuntimeOptions = {
3
+ model?: string;
4
+ apiKey?: string;
5
+ reasoningEffort?: 'minimal' | 'low' | 'medium' | 'high' | 'xhigh';
6
+ sandboxMode?: 'read-only' | 'workspace-write' | 'danger-full-access';
7
+ approvalPolicy?: 'never' | 'on-request' | 'always';
8
+ codexBin?: string;
9
+ };
10
+ export declare function codexRuntime(options?: CodexRuntimeOptions): Runtime;
11
+ //# sourceMappingURL=runtime.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"runtime.d.ts","sourceRoot":"","sources":["../src/runtime.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAuD,MAAM,eAAe,CAAA;AAIjG,MAAM,MAAM,mBAAmB,GAAG;IAChC,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,eAAe,CAAC,EAAE,SAAS,GAAG,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,OAAO,CAAA;IACjE,WAAW,CAAC,EAAE,WAAW,GAAG,iBAAiB,GAAG,oBAAoB,CAAA;IACpE,cAAc,CAAC,EAAE,OAAO,GAAG,YAAY,GAAG,QAAQ,CAAA;IAClD,QAAQ,CAAC,EAAE,MAAM,CAAA;CAClB,CAAA;AAED,wBAAgB,YAAY,CAAC,OAAO,GAAE,mBAAwB,GAAG,OAAO,CAiIvE"}
@@ -0,0 +1,135 @@
1
+ import { CodexAppServerClient } from './client.js';
2
+ import { mapCodexNotification } from './mapper.js';
3
+ export function codexRuntime(options = {}) {
4
+ const { model, apiKey, reasoningEffort = 'medium', sandboxMode = 'danger-full-access', approvalPolicy = 'never', codexBin = 'codex', } = options;
5
+ return {
6
+ name: 'codex',
7
+ async *createStream(streamOptions) {
8
+ const { contextFragments, prompt: promptIterable, sessionId, cwd, abortSignal, } = streamOptions;
9
+ if (apiKey) {
10
+ process.env.OPENAI_API_KEY = apiKey;
11
+ }
12
+ const client = new CodexAppServerClient(codexBin);
13
+ const eventQueue = [];
14
+ let resolveWait = null;
15
+ let turnDone = false;
16
+ function pushEvent(event) {
17
+ eventQueue.push(event);
18
+ resolveWait?.();
19
+ }
20
+ client.on('notification', (msg) => {
21
+ const event = mapCodexNotification(msg.method, msg.params);
22
+ if (event)
23
+ pushEvent(event);
24
+ if (msg.method === 'turn/completed') {
25
+ turnDone = true;
26
+ resolveWait?.();
27
+ }
28
+ if (msg.method === 'error') {
29
+ turnDone = true;
30
+ resolveWait?.();
31
+ }
32
+ });
33
+ // Server requests requiring client response (approval, input, etc.)
34
+ client.on('server-request', (msg) => {
35
+ switch (msg.method) {
36
+ // Official approval request methods per ServerRequest.ts
37
+ case 'item/commandExecution/requestApproval':
38
+ case 'item/fileChange/requestApproval':
39
+ case 'item/permissions/requestApproval':
40
+ case 'applyPatchApproval':
41
+ case 'execCommandApproval': {
42
+ const decision = approvalPolicy === 'never' ? 'acceptForSession' : 'accept';
43
+ client.respond(msg.id, { decision });
44
+ break;
45
+ }
46
+ default:
47
+ // Unknown server request — accept to avoid blocking
48
+ client.respond(msg.id, { decision: 'accept' });
49
+ break;
50
+ }
51
+ });
52
+ abortSignal.addEventListener('abort', () => {
53
+ client.close();
54
+ turnDone = true;
55
+ resolveWait?.();
56
+ }, { once: true });
57
+ try {
58
+ client.spawn();
59
+ await client.initialize('sena-runtime');
60
+ const baseInstructions = buildBaseInstructions(contextFragments);
61
+ const resolvedModel = streamOptions.model || model;
62
+ const threadParams = {
63
+ cwd: cwd || process.cwd(),
64
+ approvalPolicy,
65
+ sandbox: sandboxModeToCodex(sandboxMode),
66
+ baseInstructions,
67
+ };
68
+ if (resolvedModel)
69
+ threadParams.model = resolvedModel;
70
+ let threadId;
71
+ if (sessionId) {
72
+ await client.threadResume(sessionId, threadParams);
73
+ threadId = sessionId;
74
+ }
75
+ else {
76
+ const thread = await client.threadStart(threadParams);
77
+ threadId = thread.threadId;
78
+ pushEvent({ type: 'session.init', sessionId: threadId });
79
+ }
80
+ let userText = '';
81
+ for await (const msg of promptIterable) {
82
+ userText = msg.text;
83
+ break;
84
+ }
85
+ await client.turnStart(threadId, userText);
86
+ while (!turnDone) {
87
+ while (eventQueue.length > 0) {
88
+ yield eventQueue.shift();
89
+ }
90
+ if (turnDone)
91
+ break;
92
+ await new Promise((resolve) => {
93
+ resolveWait = resolve;
94
+ if (eventQueue.length > 0 || turnDone)
95
+ resolve();
96
+ });
97
+ }
98
+ while (eventQueue.length > 0) {
99
+ yield eventQueue.shift();
100
+ }
101
+ }
102
+ finally {
103
+ client.close();
104
+ }
105
+ },
106
+ };
107
+ }
108
+ function buildBaseInstructions(fragments) {
109
+ const parts = [];
110
+ for (const f of fragments.filter(f => f.role === 'system')) {
111
+ parts.push(`[${f.source}]\n${f.content}`);
112
+ }
113
+ for (const f of fragments.filter(f => f.role === 'context')) {
114
+ parts.push(`[${f.source}]\n${f.content}`);
115
+ }
116
+ return parts.join('\n\n');
117
+ }
118
+ /**
119
+ * Convert a simple sandbox mode string to the tagged-union SandboxPolicy
120
+ * format required by the Codex App Server protocol.
121
+ * @see SandboxPolicy.ts from `codex app-server generate-ts`
122
+ */
123
+ function sandboxModeToCodex(mode) {
124
+ switch (mode) {
125
+ case 'danger-full-access':
126
+ return { type: 'danger-full-access' };
127
+ case 'read-only':
128
+ return { type: 'read-only' };
129
+ case 'workspace-write':
130
+ return { type: 'workspace-write', network_access: false, exclude_tmpdir_env_var: false, exclude_slash_tmp: false };
131
+ default:
132
+ return { type: 'workspace-write', network_access: false, exclude_tmpdir_env_var: false, exclude_slash_tmp: false };
133
+ }
134
+ }
135
+ //# sourceMappingURL=runtime.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"runtime.js","sourceRoot":"","sources":["../src/runtime.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,22 @@
1
+ {
2
+ "name": "@sena-ai/runtime-codex",
3
+ "version": "0.0.1",
4
+ "type": "module",
5
+ "exports": {
6
+ ".": {
7
+ "types": "./dist/index.d.ts",
8
+ "import": "./dist/index.js"
9
+ }
10
+ },
11
+ "files": ["dist"],
12
+ "scripts": {
13
+ "build": "tsc -b",
14
+ "dev": "tsc -b --watch"
15
+ },
16
+ "dependencies": {
17
+ "@sena-ai/core": "workspace:*"
18
+ },
19
+ "devDependencies": {
20
+ "typescript": "^5.8.0"
21
+ }
22
+ }