Files
thrilltrack-explorer/supabase/functions/update-novu-preferences/index.ts
gpt-engineer-app[bot] a1280ddd05 Migrate Novu functions to wrapEdgeFunction
Refactor Phase 3 Batch 2–4 Novu-related functions to use the createEdgeFunction wrapper, replacing explicit HTTP servers with edge wrapper, adding standardized logging, tracing, and error handling across subscriber management, topic/notification, and migration/sync functions.
2025-11-11 03:55:02 +00:00

120 lines
3.8 KiB
TypeScript

import { Novu } from "npm:@novu/api@1.6.0";
import { createClient } from "https://esm.sh/@supabase/supabase-js@2.57.4";
import { corsHeaders } from '../_shared/cors.ts';
import { edgeLogger } from "../_shared/logger.ts";
import { createEdgeFunction } from '../_shared/edgeFunctionWrapper.ts';
export default createEdgeFunction(
{
name: 'update-novu-preferences',
requireAuth: false,
corsHeaders: corsHeaders
},
async (req, context) => {
const novuApiKey = Deno.env.get('NOVU_API_KEY');
const supabaseUrl = Deno.env.get('SUPABASE_URL')!;
const supabaseServiceKey = Deno.env.get('SUPABASE_SERVICE_ROLE_KEY')!;
if (!novuApiKey) {
throw new Error('NOVU_API_KEY is not configured');
}
const novu = new Novu({
secretKey: novuApiKey
});
const supabase = createClient(supabaseUrl, supabaseServiceKey);
const { userId, preferences } = await req.json();
context.span.setAttribute('action', 'update_novu_preferences');
edgeLogger.info('Updating preferences for user', { userId, requestId: context.requestId });
// Validate input
if (!userId) {
throw new Error('userId is required');
}
if (!preferences?.channelPreferences) {
throw new Error('channelPreferences is required in preferences object');
}
// Get Novu subscriber ID from database
const { data: prefData, error: prefError } = await supabase
.from('user_notification_preferences')
.select('novu_subscriber_id')
.eq('user_id', userId)
.single();
if (prefError || !prefData?.novu_subscriber_id) {
throw new Error('Novu subscriber not found');
}
const subscriberId = prefData.novu_subscriber_id;
// Update channel preferences in Novu
// Note: Novu's updatePreference updates one channel at a time for a specific workflow
const channelPrefs = preferences.channelPreferences;
const workflowId = preferences.workflowId || 'default';
const channelTypes = ['email', 'sms', 'in_app', 'push'] as const;
const results: { channel: string; success: boolean; error?: string }[] = [];
for (const channelType of channelTypes) {
if (channelPrefs[channelType] !== undefined) {
try {
await novu.subscribers.updatePreference(
subscriberId,
workflowId,
{
channel: {
type: channelType as any, // Cast to any to handle SDK type limitations
enabled: channelPrefs[channelType]
},
}
);
results.push({ channel: channelType, success: true });
} catch (channelError: any) {
edgeLogger.error('Failed to update channel preference', {
channel: channelType,
error: channelError.message,
requestId: context.requestId
});
results.push({
channel: channelType,
success: false,
error: channelError.message || 'Unknown error'
});
}
}
}
// Check if any updates failed
const failedChannels = results.filter(r => !r.success);
const allSucceeded = failedChannels.length === 0;
if (!allSucceeded) {
edgeLogger.warn('Some channel preferences failed to update', {
failedChannels: failedChannels.map(c => c.channel),
requestId: context.requestId
});
throw new Error('Some channel preferences failed to update: ' + failedChannels.map(c => c.channel).join(', '));
}
edgeLogger.info('All preferences updated successfully', {
requestId: context.requestId
});
return new Response(
JSON.stringify({
success: true,
results,
}),
{
headers: { 'Content-Type': 'application/json' },
status: 200,
}
);
}
);