- Replace connect/disconnect pattern with persistent connection pool - Add explicit database initialization on server startup - Remove all dbDisconnect() calls from API endpoints to prevent race conditions - Fix MongoNotConnectedError when scheduler runs concurrently with API requests - Add connection pooling with proper MongoDB driver options - Add safety check for recipes array in favorites utility 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
124 lines
4.3 KiB
TypeScript
124 lines
4.3 KiB
TypeScript
import type { RequestHandler } from '@sveltejs/kit';
|
|
import { RecurringPayment } from '../../../../../models/RecurringPayment';
|
|
import { Payment } from '../../../../../models/Payment';
|
|
import { PaymentSplit } from '../../../../../models/PaymentSplit';
|
|
import { dbConnect } from '../../../../../utils/db';
|
|
import { error, json } from '@sveltejs/kit';
|
|
import { calculateNextExecutionDate } from '../../../../../lib/utils/recurring';
|
|
|
|
// This endpoint is designed to be called by a cron job or external scheduler
|
|
// It processes all recurring payments that are due for execution
|
|
export const POST: RequestHandler = async ({ request }) => {
|
|
// Optional: Add basic authentication or API key validation here
|
|
const authHeader = request.headers.get('authorization');
|
|
const expectedToken = process.env.CRON_API_TOKEN;
|
|
|
|
if (expectedToken && authHeader !== `Bearer ${expectedToken}`) {
|
|
throw error(401, 'Unauthorized');
|
|
}
|
|
|
|
await dbConnect();
|
|
|
|
try {
|
|
const now = new Date();
|
|
console.log(`[Cron] Starting recurring payments processing at ${now.toISOString()}`);
|
|
|
|
// Find all active recurring payments that are due
|
|
const duePayments = await RecurringPayment.find({
|
|
isActive: true,
|
|
nextExecutionDate: { $lte: now },
|
|
$or: [
|
|
{ endDate: { $exists: false } },
|
|
{ endDate: null },
|
|
{ endDate: { $gte: now } }
|
|
]
|
|
});
|
|
|
|
console.log(`[Cron] Found ${duePayments.length} due recurring payments`);
|
|
|
|
const results = [];
|
|
let successCount = 0;
|
|
let failureCount = 0;
|
|
|
|
for (const recurringPayment of duePayments) {
|
|
try {
|
|
console.log(`[Cron] Processing recurring payment: ${recurringPayment.title} (${recurringPayment._id})`);
|
|
|
|
// Create the payment
|
|
const payment = await Payment.create({
|
|
title: `${recurringPayment.title} (Auto)`,
|
|
description: `Automatically generated from recurring payment: ${recurringPayment.description || 'No description'}`,
|
|
amount: recurringPayment.amount,
|
|
currency: recurringPayment.currency,
|
|
paidBy: recurringPayment.paidBy,
|
|
date: now,
|
|
category: recurringPayment.category,
|
|
splitMethod: recurringPayment.splitMethod,
|
|
createdBy: recurringPayment.createdBy
|
|
});
|
|
|
|
// Create payment splits
|
|
const splitPromises = recurringPayment.splits.map((split) => {
|
|
return PaymentSplit.create({
|
|
paymentId: payment._id,
|
|
username: split.username,
|
|
amount: split.amount,
|
|
proportion: split.proportion,
|
|
personalAmount: split.personalAmount
|
|
});
|
|
});
|
|
|
|
await Promise.all(splitPromises);
|
|
|
|
// Calculate next execution date
|
|
const nextExecutionDate = calculateNextExecutionDate(recurringPayment, now);
|
|
|
|
// Update the recurring payment
|
|
await RecurringPayment.findByIdAndUpdate(recurringPayment._id, {
|
|
lastExecutionDate: now,
|
|
nextExecutionDate: nextExecutionDate
|
|
});
|
|
|
|
successCount++;
|
|
results.push({
|
|
recurringPaymentId: recurringPayment._id,
|
|
paymentId: payment._id,
|
|
title: recurringPayment.title,
|
|
amount: recurringPayment.amount,
|
|
nextExecution: nextExecutionDate,
|
|
success: true
|
|
});
|
|
|
|
console.log(`[Cron] Successfully processed: ${recurringPayment.title}, next execution: ${nextExecutionDate.toISOString()}`);
|
|
|
|
} catch (paymentError) {
|
|
console.error(`[Cron] Error processing recurring payment ${recurringPayment._id}:`, paymentError);
|
|
failureCount++;
|
|
results.push({
|
|
recurringPaymentId: recurringPayment._id,
|
|
title: recurringPayment.title,
|
|
amount: recurringPayment.amount,
|
|
success: false,
|
|
error: paymentError instanceof Error ? paymentError.message : 'Unknown error'
|
|
});
|
|
}
|
|
}
|
|
|
|
console.log(`[Cron] Completed processing. Success: ${successCount}, Failures: ${failureCount}`);
|
|
|
|
return json({
|
|
success: true,
|
|
timestamp: now.toISOString(),
|
|
processed: duePayments.length,
|
|
successful: successCount,
|
|
failed: failureCount,
|
|
results: results
|
|
});
|
|
|
|
} catch (e) {
|
|
console.error('[Cron] Error executing recurring payments:', e);
|
|
throw error(500, 'Failed to execute recurring payments');
|
|
} finally {
|
|
// Connection will be reused
|
|
}
|
|
}; |