mirror of
https://github.com/pacnpal/thrilltrack-explorer.git
synced 2025-12-20 11:51:14 -05:00
feat: Implement Novu username and update functionality
This commit is contained in:
@@ -16,6 +16,7 @@ import { useAuth } from '@/hooks/useAuth';
|
|||||||
import { supabase } from '@/integrations/supabase/client';
|
import { supabase } from '@/integrations/supabase/client';
|
||||||
import { User, Upload, Trash2 } from 'lucide-react';
|
import { User, Upload, Trash2 } from 'lucide-react';
|
||||||
import { PhotoUpload } from '@/components/upload/PhotoUpload';
|
import { PhotoUpload } from '@/components/upload/PhotoUpload';
|
||||||
|
import { notificationService } from '@/lib/notificationService';
|
||||||
|
|
||||||
const profileSchema = z.object({
|
const profileSchema = z.object({
|
||||||
username: z.string().min(3).max(30).regex(/^[a-zA-Z0-9_-]+$/),
|
username: z.string().min(3).max(30).regex(/^[a-zA-Z0-9_-]+$/),
|
||||||
@@ -54,6 +55,8 @@ export function AccountProfileTab() {
|
|||||||
|
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
try {
|
try {
|
||||||
|
const usernameChanged = profile?.username !== data.username;
|
||||||
|
|
||||||
const { error } = await supabase
|
const { error } = await supabase
|
||||||
.from('profiles')
|
.from('profiles')
|
||||||
.update({
|
.update({
|
||||||
@@ -69,6 +72,15 @@ export function AccountProfileTab() {
|
|||||||
|
|
||||||
if (error) throw error;
|
if (error) throw error;
|
||||||
|
|
||||||
|
// Update Novu subscriber if username changed
|
||||||
|
if (usernameChanged && notificationService.isEnabled()) {
|
||||||
|
await notificationService.updateSubscriber({
|
||||||
|
subscriberId: user.id,
|
||||||
|
email: user.email,
|
||||||
|
firstName: data.username, // Send username as firstName to Novu
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
await refreshProfile();
|
await refreshProfile();
|
||||||
toast({
|
toast({
|
||||||
title: 'Profile updated',
|
title: 'Profile updated',
|
||||||
|
|||||||
@@ -111,6 +111,14 @@ export function NotificationsTab() {
|
|||||||
|
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
toast.success("Notification preferences saved successfully");
|
toast.success("Notification preferences saved successfully");
|
||||||
|
|
||||||
|
// Also update Novu subscriber profile to sync channel preferences
|
||||||
|
if (notificationService.isEnabled()) {
|
||||||
|
await notificationService.updateSubscriber({
|
||||||
|
subscriberId: user.id,
|
||||||
|
email: user.email,
|
||||||
|
});
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
throw new Error(result.error || 'Failed to save preferences');
|
throw new Error(result.error || 'Failed to save preferences');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -71,6 +71,30 @@ class NotificationService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update an existing Novu subscriber's profile information
|
||||||
|
*/
|
||||||
|
async updateSubscriber(subscriberData: SubscriberData): Promise<{ success: boolean; error?: string }> {
|
||||||
|
if (!this.isNovuEnabled) {
|
||||||
|
console.warn('Novu is not configured. Skipping subscriber update.');
|
||||||
|
return { success: false, error: 'Novu not configured' };
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { data, error } = await supabase.functions.invoke('update-novu-subscriber', {
|
||||||
|
body: subscriberData,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error) throw error;
|
||||||
|
|
||||||
|
console.log('Novu subscriber updated successfully');
|
||||||
|
return { success: true };
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error('Error updating Novu subscriber:', error);
|
||||||
|
return { success: false, error: error.message };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update subscriber preferences in Novu
|
* Update subscriber preferences in Novu
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -147,7 +147,7 @@ export default function Auth() {
|
|||||||
notificationService.createSubscriber({
|
notificationService.createSubscriber({
|
||||||
subscriberId: data.user.id,
|
subscriberId: data.user.id,
|
||||||
email: formData.email,
|
email: formData.email,
|
||||||
firstName: formData.displayName || formData.username,
|
firstName: formData.username, // Send username as firstName to Novu
|
||||||
data: {
|
data: {
|
||||||
username: formData.username,
|
username: formData.username,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ serve(async (req) => {
|
|||||||
|
|
||||||
const { subscriberId, email, firstName, lastName, phone, avatar, data } = await req.json();
|
const { subscriberId, email, firstName, lastName, phone, avatar, data } = await req.json();
|
||||||
|
|
||||||
console.log('Creating Novu subscriber:', { subscriberId, email });
|
console.log('Creating Novu subscriber:', { subscriberId, email, firstName });
|
||||||
|
|
||||||
const subscriber = await novu.subscribers.identify(subscriberId, {
|
const subscriber = await novu.subscribers.identify(subscriberId, {
|
||||||
email,
|
email,
|
||||||
|
|||||||
64
supabase/functions/update-novu-subscriber/index.ts
Normal file
64
supabase/functions/update-novu-subscriber/index.ts
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
import { serve } from "https://deno.land/std@0.168.0/http/server.ts";
|
||||||
|
import { Novu } from "npm:@novu/node@2.0.2";
|
||||||
|
|
||||||
|
const corsHeaders = {
|
||||||
|
'Access-Control-Allow-Origin': '*',
|
||||||
|
'Access-Control-Allow-Headers': 'authorization, x-client-info, apikey, content-type',
|
||||||
|
};
|
||||||
|
|
||||||
|
serve(async (req) => {
|
||||||
|
if (req.method === 'OPTIONS') {
|
||||||
|
return new Response(null, { headers: corsHeaders });
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const novuApiKey = Deno.env.get('NOVU_API_KEY');
|
||||||
|
|
||||||
|
if (!novuApiKey) {
|
||||||
|
throw new Error('NOVU_API_KEY is not configured');
|
||||||
|
}
|
||||||
|
|
||||||
|
const novu = new Novu(novuApiKey, {
|
||||||
|
backendUrl: Deno.env.get('VITE_NOVU_API_URL') || 'https://api.novu.co',
|
||||||
|
});
|
||||||
|
|
||||||
|
const { subscriberId, email, firstName, lastName, phone, avatar, data } = await req.json();
|
||||||
|
|
||||||
|
console.log('Updating Novu subscriber:', { subscriberId, email, firstName });
|
||||||
|
|
||||||
|
const subscriber = await novu.subscribers.update(subscriberId, {
|
||||||
|
email,
|
||||||
|
firstName,
|
||||||
|
lastName,
|
||||||
|
phone,
|
||||||
|
avatar,
|
||||||
|
data,
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log('Subscriber updated successfully:', subscriber.data);
|
||||||
|
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
success: true,
|
||||||
|
subscriberId: subscriber.data._id,
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
headers: { ...corsHeaders, 'Content-Type': 'application/json' },
|
||||||
|
status: 200,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error('Error updating Novu subscriber:', error);
|
||||||
|
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
success: false,
|
||||||
|
error: error.message,
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
headers: { ...corsHeaders, 'Content-Type': 'application/json' },
|
||||||
|
status: 500,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user