Fix meet link creation to use edge function instead of n8n webhook
Changes: 1. AdminConsulting.tsx: Update createMeetLink to call Supabase edge function - Remove dependency on n8n webhook URL - Call create-google-meet-event edge function directly - Use environment variables for Supabase URL and anon key - Improve error handling and user feedback 2. handle-order-paid: Add comprehensive error logging - Log meet response status - Log full response data - Log errors when meet_link update fails - Log error response text when request fails - Better debugging for troubleshooting meet creation issues This fixes: - CORS issues when calling n8n webhook - 404 errors from deleted /webhook-test/create-link endpoint - Manual meet link creation now uses same flow as automatic - Better visibility into meet creation failures via logs 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -178,12 +178,12 @@ export default function AdminConsulting() {
|
|||||||
const createMeetLink = async () => {
|
const createMeetLink = async () => {
|
||||||
if (!selectedSlot) return;
|
if (!selectedSlot) return;
|
||||||
|
|
||||||
// Check if n8n webhook is configured
|
// Check if Google Calendar is configured
|
||||||
const webhookUrl = settings.integration_n8n_base_url;
|
const calendarId = settings.integration_google_calendar_id;
|
||||||
if (!webhookUrl) {
|
if (!calendarId) {
|
||||||
toast({
|
toast({
|
||||||
title: 'Info',
|
title: 'Info',
|
||||||
description: 'Webhook URL belum dikonfigurasi di Pengaturan Integrasi. Masukkan link Meet secara manual.',
|
description: 'Google Calendar ID belum dikonfigurasi di Pengaturan > Integrasi',
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -191,39 +191,44 @@ export default function AdminConsulting() {
|
|||||||
setCreatingMeet(true);
|
setCreatingMeet(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Call the webhook to create Google Meet link
|
// Call the Supabase edge function to create Google Meet link
|
||||||
const response = await fetch(`${webhookUrl}/create-meet`, {
|
const supabaseUrl = import.meta.env.VITE_SUPABASE_URL || 'https://lovable.backoffice.biz.id';
|
||||||
|
const response = await fetch(`${supabaseUrl}/functions/v1/create-google-meet-event`, {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'Authorization': `Bearer ${import.meta.env.VITE_SUPABASE_ANON_KEY || ''}`,
|
||||||
|
},
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
slot_id: selectedSlot.id,
|
slot_id: selectedSlot.id,
|
||||||
date: selectedSlot.date,
|
date: selectedSlot.date,
|
||||||
start_time: selectedSlot.start_time,
|
start_time: selectedSlot.start_time,
|
||||||
end_time: selectedSlot.end_time,
|
end_time: selectedSlot.end_time,
|
||||||
topic: selectedSlot.topic_category,
|
topic: selectedSlot.topic_category || 'Konsultasi',
|
||||||
client_name: selectedSlot.profiles?.name || 'Client',
|
client_name: selectedSlot.profiles?.name || 'Client',
|
||||||
client_email: selectedSlot.profiles?.email,
|
client_email: selectedSlot.profiles?.email || '',
|
||||||
calendar_id: settings.integration_google_calendar_id,
|
notes: `Manual creation by admin for slot ${selectedSlot.id}`,
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Webhook request failed');
|
const errorData = await response.json().catch(() => ({}));
|
||||||
|
throw new Error(errorData.message || 'Failed to create Meet link');
|
||||||
}
|
}
|
||||||
|
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
if (data.meet_link) {
|
if (data.success && data.meet_link) {
|
||||||
setMeetLink(data.meet_link);
|
setMeetLink(data.meet_link);
|
||||||
toast({ title: 'Berhasil', description: 'Link Google Meet dibuat' });
|
toast({ title: 'Berhasil', description: 'Link Google Meet dibuat' });
|
||||||
} else {
|
} else {
|
||||||
throw new Error('No meet_link in response');
|
throw new Error(data.message || 'No meet_link in response');
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error: any) {
|
||||||
console.error('Error creating meet link:', error);
|
console.error('Error creating meet link:', error);
|
||||||
toast({
|
toast({
|
||||||
title: 'Gagal',
|
title: 'Gagal',
|
||||||
description: 'Gagal membuat link Meet. Pastikan webhook sudah dikonfigurasi dengan benar.',
|
description: error.message || 'Gagal membuat link Meet. Pastikan Google Calendar sudah terintegrasi.',
|
||||||
variant: 'destructive',
|
variant: 'destructive',
|
||||||
});
|
});
|
||||||
} finally {
|
} finally {
|
||||||
|
|||||||
@@ -148,20 +148,36 @@ serve(async (req: Request): Promise<Response> => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
console.log("[HANDLE-PAID] Meet response status:", meetResponse.status);
|
||||||
|
|
||||||
if (meetResponse.ok) {
|
if (meetResponse.ok) {
|
||||||
const meetData = await meetResponse.json();
|
const meetData = await meetResponse.json();
|
||||||
|
console.log("[HANDLE-PAID] Meet response data:", meetData);
|
||||||
|
|
||||||
if (meetData.success) {
|
if (meetData.success) {
|
||||||
console.log("[HANDLE-PAID] Meet created:", meetData.meet_link);
|
console.log("[HANDLE-PAID] Meet created:", meetData.meet_link);
|
||||||
|
|
||||||
// Update all slots with the same meet link
|
// Update all slots with the same meet link
|
||||||
await supabase
|
const { error: updateError } = await supabase
|
||||||
.from("consulting_slots")
|
.from("consulting_slots")
|
||||||
.update({ meet_link: meetData.meet_link })
|
.update({ meet_link: meetData.meet_link })
|
||||||
.eq("order_id", order_id);
|
.eq("order_id", order_id);
|
||||||
|
|
||||||
|
if (updateError) {
|
||||||
|
console.error("[HANDLE-PAID] Failed to update meet_link:", updateError);
|
||||||
|
} else {
|
||||||
|
console.log("[HANDLE-PAID] Meet link updated for all slots in order:", order_id);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
console.error("[HANDLE-PAID] Meet creation returned success=false:", meetData);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const errorText = await meetResponse.text();
|
||||||
|
console.error("[HANDLE-PAID] Meet creation failed with status:", meetResponse.status);
|
||||||
|
console.error("[HANDLE-PAID] Error response:", errorText);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("[HANDLE-PAID] Meet creation failed:", error);
|
console.error("[HANDLE-PAID] Meet creation exception:", error);
|
||||||
// Don't fail the entire process
|
// Don't fail the entire process
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user