From f221f4850ea36a767cf32e22eaec1fe4ad52e5f1 Mon Sep 17 00:00:00 2001 From: DhanshCOSQ Date: Sat, 5 Apr 2025 12:13:49 +0530 Subject: [PATCH 1/3] Signed url complete --- functions/src/index.ts | 86 ++++++++++++++++++++++++++++++------------ 1 file changed, 61 insertions(+), 25 deletions(-) diff --git a/functions/src/index.ts b/functions/src/index.ts index c5f1c80..99a6e85 100644 --- a/functions/src/index.ts +++ b/functions/src/index.ts @@ -9,7 +9,7 @@ import * as path from 'path'; import * as fs from 'fs'; import * as https from 'https'; import axios from "axios"; - +import { getStorage } from 'firebase-admin/storage'; const formData = require('form-data'); const Mailgun = require('mailgun.js'); const { convert } = require('html-to-text'); @@ -79,6 +79,42 @@ export const sendEmailWithAttachment = onRequest({ response.status(500).json({ success: false, error: error instanceof Error ? error.message : String(error) }); } }); + +export const accessFile = onRequest({ + region: '#{SERVICES_RGN}#' +}, async (request: Request, response: express.Response) => { + try { + const filePath = request.query.path as string; + if (!filePath) { + response.status(400).send('File path is required'); + return; + } + + const expirationMs = 30 * 24 * 60 * 60 * 1000; + + const bucket = getStorage().bucket(); + const file = bucket.file(filePath); + + const [exists] = await file.exists(); + if (!exists) { + response.status(404).send('File not found'); + return; + } + + const [signedUrl] = await file.getSignedUrl({ + action: 'read', + expires: Date.now() + expirationMs, + responseDisposition: `attachment; filename="${path.basename(filePath)}"`, + }); + + response.redirect(signedUrl); + logger.info(`File access redirect for ${filePath}`); + } catch (error) { + logger.error('Error accessing file:', error); + response.status(500).send('Error accessing file'); + } +}); + export const sendEmailMessage = onRequest({ region: process.env.SERVICES_RGN }, (request: Request, response: express.Response) => { @@ -220,12 +256,12 @@ export const createCashfreeOrder = onRequest({ const decodedToken = await admin.auth().verifyIdToken(idToken); const uid = decodedToken.uid; - const { - amount, - customerName, - customerEmail, - customerPhone, - productInfo + const { + amount, + customerName, + customerEmail, + customerPhone, + productInfo } = request.body; if (!amount || !customerEmail || !customerPhone) { @@ -236,13 +272,13 @@ export const createCashfreeOrder = onRequest({ const clientId = process.env.CASHFREE_CLIENT_ID; const clientSecret = process.env.CASHFREE_CLIENT_SECRET; const isTest = true; - - const apiUrl = isTest + + const apiUrl = isTest ? 'https://sandbox.cashfree.com/pg/orders' : 'https://api.cashfree.com/pg/orders'; - + const orderId = `order_${Date.now()}_${uid.substring(0, 6)}`; - + const cashfreeResponse = await axios.post( apiUrl, { @@ -270,7 +306,7 @@ export const createCashfreeOrder = onRequest({ } } ); - + await admin.firestore().collection('payment_orders').doc(orderId).set({ userId: uid, amount: amount, @@ -281,16 +317,16 @@ export const createCashfreeOrder = onRequest({ createdAt: new Date(), ...cashfreeResponse.data }); - + response.json({ order_id: cashfreeResponse.data.order_id, payment_session_id: cashfreeResponse.data.payment_session_id }); - + logger.info(`Cashfree order created: ${orderId}`); } catch (error: any) { logger.error('Cashfree order creation error:', error); - response.status(500).json({ + response.status(500).json({ error: 'Failed to create payment order', details: error.response?.data || error.message }); @@ -302,20 +338,20 @@ export const verifyCashfreePayment = onRequest({ }, async (request: Request, response: express.Response) => { try { const orderId = request.body.order_id || request.query.order_id; - + if (!orderId) { response.status(400).json({ error: 'Order ID is required' }); return; } - + const clientId = process.env.CASHFREE_CLIENT_ID; const clientSecret = process.env.CASHFREE_CLIENT_SECRET; const isTest = process.env.CASHFREE_ENVIRONMENT !== 'production'; - - const apiUrl = isTest + + const apiUrl = isTest ? `https://sandbox.cashfree.com/pg/orders/${orderId}` : `https://api.cashfree.com/pg/orders/${orderId}`; - + const cashfreeResponse = await axios.get( apiUrl, { @@ -326,27 +362,27 @@ export const verifyCashfreePayment = onRequest({ } } ); - + await admin.firestore().collection('payment_orders').doc(orderId).update({ orderStatus: cashfreeResponse.data.order_status, paymentDetails: cashfreeResponse.data, updatedAt: new Date() }); - + if (request.headers['x-webhook-source'] === 'cashfree') { response.status(200).send('OK'); return; } - + response.json({ status: cashfreeResponse.data.order_status, paymentDetails: cashfreeResponse.data }); - + logger.info(`Cashfree payment verified: ${orderId}`); } catch (error: any) { logger.error('Cashfree payment verification error:', error); - response.status(500).json({ + response.status(500).json({ error: 'Failed to verify payment status', details: error.response?.data || error.message }); From e553bd673a91112cfaaa1a0a16b0f9075c35ea41 Mon Sep 17 00:00:00 2001 From: DhanshCOSQ Date: Sat, 5 Apr 2025 14:22:47 +0530 Subject: [PATCH 2/3] Time changed to one hour --- functions/src/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/functions/src/index.ts b/functions/src/index.ts index 99a6e85..db89fe4 100644 --- a/functions/src/index.ts +++ b/functions/src/index.ts @@ -90,7 +90,7 @@ export const accessFile = onRequest({ return; } - const expirationMs = 30 * 24 * 60 * 60 * 1000; + const expirationMs = 60 * 60 * 1000; const bucket = getStorage().bucket(); const file = bucket.file(filePath); From b4a75f4968408de62aa0e2b70972d4861406c183 Mon Sep 17 00:00:00 2001 From: DhanshCOSQ Date: Sat, 5 Apr 2025 14:58:52 +0530 Subject: [PATCH 3/3] Changed region --- functions/src/index.ts | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/functions/src/index.ts b/functions/src/index.ts index db89fe4..a26adc6 100644 --- a/functions/src/index.ts +++ b/functions/src/index.ts @@ -116,7 +116,7 @@ export const accessFile = onRequest({ }); export const sendEmailMessage = onRequest({ - region: process.env.SERVICES_RGN + region: '#{SERVICES_RGN}#' }, (request: Request, response: express.Response) => { const mailgun = new Mailgun(formData); const mailGunClient = mailgun.client({ username: 'api', key: process.env.MAILGUN_API_KEY }); @@ -145,7 +145,7 @@ export const sendEmailMessage = onRequest({ }); export const sendSMSMessage = onRequest({ - region: process.env.SERVICES_RGN + region: '#{SERVICES_RGN}#' }, (request: Request, response: express.Response) => { const client = twilio(process.env.TWILIO_ACCOUNT_SID, process.env.TWILIO_AUTH_TOKEN); const { to, body } = request.body; @@ -174,7 +174,7 @@ interface Invitation { export const notifyInvitation = onDocumentCreated({ document: 'notifications/{notificationId}', - region: process.env.SERVICES_RGN + region: '#{SERVICES_RGN}#' }, async (event: any) => { const invitation = event.data?.data() as Invitation; const invitationId = event.params.invitationId; @@ -243,7 +243,7 @@ export const notifyInvitation = onDocumentCreated({ }); export const createCashfreeOrder = onRequest({ - region: process.env.SERVICES_RGN + region: '#{SERVICES_RGN}#' }, async (request: Request, response: express.Response) => { try { const authHeader = request.headers.authorization; @@ -334,7 +334,7 @@ export const createCashfreeOrder = onRequest({ }); export const verifyCashfreePayment = onRequest({ - region: process.env.SERVICES_RGN + region: '#{SERVICES_RGN}#' }, async (request: Request, response: express.Response) => { try { const orderId = request.body.order_id || request.query.order_id;