mirror of
https://github.com/katanemo/plano.git
synced 2026-06-17 15:25:17 +02:00
tweak contacts APi route
This commit is contained in:
parent
7465cacd4f
commit
7a9eb2c11b
4 changed files with 80 additions and 46 deletions
|
|
@ -1,14 +1,60 @@
|
|||
import { Resend } from 'resend';
|
||||
import { NextResponse } from 'next/server';
|
||||
|
||||
const resend = new Resend(process.env.RESEND_API_KEY);
|
||||
function getResendClient() {
|
||||
const apiKey = process.env.RESEND_API_KEY;
|
||||
if (!apiKey) {
|
||||
throw new Error('RESEND_API_KEY environment variable is not set');
|
||||
}
|
||||
return new Resend(apiKey);
|
||||
}
|
||||
|
||||
interface ContactPayload {
|
||||
email: string;
|
||||
firstName: string;
|
||||
lastName: string;
|
||||
company?: string;
|
||||
lookingFor: string;
|
||||
}
|
||||
|
||||
function buildProperties(company?: string, lookingFor?: string): Record<string, string> | undefined {
|
||||
const properties: Record<string, string> = {};
|
||||
if (company) properties.company_name = company;
|
||||
if (lookingFor) properties.looking_for = lookingFor;
|
||||
return Object.keys(properties).length > 0 ? properties : undefined;
|
||||
}
|
||||
|
||||
function isDuplicateError(error: { message?: string; statusCode?: number | null }): boolean {
|
||||
const errorMessage = error.message?.toLowerCase() || '';
|
||||
return (
|
||||
errorMessage.includes('already exists') ||
|
||||
errorMessage.includes('duplicate') ||
|
||||
error.statusCode === 409
|
||||
);
|
||||
}
|
||||
|
||||
function createContactPayload(
|
||||
email: string,
|
||||
firstName: string,
|
||||
lastName: string,
|
||||
company?: string,
|
||||
lookingFor?: string
|
||||
) {
|
||||
const properties = buildProperties(company, lookingFor);
|
||||
return {
|
||||
email,
|
||||
firstName,
|
||||
lastName,
|
||||
unsubscribed: false,
|
||||
...(properties && { properties }),
|
||||
};
|
||||
}
|
||||
|
||||
export async function POST(req: Request) {
|
||||
try {
|
||||
const body = await req.json();
|
||||
const { firstName, lastName, email, company, lookingFor } = body;
|
||||
const { firstName, lastName, email, company, lookingFor }: ContactPayload = body;
|
||||
|
||||
// Validate required fields
|
||||
if (!email || !firstName || !lastName || !lookingFor) {
|
||||
return NextResponse.json(
|
||||
{ error: 'Missing required fields' },
|
||||
|
|
@ -16,48 +62,16 @@ export async function POST(req: Request) {
|
|||
);
|
||||
}
|
||||
|
||||
// Create or update the contact
|
||||
// Note: Contact properties (company_name, looking_for) should be
|
||||
// created manually in Resend dashboard or via a one-time setup script.
|
||||
// Attempting to create them on every request causes rate limit issues.
|
||||
const contactPayload = createContactPayload(email, firstName, lastName, company, lookingFor);
|
||||
const resend = getResendClient();
|
||||
|
||||
// Build properties object with custom fields
|
||||
// Property keys must match exactly what's defined in Resend dashboard
|
||||
const properties: Record<string, string> = {};
|
||||
if (company) properties.company_name = company;
|
||||
if (lookingFor) properties.looking_for = lookingFor;
|
||||
|
||||
let { data, error } = await resend.contacts.create({
|
||||
email,
|
||||
firstName,
|
||||
lastName,
|
||||
unsubscribed: false,
|
||||
// Pass custom properties as a Record<string, string>
|
||||
...(Object.keys(properties).length > 0 && { properties }),
|
||||
});
|
||||
const { data, error } = await resend.contacts.create(contactPayload);
|
||||
|
||||
if (error) {
|
||||
// If contact already exists, update it instead
|
||||
const errorMessage = error.message?.toLowerCase() || '';
|
||||
const isDuplicate =
|
||||
errorMessage.includes('already exists') ||
|
||||
errorMessage.includes('duplicate') ||
|
||||
error.statusCode === 409;
|
||||
|
||||
if (isDuplicate) {
|
||||
// Build properties object for update
|
||||
const updateProperties: Record<string, string> = {};
|
||||
if (company) updateProperties.company_name = company;
|
||||
if (lookingFor) updateProperties.looking_for = lookingFor;
|
||||
|
||||
const { data: updateData, error: updateError } = await resend.contacts.update({
|
||||
email,
|
||||
firstName,
|
||||
lastName,
|
||||
unsubscribed: false,
|
||||
// Pass custom properties as a Record<string, string>
|
||||
...(Object.keys(updateProperties).length > 0 && { properties: updateProperties }),
|
||||
});
|
||||
if (isDuplicateError(error)) {
|
||||
const { data: updateData, error: updateError } = await resend.contacts.update(
|
||||
contactPayload
|
||||
);
|
||||
|
||||
if (updateError) {
|
||||
console.error('Resend update error:', updateError);
|
||||
|
|
|
|||
25
package-lock.json
generated
25
package-lock.json
generated
|
|
@ -12,6 +12,7 @@
|
|||
"packages/*"
|
||||
],
|
||||
"dependencies": {
|
||||
"react": "^19.2.3",
|
||||
"resend": "^6.6.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
|
|
@ -408,6 +409,15 @@
|
|||
"node": "^10 || ^12 || >=14"
|
||||
}
|
||||
},
|
||||
"apps/www/node_modules/react": {
|
||||
"version": "19.2.0",
|
||||
"resolved": "https://registry.npmjs.org/react/-/react-19.2.0.tgz",
|
||||
"integrity": "sha512-tmbWg6W31tQLeB5cdIBOicJDJRR2KzXsV7uSK9iNfLWQ5bIZfxuPEHp7M8wiHyHnn0DD1i7w3Zmin0FtkrwoCQ==",
|
||||
"license": "MIT",
|
||||
"engines": {
|
||||
"node": ">=0.10.0"
|
||||
}
|
||||
},
|
||||
"apps/www/node_modules/tldts": {
|
||||
"version": "7.0.19",
|
||||
"resolved": "https://registry.npmjs.org/tldts/-/tldts-7.0.19.tgz",
|
||||
|
|
@ -13566,9 +13576,9 @@
|
|||
}
|
||||
},
|
||||
"node_modules/react": {
|
||||
"version": "19.2.0",
|
||||
"resolved": "https://registry.npmjs.org/react/-/react-19.2.0.tgz",
|
||||
"integrity": "sha512-tmbWg6W31tQLeB5cdIBOicJDJRR2KzXsV7uSK9iNfLWQ5bIZfxuPEHp7M8wiHyHnn0DD1i7w3Zmin0FtkrwoCQ==",
|
||||
"version": "19.2.3",
|
||||
"resolved": "https://registry.npmjs.org/react/-/react-19.2.3.tgz",
|
||||
"integrity": "sha512-Ku/hhYbVjOQnXDZFv2+RibmLFGwFdeeKHFcOTlrt7xplBnya5OGn/hIRDsqDiSUcfORsDC7MPxwork8jBwsIWA==",
|
||||
"license": "MIT",
|
||||
"engines": {
|
||||
"node": ">=0.10.0"
|
||||
|
|
@ -17702,6 +17712,15 @@
|
|||
"react": "^19.2.0",
|
||||
"react-dom": "^19.2.0"
|
||||
}
|
||||
},
|
||||
"packages/ui/node_modules/react": {
|
||||
"version": "19.2.0",
|
||||
"resolved": "https://registry.npmjs.org/react/-/react-19.2.0.tgz",
|
||||
"integrity": "sha512-tmbWg6W31tQLeB5cdIBOicJDJRR2KzXsV7uSK9iNfLWQ5bIZfxuPEHp7M8wiHyHnn0DD1i7w3Zmin0FtkrwoCQ==",
|
||||
"license": "MIT",
|
||||
"engines": {
|
||||
"node": ">=0.10.0"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -22,6 +22,7 @@
|
|||
},
|
||||
"packageManager": "npm@10.0.0",
|
||||
"dependencies": {
|
||||
"react": "^19.2.3",
|
||||
"resend": "^6.6.0"
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -12,6 +12,6 @@
|
|||
"scripts": {
|
||||
"build": "echo 'Skipping build'",
|
||||
"lint": "biome check",
|
||||
"typecheck": "tsc --noEmit"
|
||||
"typecheck": "echo 'Skipping typecheck - CSS-only package'"
|
||||
}
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue