Skip to content

Commit 5ab1eca

Browse files
IanCallukemelia
authored andcommitted
Add test for serialising the command results when they are exported but not previously loaded
1 parent ad7ee27 commit 5ab1eca

File tree

2 files changed

+158
-0
lines changed

2 files changed

+158
-0
lines changed

packages/host/tests/acceptance/commands-test.gts

+75
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ import { baseRealm, Command } from '@cardstack/runtime-common';
1919
import {
2020
APP_BOXEL_COMMAND_MSGTYPE,
2121
APP_BOXEL_MESSAGE_MSGTYPE,
22+
APP_BOXEL_COMMAND_RESULT_EVENT_TYPE,
2223
} from '@cardstack/runtime-common/matrix-constants';
2324

2425
import CreateAIAssistantRoomCommand from '@cardstack/host/commands/create-ai-assistant-room';
@@ -362,6 +363,20 @@ module('Acceptance | Commands tests', function (hooks) {
362363
'person.gts': { Person, Meeting },
363364
'pet.gts': { Pet },
364365
'Pet/ringo.json': new Pet({ name: 'Ringo' }),
366+
'AiCommandExample/london.json': {
367+
data: {
368+
type: 'card',
369+
attributes: {
370+
location: 'London',
371+
},
372+
meta: {
373+
adoptsFrom: {
374+
module: 'http://localhost:4202/test/ai-command-example',
375+
name: 'AiCommandExample',
376+
},
377+
},
378+
},
379+
},
365380
'Person/hassan.json': new Person({
366381
firstName: 'Hassan',
367382
lastName: 'Abdel-Rahman',
@@ -867,4 +882,64 @@ module('Acceptance | Commands tests', function (hooks) {
867882
.dom('[data-test-message-idx="1"] [data-test-boxel-command-result]')
868883
.containsText('Submode: interact');
869884
});
885+
886+
test('command returns serialized result in room message', async function (assert) {
887+
await visitOperatorMode({
888+
stacks: [
889+
[
890+
{
891+
id: `${testRealmURL}AiCommandExample/london`,
892+
format: 'isolated',
893+
},
894+
],
895+
],
896+
});
897+
898+
await click('[data-test-get-weather]');
899+
await waitUntil(() => getRoomIds().length > 0);
900+
901+
let roomId = getRoomIds().pop()!;
902+
let message = getRoomEvents(roomId).pop()!;
903+
assert.strictEqual(message.content.msgtype, APP_BOXEL_MESSAGE_MSGTYPE);
904+
905+
let boxelMessageData = JSON.parse(message.content.data);
906+
assert.strictEqual(boxelMessageData.context.tools.length, 1);
907+
assert.strictEqual(boxelMessageData.context.tools[0].type, 'function');
908+
let toolName = boxelMessageData.context.tools[0].function.name;
909+
910+
simulateRemoteMessage(roomId, '@aibot:localhost', {
911+
body: 'Getting weather information for London',
912+
msgtype: APP_BOXEL_COMMAND_MSGTYPE,
913+
formatted_body: 'Getting weather information for London',
914+
format: 'org.matrix.custom.html',
915+
data: JSON.stringify({
916+
toolCall: {
917+
name: toolName,
918+
arguments: {
919+
attributes: {
920+
location: 'London',
921+
},
922+
},
923+
},
924+
eventId: '__EVENT_ID__',
925+
}),
926+
'm.relates_to': {
927+
rel_type: 'm.replace',
928+
event_id: '__EVENT_ID__',
929+
},
930+
});
931+
932+
await settled();
933+
let commandResultEvents = await getRoomEvents(roomId).filter(
934+
(event) =>
935+
event.type === APP_BOXEL_COMMAND_RESULT_EVENT_TYPE &&
936+
event.content['m.relates_to']?.rel_type === 'm.annotation' &&
937+
event.content['m.relates_to']?.key === 'applied',
938+
);
939+
assert.equal(
940+
commandResultEvents.length,
941+
1,
942+
'command result event is dispatched',
943+
);
944+
});
870945
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
import { on } from '@ember/modifier';
2+
3+
import CreateAIAssistantRoomCommand from '@cardstack/boxel-host/commands/create-ai-assistant-room';
4+
import SendAiAssistantMessageCommand from '@cardstack/boxel-host/commands/send-ai-assistant-message';
5+
6+
import { Button } from '@cardstack/boxel-ui/components';
7+
import { CardContainer } from '@cardstack/boxel-ui/components';
8+
9+
import { Command } from '@cardstack/runtime-common';
10+
11+
import {
12+
CardDef,
13+
Component,
14+
StringField,
15+
field,
16+
contains,
17+
} from 'https://cardstack.com/base/card-api';
18+
19+
export class WeatherLocationInput extends CardDef {
20+
@field location = contains(StringField);
21+
}
22+
23+
export class WeatherReport extends CardDef {
24+
@field temperature = contains(StringField);
25+
@field conditions = contains(StringField);
26+
}
27+
28+
class GetWeatherCommand extends Command<WeatherLocationInput, WeatherReport> {
29+
inputType = WeatherLocationInput;
30+
31+
async getInputType(): Promise<new (args: any) => WeatherLocationInput> {
32+
return WeatherLocationInput;
33+
}
34+
35+
protected async run(_input: WeatherLocationInput): Promise<WeatherReport> {
36+
return new WeatherReport({
37+
temperature: '25 C',
38+
conditions: 'Sunny',
39+
});
40+
}
41+
}
42+
43+
export class AiCommandExample extends CardDef {
44+
static displayName = 'AI Command Example';
45+
46+
@field location = contains(StringField);
47+
48+
static isolated = class Isolated extends Component<typeof AiCommandExample> {
49+
getWeather = async () => {
50+
let commandContext = this.args.context?.commandContext;
51+
if (!commandContext) {
52+
throw new Error('No command context found');
53+
}
54+
55+
let getWeatherCommand = new GetWeatherCommand(commandContext);
56+
57+
let createAIAssistantRoomCommand = new CreateAIAssistantRoomCommand(
58+
commandContext,
59+
);
60+
let { roomId } = await createAIAssistantRoomCommand.execute({
61+
name: 'Weather Assistant',
62+
});
63+
64+
let sendMessageCommand = new SendAiAssistantMessageCommand(
65+
commandContext,
66+
);
67+
68+
await sendMessageCommand.execute({
69+
roomId,
70+
prompt: `What is the weather in ${this.args.model.location}?`,
71+
commands: [{ command: getWeatherCommand, autoExecute: true }],
72+
});
73+
};
74+
75+
<template>
76+
<CardContainer>
77+
<Button data-test-get-weather {{on 'click' this.getWeather}}>
78+
Get Weather
79+
</Button>
80+
</CardContainer>
81+
</template>
82+
};
83+
}

0 commit comments

Comments
 (0)