phonepe (#43)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m35s

Co-authored-by: AllenTJ7 <163137620+AllenTJ7@users.noreply.github.com>
Reviewed-on: #43
This commit is contained in:
Allen T J 2025-05-21 11:31:44 +00:00
parent 4b054fac56
commit d3cc6b3710
4 changed files with 114 additions and 70 deletions

View File

@ -65,11 +65,9 @@ export const directGenerateInvoice = onRequest({
paymentMethod: paymentMethod || 'Online' paymentMethod: paymentMethod || 'Online'
}; };
// Generate the invoice without updating any payment records
const invoicePath = await invoiceService.generateInvoice(invoiceData); const invoicePath = await invoiceService.generateInvoice(invoiceData);
const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath); const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath);
// Send email if requested
let emailSent = false; let emailSent = false;
if (sendEmail && email) { if (sendEmail && email) {
emailSent = await invoiceService.sendInvoiceEmail(invoicePath, { emailSent = await invoiceService.sendInvoiceEmail(invoicePath, {

View File

@ -35,7 +35,6 @@ export const getInvoiceUrl = onRequest({
return; return;
} }
// Get a download URL for the invoice
const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath as string); const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath as string);
response.json({ response.json({

View File

@ -43,7 +43,7 @@ export class InvoiceService {
async generateInvoice(data: InvoiceData): Promise<string> { async generateInvoice(data: InvoiceData): Promise<string> {
try { try {
const tempFilePath = path.join(os.tmpdir(), `invoice_${data.invoiceNumber}.pdf`); const tempFilePath = path.join(os.tmpdir(), `invoice_${data.invoiceNumber}.pdf`);
const hasGst = data.gstNumber && data.gstNumber.length > 0; const hasGst = data.gstNumber && data.gstNumber.length > 0;
const baseAmount = hasGst ? data.amount / 1.18 : data.amount; const baseAmount = hasGst ? data.amount / 1.18 : data.amount;
const sgst = hasGst ? baseAmount * 0.09 : 0; const sgst = hasGst ? baseAmount * 0.09 : 0;
@ -51,66 +51,110 @@ export class InvoiceService {
const formattedDate = format(data.paymentDate, 'dd/MM/yyyy'); const formattedDate = format(data.paymentDate, 'dd/MM/yyyy');
const doc = new jsPDF(); const doc = new jsPDF();
doc.setFontSize(20); doc.setFontSize(20);
doc.setFont('helvetica', 'bold'); doc.setFont('helvetica', 'bold');
doc.text(data.businessName, 20, 20); doc.text(data.businessName, 20, 20);
doc.setFontSize(12); doc.setFontSize(12);
doc.setFont('helvetica', 'normal'); doc.setFont('helvetica', 'normal');
doc.text(data.address, 20, 30);
const maxWidth = 170;
if (hasGst) { const lineHeight = 5;
doc.text(`GSTIN: ${data.gstNumber}`, 20, 40);
const addressLines = doc.splitTextToSize(data.address, maxWidth);
if (addressLines.length <= 2) {
for (let i = 0; i < addressLines.length; i++) {
doc.text(addressLines[i], 20, 30 + (i * lineHeight));
}
} else {
doc.text(addressLines[0], 20, 30);
let secondLine = addressLines[1];
if (secondLine.length > 3) {
secondLine = secondLine.substring(0, secondLine.length - 3) + '...';
} else {
secondLine += '...';
}
doc.text(secondLine, 20, 35);
} }
const gstYPosition = 40;
if (hasGst) {
doc.text(`GSTIN: ${data.gstNumber}`, 20, gstYPosition);
}
doc.setFontSize(24); doc.setFontSize(24);
doc.setFont('helvetica', 'bold'); doc.setFont('helvetica', 'bold');
doc.text('RECEIPT', 190, 20, { align: 'right' }); doc.text('RECEIPT', 190, 20, { align: 'right' });
doc.setFontSize(12); doc.setFontSize(12);
doc.text(`Receipt #: ${data.invoiceNumber}`, 190, 30, { align: 'right' }); doc.text(`Receipt #: ${data.invoiceNumber}`, 190, 30, { align: 'right' });
doc.text(`Date: ${formattedDate}`, 190, 40, { align: 'right' }); doc.text(`Date: ${formattedDate}`, 190, 40, { align: 'right' });
doc.line(20, 45, 190, 45); doc.line(20, 45, 190, 45);
doc.setFontSize(12); doc.setFontSize(12);
const receiptToBoxX = 15;
const receiptToBoxY = 55;
const receiptToBoxWidth = 175;
const receiptToBoxHeight = 45;
doc.setDrawColor(0, 0, 0);
doc.setLineWidth(0.5);
doc.rect(receiptToBoxX, receiptToBoxY, receiptToBoxWidth, receiptToBoxHeight);
doc.setFont('helvetica', 'bold'); doc.setFont('helvetica', 'bold');
doc.text('Receipt To:', 20, 60); doc.text('Receipt To:', 20, 60);
doc.setFont('helvetica', 'normal'); doc.setFont('helvetica', 'normal');
doc.text(data.customerName, 20, 70); doc.text(data.customerName, 20, 70);
doc.text(`Phone: ${data.phoneNumber}`, 20, 80); doc.text(`Phone: ${data.phoneNumber}`, 20, 80);
doc.text(`Email: ${data.email}`, 20, 90); doc.text(`Email: ${data.email}`, 20, 90);
autoTable(doc,{ autoTable(doc, {
startY: 110, startY: 110,
head: [['No.', 'Description', 'HSN/SAC', 'Amount (INR)']], head: [['No.', 'Description', 'HSN/SAC', 'Amount (INR)']],
body: [ body: [
['1', `${data.planName} Subscription`, '999723', baseAmount.toFixed(2)] ['1', `${data.planName} Subscription`, '999723', baseAmount.toFixed(2)]
], ],
headStyles: { fillColor: [220, 220, 220], textColor: [0, 0, 0], fontStyle: 'bold' }, headStyles: {
styles: { halign: 'center' }, fillColor: [220, 220, 220],
textColor: [0, 0, 0],
fontStyle: 'bold',
lineWidth: 0.5,
lineColor: [0, 0, 0]
},
styles: {
halign: 'center',
lineWidth: 0.5,
lineColor: [0, 0, 0]
},
columnStyles: { columnStyles: {
0: { halign: 'center', cellWidth: 20 }, 0: { halign: 'center', cellWidth: 20 },
1: { halign: 'left' }, 1: { halign: 'left' },
2: { halign: 'center', cellWidth: 40 }, 2: { halign: 'center', cellWidth: 40 },
3: { halign: 'right', cellWidth: 40 } 3: { halign: 'right', cellWidth: 40 }
} },
theme: 'grid',
tableLineWidth: 0.5,
tableLineColor: [0, 0, 0],
}); });
const finalY = (doc as any).lastAutoTable.finalY + 20; const finalY = (doc as any).lastAutoTable.finalY + 20;
if (hasGst) { if (hasGst) {
doc.text('Taxable Amount:', 150, finalY, { align: 'right' }); doc.text('Taxable Amount:', 150, finalY, { align: 'right' });
doc.text(`${baseAmount.toFixed(2)} INR`, 190, finalY, { align: 'right' }); doc.text(`${baseAmount.toFixed(2)} INR`, 190, finalY, { align: 'right' });
doc.text('SGST (9%):', 150, finalY + 10, { align: 'right' }); doc.text('SGST (9%):', 150, finalY + 10, { align: 'right' });
doc.text(`${sgst.toFixed(2)} INR`, 190, finalY + 10, { align: 'right' }); doc.text(`${sgst.toFixed(2)} INR`, 190, finalY + 10, { align: 'right' });
doc.text('CGST (9%):', 150, finalY + 20, { align: 'right' }); doc.text('CGST (9%):', 150, finalY + 20, { align: 'right' });
doc.text(`${cgst.toFixed(2)} INR`, 190, finalY + 20, { align: 'right' }); doc.text(`${cgst.toFixed(2)} INR`, 190, finalY + 20, { align: 'right' });
doc.setFont('helvetica', 'bold'); doc.setFont('helvetica', 'bold');
doc.text('Total Amount:', 150, finalY + 30, { align: 'right' }); doc.text('Total Amount:', 150, finalY + 30, { align: 'right' });
doc.text(`${data.amount.toFixed(2)} INR`, 190, finalY + 30, { align: 'right' }); doc.text(`${data.amount.toFixed(2)} INR`, 190, finalY + 30, { align: 'right' });
@ -119,27 +163,38 @@ export class InvoiceService {
doc.text('Total Amount:', 150, finalY, { align: 'right' }); doc.text('Total Amount:', 150, finalY, { align: 'right' });
doc.text(`${data.amount.toFixed(2)} INR`, 190, finalY, { align: 'right' }); doc.text(`${data.amount.toFixed(2)} INR`, 190, finalY, { align: 'right' });
} }
const paymentY = hasGst ? finalY + 50 : finalY + 20; const paymentY = hasGst ? finalY + 50 : finalY + 20;
doc.line(20, 45, 190, 45);
const boxX = 15;
const boxY = paymentY - 5;
const boxWidth = 175;
const boxHeight = 45;
doc.setDrawColor(0, 0, 0);
doc.setLineWidth(0.5);
doc.rect(boxX, boxY, boxWidth, boxHeight);
doc.setFont('helvetica', 'bold'); doc.setFont('helvetica', 'bold');
doc.text('Payment Information:', 20, paymentY); doc.text('Payment Information:', 20, paymentY);
doc.setFont('helvetica', 'normal'); doc.setFont('helvetica', 'normal');
doc.text(`Transaction ID: ${data.transactionId}`, 20, paymentY + 10); doc.text(`Transaction ID: ${data.transactionId}`, 20, paymentY + 10);
doc.text(`Payment Method: ${data.paymentMethod}`, 20, paymentY + 20); doc.text(`Payment Method: ${data.paymentMethod}`, 20, paymentY + 20);
doc.text(`Payment Date: ${formattedDate}`, 20, paymentY + 30); doc.text(`Payment Date: ${formattedDate}`, 20, paymentY + 30);
doc.setFontSize(12); doc.setFontSize(12);
doc.setFont('helvetica', 'italic'); doc.setFont('helvetica', 'italic');
doc.text('Thank you for your business!', 105, 270, { align: 'center' }); doc.text('Thank you for your business!', 105, 270, { align: 'center' });
doc.setFontSize(10); doc.setFontSize(10);
doc.setFont('helvetica', 'normal'); doc.setFont('helvetica', 'normal');
doc.text('This is a computer-generated receipt and does not require a signature.', 105, 280, { align: 'center' }); doc.text('This is a computer-generated receipt and does not require a signature.', 105, 280, { align: 'center' });
fs.writeFileSync(tempFilePath, Buffer.from(doc.output('arraybuffer'))); fs.writeFileSync(tempFilePath, Buffer.from(doc.output('arraybuffer')));
const invoicePath = `invoices/${data.invoiceNumber}.pdf`; const invoicePath = `invoices/${data.invoiceNumber}.pdf`;
const bucket = admin.storage().bucket(); const bucket = admin.storage().bucket();
await bucket.upload(tempFilePath, { await bucket.upload(tempFilePath, {
@ -148,9 +203,9 @@ export class InvoiceService {
contentType: 'application/pdf', contentType: 'application/pdf',
}, },
}); });
fs.unlinkSync(tempFilePath); fs.unlinkSync(tempFilePath);
return invoicePath; return invoicePath;
} catch (error: any) { } catch (error: any) {
logger.error('Error generating invoice:', error); logger.error('Error generating invoice:', error);
@ -162,15 +217,15 @@ export class InvoiceService {
try { try {
const bucket = admin.storage().bucket(); const bucket = admin.storage().bucket();
const file = bucket.file(invoicePath); const file = bucket.file(invoicePath);
const expirationMs = 7 * 24 * 60 * 60 * 1000; const expirationMs = 7 * 24 * 60 * 60 * 1000;
const [signedUrl] = await file.getSignedUrl({ const [signedUrl] = await file.getSignedUrl({
action: 'read', action: 'read',
expires: Date.now() + expirationMs, expires: Date.now() + expirationMs,
responseDisposition: `attachment; filename="${path.basename(invoicePath)}"`, responseDisposition: `attachment; filename="${path.basename(invoicePath)}"`,
}); });
return signedUrl; return signedUrl;
} catch (error: any) { } catch (error: any) {
logger.error('Error getting invoice download URL:', error); logger.error('Error getting invoice download URL:', error);
@ -183,37 +238,37 @@ export class InvoiceService {
const membershipPaymentsRef = admin.firestore() const membershipPaymentsRef = admin.firestore()
.collection('membership_payments') .collection('membership_payments')
.doc(membershipId); .doc(membershipId);
const docSnapshot = await membershipPaymentsRef.get(); const docSnapshot = await membershipPaymentsRef.get();
if (!docSnapshot.exists) { if (!docSnapshot.exists) {
logger.error(`No membership payments found for membershipId: ${membershipId}`); logger.error(`No membership payments found for membershipId: ${membershipId}`);
return false; return false;
} }
const data = docSnapshot.data(); const data = docSnapshot.data();
const paymentsData = data?.payments || []; const paymentsData = data?.payments || [];
let found = false; let found = false;
for (let i = 0; i < paymentsData.length; i++) { for (let i = 0; i < paymentsData.length; i++) {
if (paymentsData[i].referenceNumber === paymentId || if (paymentsData[i].referenceNumber === paymentId ||
paymentsData[i].transactionId === paymentId) { paymentsData[i].transactionId === paymentId) {
paymentsData[i].invoicePath = invoicePath; paymentsData[i].invoicePath = invoicePath;
found = true; found = true;
break; break;
} }
} }
if (!found) { if (!found) {
logger.error(`No payment found with ID: ${paymentId} in membership: ${membershipId}`); logger.error(`No payment found with ID: ${paymentId} in membership: ${membershipId}`);
return false; return false;
} }
await membershipPaymentsRef.update({ await membershipPaymentsRef.update({
'payments': paymentsData, 'payments': paymentsData,
'updatedAt': admin.firestore.FieldValue.serverTimestamp(), 'updatedAt': admin.firestore.FieldValue.serverTimestamp(),
}); });
logger.info(`Successfully updated invoice path for payment: ${paymentId}`); logger.info(`Successfully updated invoice path for payment: ${paymentId}`);
return true; return true;
} catch (error: any) { } catch (error: any) {
@ -225,11 +280,11 @@ export class InvoiceService {
async sendInvoiceEmail(invoicePath: string, emailOptions: EmailOptions): Promise<boolean> { async sendInvoiceEmail(invoicePath: string, emailOptions: EmailOptions): Promise<boolean> {
try { try {
const downloadUrl = await this.getInvoiceDownloadUrl(invoicePath); const downloadUrl = await this.getInvoiceDownloadUrl(invoicePath);
const formattedDate = emailOptions.additionalData?.paymentDate const formattedDate = emailOptions.additionalData?.paymentDate
? new Date(emailOptions.additionalData.paymentDate).toLocaleDateString('en-GB') ? new Date(emailOptions.additionalData.paymentDate).toLocaleDateString('en-GB')
: new Date().toLocaleDateString('en-GB'); : new Date().toLocaleDateString('en-GB');
const emailHtml = emailOptions.customHtml || ` const emailHtml = emailOptions.customHtml || `
<html> <html>
<body> <body>
@ -251,7 +306,7 @@ export class InvoiceService {
</body> </body>
</html> </html>
`; `;
await sendEmailWithAttachmentUtil( await sendEmailWithAttachmentUtil(
emailOptions.recipientEmail, emailOptions.recipientEmail,
emailOptions.subject || 'Your Fitlien Membership Invoice', emailOptions.subject || 'Your Fitlien Membership Invoice',
@ -259,7 +314,7 @@ export class InvoiceService {
downloadUrl, downloadUrl,
`Invoice_${path.basename(invoicePath)}` `Invoice_${path.basename(invoicePath)}`
); );
logger.info(`Invoice email sent to ${emailOptions.recipientEmail}`); logger.info(`Invoice email sent to ${emailOptions.recipientEmail}`);
return true; return true;
} catch (error: any) { } catch (error: any) {
@ -269,7 +324,7 @@ export class InvoiceService {
} }
async processInvoice( async processInvoice(
membershipId: string, membershipId: string,
paymentId: string, paymentId: string,
invoiceData: InvoiceData, invoiceData: InvoiceData,
emailOptions?: EmailOptions emailOptions?: EmailOptions
@ -282,9 +337,9 @@ export class InvoiceService {
}> { }> {
try { try {
const invoicePath = await this.generateInvoice(invoiceData); const invoicePath = await this.generateInvoice(invoiceData);
const updateSuccess = await this.updateInvoicePath(membershipId, paymentId, invoicePath); const updateSuccess = await this.updateInvoicePath(membershipId, paymentId, invoicePath);
if (!updateSuccess) { if (!updateSuccess) {
return { return {
success: false, success: false,
@ -293,14 +348,14 @@ export class InvoiceService {
error: 'Failed to update payment with invoice path' error: 'Failed to update payment with invoice path'
}; };
} }
const downloadUrl = await this.getInvoiceDownloadUrl(invoicePath); const downloadUrl = await this.getInvoiceDownloadUrl(invoicePath);
let emailSent = false; let emailSent = false;
if (emailOptions && emailOptions.recipientEmail) { if (emailOptions && emailOptions.recipientEmail) {
emailSent = await this.sendInvoiceEmail(invoicePath, emailOptions); emailSent = await this.sendInvoiceEmail(invoicePath, emailOptions);
} }
return { return {
success: true, success: true,
invoicePath, invoicePath,

View File

@ -3,7 +3,6 @@ import { getAdmin, getLogger } from "../../shared/config";
const admin = getAdmin(); const admin = getAdmin();
const logger = getLogger(); const logger = getLogger();
// Define an interface for the payment data to avoid type errors
interface PaymentData { interface PaymentData {
id: string; id: string;
date: string; date: string;
@ -14,7 +13,7 @@ interface PaymentData {
discount: any; discount: any;
transactionId: string; transactionId: string;
createdAt: Date; createdAt: Date;
invoicePath?: string; // Make this optional invoicePath?: string;
} }
export async function updatePaymentDataAfterSuccess( export async function updatePaymentDataAfterSuccess(
@ -24,7 +23,6 @@ export async function updatePaymentDataAfterSuccess(
invoicePath?: string invoicePath?: string
): Promise<boolean> { ): Promise<boolean> {
try { try {
// Get the payment order from Firestore
const orderQuery = await admin.firestore() const orderQuery = await admin.firestore()
.collection('payment_orders') .collection('payment_orders')
.where('merchantOrderId', '==', merchantOrderId) .where('merchantOrderId', '==', merchantOrderId)
@ -39,19 +37,17 @@ export async function updatePaymentDataAfterSuccess(
const orderDoc = orderQuery.docs[0]; const orderDoc = orderQuery.docs[0];
const orderData = orderDoc.data(); const orderData = orderDoc.data();
// Extract membership ID from metaInfo
const membershipId = orderData.metaInfo?.membershipId; const membershipId = orderData.metaInfo?.membershipId;
if (!membershipId) { if (!membershipId) {
logger.error(`No membershipId found in metaInfo for order: ${merchantOrderId}`); logger.error(`No membershipId found in metaInfo for order: ${merchantOrderId}`);
return false; return false;
} }
const isoDate = new Date().toLocaleDateString('en-GB').split('/').join('-'); // DD-MM-YYYY format const isoDate = new Date().toLocaleDateString('en-GB').split('/').join('-');
const dateTimestamp = admin.firestore.Timestamp.now(); const dateTimestamp = admin.firestore.Timestamp.now();
// Create payment data object with proper typing
const paymentData: PaymentData = { const paymentData: PaymentData = {
id: admin.firestore().collection('_').doc().id, // Generate a UUID id: admin.firestore().collection('_').doc().id,
date: isoDate, date: isoDate,
dateTimestamp: dateTimestamp, dateTimestamp: dateTimestamp,
amount: orderData.amount, amount: orderData.amount,
@ -62,19 +58,16 @@ export async function updatePaymentDataAfterSuccess(
createdAt: new Date() createdAt: new Date()
}; };
// Add invoice path if provided
if (invoicePath) { if (invoicePath) {
paymentData.invoicePath = invoicePath; paymentData.invoicePath = invoicePath;
} }
// Get reference to membership payments document
const membershipPaymentsRef = admin.firestore() const membershipPaymentsRef = admin.firestore()
.collection('membership_payments') .collection('membership_payments')
.doc(membershipId); .doc(membershipId);
const docSnapshot = await membershipPaymentsRef.get(); const docSnapshot = await membershipPaymentsRef.get();
// Update or create the membership payments document
if (docSnapshot.exists) { if (docSnapshot.exists) {
await membershipPaymentsRef.update({ await membershipPaymentsRef.update({
'payments': admin.firestore.FieldValue.arrayUnion(paymentData), 'payments': admin.firestore.FieldValue.arrayUnion(paymentData),
@ -92,7 +85,6 @@ export async function updatePaymentDataAfterSuccess(
}); });
} }
// Update membership status
await updateMembershipStatus(membershipId, orderData.userId); await updateMembershipStatus(membershipId, orderData.userId);
logger.info(`Successfully updated payment data for membership: ${membershipId}`); logger.info(`Successfully updated payment data for membership: ${membershipId}`);
@ -118,7 +110,7 @@ async function updateMembershipStatus(membershipId: string, userId: string): Pro
.collection('memberships') .collection('memberships')
.doc(membershipId) .doc(membershipId)
.update({ .update({
'status': 'ACTIVE', // Assuming this matches your InvitationStatus.active 'status': 'ACTIVE',
'updatedAt': admin.firestore.FieldValue.serverTimestamp(), 'updatedAt': admin.firestore.FieldValue.serverTimestamp(),
}); });