notification-bug-fix (#84)
	
		
			
	
		
	
	
		
	
		
			All checks were successful
		
		
	
	
		
			
				
	
				Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m10s
				
			
		
		
	
	
				
					
				
			
		
			All checks were successful
		
		
	
	Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m10s
				
			Reviewed-on: #84 Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net> Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
This commit is contained in:
		
							parent
							
								
									4cf5692386
								
							
						
					
					
						commit
						b66f1603cc
					
				| @ -166,6 +166,20 @@ | |||||||
|         } |         } | ||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|  |     { | ||||||
|  |       "collectionGroup": "notifications", | ||||||
|  |       "queryScope": "COLLECTION", | ||||||
|  |       "fields": [ | ||||||
|  |         { | ||||||
|  |           "fieldPath": "recipientId", | ||||||
|  |           "order": "ASCENDING" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "fieldPath": "timestamp", | ||||||
|  |           "order": "DESCENDING" | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     }, | ||||||
|     { |     { | ||||||
|       "collectionGroup": "workout_logs", |       "collectionGroup": "workout_logs", | ||||||
|       "queryScope": "COLLECTION", |       "queryScope": "COLLECTION", | ||||||
|  | |||||||
| @ -33,6 +33,15 @@ interface PaymentData { | |||||||
|   discount?: number; |   discount?: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | interface PersonalTrainerAssign { | ||||||
|  |   id: string; | ||||||
|  |   ownerId: string; | ||||||
|  |   trainerId?: string; | ||||||
|  |   clientId: string; | ||||||
|  |   membershipId: string; | ||||||
|  |   gymId: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export const checkExpiredMemberships = onSchedule( | export const checkExpiredMemberships = onSchedule( | ||||||
|   { |   { | ||||||
|     schedule: "*/5 * * * *", |     schedule: "*/5 * * * *", | ||||||
| @ -63,10 +72,18 @@ export const checkExpiredMemberships = onSchedule( | |||||||
|         expiringMemberships.map((m) => processExpiringMembership(m.id, m.data)) |         expiringMemberships.map((m) => processExpiringMembership(m.id, m.data)) | ||||||
|       ); |       ); | ||||||
| 
 | 
 | ||||||
|       const expiredSuccessful = expiredResults.filter((r) => r.status === "fulfilled").length; |       const expiredSuccessful = expiredResults.filter( | ||||||
|       const expiredFailed = expiredResults.filter((r) => r.status === "rejected").length; |         (r) => r.status === "fulfilled" | ||||||
|       const expiringSuccessful = expiringResults.filter((r) => r.status === "fulfilled").length; |       ).length; | ||||||
|       const expiringFailed = expiringResults.filter((r) => r.status === "rejected").length; |       const expiredFailed = expiredResults.filter( | ||||||
|  |         (r) => r.status === "rejected" | ||||||
|  |       ).length; | ||||||
|  |       const expiringSuccessful = expiringResults.filter( | ||||||
|  |         (r) => r.status === "fulfilled" | ||||||
|  |       ).length; | ||||||
|  |       const expiringFailed = expiringResults.filter( | ||||||
|  |         (r) => r.status === "rejected" | ||||||
|  |       ).length; | ||||||
| 
 | 
 | ||||||
|       logger.info( |       logger.info( | ||||||
|         `Completed processing. Expired - Success: ${expiredSuccessful}, Failed: ${expiredFailed}. Expiring - Success: ${expiringSuccessful}, Failed: ${expiringFailed}` |         `Completed processing. Expired - Success: ${expiredSuccessful}, Failed: ${expiredFailed}. Expiring - Success: ${expiringSuccessful}, Failed: ${expiringFailed}` | ||||||
| @ -139,7 +156,10 @@ async function findMembershipsExpiringIn10Days(): Promise< | |||||||
|       const batchResults = await Promise.allSettled( |       const batchResults = await Promise.allSettled( | ||||||
|         batch.map(async (doc) => { |         batch.map(async (doc) => { | ||||||
|           const data = doc.data() as MembershipData; |           const data = doc.data() as MembershipData; | ||||||
|           const isExpiringIn10Days = await checkIfMembershipExpiringIn10Days(doc.id, data); |           const isExpiringIn10Days = await checkIfMembershipExpiringIn10Days( | ||||||
|  |             doc.id, | ||||||
|  |             data | ||||||
|  |           ); | ||||||
|           if (isExpiringIn10Days) { |           if (isExpiringIn10Days) { | ||||||
|             return { id: doc.id, data }; |             return { id: doc.id, data }; | ||||||
|           } |           } | ||||||
| @ -356,6 +376,66 @@ function calculateRenewalDateFromPayment( | |||||||
|   return renewalDate; |   return renewalDate; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | async function getTrainerAssignmentsForMembership( | ||||||
|  |   membershipId: string | ||||||
|  | ): Promise<PersonalTrainerAssign[]> { | ||||||
|  |   try { | ||||||
|  |     const querySnapshot = await app | ||||||
|  |       .firestore() | ||||||
|  |       .collection("personal_trainer_assignments") | ||||||
|  |       .where("membershipId", "==", membershipId) | ||||||
|  |       .get(); | ||||||
|  | 
 | ||||||
|  |     return querySnapshot.docs.map((doc) => ({ | ||||||
|  |       id: doc.id, | ||||||
|  |       ...doc.data(), | ||||||
|  |     })) as PersonalTrainerAssign[]; | ||||||
|  |   } catch (error) { | ||||||
|  |     logger.error( | ||||||
|  |       `Error getting trainer assignments for membership ${membershipId}:`, | ||||||
|  |       error | ||||||
|  |     ); | ||||||
|  |     return []; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function getTrainerName(trainerId: string): Promise<string> { | ||||||
|  |   try { | ||||||
|  |     const doc = await app | ||||||
|  |       .firestore() | ||||||
|  |       .collection("trainer_profiles") | ||||||
|  |       .doc(trainerId) | ||||||
|  |       .get(); | ||||||
|  |      | ||||||
|  |     if (!doc.exists) { | ||||||
|  |       const userDoc = await app | ||||||
|  |         .firestore() | ||||||
|  |         .collection("users") | ||||||
|  |         .doc(trainerId) | ||||||
|  |         .get(); | ||||||
|  |        | ||||||
|  |       if (userDoc.exists) { | ||||||
|  |         const userData = userDoc.data(); | ||||||
|  |         return userData?.name || userData?.displayName || "Unknown Trainer"; | ||||||
|  |       } | ||||||
|  |       return "Unknown Trainer"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const data = doc.data(); | ||||||
|  |     const fields = data?.fields; | ||||||
|  |     if (fields) { | ||||||
|  |       const firstName = fields["first-name"] || ""; | ||||||
|  |       const lastName = fields["last-name"] || ""; | ||||||
|  |       return `${firstName} ${lastName}`.trim() || "Unknown Trainer"; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     return data?.name || data?.displayName || "Unknown Trainer"; | ||||||
|  |   } catch (error) { | ||||||
|  |     logger.error(`Error getting trainer name for ${trainerId}:`, error); | ||||||
|  |     return "Unknown Trainer"; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| async function processExpiredMembership( | async function processExpiredMembership( | ||||||
|   membershipId: string, |   membershipId: string, | ||||||
|   membershipData: MembershipData |   membershipData: MembershipData | ||||||
| @ -367,7 +447,10 @@ async function processExpiredMembership( | |||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     logger.info(`Marked membership ${membershipId} as EXPIRED`); |     logger.info(`Marked membership ${membershipId} as EXPIRED`); | ||||||
|  |      | ||||||
|     await sendPlanExpiredNotification(membershipId, membershipData); |     await sendPlanExpiredNotification(membershipId, membershipData); | ||||||
|  |      | ||||||
|  |     await sendTrainerNotifications(membershipId, membershipData, "expired"); | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     logger.error(`Error processing membership ${membershipId}:`, error); |     logger.error(`Error processing membership ${membershipId}:`, error); | ||||||
|   } |   } | ||||||
| @ -379,9 +462,130 @@ async function processExpiringMembership( | |||||||
| ): Promise<void> { | ): Promise<void> { | ||||||
|   try { |   try { | ||||||
|     logger.info(`Processing expiring membership ${membershipId}`); |     logger.info(`Processing expiring membership ${membershipId}`); | ||||||
|  |      | ||||||
|     await sendPlanExpiringNotification(membershipId, membershipData); |     await sendPlanExpiringNotification(membershipId, membershipData); | ||||||
|  |      | ||||||
|  |     await sendTrainerNotifications(membershipId, membershipData, "expiring"); | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     logger.error(`Error processing expiring membership ${membershipId}:`, error); |     logger.error( | ||||||
|  |       `Error processing expiring membership ${membershipId}:`, | ||||||
|  |       error | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function sendTrainerNotifications( | ||||||
|  |   membershipId: string, | ||||||
|  |   membershipData: MembershipData, | ||||||
|  |   notificationType: "expired" | "expiring" | ||||||
|  | ): Promise<void> { | ||||||
|  |   try { | ||||||
|  |     const trainerAssignments = await getTrainerAssignmentsForMembership(membershipId); | ||||||
|  |      | ||||||
|  |     if (trainerAssignments.length === 0) { | ||||||
|  |       logger.info(`No trainer assignments found for membership ${membershipId}`); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const clientName = await getClientName(membershipId, membershipData.userId); | ||||||
|  |     const gymName = await getGymName(membershipData.gymId); | ||||||
|  | 
 | ||||||
|  |     let expiryDate: Date | undefined; | ||||||
|  |     let formattedDate = "Unknown Date"; | ||||||
|  |     let daysUntilExpiry = 0; | ||||||
|  | 
 | ||||||
|  |     const payments = await getPaymentsForMembership(membershipId); | ||||||
|  |     if (payments.length > 0) { | ||||||
|  |       const latestPayment = payments[0]; | ||||||
|  |       expiryDate = calculateRenewalDateFromPayment( | ||||||
|  |         membershipData.subscription, | ||||||
|  |         latestPayment.dateTimestamp | ||||||
|  |       ); | ||||||
|  |       formattedDate = expiryDate.toLocaleDateString("en-US", { | ||||||
|  |         year: "numeric", | ||||||
|  |         month: "long", | ||||||
|  |         day: "numeric", | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|  |       if (notificationType === "expiring") { | ||||||
|  |         const now = new Date(); | ||||||
|  |         const timeDiff = expiryDate.getTime() - now.getTime(); | ||||||
|  |         daysUntilExpiry = Math.ceil(timeDiff / (1000 * 3600 * 24)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (const assignment of trainerAssignments) { | ||||||
|  |       if (!assignment.trainerId) continue; | ||||||
|  | 
 | ||||||
|  |       try { | ||||||
|  |         const trainerName = await getTrainerName(assignment.trainerId); | ||||||
|  |          | ||||||
|  |         const notifType = notificationType === "expired" ? "trainer_client_plan_expired" : "trainer_client_plan_expiring"; | ||||||
|  |         const existing = await app | ||||||
|  |           .firestore() | ||||||
|  |           .collection("notifications") | ||||||
|  |           .where("type", "==", notifType) | ||||||
|  |           .where("recipientId", "==", assignment.trainerId) | ||||||
|  |           .where("data.membershipId", "==", membershipId) | ||||||
|  |           .where( | ||||||
|  |             "data.expiryDate", | ||||||
|  |             "==", | ||||||
|  |             expiryDate | ||||||
|  |               ? admin.firestore.Timestamp.fromDate(expiryDate) | ||||||
|  |               : admin.firestore.Timestamp.fromDate(new Date()) | ||||||
|  |           ) | ||||||
|  |           .limit(1) | ||||||
|  |           .get(); | ||||||
|  | 
 | ||||||
|  |         if (!existing.empty) { | ||||||
|  |           logger.info( | ||||||
|  |             `${notificationType} notification already sent to trainer ${assignment.trainerId} for membership ${membershipId}, skipping...` | ||||||
|  |           ); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const notificationData: any = { | ||||||
|  |           senderId: "system", | ||||||
|  |           recipientId: assignment.trainerId, | ||||||
|  |           type: notifType, | ||||||
|  |           notificationSent: false, | ||||||
|  |           timestamp: admin.firestore.FieldValue.serverTimestamp(), | ||||||
|  |           read: false, | ||||||
|  |           readBy: [], | ||||||
|  |           data: { | ||||||
|  |             planName: membershipData.subscription?.name || "Unknown Plan", | ||||||
|  |             clientName, | ||||||
|  |             membershipId, | ||||||
|  |             gymName, | ||||||
|  |             assignmentId: assignment.id, | ||||||
|  |             formattedExpiryDate: formattedDate, | ||||||
|  |             expiryDate: expiryDate | ||||||
|  |               ? admin.firestore.Timestamp.fromDate(expiryDate) | ||||||
|  |               : admin.firestore.Timestamp.fromDate(new Date()), | ||||||
|  |           }, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         if (notificationType === "expiring") { | ||||||
|  |           notificationData.data.daysUntilExpiry = daysUntilExpiry; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         await app.firestore().collection("notifications").add(notificationData); | ||||||
|  | 
 | ||||||
|  |         logger.info( | ||||||
|  |           `${notificationType} notification sent to trainer ${assignment.trainerId} (${trainerName}) for client ${clientName}'s membership ${membershipId}` | ||||||
|  |         ); | ||||||
|  |       } catch (trainerError) { | ||||||
|  |         logger.error( | ||||||
|  |           `Error sending notification to trainer ${assignment.trainerId} for membership ${membershipId}:`, | ||||||
|  |           trainerError | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     logger.error( | ||||||
|  |       `Error sending trainer notifications for membership ${membershipId}:`, | ||||||
|  |       error | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -394,19 +598,6 @@ async function sendPlanExpiredNotification( | |||||||
|     const gymOwnerId = await getGymOwnerId(membershipData.gymId); |     const gymOwnerId = await getGymOwnerId(membershipData.gymId); | ||||||
|     const gymName = await getGymName(membershipData.gymId); |     const gymName = await getGymName(membershipData.gymId); | ||||||
| 
 | 
 | ||||||
|     const existing = await app |  | ||||||
|       .firestore() |  | ||||||
|       .collection("notifications") |  | ||||||
|       .where("type", "==", "plan_expired") |  | ||||||
|       .where("data.membershipId", "==", membershipId) |  | ||||||
|       .limit(1) |  | ||||||
|       .get(); |  | ||||||
| 
 |  | ||||||
|     if (!existing.empty) { |  | ||||||
|       logger.info(`Notification already sent for ${membershipId}, skipping...`); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let expiryDate: Date | undefined; |     let expiryDate: Date | undefined; | ||||||
|     let formattedDate = "Unknown Date"; |     let formattedDate = "Unknown Date"; | ||||||
| 
 | 
 | ||||||
| @ -422,7 +613,26 @@ async function sendPlanExpiredNotification( | |||||||
|         month: "long", |         month: "long", | ||||||
|         day: "numeric", |         day: "numeric", | ||||||
|       }); |       }); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|  |     const existing = await app | ||||||
|  |       .firestore() | ||||||
|  |       .collection("notifications") | ||||||
|  |       .where("type", "==", "plan_expired") | ||||||
|  |       .where("data.membershipId", "==", membershipId) | ||||||
|  |       .where( | ||||||
|  |         "data.expiryDate", | ||||||
|  |         "==", | ||||||
|  |         expiryDate | ||||||
|  |           ? admin.firestore.Timestamp.fromDate(expiryDate) | ||||||
|  |           : admin.firestore.Timestamp.fromDate(new Date()) | ||||||
|  |       ) | ||||||
|  |       .limit(1) | ||||||
|  |       .get(); | ||||||
|  | 
 | ||||||
|  |     if (!existing.empty) { | ||||||
|  |       logger.info(`Notification already sent for ${membershipId}, skipping...`); | ||||||
|  |       return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     await app |     await app | ||||||
| @ -466,19 +676,6 @@ async function sendPlanExpiringNotification( | |||||||
|     const gymOwnerId = await getGymOwnerId(membershipData.gymId); |     const gymOwnerId = await getGymOwnerId(membershipData.gymId); | ||||||
|     const gymName = await getGymName(membershipData.gymId); |     const gymName = await getGymName(membershipData.gymId); | ||||||
| 
 | 
 | ||||||
|     const existing = await app |  | ||||||
|       .firestore() |  | ||||||
|       .collection("notifications") |  | ||||||
|       .where("type", "==", "plan_expiring_soon") |  | ||||||
|       .where("data.membershipId", "==", membershipId) |  | ||||||
|       .limit(1) |  | ||||||
|       .get(); |  | ||||||
| 
 |  | ||||||
|     if (!existing.empty) { |  | ||||||
|       logger.info(`Expiring notification already sent for ${membershipId}, skipping...`); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let expiryDate: Date | undefined; |     let expiryDate: Date | undefined; | ||||||
|     let formattedDate = "Unknown Date"; |     let formattedDate = "Unknown Date"; | ||||||
|     let daysUntilExpiry = 10; |     let daysUntilExpiry = 10; | ||||||
| @ -501,6 +698,28 @@ async function sendPlanExpiringNotification( | |||||||
|       daysUntilExpiry = Math.ceil(timeDiff / (1000 * 3600 * 24)); |       daysUntilExpiry = Math.ceil(timeDiff / (1000 * 3600 * 24)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     const existing = await app | ||||||
|  |       .firestore() | ||||||
|  |       .collection("notifications") | ||||||
|  |       .where("type", "==", "plan_expiring_soon") | ||||||
|  |       .where("data.membershipId", "==", membershipId) | ||||||
|  |       .where( | ||||||
|  |         "data.expiryDate", | ||||||
|  |         "==", | ||||||
|  |         expiryDate | ||||||
|  |           ? admin.firestore.Timestamp.fromDate(expiryDate) | ||||||
|  |           : admin.firestore.Timestamp.fromDate(new Date()) | ||||||
|  |       ) | ||||||
|  |       .limit(1) | ||||||
|  |       .get(); | ||||||
|  | 
 | ||||||
|  |     if (!existing.empty) { | ||||||
|  |       logger.info( | ||||||
|  |         `Expiring notification already sent for ${membershipId}, skipping...` | ||||||
|  |       ); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     await app |     await app | ||||||
|       .firestore() |       .firestore() | ||||||
|       .collection("notifications") |       .collection("notifications") | ||||||
| @ -530,7 +749,10 @@ async function sendPlanExpiringNotification( | |||||||
|       `Expiring notification sent for membership ${membershipId} (expires on ${formattedDate}, ${daysUntilExpiry} days remaining)` |       `Expiring notification sent for membership ${membershipId} (expires on ${formattedDate}, ${daysUntilExpiry} days remaining)` | ||||||
|     ); |     ); | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     logger.error(`Error sending expiring notification for ${membershipId}:`, error); |     logger.error( | ||||||
|  |       `Error sending expiring notification for ${membershipId}:`, | ||||||
|  |       error | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -240,14 +240,28 @@ function prepareNotificationMessage( | |||||||
|       title = notification.data?.title || "Plan Expired"; |       title = notification.data?.title || "Plan Expired"; | ||||||
|       body = |       body = | ||||||
|         notification.data?.message || |         notification.data?.message || | ||||||
|         `${notification.data?.clientName}'s membership subscription for ${notification.data?.planName} has expired.`; |         `${notification.data?.clientName}'s membership  for ${notification.data?.planName} has expired.`; | ||||||
|       break; |       break; | ||||||
| 
 | 
 | ||||||
|     case "plan_expiring_soon": |     case "plan_expiring_soon": | ||||||
|       title = notification.data?.title || "Plan Expiring Soon"; |       title = notification.data?.title || "Plan Expiring Soon"; | ||||||
|       body = |       body = | ||||||
|         notification.data?.message || |         notification.data?.message || | ||||||
|         `${notification.data?.clientName}'s membership subscription for ${notification.data?.planName} will expire on ${notification.data?.formattedExpiryDate}.`; |         `${notification.data?.clientName}'s membership  for ${notification.data?.planName} will expire on ${notification.data?.formattedExpiryDate}.`; | ||||||
|  |       break; | ||||||
|  | 
 | ||||||
|  |     case "trainer_client_plan_expired": | ||||||
|  |       title = notification.data?.title || "Client Plan Expired"; | ||||||
|  |       body = | ||||||
|  |         notification.data?.message || | ||||||
|  |         `${notification.data?.clientName}'s membership for ${notification.data?.planName} has expired.`; | ||||||
|  |       break; | ||||||
|  | 
 | ||||||
|  |     case "trainer_client_plan_expiring": | ||||||
|  |       title = notification.data?.title || "Client Plan Expiring Soon"; | ||||||
|  |       body = | ||||||
|  |         notification.data?.message || | ||||||
|  |         `${notification.data?.clientName}'s membership for ${notification.data?.planName} will expire on ${notification.data?.formattedExpiryDate}.`; | ||||||
|       break; |       break; | ||||||
| 
 | 
 | ||||||
|     case "schedule_update": |     case "schedule_update": | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user