diff --git a/packages/core/src/modules/opportunities.ts b/packages/core/src/modules/opportunities.ts
index 11e2f1da..9298dbff 100644
--- a/packages/core/src/modules/opportunities.ts
+++ b/packages/core/src/modules/opportunities.ts
@@ -461,6 +461,10 @@ const REFINE_OPPORTUNITY_PROMPT = dedent`
Follow these guidelines:
- If you cannot confidently infer a field, set it to null.
+ - If the page is not found, expired, or otherwise not a valid opportunity,
+ set all fields to null.
+ - Double check that your output is based on the website content. Don't make
+ up information that you cannot confidently infer from the website content.
Your output should be a single JSON object containing these fields. Do not
provide any explanation or text outside of the JSON object. Ensure your JSON
@@ -469,20 +473,20 @@ const REFINE_OPPORTUNITY_PROMPT = dedent`
`;
const RefineOpportunityResponse = z.object({
company: z.string().trim().min(1).nullable(),
- description: z.string().trim().min(1).max(500),
+ description: z.string().trim().min(1).max(500).nullable(),
expiresAt: z.string().nullable(),
- tags: z.array(z.string().trim().min(1)).min(1),
- title: z.string().trim().min(1).max(100),
+ tags: z.array(z.string().trim().min(1)).min(1).nullable(),
+ title: z.string().trim().min(1).max(100).nullable(),
});
type RefineOpportunityResponse = z.infer;
@@ -569,10 +573,10 @@ export async function refineOpportunity(
const opportunity = await trx
.updateTable('opportunities')
.set({
+ ...(data.description && { description: data.description }),
+ ...(data.title && { title: data.title }),
companyId,
- description: data.description,
expiresAt,
- title: data.title,
})
.where('id', '=', input.opportunityId)
.returning(['id', 'refinedAt', 'slackChannelId', 'slackMessageId'])
@@ -587,6 +591,10 @@ export async function refineOpportunity(
.where('refinedAt', 'is', null)
.executeTakeFirst();
+ if (!data.tags) {
+ return opportunity;
+ }
+
const upsertedTags = await trx
.insertInto('opportunityTags')
.values(