Compare commits

...

81 Commits

Author SHA1 Message Date
61f0d29f37 Merge branch 'qa'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 3m30s
2025-09-11 14:32:04 +05:30
209c7c65b0 Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 3m37s
2025-09-11 14:25:39 +05:30
ad31bc8e80 notification-bug-fix (#114)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m30s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #114
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-09-11 08:47:46 +00:00
f0d167a671 notification-bug-fix (#113)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m32s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #113
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-09-09 09:06:58 +00:00
e762d0fb79 notification-bug-fix (#112)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m47s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #112
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-26 06:27:20 +00:00
5ca05c6490 Merge branch 'dev'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 4m1s
2025-08-25 18:39:53 +05:30
7ec65e4ab1 Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 4m1s
2025-08-25 18:35:11 +05:30
b41431b151 notification-bug-fix (#111)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 1m48s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #111
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-25 09:57:03 +00:00
218ff1d02e notification-bug-fix (#110)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m58s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #110
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-25 09:50:08 +00:00
4f71ca273b Merge branch 'qa'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 1m40s
2025-08-22 17:39:15 +05:30
4d52d1c7f8 Merged from dev
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 1m40s
2025-08-22 17:36:45 +05:30
aee40521d3 Merge branch 'dev' into qa 2025-08-22 17:35:21 +05:30
a450e93e2b notification-bug-fix (#109)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 1m39s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #109
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-22 12:00:31 +00:00
36015d2b83 Removed npm install at root level
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 3m42s
2025-08-22 17:18:57 +05:30
51fa0825ca Removed npm install from root
Some checks failed
Deploy FitLien services to QA / Deploy to QA (push) Successful in 3m36s
Deploy FitLien services / Deploy (push) Failing after 8s
2025-08-22 17:04:54 +05:30
3e455fc83a Merge branch 'dev' into qa
Some checks failed
Deploy FitLien services to QA / Deploy to QA (push) Failing after 8s
2025-08-22 16:58:23 +05:30
0672a19a60 notification-bug-fix (#108)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 2m14s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #108
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-22 11:13:19 +00:00
172fa2edae notification-bug-fix (#107)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 1m53s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #107
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-20 13:38:54 +00:00
d0c00d8172 notification-bug-fix (#106)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m42s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #106
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-20 13:30:53 +00:00
f93931867c notification-bug-fix (#105)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m52s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #105
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-20 13:22:29 +00:00
70d76bab2e notification-bug-fix (#104)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 1m32s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #104
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 11:51:37 +00:00
308cb0fab6 notification-bug-fix (#103)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m42s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #103
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 11:43:38 +00:00
165cd74a17 notification-bug-fix (#102)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m29s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #102
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 11:18:53 +00:00
b190a371b6 notification-bug-fix (#101)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 1m34s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #101
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 10:21:00 +00:00
5543ba5e7a notification-bug-fix (#100)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m40s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #100
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 09:52:14 +00:00
5d6824a6f4 notification-bug-fix (#99)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 2m0s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #99
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 09:43:44 +00:00
abf7a04633 notification-bug-fix (#98)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m25s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #98
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 07:21:41 +00:00
835f478665 notification-bug-fix (#97)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m28s
Co-authored-by: Dhansh A S <dhanshas@cosq.net>
Reviewed-on: #97
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 07:11:25 +00:00
5f43a86036 notification-bug-fix (#96)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m24s
Reviewed-on: #96
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 07:06:26 +00:00
1e09f7a676 notification-bug-fix (#95)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m34s
Reviewed-on: #95
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 06:54:45 +00:00
5e48f695f8 notification-bug-fix (#94)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m31s
Reviewed-on: #94
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 06:51:02 +00:00
e483b7ad46 notification-bug-fix (#93)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m34s
Reviewed-on: #93
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 06:42:11 +00:00
cd59b9890d notification-bug-fix (#92)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m5s
Reviewed-on: #92
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 06:23:43 +00:00
939567f7c0 notification-bug-fix (#91)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m42s
Reviewed-on: #91
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 05:39:54 +00:00
7c494154ba notification-bug-fix (#90)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m38s
Reviewed-on: #90
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 04:59:34 +00:00
8f5956a825 notification-bug-fix (#89)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m30s
Reviewed-on: #89
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 04:49:31 +00:00
5ff7b8bb84 notification-bug-fix (#88)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m44s
Reviewed-on: #88
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-19 04:30:40 +00:00
7259e67833 REMOVED clean install from project root
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m24s
2025-08-18 19:20:36 +05:30
81c5241e95 Merge branch 'dev' of cosqnet.com:cosqnet/fitlien-services into dev
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 8s
2025-08-18 19:16:41 +05:30
209354ec6b UPDATED firebase-functions package 2025-08-18 19:16:18 +05:30
39ea6dcafb notification-bug-fix (#87)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m55s
Reviewed-on: #87
Reviewed-by: Allen T J <allentj@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-18 13:42:26 +00:00
f6b1545cf6 Merge branch 'dev' into qa
Some checks failed
Deploy FitLien services to QA / Deploy to QA (push) Failing after 1m33s
2025-08-18 19:05:38 +05:30
195262a6de Merge branch 'qa'
Some checks failed
Deploy FitLien services / Deploy (push) Failing after 2m8s
Deploy FitLien services to QA / Deploy to QA (push) Failing after 1m42s
2025-08-18 18:53:46 +05:30
d8ae223cce notification-bug-fix (#86)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m37s
Reviewed-on: #86
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-14 08:35:40 +00:00
c2914b16bb notification-bug-fix (#85)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m38s
Reviewed-on: #85
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-14 06:41:07 +00:00
b66f1603cc notification-bug-fix (#84)
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>
2025-08-14 05:30:52 +00:00
d3c9e86c7c Merge branch 'dev'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 4m10s
2025-08-08 18:53:15 +05:30
4cf5692386 plan-expiry-in-two-days (#83)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m20s
Reviewed-on: #83
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-08 07:55:52 +00:00
8b308fb9a6 Merge branch 'dev'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 5m14s
2025-08-07 20:17:17 +05:30
237dd8a263 Changes Updated (#82)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m28s
Reviewed-on: #82
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-05 14:03:40 +00:00
fb23661080 expiry-using-payment (#81)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m54s
Reviewed-on: #81
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-05 13:20:00 +00:00
cba945c282 expiry-using-payment (#80)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m5s
Reviewed-on: #80
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-05 09:22:10 +00:00
ef166a209c expiry-using-payment (#79)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 3m54s
Reviewed-on: #79
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-05 06:16:47 +00:00
5d47a78baa expiry-notification (#78)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m4s
Reviewed-on: #78
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-04 07:30:25 +00:00
b594579158 expiry-notification (#77)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m8s
Reviewed-on: #77
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-04 05:47:40 +00:00
943cff74d5 expiry-notification (#76)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m40s
Reviewed-on: #76
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-04 04:43:03 +00:00
1fc089a7cb expiry-notification (#75)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m42s
Reviewed-on: #75
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-01 16:33:09 +00:00
76a75330c8 Changes Updated (#74)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 1m32s
Reviewed-on: #74
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-08-01 13:20:36 +00:00
ca08d83f98 notification-issue (#73)
Some checks failed
Deploy FitLien services to Dev / Deploy to Dev (push) Failing after 2m3s
PLan Expiry Notification added

Reviewed-on: #73
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-07-29 13:50:50 +00:00
0a32e15d05 notification-issue (#72)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m29s
Reviewed-on: #72
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-07-29 07:46:19 +00:00
3223efc392 notification-issue (#71)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m23s
Reviewed-on: #71
Reviewed-by: Dhansh A S <dhanshas@cosq.net>
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-07-28 13:01:53 +00:00
9c2431fb7b Changes Updated (#70)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m48s
Reviewed-on: #70
Co-authored-by: Sharon Dcruz <sharondcruz@cosq.net>
Co-committed-by: Sharon Dcruz <sharondcruz@cosq.net>
2025-07-25 07:57:10 +00:00
7492cdedc1 Accesslog done (#69)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m11s
Reviewed-on: #69
Co-authored-by: DhanshCOSQ <dhanshas@cosq.net>
Co-committed-by: DhanshCOSQ <dhanshas@cosq.net>
2025-07-16 05:31:33 +00:00
d8bf928da8 feature/fitlien-updated-index (#68)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m4s
Reviewed-on: #68
Co-authored-by: DhanshCOSQ <dhanshas@cosq.net>
Co-committed-by: DhanshCOSQ <dhanshas@cosq.net>
2025-07-03 08:45:25 +00:00
9d51393aa5 Merge branch 'dev'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 3m39s
2025-07-02 21:28:42 +05:30
cc5b6d6987 Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 3m48s
2025-07-02 21:28:06 +05:30
cbe59ee4f1 updated index (#67)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 2m3s
Reviewed-on: #67
Co-authored-by: DhanshCOSQ <dhanshas@cosq.net>
Co-committed-by: DhanshCOSQ <dhanshas@cosq.net>
2025-07-02 15:56:59 +00:00
a3241afd45 phonepe (#66)
All checks were successful
Deploy FitLien services to Dev / Deploy to Dev (push) Successful in 4m0s
Co-authored-by: AllenTJ7 <163137620+AllenTJ7@users.noreply.github.com>
Reviewed-on: #66
2025-06-25 14:28:07 +00:00
ef2cd80226 Merge branch 'qa'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 4m2s
2025-06-24 15:19:54 +05:30
f19e3b012d Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 3m53s
2025-06-24 15:19:20 +05:30
f2e37e88ed Merge branch 'dev'
All checks were successful
Deploy FitLien services / Deploy (push) Successful in 4m24s
2025-06-12 19:47:27 +05:30
a0134466ee Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 3m42s
2025-06-12 19:45:40 +05:30
ecbe9d184b Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to QA / Deploy to QA (push) Successful in 3m47s
Deploy FitLien services / Deploy (push) Successful in 4m30s
2025-05-30 12:01:13 +05:30
7a796243b0 Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to Dev / Deploy to QA (push) Successful in 4m13s
2025-05-28 10:55:32 +05:30
7db9e479ad Merge branch 'dev' into qa
All checks were successful
Deploy FitLien services to Dev / Deploy to QA (push) Successful in 4m13s
2025-05-26 23:38:30 +05:30
cf6f4625ad Removed env var FITLIENHOST 2025-04-21 12:28:07 +05:30
6434f6e3fa Merge branch 'dev' into qa 2025-04-21 12:17:34 +05:30
2147963523 Merge branch 'dev' into qa 2025-04-16 18:12:16 +05:30
18569d38d3 Merge branch 'dev' into qa 2025-04-16 04:35:58 +05:30
5bc3d6dfff Merge branch 'dev' into qa 2025-04-16 03:53:43 +05:30
e8ca80df48 Merge branch 'dev' into qa 2025-04-14 15:02:28 +05:30
20 changed files with 1771 additions and 733 deletions

View File

@ -19,9 +19,6 @@ jobs:
with: with:
node-version: 22 node-version: 22
- name: Clean install
run: npm clean-install
- name: Copy .env.example to .env - name: Copy .env.example to .env
run: cp functions/.env.example functions/.env run: cp functions/.env.example functions/.env

View File

@ -19,9 +19,6 @@ jobs:
with: with:
node-version: 22 node-version: 22
- name: Clean install
run: npm clean-install
- name: Copy .env.example to .env - name: Copy .env.example to .env
run: cp functions/.env.example functions/.env run: cp functions/.env.example functions/.env

View File

@ -19,9 +19,6 @@ jobs:
with: with:
node-version: 22 node-version: 22
- name: Clean install
run: npm clean-install
- name: Copy .env.example to .env - name: Copy .env.example to .env
run: cp functions/.env.example functions/.env run: cp functions/.env.example functions/.env

View File

@ -57,11 +57,11 @@
] ]
}, },
{ {
"collectionGroup": "memberships", "collectionGroup": "gyms",
"queryScope": "COLLECTION", "queryScope": "COLLECTION",
"fields": [ "fields": [
{ {
"fieldPath": "gymId", "fieldPath": "userId",
"order": "ASCENDING" "order": "ASCENDING"
}, },
{ {
@ -71,79 +71,29 @@
] ]
}, },
{ {
"collectionGroup": "notifications", "collectionGroup": "gyms",
"queryScope": "COLLECTION", "queryScope": "COLLECTION",
"fields": [ "fields": [
{ {
"fieldPath": "clientId", "fieldPath": "isApproved",
"order": "ASCENDING" "order": "ASCENDING"
}, },
{ {
"fieldPath": "timestamp", "fieldPath": "createdAt",
"order": "DESCENDING" "order": "ASCENDING"
} }
] ]
}, },
{ {
"collectionGroup": "notifications", "collectionGroup": "memberships",
"queryScope": "COLLECTION", "queryScope": "COLLECTION",
"fields": [ "fields": [
{ {
"fieldPath": "clientId", "fieldPath": "gymId",
"order": "ASCENDING" "order": "ASCENDING"
}, },
{ {
"fieldPath": "type", "fieldPath": "createdAt",
"order": "ASCENDING"
},
{
"fieldPath": "timestamp",
"order": "DESCENDING"
}
]
},
{
"collectionGroup": "notifications",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "ownerId",
"order": "ASCENDING"
},
{
"fieldPath": "timestamp",
"order": "DESCENDING"
}
]
},
{
"collectionGroup": "notifications",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "ownerId",
"order": "ASCENDING"
},
{
"fieldPath": "type",
"order": "ASCENDING"
},
{
"fieldPath": "timestamp",
"order": "DESCENDING"
}
]
},
{
"collectionGroup": "notifications",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "trainerId",
"order": "ASCENDING"
},
{
"fieldPath": "timestamp",
"order": "DESCENDING" "order": "DESCENDING"
} }
] ]
@ -166,6 +116,66 @@
} }
] ]
}, },
{
"collectionGroup": "notifications",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "recipientId",
"order": "ASCENDING"
},
{
"fieldPath": "timestamp",
"order": "DESCENDING"
}
]
},
{
"collectionGroup": "workout_logs",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "user_id",
"order": "ASCENDING"
},
{
"fieldPath": "date",
"order": "DESCENDING"
}
]
},
{
"collectionGroup": "workout_logs",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "user_id",
"order": "ASCENDING"
},
{
"fieldPath": "date",
"order": "ASCENDING"
}
]
},
{
"collectionGroup": "workout_logs",
"queryScope": "COLLECTION",
"fields": [
{
"fieldPath": "user_id",
"order": "ASCENDING"
},
{
"fieldPath": "start_time",
"order": "ASCENDING"
},
{
"fieldPath": "date",
"order": "ASCENDING"
}
]
},
{ {
"collectionGroup": "terms_and_conditions", "collectionGroup": "terms_and_conditions",
"queryScope": "COLLECTION", "queryScope": "COLLECTION",

View File

@ -15,7 +15,7 @@
"cors": "^2.8.5", "cors": "^2.8.5",
"date-fns": "^4.1.0", "date-fns": "^4.1.0",
"firebase-admin": "^12.6.0", "firebase-admin": "^12.6.0",
"firebase-functions": "^6.0.1", "firebase-functions": "^6.4.0",
"form-data": "^4.0.1", "form-data": "^4.0.1",
"functions": "file:", "functions": "file:",
"html-to-text": "^9.0.5", "html-to-text": "^9.0.5",
@ -4897,9 +4897,10 @@
} }
}, },
"node_modules/firebase-functions": { "node_modules/firebase-functions": {
"version": "6.3.2", "version": "6.4.0",
"resolved": "https://registry.npmjs.org/firebase-functions/-/firebase-functions-6.3.2.tgz", "resolved": "https://registry.npmjs.org/firebase-functions/-/firebase-functions-6.4.0.tgz",
"integrity": "sha512-FC3A1/nhqt1ZzxRnj5HZLScQaozAcFSD/vSR8khqSoFNOfxuXgwJS6ZABTB7+v+iMD5z6Mmxw6OfqITUBuI7OQ==", "integrity": "sha512-Q/LGhJrmJEhT0dbV60J4hCkVSeOM6/r7xJS/ccmkXzTWMjo+UPAYX9zlQmGlEjotstZ0U9GtQSJSgbB2Z+TJDg==",
"license": "MIT",
"dependencies": { "dependencies": {
"@types/cors": "^2.8.5", "@types/cors": "^2.8.5",
"@types/express": "^4.17.21", "@types/express": "^4.17.21",

View File

@ -22,7 +22,7 @@
"cors": "^2.8.5", "cors": "^2.8.5",
"date-fns": "^4.1.0", "date-fns": "^4.1.0",
"firebase-admin": "^12.6.0", "firebase-admin": "^12.6.0",
"firebase-functions": "^6.0.1", "firebase-functions": "^6.4.0",
"form-data": "^4.0.1", "form-data": "^4.0.1",
"functions": "file:", "functions": "file:",
"html-to-text": "^9.0.5", "html-to-text": "^9.0.5",

View File

@ -207,10 +207,10 @@ function createGetEmployeePunchLogsRequest(username: string, password: string,
<soap12:Envelope xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:soap12="http://www.w3.org/2003/05/soap-envelope"> <soap12:Envelope xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:soap12="http://www.w3.org/2003/05/soap-envelope">
<soap12:Body> <soap12:Body>
<GetEmployeePunchLogs xmlns="http://tempuri.org/"> <GetEmployeePunchLogs xmlns="http://tempuri.org/">
<UserName>cosqclient</UserName> <UserName>${escapeXml(username)}</UserName>
<Password>3bbb58d5</Password> <Password>${escapeXml(password)}</Password>
<EmployeeCode>1</EmployeeCode> <EmployeeCode>${escapeXml(employeeCode)}</EmployeeCode>
<AttendanceDate>2025-05-24</AttendanceDate> <AttendanceDate>${escapeXml(attendanceDate)}</AttendanceDate>
</GetEmployeePunchLogs> </GetEmployeePunchLogs>
</soap12:Body> </soap12:Body>
</soap12:Envelope>`; </soap12:Envelope>`;
@ -232,9 +232,15 @@ function parseGetEmployeePunchLogsResponse(soapResponse: string, attendanceDate:
const xmlDoc = parser.parseFromString(soapResponse, "text/xml"); const xmlDoc = parser.parseFromString(soapResponse, "text/xml");
const currentElement = xmlDoc.documentElement.firstChild as HTMLElement; const currentElement = xmlDoc.documentElement.firstChild as HTMLElement;
const resultText = currentElement.textContent; const resultText = currentElement.textContent;
if (!resultText || resultText.trim() === '' || resultText.trim() === ';;' || resultText.trim() === ';') {
return [];
}
const punchLogs: Date[] = []; const punchLogs: Date[] = [];
const parts = resultText!.split(';'); const parts = resultText.split(';');
for (const part of parts) { for (const part of parts) {
if (!part || part.trim() === '') {
continue;
}
try { try {
const logDateTime = new Date(part); const logDateTime = new Date(part);
if (isNaN(logDateTime.getTime())) { if (isNaN(logDateTime.getTime())) {
@ -245,8 +251,12 @@ function parseGetEmployeePunchLogsResponse(soapResponse: string, attendanceDate:
try { try {
const timeParts = part.split(','); const timeParts = part.split(',');
for (const timePart of timeParts) { for (const timePart of timeParts) {
if (!timePart || timePart.trim() === '') {
continue;
}
try { try {
const logDateTime = createDateFromTime(rootDate, timePart); const logDateTime = createDateFromTime(rootDate, timePart.trim());
punchLogs.push(logDateTime); punchLogs.push(logDateTime);
} catch { } catch {
continue; continue;
@ -261,6 +271,7 @@ function parseGetEmployeePunchLogsResponse(soapResponse: string, attendanceDate:
return sortedLogs; return sortedLogs;
} }
async function sendSoapRequest(soapRequest: string, endpoint: string) { async function sendSoapRequest(soapRequest: string, endpoint: string) {
try { try {
const headers: any = { const headers: any = {

View File

@ -26,7 +26,7 @@ interface EmailRequest {
interface Attachment { interface Attachment {
filename: string; filename: string;
content: string | Buffer; // Base64 encoded string or Buffer content: string | Buffer;
contentType?: string; contentType?: string;
} }
@ -37,7 +37,7 @@ const stripHtml = (html: string): string => {
async function sendSimpleEmail(data: EmailRequest, recipients: string[]) { async function sendSimpleEmail(data: EmailRequest, recipients: string[]) {
const ses = new SESClient({ const ses = new SESClient({
region: '#{AWS_REGION}#', region: process.env.AWS_REGION,
credentials: { credentials: {
accessKeyId: process.env.AWS_ACCESS_KEY_ID || '', accessKeyId: process.env.AWS_ACCESS_KEY_ID || '',
secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || '' secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || ''
@ -63,7 +63,7 @@ async function sendSimpleEmail(data: EmailRequest, recipients: string[]) {
async function sendEmailWithAttachments(data: EmailRequest, recipients: string[]) { async function sendEmailWithAttachments(data: EmailRequest, recipients: string[]) {
const ses = new SESClient({ const ses = new SESClient({
region: 'ap-south-1', region: process.env.AWS_REGION,
credentials: { credentials: {
accessKeyId: process.env.AWS_ACCESS_KEY_ID || '', accessKeyId: process.env.AWS_ACCESS_KEY_ID || '',
secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || '' secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || ''

View File

@ -13,7 +13,7 @@ export * from './shared/config';
export { sendEmailSES } from './email'; export { sendEmailSES } from './email';
export { sendSMSMessage } from './sms'; export { sendSMSMessage } from './sms';
export { accessFile } from './storage'; export { accessFile } from './storage';
export { processNotificationOnCreate } from './notifications'; export { processNotificationOnCreate,checkExpiredMemberships } from './notifications';
export * from './payments'; export * from './payments';
export { getPlaceDetails, getPlacesAutocomplete } from './places'; export { getPlaceDetails, getPlacesAutocomplete } from './places';
export { registerClient } from './users'; export { registerClient } from './users';

View File

@ -1 +1,3 @@
export { processNotificationOnCreate } from './processNotification'; export { processNotificationOnCreate } from './processNotification';
export { checkExpiredMemberships } from "./membershipStatusNotifications";

File diff suppressed because it is too large Load Diff

View File

@ -1,34 +1,27 @@
import { onDocumentCreated } from "firebase-functions/v2/firestore"; import { onDocumentCreated } from "firebase-functions/v2/firestore";
import { getLogger } from "../shared/config"; import { getLogger } from "../shared/config";
import { getAdmin } from "../shared/config"; import { getAdmin } from "../shared/config";
import * as admin from 'firebase-admin'; import * as admin from "firebase-admin";
const app = getAdmin(); const app = getAdmin();
const logger = getLogger(); const logger = getLogger();
interface NotificationData { interface NotificationData {
notificationSent?: boolean; senderId?: string;
userId?: string; recipientId?: string;
clientId?: string;
invitorId?: string;
phoneNumber?: string;
message?: string;
type?: string; type?: string;
status?: string; notificationSent?: boolean;
gymName?: string; timestamp?: admin.firestore.FieldValue;
trainerName?: string; read?: boolean;
membershipId?: string; data?: { [key: string]: any };
subscriptionName?: string;
name?: string;
clientEmail?: string;
invitationId?: string;
[key: string]: any;
} }
export const processNotificationOnCreate = onDocumentCreated({ export const processNotificationOnCreate = onDocumentCreated(
region: '#{SERVICES_RGN}#', {
document: 'notifications/{notificationId}' region: "#{SERVICES_RGN}#",
}, async (event) => { document: "notifications/{notificationId}",
},
async (event) => {
try { try {
const notificationSnapshot = event.data; const notificationSnapshot = event.data;
const notificationId = event.params.notificationId; const notificationId = event.params.notificationId;
@ -39,184 +32,391 @@ export const processNotificationOnCreate = onDocumentCreated({
} }
const notification = notificationSnapshot.data() as NotificationData; const notification = notificationSnapshot.data() as NotificationData;
if (notification.notificationSent === true) { if (notification.notificationSent === true) {
logger.info(`Notification ${notificationId} already sent, skipping.`); logger.info(`Notification ${notificationId} already sent, skipping.`);
return; return;
} }
const { fcmToken } = await getUserAndFCMToken(notification); logger.info(
`Processing notification ${notificationId} of type: ${notification.type}`
);
const { userId, fcmToken } = await getUserAndFCMToken(notification);
if (!fcmToken) { if (!fcmToken) {
logger.error(`FCM token not found for notification ${notificationId}`); logger.error(
await updateNotificationWithError(notificationId, 'FCM token not found for user'); `FCM token not found for notification ${notificationId}, user: ${userId}`
);
await updateNotificationWithError(
notificationId,
"FCM token not found for user"
);
return; return;
} }
const message = prepareNotificationMessage(notification, fcmToken); const message = prepareNotificationMessage(notification, fcmToken);
try { try {
const fcmResponse = await app.messaging().send({ const fcmResponse = await app.messaging().send(message);
...message,
token: fcmToken
});
logger.info(`FCM notification sent successfully: ${fcmResponse}`); logger.info(`FCM notification sent successfully: ${fcmResponse}`);
await markNotificationAsSent(notificationId); await markNotificationAsSent(notificationId);
} catch (error) { } catch (error) {
logger.error(`Error sending notification ${notificationId}:`, error); logger.error(`Error sending notification ${notificationId}:`, error);
await updateNotificationWithError(notificationId, error instanceof Error ? error.message : String(error)); await updateNotificationWithError(
notificationId,
error instanceof Error ? error.message : String(error)
);
} }
} catch (error) { } catch (error) {
logger.error('Error processing notification:', error); logger.error("Error processing notification:", error);
} }
}); }
);
async function getUserAndFCMToken(notification: NotificationData): Promise<{ userId: string | null; fcmToken: string | null }> { async function getUserAndFCMToken(
let userId: string | null = null; notification: NotificationData
): Promise<{ userId: string | null; fcmToken: string | null }> {
let targetUserId: string | null = null;
let fcmToken: string | null = null; let fcmToken: string | null = null;
if (notification.userId) { if (notification.recipientId) {
userId = notification.userId; targetUserId = notification.recipientId;
fcmToken = await getFCMTokenFromUserDoc(userId); logger.info(`Using top-level recipientId: ${targetUserId}`);
} else if (notification.clientId) { } else if (notification.data?.phoneNumber) {
userId = notification.clientId; logger.info(
fcmToken = await getFCMTokenFromUserDoc(userId); `Looking up user by phone number from data: ${notification.data.phoneNumber}`
} else if (notification.invitorId) { );
userId = notification.invitorId;
fcmToken = await getFCMTokenFromUserDoc(userId);
} else if (notification.phoneNumber) {
const userQuery = await app const userQuery = await app
.firestore() .firestore()
.collection('users') .collection("users")
.where('phoneNumber', '==', notification.phoneNumber) .where("phoneNumber", "==", notification.data.phoneNumber)
.limit(1) .limit(1)
.get(); .get();
if (!userQuery.empty) { if (!userQuery.empty) {
const userDoc = userQuery.docs[0]; const userDoc = userQuery.docs[0];
userId = userDoc.id; targetUserId = userDoc.id;
fcmToken = userDoc.data()?.fcmToken; fcmToken = userDoc.data()?.fcmToken;
logger.info(`Found user by phone: ${targetUserId}`);
} else {
logger.warn(
`No user found with phone number from data: ${notification.data.phoneNumber}`
);
} }
} else {
logger.error("No valid user identifier found in notification or its data");
} }
return { userId, fcmToken }; if (targetUserId && !fcmToken) {
fcmToken = await getFCMTokenFromUserDoc(targetUserId);
}
if (targetUserId && !fcmToken) {
logger.warn(`User ${targetUserId} found but no FCM token available`);
}
return { userId: targetUserId, fcmToken };
} }
async function getFCMTokenFromUserDoc(userId: string): Promise<string | null> { async function getFCMTokenFromUserDoc(userId: string): Promise<string | null> {
const userDoc = await app.firestore().collection('users').doc(userId).get(); try {
return userDoc.exists ? userDoc.data()?.fcmToken : null; const userDoc = await app.firestore().collection("users").doc(userId).get();
if (userDoc.exists) {
const userData = userDoc.data();
const fcmToken = userData?.fcmToken;
if (!fcmToken) {
logger.warn(`User ${userId} exists but has no FCM token`);
}
return fcmToken;
} else {
logger.warn(`User document not found: ${userId}`);
return null;
}
} catch (error) {
logger.error(`Error fetching user ${userId}:`, error);
return null;
}
} }
function prepareNotificationMessage(notification: NotificationData, fcmToken: string): admin.messaging.Message { function prepareNotificationMessage(
let title = 'New Notification'; notification: NotificationData,
let body = notification.message || 'You have a new notification'; fcmToken: string
let data: Record<string, string> = { ): admin.messaging.TokenMessage {
type: notification.type || 'general', let title = notification.data?.title || "New Notification";
let body = notification.data?.message || "You have a new notification";
let fcmData: Record<string, string> = {
type: notification.type || "general",
notificationId: "notification_" + Date.now().toString(),
}; };
switch (notification.type) { if (notification.senderId) fcmData.senderId = notification.senderId;
case 'day_pass_entry': if (notification.recipientId) fcmData.recipientId = notification.recipientId;
const isAccepted = notification.status === 'ACCEPTED'; if (notification.read !== undefined) fcmData.read = String(notification.read);
title = isAccepted ? 'Day Pass Approved' : 'Day Pass Denied';
body = notification.message || (isAccepted ?
'Your day pass has been approved' :
'Your day pass has been denied');
data.gymName = notification.gymName || '';
break;
case 'trainer_assigned_to_client': if (notification.data) {
title = 'Trainer Assigned'; for (const key in notification.data) {
body = notification.message || `${notification.trainerName} has been assigned as your trainer`; if (Object.prototype.hasOwnProperty.call(notification.data, key)) {
data.trainerName = notification.trainerName || ''; const value = notification.data[key];
data.membershipId = notification.membershipId || ''; if (typeof value === "object" && value !== null) {
break; fcmData[key] = JSON.stringify(value);
} else {
case 'client_invitations': fcmData[key] = String(value);
if (notification.userId || notification.invitorId) { }
const isAccept = notification.status === 'ACCEPTED'; }
title = isAccept ? 'Invitation Accepted' : 'Invitation Rejected'; }
body = notification.message || (isAccept ? }
`The invitation for ${notification.subscriptionName} you shared with ${notification.name} has been accepted` :
`The invitation for ${notification.subscriptionName} you shared with ${notification.name} has been rejected`); switch (notification.type) {
} else if (notification.phoneNumber) { case "trainer_response":
const invitationStatus = getInvitationStatus(notification.status); title =
title = getInvitationTitle(invitationStatus); notification.data?.title ||
body = notification.message || getInvitationBody(invitationStatus, notification.name); (notification.data?.status === "accepted"
data.status = invitationStatus; ? "Trainer Invitation Accepted"
: "Trainer Invitation Update");
body =
notification.data?.message ||
`${
notification.data?.trainerName
} has ${notification.data?.status?.toLowerCase()} your request`;
break;
case "trainer_assignment":
title = notification.data?.title || "New Client Assignment";
body =
notification.data?.message ||
`You have been assigned to train ${notification.data?.name}.`;
break;
case "trainer_assigned_to_client":
title = notification.data?.title || "Trainer Assigned";
body =
notification.data?.message ||
`${notification.data?.trainerName} has been assigned as your trainer.`;
break;
case "trainer_update_owner":
title = notification.data?.title || "Trainer Schedule Update";
body =
notification.data?.message || "A trainer has updated their schedule";
break;
case "trainer_update_client":
title = notification.data?.title || "Schedule Update";
body =
notification.data?.message || "Your training schedule has been updated";
if (notification.data?.oldTimeSlot && notification.data?.newTimeSlot) {
body += ` from ${notification.data.oldTimeSlot} to ${notification.data.newTimeSlot}`;
if (notification.data?.formattedDate) {
body += ` on ${notification.data.formattedDate}`;
}
}
break;
case "plan_renewal":
title = notification.data?.title || "Plan Renewal";
body =
notification.data?.message ||
`Plan ${notification.data?.subscriptionName} has been renewed`;
break;
case "plan_assigned":
title = notification.data?.title || "New Plan Assigned";
body =
notification.data?.message ||
`You have been assigned ${notification.data?.subscriptionName} at ${notification.data?.gymName}`;
break;
case "plan_expired":
title = notification.data?.title || "Plan Expired";
body =
notification.data?.message ||
`${notification.data?.clientName}'s membership for ${notification.data?.planName} has expired.`;
break;
case "plan_expiring_soon":
title = notification.data?.title || "Plan Expiring Soon";
body =
notification.data?.message ||
`${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;
case "schedule_update":
title = notification.data?.title || "Schedule Update";
body =
notification.data?.message || "Your training schedule has been updated";
if (notification.data?.oldTimeSlot && notification.data?.newTimeSlot) {
body += ` from ${notification.data.oldTimeSlot} to ${notification.data.newTimeSlot}`;
if (notification.data?.formattedDate) {
body += ` on ${notification.data.formattedDate}`;
}
}
break;
case "attendance_dispute":
title = notification.data?.title || "Attendance Dispute";
body =
notification.data?.message ||
`${notification.data?.name} has disputed an attendance record`;
if (notification.data?.logTime) {
body += ` for ${notification.data.logTime}`;
}
break;
case "day_pass_entry":
const isAccepted = notification.data?.status === "ACCEPTED";
title =
notification.data?.title ||
(isAccepted ? "Day Pass Approved" : "Day Pass Denied");
body =
notification.data?.message ||
(isAccepted
? "Your day pass has been approved"
: "Your day pass has been denied");
break;
case "client_invitations":
if (notification.data?.userId || notification.data?.invitorId) {
const isAccept = notification.data?.status === "ACCEPTED";
title =
notification.data?.title ||
(isAccept ? "Invitation Accepted" : "Invitation Rejected");
body =
notification.data?.message ||
(isAccept
? `The invitation for ${notification.data?.subscriptionName} you shared with ${notification.data?.name} has been accepted`
: `The invitation for ${notification.data?.subscriptionName} you shared with ${notification.data?.name} has been rejected`);
} else if (notification.data?.phoneNumber) {
const invitationStatus = getInvitationStatus(notification.data?.status);
title =
notification.data?.title || getInvitationTitle(invitationStatus);
body =
notification.data?.message ||
getInvitationBody(invitationStatus, notification.data?.name);
fcmData.status = invitationStatus;
} }
data.gymName = notification.gymName || '';
data.clientEmail = notification.clientEmail || '';
data.clientName = notification.name || '';
data.invitationId = notification.invitationId || '';
data.subscriptionName = notification.subscriptionName || '';
break; break;
default: default:
logger.info(`Using default handling for notification type: ${notification.type}`); logger.info(
`Using default handling for notification type: ${notification.type}`
);
title =
notification.data?.title ||
(notification.type
? `${notification.type.replace("_", " ").toUpperCase()}`
: "Notification");
break; break;
} }
const notificationMessage: admin.messaging.Message = { const notificationMessage: admin.messaging.TokenMessage = {
notification: { title, body }, notification: { title, body },
data, data: fcmData,
android: { android: {
priority: 'high', priority: "high",
notification: { notification: {
channelId: 'notifications_channel', channelId: "notifications_channel",
priority: 'high', priority: "high",
defaultSound: true, defaultSound: true,
defaultVibrateTimings: true, defaultVibrateTimings: true,
icon: '@mipmap/ic_launcher', icon: "@mipmap/ic_launcher",
clickAction: 'FLUTTER_NOTIFICATION_CLICK', clickAction: "FLUTTER_NOTIFICATION_CLICK",
}, },
}, },
apns: { apns: {
payload: { payload: {
aps: { aps: {
sound: 'default', sound: "default",
badge: 1, badge: 1,
}, },
}, },
}, },
token: fcmToken, token: fcmToken,
}; };
logger.info(`Prepared notification: ${title} - ${body}`);
return notificationMessage; return notificationMessage;
} }
function getInvitationStatus(status?: string): string { function getInvitationStatus(status?: string): string {
if (status === 'ACCEPTED') return 'accepted'; if (status === "ACCEPTED") return "accepted";
if (status === 'REJECTED') return 'rejected'; if (status === "REJECTED") return "rejected";
if (status === 'PENDING') return 'pending'; if (status === "PENDING") return "pending";
return 'unknown'; return "unknown";
} }
function getInvitationTitle(status: string): string { function getInvitationTitle(status: string): string {
switch (status) { switch (status) {
case 'accepted': return 'Invitation Accepted'; case "accepted":
case 'rejected': return 'Invitation Rejected'; return "Invitation Accepted";
case 'pending': return 'New Invitation'; case "rejected":
default: return 'Invitation Update'; return "Invitation Rejected";
case "pending":
return "New Invitation";
default:
return "Invitation Update";
} }
} }
function getInvitationBody(status: string, name?: string): string { function getInvitationBody(status: string, name?: string): string {
switch (status) { switch (status) {
case 'accepted': return `You have accepted the invitation from ${name}`; case "accepted":
case 'rejected': return `You have rejected the invitation from ${name}`; return `You have accepted the invitation from ${name}`;
case 'pending': return `You have a new invitation pending from ${name}`; case "rejected":
default: return 'There is an update to your invitation'; return `You have rejected the invitation from ${name}`;
case "pending":
return `You have a new invitation pending from ${name}`;
default:
return "There is an update to your invitation";
} }
} }
async function markNotificationAsSent(notificationId: string): Promise<void> { async function markNotificationAsSent(notificationId: string): Promise<void> {
await app.firestore().collection('notifications').doc(notificationId).update({ try {
await app
.firestore()
.collection("notifications")
.doc(notificationId)
.update({
notificationSent: true, notificationSent: true,
sentAt: app.firestore.FieldValue.serverTimestamp() sentAt: admin.firestore.FieldValue.serverTimestamp(),
}); });
logger.info(`Notification ${notificationId} marked as sent`);
} catch (error) {
logger.error(`Error marking notification as sent: ${error}`);
}
} }
async function updateNotificationWithError(notificationId: string, error: string): Promise<void> { async function updateNotificationWithError(
await app.firestore().collection('notifications').doc(notificationId).update({ notificationId: string,
error: string
): Promise<void> {
try {
await app
.firestore()
.collection("notifications")
.doc(notificationId)
.update({
notificationError: error, notificationError: error,
updatedAt: app.firestore.FieldValue.serverTimestamp() notificationSent: false,
updatedAt: admin.firestore.FieldValue.serverTimestamp(),
}); });
logger.info(`Notification ${notificationId} marked with error: ${error}`);
} catch (updateError) {
logger.error(`Error updating notification with error: ${updateError}`);
}
} }

View File

@ -1,62 +1,62 @@
import { onRequest } from "firebase-functions/v2/https"; // import { onRequest } from "firebase-functions/v2/https";
import { Request } from "firebase-functions/v2/https"; // import { Request } from "firebase-functions/v2/https";
import { getCorsHandler } from "../../../shared/middleware"; // import { getCorsHandler } from "../../../shared/middleware";
import { getAdmin, getLogger } from "../../../shared/config"; // import { getAdmin, getLogger } from "../../../shared/config";
import { InvoiceService } from "./invoiceService"; // import { InvoiceService } from "./invoiceService";
const admin = getAdmin(); // const admin = getAdmin();
const logger = getLogger(); // const logger = getLogger();
const corsHandler = getCorsHandler(); // const corsHandler = getCorsHandler();
const invoiceService = new InvoiceService(); // const invoiceService = new InvoiceService();
export const getInvoiceUrl = onRequest({ // export const getInvoiceUrl = onRequest({
region: '#{SERVICES_RGN}#' // region: '#{SERVICES_RGN}#'
}, async (request: Request, response) => { // }, async (request: Request, response) => {
return corsHandler(request, response, async () => { // return corsHandler(request, response, async () => {
try { // try {
const authHeader = request.headers.authorization || ''; // const authHeader = request.headers.authorization || '';
if (!authHeader || !authHeader.startsWith('Bearer ')) { // if (!authHeader || !authHeader.startsWith('Bearer ')) {
response.status(401).json({ error: 'Unauthorized' }); // response.status(401).json({ error: 'Unauthorized' });
return; // return;
} // }
const idToken = authHeader.split('Bearer ')[1]; // const idToken = authHeader.split('Bearer ')[1];
try { // try {
await admin.auth().verifyIdToken(idToken); // await admin.auth().verifyIdToken(idToken);
const { invoicePath } = request.query; // const { invoicePath } = request.query;
if (!invoicePath) { // if (!invoicePath) {
response.status(400).json({ // response.status(400).json({
success: false, // success: false,
error: 'Missing invoice path' // error: 'Missing invoice path'
}); // });
return; // return;
} // }
const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath as string); // const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath as string);
response.json({ // response.json({
success: true, // success: true,
downloadUrl // downloadUrl
}); // });
} catch (authError: any) { // } catch (authError: any) {
logger.error('Authentication error:', authError); // logger.error('Authentication error:', authError);
response.status(401).json({ // response.status(401).json({
success: false, // success: false,
error: 'Invalid authentication token', // error: 'Invalid authentication token',
details: authError.message // details: authError.message
}); // });
} // }
} catch (error: any) { // } catch (error: any) {
logger.error('Error getting invoice URL:', error); // logger.error('Error getting invoice URL:', error);
response.status(500).json({ // response.status(500).json({
success: false, // success: false,
error: 'Failed to get invoice URL', // error: 'Failed to get invoice URL',
details: error.message // details: error.message
}); // });
} // }
}); // });
}); // });

View File

@ -1,13 +1,13 @@
import { getInvoiceUrl } from './getInvoiceUrl'; // import { getInvoiceUrl } from './getInvoiceUrl';
import { InvoiceService } from './invoiceService'; import { InvoiceService } from './invoiceService';
import { processInvoice } from './processInvoice'; // import { processInvoice } from './processInvoice';
import { sendInvoiceEmail } from './sendInvoiceEmail'; // import { sendInvoiceEmail } from './sendInvoiceEmail';
import { directGenerateInvoice } from './directInvoice'; import { directGenerateInvoice } from './directInvoice';
export { export {
getInvoiceUrl, // getInvoiceUrl,
InvoiceService, InvoiceService,
processInvoice, // processInvoice,
sendInvoiceEmail, // sendInvoiceEmail,
directGenerateInvoice, directGenerateInvoice,
}; };

View File

@ -1,83 +1,83 @@
import { onRequest } from "firebase-functions/v2/https"; // import { onRequest } from "firebase-functions/v2/https";
import { Request } from "firebase-functions/v2/https"; // import { Request } from "firebase-functions/v2/https";
import { getCorsHandler } from "../../../shared/middleware"; // import { getCorsHandler } from "../../../shared/middleware";
import { getAdmin, getLogger } from "../../../shared/config"; // import { getAdmin, getLogger } from "../../../shared/config";
import { InvoiceService } from "./invoiceService"; // import { InvoiceService } from "./invoiceService";
const admin = getAdmin(); // const admin = getAdmin();
const logger = getLogger(); // const logger = getLogger();
const corsHandler = getCorsHandler(); // const corsHandler = getCorsHandler();
const invoiceService = new InvoiceService(); // const invoiceService = new InvoiceService();
export const processInvoice = onRequest({ // export const processInvoice = onRequest({
region: '#{SERVICES_RGN}#' // region: '#{SERVICES_RGN}#'
}, async (request: Request, response) => { // }, async (request: Request, response) => {
return corsHandler(request, response, async () => { // return corsHandler(request, response, async () => {
try { // try {
const authHeader = request.headers.authorization || ''; // const authHeader = request.headers.authorization || '';
if (!authHeader || !authHeader.startsWith('Bearer ')) { // if (!authHeader || !authHeader.startsWith('Bearer ')) {
response.status(401).json({ error: 'Unauthorized' }); // response.status(401).json({ error: 'Unauthorized' });
return; // return;
} // }
const idToken = authHeader.split('Bearer ')[1]; // const idToken = authHeader.split('Bearer ')[1];
try { // try {
await admin.auth().verifyIdToken(idToken); // await admin.auth().verifyIdToken(idToken);
const { // const {
membershipId, // membershipId,
paymentId, // paymentId,
invoiceData, // invoiceData,
emailOptions // emailOptions
} = request.body; // } = request.body;
if (!membershipId || !paymentId || !invoiceData) { // if (!membershipId || !paymentId || !invoiceData) {
response.status(400).json({ // response.status(400).json({
success: false, // success: false,
error: 'Missing required fields' // error: 'Missing required fields'
}); // });
return; // return;
} // }
const result = await invoiceService.processInvoice( // const result = await invoiceService.processInvoice(
membershipId, // membershipId,
paymentId, // paymentId,
invoiceData, // invoiceData,
emailOptions // emailOptions
); // );
if (!result.success) { // if (!result.success) {
response.status(400).json({ // response.status(400).json({
success: false, // success: false,
error: result.error || 'Failed to process invoice' // error: result.error || 'Failed to process invoice'
}); // });
return; // return;
} // }
response.json({ // response.json({
success: true, // success: true,
message: 'Invoice processed successfully', // message: 'Invoice processed successfully',
invoicePath: result.invoicePath, // invoicePath: result.invoicePath,
downloadUrl: result.downloadUrl, // downloadUrl: result.downloadUrl,
emailSent: result.emailSent // emailSent: result.emailSent
}); // });
} catch (authError: any) { // } catch (authError: any) {
logger.error('Authentication error:', authError); // logger.error('Authentication error:', authError);
response.status(401).json({ // response.status(401).json({
success: false, // success: false,
error: 'Invalid authentication token', // error: 'Invalid authentication token',
details: authError.message // details: authError.message
}); // });
} // }
} catch (error: any) { // } catch (error: any) {
logger.error('Error processing invoice:', error); // logger.error('Error processing invoice:', error);
response.status(500).json({ // response.status(500).json({
success: false, // success: false,
error: 'Failed to process invoice', // error: 'Failed to process invoice',
details: error.message // details: error.message
}); // });
} // }
}); // });
}); // });

View File

@ -1,91 +1,91 @@
import { onRequest } from "firebase-functions/v2/https"; // import { onRequest } from "firebase-functions/v2/https";
import { Request } from "firebase-functions/v2/https"; // import { Request } from "firebase-functions/v2/https";
import { getCorsHandler } from "../../../shared/middleware"; // import { getCorsHandler } from "../../../shared/middleware";
import { getAdmin, getLogger } from "../../../shared/config"; // import { getAdmin, getLogger } from "../../../shared/config";
import { InvoiceService, EmailOptions } from "./invoiceService"; // import { InvoiceService, EmailOptions } from "./invoiceService";
const admin = getAdmin(); // const admin = getAdmin();
const logger = getLogger(); // const logger = getLogger();
const corsHandler = getCorsHandler(); // const corsHandler = getCorsHandler();
const invoiceService = new InvoiceService(); // const invoiceService = new InvoiceService();
export const sendInvoiceEmail = onRequest({ // export const sendInvoiceEmail = onRequest({
region: '#{SERVICES_RGN}#' // region: '#{SERVICES_RGN}#'
}, async (request: Request, response) => { // }, async (request: Request, response) => {
return corsHandler(request, response, async () => { // return corsHandler(request, response, async () => {
try { // try {
const authHeader = request.headers.authorization || ''; // const authHeader = request.headers.authorization || '';
if (!authHeader || !authHeader.startsWith('Bearer ')) { // if (!authHeader || !authHeader.startsWith('Bearer ')) {
response.status(401).json({ error: 'Unauthorized' }); // response.status(401).json({ error: 'Unauthorized' });
return; // return;
} // }
const idToken = authHeader.split('Bearer ')[1]; // const idToken = authHeader.split('Bearer ')[1];
try { // try {
await admin.auth().verifyIdToken(idToken); // await admin.auth().verifyIdToken(idToken);
const { // const {
invoicePath, // invoicePath,
recipientEmail, // recipientEmail,
recipientName, // recipientName,
subject, // subject,
customHtml, // customHtml,
gymName, // gymName,
planName, // planName,
amount, // amount,
transactionId, // transactionId,
paymentDate, // paymentDate,
paymentMethod // paymentMethod
} = request.body; // } = request.body;
if (!invoicePath || !recipientEmail) { // if (!invoicePath || !recipientEmail) {
response.status(400).json({ // response.status(400).json({
success: false, // success: false,
error: 'Missing required fields' // error: 'Missing required fields'
}); // });
return; // return;
} // }
const emailOptions: EmailOptions = { // const emailOptions: EmailOptions = {
recipientEmail, // recipientEmail,
recipientName, // recipientName,
subject, // subject,
customHtml, // customHtml,
additionalData: { // additionalData: {
gymName, // gymName,
planName, // planName,
amount, // amount,
transactionId, // transactionId,
paymentDate: paymentDate ? new Date(paymentDate) : undefined, // paymentDate: paymentDate ? new Date(paymentDate) : undefined,
paymentMethod // paymentMethod
} // }
}; // };
const emailSent = await invoiceService.sendInvoiceEmail(invoicePath, emailOptions); // const emailSent = await invoiceService.sendInvoiceEmail(invoicePath, emailOptions);
const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath); // const downloadUrl = await invoiceService.getInvoiceDownloadUrl(invoicePath);
response.json({ // response.json({
success: true, // success: true,
message: emailSent ? 'Invoice email sent successfully' : 'Failed to send email but URL generated', // message: emailSent ? 'Invoice email sent successfully' : 'Failed to send email but URL generated',
downloadUrl // downloadUrl
}); // });
} catch (authError: any) { // } catch (authError: any) {
logger.error('Authentication error:', authError); // logger.error('Authentication error:', authError);
response.status(401).json({ // response.status(401).json({
success: false, // success: false,
error: 'Invalid authentication token', // error: 'Invalid authentication token',
details: authError.message // details: authError.message
}); // });
} // }
} catch (error: any) { // } catch (error: any) {
logger.error('Error sending invoice email:', error); // logger.error('Error sending invoice email:', error);
response.status(500).json({ // response.status(500).json({
success: false, // success: false,
error: 'Failed to send invoice email', // error: 'Failed to send invoice email',
details: error.message // details: error.message
}); // });
} // }
}); // });
}); // });

View File

@ -142,7 +142,7 @@ export const phonePeWebhook = onRequest({
} else if (paymentId) { } else if (paymentId) {
await processServicePayment(payload, orderData, paymentId); await processServicePayment(payload, orderData, paymentId);
} else { } else {
logger.error(`No membershipId, bookingId, or serviceId found in metaInfo for order: ${payload.merchantOrderId}`); logger.error(`No membershipId, bookingId, or paymentId found in metaInfo for order: ${payload.merchantOrderId}`);
} }
logger.info(`Payment data updated for completed payment: ${payload.merchantOrderId}`); logger.info(`Payment data updated for completed payment: ${payload.merchantOrderId}`);
@ -639,6 +639,18 @@ async function processServicePayment(payload: any, orderData: any, paymentId: st
return; return;
} }
const serviceData = serviceDoc.data();
if (serviceData?.status === 'ACCEPTED' && serviceData?.paymentDetails?.merchantOrderId) {
logger.warn(`Service payment already processed for serviceId: ${paymentId}, merchantOrderId: ${serviceData.paymentDetails.merchantOrderId}`);
return;
}
if (serviceData?.invoicePath && serviceData?.invoiceNumber) {
logger.warn(`Invoice already exists for serviceId: ${paymentId}, invoicePath: ${serviceData.invoicePath}`);
return;
}
await serviceRef.update({ await serviceRef.update({
status: 'ACCEPTED', status: 'ACCEPTED',
paymentDetails: { paymentDetails: {
@ -651,9 +663,8 @@ async function processServicePayment(payload: any, orderData: any, paymentId: st
updatedAt: admin.firestore.FieldValue.serverTimestamp() updatedAt: admin.firestore.FieldValue.serverTimestamp()
}); });
logger.info(`Updated service booking status to 'CONFIRMED' for serviceId: ${paymentId}`); logger.info(`Updated service booking status to 'CONFIRMED' for paymentId: ${paymentId}`);
const serviceData = serviceDoc.data();
const gymId = orderData.metaInfo?.gymId || serviceData?.gymId; const gymId = orderData.metaInfo?.gymId || serviceData?.gymId;
if (gymId) { if (gymId) {
@ -681,7 +692,7 @@ async function processServicePayment(payload: any, orderData: any, paymentId: st
} }
} }
const invoiceNumber = `SRV-${payload.merchantOrderId.substring(0, 8)}`; const invoiceNumber = `INV-${payload.merchantOrderId.substring(0, 8)}`;
logger.info(`Generated invoice number for service: ${invoiceNumber}`); logger.info(`Generated invoice number for service: ${invoiceNumber}`);
const discountPercentage = orderData.metaInfo?.discount || 0; const discountPercentage = orderData.metaInfo?.discount || 0;
@ -703,7 +714,7 @@ async function processServicePayment(payload: any, orderData: any, paymentId: st
businessName: gymName, businessName: gymName,
address: gymAddress, address: gymAddress,
gstNumber: orderData.metaInfo?.gstNumber, gstNumber: orderData.metaInfo?.gstNumber,
customerName: orderData.metaInfo?.customerName || serviceData?.customerName || '', customerName: orderData.metaInfo?.customerName || serviceData?.normalizedName || '',
phoneNumber: orderData.metaInfo?.customerPhone || serviceData?.phoneNumber || '', phoneNumber: orderData.metaInfo?.customerPhone || serviceData?.phoneNumber || '',
email: orderData.metaInfo?.customerEmail || serviceData?.email || '', email: orderData.metaInfo?.customerEmail || serviceData?.email || '',
planName: orderData.metaInfo?.serviceName || serviceData?.serviceName || 'Service', planName: orderData.metaInfo?.serviceName || serviceData?.serviceName || 'Service',

View File

@ -21,7 +21,7 @@ interface EmailRequest {
interface Attachment { interface Attachment {
filename: string; filename: string;
content: string | Buffer; // Base64 encoded string or Buffer content: string | Buffer;
contentType?: string; contentType?: string;
} }
@ -32,7 +32,7 @@ const stripHtml = (html: string): string => {
async function sendSimpleEmail(data: EmailRequest, recipients: string[]) { async function sendSimpleEmail(data: EmailRequest, recipients: string[]) {
const ses = new SESClient({ const ses = new SESClient({
region: 'ap-south-1', region: process.env.AWS_REGION,
credentials: { credentials: {
accessKeyId: process.env.AWS_ACCESS_KEY_ID || '', accessKeyId: process.env.AWS_ACCESS_KEY_ID || '',
secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || '' secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || ''
@ -58,7 +58,7 @@ async function sendSimpleEmail(data: EmailRequest, recipients: string[]) {
async function sendEmailWithAttachments(data: EmailRequest, recipients: string[]) { async function sendEmailWithAttachments(data: EmailRequest, recipients: string[]) {
const ses = new SESClient({ const ses = new SESClient({
region: 'ap-south-1', region: process.env.AWS_REGION,
credentials: { credentials: {
accessKeyId: process.env.AWS_ACCESS_KEY_ID || '', accessKeyId: process.env.AWS_ACCESS_KEY_ID || '',
secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || '' secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || ''
@ -72,26 +72,21 @@ async function sendEmailWithAttachments(data: EmailRequest, recipients: string[]
rawMessage += `MIME-Version: 1.0\n`; rawMessage += `MIME-Version: 1.0\n`;
rawMessage += `Content-Type: multipart/mixed; boundary="${boundary}"\n\n`; rawMessage += `Content-Type: multipart/mixed; boundary="${boundary}"\n\n`;
// Add email body (multipart/alternative)
rawMessage += `--${boundary}\n`; rawMessage += `--${boundary}\n`;
rawMessage += `Content-Type: multipart/alternative; boundary="alt_${boundary}"\n\n`; rawMessage += `Content-Type: multipart/alternative; boundary="alt_${boundary}"\n\n`;
// Text part
if (data.text) { if (data.text) {
rawMessage += `--alt_${boundary}\n`; rawMessage += `--alt_${boundary}\n`;
rawMessage += `Content-Type: text/plain; charset=UTF-8\n\n`; rawMessage += `Content-Type: text/plain; charset=UTF-8\n\n`;
rawMessage += `${data.text}\n\n`; rawMessage += `${data.text}\n\n`;
} }
// HTML part
rawMessage += `--alt_${boundary}\n`; rawMessage += `--alt_${boundary}\n`;
rawMessage += `Content-Type: text/html; charset=UTF-8\n\n`; rawMessage += `Content-Type: text/html; charset=UTF-8\n\n`;
rawMessage += `${data.html}\n\n`; rawMessage += `${data.html}\n\n`;
// Close alternative part
rawMessage += `--alt_${boundary}--\n\n`; rawMessage += `--alt_${boundary}--\n\n`;
// Add attachments
for (const attachment of data.attachments || []) { for (const attachment of data.attachments || []) {
const contentType = attachment.contentType || const contentType = attachment.contentType ||
mime.lookup(attachment.filename) || mime.lookup(attachment.filename) ||
@ -109,7 +104,6 @@ async function sendEmailWithAttachments(data: EmailRequest, recipients: string[]
rawMessage += contentBuffer.toString('base64') + '\n\n'; rawMessage += contentBuffer.toString('base64') + '\n\n';
} }
// Close message
rawMessage += `--${boundary}--`; rawMessage += `--${boundary}--`;
const command = new SendRawEmailCommand({ const command = new SendRawEmailCommand({
@ -140,7 +134,6 @@ export async function sendEmailWithAttachmentUtil(
try { try {
logger.info(`Sending email with attachment to: ${toAddress}`); logger.info(`Sending email with attachment to: ${toAddress}`);
// Initialize data with basic fields
const data: EmailRequest = { const data: EmailRequest = {
to: toAddress, to: toAddress,
html: message, html: message,
@ -151,13 +144,11 @@ export async function sendEmailWithAttachmentUtil(
attachments: [] attachments: []
}; };
// Handle file URL if provided
if (fileUrl && fileName) { if (fileUrl && fileName) {
logger.info(`Downloading attachment from URL: ${fileUrl}`); logger.info(`Downloading attachment from URL: ${fileUrl}`);
try { try {
const fileContent = await downloadFileFromUrl(fileUrl); const fileContent = await downloadFileFromUrl(fileUrl);
// Add the downloaded file as an attachment
data.attachments!.push({ data.attachments!.push({
filename: fileName, filename: fileName,
content: fileContent, content: fileContent,

242
package-lock.json generated
View File

@ -1,242 +0,0 @@
{
"name": "fitlien-services",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
"dependencies": {
"@types/busboy": "^1.5.4",
"@types/nodemailer": "^6.4.17",
"@types/pdfkit": "^0.13.9",
"busboy": "^1.6.0",
"date-fns": "^4.1.0",
"nodemailer": "^7.0.3",
"pdfkit": "^0.17.1"
}
},
"node_modules/@swc/helpers": {
"version": "0.5.17",
"resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.17.tgz",
"integrity": "sha512-5IKx/Y13RsYd+sauPb2x+U/xZikHjolzfuDgTAl/Tdf3Q8rslRvC19NKDLgAJQ6wsqADk10ntlv08nPFw/gO/A==",
"dependencies": {
"tslib": "^2.8.0"
}
},
"node_modules/@types/busboy": {
"version": "1.5.4",
"resolved": "https://registry.npmjs.org/@types/busboy/-/busboy-1.5.4.tgz",
"integrity": "sha512-kG7WrUuAKK0NoyxfQHsVE6j1m01s6kMma64E+OZenQABMQyTJop1DumUWcLwAQ2JzpefU7PDYoRDKl8uZosFjw==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/node": {
"version": "22.10.2",
"resolved": "https://registry.npmjs.org/@types/node/-/node-22.10.2.tgz",
"integrity": "sha512-Xxr6BBRCAOQixvonOye19wnzyDiUtTeqldOOmj3CkeblonbccA12PFwlufvRdrpjXxqnmUaeiU5EOA+7s5diUQ==",
"dependencies": {
"undici-types": "~6.20.0"
}
},
"node_modules/@types/nodemailer": {
"version": "6.4.17",
"resolved": "https://registry.npmjs.org/@types/nodemailer/-/nodemailer-6.4.17.tgz",
"integrity": "sha512-I9CCaIp6DTldEg7vyUTZi8+9Vo0hi1/T8gv3C89yk1rSAAzoKQ8H8ki/jBYJSFoH/BisgLP8tkZMlQ91CIquww==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/pdfkit": {
"version": "0.13.9",
"resolved": "https://registry.npmjs.org/@types/pdfkit/-/pdfkit-0.13.9.tgz",
"integrity": "sha512-RDG8Yb1zT7I01FfpwK7nMSA433XWpblMqSCtA5vJlSyavWZb303HUYPCel6JTiDDFqwGLvtAnYbH8N/e0Cb89g==",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/base64-js": {
"version": "1.5.1",
"resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
"integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
"funding": [
{
"type": "github",
"url": "https://github.com/sponsors/feross"
},
{
"type": "patreon",
"url": "https://www.patreon.com/feross"
},
{
"type": "consulting",
"url": "https://feross.org/support"
}
]
},
"node_modules/brotli": {
"version": "1.3.3",
"resolved": "https://registry.npmjs.org/brotli/-/brotli-1.3.3.tgz",
"integrity": "sha512-oTKjJdShmDuGW94SyyaoQvAjf30dZaHnjJ8uAF+u2/vGJkJbJPJAT1gDiOJP5v1Zb6f9KEyW/1HpuaWIXtGHPg==",
"dependencies": {
"base64-js": "^1.1.2"
}
},
"node_modules/busboy": {
"version": "1.6.0",
"resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz",
"integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==",
"dependencies": {
"streamsearch": "^1.1.0"
},
"engines": {
"node": ">=10.16.0"
}
},
"node_modules/clone": {
"version": "2.1.2",
"resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz",
"integrity": "sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w==",
"engines": {
"node": ">=0.8"
}
},
"node_modules/crypto-js": {
"version": "4.2.0",
"resolved": "https://registry.npmjs.org/crypto-js/-/crypto-js-4.2.0.tgz",
"integrity": "sha512-KALDyEYgpY+Rlob/iriUtjV6d5Eq+Y191A5g4UqLAi8CyGP9N1+FdVbkc1SxKc2r4YAYqG8JzO2KGL+AizD70Q=="
},
"node_modules/date-fns": {
"version": "4.1.0",
"resolved": "https://registry.npmjs.org/date-fns/-/date-fns-4.1.0.tgz",
"integrity": "sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==",
"funding": {
"type": "github",
"url": "https://github.com/sponsors/kossnocorp"
}
},
"node_modules/dfa": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/dfa/-/dfa-1.2.0.tgz",
"integrity": "sha512-ED3jP8saaweFTjeGX8HQPjeC1YYyZs98jGNZx6IiBvxW7JG5v492kamAQB3m2wop07CvU/RQmzcKr6bgcC5D/Q=="
},
"node_modules/fast-deep-equal": {
"version": "3.1.3",
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
"integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="
},
"node_modules/fontkit": {
"version": "2.0.4",
"resolved": "https://registry.npmjs.org/fontkit/-/fontkit-2.0.4.tgz",
"integrity": "sha512-syetQadaUEDNdxdugga9CpEYVaQIxOwk7GlwZWWZ19//qW4zE5bknOKeMBDYAASwnpaSHKJITRLMF9m1fp3s6g==",
"dependencies": {
"@swc/helpers": "^0.5.12",
"brotli": "^1.3.2",
"clone": "^2.1.2",
"dfa": "^1.2.0",
"fast-deep-equal": "^3.1.3",
"restructure": "^3.0.0",
"tiny-inflate": "^1.0.3",
"unicode-properties": "^1.4.0",
"unicode-trie": "^2.0.0"
}
},
"node_modules/jpeg-exif": {
"version": "1.1.4",
"resolved": "https://registry.npmjs.org/jpeg-exif/-/jpeg-exif-1.1.4.tgz",
"integrity": "sha512-a+bKEcCjtuW5WTdgeXFzswSrdqi0jk4XlEtZlx5A94wCoBpFjfFTbo/Tra5SpNCl/YFZPvcV1dJc+TAYeg6ROQ=="
},
"node_modules/linebreak": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/linebreak/-/linebreak-1.1.0.tgz",
"integrity": "sha512-MHp03UImeVhB7XZtjd0E4n6+3xr5Dq/9xI/5FptGk5FrbDR3zagPa2DS6U8ks/3HjbKWG9Q1M2ufOzxV2qLYSQ==",
"dependencies": {
"base64-js": "0.0.8",
"unicode-trie": "^2.0.0"
}
},
"node_modules/linebreak/node_modules/base64-js": {
"version": "0.0.8",
"resolved": "https://registry.npmjs.org/base64-js/-/base64-js-0.0.8.tgz",
"integrity": "sha512-3XSA2cR/h/73EzlXXdU6YNycmYI7+kicTxks4eJg2g39biHR84slg2+des+p7iHYhbRg/udIS4TD53WabcOUkw==",
"engines": {
"node": ">= 0.4"
}
},
"node_modules/nodemailer": {
"version": "7.0.3",
"resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-7.0.3.tgz",
"integrity": "sha512-Ajq6Sz1x7cIK3pN6KesGTah+1gnwMnx5gKl3piQlQQE/PwyJ4Mbc8is2psWYxK3RJTVeqsDaCv8ZzXLCDHMTZw==",
"engines": {
"node": ">=6.0.0"
}
},
"node_modules/pako": {
"version": "0.2.9",
"resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz",
"integrity": "sha512-NUcwaKxUxWrZLpDG+z/xZaCgQITkA/Dv4V/T6bw7VON6l1Xz/VnrBqrYjZQ12TamKHzITTfOEIYUj48y2KXImA=="
},
"node_modules/pdfkit": {
"version": "0.17.1",
"resolved": "https://registry.npmjs.org/pdfkit/-/pdfkit-0.17.1.tgz",
"integrity": "sha512-Kkf1I9no14O/uo593DYph5u3QwiMfby7JsBSErN1WqeyTgCBNJE3K4pXBn3TgkdKUIVu+buSl4bYUNC+8Up4xg==",
"dependencies": {
"crypto-js": "^4.2.0",
"fontkit": "^2.0.4",
"jpeg-exif": "^1.1.4",
"linebreak": "^1.1.0",
"png-js": "^1.0.0"
}
},
"node_modules/png-js": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/png-js/-/png-js-1.0.0.tgz",
"integrity": "sha512-k+YsbhpA9e+EFfKjTCH3VW6aoKlyNYI6NYdTfDL4CIvFnvsuO84ttonmZE7rc+v23SLTH8XX+5w/Ak9v0xGY4g=="
},
"node_modules/restructure": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/restructure/-/restructure-3.0.2.tgz",
"integrity": "sha512-gSfoiOEA0VPE6Tukkrr7I0RBdE0s7H1eFCDBk05l1KIQT1UIKNc5JZy6jdyW6eYH3aR3g5b3PuL77rq0hvwtAw=="
},
"node_modules/streamsearch": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz",
"integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==",
"engines": {
"node": ">=10.0.0"
}
},
"node_modules/tiny-inflate": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/tiny-inflate/-/tiny-inflate-1.0.3.tgz",
"integrity": "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw=="
},
"node_modules/tslib": {
"version": "2.8.1",
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
"integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="
},
"node_modules/undici-types": {
"version": "6.20.0",
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.20.0.tgz",
"integrity": "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="
},
"node_modules/unicode-properties": {
"version": "1.4.1",
"resolved": "https://registry.npmjs.org/unicode-properties/-/unicode-properties-1.4.1.tgz",
"integrity": "sha512-CLjCCLQ6UuMxWnbIylkisbRj31qxHPAurvena/0iwSVbQ2G1VY5/HjV0IRabOEbDHlzZlRdCrD4NhB0JtU40Pg==",
"dependencies": {
"base64-js": "^1.3.0",
"unicode-trie": "^2.0.0"
}
},
"node_modules/unicode-trie": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/unicode-trie/-/unicode-trie-2.0.0.tgz",
"integrity": "sha512-x7bc76x0bm4prf1VLg79uhAzKw8DVboClSN5VxJuQ+LKDOVEW9CdH+VY7SP+vX7xCYQqzzgQpFqz15zeLvAtZQ==",
"dependencies": {
"pako": "^0.2.5",
"tiny-inflate": "^1.0.0"
}
}
}
}

View File

@ -1,7 +0,0 @@
{
"dependencies": {
"@types/busboy": "^1.5.4",
"busboy": "^1.6.0",
"date-fns": "^4.1.0"
}
}