Skip to content

Commit 7855a35

Browse files
committed
Merge branch 'main' into host/unicode-filenames-cs-6690
2 parents cdbb76d + 9b5eb56 commit 7855a35

30 files changed

+991
-642
lines changed

packages/ai-bot/helpers.ts

+7-10
Original file line numberDiff line numberDiff line change
@@ -171,26 +171,23 @@ export function getRelevantCards(
171171
return sortedCards;
172172
}
173173

174-
export function getFunctions(
175-
history: DiscreteMatrixEvent[],
176-
aiBotUserId: string,
177-
) {
174+
export function getTools(history: DiscreteMatrixEvent[], aiBotUserId: string) {
178175
// Just get the users messages
179176
const userMessages = history.filter((event) => event.sender !== aiBotUserId);
180177
// Get the last message
181178
if (userMessages.length === 0) {
182-
// If the user has sent no messages, there are no relevant functions to return
179+
// If the user has sent no messages, there are no relevant tools to return
183180
return [];
184181
}
185182
const lastMessage = userMessages[userMessages.length - 1];
186183
if (
187184
lastMessage.type === 'm.room.message' &&
188185
lastMessage.content.msgtype === 'org.boxel.message' &&
189-
lastMessage.content.data?.context?.functions
186+
lastMessage.content.data?.context?.tools
190187
) {
191-
return lastMessage.content.data.context.functions;
188+
return lastMessage.content.data.context.tools;
192189
} else {
193-
// If it's a different message type, or there are no functions, return an empty array
190+
// If it's a different message type, or there are no tools, return an empty array
194191
return [];
195192
}
196193
}
@@ -266,7 +263,7 @@ export function getStartOfConversation(
266263
export function getModifyPrompt(
267264
history: DiscreteMatrixEvent[],
268265
aiBotUserId: string,
269-
functions: any[] = [],
266+
tools: any[] = [],
270267
) {
271268
// Need to make sure the passed in username is a full id
272269
if (
@@ -304,7 +301,7 @@ export function getModifyPrompt(
304301
for (let card of getRelevantCards(history, aiBotUserId)) {
305302
systemMessage += `Full data: ${JSON.stringify(card)}`;
306303
}
307-
if (functions.length == 0) {
304+
if (tools.length == 0) {
308305
systemMessage +=
309306
'You are unable to edit any cards, the user has not given you access, they need to open the card on the stack and let it be auto-attached';
310307
}

packages/ai-bot/main.ts

+39-31
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ import {
1313
constructHistory,
1414
getModifyPrompt,
1515
cleanContent,
16-
getFunctions,
16+
getTools,
1717
getStartOfConversation,
1818
shouldSetRoomTitle,
1919
type OpenAIPromptMessage,
@@ -98,7 +98,7 @@ async function sendOption(
9898
) {
9999
log.info('sending option', patch);
100100
const id = patch['card_id'];
101-
const body = patch['description'];
101+
const body = patch['description'] || "Here's the change:";
102102
let messageObject = {
103103
body: body,
104104
msgtype: 'org.boxel.command',
@@ -126,9 +126,9 @@ async function sendOption(
126126
}
127127

128128
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) {
132132
return openai.beta.chat.completions.stream({
133133
model: 'gpt-4-turbo',
134134
messages: messages,
@@ -137,8 +137,8 @@ function getResponse(history: DiscreteMatrixEvent[], aiBotUsername: string) {
137137
return openai.beta.chat.completions.stream({
138138
model: 'gpt-4-turbo',
139139
messages: messages,
140-
functions: functions,
141-
function_call: 'auto',
140+
tools: tools,
141+
tool_choice: 'auto',
142142
});
143143
}
144144
}
@@ -374,6 +374,7 @@ Common issues are:
374374

375375
let unsent = 0;
376376
let sentCommands = 0;
377+
let thinkingMessageReplaced = false;
377378
const runner = getResponse(history, aiBotUserId)
378379
.on('content', async (_delta, snapshot) => {
379380
unsent += 1;
@@ -386,37 +387,44 @@ Common issues are:
386387
initialMessage.event_id,
387388
);
388389
}
390+
thinkingMessageReplaced = true;
389391
})
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+
}
412422
}
413-
return;
414423
})
415424
.on('error', async (error: OpenAIError) => {
416425
Sentry.captureException(error);
417426
return await sendError(client, room, error, initialMessage.event_id);
418427
});
419-
420428
// We also need to catch the error when getting the final content
421429
let finalContent = await runner.finalContent().catch(async (error) => {
422430
return await sendError(client, room, error, initialMessage.event_id);

0 commit comments

Comments
 (0)