Skip to content

Commit 63608cf

Browse files
committed
Improve AI choose rule prompt and add real world tests
1 parent fa52f9c commit 63608cf

File tree

3 files changed

+222
-2
lines changed

3 files changed

+222
-2
lines changed

apps/web/__tests__/ai-choose-rule.test.ts

+216
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,222 @@ describe.skipIf(!isAiTest)("aiChooseRule", () => {
9191
reason: expect.any(String),
9292
});
9393
});
94+
95+
describe("Complex real-world rule scenarios", () => {
96+
const recruiters = getRule(
97+
"Match emails from recruiters or about job opportunities",
98+
);
99+
const legal = getRule(
100+
"Match emails containing legal documents or contracts",
101+
);
102+
const requiresResponse = getRule(
103+
"Match emails that require a response or action from the recipient",
104+
);
105+
const productUpdates = getRule(
106+
"Match emails about product updates or feature announcements",
107+
);
108+
const financial = getRule(
109+
"Match emails containing financial information or invoices",
110+
);
111+
const technicalIssues = getRule(
112+
"Match emails about technical issues like server downtime or bug reports",
113+
);
114+
const marketing = getRule(
115+
"Match emails containing marketing or promotional content",
116+
);
117+
const teamUpdates = getRule(
118+
"Match emails about team updates or internal communications",
119+
);
120+
const customerFeedback = getRule(
121+
"Match emails about customer feedback or support requests",
122+
);
123+
const events = getRule(
124+
"Match emails containing event invitations or RSVPs",
125+
);
126+
const projectDeadlines = getRule(
127+
"Match emails about project deadlines or milestones",
128+
);
129+
const urgent = getRule("Match urgent emails requiring immediate attention");
130+
const catchAll = getRule("Match emails that don't fit any other category");
131+
132+
const rules = [
133+
recruiters,
134+
legal,
135+
requiresResponse,
136+
productUpdates,
137+
financial,
138+
technicalIssues,
139+
marketing,
140+
teamUpdates,
141+
customerFeedback,
142+
events,
143+
projectDeadlines,
144+
urgent,
145+
catchAll,
146+
];
147+
148+
test("Should match technical issues", async () => {
149+
const result = await aiChooseRule({
150+
rules,
151+
email: getEmail({
152+
subject: "Server downtime reported",
153+
content:
154+
"We're experiencing critical server issues affecting production.",
155+
}),
156+
user: getUser(),
157+
});
158+
159+
expect(result).toEqual({
160+
rule: technicalIssues,
161+
reason: expect.any(String),
162+
});
163+
});
164+
165+
test("Should match financial emails", async () => {
166+
const result = await aiChooseRule({
167+
rules,
168+
email: getEmail({
169+
subject: "Your invoice for March 2024",
170+
content: "Please find attached your invoice for services rendered.",
171+
}),
172+
user: getUser(),
173+
});
174+
175+
expect(result).toEqual({
176+
rule: financial,
177+
reason: expect.any(String),
178+
});
179+
});
180+
181+
test("Should match recruiter emails", async () => {
182+
const result = await aiChooseRule({
183+
rules,
184+
email: getEmail({
185+
subject: "New job opportunity at Tech Corp",
186+
content:
187+
"I came across your profile and think you'd be perfect for...",
188+
}),
189+
user: getUser(),
190+
});
191+
192+
expect(result).toEqual({
193+
rule: recruiters,
194+
reason: expect.any(String),
195+
});
196+
});
197+
198+
test("Should match legal documents", async () => {
199+
const result = await aiChooseRule({
200+
rules,
201+
email: getEmail({
202+
subject: "Please review: Contract for new project",
203+
content: "Attached is the contract for your review and signature.",
204+
}),
205+
user: getUser(),
206+
});
207+
208+
expect(result).toEqual({
209+
rule: legal,
210+
reason: expect.any(String),
211+
});
212+
});
213+
214+
test("Should match emails requiring response", async () => {
215+
const result = await aiChooseRule({
216+
rules,
217+
email: getEmail({
218+
subject: "Team lunch tomorrow?",
219+
content: "Would you like to join us for team lunch tomorrow at 12pm?",
220+
}),
221+
user: getUser(),
222+
});
223+
224+
expect(result).toEqual({
225+
rule: requiresResponse,
226+
reason: expect.any(String),
227+
});
228+
});
229+
230+
test("Should match product updates", async () => {
231+
const result = await aiChooseRule({
232+
rules,
233+
email: getEmail({
234+
subject: "New Feature Release: AI Integration",
235+
content: "We're excited to announce our new AI features...",
236+
}),
237+
user: getUser(),
238+
});
239+
240+
expect(result).toEqual({
241+
rule: productUpdates,
242+
reason: expect.any(String),
243+
});
244+
});
245+
246+
test("Should match marketing emails", async () => {
247+
const result = await aiChooseRule({
248+
rules,
249+
email: getEmail({
250+
subject: "50% off Spring Sale!",
251+
content: "Don't miss out on our biggest sale of the season!",
252+
}),
253+
user: getUser(),
254+
});
255+
256+
expect(result).toEqual({
257+
rule: marketing,
258+
reason: expect.any(String),
259+
});
260+
});
261+
262+
test("Should match team updates", async () => {
263+
const result = await aiChooseRule({
264+
rules,
265+
email: getEmail({
266+
subject: "Weekly Team Update",
267+
content: "Here's what the team accomplished this week...",
268+
}),
269+
user: getUser(),
270+
});
271+
272+
expect(result).toEqual({
273+
rule: teamUpdates,
274+
reason: expect.any(String),
275+
});
276+
});
277+
278+
test("Should match customer feedback", async () => {
279+
const result = await aiChooseRule({
280+
rules,
281+
email: getEmail({
282+
subject: "Customer Feedback: App Performance",
283+
content: "I've been experiencing slow loading times...",
284+
}),
285+
user: getUser(),
286+
});
287+
288+
expect(result).toEqual({
289+
rule: customerFeedback,
290+
reason: expect.any(String),
291+
});
292+
});
293+
294+
test("Should match event invitations", async () => {
295+
const result = await aiChooseRule({
296+
rules,
297+
email: getEmail({
298+
subject: "Invitation: Annual Tech Conference",
299+
content: "You're invited to speak at our annual conference...",
300+
}),
301+
user: getUser(),
302+
});
303+
304+
expect(result).toEqual({
305+
rule: events,
306+
reason: expect.any(String),
307+
});
308+
});
309+
});
94310
});
95311

