-
Notifications
You must be signed in to change notification settings - Fork 485
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Experimental - Stateful code interpreter #326
Changes from all commits
aed6370
37a9489
c91fa10
fded067
6d130be
9379395
4a99814
66182c8
d7aa184
0fc18ea
8766860
ea8b13a
2f3ef86
3e5e928
c370f9d
604845b
0befb64
2a0de7b
13500a9
a160308
325a33d
3dd2e92
e9cfdd2
d252f2d
d9eac38
4142a5e
72bb5b9
7f0a8f8
67efd48
a39cbac
1491f5a
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,85 @@ | ||
name: SDK Release Candidate | ||
|
||
on: | ||
pull_request: | ||
|
||
permissions: | ||
contents: write | ||
|
||
jobs: | ||
release: | ||
name: Release Candidate | ||
runs-on: ubuntu-latest | ||
|
||
steps: | ||
- name: Checkout Repo | ||
uses: actions/checkout@v4 | ||
with: | ||
ref: ${{ github.head_ref }} | ||
|
||
- uses: pnpm/action-setup@v3 | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'js-rc') }} | ||
|
||
|
||
- name: Setup Node.js 20 | ||
uses: actions/setup-node@v4 | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'js-rc') || contains( github.event.pull_request.labels.*.name, 'python-rc') }} | ||
with: | ||
node-version: 20 | ||
registry-url: 'https://registry.npmjs.org' | ||
cache: pnpm | ||
|
||
- name: Configure pnpm | ||
working-directory: packages/js-sdk | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'js-rc') }} | ||
run: | | ||
pnpm config set auto-install-peers true | ||
pnpm config set exclude-links-from-lockfile true | ||
|
||
- name: Install dependencies | ||
working-directory: packages/js-sdk | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'js-rc') }} | ||
run: pnpm install --frozen-lockfile | ||
|
||
- name: Release Candidate | ||
working-directory: packages/js-sdk | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'js-rc') }} | ||
run: | | ||
npm version prerelease --preid=${{ github.head_ref }} | ||
npm publish --tag rc | ||
env: | ||
NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }} | ||
|
||
- name: Set up Python | ||
uses: actions/setup-python@v4 | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'python-rc') }} | ||
with: | ||
python-version: "3.10" | ||
|
||
- name: Install and configure Poetry | ||
uses: snok/install-poetry@v1 | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'python-rc') }} | ||
with: | ||
version: 1.5.1 | ||
virtualenvs-create: true | ||
virtualenvs-in-project: true | ||
installer-parallel: true | ||
|
||
- name: Release Candidate | ||
if: ${{ contains( github.event.pull_request.labels.*.name, 'python-rc') }} | ||
working-directory: packages/python-sdk | ||
run: | | ||
poetry version prerelease | ||
poetry build | ||
poetry config pypi-token.pypi ${PYPI_TOKEN} && poetry publish --skip-existing | ||
env: | ||
PYPI_TOKEN: ${{ secrets.PYPI_TOKEN }} | ||
|
||
- name: Commit new versions | ||
run: | | ||
git config user.name "github-actions[bot]" | ||
git config user.email "github-actions[bot]@users.noreply.github.com" | ||
git commit -am "[skip ci] Release new versions" || exit 0 | ||
git push | ||
env: | ||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
@@ -0,0 +1,268 @@ | ||||||
import { Sandbox, SandboxOpts } from '../sandbox' | ||||||
import { ProcessMessage } from '../sandbox/process' | ||||||
import { randomBytes } from 'crypto' | ||||||
import IWebSocket from 'isomorphic-ws' | ||||||
|
||||||
interface ExecutionError { | ||||||
name: string | ||||||
value: string | ||||||
traceback: string[] | ||||||
} | ||||||
|
||||||
interface Result { | ||||||
output?: string | ||||||
stdout: string[] | ||||||
stderr: string[] | ||||||
error?: ExecutionError | ||||||
display_data: object[] | ||||||
} | ||||||
|
||||||
export class CodeInterpreterV2 extends Sandbox { | ||||||
private static template = 'code-interpreter-stateful' | ||||||
private jupyterKernelID?: string | ||||||
|
||||||
/** | ||||||
* Use `DataAnalysis.create()` instead. | ||||||
* | ||||||
* @hidden | ||||||
* @hide | ||||||
* @internal | ||||||
* @access protected | ||||||
*/ | ||||||
constructor(opts: SandboxOpts) { | ||||||
super({ template: opts.template || CodeInterpreterV2.template, ...opts }) | ||||||
} | ||||||
|
||||||
/** | ||||||
* Creates a new Sandbox from the template. | ||||||
* @returns New Sandbox | ||||||
*/ | ||||||
static override async create(): Promise<CodeInterpreterV2> | ||||||
/** | ||||||
* Creates a new Sandbox from the specified options. | ||||||
* @param opts Sandbox options | ||||||
* @returns New Sandbox | ||||||
*/ | ||||||
static override async create(opts: SandboxOpts): Promise<CodeInterpreterV2> | ||||||
static override async create(opts?: SandboxOpts) { | ||||||
const sandbox = new CodeInterpreterV2({ ...(opts ? opts : {}) }) | ||||||
await sandbox._open({ timeout: opts?.timeout }) | ||||||
sandbox.jupyterKernelID = await sandbox.getKernelID() | ||||||
|
||||||
return sandbox | ||||||
} | ||||||
|
||||||
|
||||||
/** | ||||||
* Reconnects to an existing Sandbox. | ||||||
* @param sandboxID Sandbox ID | ||||||
* @returns Existing Sandbox | ||||||
* | ||||||
* @example | ||||||
* ```ts | ||||||
* const sandbox = await Sandbox.create() | ||||||
* const sandboxID = sandbox.id | ||||||
* | ||||||
* await sandbox.keepAlive(300 * 1000) | ||||||
* await sandbox.close() | ||||||
* | ||||||
* const reconnectedSandbox = await Sandbox.reconnect(sandboxID) | ||||||
* ``` | ||||||
*/ | ||||||
static async reconnect<S extends typeof CodeInterpreterV2>(this: S, sandboxID: string): Promise<InstanceType<S>> | ||||||
/** | ||||||
* Reconnects to an existing Sandbox. | ||||||
* @param opts Sandbox options | ||||||
* @returns Existing Sandbox | ||||||
* | ||||||
* @example | ||||||
* ```ts | ||||||
* const sandbox = await Sandbox.create() | ||||||
* const sandboxID = sandbox.id | ||||||
* | ||||||
* await sandbox.keepAlive(300 * 1000) | ||||||
* await sandbox.close() | ||||||
* | ||||||
* const reconnectedSandbox = await Sandbox.reconnect({ | ||||||
* sandboxID, | ||||||
* }) | ||||||
* ``` | ||||||
*/ | ||||||
static async reconnect<S extends typeof CodeInterpreterV2>(this: S, opts: Omit<SandboxOpts, 'id' | 'template'> & { sandboxID: string }): Promise<InstanceType<S>> | ||||||
static async reconnect<S extends typeof CodeInterpreterV2>(this: S, sandboxIDorOpts: string | Omit<SandboxOpts, 'id' | 'template'> & { sandboxID: string }): Promise<InstanceType<S>> { | ||||||
let id: string | ||||||
let opts: SandboxOpts | ||||||
if (typeof sandboxIDorOpts === 'string') { | ||||||
id = sandboxIDorOpts | ||||||
opts = {} | ||||||
} else { | ||||||
id = sandboxIDorOpts.sandboxID | ||||||
opts = sandboxIDorOpts | ||||||
} | ||||||
|
||||||
const sandboxIDAndClientID = id.split('-') | ||||||
const sandboxID = sandboxIDAndClientID[0] | ||||||
const clientID = sandboxIDAndClientID[1] | ||||||
opts.__sandbox = { sandboxID, clientID, templateID: 'unknown' } | ||||||
|
||||||
const sandbox = new this(opts) as InstanceType<S> | ||||||
await sandbox._open({ timeout: opts?.timeout }) | ||||||
|
||||||
sandbox.jupyterKernelID = await sandbox.getKernelID() | ||||||
return sandbox | ||||||
} | ||||||
|
||||||
private static sendExecuteRequest(code: string) { | ||||||
const msg_id = randomBytes(16).toString('hex') | ||||||
const session = randomBytes(16).toString('hex') | ||||||
return { | ||||||
header: { | ||||||
msg_id: msg_id, | ||||||
username: 'e2b', | ||||||
session: session, | ||||||
msg_type: 'execute_request', | ||||||
version: '5.3', | ||||||
}, | ||||||
parent_header: {}, | ||||||
metadata: {}, | ||||||
content: { | ||||||
code: code, | ||||||
silent: false, | ||||||
store_history: false, | ||||||
user_expressions: {}, | ||||||
allow_stdin: false, | ||||||
}, | ||||||
} | ||||||
} | ||||||
|
||||||
async execPython( | ||||||
code: string, | ||||||
onStdout?: (out: ProcessMessage) => Promise<void> | void, | ||||||
onStderr?: (out: ProcessMessage) => Promise<void> | void, | ||||||
) { | ||||||
let resolve: () => void | ||||||
|
||||||
const finished = new Promise<void>((r) => { | ||||||
resolve = () => r() | ||||||
}) | ||||||
const result: Result = { | ||||||
stdout: [], | ||||||
stderr: [], | ||||||
display_data: [], | ||||||
} | ||||||
|
||||||
// @ts-ignore | ||||||
const ws = await this._connectKernel(result, resolve, onStdout, onStderr) | ||||||
ws.send(JSON.stringify(CodeInterpreterV2.sendExecuteRequest(code))) | ||||||
await finished | ||||||
|
||||||
ws.close() | ||||||
return result | ||||||
} | ||||||
|
||||||
private async getKernelID() { | ||||||
return await this.filesystem.read('/root/.jupyter/kernel_id') | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
There was an extra newline at the end on cloudflare workers durable objects. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks @lawrencecchen , all the fixes will be incorporated in this repo https://github.com/e2b-dev/code-interpreter |
||||||
} | ||||||
|
||||||
private async _connectKernel( | ||||||
result: Result, | ||||||
finish: () => void, | ||||||
onStdout?: (out: ProcessMessage) => Promise<void> | void, | ||||||
onStderr?: (out: ProcessMessage) => Promise<void> | void, | ||||||
) { | ||||||
const ws = new IWebSocket( | ||||||
`${this.getProtocol('ws')}://${this.getHostname(8888)}/api/kernels/${ | ||||||
this.jupyterKernelID | ||||||
}/channels`, | ||||||
) | ||||||
|
||||||
const opened = new Promise<void>((resolve) => { | ||||||
ws.onopen = () => resolve() | ||||||
}) | ||||||
await opened | ||||||
|
||||||
const helperData = { | ||||||
input_accepted: false, | ||||||
finish, | ||||||
} | ||||||
|
||||||
ws.onmessage = (e) => { | ||||||
this.onMessage(result, helperData, e.data.toString(), { | ||||||
onStdout, | ||||||
onStderr, | ||||||
}) | ||||||
} | ||||||
|
||||||
return ws | ||||||
} | ||||||
|
||||||
private onMessage( | ||||||
result: Result, | ||||||
helperData: { input_accepted: boolean; finish: () => void }, | ||||||
data: string, | ||||||
opts?: { | ||||||
onStdout?: (out: ProcessMessage) => Promise<void> | void | ||||||
onStderr?: (out: ProcessMessage) => Promise<void> | void | ||||||
}, | ||||||
) { | ||||||
const message = JSON.parse(data) | ||||||
if (message.msg_type == 'error') { | ||||||
result.error = { | ||||||
name: message.content.ename, | ||||||
value: message.content.evalue, | ||||||
traceback: message.content.traceback, | ||||||
} | ||||||
} else if (message.msg_type == 'stream') { | ||||||
if (message.content.name == 'stdout') { | ||||||
result.stdout.push(message.content.text) | ||||||
if (opts?.onStdout) { | ||||||
opts.onStdout({ | ||||||
line: message.content.text, | ||||||
timestamp: new Date().getTime() * 1_000_000, | ||||||
error: false, | ||||||
}) | ||||||
} | ||||||
} else if (message.content.name == 'stderr') { | ||||||
result.stderr.push(message.content.text) | ||||||
if (opts?.onStderr) { | ||||||
opts.onStderr({ | ||||||
line: message.content.text, | ||||||
timestamp: new Date().getTime() * 1_000_000, | ||||||
error: true, | ||||||
}) | ||||||
} | ||||||
} | ||||||
} else if (message.msg_type == 'display_data') { | ||||||
result.display_data.push(message.content.data) | ||||||
} else if (message.msg_type == 'execute_result') { | ||||||
result.output = message.content.data['text/plain'] | ||||||
} else if (message.msg_type == 'status') { | ||||||
if (message.content.execution_state == 'idle') { | ||||||
if (helperData.input_accepted) { | ||||||
helperData.finish() | ||||||
} | ||||||
} else if (message.content.execution_state == 'error') { | ||||||
result.error = { | ||||||
name: message.content.ename, | ||||||
value: message.content.evalue, | ||||||
traceback: message.content.traceback, | ||||||
} | ||||||
helperData.finish() | ||||||
} | ||||||
} else if (message.msg_type == 'execute_reply') { | ||||||
if (message.content.status == 'error') { | ||||||
result.error = { | ||||||
name: message.content.ename, | ||||||
value: message.content.evalue, | ||||||
traceback: message.content.traceback, | ||||||
} | ||||||
} else if (message.content.status == 'ok') { | ||||||
return | ||||||
} | ||||||
} else if (message.msg_type == 'execute_input') { | ||||||
helperData.input_accepted = true | ||||||
} else { | ||||||
console.log('[UNHANDLED MESSAGE TYPE]:', message.msg_type) | ||||||
} | ||||||
} | ||||||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please use web crypto for edge environment support (cf workers)