@@ -13,7 +13,7 @@ import {
13
13
constructHistory ,
14
14
getModifyPrompt ,
15
15
cleanContent ,
16
- getFunctions ,
16
+ getTools ,
17
17
getStartOfConversation ,
18
18
shouldSetRoomTitle ,
19
19
type OpenAIPromptMessage ,
@@ -98,7 +98,7 @@ async function sendOption(
98
98
) {
99
99
log . info ( 'sending option' , patch ) ;
100
100
const id = patch [ 'card_id' ] ;
101
- const body = patch [ 'description' ] ;
101
+ const body = patch [ 'description' ] || "Here's the change:" ;
102
102
let messageObject = {
103
103
body : body ,
104
104
msgtype : 'org.boxel.command' ,
@@ -126,9 +126,9 @@ async function sendOption(
126
126
}
127
127
128
128
function getResponse ( history : DiscreteMatrixEvent [ ] , aiBotUsername : string ) {
129
- let functions = getFunctions ( history , aiBotUsername ) ;
130
- let messages = getModifyPrompt ( history , aiBotUsername , functions ) ;
131
- if ( functions . length === 0 ) {
129
+ let tools = getTools ( history , aiBotUsername ) ;
130
+ let messages = getModifyPrompt ( history , aiBotUsername , tools ) ;
131
+ if ( tools . length === 0 ) {
132
132
return openai . beta . chat . completions . stream ( {
133
133
model : 'gpt-4-turbo' ,
134
134
messages : messages ,
@@ -137,8 +137,8 @@ function getResponse(history: DiscreteMatrixEvent[], aiBotUsername: string) {
137
137
return openai . beta . chat . completions . stream ( {
138
138
model : 'gpt-4-turbo' ,
139
139
messages : messages ,
140
- functions : functions ,
141
- function_call : 'auto' ,
140
+ tools : tools ,
141
+ tool_choice : 'auto' ,
142
142
} ) ;
143
143
}
144
144
}
@@ -374,6 +374,7 @@ Common issues are:
374
374
375
375
let unsent = 0 ;
376
376
let sentCommands = 0 ;
377
+ let thinkingMessageReplaced = false ;
377
378
const runner = getResponse ( history , aiBotUserId )
378
379
. on ( 'content' , async ( _delta , snapshot ) => {
379
380
unsent += 1 ;
@@ -386,37 +387,44 @@ Common issues are:
386
387
initialMessage . event_id ,
387
388
) ;
388
389
}
390
+ thinkingMessageReplaced = true ;
389
391
} )
390
- . on ( 'functionCall' , async ( functionCall ) => {
391
- console . log ( 'Function call' , functionCall ) ;
392
- let args ;
393
- try {
394
- args = JSON . parse ( functionCall . arguments ) ;
395
- } catch ( error ) {
396
- Sentry . captureException ( error ) ;
397
- return await sendError (
398
- client ,
399
- room ,
400
- error ,
401
- initialMessage . event_id ,
402
- ) ;
403
- }
404
- if ( functionCall . name === 'patchCard' ) {
405
- sentCommands += 1 ;
406
- return await sendOption (
407
- client ,
408
- room ,
409
- args ,
410
- initialMessage . event_id ,
411
- ) ;
392
+ // Messages can have both content and tool calls
393
+ // We handle tool calls here
394
+ . on ( 'message' , async ( msg ) => {
395
+ if ( msg . role === 'assistant' ) {
396
+ for ( const toolCall of msg . tool_calls || [ ] ) {
397
+ const functionCall = toolCall . function ;
398
+ console . log ( 'Function call' , toolCall ) ;
399
+ let args ;
400
+ try {
401
+ args = JSON . parse ( functionCall . arguments ) ;
402
+ } catch ( error ) {
403
+ Sentry . captureException ( error ) ;
404
+ return await sendError (
405
+ client ,
406
+ room ,
407
+ error ,
408
+ thinkingMessageReplaced ? undefined : initialMessage . event_id ,
409
+ ) ;
410
+ }
411
+ if ( functionCall . name === 'patchCard' ) {
412
+ sentCommands += 1 ;
413
+ await sendOption (
414
+ client ,
415
+ room ,
416
+ args ,
417
+ thinkingMessageReplaced ? undefined : initialMessage . event_id ,
418
+ ) ;
419
+ thinkingMessageReplaced = true ;
420
+ }
421
+ }
412
422
}
413
- return ;
414
423
} )
415
424
. on ( 'error' , async ( error : OpenAIError ) => {
416
425
Sentry . captureException ( error ) ;
417
426
return await sendError ( client , room , error , initialMessage . event_id ) ;
418
427
} ) ;
419
-
420
428
// We also need to catch the error when getting the final content
421
429
let finalContent = await runner . finalContent ( ) . catch ( async ( error ) => {
422
430
return await sendError ( client , room , error , initialMessage . event_id ) ;
0 commit comments