From 36597f544aa62e6b161dedd5b9184e4038ba0492 Mon Sep 17 00:00:00 2001 From: Joseph Shearer Date: Wed, 13 Sep 2023 16:28:25 -0400 Subject: [PATCH] feature: Introduce `get-tenant-invoice` API request to fetch the invoice metadata for a particular tenant and month --- .../billing/get_tenant_invoice_data.ts | 51 +++++++++++++++++++ supabase/functions/billing/index.ts | 3 ++ 2 files changed, 54 insertions(+) create mode 100644 supabase/functions/billing/get_tenant_invoice_data.ts diff --git a/supabase/functions/billing/get_tenant_invoice_data.ts b/supabase/functions/billing/get_tenant_invoice_data.ts new file mode 100644 index 0000000000..8dfa05dcd2 --- /dev/null +++ b/supabase/functions/billing/get_tenant_invoice_data.ts @@ -0,0 +1,51 @@ +import { customerQuery, StripeClient } from "./shared.ts"; + +export interface getTenantPaymentMethodsParams { + tenant: string; + month: string; + type: "Usage" | "Manual"; +} + +const INVOICE_TYPE_KEY = "estuary.dev/invoice_type"; +const BILLING_PERIOD_START_KEY = "estuary.dev/period_start"; + +export async function getTenantInvoiceData( + req_body: getTenantPaymentMethodsParams, + full_req: Request, +): Promise> { + const customer = (await StripeClient.customers.search({ query: customerQuery(req_body.tenant) })).data[0]; + const parsed_date = new Date(req_body.month); + + const year = new Intl.DateTimeFormat("en", { year: "numeric" }).format(parsed_date); + const month = new Intl.DateTimeFormat("en", { month: "2-digit" }).format(parsed_date); + + // We always start on the first of the month + const metadata_date = `${year}-${month}-01`; + + if (customer) { + const query = + `customer:"${customer.id}" AND metadata["${INVOICE_TYPE_KEY}"]:"${req_body.type}" AND metadata["${BILLING_PERIOD_START_KEY}"]:"${metadata_date}" AND -status:"draft"`; + const resp = await StripeClient.invoices.search({ + query, + }); + + if (resp.data[0]) { + const limited_invoice = { + id: resp.data[0].id, + amount_due: resp.data[0].amount_due, + invoice_pdf: resp.data[0].invoice_pdf, + hosted_invoice_url: resp.data[0].hosted_invoice_url, + }; + + return [JSON.stringify({ invoice: limited_invoice }), { + headers: { "Content-Type": "application/json" }, + status: 200, + }]; + } + } + + return [JSON.stringify({ invoice: null }), { + headers: { "Content-Type": "application/json" }, + status: 200, + }]; +} diff --git a/supabase/functions/billing/index.ts b/supabase/functions/billing/index.ts index 9fa8a0826a..28b5ac40e1 100644 --- a/supabase/functions/billing/index.ts +++ b/supabase/functions/billing/index.ts @@ -5,6 +5,7 @@ import { getTenantPaymentMethods } from "./get_tenant_payment_methods.ts"; import { deleteTenantPaymentMethod } from "./delete_tenant_payment_method.ts"; import { setTenantPrimaryPaymentMethod } from "./set_tenant_primary_payment_method.ts"; import { createClient } from "https://esm.sh/@supabase/supabase-js@2.0.5"; +import { getTenantInvoiceData } from "./get_tenant_invoice_data.ts"; // Now that the supabase CLI supports multiple edge functions, // we should refactor this into individual functions instead @@ -54,6 +55,8 @@ serve(async (req) => { res = await deleteTenantPaymentMethod(request, req); } else if (request.operation === "set-tenant-primary-payment-method") { res = await setTenantPrimaryPaymentMethod(request, req); + } else if (request.operation === "get-tenant-invoice") { + res = await getTenantInvoiceData(request, req); } else { res = [JSON.stringify({ error: "unknown_operation" }), { headers: { "Content-Type": "application/json" },