Skip to content

Commit ca5e6b7

Browse files
committed
Capture reasoning
1 parent 38c35a1 commit ca5e6b7

File tree

8 files changed

+119
-64
lines changed

8 files changed

+119
-64
lines changed

packages/ai-bot/lib/debug.ts

+1
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,7 @@ export async function handleDebugCommands(
6161
client,
6262
roomId,
6363
`Error parsing your debug patch, ${error} ${patchMessage}`,
64+
'',
6465
undefined,
6566
);
6667
}

packages/ai-bot/lib/matrix.ts

+14-7
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,10 @@ import { logger } from '@cardstack/runtime-common';
33
import { OpenAIError } from 'openai/error';
44
import * as Sentry from '@sentry/node';
55
import { FunctionToolCall } from '@cardstack/runtime-common/helpers/ai';
6-
import { APP_BOXEL_COMMAND_MSGTYPE } from '@cardstack/runtime-common/matrix-constants';
6+
import {
7+
APP_BOXEL_COMMAND_MSGTYPE,
8+
APP_BOXEL_REASONING_CONTENT_KEY,
9+
} from '@cardstack/runtime-common/matrix-constants';
710

811
let log = logger('ai-bot');
912

@@ -56,21 +59,24 @@ export async function updateStateEvent(
5659
export async function sendMessage(
5760
client: MatrixClient,
5861
roomId: string,
59-
content: string,
62+
body: string,
63+
reasoning: string,
6064
eventToUpdate: string | undefined,
6165
data: any = {},
6266
) {
63-
log.debug('sending message', content);
67+
log.debug('sending message', body);
6468
let messageObject: IContent = {
6569
...{
66-
body: content,
70+
body,
6771
msgtype: 'm.text',
68-
formatted_body: content,
72+
formatted_body: body,
6973
format: 'org.matrix.custom.html',
74+
[APP_BOXEL_REASONING_CONTENT_KEY]: reasoning,
7075
'm.new_content': {
71-
body: content,
76+
body,
7277
msgtype: 'm.text',
73-
formatted_body: content,
78+
formatted_body: body,
79+
[APP_BOXEL_REASONING_CONTENT_KEY]: reasoning,
7480
format: 'org.matrix.custom.html',
7581
},
7682
},
@@ -121,6 +127,7 @@ export async function sendError(
121127
client,
122128
roomId,
123129
'There was an error processing your request, please try again later',
130+
'',
124131
eventToUpdate,
125132
{
126133
isStreamingFinished: true,

packages/ai-bot/lib/responder.ts

+30-33
Original file line numberDiff line numberDiff line change
@@ -22,31 +22,29 @@ let log = logger('ai-bot');
2222
export class Responder {
2323
// internally has a debounced function that will send the text messages
2424

25-
initialMessageId: string | undefined;
25+
responseEventId: string | undefined;
2626
initialMessageReplaced = false;
2727
client: MatrixClient;
2828
roomId: string;
2929
includesFunctionToolCall = false;
30-
latestContent?: string;
30+
latestContent = '';
31+
reasoning = '';
3132
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>;
3735

3836
constructor(client: MatrixClient, roomId: string) {
3937
this.roomId = roomId;
4038
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+
4846
let dataOverrides: Record<string, string | boolean> = {
49-
isStreamingFinished: isStreamingFinished,
47+
isStreamingFinished,
5048
};
5149
if (this.includesFunctionToolCall) {
5250
dataOverrides = {
@@ -58,6 +56,7 @@ export class Responder {
5856
this.client,
5957
this.roomId,
6058
content,
59+
reasoning,
6160
eventToUpdate,
6261
dataOverrides,
6362
);
@@ -74,23 +73,26 @@ export class Responder {
7473
this.client,
7574
this.roomId,
7675
thinkingMessage,
76+
'',
7777
undefined,
7878
{ isStreamingFinished: false },
7979
);
80-
this.initialMessageId = initialMessage.event_id;
80+
this.responseEventId = initialMessage.event_id;
8181
}
8282

8383
async onChunk(chunk: OpenAI.Chat.Completions.ChatCompletionChunk) {
8484
log.debug('onChunk: ', JSON.stringify(chunk, null, 2));
8585
if (chunk.choices[0].delta?.tool_calls?.[0]?.function) {
8686
if (!this.includesFunctionToolCall) {
8787
this.includesFunctionToolCall = true;
88-
await this.debouncedMessageSender(
89-
this.latestContent || '',
90-
this.initialMessageId,
91-
);
88+
await this.sendMessageDebounced();
9289
}
9390
}
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+
}
9496
// This usage value is set *once* and *only once* at the end of the conversation
9597
// It will be null at all other times.
9698
if (chunk.usage) {
@@ -102,10 +104,8 @@ export class Responder {
102104

103105
async onContent(snapshot: string) {
104106
log.debug('onContent: ', snapshot);
105-
await this.debouncedMessageSender(
106-
cleanContent(snapshot),
107-
this.initialMessageId,
108-
);
107+
this.latestContent = cleanContent(snapshot);
108+
await this.sendMessageDebounced();
109109
this.initialMessageReplaced = true;
110110
}
111111

@@ -142,7 +142,7 @@ export class Responder {
142142
this.client,
143143
this.roomId,
144144
this.deserializeToolCall(toolCall),
145-
this.initialMessageId,
145+
this.responseEventId,
146146
);
147147
this.messagePromises.push(commandMessagePromise);
148148
await commandMessagePromise;
@@ -154,7 +154,7 @@ export class Responder {
154154
this.client,
155155
this.roomId,
156156
error,
157-
this.initialMessageId,
157+
this.responseEventId,
158158
);
159159
this.messagePromises.push(errorPromise);
160160
await errorPromise;
@@ -169,19 +169,16 @@ export class Responder {
169169
this.client,
170170
this.roomId,
171171
error,
172-
this.initialMessageId,
172+
this.responseEventId,
173173
);
174174
}
175175

176176
async finalize(finalContent: string | void | null | undefined) {
177177
log.debug('finalize: ', finalContent);
178178
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();
185182
}
186183
await Promise.all(this.messagePromises);
187184
}

packages/ai-bot/main.ts

+2
Original file line numberDiff line numberDiff line change
@@ -97,13 +97,15 @@ class Assistant {
9797
return this.openai.beta.chat.completions.stream({
9898
model: prompt.model,
9999
messages: prompt.messages as ChatCompletionMessageParam[],
100+
include_reasoning: true,
100101
});
101102
} else {
102103
return this.openai.beta.chat.completions.stream({
103104
model: prompt.model,
104105
messages: prompt.messages as ChatCompletionMessageParam[],
105106
tools: prompt.tools,
106107
tool_choice: prompt.toolChoice,
108+
include_reasoning: true,
107109
});
108110
}
109111
}

packages/ai-bot/package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,11 @@
55
"@cardstack/postgres": "workspace:*",
66
"@cardstack/billing": "workspace:*",
77
"@sentry/node": "^8.31.0",
8-
"@types/node": "^18.18.5",
8+
"@types/node": "^20.17.16",
99
"@types/stream-chain": "^2.0.1",
1010
"@types/stream-json": "^1.7.3",
1111
"matrix-js-sdk": "^31.0.0",
12-
"openai": "4.47.1",
12+
"openai": "4.81.0",
1313
"qunit": "^2.18.0",
1414
"stream-chain": "^2.2.5",
1515
"stream-json": "^1.8.0",

packages/base/matrix-event.gts

+3-1
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,16 @@ import {
66
type ToolChoice,
77
} from '@cardstack/runtime-common/helpers/ai';
88
import {
9+
APP_BOXEL_ACTIVE_LLM,
910
APP_BOXEL_CARD_FORMAT,
1011
APP_BOXEL_CARDFRAGMENT_MSGTYPE,
1112
APP_BOXEL_COMMAND_MSGTYPE,
1213
APP_BOXEL_COMMAND_RESULT_EVENT_TYPE,
1314
APP_BOXEL_COMMAND_RESULT_WITH_NO_OUTPUT_MSGTYPE,
1415
APP_BOXEL_COMMAND_RESULT_WITH_OUTPUT_MSGTYPE,
1516
APP_BOXEL_MESSAGE_MSGTYPE,
17+
APP_BOXEL_REASONING_CONTENT_KEY,
1618
APP_BOXEL_ROOM_SKILLS_EVENT_TYPE,
17-
APP_BOXEL_ACTIVE_LLM,
1819
} from '@cardstack/runtime-common/matrix-constants';
1920

2021
interface BaseMatrixEvent {
@@ -112,6 +113,7 @@ export interface MessageEvent extends BaseMatrixEvent {
112113
format: 'org.matrix.custom.html';
113114
body: string;
114115
formatted_body: string;
116+
[APP_BOXEL_REASONING_CONTENT_KEY]?: string;
115117
isStreamingFinished: boolean;
116118
errorMessage?: string;
117119
};

packages/runtime-common/matrix-constants.ts

+1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ export const APP_BOXEL_COMMAND_RESULT_WITH_NO_OUTPUT_MSGTYPE =
99
'app.boxel.commandResultWithNoOutput';
1010
export const APP_BOXEL_REALM_SERVER_EVENT_MSGTYPE =
1111
'app.boxel.realm-server-event';
12+
export const APP_BOXEL_REASONING_CONTENT_KEY = 'app.boxel.reasoning';
1213
export const APP_BOXEL_ROOM_SKILLS_EVENT_TYPE = 'app.boxel.room.skills';
1314
export const APP_BOXEL_REALMS_EVENT_TYPE = 'app.boxel.realms';
1415
export const LEGACY_APP_BOXEL_REALMS_EVENT_TYPE = 'com.cardstack.boxel.realms';

0 commit comments

Comments
 (0)