mirror of
https://github.com/renee-png/acmcc.git
synced 2026-06-21 09:50:01 +00:00
Add ACMCC app source, Supabase backend, and project config
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,93 @@
|
||||
import { createClient } from "https://esm.sh/@supabase/supabase-js@2.99.1";
|
||||
|
||||
const corsHeaders = {
|
||||
"Access-Control-Allow-Origin": "*",
|
||||
"Access-Control-Allow-Headers": "authorization, x-client-info, apikey, content-type",
|
||||
};
|
||||
|
||||
Deno.serve(async (req) => {
|
||||
if (req.method === "OPTIONS") {
|
||||
return new Response("ok", { headers: corsHeaders });
|
||||
}
|
||||
|
||||
try {
|
||||
const supabaseUrl = Deno.env.get("SUPABASE_URL")!;
|
||||
const serviceKey = Deno.env.get("SUPABASE_SERVICE_ROLE_KEY")!;
|
||||
const supabase = createClient(supabaseUrl, serviceKey);
|
||||
|
||||
const today = new Date().toISOString().split("T")[0];
|
||||
|
||||
// Find all recurring expenses where next_date <= today
|
||||
const { data: recurring, error: fetchErr } = await supabase
|
||||
.from("billable_expenses")
|
||||
.select("*")
|
||||
.eq("is_recurring", true)
|
||||
.lte("recurring_next_date", today);
|
||||
|
||||
if (fetchErr) throw fetchErr;
|
||||
if (!recurring || recurring.length === 0) {
|
||||
return new Response(JSON.stringify({ created: 0 }), {
|
||||
headers: { ...corsHeaders, "Content-Type": "application/json" },
|
||||
});
|
||||
}
|
||||
|
||||
let created = 0;
|
||||
|
||||
for (const expense of recurring) {
|
||||
// Skip if past end date
|
||||
if (expense.recurring_end_date && expense.recurring_next_date > expense.recurring_end_date) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Create the new expense entry
|
||||
const { error: insertErr } = await supabase.from("billable_expenses").insert({
|
||||
association_id: expense.association_id,
|
||||
date: expense.recurring_next_date,
|
||||
description: expense.description,
|
||||
amount: expense.amount,
|
||||
category: expense.category,
|
||||
billable_type: expense.billable_type,
|
||||
is_credit: expense.is_credit,
|
||||
credit_reason: expense.credit_reason,
|
||||
quantity: expense.quantity,
|
||||
unit_price: expense.unit_price,
|
||||
vendor_name: expense.vendor_name,
|
||||
address: expense.address,
|
||||
status: "pending",
|
||||
recurring_parent_id: expense.id,
|
||||
is_recurring: false,
|
||||
});
|
||||
|
||||
if (insertErr) {
|
||||
console.error(`Failed to create recurring expense for ${expense.id}:`, insertErr);
|
||||
continue;
|
||||
}
|
||||
|
||||
created++;
|
||||
|
||||
// Advance the next date by one month
|
||||
const nextDate = new Date(expense.recurring_next_date + "T12:00:00");
|
||||
nextDate.setMonth(nextDate.getMonth() + 1);
|
||||
// Clamp day
|
||||
const day = expense.recurring_day || 1;
|
||||
const maxDay = new Date(nextDate.getFullYear(), nextDate.getMonth() + 1, 0).getDate();
|
||||
nextDate.setDate(Math.min(day, maxDay));
|
||||
const newNextDate = nextDate.toISOString().split("T")[0];
|
||||
|
||||
await supabase
|
||||
.from("billable_expenses")
|
||||
.update({ recurring_next_date: newNextDate })
|
||||
.eq("id", expense.id);
|
||||
}
|
||||
|
||||
return new Response(JSON.stringify({ created, processed: recurring.length }), {
|
||||
headers: { ...corsHeaders, "Content-Type": "application/json" },
|
||||
});
|
||||
} catch (err) {
|
||||
console.error("Error processing recurring expenses:", err);
|
||||
return new Response(JSON.stringify({ error: err.message }), {
|
||||
status: 500,
|
||||
headers: { ...corsHeaders, "Content-Type": "application/json" },
|
||||
});
|
||||
}
|
||||
});
|
||||
Reference in New Issue
Block a user