-
Notifications
You must be signed in to change notification settings - Fork 928
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(client): add Realtime API support (#1266)
- Loading branch information
1 parent
66067d3
commit a796d21
Showing
11 changed files
with
560 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
import { OpenAIRealtimeWebSocket } from 'openai/beta/realtime/websocket'; | ||
|
||
async function main() { | ||
const rt = new OpenAIRealtimeWebSocket({ model: 'gpt-4o-realtime-preview-2024-12-17' }); | ||
|
||
// access the underlying `ws.WebSocket` instance | ||
rt.socket.addEventListener('open', () => { | ||
console.log('Connection opened!'); | ||
rt.send({ | ||
type: 'session.update', | ||
session: { | ||
modalities: ['text'], | ||
model: 'gpt-4o-realtime-preview', | ||
}, | ||
}); | ||
|
||
rt.send({ | ||
type: 'conversation.item.create', | ||
item: { | ||
type: 'message', | ||
role: 'user', | ||
content: [{ type: 'input_text', text: 'Say a couple paragraphs!' }], | ||
}, | ||
}); | ||
|
||
rt.send({ type: 'response.create' }); | ||
}); | ||
|
||
rt.on('error', (err) => { | ||
// in a real world scenario this should be logged somewhere as you | ||
// likely want to continue procesing events regardless of any errors | ||
throw err; | ||
}); | ||
|
||
rt.on('session.created', (event) => { | ||
console.log('session created!', event.session); | ||
console.log(); | ||
}); | ||
|
||
rt.on('response.text.delta', (event) => process.stdout.write(event.delta)); | ||
rt.on('response.text.done', () => console.log()); | ||
|
||
rt.on('response.done', () => rt.close()); | ||
|
||
rt.socket.addEventListener('close', () => console.log('\nConnection closed!')); | ||
} | ||
|
||
main(); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
import { OpenAIRealtimeWS } from 'openai/beta/realtime/ws'; | ||
|
||
async function main() { | ||
const rt = new OpenAIRealtimeWS({ model: 'gpt-4o-realtime-preview-2024-12-17' }); | ||
|
||
// access the underlying `ws.WebSocket` instance | ||
rt.socket.on('open', () => { | ||
console.log('Connection opened!'); | ||
rt.send({ | ||
type: 'session.update', | ||
session: { | ||
modalities: ['foo'] as any, | ||
model: 'gpt-4o-realtime-preview', | ||
}, | ||
}); | ||
rt.send({ | ||
type: 'session.update', | ||
session: { | ||
modalities: ['text'], | ||
model: 'gpt-4o-realtime-preview', | ||
}, | ||
}); | ||
|
||
rt.send({ | ||
type: 'conversation.item.create', | ||
item: { | ||
type: 'message', | ||
role: 'user', | ||
content: [{ type: 'input_text', text: 'Say a couple paragraphs!' }], | ||
}, | ||
}); | ||
|
||
rt.send({ type: 'response.create' }); | ||
}); | ||
|
||
rt.on('error', (err) => { | ||
// in a real world scenario this should be logged somewhere as you | ||
// likely want to continue procesing events regardless of any errors | ||
throw err; | ||
}); | ||
|
||
rt.on('session.created', (event) => { | ||
console.log('session created!', event.session); | ||
console.log(); | ||
}); | ||
|
||
rt.on('response.text.delta', (event) => process.stdout.write(event.delta)); | ||
rt.on('response.text.done', () => console.log()); | ||
|
||
rt.on('response.done', () => rt.close()); | ||
|
||
rt.socket.on('close', () => console.log('\nConnection closed!')); | ||
} | ||
|
||
main(); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export { OpenAIRealtimeError } from './internal-base'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,83 @@ | ||
import { RealtimeClientEvent, RealtimeServerEvent, ErrorEvent } from '../../resources/beta/realtime/realtime'; | ||
import { EventEmitter } from '../../lib/EventEmitter'; | ||
import { OpenAIError } from '../../error'; | ||
|
||
export class OpenAIRealtimeError extends OpenAIError { | ||
/** | ||
* The error data that the API sent back in an `error` event. | ||
*/ | ||
error?: ErrorEvent.Error | undefined; | ||
|
||
/** | ||
* The unique ID of the server event. | ||
*/ | ||
event_id?: string | undefined; | ||
|
||
constructor(message: string, event: ErrorEvent | null) { | ||
super(message); | ||
|
||
this.error = event?.error; | ||
this.event_id = event?.event_id; | ||
} | ||
} | ||
|
||
type Simplify<T> = { [KeyType in keyof T]: T[KeyType] } & {}; | ||
|
||
type RealtimeEvents = Simplify< | ||
{ | ||
event: (event: RealtimeServerEvent) => void; | ||
error: (error: OpenAIRealtimeError) => void; | ||
} & { | ||
[EventType in Exclude<RealtimeServerEvent['type'], 'error'>]: ( | ||
event: Extract<RealtimeServerEvent, { type: EventType }>, | ||
) => unknown; | ||
} | ||
>; | ||
|
||
export abstract class OpenAIRealtimeEmitter extends EventEmitter<RealtimeEvents> { | ||
/** | ||
* Send an event to the API. | ||
*/ | ||
abstract send(event: RealtimeClientEvent): void; | ||
|
||
/** | ||
* Close the websocket connection. | ||
*/ | ||
abstract close(props?: { code: number; reason: string }): void; | ||
|
||
protected _onError(event: null, message: string, cause: any): void; | ||
protected _onError(event: ErrorEvent, message?: string | undefined): void; | ||
protected _onError(event: ErrorEvent | null, message?: string | undefined, cause?: any): void { | ||
message = | ||
event?.error ? | ||
`${event.error.message} code=${event.error.code} param=${event.error.param} type=${event.error.type} event_id=${event.error.event_id}` | ||
: message ?? 'unknown error'; | ||
|
||
if (!this._hasListener('error')) { | ||
const error = new OpenAIRealtimeError( | ||
message + | ||
`\n\nTo resolve these unhandled rejection errors you should bind an \`error\` callback, e.g. \`rt.on('error', (error) => ...)\` `, | ||
event, | ||
); | ||
// @ts-ignore | ||
error.cause = cause; | ||
Promise.reject(error); | ||
return; | ||
} | ||
|
||
const error = new OpenAIRealtimeError(message, event); | ||
// @ts-ignore | ||
error.cause = cause; | ||
|
||
this._emit('error', error); | ||
} | ||
} | ||
|
||
export function buildRealtimeURL(props: { baseURL: string; model: string }): URL { | ||
const path = '/realtime'; | ||
|
||
const url = new URL(props.baseURL + (props.baseURL.endsWith('/') ? path.slice(1) : path)); | ||
url.protocol = 'wss'; | ||
url.searchParams.set('model', props.model); | ||
return url; | ||
} |
Oops, something went wrong.