A TypeScript/JavaScript SDK for interacting with the Codex process. This SDK provides a simple interface for managing Codex sessions, sending messages, and handling responses.
This SDK works with the native (Rust) version of Codex from the codex-rs repository. You can install it either globally or locally in your project.
The SDK requires an OpenAI API key to function. By default, it looks for the OPENAI_API_KEY
environment variable. You can:
- Set it in your environment:
export OPENAI_API_KEY='your-api-key'
- Or provide it through the session configuration (see Quick Start example below)
You have two options for installing Codex:
- Global installation (recommended for most users):
npm i -g @openai/codex@native
- Local installation (if you prefer to keep it project-specific):
npm install @openai/codex@native
Then, install the SDK:
npm install codex-js-sdk
import { CodexSDK, LogLevel } from 'codex-js-sdk';
import { CodexResponse, CodexMessageTypeEnum, ModelReasoningEffort, ModelReasoningSummary, SandboxPermission, AskForApproval } from 'codex-js-sdk';
// Create a new SDK instance
const sdk = new CodexSDK({
// Optional: Set custom working directory, could be relative to the current working directory (process.cwd())
cwd: './my-project',
// Optional: Configure logging level
logLevel: LogLevel.DEBUG,
// Optional: Specify custom path to codex binary (if installed locally)
codexPath: './node_modules/.bin/codex',
// Optional: Initial session configuration
// NOTE: Better to use configureSession method instead
config: {
model: 'codex-mini-latest'
},
// Optional: Set custom environment variables (by default, the SDK will use the process.env)
env: {
OPENAI_API_KEY: 'sk-proj-...'
}
});
// Set up response handler
sdk.onResponse((response: CodexResponse) => {
console.log('Received response:', response);
const msg = response.msg;
// Handle different response types
switch (msg.type) {
case CodexMessageTypeEnum.EXEC_APPROVAL_REQUEST: {
console.log('\nCommand execution requested:', msg.command);
// Handle command approval
sdk.handleCommand(response.id, true);
break;
}
case CodexMessageTypeEnum.APPLY_PATCH_APPROVAL_REQUEST: {
console.log('\nPatch requested:', msg.changes);
// Handle patch approval
sdk.handlePatch(response.id, true);
break;
}
case CodexMessageTypeEnum.TASK_COMPLETE: {
console.log('\nTask complete:', msg.last_agent_message);
// You can now send a new message or stop the SDK
sdk.stop();
break;
}
case CodexMessageTypeEnum.ERROR: {
console.error('\nError occurred:', msg.message);
// Handle error
break;
}
}
});
// Set up error handler
sdk.onError((response: CodexResponse) => {
console.error('Error:', response);
});
// Start the Codex process (if not started yet)
sdk.start();
// Optional: Configure session with detailed settings
// You can also use `~/.codex/config.toml` to configure the codex (https://github.com/openai/codex/blob/main/codex-rs/config.md)
await sdk.configureSession({
// Provide some instructions for the model
instructions: 'You are a helpful coding assistant, your name is Flexbe AI Bot. Provide concise and clear responses.',
// Configure the model provider (OpenAI by default)
model: 'claude-3-7-sonnet-latest',
provider: {
name: 'Anthropic',
base_url: 'https://api.anthropic.com/v1',
env_key: 'ANTHROPIC_API_KEY',
env_key_instructions: 'Create an API key (https://console.anthropic.com) and export it as an environment variable.',
wire_api: WireApi.CHAT
},
approval_policy: AskForApproval.UNLESS_ALLOW_LISTED,
sandbox_policy: { permissions: [SandboxPermission.DISK_WRITE_CWD] },
cwd: process.cwd()
});
// Send a text message
sdk.sendUserMessage([
{ type: 'text', text: 'Hello, can you help me with my code?' }
], 'run-id');
// Send a message with both text and image
sdk.sendUserMessage([
{ type: 'text', text: 'Can you analyze this screenshot?' },
// Option 1: Using a URL
{
type: 'image',
image_url: 'https://example.com/screenshot.png'
},
// Option 2: Using a local file path
{
type: 'local_image',
path: './screenshots/error.png'
}
], 'run-id-with-image');
// Handle command approvals (if not auto-approved)
sdk.handleCommand('run-id', true); // Approve a command
sdk.handleCommand('run-id', false); // Reject a command
// Handle patch approvals (if not auto-approved)
sdk.handlePatch('run-id', true); // Approve a patch
sdk.handlePatch('run-id', false); // Reject a patch
// Abort a request if needed
sdk.abort('run-id');
// Stop the process when done
sdk.stop();
// Handle process cleanup
process.on('SIGINT', () => {
console.log('\nStopping SDK...');
sdk.stop();
process.exit(0);
});
The main class for interacting with Codex.
new CodexSDK(options?: CodexProcessOptions)
Options:
-
cwd?: string
- Working directory for the Codex process -
env?: NodeJS.ProcessEnv
- Environment variables -
apiKey?: string
- OpenAI API key (defaults to OPENAI_API_KEY env variable) -
session?: SessionConfig
- Session configuration -
logLevel?: LogLevel
- Logging level (defaults to INFO) -
codexPath?: string
- Custom path to the codex binary (if not provided, will look for 'codex' in PATH)
Starts the Codex process if it's not already running.
Stops the Codex process if it's running.
Restarts the Codex process.
Updates the session configuration.
sdk.configure({
instructions: 'New instructions',
model: 'codex-mini-latest'
});
Sends a message to Codex. The message can contain text and/or images.
// InputItem type definition
type InputItem =
| { type: 'text'; text: string }
| { type: 'image'; image_url: string } // Image from URL
| { type: 'local_image'; path: string }; // Image from local file
// Examples:
// Send text only
sdk.sendUserMessage([
{ type: 'text', text: 'Hello' }
]);
// Send local image
sdk.sendUserMessage([
{
type: 'local_image',
path: './screenshots/error.png'
}
]);
// Send both text and image
const requestId = sdk.sendUserMessage([
{ type: 'text', text: 'What is your name?' },
{ type: 'text', text: 'What is in this picture?' },
{ type: 'image', image_url: '' }
]);
Parameters:
-
items: InputItem[]
- Array of input items (text messages and/or images) -
runId?: string
- Optional unique identifier for the message run. If not provided, a new UUID will be generated
Note: For images, you can either:
- Provide a URL using
type: 'image'
withimage_url
- Provide a local file path using
type: 'local_image'
withpath
(relative to the working directory)
Handles command execution requests.
// Approve command for current session only
sdk.handleCommand(requestId, true, true);
// Approve command permanently
sdk.handleCommand(requestId, true);
// Reject command
sdk.handleCommand(requestId, false);
Handles patch application requests.
// Approve patch for current session only
sdk.handlePatch(requestId, true, true);
// Approve patch permanently
sdk.handlePatch(requestId, true);
// Reject patch
sdk.handlePatch(requestId, false);
Aborts the current operation.
sdk.abort(requestId);
Registers a callback for Codex responses. The response can be of different types:
// Common response types
type CodexResponse = {
id: string;
msg: {
type: CodexMessageTypeEnum;
// ... other fields depending on type
}
};
// Example response handler
sdk.onResponse((response: CodexResponse) => {
switch (response.msg.type) {
case CodexMessageTypeEnum.TASK_COMPLETE:
console.log('Task complete:', response.msg.summary);
// Task is done, you can:
// - Send a new message
// - Stop the SDK
// - Process the results
break;
case CodexMessageTypeEnum.ERROR:
console.error('Error occurred:', response.msg.message);
// Handle error, e.g.:
// - Stop the SDK
// - Notify the user
break;
case CodexMessageTypeEnum.EXEC_APPROVAL_REQUEST:
// Handle command approval request
break;
case CodexMessageTypeEnum.APPLY_PATCH_APPROVAL_REQUEST:
// Handle patch approval request
break;
case CodexMessageTypeEnum.TASK_STARTED:
console.log('Task started:', response.msg.task_id);
break;
case CodexMessageTypeEnum.AGENT_MESSAGE:
console.log('Agent message:', response.msg.content);
break;
// ... other response types
}
});
Available logging levels:
-
LogLevel.ERROR
- Error messages only -
LogLevel.WARN
- Warning and error messages -
LogLevel.INFO
- Info, warning, and error messages -
LogLevel.DEBUG
- All messages including debug information
The SDK throws errors in the following cases:
- When trying to send a message while Codex is not started
- When Codex's stdin is not writable
- When there are issues with the Codex process
Example error handling:
try {
sdk.sendUserMessage([{ type: 'text', text: 'Hello' }]);
} catch (error) {
console.error('Failed to send message:', error);
}
The SDK uses the following environment variables:
-
OPENAI_API_KEY
- Your OpenAI API key (required for authentication)
MIT