@@ -22,31 +22,29 @@ let log = logger('ai-bot');
22
22
export class Responder {
23
23
// internally has a debounced function that will send the text messages
24
24
25
- initialMessageId : string | undefined ;
25
+ responseEventId : string | undefined ;
26
26
initialMessageReplaced = false ;
27
27
client : MatrixClient ;
28
28
roomId : string ;
29
29
includesFunctionToolCall = false ;
30
- latestContent ?: string ;
30
+ latestContent = '' ;
31
+ reasoning = '' ;
31
32
messagePromises : Promise < ISendEventResponse | void > [ ] = [ ] ;
32
- debouncedMessageSender : (
33
- content : string ,
34
- eventToUpdate : string | undefined ,
35
- isStreamingFinished ?: boolean ,
36
- ) => Promise < void > ;
33
+ isStreamingFinished = false ;
34
+ sendMessageDebounced : ( ) => Promise < void > ;
37
35
38
36
constructor ( client : MatrixClient , roomId : string ) {
39
37
this . roomId = roomId ;
40
38
this . client = client ;
41
- this . debouncedMessageSender = debounce (
42
- async (
43
- content : string ,
44
- eventToUpdate : string | undefined ,
45
- isStreamingFinished = false ,
46
- ) => {
47
- this . latestContent = content ;
39
+ this . sendMessageDebounced = debounce (
40
+ async ( ) => {
41
+ const content = this . latestContent ;
42
+ const reasoning = this . reasoning ;
43
+ const eventToUpdate = this . responseEventId ;
44
+ const isStreamingFinished = this . isStreamingFinished ;
45
+
48
46
let dataOverrides : Record < string , string | boolean > = {
49
- isStreamingFinished : isStreamingFinished ,
47
+ isStreamingFinished,
50
48
} ;
51
49
if ( this . includesFunctionToolCall ) {
52
50
dataOverrides = {
@@ -58,6 +56,7 @@ export class Responder {
58
56
this . client ,
59
57
this . roomId ,
60
58
content ,
59
+ reasoning ,
61
60
eventToUpdate ,
62
61
dataOverrides ,
63
62
) ;
@@ -74,23 +73,26 @@ export class Responder {
74
73
this . client ,
75
74
this . roomId ,
76
75
thinkingMessage ,
76
+ '' ,
77
77
undefined ,
78
78
{ isStreamingFinished : false } ,
79
79
) ;
80
- this . initialMessageId = initialMessage . event_id ;
80
+ this . responseEventId = initialMessage . event_id ;
81
81
}
82
82
83
83
async onChunk ( chunk : OpenAI . Chat . Completions . ChatCompletionChunk ) {
84
84
log . debug ( 'onChunk: ' , JSON . stringify ( chunk , null , 2 ) ) ;
85
85
if ( chunk . choices [ 0 ] . delta ?. tool_calls ?. [ 0 ] ?. function ) {
86
86
if ( ! this . includesFunctionToolCall ) {
87
87
this . includesFunctionToolCall = true ;
88
- await this . debouncedMessageSender (
89
- this . latestContent || '' ,
90
- this . initialMessageId ,
91
- ) ;
88
+ await this . sendMessageDebounced ( ) ;
92
89
}
93
90
}
91
+ // @ts -expect-error reasoning is not in the types yet
92
+ if ( chunk . choices [ 0 ] . delta ?. reasoning ) {
93
+ // @ts -expect-error reasoning is not in the types yet
94
+ this . reasoning += chunk . choices [ 0 ] . delta . reasoning ;
95
+ }
94
96
// This usage value is set *once* and *only once* at the end of the conversation
95
97
// It will be null at all other times.
96
98
if ( chunk . usage ) {
@@ -102,10 +104,8 @@ export class Responder {
102
104
103
105
async onContent ( snapshot : string ) {
104
106
log . debug ( 'onContent: ' , snapshot ) ;
105
- await this . debouncedMessageSender (
106
- cleanContent ( snapshot ) ,
107
- this . initialMessageId ,
108
- ) ;
107
+ this . latestContent = cleanContent ( snapshot ) ;
108
+ await this . sendMessageDebounced ( ) ;
109
109
this . initialMessageReplaced = true ;
110
110
}
111
111
@@ -142,7 +142,7 @@ export class Responder {
142
142
this . client ,
143
143
this . roomId ,
144
144
this . deserializeToolCall ( toolCall ) ,
145
- this . initialMessageId ,
145
+ this . responseEventId ,
146
146
) ;
147
147
this . messagePromises . push ( commandMessagePromise ) ;
148
148
await commandMessagePromise ;
@@ -154,7 +154,7 @@ export class Responder {
154
154
this . client ,
155
155
this . roomId ,
156
156
error ,
157
- this . initialMessageId ,
157
+ this . responseEventId ,
158
158
) ;
159
159
this . messagePromises . push ( errorPromise ) ;
160
160
await errorPromise ;
@@ -169,19 +169,16 @@ export class Responder {
169
169
this . client ,
170
170
this . roomId ,
171
171
error ,
172
- this . initialMessageId ,
172
+ this . responseEventId ,
173
173
) ;
174
174
}
175
175
176
176
async finalize ( finalContent : string | void | null | undefined ) {
177
177
log . debug ( 'finalize: ' , finalContent ) ;
178
178
if ( finalContent ) {
179
- finalContent = cleanContent ( finalContent ) ;
180
- await this . debouncedMessageSender (
181
- finalContent ,
182
- this . initialMessageId ,
183
- true ,
184
- ) ;
179
+ this . latestContent = cleanContent ( finalContent ) ;
180
+ this . isStreamingFinished = true ;
181
+ await this . sendMessageDebounced ( ) ;
185
182
}
186
183
await Promise . all ( this . messagePromises ) ;
187
184
}
0 commit comments