96312
// helpers

apps/web/utils/ai/choose-rule/ai-choose-rule.ts

+2
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,8 @@ async function getAiResponse(options: GetAiResponseOptions) {
3434
const system = `You are an AI assistant that helps people manage their emails.
3535
IMPORTANT: You must strictly follow the exclusions mentioned in each rule.
3636
- If a rule says to exclude certain types of emails, DO NOT select that rule for those excluded emails.
37+
- When multiple rules match, choose the more specific one that best matches the email's content.
38+
- Rules about requiring replies should be prioritized when the email clearly needs a response.
3739
- If you're unsure, select the last rule (not enough information).
3840
- It's better to select "not enough information" than to make an incorrect choice.
3941

apps/web/utils/ai/rule/create-rule-schema.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -102,10 +102,12 @@ export const getCreateRuleSchemaWithCategories = (
102102
"Whether senders in `categoryFilters` should be included or excluded",
103103
),
104104
categoryFilters: z
105-
.array(z.enum(availableCategories))
105+
.array(z.string())
106106
.optional()
107107
.describe(
108-
"The categories to match. If multiple categories are specified, the rule will match if ANY of the categories match (OR operation)",
108+
`The categories to match. If multiple categories are specified, the rule will match if ANY of the categories match (OR operation). Available categories: ${availableCategories
109+
.map((c) => `"${c}"`)
110+
.join(", ")}`,
109111
),
110112
})
111113
.optional()

0 commit comments

Comments
 (0)