Compare commits

...

87 Commits

Author SHA1 Message Date
c90f8cb1fa geofence refinement
and manual failsafe
2026-04-03 00:13:22 +02:00
71ccea85d2 geofence impemented 2026-04-02 22:47:27 +02:00
e2706118d1 dbs 2026-04-01 20:25:35 +02:00
4e322503cc db 2026-03-31 21:54:33 +02:00
e9685193a4 reduce statistics refetch log noise and dedupe requests 2026-03-31 21:54:04 +02:00
ad3eba48b0 remove attendance tab and screen from mobile navigation 2026-03-31 20:25:04 +02:00
0ccf59344e db 2026-03-31 20:16:49 +02:00
42122ac341 Merge branch 'adminRecc' 2026-03-31 20:04:37 +02:00
4dd2ed5839 dbg 2026-03-31 20:04:05 +02:00
f9a588fcd6 db 2026-03-31 20:03:52 +02:00
9330f4fd05 regenerate linked active goals when admin approves ai recommendation 2026-03-31 20:02:28 +02:00
d6683e6e5e dbs 2026-03-31 19:51:44 +02:00
bac7df33e8 Merge branch 'screen2' 2026-03-31 19:46:00 +02:00
178ad3fa90 db 2026-03-31 19:44:55 +02:00
ef9f39e564 add resilient self-plan generation with provider fallback 2026-03-31 19:43:19 +02:00
73218402f6 fix self ai plan generation authorization and error handling 2026-03-31 19:31:45 +02:00
c877577fba fix ai activity plan conversion and immediate goals refresh 2026-03-31 19:21:00 +02:00
e119f0923c pause previous ai-linked goals on self plan regeneration 2026-03-31 19:10:17 +02:00
a65b3cac08 add client self-service ai activity plans on goals screen 2026-03-31 18:51:30 +02:00
0825bb3d65 Merge branch 'workout' 2026-03-31 18:19:12 +02:00
6740dcb18f android buuild clean uo p 2026-03-31 18:18:36 +02:00
12d6c07186 fix step progress header overflow on home card 2026-03-31 18:16:59 +02:00
5010a579d6 add daily pedometer steps metric on home screen 2026-03-31 18:00:35 +02:00
2cff8eafbd add home workout quick action with attendance check-in/out 2026-03-31 17:38:12 +02:00
275248fc35 dbs 2026-03-31 17:18:41 +02:00
4c2e97b66d Merge branch 'screen1' 2026-03-31 17:05:43 +02:00
21afb085e3 scan food 2026-03-31 16:55:55 +02:00
ca64a100b6 integrate barcode scan with openfoodfacts and meal type selection 2026-03-31 16:36:18 +02:00
3c3dfb6cd6 hydration calories persistance 2026-03-31 16:17:08 +02:00
871f33bf5a dfg 2026-03-31 15:57:23 +02:00
c5dde63355 dbs 2026-03-31 13:48:39 +02:00
cd13333b52 stabilize membership loading and home motivational message 2026-03-30 20:24:13 +02:00
a620921202 Merge branch 'uifix' 2026-03-29 20:08:39 +02:00
ed14c57749 normalize mobile auth and profile ui to theme components 2026-03-29 20:07:33 +02:00
7ada05da6a db up 2026-03-29 19:54:17 +02:00
50ece15089 add membership features endpoint and use it in mobile 2026-03-29 19:51:36 +02:00
091cb5ba85 Merge branch 'planDef' 2026-03-29 16:26:38 +02:00
ebfd633a11 enforce membership feature access in api and surface plans in admin 2026-03-29 16:22:45 +02:00
1f4800c055 membership 2026-03-29 16:05:10 +02:00
0ddac10c59 Merge branch 'phase-9-release-hardening' 2026-03-29 15:54:11 +02:00
573690ab02 Merge branch 'phase-8-api-contract-tightening' 2026-03-29 15:54:00 +02:00
efbfa58c10 Merge branch 'phase-7-query-cache-consolidation' 2026-03-29 15:53:47 +02:00
7ad1e5133e Merge branch 'phase-6-coverage-expansion' 2026-03-29 15:53:27 +02:00
9c3d3f5b72 add release hardening checklist for admin and mobile 2026-03-29 15:49:47 +02:00
ff9f3d582a standardize mobile api response parsing with shared helper 2026-03-29 15:48:58 +02:00
60d7a7963d initialize react-query provider for mobile app 2026-03-29 15:46:59 +02:00
0eede3fa91 add mobile notifications api unit tests 2026-03-29 15:46:16 +02:00
b1f84722af Merge branch 'nextPhase' 2026-03-29 15:41:09 +02:00
34e88bdde5 reset mobile context caches on user identity changes 2026-03-29 15:38:51 +02:00
aa662a9b74 add mobile api unit tests for gyms and recommendations 2026-03-29 15:34:43 +02:00
80110acbf7 standardize mobile api transport for gyms and core services 2026-03-29 15:31:06 +02:00
c36cad9c54 Merge branch 'testRep' 2026-03-29 11:27:14 +02:00
2ecb8a3515 android 2026-03-29 11:25:28 +02:00
8275da687b fix mobile profile flow and onboarding edge cases 2026-03-29 11:23:49 +02:00
10b58245f5 harden admin api authorization and add authz regression tests 2026-03-29 11:23:49 +02:00
272c9b36dd todos 2026-03-29 09:52:22 +02:00
e586662c19 reports fully implemented 2026-03-19 05:04:12 +01:00
06973ccfb2 assigment and reports groundwork 2026-03-19 04:36:35 +01:00
74f0d0dbed user report gtound work 2026-03-19 03:37:15 +01:00
1be2de05fa todos 2026-03-19 01:49:07 +01:00
1be7dc2858 user id -> user name
in attendace
2026-03-19 01:37:10 +01:00
36f52b42c6 ash 2026-03-19 01:32:18 +01:00
3e30fae173 attendace, user stats 2026-03-19 01:29:09 +01:00
d6a77fcd23 various fixes
raw sql to drizzle
2026-03-19 00:57:04 +01:00
ffd3aabc55 delete gym working 2026-03-19 00:44:39 +01:00
8cac57ed67 db updated 2026-03-19 00:39:23 +01:00
d4fae890bb invitation flow finished 2026-03-18 23:58:14 +01:00
0817e8e72b invitation flow basis 2026-03-18 23:08:55 +01:00
b1f01208fa admin role enhancment
basic invitation flow
2026-03-18 13:56:51 +01:00
d112dbb122 rb fine tune 2026-03-18 13:14:47 +01:00
7043487fc2 edit local user fix 2026-03-18 12:37:21 +01:00
bb9c675421 setings page enhanced
gym managment added
2026-03-18 06:23:40 +01:00
624cdfc45c role based auth
implemented superadmin -> admin -> traniner
2026-03-18 06:06:01 +01:00
16217f46ff schema update 2026-03-18 04:27:08 +01:00
cc2dbc3423 Merge branch 'adminUI' 2026-03-18 03:57:25 +01:00
b1d6489c70 css 2026-03-18 03:55:28 +01:00
0abb41f573 admin ui enhacement 2026-03-18 02:36:12 +01:00
02c9681aca performance improvments 2026-03-18 02:04:09 +01:00
912981e3f3 dbb 2026-03-18 01:18:34 +01:00
963404434a db up 2026-03-18 01:06:55 +01:00
7f22a39886 fitness profile fix 2026-03-18 00:35:21 +01:00
0776517fb7 Merge branch 'anotherTake' 2026-03-12 20:21:41 +01:00
064dafad57 refinenments 2026-03-12 20:19:48 +01:00
5d6166df1b redesign take 2 complete
fix artefacts from previous dessign
2026-03-12 17:56:46 +01:00
aba9b1395b fitness goals
on home screen update fix
2026-03-12 17:06:10 +01:00
c3a41d2b32 actual fix 2026-03-12 16:58:14 +01:00
254a30ff93 fitness profile validation error fixed 2026-03-12 16:45:38 +01:00
335 changed files with 22065 additions and 22283 deletions

View File

@ -14,6 +14,10 @@ RESEND_API_KEY=re_your_resend_api_key_here
EMAIL_FROM=FitAI <noreply@yourdomain.com>
EMAIL_REPLY_TO=support@yourdomain.com
# Admin App URL (for invitation redirects)
# Set to your production URL in production
NEXT_PUBLIC_APP_URL=http://localhost:3000
# Database (optional - defaults to ./fitai.db)
DATABASE_PATH=./fitai.db

Binary file not shown.

View File

@ -13,6 +13,7 @@
"@fitai/shared": "file:../../packages/shared",
"@hookform/resolvers": "^5.2.2",
"@radix-ui/react-dialog": "^1.1.15",
"@radix-ui/react-select": "^2.2.6",
"@radix-ui/react-slot": "^1.2.4",
"@react-email/components": "^1.0.8",
"@react-email/render": "^2.0.4",
@ -32,6 +33,8 @@
"date-fns": "^4.1.0",
"dotenv": "^17.2.3",
"drizzle-orm": "^0.44.7",
"jspdf": "^4.2.1",
"jspdf-autotable": "^5.0.7",
"lucide-react": "^0.553.0",
"next": "^16.0.1",
"pino": "^10.3.1",
@ -574,10 +577,9 @@
}
},
"node_modules/@babel/runtime": {
"version": "7.28.4",
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.28.4.tgz",
"integrity": "sha512-Q/N6JNWvIvPnLDvjlE1OUBLPQHH6l3CltCEsHIujp45zQUSSh8K+gHnaEX45yAT1nyngnINhvWtzN+Nb9D8RAQ==",
"dev": true,
"version": "7.29.2",
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.2.tgz",
"integrity": "sha512-JiDShH45zKHWyGe4ZNVRrCjBz8Nh9TMmZG1kh4QTK8hCBTWBi8Da+i7s1fJw7/lYpM4ccepSNfqzZ/QvABBi5g==",
"license": "MIT",
"engines": {
"node": ">=6.9.0"
@ -1027,6 +1029,44 @@
"resolved": "../../packages/shared",
"link": true
},
"node_modules/@floating-ui/core": {
"version": "1.7.5",
"resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.7.5.tgz",
"integrity": "sha512-1Ih4WTWyw0+lKyFMcBHGbb5U5FtuHJuujoyyr5zTaWS5EYMeT6Jb2AuDeftsCsEuchO+mM2ij5+q9crhydzLhQ==",
"license": "MIT",
"dependencies": {
"@floating-ui/utils": "^0.2.11"
}
},
"node_modules/@floating-ui/dom": {
"version": "1.7.6",
"resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.7.6.tgz",
"integrity": "sha512-9gZSAI5XM36880PPMm//9dfiEngYoC6Am2izES1FF406YFsjvyBMmeJ2g4SAju3xWwtuynNRFL2s9hgxpLI5SQ==",
"license": "MIT",
"dependencies": {
"@floating-ui/core": "^1.7.5",
"@floating-ui/utils": "^0.2.11"
}
},
"node_modules/@floating-ui/react-dom": {
"version": "2.1.8",
"resolved": "https://registry.npmjs.org/@floating-ui/react-dom/-/react-dom-2.1.8.tgz",
"integrity": "sha512-cC52bHwM/n/CxS87FH0yWdngEZrjdtLW/qVruo68qg+prK7ZQ4YGdut2GyDVpoGeAYe/h899rVeOVm6Oi40k2A==",
"license": "MIT",
"dependencies": {
"@floating-ui/dom": "^1.7.6"
},
"peerDependencies": {
"react": ">=16.8.0",
"react-dom": ">=16.8.0"
}
},
"node_modules/@floating-ui/utils": {
"version": "0.2.11",
"resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.11.tgz",
"integrity": "sha512-RiB/yIh78pcIxl6lLMG0CgBXAZ2Y0eVHqMPYugu+9U0AeT6YBeiJpf7lbdJNIugFP5SIjwNRgo4DhR1Qxi26Gg==",
"license": "MIT"
},
"node_modules/@gar/promisify": {
"version": "1.1.3",
"resolved": "https://registry.npmjs.org/@gar/promisify/-/promisify-1.1.3.tgz",
@ -2417,12 +2457,85 @@
"url": "https://opencollective.com/pkgr"
}
},
"node_modules/@radix-ui/number": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@radix-ui/number/-/number-1.1.1.tgz",
"integrity": "sha512-MkKCwxlXTgz6CFoJx3pCwn07GKp36+aZyu/u2Ln2VrA5DcdyCZkASEDBTd8x5whTQQL5CiYf4prXKLcgQdv29g==",
"license": "MIT"
},
"node_modules/@radix-ui/primitive": {
"version": "1.1.3",
"resolved": "https://registry.npmjs.org/@radix-ui/primitive/-/primitive-1.1.3.tgz",
"integrity": "sha512-JTF99U/6XIjCBo0wqkU5sK10glYe27MRRsfwoiq5zzOEZLHU3A3KCMa5X/azekYRCJ0HlwI0crAXS/5dEHTzDg==",
"license": "MIT"
},
"node_modules/@radix-ui/react-arrow": {
"version": "1.1.7",
"resolved": "https://registry.npmjs.org/@radix-ui/react-arrow/-/react-arrow-1.1.7.tgz",
"integrity": "sha512-F+M1tLhO+mlQaOWspE8Wstg+z6PwxwRd8oQ8IXceWz92kfAmalTRf0EjrouQeo7QssEPfCn05B4Ihs1K9WQ/7w==",
"license": "MIT",
"dependencies": {
"@radix-ui/react-primitive": "2.1.3"
},
"peerDependencies": {
"@types/react": "*",
"@types/react-dom": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
"react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
},
"@types/react-dom": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-collection": {
"version": "1.1.7",
"resolved": "https://registry.npmjs.org/@radix-ui/react-collection/-/react-collection-1.1.7.tgz",
"integrity": "sha512-Fh9rGN0MoI4ZFUNyfFVNU4y9LUz93u9/0K+yLgA2bwRojxM8JU1DyvvMBabnZPBgMWREAJvU2jjVzq+LrFUglw==",
"license": "MIT",
"dependencies": {
"@radix-ui/react-compose-refs": "1.1.2",
"@radix-ui/react-context": "1.1.2",
"@radix-ui/react-primitive": "2.1.3",
"@radix-ui/react-slot": "1.2.3"
},
"peerDependencies": {
"@types/react": "*",
"@types/react-dom": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
"react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
},
"@types/react-dom": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-collection/node_modules/@radix-ui/react-slot": {
"version": "1.2.3",
"resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.3.tgz",
"integrity": "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A==",
"license": "MIT",
"dependencies": {
"@radix-ui/react-compose-refs": "1.1.2"
},
"peerDependencies": {
"@types/react": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-compose-refs": {
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.2.tgz",
@ -2507,6 +2620,21 @@
}
}
},
"node_modules/@radix-ui/react-direction": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@radix-ui/react-direction/-/react-direction-1.1.1.tgz",
"integrity": "sha512-1UEWRX6jnOA2y4H5WczZ44gOOjTEmlqv1uNW4GAJEO5+bauCBhv8snY65Iw5/VOS/ghKN9gr2KjnLKxrsvoMVw==",
"license": "MIT",
"peerDependencies": {
"@types/react": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-dismissable-layer": {
"version": "1.1.11",
"resolved": "https://registry.npmjs.org/@radix-ui/react-dismissable-layer/-/react-dismissable-layer-1.1.11.tgz",
@ -2592,6 +2720,38 @@
}
}
},
"node_modules/@radix-ui/react-popper": {
"version": "1.2.8",
"resolved": "https://registry.npmjs.org/@radix-ui/react-popper/-/react-popper-1.2.8.tgz",
"integrity": "sha512-0NJQ4LFFUuWkE7Oxf0htBKS6zLkkjBH+hM1uk7Ng705ReR8m/uelduy1DBo0PyBXPKVnBA6YBlU94MBGXrSBCw==",
"license": "MIT",
"dependencies": {
"@floating-ui/react-dom": "^2.0.0",
"@radix-ui/react-arrow": "1.1.7",
"@radix-ui/react-compose-refs": "1.1.2",
"@radix-ui/react-context": "1.1.2",
"@radix-ui/react-primitive": "2.1.3",
"@radix-ui/react-use-callback-ref": "1.1.1",
"@radix-ui/react-use-layout-effect": "1.1.1",
"@radix-ui/react-use-rect": "1.1.1",
"@radix-ui/react-use-size": "1.1.1",
"@radix-ui/rect": "1.1.1"
},
"peerDependencies": {
"@types/react": "*",
"@types/react-dom": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
"react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
},
"@types/react-dom": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-portal": {
"version": "1.1.9",
"resolved": "https://registry.npmjs.org/@radix-ui/react-portal/-/react-portal-1.1.9.tgz",
@ -2681,6 +2841,67 @@
}
}
},
"node_modules/@radix-ui/react-select": {
"version": "2.2.6",
"resolved": "https://registry.npmjs.org/@radix-ui/react-select/-/react-select-2.2.6.tgz",
"integrity": "sha512-I30RydO+bnn2PQztvo25tswPH+wFBjehVGtmagkU78yMdwTwVf12wnAOF+AeP8S2N8xD+5UPbGhkUfPyvT+mwQ==",
"license": "MIT",
"dependencies": {
"@radix-ui/number": "1.1.1",
"@radix-ui/primitive": "1.1.3",
"@radix-ui/react-collection": "1.1.7",
"@radix-ui/react-compose-refs": "1.1.2",
"@radix-ui/react-context": "1.1.2",
"@radix-ui/react-direction": "1.1.1",
"@radix-ui/react-dismissable-layer": "1.1.11",
"@radix-ui/react-focus-guards": "1.1.3",
"@radix-ui/react-focus-scope": "1.1.7",
"@radix-ui/react-id": "1.1.1",
"@radix-ui/react-popper": "1.2.8",
"@radix-ui/react-portal": "1.1.9",
"@radix-ui/react-primitive": "2.1.3",
"@radix-ui/react-slot": "1.2.3",
"@radix-ui/react-use-callback-ref": "1.1.1",
"@radix-ui/react-use-controllable-state": "1.2.2",
"@radix-ui/react-use-layout-effect": "1.1.1",
"@radix-ui/react-use-previous": "1.1.1",
"@radix-ui/react-visually-hidden": "1.2.3",
"aria-hidden": "^1.2.4",
"react-remove-scroll": "^2.6.3"
},
"peerDependencies": {
"@types/react": "*",
"@types/react-dom": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
"react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
},
"@types/react-dom": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-select/node_modules/@radix-ui/react-slot": {
"version": "1.2.3",
"resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.3.tgz",
"integrity": "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A==",
"license": "MIT",
"dependencies": {
"@radix-ui/react-compose-refs": "1.1.2"
},
"peerDependencies": {
"@types/react": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-slot": {
"version": "1.2.4",
"resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.4.tgz",
@ -2784,6 +3005,86 @@
}
}
},
"node_modules/@radix-ui/react-use-previous": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@radix-ui/react-use-previous/-/react-use-previous-1.1.1.tgz",
"integrity": "sha512-2dHfToCj/pzca2Ck724OZ5L0EVrr3eHRNsG/b3xQJLA2hZpVCS99bLAX+hm1IHXDEnzU6by5z/5MIY794/a8NQ==",
"license": "MIT",
"peerDependencies": {
"@types/react": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-use-rect": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@radix-ui/react-use-rect/-/react-use-rect-1.1.1.tgz",
"integrity": "sha512-QTYuDesS0VtuHNNvMh+CjlKJ4LJickCMUAqjlE3+j8w+RlRpwyX3apEQKGFzbZGdo7XNG1tXa+bQqIE7HIXT2w==",
"license": "MIT",
"dependencies": {
"@radix-ui/rect": "1.1.1"
},
"peerDependencies": {
"@types/react": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-use-size": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@radix-ui/react-use-size/-/react-use-size-1.1.1.tgz",
"integrity": "sha512-ewrXRDTAqAXlkl6t/fkXWNAhFX9I+CkKlw6zjEwk86RSPKwZr3xpBRso655aqYafwtnbpHLj6toFzmd6xdVptQ==",
"license": "MIT",
"dependencies": {
"@radix-ui/react-use-layout-effect": "1.1.1"
},
"peerDependencies": {
"@types/react": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
}
}
},
"node_modules/@radix-ui/react-visually-hidden": {
"version": "1.2.3",
"resolved": "https://registry.npmjs.org/@radix-ui/react-visually-hidden/-/react-visually-hidden-1.2.3.tgz",
"integrity": "sha512-pzJq12tEaaIhqjbzpCuv/OypJY/BPavOofm+dbab+MHLajy277+1lLm6JFcGgF5eskJ6mquGirhXY2GD/8u8Ug==",
"license": "MIT",
"dependencies": {
"@radix-ui/react-primitive": "2.1.3"
},
"peerDependencies": {
"@types/react": "*",
"@types/react-dom": "*",
"react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
"react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
},
"peerDependenciesMeta": {
"@types/react": {
"optional": true
},
"@types/react-dom": {
"optional": true
}
}
},
"node_modules/@radix-ui/rect": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@radix-ui/rect/-/rect-1.1.1.tgz",
"integrity": "sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw==",
"license": "MIT"
},
"node_modules/@react-email/body": {
"version": "0.2.1",
"resolved": "https://registry.npmjs.org/@react-email/body/-/body-0.2.1.tgz",
@ -3019,7 +3320,6 @@
"resolved": "https://registry.npmjs.org/@react-email/render/-/render-2.0.4.tgz",
"integrity": "sha512-kht2oTFQ1SwrLpd882ahTvUtNa9s53CERHstiTbzhm6aR2Hbykp/mQ4tpPvsBGkKAEvKRlDEoooh60Uk6nHK1g==",
"license": "MIT",
"peer": true,
"dependencies": {
"html-to-text": "^9.0.5",
"prettier": "^3.5.3"
@ -3595,6 +3895,19 @@
"undici-types": "~7.16.0"
}
},
"node_modules/@types/pako": {
"version": "2.0.4",
"resolved": "https://registry.npmjs.org/@types/pako/-/pako-2.0.4.tgz",
"integrity": "sha512-VWDCbrLeVXJM9fihYodcLiIv0ku+AlOa/TQ1SvYOaBuyrSKgEcro95LJyIsJ4vSo6BXIxOKxiJAat04CmST9Fw==",
"license": "MIT"
},
"node_modules/@types/raf": {
"version": "3.4.3",
"resolved": "https://registry.npmjs.org/@types/raf/-/raf-3.4.3.tgz",
"integrity": "sha512-c4YAvMedbPZ5tEyxzQdMoOhhJ4RD3rngZIdwC2/qDN3d7JpEhB6fiBRKVY1lg5B7Wk+uPBjn5f39j1/2MY1oOw==",
"license": "MIT",
"optional": true
},
"node_modules/@types/react": {
"version": "19.2.2",
"resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.2.tgz",
@ -3641,6 +3954,13 @@
"dev": true,
"license": "MIT"
},
"node_modules/@types/trusted-types": {
"version": "2.0.7",
"resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz",
"integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==",
"license": "MIT",
"optional": true
},
"node_modules/@types/use-sync-external-store": {
"version": "0.0.6",
"resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.6.tgz",
@ -4891,6 +5211,16 @@
"integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
"license": "MIT"
},
"node_modules/base64-arraybuffer": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz",
"integrity": "sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ==",
"license": "MIT",
"optional": true,
"engines": {
"node": ">= 0.6.0"
}
},
"node_modules/base64-js": {
"version": "1.5.1",
"resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
@ -5268,6 +5598,26 @@
],
"license": "CC-BY-4.0"
},
"node_modules/canvg": {
"version": "3.0.11",
"resolved": "https://registry.npmjs.org/canvg/-/canvg-3.0.11.tgz",
"integrity": "sha512-5ON+q7jCTgMp9cjpu4Jo6XbvfYwSB2Ow3kzHKfIyJfaCAOHLbdKPQqGKgfED/R5B+3TFFfe8pegYA+b423SRyA==",
"license": "MIT",
"optional": true,
"dependencies": {
"@babel/runtime": "^7.12.5",
"@types/raf": "^3.4.0",
"core-js": "^3.8.3",
"raf": "^3.4.1",
"regenerator-runtime": "^0.13.7",
"rgbcolor": "^1.0.1",
"stackblur-canvas": "^2.0.0",
"svg-pathdata": "^6.0.3"
},
"engines": {
"node": ">=10.0.0"
}
},
"node_modules/chalk": {
"version": "4.1.2",
"resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
@ -5569,6 +5919,18 @@
"node": ">=18"
}
},
"node_modules/core-js": {
"version": "3.49.0",
"resolved": "https://registry.npmjs.org/core-js/-/core-js-3.49.0.tgz",
"integrity": "sha512-es1U2+YTtzpwkxVLwAFdSpaIMyQaq0PBgm3YD1W3Qpsn1NAmO3KSgZfu+oGSWVu6NvLHoHCV/aYcsE5wiB7ALg==",
"hasInstallScript": true,
"license": "MIT",
"optional": true,
"funding": {
"type": "opencollective",
"url": "https://opencollective.com/core-js"
}
},
"node_modules/cross-spawn": {
"version": "7.0.6",
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
@ -5583,6 +5945,16 @@
"node": ">= 8"
}
},
"node_modules/css-line-break": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/css-line-break/-/css-line-break-2.1.0.tgz",
"integrity": "sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w==",
"license": "MIT",
"optional": true,
"dependencies": {
"utrie": "^1.0.2"
}
},
"node_modules/css.escape": {
"version": "1.5.1",
"resolved": "https://registry.npmjs.org/css.escape/-/css.escape-1.5.1.tgz",
@ -6092,6 +6464,16 @@
"url": "https://github.com/fb55/domhandler?sponsor=1"
}
},
"node_modules/dompurify": {
"version": "3.3.3",
"resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.3.3.tgz",
"integrity": "sha512-Oj6pzI2+RqBfFG+qOaOLbFXLQ90ARpcGG6UePL82bJLtdsa6CYJD7nmiU8MW9nQNOtCHV3lZ/Bzq1X0QYbBZCA==",
"license": "(MPL-2.0 OR Apache-2.0)",
"optional": true,
"optionalDependencies": {
"@types/trusted-types": "^2.0.7"
}
},
"node_modules/domutils": {
"version": "3.2.2",
"resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz",
@ -7135,6 +7517,17 @@
"integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
"license": "MIT"
},
"node_modules/fast-png": {
"version": "6.4.0",
"resolved": "https://registry.npmjs.org/fast-png/-/fast-png-6.4.0.tgz",
"integrity": "sha512-kAqZq1TlgBjZcLr5mcN6NP5Rv4V2f22z00c3g8vRrwkcqjerx7BEhPbOnWCPqaHUl2XWQBJQvOT/FQhdMT7X/Q==",
"license": "MIT",
"dependencies": {
"@types/pako": "^2.0.3",
"iobuffer": "^5.3.2",
"pako": "^2.1.0"
}
},
"node_modules/fast-safe-stringify": {
"version": "2.1.1",
"resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz",
@ -7166,6 +7559,12 @@
"bser": "2.1.1"
}
},
"node_modules/fflate": {
"version": "0.8.2",
"resolved": "https://registry.npmjs.org/fflate/-/fflate-0.8.2.tgz",
"integrity": "sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A==",
"license": "MIT"
},
"node_modules/file-entry-cache": {
"version": "8.0.0",
"resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz",
@ -7907,6 +8306,20 @@
"node": ">=14"
}
},
"node_modules/html2canvas": {
"version": "1.4.1",
"resolved": "https://registry.npmjs.org/html2canvas/-/html2canvas-1.4.1.tgz",
"integrity": "sha512-fPU6BHNpsyIhr8yyMpTLLxAbkaK8ArIBcmZIRiBLiDhjeqvXolaEmDGmELFuX9I4xDcaKKcJl+TKZLqruBbmWA==",
"license": "MIT",
"optional": true,
"dependencies": {
"css-line-break": "^2.1.0",
"text-segmentation": "^1.0.3"
},
"engines": {
"node": ">=8.0.0"
}
},
"node_modules/htmlparser2": {
"version": "8.0.2",
"resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.2.tgz",
@ -8156,6 +8569,12 @@
"node": ">=12"
}
},
"node_modules/iobuffer": {
"version": "5.4.0",
"resolved": "https://registry.npmjs.org/iobuffer/-/iobuffer-5.4.0.tgz",
"integrity": "sha512-DRebOWuqDvxunfkNJAlc3IzWIPD5xVxwUNbHr7xKB8E6aLJxIPfNX3CoMJghcFjpv6RWQsrcJbghtEwSPoJqMA==",
"license": "MIT"
},
"node_modules/ip-address": {
"version": "10.1.0",
"resolved": "https://registry.npmjs.org/ip-address/-/ip-address-10.1.0.tgz",
@ -9897,6 +10316,33 @@
"node": ">=6"
}
},
"node_modules/jspdf": {
"version": "4.2.1",
"resolved": "https://registry.npmjs.org/jspdf/-/jspdf-4.2.1.tgz",
"integrity": "sha512-YyAXyvnmjTbR4bHQRLzex3CuINCDlQnBqoSYyjJwTP2x9jDLuKDzy7aKUl0hgx3uhcl7xzg32agn5vlie6HIlQ==",
"license": "MIT",
"peer": true,
"dependencies": {
"@babel/runtime": "^7.28.6",
"fast-png": "^6.2.0",
"fflate": "^0.8.1"
},
"optionalDependencies": {
"canvg": "^3.0.11",
"core-js": "^3.6.0",
"dompurify": "^3.3.1",
"html2canvas": "^1.0.0-rc.5"
}
},
"node_modules/jspdf-autotable": {
"version": "5.0.7",
"resolved": "https://registry.npmjs.org/jspdf-autotable/-/jspdf-autotable-5.0.7.tgz",
"integrity": "sha512-2wr7H6liNDBYNwt25hMQwXkEWFOEopgKIvR1Eukuw6Zmprm/ZcnmLTQEjW7Xx3FCbD3v7pflLcnMAv/h1jFDQw==",
"license": "MIT",
"peerDependencies": {
"jspdf": "^2 || ^3 || ^4"
}
},
"node_modules/jsx-ast-utils": {
"version": "3.3.5",
"resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz",
@ -11119,6 +11565,12 @@
"dev": true,
"license": "BlueOak-1.0.0"
},
"node_modules/pako": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/pako/-/pako-2.1.0.tgz",
"integrity": "sha512-w+eufiZ1WuJYgPXbV/PO3NCMEc3xqylkKHzp8bxp1uW4qaSNQUkwmLLEc3kKsfz8lpV1F8Ht3U1Cm+9Srog2ug==",
"license": "(MIT AND Zlib)"
},
"node_modules/parent-module": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
@ -11243,6 +11695,13 @@
"url": "https://ko-fi.com/killymxi"
}
},
"node_modules/performance-now": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
"integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==",
"license": "MIT",
"optional": true
},
"node_modules/picocolors": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
@ -11810,6 +12269,16 @@
"integrity": "sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg==",
"license": "MIT"
},
"node_modules/raf": {
"version": "3.4.1",
"resolved": "https://registry.npmjs.org/raf/-/raf-3.4.1.tgz",
"integrity": "sha512-Sq4CW4QhwOHE8ucn6J34MqtZCeWFP2aQSmrlroYgqAV1PjStIhJXxYuTgUIfkEk7zTLjmIjLmU5q+fbD1NnOJA==",
"license": "MIT",
"optional": true,
"dependencies": {
"performance-now": "^2.1.0"
}
},
"node_modules/rc": {
"version": "1.2.8",
"resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
@ -12102,6 +12571,13 @@
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/regenerator-runtime": {
"version": "0.13.11",
"resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz",
"integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==",
"license": "MIT",
"optional": true
},
"node_modules/regexp.prototype.flags": {
"version": "1.5.4",
"resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz",
@ -12242,6 +12718,16 @@
"node": ">=0.10.0"
}
},
"node_modules/rgbcolor": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/rgbcolor/-/rgbcolor-1.0.1.tgz",
"integrity": "sha512-9aZLIrhRaD97sgVhtJOW6ckOEh6/GnvQtdVNfdZ6s67+3/XwLS9lBcQYzEEhYVeUowN7pRzMLsyGhK2i/xvWbw==",
"license": "MIT OR SEE LICENSE IN FEEL-FREE.md",
"optional": true,
"engines": {
"node": ">= 0.8.15"
}
},
"node_modules/rimraf": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
@ -12943,6 +13429,16 @@
"node": ">=8"
}
},
"node_modules/stackblur-canvas": {
"version": "2.7.0",
"resolved": "https://registry.npmjs.org/stackblur-canvas/-/stackblur-canvas-2.7.0.tgz",
"integrity": "sha512-yf7OENo23AGJhBriGx0QivY5JP6Y1HbrrDI6WLt6C5auYZXlQrheoY8hD4ibekFKz1HOfE48Ww8kMWMnJD/zcQ==",
"license": "MIT",
"optional": true,
"engines": {
"node": ">=0.1.14"
}
},
"node_modules/standardwebhooks": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/standardwebhooks/-/standardwebhooks-1.0.0.tgz",
@ -13333,6 +13829,16 @@
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/svg-pathdata": {
"version": "6.0.3",
"resolved": "https://registry.npmjs.org/svg-pathdata/-/svg-pathdata-6.0.3.tgz",
"integrity": "sha512-qsjeeq5YjBZ5eMdFuUa4ZosMLxgr5RZ+F+Y1OrDhuOCEInRMA3x74XdBtggJcj9kOeInz0WE+LgCPDkZFlBYJw==",
"license": "MIT",
"optional": true,
"engines": {
"node": ">=12.0.0"
}
},
"node_modules/svix": {
"version": "1.84.1",
"resolved": "https://registry.npmjs.org/svix/-/svix-1.84.1.tgz",
@ -13566,6 +14072,16 @@
"url": "https://github.com/sponsors/isaacs"
}
},
"node_modules/text-segmentation": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/text-segmentation/-/text-segmentation-1.0.3.tgz",
"integrity": "sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw==",
"license": "MIT",
"optional": true,
"dependencies": {
"utrie": "^1.0.2"
}
},
"node_modules/thenify": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
@ -14212,6 +14728,16 @@
"integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
"license": "MIT"
},
"node_modules/utrie": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/utrie/-/utrie-1.0.2.tgz",
"integrity": "sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw==",
"license": "MIT",
"optional": true,
"dependencies": {
"base64-arraybuffer": "^1.0.2"
}
},
"node_modules/uuid": {
"version": "10.0.0",
"resolved": "https://registry.npmjs.org/uuid/-/uuid-10.0.0.tgz",

View File

@ -19,6 +19,7 @@
"@fitai/shared": "file:../../packages/shared",
"@hookform/resolvers": "^5.2.2",
"@radix-ui/react-dialog": "^1.1.15",
"@radix-ui/react-select": "^2.2.6",
"@radix-ui/react-slot": "^1.2.4",
"@react-email/components": "^1.0.8",
"@react-email/render": "^2.0.4",
@ -38,6 +39,8 @@
"date-fns": "^4.1.0",
"dotenv": "^17.2.3",
"drizzle-orm": "^0.44.7",
"jspdf": "^4.2.1",
"jspdf-autotable": "^5.0.7",
"lucide-react": "^0.553.0",
"next": "^16.0.1",
"pino": "^10.3.1",

View File

@ -0,0 +1,168 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import { successResponse } from "@/lib/api/responses";
import { db as rawDb, sql } from "@fitai/database";
import { getUsersByGym, getClientsByGym } from "@/lib/gym-context";
import log from "@/lib/logger";
interface UserGrowthPoint {
label: string;
value: number;
}
interface MembershipDistributionPoint {
label: string;
value: number;
color: string;
}
interface RevenuePoint {
label: string;
value: number;
color: string;
}
interface AnalyticsData {
userGrowth: UserGrowthPoint[];
membershipDistribution: MembershipDistributionPoint[];
revenue: RevenuePoint[];
}
export async function GET(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const database = await getDatabase();
const user = await ensureUserSynced(userId, database);
if (!user || (user.role !== "admin" && user.role !== "superAdmin")) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const url = new URL(req.url);
const months = parseInt(url.searchParams.get("months") || "6");
// Get target gym based on role
const targetGymId =
user.role === "superAdmin"
? (url.searchParams.get("gymId") ?? undefined)
: (user.gymId ?? undefined);
// Validate gym access for non-superAdmins
if (user.role !== "superAdmin" && !targetGymId) {
return NextResponse.json(
{ error: "Forbidden - No gym assigned" },
{ status: 403 },
);
}
// Get gym-scoped data
const allUsers = targetGymId
? await getUsersByGym(targetGymId)
: await database.getAllUsers();
const allClients = targetGymId
? await getClientsByGym(targetGymId)
: await database.getAllClients();
// For payments, we'd need a similar filter - for now, skip payments if gym-scoped
// TODO: Add getPaymentsByGym when needed
const paymentsRaw = targetGymId
? [] // Skip payments for gym-scoped for now
: await rawDb.all(
sql`SELECT * FROM payments WHERE status = 'completed' AND paid_at IS NOT NULL`,
);
const payments: any[] = paymentsRaw || [];
const now = new Date();
const userGrowth: UserGrowthPoint[] = [];
for (let i = months - 1; i >= 0; i--) {
const date = new Date(now.getFullYear(), now.getMonth() - i, 1);
const monthLabel = date.toLocaleDateString("en-US", { month: "short" });
const usersCreatedByMonth = allUsers.filter((u: any) => {
const createdAt = new Date(u.createdAt);
return (
createdAt.getFullYear() === date.getFullYear() &&
createdAt.getMonth() === date.getMonth()
);
});
userGrowth.push({
label: monthLabel,
value: usersCreatedByMonth.length,
});
}
let runningTotal = 0;
for (const point of userGrowth) {
runningTotal += point.value;
point.value = runningTotal;
}
const membershipCounts: Record<string, number> = {
basic: 0,
premium: 0,
vip: 0,
};
for (const client of allClients) {
const membershipType = (client as any).membershipType || "basic";
if (membershipCounts[membershipType] !== undefined) {
membershipCounts[membershipType]++;
}
}
const membershipDistribution: MembershipDistributionPoint[] = [
{ label: "Basic", value: membershipCounts.basic, color: "#6b7280" },
{ label: "Premium", value: membershipCounts.premium, color: "#3b82f6" },
{ label: "VIP", value: membershipCounts.vip, color: "#f59e0b" },
];
const revenue: RevenuePoint[] = [];
for (let i = months - 1; i >= 0; i--) {
const date = new Date(now.getFullYear(), now.getMonth() - i, 1);
const monthLabel = date.toLocaleDateString("en-US", { month: "short" });
const nextMonthDate = new Date(
now.getFullYear(),
now.getMonth() - i + 1,
1,
);
const monthlyRevenue = payments
.filter((p: any) => {
const paidAt = p.paid_at ? new Date(p.paid_at) : null;
if (!paidAt) return false;
return paidAt >= date && paidAt < nextMonthDate;
})
.reduce((sum: number, p: any) => sum + (p.amount || 0), 0);
revenue.push({
label: monthLabel,
value: monthlyRevenue,
color: "#10b981",
});
}
const analyticsData: AnalyticsData = {
userGrowth,
membershipDistribution,
revenue,
};
return successResponse({ analytics: analyticsData });
} catch (error) {
log.error("Analytics error", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -1,27 +1,68 @@
import { auth } from '@clerk/nextjs/server'
import { NextResponse } from 'next/server'
import { getDatabase } from '@/lib/database'
import { ensureUserSynced } from '@/lib/sync-user'
import { auth } from "@clerk/nextjs/server";
import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import { successResponse } from "@/lib/api/responses";
import { getAttendanceByGym } from "@/lib/gym-context";
export async function GET(req: Request) {
try {
const { userId } = await auth()
if (!userId) return new NextResponse('Unauthorized', { status: 401 })
export async function GET(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase()
const db = await getDatabase();
const user = await ensureUserSynced(userId, db);
// Ensure user is synced (handles seed script ID mismatch)
// We need to import ensureUserSynced
const user = await ensureUserSynced(userId, db)
if (!user || (user.role !== 'admin' && user.role !== 'superAdmin')) {
return new NextResponse('Forbidden', { status: 403 })
}
const attendance = await db.getAllAttendance()
return NextResponse.json(attendance)
} catch (error) {
console.error('Admin attendance error:', error)
return new NextResponse('Internal Server Error', { status: 500 })
if (!user || (user.role !== "admin" && user.role !== "superAdmin")) {
return new NextResponse("Forbidden", { status: 403 });
}
// Get target gym based on role
const url = new URL(req.url);
const targetGymId =
user.role === "superAdmin"
? (url.searchParams.get("gymId") ?? undefined)
: (user.gymId ?? undefined);
// Validate gym access for non-superAdmins
if (user.role !== "superAdmin" && !targetGymId) {
return new NextResponse("Forbidden - No gym assigned", { status: 403 });
}
// Get attendance filtered by gym
const attendance = targetGymId
? await getAttendanceByGym(targetGymId)
: await db.getAllAttendance();
// Get all users to enrich attendance with user names
const allUsers = await db.getAllUsers();
const userMap = new Map(
allUsers.map((u) => [
u.id,
{
firstName: u.firstName,
lastName: u.lastName,
email: u.email,
},
]),
);
// Enrich attendance records with user information
const enrichedAttendance = attendance.map((record) => {
const userInfo = userMap.get(record.userId);
return {
...record,
userName: userInfo
? `${userInfo.firstName} ${userInfo.lastName}`.trim() ||
userInfo.email
: record.userId,
userEmail: userInfo?.email,
};
});
return successResponse({ records: enrichedAttendance });
} catch (error) {
console.error("Admin attendance error:", error);
return new NextResponse("Internal Server Error", { status: 500 });
}
}

View File

@ -0,0 +1,103 @@
/**
* @jest-environment node
*/
import { NextRequest } from "next/server";
import { POST } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
}));
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@/lib/clerk-helpers", () => ({
setUserRole: jest.fn(),
}));
describe("POST /api/admin/set-role", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database").getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
const mockSetUserRole = require("@/lib/clerk-helpers")
.setUserRole as jest.Mock;
const mockDb = {
getUserById: jest.fn(),
};
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue(mockDb);
});
it("returns 403 when admin tries to assign role across gyms", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
mockDb.getUserById.mockResolvedValue({
id: "user_2",
role: "client",
gymId: "gym_b",
});
const request = new NextRequest("http://localhost/api/admin/set-role", {
method: "POST",
body: JSON.stringify({
targetUserId: "user_2",
role: "trainer",
}),
});
const response = await POST(request);
expect(response.status).toBe(403);
expect(mockSetUserRole).not.toHaveBeenCalled();
});
it("allows superAdmin to assign roles across gyms", async () => {
mockAuth.mockResolvedValue({ userId: "super_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "super_1",
role: "superAdmin",
gymId: null,
});
mockDb.getUserById.mockResolvedValue({
id: "user_2",
role: "client",
gymId: "gym_b",
});
mockSetUserRole.mockResolvedValue({
id: "user_2",
emailAddresses: [{ emailAddress: "user2@example.com" }],
firstName: "User",
lastName: "Two",
publicMetadata: { role: "admin" },
});
const request = new NextRequest("http://localhost/api/admin/set-role", {
method: "POST",
body: JSON.stringify({
targetUserId: "user_2",
role: "admin",
}),
});
const response = await POST(request);
const payload = await response.json();
expect(response.status).toBe(200);
expect(payload.success).toBe(true);
expect(mockSetUserRole).toHaveBeenCalledWith("user_2", "admin");
});
});

View File

@ -1,6 +1,9 @@
import { auth } from '@clerk/nextjs/server';
import { NextResponse } from 'next/server';
import { setUserRole, isAdmin, type UserRole } from '@/lib/clerk-helpers';
import { auth } from "@clerk/nextjs/server";
import { NextResponse } from "next/server";
import { USER_ROLES, type UserRole } from "@fitai/shared";
import { setUserRole } from "@/lib/clerk-helpers";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
export async function POST(req: Request) {
try {
@ -8,16 +11,27 @@ export async function POST(req: Request) {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(userId, db);
if (!currentUser) {
return NextResponse.json(
{ error: "Forbidden: user not found" },
{ status: 403 },
);
}
// Check if the requesting user is an admin
const requestingUserIsAdmin = await isAdmin(userId);
const requestingUserIsAdmin =
currentUser.role === "admin" || currentUser.role === "superAdmin";
if (!requestingUserIsAdmin) {
return NextResponse.json(
{ error: 'Forbidden: Admin access required' },
{ status: 403 }
{ error: "Forbidden: Admin access required" },
{ status: 403 },
);
}
@ -26,25 +40,57 @@ export async function POST(req: Request) {
const { targetUserId, role } = body;
// Validate inputs
if (!targetUserId || typeof targetUserId !== 'string') {
if (!targetUserId || typeof targetUserId !== "string") {
return NextResponse.json(
{ error: 'Invalid or missing targetUserId' },
{ status: 400 }
{ error: "Invalid or missing targetUserId" },
{ status: 400 },
);
}
if (!role || !['admin', 'trainer', 'client'].includes(role)) {
if (!role || !USER_ROLES.includes(role as UserRole)) {
return NextResponse.json(
{ error: 'Invalid role. Must be admin, trainer, or client' },
{ status: 400 }
{
error: `Invalid role. Must be one of: ${USER_ROLES.join(", ")}`,
},
{ status: 400 },
);
}
const allowedRolesByRequester: Record<UserRole, UserRole[]> = {
superAdmin: ["superAdmin", "admin", "trainer", "client"],
admin: ["admin", "trainer", "client"],
trainer: [],
client: [],
};
const allowedTargetRoles = allowedRolesByRequester[currentUser.role];
if (!allowedTargetRoles.includes(role as UserRole)) {
return NextResponse.json(
{ error: `Forbidden: cannot assign role '${role}'` },
{ status: 403 },
);
}
// Prevent admin from changing their own role
if (userId === targetUserId) {
return NextResponse.json(
{ error: 'Cannot change your own role' },
{ status: 400 }
{ error: "Cannot change your own role" },
{ status: 400 },
);
}
const targetUser = await db.getUserById(targetUserId);
if (!targetUser) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
if (
currentUser.role !== "superAdmin" &&
(!currentUser.gymId || targetUser.gymId !== currentUser.gymId)
) {
return NextResponse.json(
{ error: "Cannot change roles for users from other gyms" },
{ status: 403 },
);
}
@ -63,15 +109,15 @@ export async function POST(req: Request) {
},
});
} catch (error) {
console.error('Error setting user role:', error);
console.error("Error setting user role:", error);
if (error instanceof Error && error.message.includes('not found')) {
return NextResponse.json({ error: 'User not found' }, { status: 404 });
if (error instanceof Error && error.message.includes("not found")) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
return NextResponse.json(
{ error: 'Internal server error' },
{ status: 500 }
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -2,20 +2,27 @@ import { auth } from "@clerk/nextjs/server";
import { NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import { successResponse } from "@/lib/api/responses";
import log from "@/lib/logger";
export async function GET(req: Request) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const user = await ensureUserSynced(userId, db);
if (!user || (user.role !== "admin" && user.role !== "superAdmin")) {
return new NextResponse("Forbidden", { status: 403 });
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
if (user.role === "admin" && !user.gymId) {
return new NextResponse("Admin gymId not set", { status: 400 });
return NextResponse.json(
{ error: "Admin gymId not set" },
{ status: 400 },
);
}
const url = new URL(req.url);
@ -51,9 +58,12 @@ export async function GET(req: Request) {
revenueGrowth: 0,
};
return NextResponse.json(stats);
return successResponse({ stats });
} catch (error) {
console.error("Dashboard stats error:", error);
return new NextResponse("Internal Server Error", { status: 500 });
log.error("Dashboard stats error", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -1,137 +1,180 @@
/**
* @jest-environment node
*/
import { POST as checkIn } from '../check-in/route'
import { POST as checkOut } from '../check-out/route'
import { GET as history } from '../history/route'
import { NextRequest } from 'next/server'
import { POST as checkIn } from "../check-in/route";
import { POST as checkOut } from "../check-out/route";
import { GET as history } from "../history/route";
import { NextRequest } from "next/server";
// Mock dependencies
jest.mock('@clerk/nextjs/server', () => ({
auth: jest.fn(() => Promise.resolve({ userId: 'test_user_id' })),
currentUser: jest.fn(() => Promise.resolve({ id: 'test_user_id', emailAddresses: [{ emailAddress: 'test@example.com' }] }))
}))
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(() => Promise.resolve({ userId: "test_user_id" })),
currentUser: jest.fn(() =>
Promise.resolve({
id: "test_user_id",
emailAddresses: [{ emailAddress: "test@example.com" }],
}),
),
}));
jest.mock('@/lib/sync-user', () => ({
ensureUserSynced: jest.fn()
}))
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@/lib/geofence", () => ({
getUserGymGeofence: jest.fn(() =>
Promise.resolve({
id: "gym_1",
name: "Test Gym",
latitude: 1,
longitude: 1,
geofenceRadiusMeters: 30,
geofenceEnabled: true,
}),
),
parseUserLocation: jest.fn(() => ({
latitude: 1,
longitude: 1,
accuracy: 10,
})),
validateGeofence: jest.fn(() => ({ ok: true })),
validateGeofenceWithFallback: jest.fn(() => ({ ok: true })),
validateCheckInGeofence: jest.fn(() => ({ ok: true })),
}));
const mockDb = {
checkIn: jest.fn(),
checkOut: jest.fn(),
getAttendanceHistory: jest.fn(),
getActiveCheckIn: jest.fn(),
getUserById: jest.fn(),
createUser: jest.fn(),
getClientByUserId: jest.fn(),
createClient: jest.fn(),
getFitnessProfileByUserId: jest.fn(),
createFitnessProfile: jest.fn(),
}
checkIn: jest.fn(),
checkOut: jest.fn(),
getAttendanceHistory: jest.fn(),
getActiveCheckIn: jest.fn(),
getUserById: jest.fn(),
createUser: jest.fn(),
getClientByUserId: jest.fn(),
createClient: jest.fn(),
getFitnessProfileByUserId: jest.fn(),
createFitnessProfile: jest.fn(),
};
jest.mock('@/lib/database', () => ({
getDatabase: jest.fn(() => Promise.resolve(mockDb))
}))
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(() => Promise.resolve(mockDb)),
}));
describe('Attendance API', () => {
beforeEach(() => {
jest.clearAllMocks()
})
describe("Attendance API", () => {
beforeEach(() => {
jest.clearAllMocks();
});
describe('POST /api/attendance/check-in', () => {
it('should successfully check in', async () => {
mockDb.getUserById.mockResolvedValue({ id: 'test_user_id' })
mockDb.getActiveCheckIn.mockResolvedValue(null)
mockDb.checkIn.mockResolvedValue({
id: 'attendance_id',
userId: 'test_user_id',
checkInTime: new Date(),
type: 'gym'
})
describe("POST /api/attendance/check-in", () => {
it("should successfully check in", async () => {
mockDb.getUserById.mockResolvedValue({ id: "test_user_id" });
mockDb.getActiveCheckIn.mockResolvedValue(null);
mockDb.checkIn.mockResolvedValue({
id: "attendance_id",
userId: "test_user_id",
checkInTime: new Date(),
type: "gym",
});
const req = new NextRequest('http://localhost/api/attendance/check-in', {
method: 'POST',
body: JSON.stringify({ type: 'gym', notes: 'Test check-in' })
})
const req = new NextRequest("http://localhost/api/attendance/check-in", {
method: "POST",
body: JSON.stringify({
type: "gym",
notes: "Test check-in",
location: { latitude: 1, longitude: 1, accuracy: 10 },
}),
});
const res = await checkIn(req)
const data = await res.json()
const res = await checkIn(req);
const data = await res.json();
expect(res.status).toBe(200)
expect(data.id).toBe('attendance_id')
expect(data.userId).toBe('test_user_id')
expect(mockDb.checkIn).toHaveBeenCalledWith('test_user_id', 'gym', 'Test check-in')
})
expect(res.status).toBe(200);
expect(data.id).toBe("attendance_id");
expect(data.userId).toBe("test_user_id");
expect(mockDb.checkIn).toHaveBeenCalledWith(
"test_user_id",
"gym",
"Test check-in",
);
});
it('should fail if already checked in', async () => {
mockDb.getUserById.mockResolvedValue({ id: 'test_user_id' })
mockDb.getActiveCheckIn.mockResolvedValue({ id: 'existing_id' })
it("should fail if already checked in", async () => {
mockDb.getUserById.mockResolvedValue({ id: "test_user_id" });
mockDb.getActiveCheckIn.mockResolvedValue({ id: "existing_id" });
const req = new NextRequest('http://localhost/api/attendance/check-in', {
method: 'POST',
body: JSON.stringify({ type: 'gym' })
})
const req = new NextRequest("http://localhost/api/attendance/check-in", {
method: "POST",
body: JSON.stringify({
type: "gym",
location: { latitude: 1, longitude: 1, accuracy: 10 },
}),
});
const res = await checkIn(req)
const text = await res.text()
const res = await checkIn(req);
const text = await res.text();
expect(res.status).toBe(400)
expect(text).toBe('Already checked in')
})
})
expect(res.status).toBe(400);
expect(text).toBe("Already checked in");
});
});
describe('POST /api/attendance/check-out', () => {
it('should successfully check out', async () => {
mockDb.getActiveCheckIn.mockResolvedValue({ id: 'attendance_id' })
mockDb.checkOut.mockResolvedValue({
id: 'attendance_id',
checkOutTime: new Date()
})
describe("POST /api/attendance/check-out", () => {
it("should successfully check out", async () => {
mockDb.getActiveCheckIn.mockResolvedValue({ id: "attendance_id" });
mockDb.checkOut.mockResolvedValue({
id: "attendance_id",
checkOutTime: new Date(),
});
const req = new NextRequest('http://localhost/api/attendance/check-out', {
method: 'POST'
})
const req = new NextRequest("http://localhost/api/attendance/check-out", {
method: "POST",
body: JSON.stringify({
location: { latitude: 1, longitude: 1, accuracy: 10 },
}),
});
const res = await checkOut(req)
const data = await res.json()
const res = await checkOut(req);
const data = await res.json();
expect(res.status).toBe(200)
expect(data.id).toBe('attendance_id')
expect(data.checkOutTime).toBeDefined()
expect(mockDb.checkOut).toHaveBeenCalledWith('attendance_id')
})
expect(res.status).toBe(200);
expect(data.id).toBe("attendance_id");
expect(data.checkOutTime).toBeDefined();
expect(mockDb.checkOut).toHaveBeenCalledWith("attendance_id");
});
it('should fail if not checked in', async () => {
mockDb.getActiveCheckIn.mockResolvedValue(null)
it("should fail if not checked in", async () => {
mockDb.getActiveCheckIn.mockResolvedValue(null);
const req = new NextRequest('http://localhost/api/attendance/check-out', {
method: 'POST'
})
const req = new NextRequest("http://localhost/api/attendance/check-out", {
method: "POST",
body: JSON.stringify({
location: { latitude: 1, longitude: 1, accuracy: 10 },
}),
});
const res = await checkOut(req)
const text = await res.text()
const res = await checkOut(req);
const text = await res.text();
expect(res.status).toBe(404)
expect(text).toBe('No active check-in found')
})
})
expect(res.status).toBe(404);
expect(text).toBe("No active check-in found");
});
});
describe('GET /api/attendance/history', () => {
it('should return attendance history', async () => {
const historyData = [
{ id: '1', checkInTime: new Date() },
{ id: '2', checkInTime: new Date() }
]
mockDb.getUserById.mockResolvedValue({ id: 'test_user_id' })
mockDb.getAttendanceHistory.mockResolvedValue(historyData)
describe("GET /api/attendance/history", () => {
it("should return attendance history", async () => {
const historyData = [
{ id: "1", checkInTime: new Date() },
{ id: "2", checkInTime: new Date() },
];
mockDb.getUserById.mockResolvedValue({ id: "test_user_id" });
mockDb.getAttendanceHistory.mockResolvedValue(historyData);
const req = new NextRequest('http://localhost/api/attendance/history')
const res = await history(req)
const data = await res.json()
const req = new NextRequest("http://localhost/api/attendance/history");
const res = await history(req);
const data = await res.json();
expect(res.status).toBe(200)
expect(data).toEqual(JSON.parse(JSON.stringify(historyData))) // Handle date serialization
expect(mockDb.getAttendanceHistory).toHaveBeenCalledWith('test_user_id')
})
})
})
expect(res.status).toBe(200);
expect(data).toEqual(JSON.parse(JSON.stringify(historyData))); // Handle date serialization
expect(mockDb.getAttendanceHistory).toHaveBeenCalledWith("test_user_id");
});
});
});

View File

@ -2,12 +2,12 @@ import { auth } from "@clerk/nextjs/server";
import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
import { checkInSchema } from "@/lib/validation/schemas";
import {
validateRequestBody,
validationErrorResponse,
} from "@/lib/validation/helpers";
getUserGymGeofence,
parseUserLocation,
validateCheckInGeofence,
} from "@/lib/geofence";
import log from "@/lib/logger";
export async function POST(req: NextRequest) {
try {
@ -25,8 +25,26 @@ export async function POST(req: NextRequest) {
return new NextResponse("Already checked in", { status: 400 });
}
const body = await req.json();
const body = await req.json().catch(() => ({}));
const { type = "gym", notes } = body;
const fallbackRequested = Boolean(body.fallbackRequested);
const gym = await getUserGymGeofence(userId);
if (!gym) {
return NextResponse.json(
{ error: "No gym assigned for this user" },
{ status: 400 },
);
}
const location = parseUserLocation(body.location);
const geofence = validateCheckInGeofence(gym, location, fallbackRequested);
if (!geofence.ok) {
return NextResponse.json(
{ error: geofence.error },
{ status: geofence.status },
);
}
const attendance = await db.checkIn(userId, type, notes);
return NextResponse.json(attendance);

View File

@ -1,6 +1,11 @@
import { auth } from "@clerk/nextjs/server";
import { NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import {
getUserGymGeofence,
parseUserLocation,
validateGeofenceWithFallback,
} from "@/lib/geofence";
import log from "@/lib/logger";
export async function POST(req: Request) {
@ -15,6 +20,30 @@ export async function POST(req: Request) {
return new NextResponse("No active check-in found", { status: 404 });
}
const body = await req.json().catch(() => ({}));
const fallbackRequested = Boolean(body.fallbackRequested);
const gym = await getUserGymGeofence(userId);
if (!gym) {
return NextResponse.json(
{ error: "No gym assigned for this user" },
{ status: 400 },
);
}
const location = parseUserLocation(body.location);
const geofence = validateGeofenceWithFallback(
gym,
location,
fallbackRequested,
);
if (!geofence.ok) {
return NextResponse.json(
{ error: geofence.error },
{ status: geofence.status },
);
}
const attendance = await db.checkOut(activeCheckIn.id);
return NextResponse.json(attendance);
} catch (error) {

View File

@ -1,5 +1,6 @@
import { NextRequest, NextResponse } from "next/server";
import bcrypt from "bcryptjs";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "../../../../lib/database/index";
import log from "@/lib/logger";
import { userSchema } from "@/lib/validation/schemas";
@ -7,6 +8,8 @@ import {
validateRequestBody,
validationErrorResponse,
} from "@/lib/validation/helpers";
import { ensureUserSynced } from "@/lib/sync-user";
import { getUsersByGym } from "@/lib/gym-context";
export async function POST(request: NextRequest) {
try {
@ -68,8 +71,31 @@ export async function POST(request: NextRequest) {
export async function GET() {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const allUsers = await db.getAllUsers();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const canViewUsers =
currentUser.role === "superAdmin" || currentUser.role === "admin";
if (!canViewUsers) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const allUsers =
currentUser.role === "superAdmin"
? await db.getAllUsers()
: currentUser.gymId
? await getUsersByGym(currentUser.gymId)
: [];
const usersWithoutPassword = allUsers.map(
({ password: _, ...user }) => user,
);

View File

@ -0,0 +1,52 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
/**
* GET /api/fitness-profile/history
* Get fitness profile history for a user
*
* Query params:
* - userId: string (required)
* - startDate: YYYY-MM-DD (optional)
* - endDate: YYYY-MM-DD (optional)
*/
export async function GET(request: NextRequest) {
try {
const { userId: authUserId } = await auth();
if (!authUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { searchParams } = new URL(request.url);
const userId = searchParams.get("userId");
const startDateStr = searchParams.get("startDate");
const endDateStr = searchParams.get("endDate");
if (!userId) {
return NextResponse.json(
{ error: "userId is required" },
{ status: 400 },
);
}
// Convert date strings to Date objects if provided
const startDate = startDateStr ? new Date(startDateStr) : undefined;
const endDate = endDateStr ? new Date(endDateStr) : undefined;
const db = await getDatabase();
const history = await db.getFitnessProfileHistory(
userId,
startDate,
endDate,
);
return NextResponse.json({ history });
} catch (error) {
console.error("Get fitness profile history error:", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,156 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getUserMembershipContext } from "@/lib/membership/access";
import log from "@/lib/logger";
interface OpenFoodFactsProduct {
product_name?: string;
product_name_en?: string;
brands?: string;
image_url?: string;
image_front_url?: string;
serving_size?: string;
nutriments?: {
[key: string]: number | string | undefined;
"energy-kcal_serving"?: number;
"energy-kcal_100g"?: number;
proteins_serving?: number;
proteins_100g?: number;
carbohydrates_serving?: number;
carbohydrates_100g?: number;
fat_serving?: number;
fat_100g?: number;
};
}
interface OpenFoodFactsResponse {
status: number;
code: string;
product?: OpenFoodFactsProduct;
}
function normalizeBarcode(rawCode: string): string {
return rawCode.replace(/\D/g, "").trim();
}
function isSupportedBarcode(code: string): boolean {
return [8, 12, 13].includes(code.length);
}
function getNumber(value: unknown): number | undefined {
if (typeof value === "number" && Number.isFinite(value)) return value;
if (typeof value === "string") {
const parsed = Number(value);
if (Number.isFinite(parsed)) return parsed;
}
return undefined;
}
function buildProductPayload(code: string, product: OpenFoodFactsProduct) {
const caloriesPerServing =
getNumber(product.nutriments?.["energy-kcal_serving"]) ??
getNumber(product.nutriments?.["energy-kcal_100g"]) ??
0;
const protein =
getNumber(product.nutriments?.proteins_serving) ??
getNumber(product.nutriments?.proteins_100g);
const carbs =
getNumber(product.nutriments?.carbohydrates_serving) ??
getNumber(product.nutriments?.carbohydrates_100g);
const fat =
getNumber(product.nutriments?.fat_serving) ??
getNumber(product.nutriments?.fat_100g);
return {
barcode: code,
name: product.product_name || product.product_name_en || "Unknown Product",
brand: product.brands || null,
imageUrl: product.image_url || product.image_front_url || null,
servingSize: product.serving_size || "1 serving",
caloriesPerServing: Math.max(0, Math.round(caloriesPerServing)),
macros: {
protein: protein ?? null,
carbs: carbs ?? null,
fat: fat ?? null,
},
source: "openfoodfacts" as const,
};
}
export async function GET(
_request: Request,
{ params }: { params: Promise<{ code: string }> },
) {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Barcode food scan is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const { code: rawCode } = await params;
const code = normalizeBarcode(rawCode);
if (!isSupportedBarcode(code)) {
return NextResponse.json(
{ error: "Invalid barcode. Use EAN-8, UPC-A, or EAN-13 formats." },
{ status: 400 },
);
}
const response = await fetch(
`https://world.openfoodfacts.org/api/v2/product/${code}.json`,
{
headers: {
"User-Agent": "FitAI/1.0 (fitai.app)",
},
cache: "no-store",
},
);
if (!response.ok) {
log.warn("OpenFoodFacts lookup failed", {
status: response.status,
barcode: code,
});
return NextResponse.json(
{ error: "Food lookup service unavailable. Please try again." },
{ status: 503 },
);
}
const payload = (await response.json()) as OpenFoodFactsResponse;
if (payload.status !== 1 || !payload.product) {
return NextResponse.json(
{ error: "Product not found in OpenFoodFacts" },
{ status: 404 },
);
}
return NextResponse.json({
success: true,
data: buildProductPayload(code, payload.product),
meta: {
timestamp: new Date().toISOString(),
},
});
} catch (error) {
log.error("Failed barcode food lookup", error);
return NextResponse.json(
{ error: "Failed to lookup food barcode" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,250 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { eq, sql } from "@fitai/database";
import { db, users as usersTable, gyms as gymsTable } from "@fitai/database";
import { ensureUserSynced } from "@/lib/sync-user";
import { getDatabase } from "@/lib/database";
import log from "@/lib/logger";
async function ensureGymsTable() {
await db.run(sql`
CREATE TABLE IF NOT EXISTS gyms (
id TEXT PRIMARY KEY,
name TEXT NOT NULL,
location TEXT,
status TEXT NOT NULL CHECK (status IN ('active','inactive')) DEFAULT 'active',
admin_user_id TEXT NOT NULL,
created_at INTEGER NOT NULL,
updated_at INTEGER NOT NULL
)
`);
const columns = await db.all(sql`PRAGMA table_info('gyms')`);
const columnNames = new Set(
(columns as Array<{ name?: string }>)
.map((col) => col.name)
.filter(Boolean),
);
if (!columnNames.has("latitude")) {
await db.run(sql`ALTER TABLE gyms ADD COLUMN latitude REAL`);
}
if (!columnNames.has("longitude")) {
await db.run(sql`ALTER TABLE gyms ADD COLUMN longitude REAL`);
}
if (!columnNames.has("geofence_radius_meters")) {
await db.run(
sql`ALTER TABLE gyms ADD COLUMN geofence_radius_meters REAL NOT NULL DEFAULT 30`,
);
}
if (!columnNames.has("geofence_enabled")) {
await db.run(
sql`ALTER TABLE gyms ADD COLUMN geofence_enabled INTEGER NOT NULL DEFAULT 1`,
);
}
}
// PATCH /api/gyms/[id]
// Update gym details and geofence configuration
export async function PATCH(
request: Request,
{ params }: { params: Promise<{ id: string }> },
) {
try {
const { id: gymId } = await params;
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const appDb = await getDatabase();
const currentUser = await ensureUserSynced(userId, appDb);
if (
!currentUser ||
(currentUser.role !== "superAdmin" && currentUser.role !== "admin")
) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
await ensureGymsTable();
const existingGym = await db
.select()
.from(gymsTable)
.where(eq(gymsTable.id, gymId))
.get();
if (!existingGym) {
return NextResponse.json({ error: "Gym not found" }, { status: 404 });
}
if (
currentUser.role === "admin" &&
currentUser.gymId &&
currentUser.gymId !== gymId
) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const body = await request.json().catch(() => null);
if (!body || typeof body !== "object") {
return NextResponse.json({ error: "Invalid body" }, { status: 400 });
}
const latitude =
body.latitude === undefined || body.latitude === null
? null
: Number(body.latitude);
const longitude =
body.longitude === undefined || body.longitude === null
? null
: Number(body.longitude);
const geofenceRadiusMeters =
body.geofenceRadiusMeters === undefined ||
body.geofenceRadiusMeters === null
? 30
: Number(body.geofenceRadiusMeters);
const geofenceEnabled =
body.geofenceEnabled === undefined ? true : Boolean(body.geofenceEnabled);
if (
latitude !== null &&
(!Number.isFinite(latitude) || latitude < -90 || latitude > 90)
) {
return NextResponse.json(
{ error: "latitude must be between -90 and 90" },
{ status: 400 },
);
}
if (
longitude !== null &&
(!Number.isFinite(longitude) || longitude < -180 || longitude > 180)
) {
return NextResponse.json(
{ error: "longitude must be between -180 and 180" },
{ status: 400 },
);
}
if (!Number.isFinite(geofenceRadiusMeters) || geofenceRadiusMeters <= 0) {
return NextResponse.json(
{ error: "geofenceRadiusMeters must be a positive number" },
{ status: 400 },
);
}
await db.run(sql`
UPDATE gyms
SET
latitude = ${latitude},
longitude = ${longitude},
geofence_radius_meters = ${geofenceRadiusMeters},
geofence_enabled = ${geofenceEnabled ? 1 : 0},
updated_at = ${Math.floor(Date.now() / 1000)}
WHERE id = ${gymId}
`);
const updatedRows = await db.all(sql`
SELECT
id,
name,
location,
latitude,
longitude,
geofence_radius_meters as geofenceRadiusMeters,
geofence_enabled as geofenceEnabled,
status,
admin_user_id as adminUserId,
created_at as createdAt,
updated_at as updatedAt
FROM gyms
WHERE id = ${gymId}
LIMIT 1
`);
const updated = updatedRows?.[0]
? {
...updatedRows[0],
geofenceEnabled:
typeof (updatedRows[0] as { geofenceEnabled?: unknown })
.geofenceEnabled === "boolean"
? (updatedRows[0] as { geofenceEnabled: boolean }).geofenceEnabled
: Boolean(
(updatedRows[0] as { geofenceEnabled?: unknown })
.geofenceEnabled,
),
}
: null;
return NextResponse.json(updated);
} catch (error) {
log.error("Failed to update gym", error);
return NextResponse.json(
{ error: "Internal Server Error" },
{ status: 500 },
);
}
}
// DELETE /api/gyms/[id]
// Delete a gym (soft delete - mark as inactive)
export async function DELETE(
request: Request,
{ params }: { params: Promise<{ id: string }> },
) {
try {
const { id: gymId } = await params;
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const appDb = await getDatabase();
const currentUser = await ensureUserSynced(userId, appDb);
// Only superAdmin can delete gyms
if (!currentUser || currentUser.role !== "superAdmin") {
return NextResponse.json(
{ error: "Forbidden - Only superAdmin can delete gyms" },
{ status: 403 },
);
}
await ensureGymsTable();
// Check if gym exists using Drizzle ORM
const existingGym = await db
.select()
.from(gymsTable)
.where(eq(gymsTable.id, gymId))
.get();
if (!existingGym) {
return NextResponse.json({ error: "Gym not found" }, { status: 404 });
}
// Soft delete - mark as inactive using Drizzle ORM
await db
.update(gymsTable)
.set({ status: "inactive", updatedAt: new Date() })
.where(eq(gymsTable.id, gymId));
return NextResponse.json({
success: true,
message: "Gym deleted successfully",
});
} catch (error) {
log.error("Failed to delete gym", error);
return NextResponse.json(
{ error: "Internal Server Error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,142 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { eq, sql } from "@fitai/database";
import { db, gyms as gymsTable } from "@fitai/database";
import log from "@/lib/logger";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
async function ensureGymsTable() {
await db.run(sql`
CREATE TABLE IF NOT EXISTS gyms (
id TEXT PRIMARY KEY,
name TEXT NOT NULL,
location TEXT,
status TEXT NOT NULL CHECK (status IN ('active','inactive')) DEFAULT 'active',
admin_user_id TEXT NOT NULL,
created_at INTEGER NOT NULL,
updated_at INTEGER NOT NULL
)
`);
}
// GET /api/gyms/[id]/stats
// Get stats for a specific gym
export async function GET(
request: Request,
{ params }: { params: Promise<{ id: string }> },
) {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const appDb = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, appDb);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const { id: gymId } = await params;
const canViewGymStats =
currentUser.role === "superAdmin" || currentUser.role === "admin";
if (!canViewGymStats) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
if (currentUser.role !== "superAdmin" && currentUser.gymId !== gymId) {
return NextResponse.json(
{ error: "Forbidden - Cannot access other gym's data" },
{ status: 403 },
);
}
await ensureGymsTable();
// Get gym info using Drizzle ORM
const gym = await db
.select()
.from(gymsTable)
.where(eq(gymsTable.id, gymId))
.get();
if (!gym) {
return NextResponse.json({ error: "Gym not found" }, { status: 404 });
}
// Get user counts
const usersResult = await db.all(
sql`SELECT role, COUNT(*) as count FROM users WHERE gym_id = ${gymId} GROUP BY role`,
);
const userCounts: Record<string, number> = {
admin: 0,
trainer: 0,
client: 0,
};
for (const row of usersResult as any[]) {
if (row.role in userCounts) {
userCounts[row.role] = row.count;
}
}
// Get client stats
const clientsResult = (await db.all(
sql`SELECT
membership_type,
membership_status,
COUNT(*) as count
FROM clients
WHERE user_id IN (SELECT id FROM users WHERE gym_id = ${gymId})
GROUP BY membership_type, membership_status`,
)) as any[];
const membershipStats: Record<string, number> = {
basic: 0,
premium: 0,
vip: 0,
};
const activeClients = clientsResult.filter(
(c: any) => c.membership_status === "active",
);
for (const row of activeClients) {
if (row.membership_type in membershipStats) {
membershipStats[row.membership_type] = row.count;
}
}
// Get recent activity (attendance in last 30 days)
// Database stores timestamps in seconds, so convert milliseconds to seconds
const thirtyDaysAgo = Math.floor(Date.now() / 1000) - 30 * 24 * 60 * 60;
const attendanceResult = (await db.all(
sql`SELECT COUNT(*) as count FROM attendance
WHERE user_id IN (SELECT id FROM users WHERE gym_id = ${gymId})
AND check_in_time > ${thirtyDaysAgo}`,
)) as any[];
const attendanceCount = attendanceResult[0]?.count || 0;
const stats = {
totalUsers: userCounts.admin + userCounts.trainer + userCounts.client,
admins: userCounts.admin,
trainers: userCounts.trainer,
clients: userCounts.client,
membershipStats,
activeClients: activeClients.reduce(
(sum: number, c: any) => sum + c.count,
0,
),
attendanceLast30Days: attendanceCount,
};
return NextResponse.json({ gym, stats });
} catch (error) {
log.error("Failed to get gym stats", error);
return NextResponse.json(
{ error: "Internal Server Error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,88 @@
/**
* @jest-environment node
*/
import { GET } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
}));
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@fitai/database", () => ({
eq: jest.fn(() => ({})),
sql: jest.fn((strings: TemplateStringsArray) => strings.join("")),
db: {
run: jest.fn(),
select: jest.fn(() => ({
from: jest.fn(() => ({
where: jest.fn(() => ({
orderBy: jest.fn(() => ({
all: jest.fn().mockResolvedValue([
{ id: "gym_a", status: "active", name: "Gym A" },
{ id: "gym_b", status: "active", name: "Gym B" },
]),
})),
})),
})),
})),
gyms: {
status: "active",
},
users: {},
},
gyms: {
status: "active",
},
users: {},
}));
describe("GET /api/gyms authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database").getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue({});
});
it("returns only own gym for admin", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
const response = await GET();
const data = await response.json();
expect(response.status).toBe(200);
expect(data).toHaveLength(1);
expect(data[0].id).toBe("gym_a");
});
it("returns all gyms for superAdmin", async () => {
mockAuth.mockResolvedValue({ userId: "super_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "super_1",
role: "superAdmin",
gymId: null,
});
const response = await GET();
const data = await response.json();
expect(response.status).toBe(200);
expect(data).toHaveLength(2);
});
});

View File

@ -1,8 +1,9 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { eq, sql } from "@fitai/database";
import { db, users as usersTable } from "@fitai/database";
import { db, users as usersTable, gyms as gymsTable } from "@fitai/database";
import { ensureUserSynced } from "@/lib/sync-user";
import { getDatabase } from "@/lib/database";
import log from "@/lib/logger";
async function ensureGymsTable() {
@ -17,18 +18,102 @@ async function ensureGymsTable() {
updated_at INTEGER NOT NULL
)
`);
const columns = await db.all(sql`PRAGMA table_info('gyms')`);
const columnNames = new Set(
(columns as Array<{ name?: string }>)
.map((col) => col.name)
.filter(Boolean),
);
if (!columnNames.has("latitude")) {
await db.run(sql`ALTER TABLE gyms ADD COLUMN latitude REAL`);
}
if (!columnNames.has("longitude")) {
await db.run(sql`ALTER TABLE gyms ADD COLUMN longitude REAL`);
}
if (!columnNames.has("geofence_radius_meters")) {
await db.run(
sql`ALTER TABLE gyms ADD COLUMN geofence_radius_meters REAL NOT NULL DEFAULT 30`,
);
}
if (!columnNames.has("geofence_enabled")) {
await db.run(
sql`ALTER TABLE gyms ADD COLUMN geofence_enabled INTEGER NOT NULL DEFAULT 1`,
);
}
}
// GET /api/gyms
// Lists active gyms for selection (grid)
export async function GET() {
try {
await ensureGymsTable();
const rows = await db.all(
sql`SELECT * FROM gyms WHERE status = 'active' ORDER BY created_at DESC`,
);
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
return NextResponse.json(rows);
const appDb = await getDatabase();
const currentUser = await ensureUserSynced(userId, appDb);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
if (currentUser.role !== "admin" && currentUser.role !== "superAdmin") {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
await ensureGymsTable();
let rows = (await db.all(sql`
SELECT
id,
name,
location,
latitude,
longitude,
geofence_radius_meters as geofenceRadiusMeters,
geofence_enabled as geofenceEnabled,
status,
admin_user_id as adminUserId,
created_at as createdAt,
updated_at as updatedAt
FROM gyms
WHERE status = 'active'
ORDER BY created_at DESC
`)) as Array<{
id: string;
name: string;
location: string | null;
latitude: number | null;
longitude: number | null;
geofenceRadiusMeters: number | null;
geofenceEnabled: number | boolean | null;
status: "active" | "inactive";
adminUserId: string;
createdAt: number;
updatedAt: number;
}>;
if (currentUser.role !== "superAdmin") {
if (!currentUser.gymId) {
return NextResponse.json({ error: "No gym assigned" }, { status: 403 });
}
rows = rows.filter((row) => row.id === currentUser.gymId);
}
return NextResponse.json(
rows.map((row) => ({
...row,
geofenceEnabled:
typeof row.geofenceEnabled === "boolean"
? row.geofenceEnabled
: Boolean(row.geofenceEnabled),
})),
);
} catch (error) {
log.error("Failed to get gyms", error);
return new NextResponse("Internal Server Error", { status: 500 });
@ -45,60 +130,8 @@ export async function POST(req: Request) {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
// Ensure our local DB has the user synced (role, etc.)
const currentUser = await ensureUserSynced(userId, {
// minimal facade for ensureUserSynced to work: it expects an object implementing part of IDatabase
getUserById: async (id: string) => {
const row = await db
.select()
.from(usersTable)
.where(eq(usersTable.id, id))
.get();
return row
? {
id: row.id,
email: row.email,
firstName: row.firstName,
lastName: row.lastName,
password: row.password ?? "",
phone: row.phone ?? undefined,
role: row.role,
imageUrl: undefined,
createdAt: new Date(row.createdAt),
updatedAt: new Date(row.updatedAt),
}
: null;
},
updateUser: async (id: string, updates: any) => {
await db
.update(usersTable)
.set({
...updates,
updatedAt: new Date(),
})
.where(eq(usersTable.id, id))
.run();
const row = await db
.select()
.from(usersTable)
.where(eq(usersTable.id, id))
.get();
return row
? {
id: row.id,
email: row.email,
firstName: row.firstName,
lastName: row.lastName,
password: row.password ?? "",
phone: row.phone ?? undefined,
role: row.role,
imageUrl: undefined,
createdAt: new Date(row.createdAt),
updatedAt: new Date(row.updatedAt),
}
: null;
},
} as any);
const appDb = await getDatabase();
const currentUser = await ensureUserSynced(userId, appDb);
if (
!currentUser ||
@ -114,6 +147,21 @@ export async function POST(req: Request) {
const name = String(body.name ?? "").trim();
const location = body.location ? String(body.location).trim() : null;
const latitude =
body.latitude === undefined || body.latitude === null
? null
: Number(body.latitude);
const longitude =
body.longitude === undefined || body.longitude === null
? null
: Number(body.longitude);
const geofenceRadiusMeters =
body.geofenceRadiusMeters === undefined ||
body.geofenceRadiusMeters === null
? 30
: Number(body.geofenceRadiusMeters);
const geofenceEnabled =
body.geofenceEnabled === undefined ? true : Boolean(body.geofenceEnabled);
let adminUserId: string | null = body.adminUserId
? String(body.adminUserId)
: null;
@ -122,6 +170,33 @@ export async function POST(req: Request) {
return NextResponse.json({ error: "name is required" }, { status: 400 });
}
if (
latitude !== null &&
(!Number.isFinite(latitude) || latitude < -90 || latitude > 90)
) {
return NextResponse.json(
{ error: "latitude must be between -90 and 90" },
{ status: 400 },
);
}
if (
longitude !== null &&
(!Number.isFinite(longitude) || longitude < -180 || longitude > 180)
) {
return NextResponse.json(
{ error: "longitude must be between -180 and 180" },
{ status: 400 },
);
}
if (!Number.isFinite(geofenceRadiusMeters) || geofenceRadiusMeters <= 0) {
return NextResponse.json(
{ error: "geofenceRadiusMeters must be a positive number" },
{ status: 400 },
);
}
// Enforce admin ownership rules
if (currentUser.role === "admin") {
adminUserId = currentUser.id;
@ -146,19 +221,73 @@ export async function POST(req: Request) {
}
const id = generateId();
const nowTs = Date.now();
const nowTs = new Date();
await db.run(
sql`INSERT INTO gyms (id, name, location, status, admin_user_id, created_at, updated_at)
VALUES (${id}, ${name}, ${location ?? null}, 'active', ${adminUserId!}, ${nowTs}, ${nowTs})`,
);
// Use Drizzle's insert method instead of raw SQL
await db.run(sql`
INSERT INTO gyms (
id,
name,
location,
latitude,
longitude,
geofence_radius_meters,
geofence_enabled,
status,
admin_user_id,
created_at,
updated_at
) VALUES (
${id},
${name},
${location ?? null},
${latitude},
${longitude},
${geofenceRadiusMeters},
${geofenceEnabled ? 1 : 0},
${"active"},
${adminUserId!},
${Math.floor(nowTs.getTime() / 1000)},
${Math.floor(nowTs.getTime() / 1000)}
)
`);
// Assign the admin to this gym immediately after creation
await db.run(
sql`UPDATE users SET gym_id = ${id}, updated_at = ${nowTs} WHERE id = ${adminUserId!}`,
);
await db
.update(usersTable)
.set({ gymId: id, updatedAt: nowTs })
.where(eq(usersTable.id, adminUserId!));
const created = await db.get(sql`SELECT * FROM gyms WHERE id = ${id}`);
const rowsCreated = await db.all(sql`
SELECT
id,
name,
location,
latitude,
longitude,
geofence_radius_meters as geofenceRadiusMeters,
geofence_enabled as geofenceEnabled,
status,
admin_user_id as adminUserId,
created_at as createdAt,
updated_at as updatedAt
FROM gyms
WHERE id = ${id}
LIMIT 1
`);
const createdRow = rowsCreated?.[0] ?? null;
const created = createdRow
? {
...createdRow,
geofenceEnabled:
typeof (createdRow as { geofenceEnabled?: unknown })
.geofenceEnabled === "boolean"
? (createdRow as { geofenceEnabled: boolean }).geofenceEnabled
: Boolean(
(createdRow as { geofenceEnabled?: unknown }).geofenceEnabled,
),
}
: null;
return NextResponse.json(created, { status: 201 });
} catch (error) {
log.error("Failed to create gym", error);

View File

@ -0,0 +1,175 @@
import { auth } from "@clerk/nextjs/server";
import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
import { getUserMembershipContext } from "@/lib/membership/access";
export async function POST(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.hydrationTracking) {
return NextResponse.json(
{
error:
"Hydration tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const body = await req.json();
const { date, entries, totalWater, waterGoal } = body;
if (!date) {
return NextResponse.json({ error: "Date is required" }, { status: 400 });
}
// Check if entry already exists for this date
const existing = await db.getDailyHydration(userId, date);
let result;
if (existing) {
// Update existing entry
result = await db.updateDailyHydration(existing.id, {
entries,
totalWater: totalWater ?? existing.totalWater,
waterGoal: waterGoal ?? existing.waterGoal,
});
} else {
// Create new entry
result = await db.createDailyHydration({
userId,
date,
entries: entries || [],
totalWater: totalWater || 0,
waterGoal: waterGoal || 2000,
});
}
return NextResponse.json(result);
} catch (error) {
log.error("Failed to save hydration data", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function GET(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.hydrationTracking) {
return NextResponse.json(
{
error:
"Hydration tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const url = new URL(req.url);
const date = url.searchParams.get("date");
const startDate = url.searchParams.get("startDate");
const endDate = url.searchParams.get("endDate");
// Single date query
if (date) {
const result = await db.getDailyHydration(userId, date);
return NextResponse.json(result);
}
// Date range query
if (startDate && endDate) {
const results = await db.getDailyHydrationRange(
userId,
startDate,
endDate,
);
return NextResponse.json(results);
}
return NextResponse.json(
{ error: "Either 'date' or 'startDate' and 'endDate' are required" },
{ status: 400 },
);
} catch (error) {
log.error("Failed to fetch hydration data", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function DELETE(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.hydrationTracking) {
return NextResponse.json(
{
error:
"Hydration tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const url = new URL(req.url);
const id = url.searchParams.get("id");
if (!id) {
return NextResponse.json({ error: "ID is required" }, { status: 400 });
}
// Verify ownership before deletion
const existing = await db.getDailyHydrationById(id);
if (!existing) {
return NextResponse.json({ error: "Not found" }, { status: 404 });
}
if (existing.userId !== userId) {
return NextResponse.json(
{ error: "Forbidden: You can only delete your own hydration data" },
{ status: 403 },
);
}
const success = await db.deleteDailyHydration(id);
if (success) {
return NextResponse.json({ success: true });
} else {
return NextResponse.json({ error: "Not found" }, { status: 404 });
}
} catch (error) {
log.error("Failed to delete hydration data", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,103 @@
import { NextRequest, NextResponse } from "next/server";
import { auth, clerkClient } from "@clerk/nextjs/server";
import { getAuthContext } from "@/lib/auth/context";
import log from "@/lib/logger";
/**
* POST /api/invitations/[id]/resend
*
* Resend an invitation with same parameters
*/
export async function POST(
request: NextRequest,
{ params }: { params: Promise<{ id: string }> },
) {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { id: invitationId } = await params;
const authContext = await getAuthContext();
// Fetch pending invitations to find the one being resent
const client = await clerkClient();
const invitationList = await client.invitations.getInvitationList({
status: "pending",
});
// Find the invitation
const invitation = invitationList.data.find(
(inv) => inv.id === invitationId,
);
if (!invitation) {
return NextResponse.json(
{ error: "Invitation not found or already processed" },
{ status: 404 },
);
}
const metadata = invitation.publicMetadata as any;
const invitationGymId =
(metadata?.gymId as string | null | undefined) ?? null;
const createdBy = (metadata?.createdBy as string | undefined) ?? undefined;
const canManageByRole =
authContext.role === "superAdmin" ||
(authContext.role === "admin" &&
authContext.gymId !== null &&
invitationGymId === authContext.gymId);
// Check if current user created this invitation
if (createdBy !== userId && !canManageByRole) {
return NextResponse.json(
{
error:
"Forbidden - You can only resend invitations you created or manage within your scope",
},
{ status: 403 },
);
}
// Create new invitation with same parameters
const role = metadata?.role;
// Determine redirect URL based on role
const isStaffRole = ["admin", "trainer", "superAdmin"].includes(role);
const redirectUrl = isStaffRole
? `${process.env.NEXT_PUBLIC_APP_URL || "http://localhost:3000"}/sign-up`
: undefined;
const newInvitation = await client.invitations.createInvitation({
emailAddress: invitation.emailAddress,
publicMetadata: invitation.publicMetadata || undefined,
redirectUrl,
ignoreExisting: true,
});
log.info("Invitation resent", {
originalId: invitationId,
newId: newInvitation.id,
email: invitation.emailAddress,
resentBy: userId,
});
return NextResponse.json({
data: {
invitation: {
id: newInvitation.id,
email: newInvitation.emailAddress,
status: newInvitation.status,
},
},
});
} catch (error) {
log.error("POST /api/invitations/[id]/resend error:", error);
return NextResponse.json(
{ error: "Failed to resend invitation" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,83 @@
import { NextRequest, NextResponse } from "next/server";
import { auth, clerkClient } from "@clerk/nextjs/server";
import { getAuthContext } from "@/lib/auth/context";
import log from "@/lib/logger";
/**
* DELETE /api/invitations/[id]
*
* Revoke a pending invitation (creator-only permission)
*/
export async function DELETE(
request: NextRequest,
{ params }: { params: Promise<{ id: string }> },
) {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { id: invitationId } = await params;
const authContext = await getAuthContext();
// Fetch pending invitations to find and verify the one being revoked
const client = await clerkClient();
const invitationList = await client.invitations.getInvitationList({
status: "pending",
});
// Find the invitation
const invitation = invitationList.data.find(
(inv) => inv.id === invitationId,
);
if (!invitation) {
return NextResponse.json(
{ error: "Invitation not found or already processed" },
{ status: 404 },
);
}
const metadata = invitation.publicMetadata as any;
const invitationGymId =
(metadata?.gymId as string | null | undefined) ?? null;
const createdBy = (metadata?.createdBy as string | undefined) ?? undefined;
const canManageByRole =
authContext.role === "superAdmin" ||
(authContext.role === "admin" &&
authContext.gymId !== null &&
invitationGymId === authContext.gymId);
// Check if current user created this invitation
if (createdBy !== userId && !canManageByRole) {
return NextResponse.json(
{
error:
"Forbidden - You can only cancel invitations you created or manage within your scope",
},
{ status: 403 },
);
}
// Revoke the invitation
await client.invitations.revokeInvitation(invitationId);
log.info("Invitation revoked", {
invitationId,
revokedBy: userId,
email: invitation.emailAddress,
});
return NextResponse.json({
data: { success: true },
});
} catch (error) {
log.error("DELETE /api/invitations/[id] error:", error);
return NextResponse.json(
{ error: "Failed to revoke invitation" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,92 @@
/**
* @jest-environment node
*/
import { NextRequest } from "next/server";
import { POST } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
clerkClient: jest.fn(),
}));
jest.mock("@/lib/auth/context", () => ({
getAuthContext: jest.fn(),
}));
jest.mock("@/lib/logger", () => ({
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
debug: jest.fn(),
}));
describe("POST /api/invitations authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockClerkClient = require("@clerk/nextjs/server")
.clerkClient as jest.Mock;
const mockGetAuthContext = require("@/lib/auth/context")
.getAuthContext as jest.Mock;
const createInvitation = jest.fn();
beforeEach(() => {
jest.clearAllMocks();
mockClerkClient.mockResolvedValue({
invitations: {
createInvitation,
},
});
});
it("blocks admin from inviting into another gym", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockGetAuthContext.mockResolvedValue({
userId: "admin_1",
role: "admin",
gymId: "gym_a",
});
const request = new NextRequest("http://localhost/api/invitations", {
method: "POST",
body: JSON.stringify({
inviteeEmail: "test@example.com",
roleAssigned: "trainer",
gymId: "gym_b",
}),
});
const response = await POST(request);
expect(response.status).toBe(403);
expect(createInvitation).not.toHaveBeenCalled();
});
it("allows superAdmin to invite with explicit gym", async () => {
mockAuth.mockResolvedValue({ userId: "super_1" });
mockGetAuthContext.mockResolvedValue({
userId: "super_1",
role: "superAdmin",
gymId: null,
});
createInvitation.mockResolvedValue({ id: "inv_1" });
const request = new NextRequest("http://localhost/api/invitations", {
method: "POST",
body: JSON.stringify({
inviteeEmail: "test@example.com",
roleAssigned: "admin",
gymId: "gym_b",
}),
});
const response = await POST(request);
expect(response.status).toBe(201);
expect(createInvitation).toHaveBeenCalledWith(
expect.objectContaining({
emailAddress: "test@example.com",
}),
);
});
});

View File

@ -1,5 +1,90 @@
import { NextResponse } from "next/server";
import { NextRequest, NextResponse } from "next/server";
import { auth, clerkClient } from "@clerk/nextjs/server";
import { getAuthContext } from "@/lib/auth/context";
import { getInvitableRoles, validateGymAccess } from "@/lib/auth/permissions";
import log from "@/lib/logger";
/**
* GET /api/invitations
*
* Fetch pending invitations with gym and creator filtering.
* Users can only see invitations they created, scoped to their gym access.
*
* Query params:
* - gymId: Optional gym filter (superAdmin only)
*/
export async function GET(request: NextRequest) {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
// Get auth context
const authContext = await getAuthContext();
const { role, gymId: userGymId } = authContext;
// Get query params
const { searchParams } = new URL(request.url);
const gymIdParam = searchParams.get("gymId");
// Validate gym access
const targetGymId = gymIdParam || undefined;
const accessError = validateGymAccess(role, userGymId, targetGymId);
if (accessError) return accessError;
// Fetch all pending invitations from Clerk
const client = await clerkClient();
const invitationList = await client.invitations.getInvitationList({
status: "pending",
});
// Filter invitations based on:
// 1. Creator (only show invitations created by current user)
// 2. Gym (apply gym-scoping rules)
const filteredInvitations = invitationList.data
.filter((inv) => {
const metadata = inv.publicMetadata as any;
// Creator filter: only show if user created it
if (metadata?.createdBy !== userId) {
return false;
}
// Gym filter: SuperAdmin can see all, others only their gym
if (role === "superAdmin") {
// If gymId param provided, filter by it
if (targetGymId && (metadata?.gymId ?? null) !== targetGymId) {
return false;
}
return true;
} else {
// Non-superAdmins: must match their gym (normalize null/undefined)
return (metadata?.gymId ?? null) === (userGymId ?? null);
}
})
.map((inv) => ({
id: inv.id,
emailAddress: inv.emailAddress,
publicMetadata: inv.publicMetadata,
status: inv.status,
url: inv.url,
createdAt: inv.createdAt,
updatedAt: inv.updatedAt,
revoked: inv.revoked,
}));
return NextResponse.json({
data: { invitations: filteredInvitations },
});
} catch (error) {
log.error("GET /api/invitations error:", error);
return NextResponse.json(
{ error: "Failed to fetch invitations" },
{ status: 500 },
);
}
}
/**
* POST /api/invitations
@ -47,91 +132,51 @@ export async function POST(req: Request) {
);
}
// Fetch inviter user from Clerk
const client = await clerkClient();
const inviter = await client.users.getUser(userId);
const inviterRole =
(inviter.publicMetadata?.role as
| "superAdmin"
| "admin"
| "trainer"
| "client"
| "generalUser") ?? "client";
const inviterGymId =
(inviter.publicMetadata?.gymId as string | undefined) ?? undefined;
const authContext = await getAuthContext();
const { role: inviterRole, gymId: inviterGymId } = authContext;
const allowedRoles = getInvitableRoles(inviterRole);
if (!allowedRoles.includes(roleAssigned)) {
return NextResponse.json(
{ error: `Forbidden - Cannot invite role '${roleAssigned}'` },
{ status: 403 },
);
}
// Enforce role-based rules and resolve target gymId for the invitation
let gymIdForInvite: string | null = null;
switch (inviterRole) {
case "admin": {
if (roleAssigned !== "trainer" && roleAssigned !== "client") {
return NextResponse.json(
{ error: "Admin can only invite trainer or client" },
{ status: 403 },
);
}
if (!inviterGymId) {
return NextResponse.json(
{ error: "Inviter admin must be assigned to a gym" },
{ status: 400 },
);
}
gymIdForInvite = inviterGymId;
break;
}
case "trainer": {
if (roleAssigned !== "client") {
return NextResponse.json(
{ error: "Trainer can only invite client" },
{ status: 403 },
);
}
if (!inviterGymId) {
return NextResponse.json(
{ error: "Inviter trainer must be assigned to a gym" },
{ status: 400 },
);
}
gymIdForInvite = inviterGymId;
break;
}
case "superAdmin": {
if (
roleAssigned !== "admin" &&
roleAssigned !== "trainer" &&
roleAssigned !== "client"
) {
return NextResponse.json(
{ error: "Invalid roleAssigned for SuperAdmin" },
{ status: 400 },
);
}
// Prefer explicitly provided gymId, otherwise fall back to inviter's gymId if present
gymIdForInvite = requestedGymId || inviterGymId || null;
if (!gymIdForInvite) {
return NextResponse.json(
{ error: "gymId is required for SuperAdmin when inviting" },
{ status: 400 },
);
}
break;
}
default: {
if (inviterRole === "superAdmin") {
gymIdForInvite = requestedGymId || inviterGymId || null;
if (!gymIdForInvite) {
return NextResponse.json(
{ error: "Inviter role not permitted to create invitations" },
{ error: "gymId is required for superAdmin invitations" },
{ status: 400 },
);
}
} else {
if (!inviterGymId) {
return NextResponse.json(
{ error: "Inviter must be assigned to a gym" },
{ status: 400 },
);
}
if (requestedGymId && requestedGymId !== inviterGymId) {
return NextResponse.json(
{ error: "Cannot invite users into another gym" },
{ status: 403 },
);
}
gymIdForInvite = inviterGymId;
}
// Create Clerk invitation with metadata needed by webhook to assign role & gym
// reuse existing Clerk client instance
const client = await clerkClient();
const invitation = await client.invitations.createInvitation({
emailAddress: inviteeEmail,
publicMetadata: {
roleAssigned,
role: roleAssigned,
gymId: gymIdForInvite,
inviterUserId: inviter.id,
createdBy: userId,
},
});

View File

@ -0,0 +1,32 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { MEMBERSHIP_FEATURES } from "@/lib/membership/features";
import { getUserMembershipContext } from "@/lib/membership/access";
export async function GET() {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { membershipType, features } = await getUserMembershipContext(userId);
return NextResponse.json({
success: true,
data: {
membershipType,
currentFeatures: features,
plans: MEMBERSHIP_FEATURES,
},
meta: {
timestamp: new Date().toISOString(),
},
});
} catch {
return NextResponse.json(
{ error: "Failed to load membership features" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,87 @@
/**
* @jest-environment node
*/
import { NextRequest } from "next/server";
import { POST } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
}));
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@/lib/logger", () => ({
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
debug: jest.fn(),
}));
describe("POST /api/notifications authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database").getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
const mockDb = {
getUserById: jest.fn(),
createNotification: jest.fn(),
};
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue(mockDb);
});
it("returns 403 for non-staff user", async () => {
mockAuth.mockResolvedValue({ userId: "client_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "client_1",
role: "client",
gymId: "gym_a",
});
const request = new NextRequest("http://localhost/api/notifications", {
method: "POST",
body: JSON.stringify({
targetUserId: "client_2",
title: "Hello",
message: "Test",
type: "system",
}),
});
const response = await POST(request);
expect(response.status).toBe(403);
});
it("returns 403 for cross-gym notify by admin", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
mockDb.getUserById.mockResolvedValue({ id: "client_2", gymId: "gym_b" });
const request = new NextRequest("http://localhost/api/notifications", {
method: "POST",
body: JSON.stringify({
targetUserId: "client_2",
title: "Hello",
message: "Test",
type: "system",
}),
});
const response = await POST(request);
expect(response.status).toBe(403);
});
});

View File

@ -2,6 +2,7 @@ import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import log from "@/lib/logger";
import { ensureUserSynced } from "@/lib/sync-user";
/**
* GET /api/notifications
@ -84,6 +85,39 @@ export async function POST(req: NextRequest) {
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(userId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const canCreateNotifications =
currentUser.role === "superAdmin" ||
currentUser.role === "admin" ||
currentUser.role === "trainer";
if (!canCreateNotifications) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const targetUser = await db.getUserById(targetUserId);
if (!targetUser) {
return NextResponse.json(
{ error: "Target user not found" },
{ status: 404 },
);
}
if (
currentUser.role !== "superAdmin" &&
(!currentUser.gymId || targetUser.gymId !== currentUser.gymId)
) {
return NextResponse.json(
{ error: "Forbidden - Cannot notify users from other gyms" },
{ status: 403 },
);
}
const notification = await db.createNotification({
id: crypto.randomUUID(),
userId: targetUserId,

View File

@ -0,0 +1,163 @@
import { auth } from "@clerk/nextjs/server";
import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
import { getUserMembershipContext } from "@/lib/membership/access";
export async function POST(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Nutrition tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const body = await req.json();
const {
dailyNutritionId,
mealType,
foodName,
calories,
protein,
carbs,
fats,
} = body;
if (!mealType || !foodName || calories === undefined) {
return NextResponse.json(
{ error: "mealType, foodName, and calories are required" },
{ status: 400 },
);
}
const result = await db.createMealEntry({
userId,
dailyNutritionId,
mealType,
foodName,
calories,
protein,
carbs,
fats,
timestamp: new Date(),
});
return NextResponse.json(result);
} catch (error) {
log.error("Failed to create meal entry", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function GET(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Nutrition tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const url = new URL(req.url);
const date = url.searchParams.get("date");
if (!date) {
return NextResponse.json(
{ error: "date is required (YYYY-MM-DD format)" },
{ status: 400 },
);
}
const results = await db.getMealEntriesByDate(userId, date);
return NextResponse.json(results);
} catch (error) {
log.error("Failed to fetch meal entries", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function DELETE(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Nutrition tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const url = new URL(req.url);
const id = url.searchParams.get("id");
if (!id) {
return NextResponse.json({ error: "ID is required" }, { status: 400 });
}
// Verify ownership before deletion
const existing = await db.getMealEntryById(id);
if (!existing) {
return NextResponse.json({ error: "Not found" }, { status: 404 });
}
if (existing.userId !== userId) {
return NextResponse.json(
{ error: "Forbidden: You can only delete your own meal entries" },
{ status: 403 },
);
}
const success = await db.deleteMealEntry(id);
if (success) {
return NextResponse.json({ success: true });
} else {
return NextResponse.json({ error: "Not found" }, { status: 404 });
}
} catch (error) {
log.error("Failed to delete meal entry", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,175 @@
import { auth } from "@clerk/nextjs/server";
import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
import { getUserMembershipContext } from "@/lib/membership/access";
export async function POST(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Nutrition tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const body = await req.json();
const { date, meals, totalCalories, calorieGoal } = body;
if (!date) {
return NextResponse.json({ error: "Date is required" }, { status: 400 });
}
// Check if entry already exists for this date
const existing = await db.getDailyNutrition(userId, date);
let result;
if (existing) {
// Update existing entry
result = await db.updateDailyNutrition(existing.id, {
meals,
totalCalories: totalCalories ?? existing.totalCalories,
calorieGoal: calorieGoal ?? existing.calorieGoal,
});
} else {
// Create new entry
result = await db.createDailyNutrition({
userId,
date,
meals: meals || [],
totalCalories: totalCalories || 0,
calorieGoal: calorieGoal || 2000,
});
}
return NextResponse.json(result);
} catch (error) {
log.error("Failed to save nutrition data", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function GET(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Nutrition tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const url = new URL(req.url);
const date = url.searchParams.get("date");
const startDate = url.searchParams.get("startDate");
const endDate = url.searchParams.get("endDate");
// Single date query
if (date) {
const result = await db.getDailyNutrition(userId, date);
return NextResponse.json(result);
}
// Date range query
if (startDate && endDate) {
const results = await db.getDailyNutritionRange(
userId,
startDate,
endDate,
);
return NextResponse.json(results);
}
return NextResponse.json(
{ error: "Either 'date' or 'startDate' and 'endDate' are required" },
{ status: 400 },
);
} catch (error) {
log.error("Failed to fetch nutrition data", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function DELETE(req: NextRequest) {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const db = await getDatabase();
await ensureUserSynced(userId, db);
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Nutrition tracking is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const url = new URL(req.url);
const id = url.searchParams.get("id");
if (!id) {
return NextResponse.json({ error: "ID is required" }, { status: 400 });
}
// Verify ownership before deletion
const existing = await db.getDailyNutritionById(id);
if (!existing) {
return NextResponse.json({ error: "Not found" }, { status: 404 });
}
if (existing.userId !== userId) {
return NextResponse.json(
{ error: "Forbidden: You can only delete your own nutrition data" },
{ status: 403 },
);
}
const success = await db.deleteDailyNutrition(id);
if (success) {
return NextResponse.json({ success: true });
} else {
return NextResponse.json({ error: "Not found" }, { status: 404 });
}
} catch (error) {
log.error("Failed to delete nutrition data", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -1,11 +1,14 @@
import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from "../../../../lib/database/index";
import { auth } from "@clerk/nextjs/server";
import log from "@/lib/logger";
import { fitnessProfileSchema } from "@/lib/validation/schemas";
import {
validateRequestBody,
validationErrorResponse,
} from "@/lib/validation/helpers";
import { getFitnessProfilesByGym } from "@/lib/gym-context";
import { ensureUserSynced } from "@/lib/sync-user";
export async function POST(request: NextRequest) {
try {
@ -63,12 +66,47 @@ export async function POST(request: NextRequest) {
export async function GET(request: NextRequest) {
try {
const db = await getDatabase();
const { searchParams } = new URL(request.url);
const userId = searchParams.get("userId");
// First authenticate
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
if (userId) {
const profile = await db.getFitnessProfileByUserId(userId);
const db = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const { searchParams } = new URL(request.url);
const targetUserId = searchParams.get("userId");
// If accessing another user's profile, verify gym access
if (targetUserId && targetUserId !== clerkUserId) {
const isStaff =
currentUser.role === "admin" ||
currentUser.role === "superAdmin" ||
currentUser.role === "trainer";
if (!isStaff) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
// Staff need to verify target user is in same gym
if (currentUser.role !== "superAdmin") {
const targetUser = await db.getUserById(targetUserId);
if (!targetUser || targetUser.gymId !== currentUser.gymId) {
return NextResponse.json(
{ error: "Forbidden - Cannot access users from other gyms" },
{ status: 403 },
);
}
}
}
if (targetUserId) {
const profile = await db.getFitnessProfileByUserId(targetUserId);
if (!profile) {
return NextResponse.json(
{ error: "Profile not found" },
@ -78,8 +116,28 @@ export async function GET(request: NextRequest) {
return NextResponse.json({ profile });
}
const profiles = await db.getAllFitnessProfiles();
return NextResponse.json({ profiles });
// Staff get gym-scoped profiles
const isStaff =
currentUser.role === "admin" ||
currentUser.role === "superAdmin" ||
currentUser.role === "trainer";
if (isStaff) {
const targetGymId =
currentUser.role === "superAdmin"
? (searchParams.get("gymId") ?? undefined)
: (currentUser.gymId ?? undefined);
const profiles = targetGymId
? await getFitnessProfilesByGym(targetGymId)
: await db.getAllFitnessProfiles();
return NextResponse.json({ profiles });
}
// Regular users only get their own
const profile = await db.getFitnessProfileByUserId(clerkUserId);
return NextResponse.json({ profile: profile ? [profile] : [] });
} catch (error) {
log.error("Failed to get fitness profiles", error);
return NextResponse.json(

View File

@ -0,0 +1,83 @@
/**
* @jest-environment node
*/
import { POST } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
}));
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@/lib/logger", () => ({
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
debug: jest.fn(),
}));
describe("POST /api/recommendations/approve authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database").getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
const mockDb = {
getAllRecommendations: jest.fn(),
getUserById: jest.fn(),
updateRecommendation: jest.fn(),
createNotification: jest.fn(),
};
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue(mockDb);
});
it("returns 403 for non-staff role", async () => {
mockAuth.mockResolvedValue({ userId: "client_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "client_1",
role: "client",
gymId: "gym_a",
});
const req = new Request("http://localhost/api/recommendations/approve", {
method: "POST",
body: JSON.stringify({ recommendationId: "rec_1", status: "approved" }),
headers: { "Content-Type": "application/json" },
});
const res = await POST(req);
expect(res.status).toBe(403);
});
it("returns 403 for cross-gym approval by admin", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
mockDb.getAllRecommendations.mockResolvedValue([
{ id: "rec_1", userId: "client_1" },
]);
mockDb.getUserById.mockResolvedValue({ id: "client_1", gymId: "gym_b" });
const req = new Request("http://localhost/api/recommendations/approve", {
method: "POST",
body: JSON.stringify({ recommendationId: "rec_1", status: "approved" }),
headers: { "Content-Type": "application/json" },
});
const res = await POST(req);
expect(res.status).toBe(403);
});
});

View File

@ -1,13 +1,115 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import log from "@/lib/logger";
import { ensureUserSynced } from "@/lib/sync-user";
const AI_LINK_PREFIX = "[AI_LINKED]";
type GoalType =
| "weight_target"
| "strength_milestone"
| "endurance_target"
| "flexibility_goal"
| "habit_building"
| "custom";
interface ParsedPlanItem {
id: string;
title: string;
description: string;
goalType: GoalType;
}
function inferGoalType(text: string): GoalType {
const normalized = text.toLowerCase();
if (
normalized.includes("strength") ||
normalized.includes("bench") ||
normalized.includes("squat") ||
normalized.includes("deadlift") ||
normalized.includes("weights")
) {
return "strength_milestone";
}
if (
normalized.includes("run") ||
normalized.includes("cardio") ||
normalized.includes("endurance") ||
normalized.includes("cycle")
) {
return "endurance_target";
}
if (
normalized.includes("stretch") ||
normalized.includes("mobility") ||
normalized.includes("yoga") ||
normalized.includes("flexibility")
) {
return "flexibility_goal";
}
if (
normalized.includes("daily") ||
normalized.includes("routine") ||
normalized.includes("habit")
) {
return "habit_building";
}
return "custom";
}
function parseActivityPlanToItems(activityPlan: string): ParsedPlanItem[] {
const lines = activityPlan
.replace(/\r\n/g, "\n")
.split(/\n+/)
.flatMap((line) => line.split(/(?<=[.!?])\s+(?=[A-Z0-9])/g))
.map((line) => line.trim())
.filter(Boolean)
.map((line) => line.replace(/^[-*•\d.)\s]+/, "").trim())
.filter((line) => line.length > 10)
.slice(0, 8);
const uniqueLines = Array.from(new Set(lines));
return uniqueLines.map((line) => ({
id: crypto.randomUUID(),
title: line.length > 72 ? `${line.slice(0, 69)}...` : line,
description: line,
goalType: inferGoalType(line),
}));
}
function getDefaultPlanItems(): ParsedPlanItem[] {
const defaults = [
"Complete 3 strength sessions this week with progressive overload.",
"Add 2 cardio sessions of 25-30 minutes for endurance.",
"Do a 10-minute mobility routine daily after training.",
];
return defaults.map((line) => ({
id: crypto.randomUUID(),
title: line.length > 72 ? `${line.slice(0, 69)}...` : line,
description: line,
goalType: inferGoalType(line),
}));
}
export async function POST(req: Request) {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const body = await req.json();
log.debug("Approve recommendation request body", { body });
const { recommendationId, status, approvedBy } = body;
const { recommendationId, status } = body;
if (!recommendationId || !status) {
log.error("Missing required fields", {
@ -22,12 +124,52 @@ export async function POST(req: Request) {
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const canApproveRecommendations =
currentUser.role === "superAdmin" ||
currentUser.role === "admin" ||
currentUser.role === "trainer";
if (!canApproveRecommendations) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const existingRecommendation = (await db.getAllRecommendations()).find(
(recommendation) => recommendation.id === recommendationId,
);
if (!existingRecommendation) {
return NextResponse.json(
{ error: "Recommendation not found" },
{ status: 404 },
);
}
if (currentUser.role !== "superAdmin") {
const targetUser = await db.getUserById(existingRecommendation.userId);
if (
!currentUser.gymId ||
!targetUser ||
targetUser.gymId !== currentUser.gymId
) {
return NextResponse.json(
{ error: "Forbidden - Cannot access users from other gyms" },
{ status: 403 },
);
}
}
// Update recommendation status
const updates: any = {
status,
approvedAt: status === "approved" ? new Date() : undefined,
approvedBy: status === "approved" ? approvedBy : undefined,
approvedBy: status === "approved" ? clerkUserId : undefined,
};
// Remove undefined keys
@ -47,8 +189,103 @@ export async function POST(req: Request) {
);
}
// If approved, create a notification for the user
let pausedGoalsCount = 0;
let createdGoalsCount = 0;
// If approved, regenerate linked AI goals and create a notification for the user
if (status === "approved") {
try {
const existingActiveGoals = await db.getFitnessGoalsByUserId(
updatedRecommendation.userId,
"active",
);
const linkedGoals = existingActiveGoals.filter((goal) =>
goal.notes?.startsWith(AI_LINK_PREFIX),
);
pausedGoalsCount = linkedGoals.length;
await Promise.all(
linkedGoals.map((goal) =>
db.updateFitnessGoal(goal.id, {
status: "paused",
notes: `${goal.notes || ""}\nPaused due to recommendation approval on ${new Date().toISOString()}`,
}),
),
);
let planItems = parseActivityPlanToItems(
updatedRecommendation.activityPlan || "",
);
if (
planItems.length === 0 &&
updatedRecommendation.recommendationText
) {
planItems = parseActivityPlanToItems(
updatedRecommendation.recommendationText,
);
}
if (planItems.length === 0) {
planItems = getDefaultPlanItems();
}
const fitnessProfileId =
updatedRecommendation.fitnessProfileId ||
(await db.getFitnessProfileByUserId(updatedRecommendation.userId))
?.id;
if (!fitnessProfileId) {
log.warn("No fitness profile available for AI goal creation", {
recommendationId,
userId: updatedRecommendation.userId,
});
} else {
const createdGoals = await Promise.all(
planItems.map((item) =>
db.createFitnessGoal({
id: crypto.randomUUID(),
userId: updatedRecommendation.userId,
fitnessProfileId,
goalType: item.goalType,
title: item.title,
description: item.description,
targetValue: undefined,
currentValue: 0,
unit: undefined,
startDate: new Date(),
targetDate: undefined,
completedDate: undefined,
status: "active",
progress: 0,
priority: "medium",
notes: `${AI_LINK_PREFIX} recommendationId=${updatedRecommendation.id}; itemId=${item.id}`,
}),
),
);
createdGoalsCount = createdGoals.length;
}
log.info("Regenerated linked AI goals from approved recommendation", {
recommendationId: updatedRecommendation.id,
userId: updatedRecommendation.userId,
pausedGoals: pausedGoalsCount,
createdGoals: createdGoalsCount,
});
} catch (goalConversionError) {
log.error(
"Failed to regenerate linked goals for approved recommendation",
goalConversionError,
{
recommendationId,
userId: updatedRecommendation.userId,
},
);
}
try {
await db.createNotification({
id: crypto.randomUUID(),
@ -75,6 +312,8 @@ export async function POST(req: Request) {
data: updatedRecommendation,
meta: {
timestamp: new Date().toISOString(),
pausedGoals: pausedGoalsCount,
createdGoals: createdGoalsCount,
},
});
} catch (error) {

View File

@ -0,0 +1,455 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { buildAIContext } from "@/lib/ai/ai-context";
import { buildEnhancedPrompt, buildBasicPrompt } from "@/lib/ai/prompt-builder";
import log from "@/lib/logger";
import { ensureUserSynced } from "@/lib/sync-user";
import { getUserMembershipContext } from "@/lib/membership/access";
const AI_LINK_PREFIX = "[AI_LINKED]";
interface ParsedPlanItem {
id: string;
title: string;
description: string;
goalType:
| "weight_target"
| "strength_milestone"
| "endurance_target"
| "flexibility_goal"
| "habit_building"
| "custom";
}
interface GeneratedPlanContent {
recommendationText?: string;
activityPlan?: string;
dietPlan?: string;
}
function buildFallbackPlan(profile: {
activityLevel?: string;
fitnessGoals?: string[] | string;
medicalConditions?: string;
}): GeneratedPlanContent {
const goals = Array.isArray(profile.fitnessGoals)
? profile.fitnessGoals
: typeof profile.fitnessGoals === "string" && profile.fitnessGoals
? [profile.fitnessGoals]
: ["general fitness"];
const primaryGoal = goals[0] || "general fitness";
const activityLevel = profile.activityLevel || "moderate";
const hasMedicalNotes = Boolean(profile.medicalConditions?.trim());
return {
recommendationText:
`Personalized starter plan focused on ${primaryGoal} with ${activityLevel} activity pacing.` +
(hasMedicalNotes
? " Medical notes detected, so keep intensity conservative and progress gradually."
: ""),
activityPlan:
"- 3 strength sessions per week (full-body, 35-45 min)\n" +
"- 2 cardio sessions per week (20-30 min brisk walk/run/cycle)\n" +
"- 10 minutes daily mobility/stretching after workouts\n" +
"- 1 full recovery day each week",
dietPlan:
"- Build meals around lean protein, vegetables, whole grains, and hydration\n" +
"- Keep portions consistent and avoid skipping meals\n" +
"- Track intake daily and adjust calories based on weekly progress",
};
}
function inferGoalType(text: string): ParsedPlanItem["goalType"] {
const normalized = text.toLowerCase();
if (
normalized.includes("strength") ||
normalized.includes("bench") ||
normalized.includes("squat") ||
normalized.includes("deadlift") ||
normalized.includes("weights")
) {
return "strength_milestone";
}
if (
normalized.includes("run") ||
normalized.includes("cardio") ||
normalized.includes("endurance") ||
normalized.includes("cycle")
) {
return "endurance_target";
}
if (
normalized.includes("stretch") ||
normalized.includes("mobility") ||
normalized.includes("yoga") ||
normalized.includes("flexibility")
) {
return "flexibility_goal";
}
if (
normalized.includes("daily") ||
normalized.includes("routine") ||
normalized.includes("habit")
) {
return "habit_building";
}
return "custom";
}
function parseActivityPlanToItems(activityPlan: string): ParsedPlanItem[] {
const lines = activityPlan
.replace(/\r\n/g, "\n")
.split(/\n+/)
.flatMap((line) => line.split(/(?<=[.!?])\s+(?=[A-Z0-9])/g))
.map((line) => line.trim())
.filter(Boolean)
.map((line) => line.replace(/^[-*•\d.)\s]+/, "").trim())
.filter((line) => line.length > 10)
.slice(0, 8);
const uniqueLines = Array.from(new Set(lines));
return uniqueLines.map((line) => ({
id: crypto.randomUUID(),
title: line.length > 72 ? `${line.slice(0, 69)}...` : line,
description: line,
goalType: inferGoalType(line),
}));
}
function getDefaultPlanItems(): ParsedPlanItem[] {
const defaults = [
"Complete 3 strength sessions this week with progressive overload.",
"Add 2 cardio sessions of 25-30 minutes for endurance.",
"Do a 10-minute mobility routine daily after training.",
];
return defaults.map((line) => ({
id: crypto.randomUUID(),
title: line.length > 72 ? `${line.slice(0, 69)}...` : line,
description: line,
goalType: inferGoalType(line),
}));
}
function parseJsonPayload(content: string): GeneratedPlanContent {
let cleanResponse = content.trim();
if (cleanResponse.startsWith("```json")) {
cleanResponse = cleanResponse
.replace(/^```json\s*/, "")
.replace(/\s*```$/, "");
} else if (cleanResponse.startsWith("```")) {
cleanResponse = cleanResponse.replace(/^```\s*/, "").replace(/\s*```$/, "");
}
const firstBrace = cleanResponse.indexOf("{");
const lastBrace = cleanResponse.lastIndexOf("}");
if (firstBrace !== -1 && lastBrace !== -1) {
cleanResponse = cleanResponse.substring(firstBrace, lastBrace + 1);
}
return JSON.parse(cleanResponse) as GeneratedPlanContent;
}
async function generateWithOpenAI(
openaiApiKey: string,
prompt: string,
): Promise<GeneratedPlanContent> {
const response = await fetch("https://api.openai.com/v1/chat/completions", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${openaiApiKey}`,
},
body: JSON.stringify({
model: "gpt-4o-mini",
messages: [
{
role: "system",
content:
'You are a professional fitness trainer and nutritionist. Always respond with valid JSON only, no markdown or code blocks. The response must have this exact structure: {"recommendationText": "string", "activityPlan": "string", "dietPlan": "string"}',
},
{
role: "user",
content: prompt,
},
],
temperature: 0.7,
max_tokens: 1500,
response_format: { type: "json_object" },
}),
});
if (!response.ok) {
const errorText = await response.text();
throw new Error(`OpenAI failed: ${response.status} ${errorText}`);
}
const data = await response.json();
return parseJsonPayload(data.choices[0].message.content as string);
}
async function generateWithDeepSeek(
deepseekApiKey: string,
prompt: string,
): Promise<GeneratedPlanContent> {
const response = await fetch("https://api.deepseek.com/v1/chat/completions", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${deepseekApiKey}`,
},
body: JSON.stringify({
model: "deepseek-chat",
messages: [
{
role: "system",
content:
"You are a professional fitness trainer and nutritionist. Always respond with valid JSON only, no markdown or code blocks.",
},
{
role: "user",
content: prompt,
},
],
temperature: 0.7,
max_tokens: 1200,
}),
});
if (!response.ok) {
const errorText = await response.text();
throw new Error(`DeepSeek failed: ${response.status} ${errorText}`);
}
const data = await response.json();
return parseJsonPayload(data.choices[0].message.content as string);
}
async function generateWithOllama(
prompt: string,
): Promise<GeneratedPlanContent> {
const response = await fetch("http://localhost:11434/api/generate", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
model: "gemma3:latest",
prompt,
stream: false,
format: "json",
}),
});
if (!response.ok) {
const errorText = await response.text();
throw new Error(`Ollama failed: ${response.status} ${errorText}`);
}
const data = await response.json();
return parseJsonPayload(data.response as string);
}
export async function POST() {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(userId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const { membershipType, features } = await getUserMembershipContext(userId);
if (membershipType === "basic") {
return NextResponse.json(
{
error:
"AI plan generation is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
if (features.recommendationsPerMonth > 0) {
const currentMonth = new Date();
const monthStart = new Date(
currentMonth.getFullYear(),
currentMonth.getMonth(),
1,
);
const monthEnd = new Date(
currentMonth.getFullYear(),
currentMonth.getMonth() + 1,
1,
);
const existingRecommendations =
await db.getRecommendationsByUserId(userId);
const recommendationsThisMonth = existingRecommendations.filter(
(recommendation) =>
recommendation.generatedAt >= monthStart &&
recommendation.generatedAt < monthEnd,
).length;
if (recommendationsThisMonth >= features.recommendationsPerMonth) {
return NextResponse.json(
{
error: `Your ${membershipType} plan includes ${features.recommendationsPerMonth} AI recommendation(s) per month`,
membershipType,
},
{ status: 403 },
);
}
}
const profile = await db.getFitnessProfileByUserId(userId);
if (!profile) {
return NextResponse.json(
{ error: "Complete your fitness profile before generating a plan" },
{ status: 404 },
);
}
let prompt: string;
try {
const context = await buildAIContext(userId);
prompt = buildEnhancedPrompt(context);
} catch (error) {
log.warn("Failed to build AI context for self-generate", {
userId,
error: error instanceof Error ? error.message : String(error),
});
prompt = buildBasicPrompt(profile);
}
const openaiApiKey = process.env.OPENAI_API_KEY;
const deepseekApiKey = process.env.DEEPSEEK_API_KEY;
let parsedResponse: GeneratedPlanContent;
let usedFallbackPlan = false;
try {
if (openaiApiKey) {
parsedResponse = await generateWithOpenAI(openaiApiKey, prompt);
} else if (deepseekApiKey) {
parsedResponse = await generateWithDeepSeek(deepseekApiKey, prompt);
} else {
parsedResponse = await generateWithOllama(prompt);
}
} catch (providerError) {
log.error("Self-generate provider failed", providerError, {
userId,
hasOpenAI: Boolean(openaiApiKey),
hasDeepSeek: Boolean(deepseekApiKey),
});
parsedResponse = buildFallbackPlan({
activityLevel: profile.activityLevel,
fitnessGoals: profile.fitnessGoals,
medicalConditions: profile.medicalConditions,
});
usedFallbackPlan = true;
}
const recommendation = await db.createRecommendation({
id: crypto.randomUUID(),
userId,
fitnessProfileId: profile.id,
recommendationText: parsedResponse.recommendationText || "",
activityPlan: parsedResponse.activityPlan || "",
dietPlan: parsedResponse.dietPlan || "",
status: "approved",
generatedAt: new Date(),
updatedAt: new Date(),
});
const existingActiveGoals = await db.getFitnessGoalsByUserId(
userId,
"active",
);
const linkedGoals = existingActiveGoals.filter((goal) =>
goal.notes?.startsWith(AI_LINK_PREFIX),
);
await Promise.all(
linkedGoals.map((goal) =>
db.updateFitnessGoal(goal.id, {
status: "paused",
notes: `${goal.notes || ""}\nPaused due to new AI plan generation on ${new Date().toISOString()}`,
}),
),
);
let planItems = parseActivityPlanToItems(parsedResponse.activityPlan || "");
if (planItems.length === 0 && parsedResponse.recommendationText) {
planItems = parseActivityPlanToItems(parsedResponse.recommendationText);
}
if (planItems.length === 0) {
planItems = getDefaultPlanItems();
}
log.debug("AI plan parsed into goal items", {
recommendationId: recommendation.id,
userId,
parsedItems: planItems.length,
});
const createdGoals = await Promise.all(
planItems.map((item) =>
db.createFitnessGoal({
id: crypto.randomUUID(),
userId,
fitnessProfileId: profile.id,
goalType: item.goalType,
title: item.title,
description: item.description,
targetValue: undefined,
currentValue: 0,
unit: undefined,
startDate: new Date(),
targetDate: undefined,
completedDate: undefined,
status: "active",
progress: 0,
priority: "medium",
notes: `${AI_LINK_PREFIX} recommendationId=${recommendation.id}; itemId=${item.id}`,
}),
),
);
return NextResponse.json({
success: true,
data: recommendation,
meta: {
timestamp: new Date().toISOString(),
createdGoals: createdGoals.length,
pausedGoals: linkedGoals.length,
usedFallbackPlan,
},
});
} catch (error) {
log.error("Failed to self-generate recommendation", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,85 @@
/**
* @jest-environment node
*/
import { POST } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
}));
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@/lib/ai/ai-context", () => ({
buildAIContext: jest.fn(),
}));
jest.mock("@/lib/ai/prompt-builder", () => ({
buildEnhancedPrompt: jest.fn(),
buildBasicPrompt: jest.fn(),
}));
jest.mock("@/lib/logger", () => ({
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
debug: jest.fn(),
}));
describe("POST /api/recommendations/generate authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database").getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
const mockDb = {
getUserById: jest.fn(),
getFitnessProfileByUserId: jest.fn(),
};
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue(mockDb);
});
it("returns 401 when unauthenticated", async () => {
mockAuth.mockResolvedValue({ userId: null });
const req = new Request("http://localhost/api/recommendations/generate", {
method: "POST",
body: JSON.stringify({ userId: "client_1" }),
headers: { "Content-Type": "application/json" },
});
const res = await POST(req);
expect(res.status).toBe(401);
});
it("returns 403 when staff accesses user from another gym", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
mockDb.getUserById.mockResolvedValue({
id: "client_1",
gymId: "gym_b",
});
const req = new Request("http://localhost/api/recommendations/generate", {
method: "POST",
body: JSON.stringify({ userId: "client_1" }),
headers: { "Content-Type": "application/json" },
});
const res = await POST(req);
expect(res.status).toBe(403);
});
});

View File

@ -1,11 +1,19 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { buildAIContext } from "@/lib/ai/ai-context";
import { buildEnhancedPrompt, buildBasicPrompt } from "@/lib/ai/prompt-builder";
import log from "@/lib/logger";
import { ensureUserSynced } from "@/lib/sync-user";
import { getUserMembershipContext } from "@/lib/membership/access";
export async function POST(req: Request) {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { userId, useExternalModel, modelProvider } = await req.json();
if (!userId) {
@ -22,6 +30,69 @@ export async function POST(req: Request) {
});
const db = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const canGenerateRecommendations =
currentUser.role === "superAdmin" ||
currentUser.role === "admin" ||
currentUser.role === "trainer";
if (!canGenerateRecommendations) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const targetUser = await db.getUserById(userId);
if (!targetUser) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
const { membershipType, features } = await getUserMembershipContext(userId);
if (features.recommendationsPerMonth === 1) {
const currentMonth = new Date();
const monthStart = new Date(
currentMonth.getFullYear(),
currentMonth.getMonth(),
1,
);
const monthEnd = new Date(
currentMonth.getFullYear(),
currentMonth.getMonth() + 1,
1,
);
const existingRecommendations =
await db.getRecommendationsByUserId(userId);
const recommendationsThisMonth = existingRecommendations.filter(
(recommendation) =>
recommendation.generatedAt >= monthStart &&
recommendation.generatedAt < monthEnd,
).length;
if (recommendationsThisMonth >= 1) {
return NextResponse.json(
{
error:
"Basic membership includes 1 recommendation per month. Upgrade to Premium or VIP for unlimited recommendations.",
membershipType,
},
{ status: 403 },
);
}
}
if (currentUser.role !== "superAdmin") {
if (!currentUser.gymId || targetUser.gymId !== currentUser.gymId) {
return NextResponse.json(
{ error: "Forbidden - Cannot access users from other gyms" },
{ status: 403 },
);
}
}
// Fetch fitness profile
const profile = await db.getFitnessProfileByUserId(userId);

View File

@ -17,6 +17,8 @@ import {
badRequestResponse,
internalErrorResponse,
} from "@/lib/api/responses";
import { getRecommendationsByGym } from "@/lib/gym-context";
import { ensureUserSynced } from "@/lib/sync-user";
export async function GET(request: NextRequest) {
try {
@ -25,38 +27,59 @@ export async function GET(request: NextRequest) {
return unauthorizedResponse();
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(currentUserId, db);
if (!currentUser) {
return forbiddenResponse("User not found");
}
const { searchParams } = new URL(request.url);
const targetUserId = searchParams.get("userId");
const db = await getDatabase();
// If no userId provided, check if staff and return all recommendations
// If no userId provided, staff gets gym-scoped recommendations
if (!targetUserId) {
const currentUser = await db.getUserById(currentUserId);
const isStaff =
currentUser?.role === "admin" ||
currentUser?.role === "superAdmin" ||
currentUser?.role === "trainer";
currentUser.role === "admin" ||
currentUser.role === "superAdmin" ||
currentUser.role === "trainer";
if (!isStaff) {
return badRequestResponse("User ID is required");
}
const recommendations = await db.getAllRecommendations();
// Get target gym based on role
const targetGymId =
currentUser.role === "superAdmin"
? (searchParams.get("gymId") ?? undefined)
: (currentUser.gymId ?? undefined);
// Get recommendations filtered by gym
const recommendations = targetGymId
? await getRecommendationsByGym(targetGymId)
: await db.getAllRecommendations();
return successResponse({ recommendations });
}
// Check permissions: Users can view their own, Admins/Trainers can view anyone's
const currentUser = await db.getUserById(currentUserId);
// Check permissions: Users can view their own, Admins/Trainers can view anyone's in their gym
const isStaff =
currentUser?.role === "admin" ||
currentUser?.role === "superAdmin" ||
currentUser?.role === "trainer";
currentUser.role === "admin" ||
currentUser.role === "superAdmin" ||
currentUser.role === "trainer";
if (currentUserId !== targetUserId) {
if (!isStaff) {
return forbiddenResponse();
}
// Staff need to verify target user is in same gym
if (currentUser.role !== "superAdmin") {
const targetUser = await db.getUserById(targetUserId);
if (!targetUser || targetUser.gymId !== currentUser.gymId) {
return forbiddenResponse("Cannot access users from other gyms");
}
}
}
let recommendations = await db.getRecommendationsByUserId(targetUserId);
@ -83,7 +106,7 @@ export async function POST(request: NextRequest) {
}
const db = await getDatabase();
const currentUser = await db.getUserById(currentUserId);
const currentUser = await ensureUserSynced(currentUserId, db);
const isStaff =
currentUser?.role === "admin" ||
currentUser?.role === "superAdmin" ||
@ -117,6 +140,18 @@ export async function POST(request: NextRequest) {
content,
} = validation.data;
const targetUser = await db.getUserById(userId);
if (!targetUser) {
return badRequestResponse("Target user not found");
}
if (
currentUser?.role !== "superAdmin" &&
(!currentUser?.gymId || targetUser.gymId !== currentUser.gymId)
) {
return forbiddenResponse("Cannot create recommendations for other gyms");
}
// Handle AI Plan (Legacy/Specific)
if (recommendationText && activityPlan && dietPlan && fitnessProfileId) {
const recommendation = await db.createRecommendation({
@ -175,6 +210,41 @@ export async function PUT(request: NextRequest) {
validation.data;
const db = await getDatabase();
const currentUser = await ensureUserSynced(currentUserId, db);
if (!currentUser) {
return forbiddenResponse("User not found");
}
const isStaff =
currentUser.role === "admin" ||
currentUser.role === "superAdmin" ||
currentUser.role === "trainer";
if (!isStaff) {
return forbiddenResponse();
}
const existingRecommendation = (await db.getAllRecommendations()).find(
(recommendation) => recommendation.id === id,
);
if (!existingRecommendation) {
return badRequestResponse("Recommendation not found");
}
if (currentUser.role !== "superAdmin") {
const targetUser = await db.getUserById(existingRecommendation.userId);
if (
!currentUser.gymId ||
!targetUser ||
targetUser.gymId !== currentUser.gymId
) {
return forbiddenResponse(
"Cannot modify recommendations for other gyms",
);
}
}
const updated = await db.updateRecommendation(id, {
...(status && { status }),

View File

@ -0,0 +1,373 @@
/**
* Report Generation API Tests
*
* Tests for /api/reports/user/[userId] endpoint
*
* Test Coverage:
* 1. Access Control (client, trainer, admin, superAdmin)
* 2. Data Aggregation (attendance, nutrition, hydration, goals)
* 3. PDF Generation
* 4. Date Range Validation
*/
import { describe, it, expect, beforeAll, afterAll } from "@jest/globals";
// Mock database
const mockDatabase = {
getUserById: jest.fn(),
getClientByUserId: jest.fn(),
getFitnessProfileByUserId: jest.fn(),
getAttendanceByWeek: jest.fn(),
getDailyNutritionRange: jest.fn(),
getDailyHydrationRange: jest.fn(),
getFitnessGoalsByUserId: jest.fn(),
getFitnessProfileHistory: jest.fn(),
getRecommendationsByUserId: jest.fn(),
getTrainerClientAssignments: jest.fn(),
};
// Mock user data
const mockUsers = {
client: {
id: "user_client_123",
email: "client@example.com",
firstName: "John",
lastName: "Client",
role: "client",
gymId: "gym_123",
},
trainer: {
id: "user_trainer_456",
email: "trainer@example.com",
firstName: "Jane",
lastName: "Trainer",
role: "trainer",
gymId: "gym_123",
},
admin: {
id: "user_admin_789",
email: "admin@example.com",
firstName: "Admin",
lastName: "User",
role: "admin",
gymId: "gym_123",
},
superAdmin: {
id: "user_superadmin_000",
email: "superadmin@example.com",
firstName: "Super",
lastName: "Admin",
role: "superAdmin",
gymId: undefined,
},
};
describe("Report Generation API", () => {
describe("Access Control", () => {
it("should allow clients to view their own reports", async () => {
const currentUser = mockUsers.client;
const targetUserId = mockUsers.client.id;
// Users can always view their own reports
const hasAccess = currentUser.id === targetUserId;
expect(hasAccess).toBe(true);
});
it("should allow trainers to view assigned client reports", async () => {
const currentUser = mockUsers.trainer;
const targetUserId = mockUsers.client.id;
// Mock assignment exists
mockDatabase.getTrainerClientAssignments.mockResolvedValue([
{
trainerId: currentUser.id,
clientId: targetUserId,
isActive: true,
},
]);
const assignments = await mockDatabase.getTrainerClientAssignments(
currentUser.id,
);
const assignment = assignments.find(
(a: any) => a.clientId === targetUserId && a.isActive,
);
const hasAccess = !!assignment;
expect(hasAccess).toBe(true);
});
it("should deny trainers access to non-assigned client reports", async () => {
const currentUser = mockUsers.trainer;
const targetUserId = "user_other_client_999";
mockDatabase.getTrainerClientAssignments.mockResolvedValue([]);
const assignments = await mockDatabase.getTrainerClientAssignments(
currentUser.id,
);
const assignment = assignments.find(
(a: any) => a.clientId === targetUserId && a.isActive,
);
const hasAccess = !!assignment;
expect(hasAccess).toBe(false);
});
it("should allow admins to view clients in their gym", async () => {
const currentUser = mockUsers.admin;
const targetUser = { ...mockUsers.client, gymId: "gym_123" };
mockDatabase.getUserById.mockResolvedValue(targetUser);
const user = await mockDatabase.getUserById(targetUser.id);
const hasAccess = user?.gymId === currentUser.gymId;
expect(hasAccess).toBe(true);
});
it("should deny admins access to clients outside their gym", async () => {
const currentUser = mockUsers.admin;
const targetUser = { ...mockUsers.client, gymId: "gym_other_999" };
mockDatabase.getUserById.mockResolvedValue(targetUser);
const user = await mockDatabase.getUserById(targetUser.id);
const hasAccess = user?.gymId === currentUser.gymId;
expect(hasAccess).toBe(false);
});
it("should allow superAdmins to view any user", async () => {
const currentUser = mockUsers.superAdmin;
// SuperAdmins bypass gym checks
const hasAccess = currentUser.role === "superAdmin";
expect(hasAccess).toBe(true);
});
it("should deny clients access to other users reports", async () => {
const currentUser = mockUsers.client;
const targetUserId = "user_other_123";
// Only own ID or specific permissions
const hasAccess = currentUser.id === targetUserId;
expect(hasAccess).toBe(false);
});
});
describe("Date Range Validation", () => {
it("should accept valid date format YYYY-MM-DD", () => {
const validDates = ["2024-01-01", "2024-12-31", "2024-06-15"];
validDates.forEach((date) => {
const parsed = Date.parse(date);
expect(isNaN(parsed)).toBe(false);
});
});
it("should reject invalid date formats", () => {
const invalidDates = ["01-01-2024", "2024/01/01", "2024-1-1", "invalid"];
invalidDates.forEach((date) => {
const parsed = Date.parse(date);
expect(isNaN(parsed)).toBe(true);
});
});
it("should default to last 30 days if no dates provided", () => {
const endDate = new Date();
const startDate = new Date(Date.now() - 30 * 24 * 60 * 60 * 1000);
const daysDiff = Math.ceil(
(endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24),
);
expect(daysDiff).toBe(30);
});
});
describe("Data Aggregation", () => {
it("should aggregate weekly check-ins correctly", () => {
const attendanceRecords = [
{
checkInTime: new Date("2024-01-02T10:00:00"),
checkOutTime: new Date("2024-01-02T11:00:00"),
type: "gym",
},
{
checkInTime: new Date("2024-01-03T14:00:00"),
checkOutTime: new Date("2024-01-03T15:30:00"),
type: "gym",
},
{
checkInTime: new Date("2024-01-04T09:00:00"),
checkOutTime: null, // Still checked in
type: "class",
},
];
const totalCheckIns = attendanceRecords.length;
const completedCheckIns = attendanceRecords.filter(
(r) => r.checkOutTime,
).length;
expect(totalCheckIns).toBe(3);
expect(completedCheckIns).toBe(2);
});
it("should calculate total time spent correctly", () => {
const attendanceRecords = [
{
checkInTime: new Date("2024-01-02T10:00:00"),
checkOutTime: new Date("2024-01-02T11:30:00"), // 90 minutes
},
{
checkInTime: new Date("2024-01-03T14:00:00"),
checkOutTime: new Date("2024-01-03T15:00:00"), // 60 minutes
},
];
const totalMinutes = attendanceRecords.reduce((sum, record) => {
if (record.checkOutTime) {
const duration =
record.checkOutTime.getTime() - record.checkInTime.getTime();
return sum + Math.floor(duration / 60000);
}
return sum;
}, 0);
expect(totalMinutes).toBe(150);
});
it("should handle nutrition goal achievement correctly", () => {
const nutritionRecords = [
{ date: "2024-01-01", totalCalories: 2100, calorieGoal: 2000 },
{ date: "2024-01-02", totalCalories: 1900, calorieGoal: 2000 },
{ date: "2024-01-03", totalCalories: 2200, calorieGoal: 2000 },
];
const daysMetGoal = nutritionRecords.filter((record) => {
const lowerBound = record.calorieGoal * 0.9;
const upperBound = record.calorieGoal * 1.1;
return (
record.totalCalories >= lowerBound &&
record.totalCalories <= upperBound
);
}).length;
expect(daysMetGoal).toBe(3); // All within ±10%
});
it("should handle hydration goal achievement correctly", () => {
const hydrationRecords = [
{ date: "2024-01-01", totalWater: 2500, waterGoal: 2000 }, // 125%
{ date: "2024-01-02", totalWater: 1800, waterGoal: 2000 }, // 90%
{ date: "2024-01-03", totalWater: 2000, waterGoal: 2000 }, // 100%
];
const daysMetGoal = hydrationRecords.filter(
(record) => record.totalWater / record.waterGoal >= 1,
).length;
expect(daysMetGoal).toBe(2); // Days 1 and 3 meet goal
});
});
describe("Report Structure", () => {
it("should include all required sections", () => {
const requiredSections = [
"userId",
"user",
"client",
"fitnessProfile",
"reportPeriod",
"weeklyCheckIns",
"nutrition",
"hydration",
"goals",
"profileHistory",
"recommendations",
"generatedAt",
];
// Mock report object
const report = {
userId: "user_123",
user: {},
client: null,
fitnessProfile: null,
reportPeriod: { startDate: "", endDate: "" },
weeklyCheckIns: [],
nutrition: {
dailySummaries: [],
averageDailyCalories: 0,
totalDays: 0,
daysMetGoal: 0,
},
hydration: {
dailySummaries: [],
averageDailyWater: 0,
totalDays: 0,
daysMetGoal: 0,
},
goals: {
active: [],
completed: [],
totalActive: 0,
totalCompleted: 0,
averageProgress: 0,
},
profileHistory: [],
recommendations: {
accepted: [],
rejected: [],
pending: [],
totalAccepted: 0,
totalRejected: 0,
totalPending: 0,
},
generatedAt: new Date(),
};
requiredSections.forEach((section) => {
expect(report).toHaveProperty(section);
});
});
});
});
describe("ISO Week Calculations", () => {
it("should start week on Monday", () => {
// Test dates
const dates = [
{ date: "2024-01-01", expectedMonday: "2024-01-01" }, // Monday
{ date: "2024-01-02", expectedMonday: "2024-01-01" }, // Tuesday
{ date: "2024-01-03", expectedMonday: "2024-01-01" }, // Wednesday
{ date: "2024-01-07", expectedMonday: "2024-01-01" }, // Sunday
{ date: "2024-01-08", expectedMonday: "2024-01-08" }, // Monday (next week)
];
dates.forEach(({ date, expectedMonday }) => {
const current = new Date(date);
const day = current.getDay();
const diff = current.getDate() - day + (day === 0 ? -6 : 1);
const monday = new Date(current);
monday.setDate(diff);
const mondayStr = monday.toISOString().split("T")[0];
expect(mondayStr).toBe(expectedMonday);
});
});
it("should end week on Sunday", () => {
const monday = new Date("2024-01-01");
const sunday = new Date(monday);
sunday.setDate(sunday.getDate() + 6);
const sundayStr = sunday.toISOString().split("T")[0];
expect(sundayStr).toBe("2024-01-07");
});
});

View File

@ -0,0 +1,425 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
import type {
UserReport,
WeeklyCheckInStats,
NutritionSummary,
HydrationSummary,
GoalSummary,
ProfileChangeSummary,
RecommendationSummary,
AttendanceType,
} from "@fitai/shared";
import { ATTENDANCE_TYPES } from "@fitai/shared";
import { generateReportPDFBase64 } from "@/lib/pdf/report-helpers";
/**
* GET /api/reports/user/[userId]
* Generate a comprehensive user report
*
* Query params:
* - userId: string (required) - Target user ID
* - startDate: YYYY-MM-DD (optional, default: 30 days ago)
* - endDate: YYYY-MM-DD (optional, default: today)
*
* Access Control:
* - Admins: Can generate reports for any user in their gym
* - Trainers: Can generate reports for their assigned clients only
* - Users: Can only generate reports for themselves
*/
export async function GET(
request: NextRequest,
{ params }: { params: Promise<{ userId: string }> },
) {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const { userId: targetUserId } = await params;
// Access Control Check
const hasAccess = await checkReportAccess(currentUser, targetUserId, db);
if (!hasAccess) {
return NextResponse.json(
{ error: "You don't have permission to view this report" },
{ status: 403 },
);
}
// Parse date range from query params
const { searchParams } = new URL(request.url);
const endDate =
searchParams.get("endDate") || new Date().toISOString().split("T")[0];
const startDate =
searchParams.get("startDate") ||
new Date(Date.now() - 30 * 24 * 60 * 60 * 1000)
.toISOString()
.split("T")[0];
const format = searchParams.get("format") || "json";
// Validate dates
if (isNaN(Date.parse(startDate)) || isNaN(Date.parse(endDate))) {
return NextResponse.json(
{ error: "Invalid date format. Use YYYY-MM-DD" },
{ status: 400 },
);
}
// Fetch all user data
const [user, client, fitnessProfile] = await Promise.all([
db.getUserById(targetUserId),
db.getClientByUserId(targetUserId),
db.getFitnessProfileByUserId(targetUserId),
]);
if (!user) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
// Aggregate data in parallel
const [
nutritionRecords,
hydrationRecords,
fitnessGoals,
profileHistory,
recommendations,
] = await Promise.all([
db.getDailyNutritionRange(targetUserId, startDate, endDate),
db.getDailyHydrationRange(targetUserId, startDate, endDate),
db.getFitnessGoalsByUserId(targetUserId),
db.getFitnessProfileHistory(
targetUserId,
new Date(startDate),
new Date(endDate),
),
db.getRecommendationsByUserId(targetUserId),
]);
// Process and aggregate data
const weeklyCheckIns = await processWeeklyCheckIns(
db,
targetUserId,
startDate,
endDate,
);
const nutrition = processNutrition(nutritionRecords);
const hydration = processHydration(hydrationRecords);
const goals = processGoals(fitnessGoals);
const profileChanges = processProfileHistory(profileHistory);
const recs = processRecommendations(recommendations);
const report: UserReport = {
userId: targetUserId,
user,
client,
fitnessProfile,
reportPeriod: {
startDate,
endDate,
},
weeklyCheckIns,
nutrition,
hydration,
goals,
profileHistory: profileChanges,
recommendations: recs,
generatedAt: new Date(),
};
// Return PDF if requested
if (format === "pdf") {
try {
const pdfBase64 = generateReportPDFBase64(report);
const pdfBuffer = Buffer.from(pdfBase64, "base64");
return new NextResponse(pdfBuffer, {
status: 200,
headers: {
"Content-Type": "application/pdf",
"Content-Disposition": `attachment; filename="FitAI_Report_${report.user.firstName}_${report.user.lastName}_${startDate}_${endDate}.pdf"`,
"Content-Length": pdfBuffer.length.toString(),
},
});
} catch (pdfError) {
log.error("Failed to generate PDF", pdfError);
return NextResponse.json(
{ error: "Failed to generate PDF" },
{ status: 500 },
);
}
}
return NextResponse.json(report);
} catch (error) {
log.error("Failed to generate user report", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
/**
* Check if the current user has access to view the target user's report
*/
async function checkReportAccess(
currentUser: { id: string; role: string; gymId?: string },
targetUserId: string,
db: Awaited<ReturnType<typeof getDatabase>>,
): Promise<boolean> {
// Users can always view their own reports
if (currentUser.id === targetUserId) {
return true;
}
// Admins can view any user's report in their gym
if (currentUser.role === "admin" || currentUser.role === "superAdmin") {
const targetUser = await db.getUserById(targetUserId);
if (!targetUser) return false;
// SuperAdmins can view all users
if (currentUser.role === "superAdmin") return true;
// Admins can only view users in their gym
if (currentUser.gymId && targetUser.gymId === currentUser.gymId) {
return true;
}
return false;
}
// Trainers can only view reports for their assigned clients
if (currentUser.role === "trainer") {
const assignments = await db.getTrainerClientAssignments(currentUser.id);
const assignment = assignments.find(
(a) => a.clientId === targetUserId && a.isActive,
);
return !!assignment;
}
// Regular users cannot view other users' reports
return false;
}
/**
* Process attendance records into weekly check-in stats
*/
async function processWeeklyCheckIns(
db: Awaited<ReturnType<typeof getDatabase>>,
userId: string,
startDate: string,
endDate: string,
): Promise<WeeklyCheckInStats[]> {
const weeks: WeeklyCheckInStats[] = [];
// Generate all weeks in the range
const current = new Date(startDate);
const end = new Date(endDate);
while (current <= end) {
// Get Monday of current week (ISO week)
const monday = new Date(current);
const day = monday.getDay();
const diff = monday.getDate() - day + (day === 0 ? -6 : 1);
monday.setDate(diff);
const weekStart = monday.toISOString().split("T")[0];
const sunday = new Date(monday);
sunday.setDate(sunday.getDate() + 6);
const weekEnd = sunday.toISOString().split("T")[0];
weeks.push({
weekStart,
weekEnd,
totalCheckIns: 0,
totalTimeMinutes: 0,
averageDurationMinutes: 0,
checkInsByType: ATTENDANCE_TYPES.map((type) => ({ type, count: 0 })),
});
current.setDate(current.getDate() + 7);
}
// Fetch attendance for each week and aggregate
for (let i = 0; i < weeks.length; i++) {
const week = weeks[i];
const attendanceRecords = await db.getAttendanceByWeek(
userId,
new Date(week.weekStart),
);
// Filter records to only include those within the week
const weekStartDate = new Date(week.weekStart);
const weekEndDate = new Date(week.weekEnd);
weekEndDate.setHours(23, 59, 59, 999);
const filteredRecords = attendanceRecords.filter((record) => {
const checkInTime = new Date(record.checkInTime);
return checkInTime >= weekStartDate && checkInTime <= weekEndDate;
});
week.totalCheckIns = filteredRecords.length;
// Calculate time spent
for (const record of filteredRecords) {
if (record.checkOutTime) {
const checkIn = new Date(record.checkInTime);
const checkOut = new Date(record.checkOutTime);
const durationMs = checkOut.getTime() - checkIn.getTime();
const durationMinutes = Math.floor(durationMs / 60000);
week.totalTimeMinutes += durationMinutes;
}
// Count by type
const typeIndex = week.checkInsByType.findIndex(
(t) => t.type === record.type,
);
if (typeIndex !== -1) {
week.checkInsByType[typeIndex].count++;
}
}
// Calculate average duration
week.averageDurationMinutes =
week.totalCheckIns > 0
? Math.round(week.totalTimeMinutes / week.totalCheckIns)
: 0;
}
return weeks.sort((a, b) => a.weekStart.localeCompare(b.weekStart));
}
/**
* Process nutrition records into summaries
*/
function processNutrition(records: any[]): UserReport["nutrition"] {
const dailySummaries: NutritionSummary[] = records.map((record) => ({
date: record.date,
totalCalories: record.totalCalories,
calorieGoal: record.calorieGoal,
caloriesDelta: record.totalCalories - record.calorieGoal,
mealsCount: record.meals?.length || 0,
}));
const totalDays = dailySummaries.length;
const totalCalories = dailySummaries.reduce(
(sum, day) => sum + day.totalCalories,
0,
);
const averageDailyCalories =
totalDays > 0 ? Math.round(totalCalories / totalDays) : 0;
// Count days where calories were within ±10% of goal
const daysMetGoal = dailySummaries.filter((day) => {
const lowerBound = day.calorieGoal * 0.9;
const upperBound = day.calorieGoal * 1.1;
return day.totalCalories >= lowerBound && day.totalCalories <= upperBound;
}).length;
return {
dailySummaries,
averageDailyCalories,
totalDays,
daysMetGoal,
};
}
/**
* Process hydration records into summaries
*/
function processHydration(records: any[]): UserReport["hydration"] {
const dailySummaries: HydrationSummary[] = records.map((record) => ({
date: record.date,
totalWater: record.totalWater,
waterGoal: record.waterGoal,
hydrationPercentage:
record.waterGoal > 0
? Math.round((record.totalWater / record.waterGoal) * 100)
: 0,
}));
const totalDays = dailySummaries.length;
const totalWater = dailySummaries.reduce(
(sum, day) => sum + day.totalWater,
0,
);
const averageDailyWater =
totalDays > 0 ? Math.round(totalWater / totalDays) : 0;
// Count days where water goal was met (>= 100%)
const daysMetGoal = dailySummaries.filter(
(day) => day.hydrationPercentage >= 100,
).length;
return {
dailySummaries,
averageDailyWater,
totalDays,
daysMetGoal,
};
}
/**
* Process fitness goals into summaries
*/
function processGoals(goals: any[]): GoalSummary {
const active = goals.filter((g) => g.status === "active");
const completed = goals.filter((g) => g.status === "completed");
const averageProgress =
active.length > 0
? Math.round(
active.reduce((sum, g) => sum + g.progress, 0) / active.length,
)
: 0;
return {
active,
completed,
totalActive: active.length,
totalCompleted: completed.length,
averageProgress,
};
}
/**
* Process fitness profile history into summaries
*/
function processProfileHistory(history: any[]): ProfileChangeSummary[] {
return history.map((record) => ({
changeType: record.changeType,
fieldName: record.fieldName,
previousValue: record.previousValue,
newValue: record.newValue,
changedAt: record.changedAt,
}));
}
/**
* Process recommendations into summaries
*/
function processRecommendations(recommendations: any[]): RecommendationSummary {
const accepted = recommendations.filter((r) => r.status === "accepted");
const rejected = recommendations.filter((r) => r.status === "rejected");
const pending = recommendations.filter((r) => r.status === "pending");
return {
accepted,
rejected,
pending,
totalAccepted: accepted.length,
totalRejected: rejected.length,
totalPending: pending.length,
};
}

View File

@ -0,0 +1,97 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
/**
* DELETE /api/trainer-client/[id]
* Delete (deactivate) a trainer-client assignment
*/
export async function DELETE(
request: NextRequest,
{ params }: { params: Promise<{ id: string }> },
) {
try {
const { userId: authUserId } = await auth();
if (!authUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(authUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
// Only admins can delete assignments
if (currentUser.role !== "admin" && currentUser.role !== "superAdmin") {
return NextResponse.json(
{ error: "Only admins can remove trainer-client assignments" },
{ status: 403 },
);
}
const { id } = await params;
const allAssignments = await db.getAllTrainerClientAssignments();
const assignment = allAssignments.find((a) => a.id === id);
if (!assignment) {
return NextResponse.json(
{ error: "Assignment not found" },
{ status: 404 },
);
}
if (currentUser.role !== "superAdmin") {
if (!currentUser.gymId) {
return NextResponse.json({ error: "No gym assigned" }, { status: 403 });
}
const [trainer, client] = await Promise.all([
db.getUserById(assignment.trainerId),
db.getUserById(assignment.clientId),
]);
if (
!trainer ||
!client ||
trainer.gymId !== currentUser.gymId ||
client.gymId !== currentUser.gymId
) {
return NextResponse.json(
{ error: "Cannot modify assignments from other gyms" },
{ status: 403 },
);
}
}
// Deactivate the assignment
const result = await db.deactivateTrainerClientAssignment(id);
if (!result) {
return NextResponse.json(
{ error: "Failed to deactivate assignment" },
{ status: 500 },
);
}
log.info("Trainer-client assignment deactivated", {
assignmentId: id,
deactivatedBy: currentUser.id,
});
return NextResponse.json({
success: true,
message: "Assignment deactivated successfully",
});
} catch (error) {
log.error("Failed to delete trainer-client assignment", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,89 @@
/**
* @jest-environment node
*/
import { NextRequest } from "next/server";
import { GET, POST } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
}));
jest.mock("@/lib/database", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@/lib/logger", () => ({
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
debug: jest.fn(),
}));
describe("/api/trainer-client authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database").getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
const mockDb = {
getUserById: jest.fn(),
getTrainerClientAssignments: jest.fn(),
getAllTrainerClientAssignments: jest.fn(),
createTrainerClientAssignment: jest.fn(),
};
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue(mockDb);
});
it("blocks non-admin users from listing assignments", async () => {
mockAuth.mockResolvedValue({ userId: "trainer_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "trainer_1",
role: "trainer",
gymId: "gym_a",
});
const request = new NextRequest("http://localhost/api/trainer-client");
const response = await GET(request);
expect(response.status).toBe(403);
});
it("blocks admins from creating assignments across gyms", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
mockDb.getUserById
.mockResolvedValueOnce({
id: "trainer_2",
role: "trainer",
gymId: "gym_b",
})
.mockResolvedValueOnce({
id: "client_2",
role: "client",
gymId: "gym_b",
});
const request = new NextRequest("http://localhost/api/trainer-client", {
method: "POST",
body: JSON.stringify({ trainerId: "trainer_2", clientId: "client_2" }),
});
const response = await POST(request);
expect(response.status).toBe(403);
expect(mockDb.createTrainerClientAssignment).not.toHaveBeenCalled();
});
});

View File

@ -0,0 +1,264 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
import log from "@/lib/logger";
/**
* GET /api/trainer-client
* Get trainer-client assignments
*
* Query params:
* - trainerId: string (optional) - Filter by trainer
* - clientId: string (optional) - Filter by client
*/
export async function GET(request: NextRequest) {
try {
const { userId: authUserId } = await auth();
if (!authUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(authUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
// Only admins and superadmins can view all assignments
if (currentUser.role !== "admin" && currentUser.role !== "superAdmin") {
return NextResponse.json(
{ error: "Only admins can view trainer-client assignments" },
{ status: 403 },
);
}
const { searchParams } = new URL(request.url);
const trainerId = searchParams.get("trainerId");
const clientId = searchParams.get("clientId");
if (trainerId && clientId) {
const [trainer, client] = await Promise.all([
db.getUserById(trainerId),
db.getUserById(clientId),
]);
if (!trainer || !client) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
if (
currentUser.role !== "superAdmin" &&
(!currentUser.gymId ||
trainer.gymId !== currentUser.gymId ||
client.gymId !== currentUser.gymId)
) {
return NextResponse.json(
{ error: "Cannot access assignments from other gyms" },
{ status: 403 },
);
}
}
if (trainerId && !clientId) {
const trainer = await db.getUserById(trainerId);
if (!trainer) {
return NextResponse.json(
{ error: "Trainer not found" },
{ status: 404 },
);
}
if (
currentUser.role !== "superAdmin" &&
(!currentUser.gymId || trainer.gymId !== currentUser.gymId)
) {
return NextResponse.json(
{ error: "Cannot access assignments from other gyms" },
{ status: 403 },
);
}
}
if (!trainerId && clientId) {
const client = await db.getUserById(clientId);
if (!client) {
return NextResponse.json(
{ error: "Client not found" },
{ status: 404 },
);
}
if (
currentUser.role !== "superAdmin" &&
(!currentUser.gymId || client.gymId !== currentUser.gymId)
) {
return NextResponse.json(
{ error: "Cannot access assignments from other gyms" },
{ status: 403 },
);
}
}
let assignments = trainerId
? await db.getTrainerClientAssignments(trainerId)
: await db.getAllTrainerClientAssignments();
if (clientId) {
assignments = assignments.filter((a) => a.clientId === clientId);
}
if (currentUser.role !== "superAdmin") {
if (!currentUser.gymId) {
return NextResponse.json({ error: "No gym assigned" }, { status: 403 });
}
const involvedUserIds = Array.from(
new Set(
assignments.flatMap((assignment) => [
assignment.trainerId,
assignment.clientId,
]),
),
);
const users = await Promise.all(
involvedUserIds.map((userId) => db.getUserById(userId)),
);
const gymByUserId = new Map(
users
.filter((user): user is NonNullable<typeof user> => !!user)
.map((user) => [user.id, user.gymId]),
);
assignments = assignments.filter((assignment) => {
const trainerGymId = gymByUserId.get(assignment.trainerId);
const clientGymId = gymByUserId.get(assignment.clientId);
return (
trainerGymId === currentUser.gymId &&
clientGymId === currentUser.gymId
);
});
}
return NextResponse.json({ assignments });
} catch (error) {
log.error("Failed to get trainer-client assignments", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
/**
* POST /api/trainer-client
* Create a new trainer-client assignment
*/
export async function POST(request: NextRequest) {
try {
const { userId: authUserId } = await auth();
if (!authUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(authUserId, db);
if (!currentUser) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
// Only admins can create assignments
if (currentUser.role !== "admin" && currentUser.role !== "superAdmin") {
return NextResponse.json(
{ error: "Only admins can assign trainers to clients" },
{ status: 403 },
);
}
const body = await request.json();
const { trainerId, clientId } = body;
if (!trainerId || !clientId) {
return NextResponse.json(
{ error: "trainerId and clientId are required" },
{ status: 400 },
);
}
// Verify trainer exists and has trainer role
const trainer = await db.getUserById(trainerId);
if (!trainer) {
return NextResponse.json({ error: "Trainer not found" }, { status: 404 });
}
if (trainer.role !== "trainer" && trainer.role !== "admin") {
return NextResponse.json(
{ error: "User must be a trainer or admin" },
{ status: 400 },
);
}
// Verify client exists
const client = await db.getUserById(clientId);
if (!client) {
return NextResponse.json({ error: "Client not found" }, { status: 404 });
}
if (client.role !== "client") {
return NextResponse.json(
{ error: "User must be a client" },
{ status: 400 },
);
}
if (
currentUser.role !== "superAdmin" &&
(!currentUser.gymId ||
trainer.gymId !== currentUser.gymId ||
client.gymId !== currentUser.gymId)
) {
return NextResponse.json(
{ error: "Cannot assign users from other gyms" },
{ status: 403 },
);
}
// Check if assignment already exists
const existingAssignments = await db.getTrainerClientAssignments(trainerId);
const existingAssignment = existingAssignments.find(
(a) => a.clientId === clientId && a.isActive,
);
if (existingAssignment) {
return NextResponse.json(
{ error: "Trainer is already assigned to this client" },
{ status: 409 },
);
}
// Create the assignment
const assignment = await db.createTrainerClientAssignment({
trainerId,
clientId,
assignedAt: new Date(),
assignedBy: currentUser.id,
isActive: true,
});
log.info("Trainer-client assignment created", {
trainerId,
clientId,
assignedBy: currentUser.id,
});
return NextResponse.json({ assignment }, { status: 201 });
} catch (error) {
log.error("Failed to create trainer-client assignment", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,109 @@
/**
* @jest-environment node
*/
import { NextRequest } from "next/server";
import { DELETE } from "../route";
jest.mock("@clerk/nextjs/server", () => ({
auth: jest.fn(),
clerkClient: jest.fn(),
}));
jest.mock("@/lib/database/index", () => ({
getDatabase: jest.fn(),
}));
jest.mock("@/lib/sync-user", () => ({
ensureUserSynced: jest.fn(),
}));
jest.mock("@fitai/database", () => ({
db: {
all: jest.fn(),
get: jest.fn(),
run: jest.fn(),
},
sql: jest.fn((strings: TemplateStringsArray) => strings.join("")),
}));
jest.mock("@/lib/logger", () => ({
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
debug: jest.fn(),
}));
describe("DELETE /api/users authz", () => {
const mockAuth = require("@clerk/nextjs/server").auth as jest.Mock;
const mockGetDatabase = require("@/lib/database/index")
.getDatabase as jest.Mock;
const mockEnsureUserSynced = require("@/lib/sync-user")
.ensureUserSynced as jest.Mock;
const mockDb = {
getUserById: jest.fn(),
deleteUser: jest.fn(),
};
beforeEach(() => {
jest.clearAllMocks();
mockGetDatabase.mockResolvedValue(mockDb);
});
it("blocks self deletion", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
const request = new NextRequest("http://localhost/api/users?id=admin_1", {
method: "DELETE",
});
const response = await DELETE(request);
expect(response.status).toBe(403);
expect(mockDb.deleteUser).not.toHaveBeenCalled();
});
it("blocks cross-gym deletion for admin", async () => {
mockAuth.mockResolvedValue({ userId: "admin_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "admin_1",
role: "admin",
gymId: "gym_a",
});
mockDb.getUserById.mockResolvedValue({ id: "user_2", gymId: "gym_b" });
const request = new NextRequest("http://localhost/api/users?id=user_2", {
method: "DELETE",
});
const response = await DELETE(request);
expect(response.status).toBe(403);
expect(mockDb.deleteUser).not.toHaveBeenCalled();
});
it("allows superAdmin cross-gym deletion", async () => {
mockAuth.mockResolvedValue({ userId: "super_1" });
mockEnsureUserSynced.mockResolvedValue({
id: "super_1",
role: "superAdmin",
gymId: null,
});
mockDb.getUserById.mockResolvedValue({ id: "user_2", gymId: "gym_b" });
const request = new NextRequest("http://localhost/api/users?id=user_2", {
method: "DELETE",
});
const response = await DELETE(request);
const payload = await response.json();
expect(response.status).toBe(200);
expect(payload.success).toBe(true);
expect(mockDb.deleteUser).toHaveBeenCalledWith("user_2");
});
});

View File

@ -8,6 +8,8 @@ import {
errorResponse,
unauthorizedResponse,
} from "@/lib/api/responses";
import { getAuthContext } from "@/lib/auth/context";
import { getInvitableRoles } from "@/lib/auth/permissions";
import log from "@/lib/logger";
/**
@ -26,6 +28,10 @@ export async function POST(request: NextRequest) {
return unauthorizedResponse("Unauthorized");
}
// Get full auth context (role and gymId)
const authContext = await getAuthContext();
const { role: requesterRole, gymId: requesterGymId } = authContext;
const body = await request.json();
const validationResult = createUserSchema.safeParse(body);
@ -41,6 +47,33 @@ export async function POST(request: NextRequest) {
const data = validationResult.data;
// Validate role creation permission
const allowedRoles = getInvitableRoles(requesterRole);
if (!allowedRoles.includes(data.role)) {
const roleMessages: Record<string, string> = {
trainer:
"Trainers can only create clients. Contact an admin to create other roles.",
admin:
"Admins can create trainers and clients only. Contact a superAdmin to create other admins.",
client: "Clients cannot create users.",
};
return errorResponse(
roleMessages[requesterRole] ||
"You are not authorized to create this role",
{ status: 403 },
);
}
// Auto-assign gym for non-superAdmins
// SuperAdmins can choose any gym, others must use their own gym
const assignedGymId =
requesterRole === "superAdmin" ? data.gymId : requesterGymId;
// Validate that non-superAdmins have a gym assigned
if (!assignedGymId && requesterRole !== "superAdmin") {
return errorResponse("You are not assigned to a gym", { status: 400 });
}
// Note: Email is required in current schema
// TODO: Add schema migration to make email optional for direct creation
if (!data.email) {
@ -51,18 +84,37 @@ export async function POST(request: NextRequest) {
if (data.sendInvitation && data.email) {
try {
const client = await clerkClient();
// Build publicMetadata with consistent field names
const publicMetadata: Record<string, any> = {
role: data.role,
createdBy: userId,
gymId: assignedGymId ?? null,
};
// Determine redirect URL based on role
// Staff (admin, trainer, superAdmin) → Admin web app
// Clients → Mobile app (handled by Clerk's default)
const isStaffRole = ["admin", "trainer", "superAdmin"].includes(
data.role,
);
const redirectUrl = isStaffRole
? `${process.env.NEXT_PUBLIC_APP_URL || "http://localhost:3000"}/sign-up`
: undefined; // Clients use default Clerk redirect
const invitation = await client.invitations.createInvitation({
emailAddress: data.email,
publicMetadata: {
role: data.role,
gymId: data.gymId,
},
publicMetadata,
redirectUrl,
ignoreExisting: true, // Don't fail if invitation already exists
});
log.info("Clerk invitation sent", {
email: data.email,
role: data.role,
gymId: assignedGymId,
invitationId: invitation.id,
redirectUrl: redirectUrl || "default (mobile app)",
});
// Send custom invitation email (in addition to Clerk's)
@ -83,7 +135,22 @@ export async function POST(request: NextRequest) {
} catch (error: unknown) {
const errorMessage =
error instanceof Error ? error.message : "Unknown error";
log.error("Failed to send Clerk invitation", error);
// Extract Clerk-specific error details
const clerkError = error as any;
log.error("Failed to send Clerk invitation", error, {
email: data.email,
role: data.role,
gymId: assignedGymId,
publicMetadata: assignedGymId
? { role: data.role, gymId: assignedGymId }
: { role: data.role },
clerkStatus: clerkError?.status,
clerkErrors: clerkError?.errors,
clerkTraceId: clerkError?.clerkTraceId,
});
return errorResponse(`Failed to send invitation: ${errorMessage}`, {
status: 500,
});
@ -106,13 +173,14 @@ export async function POST(request: NextRequest) {
lastName: data.lastName,
role: data.role,
phone: data.phone,
gymId: data.gymId,
gymId: assignedGymId,
})
.returning();
log.info("User created in database", {
userId: newUser.id,
role: data.role,
gymId: assignedGymId,
hasEmail: !!data.email,
});

View File

@ -1,8 +1,78 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { db, users as usersTable, eq, sql } from "@fitai/database";
import { ensureGymsGeofenceColumns } from "@/lib/geofence";
import log from "@/lib/logger";
export async function GET() {
try {
const { userId } = await auth();
if (!userId) return new NextResponse("Unauthorized", { status: 401 });
const user = await db
.select()
.from(usersTable)
.where(eq(usersTable.id, userId))
.get();
if (!user) {
return new NextResponse("User not found", { status: 404 });
}
if (!user.gymId) {
return NextResponse.json({ gymId: null, gym: null });
}
await ensureGymsGeofenceColumns();
const rows = await db.all(sql`
SELECT
id,
name,
location,
latitude,
longitude,
geofence_radius_meters as geofenceRadiusMeters,
geofence_enabled as geofenceEnabled,
status
FROM gyms
WHERE id = ${user.gymId}
LIMIT 1
`);
const gym = rows?.[0] as
| {
id: string;
name: string;
location: string | null;
latitude: number | null;
longitude: number | null;
geofenceRadiusMeters: number | null;
geofenceEnabled: number | boolean | null;
status: "active" | "inactive";
}
| undefined;
if (!gym || gym.status !== "active") {
return NextResponse.json({ gymId: user.gymId, gym: null });
}
return NextResponse.json({
gymId: user.gymId,
gym: {
...gym,
geofenceEnabled:
typeof gym.geofenceEnabled === "boolean"
? gym.geofenceEnabled
: Boolean(gym.geofenceEnabled),
},
});
} catch (error) {
log.error("Failed to fetch current user gym", error);
return new NextResponse("Internal Server Error", { status: 500 });
}
}
/**
* PATCH /api/users/gym
* Body: { gymId: string | null }

View File

@ -0,0 +1,32 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getDatabase } from "@/lib/database";
import { ensureUserSynced } from "@/lib/sync-user";
/**
* GET /api/users/me
* Get current authenticated user's information
*/
export async function GET() {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const db = await getDatabase();
const user = await ensureUserSynced(clerkUserId, db);
if (!user) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
return NextResponse.json({ user });
} catch (error) {
console.error("Failed to get current user:", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}

View File

@ -24,14 +24,60 @@ import {
internalErrorResponse,
badRequestResponse,
} from "@/lib/api/responses";
import { getUsersByGym } from "@/lib/gym-context";
import { ensureUserSynced } from "@/lib/sync-user";
export async function GET(request: NextRequest) {
try {
// First authenticate the user
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return unauthorizedResponse();
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return forbiddenResponse("User not found");
}
const { searchParams } = new URL(request.url);
const role = searchParams.get("role");
let users = await db.getAllUsers();
log.debug("User API called", {
currentUserId: currentUser.id,
currentUserRole: currentUser.role,
currentUserGymId: currentUser.gymId,
});
// Get target gym based on role
const targetGymId =
currentUser.role === "superAdmin"
? (searchParams.get("gymId") ?? undefined)
: (currentUser.gymId ?? undefined);
log.debug("Target gym calculation", {
targetGymId,
currentUserRole: currentUser.role,
currentUserGymId: currentUser.gymId,
});
// Validate gym access for non-superAdmins
if (currentUser.role !== "superAdmin" && !targetGymId) {
return forbiddenResponse("No gym assigned");
}
// Get users filtered by gym
let users = targetGymId
? await getUsersByGym(targetGymId)
: await db.getAllUsers();
log.debug("Users fetched from database", {
targetGymId,
totalUsers: Array.isArray(users) ? users.length : 0,
sampleGymId: users && users[0] ? (users[0] as any).gymId : null,
});
// Hydrate gymId from raw DB to ensure consistency with writes
const rawUserRows = await rawDb.all(sql`SELECT id, gym_id FROM users`);
@ -62,12 +108,12 @@ export async function GET(request: NextRequest) {
log.debug("Applied role filter", {
role,
usersAfterFilter: Array.isArray(users) ? users.length : 0,
sample:
sampleUser:
users && users[0]
? {
id: users[0].id,
role: users[0].role,
gymId: (users as any)[0].gymId,
gymId: (users[0] as any).gymId,
}
: null,
});
@ -303,8 +349,17 @@ export async function PUT(request: NextRequest) {
return validationErrorResponse(validation.errors);
}
const { id, email, firstName, lastName, role, phone, gymId } =
validation.data;
const {
id,
email,
firstName,
lastName,
role,
phone,
gymId,
membershipType,
membershipStatus,
} = validation.data;
log.debug("Updating user", {
id,
@ -314,6 +369,8 @@ export async function PUT(request: NextRequest) {
role,
phone,
gymId,
membershipType,
membershipStatus,
});
const db = await getDatabase();
@ -335,6 +392,27 @@ export async function PUT(request: NextRequest) {
return notFoundResponse("User not found");
}
log.debug("Edit authorization check", {
requesterId,
requesterRole: requester.role,
requesterGymId: requester.gymId,
targetUserId: id,
targetRole: existingUser.role,
targetGymId: existingUser.gymId,
requestedRole: role,
});
// Authorization: check gym-based permissions
// Non-superAdmins can only edit users in their own gym
if (requester.role !== "superAdmin") {
if (!requester.gymId) {
return forbiddenResponse("No gym assigned to requester");
}
if (existingUser.gymId !== requester.gymId) {
return forbiddenResponse("Cannot edit users from other gyms");
}
}
// Authorization: determine allowed role changes
const requesterRole = requester.role;
const allowedByRole: Record<string, string[]> = {
@ -345,10 +423,27 @@ export async function PUT(request: NextRequest) {
generalUser: [], // general users cannot change roles
};
if (role && !allowedByRole[requesterRole]?.includes(role)) {
// Only check authorization if the role is actually being changed
if (
role &&
role !== existingUser.role &&
!allowedByRole[requesterRole]?.includes(role)
) {
return forbiddenResponse(`Not authorized to assign role '${role}'`);
}
// Authorization: trainers and admins cannot reassign users to different gyms
// Only superAdmins can reassign gyms
if (
requesterRole !== "superAdmin" &&
gymId !== undefined &&
gymId !== existingUser.gymId
) {
return forbiddenResponse(
"Only superAdmins can reassign users to different gyms",
);
}
// Check if email is being changed and if it's already taken
if (email && email !== existingUser.email) {
const userWithEmail = await db.getUserByEmail(email);
@ -359,24 +454,27 @@ export async function PUT(request: NextRequest) {
// Update Clerk publicMetadata (role/gymId) to propagate via webhook
// Note: Only update metadata when a change is requested
try {
const client = await clerkClient();
const publicMetadata: Record<string, unknown> = {};
log.debug("Preparing Clerk metadata update", {
targetUserId: id,
role,
gymId,
});
const client = await clerkClient();
const publicMetadata: Record<string, unknown> = {};
log.debug("Preparing Clerk metadata update", {
targetUserId: id,
role,
gymId,
});
if (role) {
publicMetadata.role = role;
}
if (gymId !== undefined) {
publicMetadata.gymId = gymId === null ? null : String(gymId);
}
if (role) {
publicMetadata.role = role;
}
if (gymId !== undefined) {
publicMetadata.gymId = gymId === null ? null : String(gymId);
}
if (Object.keys(publicMetadata).length > 0) {
log.debug("Updating Clerk user metadata", { publicMetadata });
try {
// Check if user exists in Clerk first
await client.users.getUser(id);
if (Object.keys(publicMetadata).length > 0) {
log.debug("Updating Clerk user metadata", { publicMetadata });
const clerkResult = await client.users.updateUser(id, {
publicMetadata,
});
@ -385,14 +483,20 @@ export async function PUT(request: NextRequest) {
role: clerkResult.publicMetadata?.role,
gymId: clerkResult.publicMetadata?.gymId,
});
} else {
log.debug("No Clerk metadata changes requested");
} catch (clerkErr: any) {
// User might not exist in Clerk yet - that's OK for local users
if (
clerkErr?.status === 404 ||
clerkErr?.errors?.[0]?.code === "resource_not_found"
) {
log.debug(
"User not found in Clerk, skipping metadata update (local-only user)",
);
} else {
log.error("Clerk metadata update failed", clerkErr, { userId: id });
// Don't fail the whole request - local DB update can still proceed
}
}
} catch (clerkErr: any) {
log.error("Clerk metadata update failed", clerkErr, { userId: id });
return internalErrorResponse(
"Failed to update role/gym in identity provider",
);
}
// Update local DB for immediate UI feedback (webhook will also sync)
@ -414,6 +518,43 @@ export async function PUT(request: NextRequest) {
);
log.debug("User updated in database", { updatedRow });
// If the user is a client, update membership fields in clients table
const finalRole = role ?? existingUser.role;
if (
finalRole === "client" &&
(membershipType !== undefined || membershipStatus !== undefined)
) {
log.debug("Updating client membership fields", {
userId: id,
membershipType,
membershipStatus,
});
// Check if client record exists
const existingClient = await rawDb.get(
sql`SELECT * FROM clients WHERE user_id = ${id}`,
);
if (existingClient) {
// Update existing client record
await rawDb.run(
sql`UPDATE clients
SET membership_type = ${membershipType ?? (existingClient as any).membership_type},
membership_status = ${membershipStatus ?? (existingClient as any).membership_status},
updated_at = ${Date.now()}
WHERE user_id = ${id}`,
);
log.debug("Client membership updated");
} else {
// Create client record if it doesn't exist
await rawDb.run(
sql`INSERT INTO clients (id, user_id, membership_type, membership_status, join_date, created_at, updated_at)
VALUES (${`client_${id}_${Date.now()}`}, ${id}, ${membershipType ?? "basic"}, ${membershipStatus ?? "active"}, ${Date.now()}, ${Date.now()}, ${Date.now()})`,
);
log.debug("Client record created");
}
}
const updatedUser = {
...existingUser,
email: email ?? existingUser.email,
@ -439,26 +580,76 @@ export async function PUT(request: NextRequest) {
export async function DELETE(request: NextRequest) {
try {
const { userId: clerkUserId } = await auth();
if (!clerkUserId) {
return unauthorizedResponse();
}
const db = await getDatabase();
const currentUser = await ensureUserSynced(clerkUserId, db);
if (!currentUser) {
return forbiddenResponse("Current user not found");
}
const canDeleteUsers =
currentUser.role === "admin" || currentUser.role === "superAdmin";
if (!canDeleteUsers) {
return forbiddenResponse("Only admins can delete users");
}
const { searchParams } = new URL(request.url);
const id = searchParams.get("id");
const body = await request.json().catch(() => ({}));
const { ids } = body;
const targetIds: string[] = Array.isArray(ids)
? ids.filter(
(userId: unknown): userId is string => typeof userId === "string",
)
: id
? [id]
: [];
if (targetIds.length === 0) {
return badRequestResponse("User ID or IDs array required");
}
if (targetIds.includes(clerkUserId)) {
return forbiddenResponse("Cannot delete your own account");
}
const targetUsers = await Promise.all(
targetIds.map((targetId) => db.getUserById(targetId)),
);
const missingTargets = targetUsers.filter((user) => !user).length;
if (missingTargets > 0) {
return notFoundResponse("One or more users were not found");
}
if (currentUser.role !== "superAdmin") {
if (!currentUser.gymId) {
return forbiddenResponse("No gym assigned to current user");
}
const hasCrossGymTarget = targetUsers.some(
(targetUser) => targetUser && targetUser.gymId !== currentUser.gymId,
);
if (hasCrossGymTarget) {
return forbiddenResponse("Cannot delete users from other gyms");
}
}
if (ids && Array.isArray(ids)) {
// Bulk delete
await Promise.all(ids.map((userId: string) => db.deleteUser(userId)));
return successResponse({ deleted: ids.length });
} else if (id) {
// Single delete
const user = await db.getUserById(id);
if (!user) {
return notFoundResponse("User not found");
}
await db.deleteUser(id);
return successResponse({ deleted: 1 });
} else {
return badRequestResponse("User ID or IDs array required");
await db.deleteUser(id as string);
return successResponse({ deleted: 1 });
}
} catch (error) {
log.error("Failed to delete user(s)", error);

View File

@ -1,96 +1,102 @@
'use client'
"use client";
import { useState, useEffect } from 'react'
import { format } from 'date-fns'
interface Attendance {
id: string
clientId: string
checkInTime: string
checkOutTime?: string
type: string
notes?: string
}
import { useAttendance } from "@/hooks/use-api";
import { PageHeader } from "@/components/ui/PageHeader";
import { Badge } from "@/components/ui/badge";
export default function AttendancePage() {
const [attendance, setAttendance] = useState<Attendance[]>([])
const [loading, setLoading] = useState(true)
useEffect(() => {
const fetchAttendance = async () => {
try {
const res = await fetch('/api/admin/attendance')
if (res.ok) {
const data = await res.json()
setAttendance(data)
}
} catch (error) {
console.error('Error fetching attendance:', error)
} finally {
setLoading(false)
}
}
fetchAttendance()
}, [])
if (loading) {
return <div className="p-8">Loading...</div>
}
const { data: attendance = [], isLoading } = useAttendance();
if (isLoading) {
return (
<div className="p-8">
<h1 className="text-2xl font-bold mb-6">Attendance Monitoring</h1>
<div className="bg-white rounded-lg shadow overflow-hidden">
<table className="min-w-full divide-y divide-gray-200">
<thead className="bg-gray-50">
<tr>
<th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
Client ID
</th>
<th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
Type
</th>
<th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
Check In
</th>
<th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
Check Out
</th>
<th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
Status
</th>
</tr>
</thead>
<tbody className="bg-white divide-y divide-gray-200">
{attendance.map((record) => (
<tr key={record.id}>
<td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900">
{record.clientId}
</td>
<td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500 capitalize">
{record.type}
</td>
<td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
{format(new Date(record.checkInTime), 'PP p')}
</td>
<td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
{record.checkOutTime ? format(new Date(record.checkOutTime), 'PP p') : '-'}
</td>
<td className="px-6 py-4 whitespace-nowrap">
<span className={`px-2 inline-flex text-xs leading-5 font-semibold rounded-full ${record.checkOutTime
? 'bg-gray-100 text-gray-800'
: 'bg-green-100 text-green-800'
}`}>
{record.checkOutTime ? 'Completed' : 'Active'}
</span>
</td>
</tr>
))}
</tbody>
</table>
</div>
<div className="space-y-8">
<PageHeader
title="Attendance"
description="Monitor gym check-ins and check-outs"
breadcrumbs={[{ label: "Attendance" }]}
/>
<div className="card-modern">
<div className="animate-pulse space-y-4">
{[...Array(5)].map((_, i) => (
<div key={i} className="h-12 bg-muted rounded" />
))}
</div>
</div>
)
</div>
);
}
return (
<div className="space-y-8">
<PageHeader
title="Attendance"
description="Monitor gym check-ins and check-outs"
breadcrumbs={[{ label: "Attendance" }]}
/>
<div className="card-modern">
<div className="overflow-x-auto">
<table className="w-full">
<thead>
<tr className="border-b border-border">
<th className="text-left py-3 px-4 text-sm font-medium text-muted-foreground">
Client
</th>
<th className="text-left py-3 px-4 text-sm font-medium text-muted-foreground">
Type
</th>
<th className="text-left py-3 px-4 text-sm font-medium text-muted-foreground">
Check In
</th>
<th className="text-left py-3 px-4 text-sm font-medium text-muted-foreground">
Check Out
</th>
<th className="text-left py-3 px-4 text-sm font-medium text-muted-foreground">
Status
</th>
</tr>
</thead>
<tbody>
{attendance.map((record) => (
<tr
key={record.id}
className="border-b border-border/50 hover:bg-muted/50 transition-colors"
>
<td className="py-3 px-4 text-sm">
{record.userName || record.userId.substring(0, 8) + "..."}
</td>
<td className="py-3 px-4 text-sm capitalize text-muted-foreground">
{record.type}
</td>
<td className="py-3 px-4 text-sm text-muted-foreground">
{new Date(record.checkInTime).toLocaleString()}
</td>
<td className="py-3 px-4 text-sm text-muted-foreground">
{record.checkOutTime
? new Date(record.checkOutTime).toLocaleString()
: "-"}
</td>
<td className="py-3 px-4">
<Badge variant={record.checkOutTime ? "gray" : "success"}>
{record.checkOutTime ? "Completed" : "Active"}
</Badge>
</td>
</tr>
))}
{attendance.length === 0 && (
<tr>
<td
colSpan={5}
className="py-8 text-center text-muted-foreground"
>
No attendance records found
</td>
</tr>
)}
</tbody>
</table>
</div>
</div>
</div>
);
}

View File

@ -1,76 +1,248 @@
@tailwind base;
@tailwind components;
@tailwind utilities;
@layer base {
:root {
--background: 0 0% 100%;
--background: 0 0% 98%;
--foreground: 222.2 84% 4.9%;
--card: 0 0% 100%;
--card-foreground: 222.2 84% 4.9%;
--popover: 0 0% 100%;
--popover-foreground: 222.2 84% 4.9%;
--primary: 222.2 47.4% 11.2%;
--primary: 221.2 83.2% 53.3%;
--primary-foreground: 210 40% 98%;
--secondary: 210 40% 96.1%;
--secondary-foreground: 222.2 47.4% 11.2%;
--muted: 210 40% 96.1%;
--muted-foreground: 215.4 16.3% 46.9%;
--accent: 210 40% 96.1%;
--accent-foreground: 222.2 47.4% 11.2%;
--destructive: 0 84.2% 60.2%;
--destructive-foreground: 210 40% 98%;
--success: 142.1 76.2% 36.3%;
--success-foreground: 355.7 100% 97.3%;
--warning: 37.7 93.1% 50.2%;
--warning-foreground: 222.2 47.4% 11.2%;
--info: 199.4 86.4% 48.4%;
--info-foreground: 210 40% 98%;
--border: 214.3 31.8% 91.4%;
--input: 214.3 31.8% 91.4%;
--ring: 222.2 84% 4.9%;
--radius: 0.5rem;
--ring: 221.2 83.2% 53.3%;
--radius: 0.625rem;
--sidebar-background: 222.2 47.4% 7.8%;
--sidebar-foreground: 210 40% 98%;
--sidebar-primary: 221.2 83.2% 53.3%;
--sidebar-primary-foreground: 210 40% 98%;
--sidebar-accent: 217.2 32.6% 17.5%;
--sidebar-accent-foreground: 210 40% 98%;
--sidebar-border: 217.2 32.6% 17.5%;
--sidebar-ring: 217.2 91.6% 59.8%;
}
.dark {
--background: 222.2 84% 4.9%;
--foreground: 210 40% 98%;
--card: 222.2 84% 4.9%;
--card: 222.2 47.4% 10.2%;
--card-foreground: 210 40% 98%;
--popover: 222.2 84% 4.9%;
--popover: 222.2 47.4% 10.2%;
--popover-foreground: 210 40% 98%;
--primary: 210 40% 98%;
--primary: 217.2 91.2% 59.8%;
--primary-foreground: 222.2 47.4% 11.2%;
--secondary: 217.2 32.6% 17.5%;
--secondary-foreground: 210 40% 98%;
--muted: 217.2 32.6% 17.5%;
--muted-foreground: 215 20.2% 65.1%;
--accent: 217.2 32.6% 17.5%;
--accent-foreground: 210 40% 98%;
--destructive: 0 62.8% 30.6%;
--destructive-foreground: 210 40% 98%;
--success: 142.1 76.2% 36.3%;
--success-foreground: 355.7 100% 97.3%;
--warning: 37.7 93.1% 50.2%;
--warning-foreground: 222.2 47.4% 11.2%;
--info: 199.4 86.4% 48.4%;
--info-foreground: 210 40% 98%;
--border: 217.2 32.6% 17.5%;
--input: 217.2 32.6% 17.5%;
--ring: 212.7 26.8% 83.9%;
--ring: 224.3 76.3% 48%;
--sidebar-background: 222.2 47.4% 7.8%;
--sidebar-foreground: 210 40% 98%;
--sidebar-primary: 217.2 91.2% 59.8%;
--sidebar-primary-foreground: 222.2 47.4% 11.2%;
--sidebar-accent: 217.2 32.6% 17.5%;
--sidebar-accent-foreground: 210 40% 98%;
--sidebar-border: 217.2 32.6% 17.5%;
--sidebar-ring: 217.2 91.6% 59.8%;
}
}
@layer base {
* {
@apply border-border;
}
body {
@apply bg-background text-foreground;
@apply bg-background text-foreground antialiased;
}
}
}
@layer utilities {
.gradient-mesh {
background:
radial-gradient(
at 40% 20%,
hsla(221, 83%, 53%, 0.1) 0px,
transparent 50%
),
radial-gradient(at 80% 0%, hsla(189, 97%, 66%, 0.1) 0px, transparent 50%),
radial-gradient(at 0% 50%, hsla(355, 85%, 93%, 0.3) 0px, transparent 50%),
radial-gradient(
at 80% 50%,
hsla(240, 75%, 98%, 0.3) 0px,
transparent 50%
),
radial-gradient(at 0% 100%, hsla(22, 100%, 92%, 0.2) 0px, transparent 50%);
}
.glass {
@apply backdrop-blur-xl bg-white/70 dark:bg-black/70;
}
.text-gradient {
@apply bg-clip-text text-transparent bg-gradient-to-r from-primary to-primary/60;
}
.animate-fade-in {
animation: fadeIn 0.5s ease-out forwards;
}
.animate-slide-up {
animation: slideUp 0.5s ease-out forwards;
}
.animate-scale-in {
animation: scaleIn 0.3s ease-out forwards;
}
@keyframes fadeIn {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@keyframes slideUp {
from {
opacity: 0;
transform: translateY(20px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
@keyframes scaleIn {
from {
opacity: 0;
transform: scale(0.95);
}
to {
opacity: 1;
transform: scale(1);
}
}
.stagger-1 {
animation-delay: 0.1s;
}
.stagger-2 {
animation-delay: 0.2s;
}
.stagger-3 {
animation-delay: 0.3s;
}
.stagger-4 {
animation-delay: 0.4s;
}
.stagger-5 {
animation-delay: 0.5s;
}
}
@layer components {
.btn-primary {
@apply inline-flex items-center justify-center rounded-lg bg-primary px-4 py-2 text-sm font-medium text-primary-foreground shadow-sm transition-all duration-200 hover:bg-primary/90 hover:shadow-md focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50;
}
.btn-secondary {
@apply inline-flex items-center justify-center rounded-lg bg-secondary px-4 py-2 text-sm font-medium text-secondary-foreground shadow-sm transition-all duration-200 hover:bg-secondary/80 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50;
}
.btn-ghost {
@apply inline-flex items-center justify-center rounded-lg px-4 py-2 text-sm font-medium text-muted-foreground transition-all duration-200 hover:bg-accent hover:text-accent-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50;
}
.btn-danger {
@apply inline-flex items-center justify-center rounded-lg bg-destructive px-4 py-2 text-sm font-medium text-destructive-foreground shadow-sm transition-all duration-200 hover:bg-destructive/90 hover:shadow-md focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50;
}
.card-modern {
@apply rounded-xl border border-border/50 bg-card p-6 shadow-sm transition-all duration-200 hover:shadow-md;
}
.card-elevated {
@apply rounded-xl border border-border/50 bg-card p-6 shadow-lg transition-all duration-200 hover:shadow-xl hover:-translate-y-0.5;
}
.input-modern {
@apply flex h-10 w-full rounded-lg border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 transition-all duration-200;
}
.badge-success {
@apply inline-flex items-center rounded-full bg-green-100 px-2.5 py-0.5 text-xs font-medium text-green-800 dark:bg-green-900/30 dark:text-green-400;
}
.badge-warning {
@apply inline-flex items-center rounded-full bg-yellow-100 px-2.5 py-0.5 text-xs font-medium text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-400;
}
.badge-error {
@apply inline-flex items-center rounded-full bg-red-100 px-2.5 py-0.5 text-xs font-medium text-red-800 dark:bg-red-900/30 dark:text-red-400;
}
.badge-info {
@apply inline-flex items-center rounded-full bg-blue-100 px-2.5 py-0.5 text-xs font-medium text-blue-800 dark:bg-blue-900/30 dark:text-blue-400;
}
.badge-default {
@apply inline-flex items-center rounded-full bg-gray-100 px-2.5 py-0.5 text-xs font-medium text-gray-800 dark:bg-gray-800 dark:text-gray-300;
}
}

View File

@ -9,7 +9,8 @@ import {
UserButton,
} from "@clerk/nextjs";
import { Sidebar } from "@/components/ui/Sidebar";
import { Toaster } from "sonner";
import { ToasterProvider } from "@/components/providers/ToasterProvider";
import { QueryProvider } from "@/components/providers/QueryProvider";
const inter = Inter({ subsets: ["latin"] });
@ -25,15 +26,19 @@ export default function RootLayout({
}) {
return (
<ClerkProvider>
<html lang="en">
<body className={inter.className}>
<div className="flex min-h-screen bg-slate-50">
<Sidebar />
<main className="flex-1 ml-20 p-8">{children}</main>
</div>
<Toaster richColors position="top-right" />
</body>
</html>
<QueryProvider>
<html lang="en">
<body className={inter.className}>
<div className="flex min-h-screen bg-background">
<Sidebar />
<main className="flex-1 p-6 lg:p-8 min-h-screen">
<div className="max-w-7xl mx-auto space-y-8">{children}</div>
</main>
</div>
<ToasterProvider />
</body>
</html>
</QueryProvider>
</ClerkProvider>
);
}

View File

@ -1,100 +1,150 @@
"use client";
import { useEffect, useState } from "react";
import { Users, CreditCard, CalendarCheck, TrendingUp } from "lucide-react";
import { StatsCard } from "@/components/ui/StatsCard";
import {
Users,
CreditCard,
CalendarCheck,
TrendingUp,
RefreshCw,
} from "lucide-react";
import { StatsCard, StatsCardSkeleton } from "@/components/ui/StatsCard";
import { PageHeader } from "@/components/ui/PageHeader";
import { UserManagement } from "@/components/users/UserManagement";
import { AnalyticsDashboard } from "@/components/analytics/AnalyticsDashboard";
import axios from "axios";
interface DashboardStats {
totalUsers: number;
activeClients: number;
totalRevenue: number;
revenueGrowth: number;
}
import { useDashboardStats } from "@/hooks/use-api";
import { useQueryClient } from "@tanstack/react-query";
import { Button } from "@/components/ui/button";
import { useUser } from "@clerk/nextjs";
import { useSearchParams } from "next/navigation";
import { GymSelector } from "@/components/gym/GymSelector";
export default function Home() {
const [stats, setStats] = useState<DashboardStats>({
totalUsers: 0,
activeClients: 0,
totalRevenue: 0,
revenueGrowth: 0,
});
const [loading, setLoading] = useState(true);
const { user } = useUser();
const searchParams = useSearchParams();
const gymId = searchParams.get("gymId") ?? undefined;
useEffect(() => {
const fetchStats = async () => {
try {
const response = await axios.get("/api/admin/stats");
setStats(response.data);
} catch (error) {
console.error("Failed to fetch dashboard stats:", error);
} finally {
setLoading(false);
}
};
const {
data: stats,
isLoading,
refetch,
isFetching,
} = useDashboardStats(gymId);
const queryClient = useQueryClient();
fetchStats();
}, []);
// Get user role from metadata
const userRole = (user?.publicMetadata?.role as string) ?? "client";
const handleRefresh = () => {
queryClient.invalidateQueries({ queryKey: ["dashboard-stats"] });
queryClient.invalidateQueries({ queryKey: ["analytics"] });
refetch();
};
const formatCurrency = (value: number) => {
return new Intl.NumberFormat("en-US", {
style: "currency",
currency: "USD",
}).format(value);
minimumFractionDigits: 0,
maximumFractionDigits: 0,
}).format(value || 0);
};
return (
<div className="space-y-8">
<div>
<h2 className="text-3xl font-bold text-slate-900">Dashboard</h2>
<p className="text-slate-500 mt-2">Welcome back, here's what's happening today.</p>
<PageHeader
title="Dashboard"
description="Welcome back! Here's what's happening with your gym today."
actions={
<div className="flex items-center gap-3">
<GymSelector userRole={userRole} />
<Button
variant="outline"
size="sm"
onClick={handleRefresh}
disabled={isFetching}
className="gap-2"
>
<RefreshCw
className={`h-4 w-4 ${isFetching ? "animate-spin" : ""}`}
/>
Refresh
</Button>
</div>
}
/>
{/* Stats Grid */}
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
{isLoading ? (
<>
<StatsCardSkeleton />
<StatsCardSkeleton />
<StatsCardSkeleton />
<StatsCardSkeleton />
</>
) : (
<>
<StatsCard
title="Total Users"
value={stats?.totalUsers ?? 0}
change="+12%"
trend="up"
icon={Users}
color="blue"
/>
<StatsCard
title="Active Clients"
value={stats?.activeClients ?? 0}
change="+5%"
trend="up"
icon={CalendarCheck}
color="green"
/>
<StatsCard
title="Revenue"
value={formatCurrency(stats?.totalRevenue ?? 0)}
change={`${(stats?.revenueGrowth ?? 0) > 0 ? "+" : ""}${stats?.revenueGrowth ?? 0}%`}
trend={(stats?.revenueGrowth ?? 0) >= 0 ? "up" : "down"}
icon={CreditCard}
color="purple"
/>
<StatsCard
title="Growth"
value="24%"
change="-2%"
trend="down"
icon={TrendingUp}
color="orange"
/>
</>
)}
</div>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6">
<StatsCard
title="Total Users"
value={loading ? "..." : stats.totalUsers}
change="+12%" // Placeholder for now as we don't track historical growth yet
trend="up"
icon={Users}
color="blue"
/>
<StatsCard
title="Active Clients"
value={loading ? "..." : stats.activeClients}
change="+5%"
trend="up"
icon={CalendarCheck}
color="green"
/>
<StatsCard
title="Revenue"
value={loading ? "..." : formatCurrency(stats.totalRevenue)}
change={`${stats.revenueGrowth > 0 ? "+" : ""}${stats.revenueGrowth}%`}
trend={stats.revenueGrowth >= 0 ? "up" : "down"}
icon={CreditCard}
color="purple"
/>
<StatsCard
title="Growth"
value="24%" // Placeholder
change="-2%"
trend="down"
icon={TrendingUp}
color="orange"
/>
</div>
{/* Main Content Grid */}
<div className="grid grid-cols-1 xl:grid-cols-3 gap-6">
{/* <div className="xl:col-span-2"> */}
{/* <div className="card-modern"> */}
{/* <div className="mb-6"> */}
{/* <h3 className="text-lg font-semibold">Recent Activity</h3> */}
{/* <p className="text-sm text-muted-foreground"> */}
{/* Manage and view your users */}
{/* </p> */}
{/* </div> */}
{/* <UserManagement /> */}
{/* </div> */}
{/* </div> */}
<div className="flex flex-col gap-8">
<div className="bg-white rounded-xl shadow-sm border border-slate-100 p-6">
<h3 className="text-xl font-bold text-slate-900 mb-6">Recent Activity</h3>
<UserManagement />
</div>
<div className="bg-white rounded-xl shadow-sm border border-slate-100 p-6">
<h3 className="text-xl font-bold text-slate-900 mb-6">Quick Analytics</h3>
<AnalyticsDashboard />
{/* Analytics Sidebar */}
<div className="xl:col-span-3">
<div className="card-modern">
<div className="mb-6">
<h3 className="text-lg font-semibold">Quick Analytics</h3>
<p className="text-sm text-muted-foreground">
Overview of your gym metrics
</p>
</div>
<AnalyticsDashboard gymId={gymId} />
</div>
</div>
</div>
</div>

View File

@ -55,7 +55,7 @@ export default function ProfilePage() {
<div className="flex justify-between items-center mb-6">
<h1 className="text-2xl font-bold">Profile</h1>
<Button
variant={isEditing ? "primary" : "secondary"}
variant={isEditing ? "default" : "secondary"}
onClick={() => setIsEditing(!isEditing)}
>
{isEditing ? "Cancel" : "Edit Profile"}

View File

@ -1,86 +1,41 @@
"use client";
import { useEffect, useState } from "react";
import { useState } from "react";
import { useUser } from "@clerk/nextjs";
import log from "@/lib/logger";
import { toast } from "@/lib/toast";
interface User {
id: string;
firstName: string;
lastName: string;
email: string;
}
interface Recommendation {
id: string;
userId: string;
content: string;
recommendationText: string;
activityPlan: string;
dietPlan: string;
status: string;
createdAt: Date;
}
import {
useUsers,
useRecommendations,
useGenerateRecommendations,
useApproveRecommendation,
useUpdateRecommendation,
type Recommendation,
} from "@/hooks/use-api";
import { PageHeader } from "@/components/ui/PageHeader";
export default function RecommendationsPage() {
const { user } = useUser();
const [users, setUsers] = useState<User[]>([]);
const [pendingRecommendations, setPendingRecommendations] = useState<
Recommendation[]
>([]);
const [loading, setLoading] = useState(true);
const [generating, setGenerating] = useState<string | null>(null);
const [useExternalModel, setUseExternalModel] = useState(false);
useEffect(() => {
fetchData();
}, []);
const { data: users = [], isLoading: usersLoading } = useUsers();
const { data: allRecommendations = [], isLoading: recsLoading } =
useRecommendations();
const pendingRecommendations = allRecommendations.filter(
(r) => r.status === "pending",
);
const fetchData = async () => {
try {
// Fetch users - API returns { success: true, data: { users: [...] }, meta: {...} }
const usersRes = await fetch("/api/users");
const usersResult = await usersRes.json();
const usersArray = usersResult.data?.users || usersResult.users || [];
setUsers(usersArray);
// Fetch pending recommendations - API returns { success: true, data: { recommendations: [...] }, meta: {...} }
const recsRes = await fetch("/api/recommendations");
const recsResult = await recsRes.json();
const allRecs =
recsResult.data?.recommendations || recsResult.recommendations || [];
setPendingRecommendations(
allRecs.filter((r: Recommendation) => r.status === "pending"),
);
} catch (error) {
log.error("Failed to fetch data", error);
} finally {
setLoading(false);
}
};
const generateRec = useGenerateRecommendations();
const approveRec = useApproveRecommendation();
const updateRec = useUpdateRecommendation();
const handleGenerate = async (userId: string) => {
setGenerating(userId);
try {
const res = await fetch("/api/recommendations/generate", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ userId, useExternalModel }),
});
if (!res.ok) {
const error = await res.json();
toast.error(`Error: ${error.error}`);
} else {
toast.success("Recommendation generated successfully!");
fetchData(); // Refresh data
}
await generateRec.mutateAsync(userId);
toast.success("Recommendation generated successfully!");
} catch (error) {
log.error("Failed to generate recommendation", error);
toast.error("Failed to generate recommendation.");
} finally {
setGenerating(null);
}
};
@ -89,25 +44,11 @@ export default function RecommendationsPage() {
status: "approved" | "rejected",
) => {
try {
const res = await fetch("/api/recommendations/approve", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
recommendationId,
status,
approvedBy: user?.id || "admin",
}),
await approveRec.mutateAsync({
recommendationId,
approved: status === "approved",
});
if (!res.ok) {
const errorData = await res.json();
toast.error(
`Failed to update status: ${errorData.error || "Unknown error"}`,
);
} else {
toast.success("Recommendation status updated");
fetchData(); // Refresh data
}
toast.success("Recommendation status updated");
} catch (error) {
log.error("Failed to approve recommendation", error);
toast.error("Error processing request");
@ -124,168 +65,137 @@ export default function RecommendationsPage() {
newActivityPlan === null ||
newDietPlan === null
) {
// User cancelled one of the prompts
return;
}
try {
const res = await fetch("/api/recommendations", {
method: "PUT",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
id: rec.id,
content: newContent,
activityPlan: newActivityPlan,
dietPlan: newDietPlan,
}),
await updateRec.mutateAsync({
id: rec.id,
content: newContent,
activityPlan: newActivityPlan,
dietPlan: newDietPlan,
});
if (!res.ok) {
const errorData = await res.json();
toast.error(
`Failed to update recommendation: ${errorData.error || "Unknown error"}`,
);
} else {
toast.success("Recommendation updated successfully!");
fetchData(); // Refresh data
}
toast.success("Recommendation updated successfully!");
} catch (error) {
log.error("Failed to update recommendation", error);
toast.error("Failed to update recommendation.");
}
};
if (loading) {
if (usersLoading || recsLoading) {
return (
<div className="flex items-center justify-center h-screen">
<div className="text-xl">Loading...</div>
<div className="flex items-center justify-center h-64">
<div className="text-lg">Loading...</div>
</div>
);
}
return (
<div className="container mx-auto py-10 px-4">
<div className="flex justify-between items-center mb-8">
<h1 className="text-3xl font-bold">AI Recommendations</h1>
<div className="space-y-8">
<PageHeader
title="AI Recommendations"
description="Generate and manage AI-powered fitness recommendations"
breadcrumbs={[{ label: "AI Recommendations" }]}
/>
{/* Model Selection Toggle */}
<div className="flex items-center gap-3 bg-white px-4 py-2 rounded-lg shadow">
<span className="text-sm font-medium text-gray-700">
{useExternalModel ? "DeepSeek AI" : "Local Ollama"}
</span>
<button
onClick={() => setUseExternalModel(!useExternalModel)}
className={`relative inline-flex h-6 w-11 items-center rounded-full transition-colors ${
useExternalModel ? "bg-blue-600" : "bg-gray-300"
}`}
>
<span
className={`inline-block h-4 w-4 transform rounded-full bg-white transition-transform ${
useExternalModel ? "translate-x-6" : "translate-x-1"
}`}
/>
</button>
<span className="text-xs text-gray-500">
{useExternalModel ? "External" : "Local"}
</span>
</div>
</div>
<div className="grid grid-cols-1 lg:grid-cols-2 gap-8">
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
{/* Generate Section */}
<div>
<h2 className="text-2xl font-semibold mb-4">
Generate Recommendations
</h2>
<div className="bg-white shadow rounded-lg p-6">
<p className="mb-4 text-gray-600">
<div className="card-modern">
<div className="mb-6">
<h3 className="text-lg font-semibold">Generate Recommendations</h3>
<p className="text-sm text-muted-foreground">
Select a user to generate a new daily recommendation.
</p>
<ul className="space-y-4">
{users.map((user) => (
<li
key={user.id}
className="flex items-center justify-between border-b pb-2"
>
<div>
<p className="font-medium">
{user.firstName} {user.lastName}
</p>
<p className="text-sm text-gray-500">{user.email}</p>
</div>
<button
onClick={() => handleGenerate(user.id)}
disabled={generating === user.id}
className="bg-blue-600 text-white px-4 py-2 rounded hover:bg-blue-700 disabled:opacity-50"
>
{generating === user.id ? "Generating..." : "Generate"}
</button>
</li>
))}
{users.length === 0 && (
<p className="text-gray-500 italic">No users found.</p>
)}
</ul>
</div>
<ul className="space-y-4">
{users.map((user) => (
<li
key={user.id}
className="flex items-center justify-between border-b border-border/50 pb-4 last:border-0 last:pb-0"
>
<div>
<p className="font-medium">
{user.firstName} {user.lastName}
</p>
<p className="text-sm text-muted-foreground">{user.email}</p>
</div>
<button
onClick={() => handleGenerate(user.id)}
disabled={generateRec.isPending}
className="btn-primary"
>
{generateRec.isPending ? "Generating..." : "Generate"}
</button>
</li>
))}
{users.length === 0 && (
<p className="text-muted-foreground italic">No users found.</p>
)}
</ul>
</div>
{/* Pending Approvals Section */}
<div>
<h2 className="text-2xl font-semibold mb-4">Pending Approvals</h2>
<div className="bg-white shadow rounded-lg p-6">
{pendingRecommendations.length === 0 ? (
<p className="text-gray-500 italic">
No pending recommendations.
</p>
) : (
<ul className="space-y-6">
{pendingRecommendations.map((rec) => (
<li key={rec.id} className="border rounded p-4">
<div className="flex justify-between items-start mb-2">
<h3 className="font-bold">For: User {rec.userId}</h3>
<span className="text-xs text-gray-500">
{new Date(rec.createdAt).toLocaleDateString()}
</span>
</div>
<div className="space-y-2 text-sm mb-4">
<div>
<span className="font-semibold">Advice:</span>{" "}
{rec.recommendationText}
</div>
<div>
<span className="font-semibold">Activity:</span>{" "}
{rec.activityPlan}
</div>
<div>
<span className="font-semibold">Diet:</span>{" "}
{rec.dietPlan}
</div>
</div>
<div className="flex space-x-2">
<button
onClick={() => handleEdit(rec)}
className="flex-1 bg-blue-500 text-white py-2 rounded hover:bg-blue-600"
>
Edit
</button>
<button
onClick={() => handleApprove(rec.id, "approved")}
className="flex-1 bg-green-600 text-white py-2 rounded hover:bg-green-700"
>
Approve
</button>
<button
onClick={() => handleApprove(rec.id, "rejected")}
className="flex-1 bg-red-600 text-white py-2 rounded hover:bg-red-700"
>
Reject
</button>
</div>
</li>
))}
</ul>
)}
<div className="card-modern">
<div className="mb-6">
<h3 className="text-lg font-semibold">Pending Approvals</h3>
<p className="text-sm text-muted-foreground">
Review and approve AI-generated recommendations
</p>
</div>
{pendingRecommendations.length === 0 ? (
<p className="text-muted-foreground italic">
No pending recommendations.
</p>
) : (
<ul className="space-y-4">
{pendingRecommendations.map((rec) => (
<li
key={rec.id}
className="border border-border rounded-lg p-4"
>
<div className="flex justify-between items-start mb-3">
<h4 className="font-semibold">For: User {rec.userId}</h4>
<span className="text-xs text-muted-foreground">
{new Date(rec.createdAt).toLocaleDateString()}
</span>
</div>
<div className="space-y-2 text-sm mb-4">
<div>
<span className="font-medium">Advice:</span>{" "}
{rec.recommendationText}
</div>
<div>
<span className="font-medium">Activity:</span>{" "}
{rec.activityPlan}
</div>
<div>
<span className="font-medium">Diet:</span> {rec.dietPlan}
</div>
</div>
<div className="flex gap-2">
<button
onClick={() => handleEdit(rec)}
className="btn-secondary flex-1"
>
Edit
</button>
<button
onClick={() => handleApprove(rec.id, "approved")}
className="btn-primary flex-1 bg-emerald-600 hover:bg-emerald-700"
>
Approve
</button>
<button
onClick={() => handleApprove(rec.id, "rejected")}
className="btn-danger flex-1"
>
Reject
</button>
</div>
</li>
))}
</ul>
)}
</div>
</div>
</div>

View File

@ -0,0 +1,52 @@
"use client";
import { useState } from "react";
import { UserReport } from "@/components/reports/UserReport";
import { ReportFilters } from "@/components/reports/ReportFilters";
import { Card, CardContent } from "@/components/ui/card";
export default function ReportsPage() {
const [selectedUserId, setSelectedUserId] = useState<string>("");
const [dateRange, setDateRange] = useState<{
startDate: string;
endDate: string;
}>({
startDate: new Date(Date.now() - 30 * 24 * 60 * 60 * 1000)
.toISOString()
.split("T")[0],
endDate: new Date().toISOString().split("T")[0],
});
return (
<div className="min-h-screen bg-gray-50">
<div className="container mx-auto px-4 py-8 space-y-6">
<div className="flex justify-between items-center">
<h1 className="text-3xl font-bold text-gray-900">User Reports</h1>
</div>
<ReportFilters
selectedUserId={selectedUserId}
onUserChange={setSelectedUserId}
dateRange={dateRange}
onDateRangeChange={setDateRange}
/>
{selectedUserId ? (
<UserReport
userId={selectedUserId}
startDate={dateRange.startDate}
endDate={dateRange.endDate}
/>
) : (
<Card>
<CardContent className="flex items-center justify-center h-64">
<p className="text-gray-500 text-lg">
Select a user to view their report
</p>
</CardContent>
</Card>
)}
</div>
</div>
);
}

View File

@ -9,10 +9,15 @@ import {
AlertTriangle,
Check,
Loader2,
Trash2,
Users,
CalendarCheck,
TrendingUp,
} from "lucide-react";
import { Button } from "@/components/ui/button";
import { useUser } from "@clerk/nextjs";
import log from "@/lib/logger";
import { MEMBERSHIP_FEATURES } from "@/lib/membership/features";
interface Backup {
name: string;
@ -24,8 +29,27 @@ interface Gym {
id: string;
name: string;
location?: string | null;
latitude?: number | null;
longitude?: number | null;
geofenceRadiusMeters?: number | null;
geofenceEnabled?: boolean;
status: "active" | "inactive";
adminUserId: string;
createdAt?: number;
}
interface GymStats {
totalUsers: number;
admins: number;
trainers: number;
clients: number;
membershipStats: {
basic: number;
premium: number;
vip: number;
};
activeClients: number;
attendanceLast30Days: number;
}
export default function SettingsPage() {
@ -46,6 +70,18 @@ export default function SettingsPage() {
type: "success" | "error";
text: string;
} | null>(null);
// Selected gym for details
const [selectedGym, setSelectedGym] = useState<Gym | null>(null);
const [gymStats, setGymStats] = useState<GymStats | null>(null);
const [statsLoading, setStatsLoading] = useState(false);
const [deletingGym, setDeletingGym] = useState(false);
const [savingGeofence, setSavingGeofence] = useState(false);
const [geofenceLatitude, setGeofenceLatitude] = useState("");
const [geofenceLongitude, setGeofenceLongitude] = useState("");
const [geofenceRadiusMeters, setGeofenceRadiusMeters] = useState("30");
const [geofenceEnabled, setGeofenceEnabled] = useState(true);
// Create Gym modal state
const [showCreateGym, setShowCreateGym] = useState(false);
const [gymName, setGymName] = useState("");
@ -77,11 +113,33 @@ export default function SettingsPage() {
}
};
const fetchGymStats = async (gymId: string) => {
setStatsLoading(true);
try {
const res = await axios.get(`/api/gyms/${gymId}/stats`);
if (res.data?.stats) {
setGymStats(res.data.stats);
}
} catch (error) {
log.error("Failed to fetch gym stats", error);
} finally {
setStatsLoading(false);
}
};
useEffect(() => {
fetchBackups();
fetchGyms();
}, []);
useEffect(() => {
if (selectedGym) {
fetchGymStats(selectedGym.id);
} else {
setGymStats(null);
}
}, [selectedGym]);
const handleCreateBackup = async () => {
setCreatingBackup(true);
setMessage(null);
@ -111,7 +169,6 @@ export default function SettingsPage() {
try {
await axios.post("/api/admin/backups/restore", { filename });
setMessage({ type: "success", text: "Database restored successfully" });
// Optional: Refresh page or force re-login if session is invalidated
} catch (error) {
log.error("Failed to restore backup", error);
setMessage({ type: "error", text: "Failed to restore backup" });
@ -135,21 +192,126 @@ export default function SettingsPage() {
return new Date(dateString).toLocaleString();
};
const handleSelectGym = async (gymId: string | null) => {
setGymMessage(null);
try {
// Update current user's gym selection
await axios.patch("/api/users/gym", { gymId });
setGymMessage({
type: "success",
text: gymId ? "Gym selected successfully" : "Proceeding without gym",
});
} catch (error) {
log.error("Failed to set gym", error);
setGymMessage({ type: "error", text: "Failed to set gym" });
const handleSelectGym = async (gym: Gym | null) => {
setSelectedGym(gym);
setGymStats(null);
if (gym) {
setGeofenceLatitude(
gym.latitude !== null && gym.latitude !== undefined
? String(gym.latitude)
: "",
);
setGeofenceLongitude(
gym.longitude !== null && gym.longitude !== undefined
? String(gym.longitude)
: "",
);
setGeofenceRadiusMeters(String(gym.geofenceRadiusMeters ?? 30));
setGeofenceEnabled(gym.geofenceEnabled ?? true);
}
};
const handleSaveGeofence = async () => {
if (!selectedGym) return;
const latitude =
geofenceLatitude.trim() === "" ? null : Number(geofenceLatitude);
const longitude =
geofenceLongitude.trim() === "" ? null : Number(geofenceLongitude);
const radius = Number(geofenceRadiusMeters);
if (
latitude !== null &&
(!Number.isFinite(latitude) || latitude < -90 || latitude > 90)
) {
setGymMessage({
type: "error",
text: "Latitude must be between -90 and 90",
});
return;
}
if (
longitude !== null &&
(!Number.isFinite(longitude) || longitude < -180 || longitude > 180)
) {
setGymMessage({
type: "error",
text: "Longitude must be between -180 and 180",
});
return;
}
if (!Number.isFinite(radius) || radius <= 0) {
setGymMessage({
type: "error",
text: "Radius must be a positive number",
});
return;
}
setSavingGeofence(true);
setGymMessage(null);
try {
const response = await axios.patch(`/api/gyms/${selectedGym.id}`, {
latitude,
longitude,
geofenceRadiusMeters: radius,
geofenceEnabled,
});
setGymMessage({ type: "success", text: "Geofence settings updated" });
const updatedGym = response.data as Gym;
setSelectedGym(updatedGym);
setGyms((prev) =>
prev.map((gym) => (gym.id === updatedGym.id ? updatedGym : gym)),
);
} catch (error) {
log.error("Failed to update geofence settings", error);
setGymMessage({
type: "error",
text: "Failed to update geofence settings",
});
} finally {
setSavingGeofence(false);
}
};
const handleDeleteGym = async (gymId: string) => {
if (
!window.confirm(
"Are you sure you want to delete this gym? This action cannot be undone.",
)
) {
return;
}
setDeletingGym(true);
setGymMessage(null);
try {
const response = await axios.delete(`/api/gyms/${gymId}`);
log.info("Delete gym response:", response.data);
setGymMessage({ type: "success", text: "Gym deleted successfully" });
setSelectedGym(null);
setGymStats(null);
await fetchGyms();
} catch (error: any) {
log.error("Failed to delete gym", error);
const errorMessage =
error.response?.data?.error ||
error.response?.data ||
error.message ||
"Failed to delete gym";
setGymMessage({ type: "error", text: errorMessage });
} finally {
setDeletingGym(false);
}
};
const userRole = (user?.publicMetadata?.role as string) ?? "client";
const isSuperAdmin = userRole === "superAdmin";
return (
<div className="space-y-8 p-8">
<div>
@ -159,7 +321,7 @@ export default function SettingsPage() {
</p>
</div>
{/* Gym Picker */}
{/* Gym Management */}
<div className="bg-white rounded-xl shadow-sm border border-slate-100 p-6">
<div className="flex items-center justify-between mb-6">
<div className="flex items-center gap-3">
@ -168,10 +330,12 @@ export default function SettingsPage() {
</div>
<div>
<h3 className="text-xl font-bold text-slate-900">
Gym Selection
Gym Management
</h3>
<p className="text-sm text-slate-500">
Select your gym or proceed without a gym
{isSuperAdmin
? "Select a gym to view details, create or delete gyms"
: "Select your gym or proceed without a gym"}
</p>
<p className="text-xs text-blue-600 mt-1">
{user ? (
@ -180,11 +344,6 @@ export default function SettingsPage() {
<span className="font-medium">
{String(user.publicMetadata?.role ?? "unknown")}
</span>
{" • "}
Gym ID:{" "}
<span className="font-medium">
{String(user.publicMetadata?.gymId ?? "none")}
</span>
</>
) : (
"Loading user metadata..."
@ -196,6 +355,8 @@ export default function SettingsPage() {
<Button
onClick={fetchGyms}
disabled={gymsLoading}
variant="outline"
size="sm"
className="flex items-center gap-2"
>
{gymsLoading ? (
@ -203,14 +364,17 @@ export default function SettingsPage() {
) : (
<RefreshCw className="w-4 h-4" />
)}
Refresh Gyms
</Button>
<Button
onClick={() => setShowCreateGym(true)}
className="flex items-center gap-2"
>
Create Gym
Refresh
</Button>
{isSuperAdmin && (
<Button
onClick={() => setShowCreateGym(true)}
size="sm"
className="flex items-center gap-2"
>
Create Gym
</Button>
)}
</div>
</div>
@ -226,6 +390,7 @@ export default function SettingsPage() {
{gymMessage.text}
</div>
)}
{showCreateGym && (
<div className="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
<div className="bg-white p-6 rounded-lg shadow-lg max-w-md w-full">
@ -310,59 +475,379 @@ export default function SettingsPage() {
</div>
)}
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4">
<div className="border rounded-lg p-4 flex flex-col justify-between">
<div>
<h4 className="font-semibold text-slate-900">
Proceed without gym
</h4>
<p className="text-sm text-slate-600 mt-1">
You can select a gym later.
</p>
</div>
<Button
variant="outline"
className="mt-4"
onClick={() => handleSelectGym(null)}
>
Proceed without gym
</Button>
<div className="grid grid-cols-1 lg:grid-cols-3 gap-6">
{/* Gym List */}
<div className="lg:col-span-1 space-y-4">
<h4 className="font-semibold text-slate-900">Gyms</h4>
{gymsLoading ? (
<div className="flex items-center justify-center p-8 text-slate-500">
<Loader2 className="w-5 h-5 animate-spin mr-2" />
Loading...
</div>
) : gyms.length === 0 ? (
<div className="p-4 text-center text-slate-500">
No active gyms found.
</div>
) : (
<div className="space-y-2 max-h-96 overflow-y-auto">
{gyms.map((gym) => (
<div
key={gym.id}
className={`border rounded-lg p-3 cursor-pointer transition-colors ${
selectedGym?.id === gym.id
? "border-blue-500 bg-blue-50"
: "hover:bg-slate-50"
}`}
onClick={() => handleSelectGym(gym)}
>
<div className="flex items-center justify-between">
<div>
<h5 className="font-medium text-slate-900">
{gym.name}
</h5>
<p className="text-xs text-slate-500">
{gym.location || "No location"}
</p>
</div>
<span
className={`text-xs px-2 py-1 rounded ${
gym.status === "active"
? "bg-green-100 text-green-700"
: "bg-red-100 text-red-700"
}`}
>
{gym.status}
</span>
</div>
</div>
))}
</div>
)}
</div>
{gymsLoading ? (
<div className="col-span-full flex items-center justify-center p-8 text-slate-500">
<Loader2 className="w-5 h-5 animate-spin mr-2" />
Loading gyms...
</div>
) : gyms.length === 0 ? (
<div className="col-span-full p-8 text-center text-slate-500">
No active gyms found.
</div>
) : (
gyms.map((gym) => (
<div
key={gym.id}
className="border rounded-lg p-4 flex flex-col justify-between"
>
<div>
<h4 className="font-semibold text-slate-900">{gym.name}</h4>
<p className="text-sm text-slate-600 mt-1">
{gym.location || "No location provided"}
</p>
{/* Gym Details / Stats */}
<div className="lg:col-span-2">
{selectedGym ? (
<div className="space-y-4">
<div className="flex items-center justify-between">
<h4 className="font-semibold text-slate-900">
{selectedGym.name} - Details
</h4>
{isSuperAdmin && (
<Button
variant="ghost"
size="sm"
className="text-red-600 hover:text-red-700 hover:bg-red-50"
onClick={() => handleDeleteGym(selectedGym.id)}
disabled={deletingGym}
>
{deletingGym ? (
<Loader2 className="w-4 h-4 animate-spin" />
) : (
<Trash2 className="w-4 h-4 mr-1" />
)}
Delete Gym
</Button>
)}
</div>
{/* Gym Info */}
<div className="grid grid-cols-2 gap-4 p-4 bg-slate-50 rounded-lg">
<div>
<p className="text-xs text-slate-500">Location</p>
<p className="font-medium">
{selectedGym.location || "Not specified"}
</p>
</div>
<div>
<p className="text-xs text-slate-500">Status</p>
<p className="font-medium capitalize">
{selectedGym.status}
</p>
</div>
<div>
<p className="text-xs text-slate-500">Geofence</p>
<p className="font-medium">
{selectedGym.geofenceEnabled === false
? "Disabled"
: `${selectedGym.geofenceRadiusMeters ?? 30}m`}
</p>
</div>
</div>
{/* Geofence Settings */}
<div className="p-4 border rounded-lg space-y-3">
<div className="flex items-center justify-between">
<h5 className="text-sm font-medium text-slate-700">
Attendance Geofence
</h5>
<label className="inline-flex items-center gap-2 text-sm text-slate-700">
<input
type="checkbox"
checked={geofenceEnabled}
onChange={(e) => setGeofenceEnabled(e.target.checked)}
/>
Enabled
</label>
</div>
<div className="grid grid-cols-1 md:grid-cols-3 gap-3">
<div>
<label className="block text-xs text-slate-500 mb-1">
Latitude
</label>
<input
type="number"
step="any"
value={geofenceLatitude}
onChange={(e) => setGeofenceLatitude(e.target.value)}
className="w-full border border-gray-300 rounded px-3 py-2"
placeholder="e.g. 37.7749"
/>
</div>
<div>
<label className="block text-xs text-slate-500 mb-1">
Longitude
</label>
<input
type="number"
step="any"
value={geofenceLongitude}
onChange={(e) => setGeofenceLongitude(e.target.value)}
className="w-full border border-gray-300 rounded px-3 py-2"
placeholder="e.g. -122.4194"
/>
</div>
<div>
<label className="block text-xs text-slate-500 mb-1">
Radius (meters)
</label>
<input
type="number"
min="1"
value={geofenceRadiusMeters}
onChange={(e) =>
setGeofenceRadiusMeters(e.target.value)
}
className="w-full border border-gray-300 rounded px-3 py-2"
/>
</div>
</div>
<div className="flex items-center justify-between">
<p className="text-xs text-slate-500">
Default radius is 30m and geofence is enabled by default.
</p>
<Button
size="sm"
onClick={handleSaveGeofence}
disabled={savingGeofence}
>
{savingGeofence ? (
<Loader2 className="w-4 h-4 animate-spin" />
) : (
"Save Geofence"
)}
</Button>
</div>
</div>
{/* Stats */}
{statsLoading ? (
<div className="flex items-center justify-center p-8">
<Loader2 className="w-6 h-6 animate-spin text-slate-400" />
</div>
) : gymStats ? (
<div className="grid grid-cols-2 md:grid-cols-4 gap-4">
<div className="p-4 bg-blue-50 rounded-lg">
<div className="flex items-center gap-2 mb-1">
<Users className="w-4 h-4 text-blue-600" />
<span className="text-xs text-blue-600">
Total Users
</span>
</div>
<p className="text-2xl font-bold text-blue-700">
{gymStats.totalUsers}
</p>
</div>
<div className="p-4 bg-green-50 rounded-lg">
<div className="flex items-center gap-2 mb-1">
<CalendarCheck className="w-4 h-4 text-green-600" />
<span className="text-xs text-green-600">
Active Clients
</span>
</div>
<p className="text-2xl font-bold text-green-700">
{gymStats.activeClients}
</p>
</div>
<div className="p-4 bg-purple-50 rounded-lg">
<div className="flex items-center gap-2 mb-1">
<TrendingUp className="w-4 h-4 text-purple-600" />
<span className="text-xs text-purple-600">
Check-ins (30d)
</span>
</div>
<p className="text-2xl font-bold text-purple-700">
{gymStats.attendanceLast30Days}
</p>
</div>
<div className="p-4 bg-orange-50 rounded-lg">
<div className="flex items-center gap-2 mb-1">
<Users className="w-4 h-4 text-orange-600" />
<span className="text-xs text-orange-600">
Trainers
</span>
</div>
<p className="text-2xl font-bold text-orange-700">
{gymStats.trainers}
</p>
</div>
</div>
) : null}
{/* Membership Distribution */}
{gymStats && (
<div className="mt-4">
<h5 className="text-sm font-medium text-slate-700 mb-2">
Membership Distribution
</h5>
<div className="flex gap-4">
<div className="flex-1 p-3 bg-slate-100 rounded-lg text-center">
<p className="text-2xl font-bold text-slate-700">
{gymStats.membershipStats.basic}
</p>
<p className="text-xs text-slate-500">Basic</p>
</div>
<div className="flex-1 p-3 bg-blue-50 rounded-lg text-center">
<p className="text-2xl font-bold text-blue-700">
{gymStats.membershipStats.premium}
</p>
<p className="text-xs text-blue-500">Premium</p>
</div>
<div className="flex-1 p-3 bg-yellow-50 rounded-lg text-center">
<p className="text-2xl font-bold text-yellow-700">
{gymStats.membershipStats.vip}
</p>
<p className="text-xs text-yellow-600">VIP</p>
</div>
</div>
</div>
)}
{/* Membership Feature Access */}
<div className="mt-6">
<h5 className="text-sm font-medium text-slate-700 mb-2">
Membership Feature Access
</h5>
<div className="overflow-x-auto border rounded-lg">
<table className="w-full text-left text-sm">
<thead className="bg-slate-50 border-b">
<tr>
<th className="px-4 py-3 font-semibold text-slate-900">
Feature
</th>
<th className="px-4 py-3 font-semibold text-slate-900">
Basic
</th>
<th className="px-4 py-3 font-semibold text-slate-900">
Premium
</th>
<th className="px-4 py-3 font-semibold text-slate-900">
VIP
</th>
</tr>
</thead>
<tbody className="divide-y">
<tr>
<td className="px-4 py-3 text-slate-700">
Recommendations per month
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.basic.recommendationsPerMonth}
</td>
<td className="px-4 py-3 text-slate-700">
Unlimited
</td>
<td className="px-4 py-3 text-slate-700">
Unlimited
</td>
</tr>
<tr>
<td className="px-4 py-3 text-slate-700">
Nutrition tracking
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.basic.nutritionTracking
? "Yes"
: "No"}
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.premium.nutritionTracking
? "Yes"
: "No"}
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.vip.nutritionTracking
? "Yes"
: "No"}
</td>
</tr>
<tr>
<td className="px-4 py-3 text-slate-700">
Hydration tracking
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.basic.hydrationTracking
? "Yes"
: "No"}
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.premium.hydrationTracking
? "Yes"
: "No"}
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.vip.hydrationTracking
? "Yes"
: "No"}
</td>
</tr>
<tr>
<td className="px-4 py-3 text-slate-700">
Advanced statistics
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.basic.advancedStatistics
? "Yes"
: "No"}
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.premium.advancedStatistics
? "Yes"
: "No"}
</td>
<td className="px-4 py-3 text-slate-700">
{MEMBERSHIP_FEATURES.vip.advancedStatistics
? "Yes"
: "No"}
</td>
</tr>
</tbody>
</table>
</div>
</div>
<Button
variant="default"
className="mt-4"
onClick={() => handleSelectGym(gym.id)}
>
Select this gym
</Button>
</div>
))
)}
) : (
<div className="flex items-center justify-center h-64 bg-slate-50 rounded-lg">
<p className="text-slate-500">Select a gym to view details</p>
</div>
)}
</div>
</div>
</div>
{/* Database Management */}
<div className="bg-white rounded-xl shadow-sm border border-slate-100 p-6">
<div className="flex items-center justify-between mb-6">
<div className="flex items-center gap-3">

View File

@ -0,0 +1,298 @@
"use client";
import { useState, useEffect } from "react";
import { User, TrainerClientAssignment } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import { Button } from "@/components/ui/button";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import { Badge } from "@/components/ui/badge";
import { Plus, UserCheck, UserX } from "lucide-react";
export default function TrainerClientsPage() {
const [trainers, setTrainers] = useState<User[]>([]);
const [clients, setClients] = useState<User[]>([]);
const [assignments, setAssignments] = useState<
(TrainerClientAssignment & { trainer?: User; client?: User })[]
>([]);
const [loading, setLoading] = useState(true);
const [selectedTrainer, setSelectedTrainer] = useState<string>("");
const [selectedClient, setSelectedClient] = useState<string>("");
useEffect(() => {
fetchData();
}, []);
const fetchData = async () => {
try {
setLoading(true);
const [trainersRes, clientsRes, assignmentsRes] = await Promise.all([
fetch("/api/users?role=trainer"),
fetch("/api/users?role=client"),
fetch("/api/trainer-client"),
]);
const fetchedTrainers: User[] = trainersRes.ok
? (await trainersRes.json()).data?.users || []
: [];
const fetchedClients: User[] = clientsRes.ok
? (await clientsRes.json()).data?.users || []
: [];
setTrainers(fetchedTrainers);
setClients(fetchedClients);
if (assignmentsRes.ok) {
const assignmentsData = await assignmentsRes.json();
const enrichedAssignments = (assignmentsData.assignments || []).map(
(assignment: TrainerClientAssignment) => {
return {
...assignment,
trainer: fetchedTrainers.find(
(t) => t.id === assignment.trainerId,
),
client: fetchedClients.find((c) => c.id === assignment.clientId),
};
},
);
setAssignments(enrichedAssignments);
}
} catch (error) {
console.error("Failed to fetch data:", error);
} finally {
setLoading(false);
}
};
const handleAssign = async () => {
if (!selectedTrainer || !selectedClient) {
alert("Please select both a trainer and a client");
return;
}
try {
const response = await fetch("/api/trainer-client", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
trainerId: selectedTrainer,
clientId: selectedClient,
}),
});
if (response.ok) {
alert("Assignment created successfully!");
setSelectedTrainer("");
setSelectedClient("");
fetchData();
} else {
const error = await response.json();
alert(error.error || "Failed to create assignment");
}
} catch (error) {
console.error("Failed to create assignment:", error);
alert("Failed to create assignment");
}
};
const handleRemoveAssignment = async (id: string) => {
if (!confirm("Are you sure you want to remove this assignment?")) {
return;
}
try {
const response = await fetch(`/api/trainer-client/${id}`, {
method: "DELETE",
});
if (response.ok) {
alert("Assignment removed successfully!");
fetchData();
} else {
const error = await response.json();
alert(error.error || "Failed to remove assignment");
}
} catch (error) {
console.error("Failed to remove assignment:", error);
alert("Failed to remove assignment");
}
};
const getActiveAssignments = () => assignments.filter((a) => a.isActive);
const getInactiveAssignments = () => assignments.filter((a) => !a.isActive);
if (loading) {
return (
<div className="min-h-screen bg-gray-50">
<div className="container mx-auto px-4 py-8">
<div className="flex items-center justify-center h-64">
<div className="text-lg">Loading...</div>
</div>
</div>
</div>
);
}
return (
<div className="min-h-screen bg-gray-50">
<div className="container mx-auto px-4 py-8 space-y-6">
<div className="flex justify-between items-center">
<h1 className="text-3xl font-bold text-gray-900">
Trainer-Client Assignments
</h1>
</div>
<Card>
<CardHeader>
<CardTitle>Assign Trainer to Client</CardTitle>
</CardHeader>
<CardContent>
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
<div className="space-y-2">
<label className="text-sm font-medium">Select Trainer</label>
<Select
value={selectedTrainer}
onValueChange={setSelectedTrainer}
>
<SelectTrigger>
<SelectValue placeholder="Choose a trainer..." />
</SelectTrigger>
<SelectContent>
{trainers.map((trainer) => (
<SelectItem key={trainer.id} value={trainer.id}>
{trainer.firstName} {trainer.lastName}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Select Client</label>
<Select
value={selectedClient}
onValueChange={setSelectedClient}
>
<SelectTrigger>
<SelectValue placeholder="Choose a client..." />
</SelectTrigger>
<SelectContent>
{clients.map((client) => (
<SelectItem key={client.id} value={client.id}>
{client.firstName} {client.lastName}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="flex items-end">
<Button onClick={handleAssign} className="w-full">
<Plus className="w-4 h-4 mr-2" />
Assign
</Button>
</div>
</div>
</CardContent>
</Card>
<Card>
<CardHeader>
<div className="flex justify-between items-center">
<CardTitle>Active Assignments</CardTitle>
<Badge variant="default">{getActiveAssignments().length}</Badge>
</div>
</CardHeader>
<CardContent>
{getActiveAssignments().length === 0 ? (
<p className="text-gray-500 text-center py-8">
No active assignments
</p>
) : (
<div className="space-y-2">
{getActiveAssignments().map((assignment) => (
<div
key={assignment.id}
className="flex items-center justify-between p-4 border rounded-lg hover:bg-gray-50"
>
<div className="flex items-center space-x-4">
<div className="flex items-center space-x-2">
<UserCheck className="w-4 h-4 text-green-600" />
<span className="font-medium">
{assignment.trainer?.firstName}{" "}
{assignment.trainer?.lastName}
</span>
</div>
<span className="text-gray-400"></span>
<span>
{assignment.client?.firstName}{" "}
{assignment.client?.lastName}
</span>
</div>
<div className="flex items-center space-x-2">
<span className="text-xs text-gray-500">
Assigned{" "}
{new Date(assignment.assignedAt).toLocaleDateString()}
</span>
<Button
variant="ghost"
size="sm"
onClick={() => handleRemoveAssignment(assignment.id)}
>
<UserX className="w-4 h-4 text-red-600" />
</Button>
</div>
</div>
))}
</div>
)}
</CardContent>
</Card>
{getInactiveAssignments().length > 0 && (
<Card>
<CardHeader>
<div className="flex justify-between items-center">
<CardTitle className="text-gray-500">
Inactive Assignments
</CardTitle>
<Badge variant="secondary">
{getInactiveAssignments().length}
</Badge>
</div>
</CardHeader>
<CardContent>
<div className="space-y-2">
{getInactiveAssignments().map((assignment) => (
<div
key={assignment.id}
className="flex items-center justify-between p-4 border rounded-lg bg-gray-50 opacity-60"
>
<div className="flex items-center space-x-4">
<span className="font-medium text-gray-600">
{assignment.trainer?.firstName}{" "}
{assignment.trainer?.lastName}
</span>
<span className="text-gray-400"></span>
<span className="text-gray-600">
{assignment.client?.firstName}{" "}
{assignment.client?.lastName}
</span>
</div>
<Badge variant="secondary">Inactive</Badge>
</div>
))}
</div>
</CardContent>
</Card>
)}
</div>
</div>
);
}

View File

@ -1,11 +1,30 @@
import { UserManagement } from '@/components/users/UserManagement'
"use client";
import { UserManagement } from "@/components/users/UserManagement";
import { PageHeader } from "@/components/ui/PageHeader";
import { useUser } from "@clerk/nextjs";
import { useSearchParams } from "next/navigation";
import { GymSelector } from "@/components/gym/GymSelector";
export default function UsersPage() {
const { user } = useUser();
const searchParams = useSearchParams();
const userRole = (user?.publicMetadata?.role as string) ?? "client";
const gymId = searchParams.get("gymId") ?? undefined;
return (
<main className="min-h-screen bg-gray-50">
<div className="container mx-auto px-4 py-8">
<UserManagement />
<div className="space-y-8">
<PageHeader
title="Users"
description="Manage your gym members, trainers, and administrators"
breadcrumbs={[{ label: "Users", href: "/users" }]}
actions={<GymSelector userRole={userRole} />}
/>
<div className="card-modern">
<UserManagement gymId={gymId} />
</div>
</main>
)
}
</div>
);
}

View File

@ -71,11 +71,11 @@ export function Navigation(): ReactElement {
<li key={item.href}>
<Link href={item.href} className="flex items-center gap-2">
<Button
variant={pathname === item.href ? "primary" : "secondary"}
variant={pathname === item.href ? "default" : "secondary"}
className={cn(
"h-9 px-4 py-2",
pathname === item.href &&
"bg-primary text-primary-foreground",
"bg-primary text-primary-foreground",
)}
aria-current={pathname === item.href ? "page" : undefined}
>

View File

@ -1,75 +1,38 @@
'use client'
"use client";
import { useState, useEffect } from 'react'
import { UserGrowthChart } from '@/components/charts/UserGrowthChart'
import { MembershipDistributionChart } from '@/components/charts/MembershipDistributionChart'
import { RevenueChart } from '@/components/charts/RevenueChart'
import { Card, CardHeader, CardContent } from '@/components/ui/card'
import { UserGrowthChart } from "@/components/charts/UserGrowthChart";
import { MembershipDistributionChart } from "@/components/charts/MembershipDistributionChart";
import { RevenueChart } from "@/components/charts/RevenueChart";
import { Card, CardHeader, CardContent } from "@/components/ui/card";
import { useAnalytics } from "@/hooks/use-api";
interface ChartData {
label: string
value: number
color?: string
interface AnalyticsDashboardProps {
gymId?: string;
}
export function AnalyticsDashboard() {
const [userGrowthData, setUserGrowthData] = useState<ChartData[]>([])
const [membershipData, setMembershipData] = useState<ChartData[]>([])
const [revenueData, setRevenueData] = useState<ChartData[]>([])
const [loading, setLoading] = useState(true)
export function AnalyticsDashboard({ gymId }: AnalyticsDashboardProps) {
const { data: analytics, isLoading } = useAnalytics(6, gymId);
useEffect(() => {
fetchAnalyticsData()
}, [])
const userGrowthData = analytics?.userGrowth ?? [];
const membershipData = analytics?.membershipDistribution ?? [];
const revenueData = analytics?.revenue ?? [];
const fetchAnalyticsData = async () => {
setLoading(true)
try {
// Mock data for demonstration - replace with real API calls
const mockUserGrowth = [
{ label: 'Jan', value: 45 },
{ label: 'Feb', value: 52 },
{ label: 'Mar', value: 61 },
{ label: 'Apr', value: 58 },
{ label: 'May', value: 67 },
{ label: 'Jun', value: 74 },
]
const totalUsers =
userGrowthData.length > 0
? userGrowthData[userGrowthData.length - 1].value
: 0;
const totalRevenue = revenueData.reduce((sum, item) => sum + item.value, 0);
const activeMembers = membershipData.reduce(
(sum, item) => sum + item.value,
0,
);
const mockMembershipData = [
{ label: 'Basic', value: 45, color: '#6b7280' },
{ label: 'Premium', value: 28, color: '#3b82f6' },
{ label: 'VIP', value: 12, color: '#f59e0b' },
]
const mockRevenueData = [
{ label: 'Jan', value: 12500, color: '#10b981' },
{ label: 'Feb', value: 14200, color: '#10b981' },
{ label: 'Mar', value: 16800, color: '#10b981' },
{ label: 'Apr', value: 15900, color: '#10b981' },
{ label: 'May', value: 18200, color: '#10b981' },
{ label: 'Jun', value: 19400, color: '#10b981' },
]
setUserGrowthData(mockUserGrowth)
setMembershipData(mockMembershipData)
setRevenueData(mockRevenueData)
} catch (error) {
console.error('Failed to fetch analytics data:', error)
} finally {
setLoading(false)
}
}
const totalUsers = userGrowthData.length > 0 ? userGrowthData[userGrowthData.length - 1].value : 0
const totalRevenue = revenueData.reduce((sum, item) => sum + item.value, 0)
const activeMembers = membershipData.reduce((sum, item) => sum + item.value, 0)
if (loading) {
if (isLoading) {
return (
<div className="flex justify-center items-center h-64">
<div className="text-lg">Loading analytics...</div>
</div>
)
);
}
return (
@ -81,7 +44,9 @@ export function AnalyticsDashboard() {
<Card>
<CardContent>
<div className="text-center">
<div className="text-3xl font-bold text-blue-600">{totalUsers}</div>
<div className="text-3xl font-bold text-blue-600">
{totalUsers}
</div>
<div className="text-gray-600">Total Users</div>
</div>
</CardContent>
@ -90,7 +55,9 @@ export function AnalyticsDashboard() {
<Card>
<CardContent>
<div className="text-center">
<div className="text-3xl font-bold text-green-600">${totalRevenue.toLocaleString()}</div>
<div className="text-3xl font-bold text-green-600">
${totalRevenue.toLocaleString()}
</div>
<div className="text-gray-600">Total Revenue</div>
</div>
</CardContent>
@ -99,7 +66,9 @@ export function AnalyticsDashboard() {
<Card>
<CardContent>
<div className="text-center">
<div className="text-3xl font-bold text-purple-600">{activeMembers}</div>
<div className="text-3xl font-bold text-purple-600">
{activeMembers}
</div>
<div className="text-gray-600">Active Members</div>
</div>
</CardContent>
@ -133,14 +102,14 @@ export function AnalyticsDashboard() {
</CardHeader>
<CardContent>
<RevenueChart
data={revenueData.map(item => ({
data={revenueData.map((item) => ({
category: item.label,
value: item.value,
color: item.color
color: item.color,
}))}
/>
</CardContent>
</Card>
</div>
)
);
}

View File

@ -0,0 +1,132 @@
"use client";
import React from "react";
import { AgCharts } from "ag-charts-react";
import {
AgBarSeriesOptions,
AgChartOptions,
AgLineSeriesOptions,
AgPieSeriesOptions,
} from "ag-charts-community";
export function BarChart({
data,
xKey,
yKey,
color = "#3c82e6",
height = 300,
}: {
data: any[];
xKey: string;
yKey: string;
color?: string;
height?: number;
}) {
const options: AgChartOptions = {
data,
series: [
{
type: "bar",
xKey,
yKey,
fill: color,
} as AgBarSeriesOptions,
],
height,
};
return <AgCharts options={options} />;
}
export function LineChart({
data,
xKey,
yKey,
color = "#3b82f6",
height = 300,
}: {
data: any[];
xKey: string;
yKey: string;
color?: string;
height?: number;
}) {
const options: AgChartOptions = {
data,
series: [
{
type: "line",
xKey,
yKey,
stroke: color,
strokeWidth: 3,
marker: {
size: 6,
fill: color,
stroke: "#ffffff",
strokeWidth: 2,
},
} as AgLineSeriesOptions,
],
height,
};
return <AgCharts options={options} />;
}
export function PieChart({
data,
labelKey,
valueKey,
height = 300,
}: {
data: any[];
labelKey: string;
valueKey: string;
height?: number;
}) {
const options: AgChartOptions = {
data,
series: [
{
type: "pie",
labelKey,
angleKey: valueKey,
} as AgPieSeriesOptions,
],
height,
};
return <AgCharts options={options} />;
}
export function AreaChart({
data,
xKey,
yKey,
color = "#3b82f6",
height = 300,
}: {
data: any[];
xKey: string;
yKey: string;
color?: string;
height?: number;
}) {
const options: AgChartOptions = {
data,
series: [
{
type: "area",
xKey,
yKey,
fill: color,
stroke: color,
strokeWidth: 2,
},
],
height,
};
return <AgCharts options={options} />;
}

View File

@ -0,0 +1,91 @@
"use client";
import { useRouter, useSearchParams } from "next/navigation";
import { useEffect, useState } from "react";
interface Gym {
id: string;
name: string;
}
interface GymSelectorProps {
currentGymId?: string;
userRole: string;
}
export function GymSelector({ userRole }: GymSelectorProps) {
const router = useRouter();
const searchParams = useSearchParams();
const [gyms, setGyms] = useState<Gym[]>([]);
const [loading, setLoading] = useState(true);
useEffect(() => {
// Only fetch gyms for superAdmin
if (userRole !== "superAdmin") {
setLoading(false);
return;
}
async function fetchGyms() {
try {
const response = await fetch("/api/gyms");
const data = await response.json();
// API returns array directly
if (Array.isArray(data)) {
setGyms(data);
}
} catch (error) {
console.error("Failed to fetch gyms:", error);
} finally {
setLoading(false);
}
}
fetchGyms();
}, [userRole]);
const handleGymChange = (gymId: string) => {
const params = new URLSearchParams(searchParams.toString());
if (gymId === "all") {
params.delete("gymId");
} else {
params.set("gymId", gymId);
}
router.push(`?${params.toString()}`);
};
// Only show for superAdmin
if (userRole !== "superAdmin") {
return null;
}
const selectedGymId = searchParams.get("gymId") ?? "all";
if (loading) {
return (
<div className="flex items-center gap-2">
<div className="h-9 w-40 animate-pulse rounded-md bg-muted" />
</div>
);
}
return (
<div className="flex items-center gap-2">
<select
value={selectedGymId}
onChange={(e) => handleGymChange(e.target.value)}
className="h-9 rounded-md border border-input bg-background px-3 py-1 text-sm shadow-sm transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring"
>
<option value="all">All Gyms</option>
{gyms.map((gym) => (
<option key={gym.id} value={gym.id}>
{gym.name}
</option>
))}
</select>
</div>
);
}

View File

@ -0,0 +1,26 @@
"use client";
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
import { useState, type ReactNode } from "react";
interface QueryProviderProps {
children: ReactNode;
}
export function QueryProvider({ children }: QueryProviderProps) {
const [queryClient] = useState(
() =>
new QueryClient({
defaultOptions: {
queries: {
staleTime: 60 * 1000,
refetchOnWindowFocus: false,
},
},
}),
);
return (
<QueryClientProvider client={queryClient}>{children}</QueryClientProvider>
);
}

View File

@ -0,0 +1,7 @@
"use client";
import { Toaster } from "sonner";
export function ToasterProvider() {
return <Toaster richColors position="top-right" />;
}

View File

@ -0,0 +1,108 @@
"use client";
import { GoalSummary } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import { Badge } from "@/components/ui/badge";
interface GoalsSummaryCardProps {
goals: GoalSummary;
}
export function GoalsSummaryCard({ goals }: GoalsSummaryCardProps) {
return (
<Card>
<CardHeader>
<CardTitle>Fitness Goals</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
{/* Summary Stats */}
<div className="grid grid-cols-3 gap-4">
<div className="text-center">
<div className="text-2xl font-bold text-blue-600">
{goals.totalActive}
</div>
<div className="text-xs text-gray-600">Active Goals</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-green-600">
{goals.totalCompleted}
</div>
<div className="text-xs text-gray-600">Completed</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-purple-600">
{goals.averageProgress}%
</div>
<div className="text-xs text-gray-600">Avg Progress</div>
</div>
</div>
{/* Active Goals */}
{goals.active.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold">Active Goals</h4>
<div className="space-y-2">
{goals.active.map((goal) => (
<div key={goal.id} className="border rounded-lg p-3">
<div className="flex justify-between items-start mb-2">
<div className="font-medium">{goal.title}</div>
<Badge
variant={
goal.priority === "high" ? "destructive" : "secondary"
}
>
{goal.priority}
</Badge>
</div>
<div className="text-xs text-gray-600 mb-2">
{goal.goalType.replace("_", " ")}
</div>
{/* Progress Bar */}
<div className="w-full bg-gray-200 rounded-full h-2">
<div
className="bg-blue-600 h-2 rounded-full transition-all"
style={{ width: `${goal.progress}%` }}
/>
</div>
<div className="text-xs text-gray-600 mt-1">
{goal.progress}% complete
</div>
</div>
))}
</div>
</div>
)}
{/* Completed Goals */}
{goals.completed.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold">Completed Goals</h4>
<div className="space-y-1">
{goals.completed.map((goal) => (
<div
key={goal.id}
className="flex justify-between items-center text-sm p-2 bg-green-50 rounded"
>
<span className="font-medium">{goal.title}</span>
<span className="text-xs text-green-700">
{" "}
{goal.completedDate
? new Date(goal.completedDate).toLocaleDateString()
: "N/A"}
</span>
</div>
))}
</div>
</div>
)}
{/* Empty State */}
{goals.active.length === 0 && goals.completed.length === 0 && (
<div className="text-center text-gray-500 py-8">
No goals found for this period
</div>
)}
</CardContent>
</Card>
);
}

View File

@ -0,0 +1,119 @@
"use client";
import { HydrationSummary } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import {
PieChart,
Pie,
Cell,
ResponsiveContainer,
Legend,
Tooltip,
} from "recharts";
interface HydrationSummaryCardProps {
hydration: {
dailySummaries: HydrationSummary[];
averageDailyWater: number;
totalDays: number;
daysMetGoal: number;
};
}
export function HydrationSummaryCard({ hydration }: HydrationSummaryCardProps) {
const goalMetPercentage =
hydration.totalDays > 0
? Math.round((hydration.daysMetGoal / hydration.totalDays) * 100)
: 0;
const chartData = [
{ name: "Days Met Goal", value: hydration.daysMetGoal, color: "#06b6d4" },
{
name: "Days Under Goal",
value: hydration.totalDays - hydration.daysMetGoal,
color: "#f59e0b",
},
];
return (
<Card>
<CardHeader>
<CardTitle>Hydration Summary</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
{/* Summary Stats */}
<div className="grid grid-cols-2 gap-4">
<div className="text-center">
<div className="text-2xl font-bold text-blue-600">
{(hydration.averageDailyWater / 1000).toFixed(1)}L
</div>
<div className="text-xs text-gray-600">Avg Daily Water</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-cyan-600">
{hydration.totalDays}
</div>
<div className="text-xs text-gray-600">Days Tracked</div>
</div>
</div>
{/* Goal Achievement */}
<div className="text-center">
<div className="text-3xl font-bold text-green-600">
{goalMetPercentage}%
</div>
<div className="text-xs text-gray-600">Days Met Hydration Goal</div>
</div>
{/* Chart */}
{hydration.totalDays > 0 && (
<div className="h-48">
<ResponsiveContainer width="100%" height="100%">
<PieChart>
<Pie
data={chartData}
cx="50%"
cy="50%"
innerRadius={40}
outerRadius={70}
paddingAngle={5}
dataKey="value"
>
{chartData.map((entry, index) => (
<Cell key={`cell-${index}`} fill={entry.color} />
))}
</Pie>
<Tooltip />
<Legend />
</PieChart>
</ResponsiveContainer>
</div>
)}
{/* Recent Days */}
{hydration.dailySummaries.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold">Recent Days</h4>
<div className="max-h-32 overflow-y-auto space-y-1">
{hydration.dailySummaries
.slice(-5)
.reverse()
.map((day) => (
<div
key={day.date}
className="flex justify-between items-center text-sm"
>
<span className="text-gray-600">{day.date}</span>
<span className="font-medium">
{day.totalWater}ml / {day.waterGoal}ml (
{day.hydrationPercentage}%)
</span>
</div>
))}
</div>
</div>
)}
</CardContent>
</Card>
);
}

View File

@ -0,0 +1,121 @@
"use client";
import { NutritionSummary } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import {
PieChart,
Pie,
Cell,
ResponsiveContainer,
Legend,
Tooltip,
} from "recharts";
interface NutritionSummaryCardProps {
nutrition: {
dailySummaries: NutritionSummary[];
averageDailyCalories: number;
totalDays: number;
daysMetGoal: number;
};
}
export function NutritionSummaryCard({ nutrition }: NutritionSummaryCardProps) {
const goalMetPercentage =
nutrition.totalDays > 0
? Math.round((nutrition.daysMetGoal / nutrition.totalDays) * 100)
: 0;
const chartData = [
{ name: "Days Met Goal", value: nutrition.daysMetGoal, color: "#10b981" },
{
name: "Days Over/Under",
value: nutrition.totalDays - nutrition.daysMetGoal,
color: "#f59e0b",
},
];
return (
<Card>
<CardHeader>
<CardTitle>Nutrition Summary</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
{/* Summary Stats */}
<div className="grid grid-cols-2 gap-4">
<div className="text-center">
<div className="text-2xl font-bold text-blue-600">
{nutrition.averageDailyCalories.toLocaleString()}
</div>
<div className="text-xs text-gray-600">Avg Daily Calories</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-green-600">
{nutrition.totalDays}
</div>
<div className="text-xs text-gray-600">Days Tracked</div>
</div>
</div>
{/* Goal Achievement */}
<div className="text-center">
<div className="text-3xl font-bold text-purple-600">
{goalMetPercentage}%
</div>
<div className="text-xs text-gray-600">
Days Met Calorie Goal (±10%)
</div>
</div>
{/* Chart */}
{nutrition.totalDays > 0 && (
<div className="h-48">
<ResponsiveContainer width="100%" height="100%">
<PieChart>
<Pie
data={chartData}
cx="50%"
cy="50%"
innerRadius={40}
outerRadius={70}
paddingAngle={5}
dataKey="value"
>
{chartData.map((entry, index) => (
<Cell key={`cell-${index}`} fill={entry.color} />
))}
</Pie>
<Tooltip />
<Legend />
</PieChart>
</ResponsiveContainer>
</div>
)}
{/* Recent Days */}
{nutrition.dailySummaries.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold">Recent Days</h4>
<div className="max-h-32 overflow-y-auto space-y-1">
{nutrition.dailySummaries
.slice(-5)
.reverse()
.map((day) => (
<div
key={day.date}
className="flex justify-between items-center text-sm"
>
<span className="text-gray-600">{day.date}</span>
<span className="font-medium">
{day.totalCalories.toLocaleString()} /{" "}
{day.calorieGoal.toLocaleString()} cal
</span>
</div>
))}
</div>
</div>
)}
</CardContent>
</Card>
);
}

View File

@ -0,0 +1,130 @@
"use client";
import { RecommendationSummary } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import { Badge } from "@/components/ui/badge";
interface RecommendationsCardProps {
recommendations: RecommendationSummary;
}
export function RecommendationsCard({
recommendations,
}: RecommendationsCardProps) {
return (
<Card>
<CardHeader>
<CardTitle>AI Recommendations</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
{/* Summary Stats */}
<div className="grid grid-cols-3 gap-4">
<div className="text-center">
<div className="text-2xl font-bold text-green-600">
{recommendations.totalAccepted}
</div>
<div className="text-xs text-gray-600">Accepted</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-yellow-600">
{recommendations.totalPending}
</div>
<div className="text-xs text-gray-600">Pending</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-red-600">
{recommendations.totalRejected}
</div>
<div className="text-xs text-gray-600">Rejected</div>
</div>
</div>
{/* Accepted Recommendations */}
{recommendations.accepted.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold text-green-700">
Accepted Recommendations
</h4>
<div className="space-y-2">
{recommendations.accepted.map((rec) => (
<div
key={rec.id}
className="border border-green-200 bg-green-50 rounded-lg p-3"
>
<div className="flex justify-between items-start mb-2">
<Badge variant="default" className="bg-green-600">
Accepted
</Badge>
<span className="text-xs text-gray-500">
{new Date(rec.generatedAt).toLocaleDateString()}
</span>
</div>
<div className="text-sm">{rec.recommendationText}</div>
</div>
))}
</div>
</div>
)}
{/* Pending Recommendations */}
{recommendations.pending.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold text-yellow-700">
Pending Recommendations
</h4>
<div className="space-y-2">
{recommendations.pending.map((rec) => (
<div
key={rec.id}
className="border border-yellow-200 bg-yellow-50 rounded-lg p-3"
>
<div className="flex justify-between items-start mb-2">
<Badge variant="secondary">Pending</Badge>
<span className="text-xs text-gray-500">
{new Date(rec.generatedAt).toLocaleDateString()}
</span>
</div>
<div className="text-sm">{rec.recommendationText}</div>
</div>
))}
</div>
</div>
)}
{/* Rejected Recommendations */}
{recommendations.rejected.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold text-red-700">
Rejected Recommendations
</h4>
<div className="space-y-2">
{recommendations.rejected.map((rec) => (
<div
key={rec.id}
className="border border-red-200 bg-red-50 rounded-lg p-3"
>
<div className="flex justify-between items-start mb-2">
<Badge variant="destructive">Rejected</Badge>
<span className="text-xs text-gray-500">
{new Date(rec.generatedAt).toLocaleDateString()}
</span>
</div>
<div className="text-sm">{rec.recommendationText}</div>
</div>
))}
</div>
</div>
)}
{/* Empty State */}
{recommendations.totalAccepted === 0 &&
recommendations.totalPending === 0 &&
recommendations.totalRejected === 0 && (
<div className="text-center text-gray-500 py-8">
No recommendations found for this period
</div>
)}
</CardContent>
</Card>
);
}

View File

@ -0,0 +1,234 @@
"use client";
import { useState, useEffect } from "react";
import { User, Client } from "@fitai/shared";
import { Button } from "@/components/ui/button";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import { Label } from "@/components/ui/label";
import { Card, CardContent } from "@/components/ui/card";
import { useUser } from "@clerk/nextjs";
import log from "@/lib/logger";
interface ReportFiltersProps {
selectedUserId: string;
onUserChange: (userId: string) => void;
dateRange: {
startDate: string;
endDate: string;
};
onDateRangeChange: (range: { startDate: string; endDate: string }) => void;
}
export function ReportFilters({
selectedUserId,
onUserChange,
dateRange,
onDateRangeChange,
}: ReportFiltersProps) {
const { user: clerkUser } = useUser();
const [users, setUsers] = useState<(User & { client?: Client | null })[]>([]);
const [loading, setLoading] = useState(true);
const [currentUser, setCurrentUser] = useState<User | null>(null);
useEffect(() => {
fetchCurrentUserAndClients();
}, []);
const fetchCurrentUserAndClients = async () => {
try {
setLoading(true);
const userResponse = await fetch("/api/users/me");
if (!userResponse.ok) {
setLoading(false);
return;
}
const userData = await userResponse.json();
setCurrentUser(userData.user);
const currentRole = userData.user.role;
let allUsers: (User & { client?: Client | null })[] = [];
if (currentRole === "client") {
setUsers([userData.user]);
onUserChange(userData.user.id);
setLoading(false);
return;
}
if (currentRole === "trainer") {
allUsers.push(userData.user);
const assignmentsRes = await fetch("/api/trainer-client");
if (assignmentsRes.ok) {
const assignmentsData = await assignmentsRes.json();
const assignedClientIds = (assignmentsData.assignments || [])
.filter((a: any) => a.isActive)
.map((a: any) => a.clientId);
if (assignedClientIds.length > 0) {
const clientsRes = await fetch(
`/api/users?role=client&ids=${assignedClientIds.join(",")}`,
);
if (clientsRes.ok) {
const clientsData = await clientsRes.json();
allUsers = [...allUsers, ...(clientsData.data?.users || [])];
}
}
}
setUsers(allUsers);
setLoading(false);
return;
}
if (currentRole === "admin") {
allUsers.push(userData.user);
}
if (currentRole === "superAdmin" || currentRole === "admin") {
const [adminsRes, trainersRes, clientsRes] = await Promise.all([
fetch("/api/users?role=admin"),
fetch("/api/users?role=trainer"),
fetch("/api/users?role=client"),
]);
const [adminsData, trainersData, clientsData] = await Promise.all([
adminsRes.ok ? adminsRes.json() : { data: { users: [] } },
trainersRes.ok ? trainersRes.json() : { data: { users: [] } },
clientsRes.ok ? clientsRes.json() : { data: { users: [] } },
]);
allUsers = [
...allUsers,
...(adminsData.data?.users || []),
...(trainersData.data?.users || []),
...(clientsData.data?.users || []),
];
if (currentRole === "superAdmin") {
const superAdminsRes = await fetch("/api/users?role=superAdmin");
if (superAdminsRes.ok) {
const superAdminsData = await superAdminsRes.json();
allUsers = [...(superAdminsData.data?.users || []), ...allUsers];
}
}
setUsers(allUsers);
setLoading(false);
return;
}
setUsers(allUsers);
} catch (error) {
log.error("Failed to fetch users:", error);
} finally {
setLoading(false);
}
};
const handlePresetRange = (days: number) => {
const end = new Date();
const start = new Date(Date.now() - days * 24 * 60 * 60 * 1000);
onDateRangeChange({
startDate: start.toISOString().split("T")[0],
endDate: end.toISOString().split("T")[0],
});
};
return (
<Card>
<CardContent className="p-6">
<div className="grid grid-cols-1 md:grid-cols-4 gap-4">
{/* User Selector */}
<div className="space-y-2">
<Label htmlFor="user-select">Select User</Label>
<Select
value={selectedUserId}
onValueChange={onUserChange}
disabled={loading}
>
<SelectTrigger id="user-select">
<SelectValue placeholder="Choose a user..." />
</SelectTrigger>
<SelectContent>
{users.map((user) => (
<SelectItem key={user.id} value={user.id}>
{user.firstName} {user.lastName} ({user.role})
</SelectItem>
))}
</SelectContent>
</Select>
</div>
{/* Start Date */}
<div className="space-y-2">
<Label htmlFor="start-date">Start Date</Label>
<input
id="start-date"
type="date"
className="flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50"
value={dateRange.startDate}
onChange={(e) =>
onDateRangeChange({
...dateRange,
startDate: e.target.value,
})
}
/>
</div>
{/* End Date */}
<div className="space-y-2">
<Label htmlFor="end-date">End Date</Label>
<input
id="end-date"
type="date"
className="flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50"
value={dateRange.endDate}
onChange={(e) =>
onDateRangeChange({
...dateRange,
endDate: e.target.value,
})
}
/>
</div>
{/* Preset Ranges */}
<div className="space-y-2">
<Label>Quick Select</Label>
<div className="flex gap-2">
<Button
variant="outline"
size="sm"
onClick={() => handlePresetRange(7)}
>
7 Days
</Button>
<Button
variant="outline"
size="sm"
onClick={() => handlePresetRange(30)}
>
30 Days
</Button>
<Button
variant="outline"
size="sm"
onClick={() => handlePresetRange(90)}
>
90 Days
</Button>
</div>
</div>
</div>
</CardContent>
</Card>
);
}

View File

@ -0,0 +1,136 @@
"use client";
import { useState, useEffect } from "react";
import { UserReport as UserReportType } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import { WeeklyCheckInsCard } from "./WeeklyCheckInsCard";
import { NutritionSummaryCard } from "./NutritionSummaryCard";
import { HydrationSummaryCard } from "./HydrationSummaryCard";
import { GoalsSummaryCard } from "./GoalsSummaryCard";
import { RecommendationsCard } from "./RecommendationsCard";
import { Button } from "@/components/ui/button";
import { Download } from "lucide-react";
interface UserReportProps {
userId: string;
startDate: string;
endDate: string;
}
export function UserReport({ userId, startDate, endDate }: UserReportProps) {
const [report, setReport] = useState<UserReportType | null>(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
useEffect(() => {
fetchReport();
}, [userId, startDate, endDate]);
const fetchReport = async () => {
try {
setLoading(true);
setError(null);
const response = await fetch(
`/api/reports/user/${userId}?startDate=${startDate}&endDate=${endDate}`,
);
if (!response.ok) {
const errorData = await response.json();
throw new Error(errorData.error || "Failed to fetch report");
}
const data = await response.json();
setReport(data);
} catch (err) {
setError(err instanceof Error ? err.message : "Failed to load report");
console.error("Report fetch error:", err);
} finally {
setLoading(false);
}
};
const handleExportPDF = () => {
if (!report) return;
const filename = `FitAI_Report_${report.user.firstName}_${report.user.lastName}_${startDate}_${endDate}.pdf`;
const url = `/api/reports/user/${userId}?startDate=${startDate}&endDate=${endDate}&format=pdf`;
// Open in new tab to trigger download
window.open(url, "_blank");
};
if (loading) {
return (
<Card>
<CardContent className="flex items-center justify-center h-64">
<div className="text-lg text-gray-600">Loading report...</div>
</CardContent>
</Card>
);
}
if (error) {
return (
<Card>
<CardContent className="flex items-center justify-center h-64">
<div className="text-lg text-red-600">{error}</div>
</CardContent>
</Card>
);
}
if (!report) {
return (
<Card>
<CardContent className="flex items-center justify-center h-64">
<div className="text-lg text-gray-600">No report data available</div>
</CardContent>
</Card>
);
}
return (
<div className="space-y-6">
{/* Report Header */}
<Card>
<CardHeader>
<div className="flex justify-between items-center">
<div>
<CardTitle className="text-2xl">
{report.user.firstName} {report.user.lastName}
</CardTitle>
<p className="text-sm text-gray-600 mt-1">
{report.user.email} {report.client?.membershipType || "N/A"}{" "}
Member
</p>
<p className="text-xs text-gray-500 mt-1">
Report Period: {report.reportPeriod.startDate} to{" "}
{report.reportPeriod.endDate}
</p>
</div>
<Button onClick={handleExportPDF} variant="default">
<Download className="w-4 h-4 mr-2" />
Export PDF
</Button>
</div>
</CardHeader>
</Card>
{/* Report Sections */}
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<WeeklyCheckInsCard weeklyCheckIns={report.weeklyCheckIns} />
<NutritionSummaryCard nutrition={report.nutrition} />
</div>
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<HydrationSummaryCard hydration={report.hydration} />
<GoalsSummaryCard goals={report.goals} />
</div>
<div className="grid grid-cols-1 gap-6">
<RecommendationsCard recommendations={report.recommendations} />
</div>
</div>
);
}

View File

@ -0,0 +1,110 @@
"use client";
import { WeeklyCheckInStats } from "@fitai/shared";
import { Card, CardHeader, CardTitle, CardContent } from "@/components/ui/card";
import {
BarChart,
Bar,
XAxis,
YAxis,
CartesianGrid,
Tooltip,
Legend,
ResponsiveContainer,
} from "recharts";
interface WeeklyCheckInsCardProps {
weeklyCheckIns: WeeklyCheckInStats[];
}
export function WeeklyCheckInsCard({
weeklyCheckIns,
}: WeeklyCheckInsCardProps) {
const totalCheckIns = weeklyCheckIns.reduce(
(sum, week) => sum + week.totalCheckIns,
0,
);
const totalTimeMinutes = weeklyCheckIns.reduce(
(sum, week) => sum + week.totalTimeMinutes,
0,
);
const avgTimeMinutes =
totalCheckIns > 0 ? Math.round(totalTimeMinutes / totalCheckIns) : 0;
const chartData = weeklyCheckIns.map((week) => ({
week: week.weekStart,
checkIns: week.totalCheckIns,
timeMinutes: week.totalTimeMinutes,
}));
return (
<Card>
<CardHeader>
<CardTitle>Weekly Check-ins</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<div className="grid grid-cols-3 gap-4">
<div className="text-center">
<div className="text-2xl font-bold text-blue-600">
{totalCheckIns}
</div>
<div className="text-xs text-gray-600">Total Check-ins</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-green-600">
{Math.floor(totalTimeMinutes / 60)}h {totalTimeMinutes % 60}m
</div>
<div className="text-xs text-gray-600">Total Time</div>
</div>
<div className="text-center">
<div className="text-2xl font-bold text-purple-600">
{avgTimeMinutes}m
</div>
<div className="text-xs text-gray-600">Avg Duration</div>
</div>
</div>
{chartData.length > 0 ? (
<div className="h-64">
<ResponsiveContainer width="100%" height="100%">
<BarChart data={chartData}>
<CartesianGrid strokeDasharray="3 3" />
<XAxis dataKey="week" />
<YAxis />
<Tooltip />
<Legend />
<Bar dataKey="checkIns" fill="#3c82e6" name="Check-ins" />
<Bar dataKey="timeMinutes" fill="#10b981" name="Minutes" />
</BarChart>
</ResponsiveContainer>
</div>
) : (
<div className="h-64 flex items-center justify-center text-gray-500">
No check-in data available
</div>
)}
{weeklyCheckIns.length > 0 && (
<div className="space-y-2">
<h4 className="text-sm font-semibold">Weekly Breakdown</h4>
<div className="space-y-1">
{weeklyCheckIns.slice(-4).map((week) => (
<div
key={week.weekStart}
className="flex justify-between items-center text-sm"
>
<span className="text-gray-600">
{week.weekStart} - {week.weekEnd}
</span>
<span className="font-medium">
{week.totalCheckIns} check-ins {week.totalTimeMinutes} min
</span>
</div>
))}
</div>
</div>
)}
</CardContent>
</Card>
);
}

View File

@ -0,0 +1,101 @@
import { cn } from "@/lib/utils";
import { ChevronRight, Home } from "lucide-react";
import Link from "next/link";
interface Breadcrumb {
label: string;
href?: string;
}
interface PageHeaderProps {
title: string;
description?: string;
breadcrumbs?: Breadcrumb[];
actions?: React.ReactNode;
className?: string;
}
export function PageHeader({
title,
description,
breadcrumbs,
actions,
className,
}: PageHeaderProps) {
return (
<div className={cn("space-y-4", className)}>
{breadcrumbs && breadcrumbs.length > 0 && (
<nav className="flex items-center gap-1 text-sm">
<Link
href="/"
className="flex items-center gap-1 text-muted-foreground hover:text-foreground transition-colors"
>
<Home className="h-4 w-4" />
<span>Home</span>
</Link>
{breadcrumbs.map((crumb, index) => (
<span key={index} className="flex items-center gap-1">
<ChevronRight className="h-4 w-4 text-muted-foreground" />
{crumb.href ? (
<Link
href={crumb.href}
className="text-muted-foreground hover:text-foreground transition-colors"
>
{crumb.label}
</Link>
) : (
<span className="font-medium text-foreground">
{crumb.label}
</span>
)}
</span>
))}
</nav>
)}
<div className="flex items-start justify-between gap-4">
<div className="space-y-1">
<h1 className="text-3xl font-bold tracking-tight">{title}</h1>
{description && (
<p className="text-muted-foreground">{description}</p>
)}
</div>
{actions && <div className="flex items-center gap-3">{actions}</div>}
</div>
</div>
);
}
interface PageSectionProps {
title?: string;
description?: string;
children: React.ReactNode;
actions?: React.ReactNode;
className?: string;
}
export function PageSection({
title,
description,
children,
actions,
className,
}: PageSectionProps) {
return (
<section className={cn("space-y-4", className)}>
{(title || actions) && (
<div className="flex items-start justify-between gap-4">
<div className="space-y-1">
{title && (
<h2 className="text-xl font-semibold tracking-tight">{title}</h2>
)}
{description && (
<p className="text-sm text-muted-foreground">{description}</p>
)}
</div>
{actions && <div className="flex items-center gap-2">{actions}</div>}
</div>
)}
{children}
</section>
);
}

View File

@ -9,60 +9,48 @@ import {
CalendarCheck,
CreditCard,
Settings,
LogOut,
Brain,
ChevronLeft,
Activity,
} from "lucide-react";
import { UserButton, useUser } from "@clerk/nextjs";
import { usePendingRecommendationsCount } from "@/hooks/use-api";
import { cn } from "@/lib/utils";
interface Recommendation {
id: string;
status: string;
interface NavItem {
icon: React.ElementType;
label: string;
href: string;
badge?: number;
}
export function Sidebar() {
const pathname = usePathname();
const { user } = useUser();
const [pendingCount, setPendingCount] = useState(0);
const { data: pendingCount = 0 } = usePendingRecommendationsCount();
const [mounted, setMounted] = useState(false);
const [isCollapsed, setIsCollapsed] = useState(false);
useEffect(() => {
setMounted(true);
const saved = localStorage.getItem("sidebar-collapsed");
if (saved) setIsCollapsed(JSON.parse(saved));
}, []);
useEffect(() => {
const fetchPending = async () => {
try {
const res = await fetch("/api/recommendations");
if (res.ok) {
const data = await res.json();
const pending = (data.recommendations || []).filter(
(r: Recommendation) => r.status === "pending",
);
setPendingCount(pending.length);
}
} catch (e) {
console.error("Failed to fetch pending recommendations", e);
}
};
fetchPending();
}, []);
const handleToggle = () => {
const newState = !isCollapsed;
setIsCollapsed(newState);
localStorage.setItem("sidebar-collapsed", JSON.stringify(newState));
};
const menuItems = [
const navItems: NavItem[] = [
{ icon: LayoutDashboard, label: "Dashboard", href: "/" },
{ icon: Users, label: "Users", href: "/users" },
{
icon: Brain,
label: (
<span className="flex items-center">
AI Recommendations
{pendingCount > 0 && (
<span className="ml-2 inline-block bg-red-600 text-xs rounded-full px-2 py-0.5">
{pendingCount}
</span>
)}
</span>
),
label: "AI Recommendations",
href: "/recommendations",
badge: pendingCount > 0 ? pendingCount : undefined,
},
{ icon: CalendarCheck, label: "Attendance", href: "/attendance" },
{ icon: CreditCard, label: "Payments", href: "/payments" },
@ -70,59 +58,126 @@ export function Sidebar() {
];
return (
<aside className="w-20 hover:w-64 bg-slate-900 text-white h-screen fixed left-0 top-0 flex flex-col border-r border-slate-800 transition-all duration-300 z-50 group overflow-hidden">
<div className="p-6 border-b border-slate-800 flex items-center overflow-hidden whitespace-nowrap">
<h1 className="text-2xl font-bold bg-gradient-to-r from-blue-400 to-indigo-400 bg-clip-text text-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300">
FitAI Admin
</h1>
</div>
<aside
className={cn(
"fixed left-0 top-0 z-40 h-screen transition-all duration-300 ease-in-out",
isCollapsed ? "w-[72px]" : "w-[260px]",
"bg-gradient-to-b from-[var(--sidebar-background)] to-[#0f172a]",
)}
>
<div className="flex h-full flex-col border-r border-white/5">
{/* Logo */}
<div
className={cn(
"flex h-16 items-center border-b border-white/5 px-4",
isCollapsed ? "justify-center" : "justify-between",
)}
>
{!isCollapsed && (
<Link href="/" className="flex items-center gap-2">
<div className="flex h-9 w-9 items-center justify-center rounded-lg bg-gradient-to-br from-blue-500 to-indigo-600 shadow-lg shadow-blue-500/25">
<Activity className="h-5 w-5 text-white" />
</div>
<span className="text-lg font-bold text-white">FitAI</span>
</Link>
)}
{isCollapsed && (
<div className="flex h-9 w-9 items-center justify-center rounded-lg bg-gradient-to-br from-blue-500 to-indigo-600 shadow-lg shadow-blue-500/25">
<Activity className="h-5 w-5 text-white" />
</div>
)}
</div>
<nav className="flex-1 p-4 space-y-2">
{menuItems.map((item) => {
const Icon = item.icon;
const isActive = pathname === item.href;
const label =
typeof item.label === "string" ? item.label : item.label;
return (
<Link
key={item.href}
href={item.href}
className={`flex items-center gap-3 px-4 py-3 rounded-lg transition-all duration-200 ${
isActive
? "bg-blue-600 text-white shadow-lg shadow-blue-900/20"
: "text-slate-400 hover:bg-slate-800 hover:text-white"
}`}
>
<div className="min-w-[20px]">
{/* Toggle Button */}
<button
onClick={handleToggle}
className={cn(
"absolute -right-3 top-20 z-50 flex h-6 w-6 items-center justify-center rounded-full border border-white/10 bg-slate-800 text-white/60 shadow-lg transition-all hover:bg-slate-700 hover:text-white",
isCollapsed ? "left-[60px]" : "left-[244px]",
)}
>
<ChevronLeft
className={cn(
"h-3.5 w-3.5 transition-transform",
isCollapsed && "rotate-180",
)}
/>
</button>
{/* Navigation */}
<nav className="flex-1 space-y-1 overflow-y-auto p-3">
{navItems.map((item) => {
const Icon = item.icon;
const isActive =
pathname === item.href ||
(item.href !== "/" && pathname.startsWith(item.href));
return (
<Link
key={item.href}
href={item.href}
className={cn(
"group flex items-center gap-3 rounded-lg px-3 py-2.5 text-sm font-medium transition-all duration-200",
isActive
? "bg-gradient-to-r from-blue-600 to-blue-500 text-white shadow-lg shadow-blue-500/25"
: "text-slate-400 hover:bg-white/5 hover:text-white",
isCollapsed && "justify-center px-2",
)}
>
<Icon
size={20}
className={
className={cn(
"h-5 w-5 flex-shrink-0 transition-colors",
isActive
? "text-white"
: "text-slate-500 group-hover:text-white"
}
: "text-slate-500 group-hover:text-white",
)}
/>
</div>
<span className="font-medium opacity-0 group-hover:opacity-100 transition-opacity duration-300 whitespace-nowrap overflow-hidden">
{label}
</span>
</Link>
);
})}
</nav>
{!isCollapsed && (
<>
<span className="flex-1">{item.label}</span>
{item.badge !== undefined && (
<span className="flex h-5 min-w-[20px] items-center justify-center rounded-full bg-red-500 px-1.5 text-xs font-bold text-white">
{item.badge > 99 ? "99+" : item.badge}
</span>
)}
</>
)}
{isCollapsed && item.badge !== undefined && (
<span className="absolute -right-1 -top-1 flex h-4 min-w-[16px] items-center justify-center rounded-full bg-red-500 px-1 text-[10px] font-bold text-white">
{item.badge > 9 ? "9+" : item.badge}
</span>
)}
</Link>
);
})}
</nav>
<div className="p-4 border-t border-slate-800">
<div className="flex items-center gap-3 px-2 py-3 rounded-lg bg-slate-800/50 overflow-hidden whitespace-nowrap">
<div className="min-w-[32px]">
{mounted && <UserButton afterSignOutUrl="/" />}
</div>
<div className="flex-1 min-w-0 opacity-0 group-hover:opacity-100 transition-opacity duration-300">
<p className="text-sm font-medium text-white truncate">
{user?.fullName || "Admin User"}
</p>
<p className="text-xs text-slate-400 truncate">
{user?.primaryEmailAddress?.emailAddress}
</p>
{/* User Section */}
<div
className={cn(
"border-t border-white/5 p-3",
isCollapsed && "flex justify-center",
)}
>
<div
className={cn(
"flex items-center gap-3 rounded-lg bg-white/5 p-2 transition-colors hover:bg-white/10",
isCollapsed && "justify-center p-2",
)}
>
<div className={cn("flex-shrink-0", isCollapsed ? "" : "ml-1")}>
{mounted && <UserButton afterSignOutUrl="/" />}
</div>
{!isCollapsed && (
<div className="min-w-0 flex-1">
<p className="truncate text-sm font-medium text-white">
{user?.fullName || "Admin"}
</p>
<p className="truncate text-xs text-slate-400">
{user?.primaryEmailAddress?.emailAddress}
</p>
</div>
)}
</div>
</div>
</div>

View File

@ -1,51 +1,139 @@
import { LucideIcon } from "lucide-react";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import { cn } from "@/lib/utils";
import { TrendingUp, TrendingDown, Minus } from "lucide-react";
interface StatsCardProps {
title: string;
value: string | number;
change?: string;
trend?: "up" | "down" | "neutral";
icon: LucideIcon;
color?: "blue" | "green" | "purple" | "orange";
title: string;
value: string | number;
change?: string;
trend?: "up" | "down" | "neutral";
icon: LucideIcon;
color?: "blue" | "green" | "purple" | "orange" | "red";
className?: string;
}
export function StatsCard({ title, value, change, trend, icon: Icon, color = "blue" }: StatsCardProps) {
const colorStyles = {
blue: "bg-blue-50 text-blue-600",
green: "bg-green-50 text-green-600",
purple: "bg-purple-50 text-purple-600",
orange: "bg-orange-50 text-orange-600",
};
export function StatsCard({
title,
value,
change,
trend,
icon: Icon,
color = "blue",
className,
}: StatsCardProps) {
const colorStyles = {
blue: {
gradient: "from-blue-500/10 to-blue-600/5",
iconBg: "bg-gradient-to-br from-blue-500 to-blue-600",
iconShadow: "shadow-blue-500/25",
text: "text-blue-600",
trendUp: "text-emerald-600",
trendDown: "text-red-600",
},
green: {
gradient: "from-emerald-500/10 to-emerald-600/5",
iconBg: "bg-gradient-to-br from-emerald-500 to-emerald-600",
iconShadow: "shadow-emerald-500/25",
text: "text-emerald-600",
trendUp: "text-emerald-600",
trendDown: "text-red-600",
},
purple: {
gradient: "from-violet-500/10 to-violet-600/5",
iconBg: "bg-gradient-to-br from-violet-500 to-violet-600",
iconShadow: "shadow-violet-500/25",
text: "text-violet-600",
trendUp: "text-emerald-600",
trendDown: "text-red-600",
},
orange: {
gradient: "from-amber-500/10 to-amber-600/5",
iconBg: "bg-gradient-to-br from-amber-500 to-orange-600",
iconShadow: "shadow-amber-500/25",
text: "text-amber-600",
trendUp: "text-emerald-600",
trendDown: "text-red-600",
},
red: {
gradient: "from-rose-500/10 to-rose-600/5",
iconBg: "bg-gradient-to-br from-rose-500 to-rose-600",
iconShadow: "shadow-rose-500/25",
text: "text-rose-600",
trendUp: "text-emerald-600",
trendDown: "text-red-600",
},
};
return (
<Card>
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-muted-foreground">
{title}
</CardTitle>
<div className={`p-2 rounded-lg ${colorStyles[color]}`}>
<Icon size={16} />
</div>
</CardHeader>
<CardContent>
<div className="text-2xl font-bold">{value}</div>
{change && (
<p className="text-xs text-muted-foreground mt-1">
<span
className={`font-medium ${trend === "up"
? "text-green-600"
: trend === "down"
? "text-red-600"
: "text-slate-600"
}`}
>
{change}
</span>{" "}
vs last month
</p>
const styles = colorStyles[color];
const TrendIcon =
trend === "up" ? TrendingUp : trend === "down" ? TrendingDown : Minus;
return (
<div
className={cn(
"group relative overflow-hidden rounded-2xl border border-border/50 bg-gradient-to-br p-6 transition-all duration-300 hover:shadow-lg hover:-translate-y-1",
styles.gradient,
className,
)}
>
<div className="flex items-start justify-between">
<div className="space-y-2">
<p className="text-sm font-medium text-muted-foreground">{title}</p>
<div className="flex items-baseline gap-2">
<span className="text-3xl font-bold tracking-tight">{value}</span>
</div>
{change && (
<div className="flex items-center gap-1 text-sm">
<TrendIcon
className={cn(
"h-4 w-4",
trend === "up" && styles.trendUp,
trend === "down" && styles.trendDown,
trend === "neutral" && "text-muted-foreground",
)}
</CardContent>
</Card>
);
/>
<span
className={cn(
"font-medium",
trend === "up" && styles.trendUp,
trend === "down" && styles.trendDown,
trend === "neutral" && "text-muted-foreground",
)}
>
{change}
</span>
<span className="text-muted-foreground">vs last month</span>
</div>
)}
</div>
<div
className={cn(
"flex h-12 w-12 items-center justify-center rounded-xl bg-gradient-to-br shadow-lg transition-transform duration-300 group-hover:scale-110",
styles.iconBg,
styles.iconShadow,
)}
>
<Icon className="h-6 w-6 text-white" />
</div>
</div>
{/* Decorative corner gradient */}
<div className="absolute -right-8 -top-8 h-32 w-32 rounded-full bg-gradient-to-br from-white/20 to-transparent opacity-0 transition-opacity duration-300 group-hover:opacity-100" />
</div>
);
}
export function StatsCardSkeleton() {
return (
<div className="animate-pulse rounded-2xl border border-border/50 bg-gradient-to-br from-muted/30 to-muted/20 p-6">
<div className="flex items-start justify-between">
<div className="space-y-3">
<div className="h-4 w-24 rounded bg-muted" />
<div className="h-9 w-32 rounded bg-muted" />
<div className="h-4 w-20 rounded bg-muted" />
</div>
<div className="h-12 w-12 rounded-xl bg-muted" />
</div>
</div>
);
}

View File

@ -0,0 +1,62 @@
import { cn } from "@/lib/utils";
import { cva, type VariantProps } from "class-variance-authority";
const badgeVariants = cva(
"inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2",
{
variants: {
variant: {
default: "bg-primary/10 text-primary",
secondary: "bg-secondary text-secondary-foreground",
success:
"bg-emerald-100 text-emerald-800 dark:bg-emerald-900/30 dark:text-emerald-400",
warning:
"bg-amber-100 text-amber-800 dark:bg-amber-900/30 dark:text-amber-400",
destructive:
"bg-red-100 text-red-800 dark:bg-red-900/30 dark:text-red-400",
info: "bg-blue-100 text-blue-800 dark:bg-blue-900/30 dark:text-blue-400",
outline: "border border-input text-foreground",
gray: "bg-slate-100 text-slate-800 dark:bg-slate-800 dark:text-slate-300",
},
},
defaultVariants: {
variant: "default",
},
},
);
export interface BadgeProps
extends React.HTMLAttributes<HTMLDivElement>,
VariantProps<typeof badgeVariants> {}
export function Badge({ className, variant, ...props }: BadgeProps) {
return (
<div className={cn(badgeVariants({ variant }), className)} {...props} />
);
}
export function StatusBadge({ status }: { status: string }) {
const statusConfig: Record<
string,
{ variant: BadgeProps["variant"]; label: string }
> = {
active: { variant: "success", label: "Active" },
inactive: { variant: "gray", label: "Inactive" },
pending: { variant: "warning", label: "Pending" },
approved: { variant: "success", label: "Approved" },
rejected: { variant: "destructive", label: "Rejected" },
completed: { variant: "success", label: "Completed" },
failed: { variant: "destructive", label: "Failed" },
suspended: { variant: "destructive", label: "Suspended" },
basic: { variant: "default", label: "Basic" },
premium: { variant: "info", label: "Premium" },
vip: { variant: "warning", label: "VIP" },
};
const config = statusConfig[status.toLowerCase()] || {
variant: "outline",
label: status,
};
return <Badge variant={config.variant}>{config.label}</Badge>;
}

View File

@ -1,49 +1,60 @@
import React from 'react'
import { cn } from '@/lib/utils'
import { Loader2 } from 'lucide-react'
import React from "react";
import { cn } from "@/lib/utils";
import { Loader2 } from "lucide-react";
import { cva, type VariantProps } from "class-variance-authority";
interface ButtonProps extends React.ButtonHTMLAttributes<HTMLButtonElement> {
variant?: 'primary' | 'secondary' | 'ghost' | 'destructive' | 'outline' | 'default'
size?: 'default' | 'sm' | 'lg' | 'icon'
isLoading?: boolean
children: React.ReactNode
const buttonVariants = cva(
"inline-flex items-center justify-center rounded-lg text-sm font-medium transition-all duration-200 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50",
{
variants: {
variant: {
default:
"bg-primary text-primary-foreground shadow-sm hover:bg-primary/90 hover:shadow",
destructive:
"bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90 hover:shadow",
outline:
"border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground",
secondary:
"bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80",
ghost: "hover:bg-accent hover:text-accent-foreground",
link: "text-primary underline-offset-4 hover:underline",
},
size: {
default: "h-10 px-4 py-2",
sm: "h-9 rounded-md px-3",
lg: "h-11 rounded-md px-8",
icon: "h-10 w-10",
},
},
defaultVariants: {
variant: "default",
size: "default",
},
},
);
export interface ButtonProps
extends React.ButtonHTMLAttributes<HTMLButtonElement>,
VariantProps<typeof buttonVariants> {
isLoading?: boolean;
}
export function Button({
variant = 'primary',
size = 'default',
isLoading = false,
children,
className = '',
disabled,
...props
}: ButtonProps) {
const baseClasses = 'inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50'
const variantClasses = {
primary: 'bg-blue-600 text-white hover:bg-blue-700 shadow hover:bg-blue-700/90',
default: 'bg-blue-600 text-white hover:bg-blue-700 shadow hover:bg-blue-700/90',
secondary: 'bg-slate-100 text-slate-900 hover:bg-slate-200/80',
ghost: 'hover:bg-slate-100 hover:text-slate-900',
destructive: 'bg-red-500 text-white hover:bg-red-600/90',
outline: 'border border-input bg-transparent shadow-sm hover:bg-slate-100 hover:text-slate-900'
}
const sizeClasses = {
default: 'h-9 px-4 py-2',
sm: 'h-8 rounded-md px-3 text-xs',
lg: 'h-10 rounded-md px-8',
icon: 'h-9 w-9'
}
return (
<button
className={cn(baseClasses, variantClasses[variant], sizeClasses[size], className)}
disabled={disabled || isLoading}
{...props}
>
{isLoading && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
{children}
</button>
)
}
export const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
(
{ className, variant, size, isLoading, children, disabled, ...props },
ref,
) => {
return (
<button
className={cn(buttonVariants({ variant, size, className }))}
ref={ref}
disabled={disabled || isLoading}
{...props}
>
{isLoading && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
{children}
</button>
);
},
);
Button.displayName = "Button";

View File

@ -0,0 +1,22 @@
import * as React from "react";
import { cn } from "@/lib/utils";
export interface LabelProps
extends React.LabelHTMLAttributes<HTMLLabelElement> {}
const Label = React.forwardRef<HTMLLabelElement, LabelProps>(
({ className, ...props }, ref) => (
<label
ref={ref}
className={cn(
"text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70",
className,
)}
{...props}
/>
),
);
Label.displayName = "Label";
export { Label };

View File

@ -0,0 +1,160 @@
"use client";
import * as React from "react";
import * as SelectPrimitive from "@radix-ui/react-select";
import { Check, ChevronDown, ChevronUp } from "lucide-react";
import { cn } from "@/lib/utils";
const Select = SelectPrimitive.Root;
const SelectGroup = SelectPrimitive.Group;
const SelectValue = SelectPrimitive.Value;
const SelectTrigger = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Trigger>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Trigger>
>(({ className, children, ...props }, ref) => (
<SelectPrimitive.Trigger
ref={ref}
className={cn(
"flex h-10 w-full items-center justify-between rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background placeholder:text-muted-foreground focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 [&>span]:line-clamp-1",
className,
)}
{...props}
>
{children}
<SelectPrimitive.Icon asChild>
<ChevronDown className="h-4 w-4 opacity-50" />
</SelectPrimitive.Icon>
</SelectPrimitive.Trigger>
));
SelectTrigger.displayName = SelectPrimitive.Trigger.displayName;
const SelectScrollUpButton = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.ScrollUpButton>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.ScrollUpButton>
>(({ className, ...props }, ref) => (
<SelectPrimitive.ScrollUpButton
ref={ref}
className={cn(
"flex cursor-default items-center justify-center py-1",
className,
)}
{...props}
>
<ChevronUp className="h-4 w-4" />
</SelectPrimitive.ScrollUpButton>
));
SelectScrollUpButton.displayName = SelectPrimitive.ScrollUpButton.displayName;
const SelectScrollDownButton = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.ScrollDownButton>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.ScrollDownButton>
>(({ className, ...props }, ref) => (
<SelectPrimitive.ScrollDownButton
ref={ref}
className={cn(
"flex cursor-default items-center justify-center py-1",
className,
)}
{...props}
>
<ChevronDown className="h-4 w-4" />
</SelectPrimitive.ScrollDownButton>
));
SelectScrollDownButton.displayName =
SelectPrimitive.ScrollDownButton.displayName;
const SelectContent = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Content>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Content>
>(({ className, children, position = "popper", ...props }, ref) => (
<SelectPrimitive.Portal>
<SelectPrimitive.Content
ref={ref}
className={cn(
"relative z-50 max-h-96 min-w-[8rem] overflow-hidden rounded-md border bg-popover text-popover-foreground shadow-md data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
position === "popper" &&
"data-[side=bottom]:translate-y-1 data-[side=left]:-translate-x-1 data-[side=right]:translate-x-1 data-[side=top]:-translate-y-1",
className,
)}
position={position}
{...props}
>
<SelectScrollUpButton />
<SelectPrimitive.Viewport
className={cn(
"p-1",
position === "popper" &&
"h-[var(--radix-select-trigger-height)] w-full min-w-[var(--radix-select-trigger-width)]",
)}
>
{children}
</SelectPrimitive.Viewport>
<SelectScrollDownButton />
</SelectPrimitive.Content>
</SelectPrimitive.Portal>
));
SelectContent.displayName = SelectPrimitive.Content.displayName;
const SelectLabel = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Label>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Label>
>(({ className, ...props }, ref) => (
<SelectPrimitive.Label
ref={ref}
className={cn("py-1.5 pl-8 pr-2 text-sm font-semibold", className)}
{...props}
/>
));
SelectLabel.displayName = SelectPrimitive.Label.displayName;
const SelectItem = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Item>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Item>
>(({ className, children, ...props }, ref) => (
<SelectPrimitive.Item
ref={ref}
className={cn(
"relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
className,
)}
{...props}
>
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
<SelectPrimitive.ItemIndicator>
<Check className="h-4 w-4" />
</SelectPrimitive.ItemIndicator>
</span>
<SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
</SelectPrimitive.Item>
));
SelectItem.displayName = SelectPrimitive.Item.displayName;
const SelectSeparator = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Separator>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Separator>
>(({ className, ...props }, ref) => (
<SelectPrimitive.Separator
ref={ref}
className={cn("-mx-1 my-1 h-px bg-muted", className)}
{...props}
/>
));
SelectSeparator.displayName = SelectPrimitive.Separator.displayName;
export {
Select,
SelectGroup,
SelectValue,
SelectTrigger,
SelectContent,
SelectLabel,
SelectItem,
SelectSeparator,
SelectScrollUpButton,
SelectScrollDownButton,
};

View File

@ -0,0 +1,57 @@
import { cn } from "@/lib/utils";
interface SkeletonProps {
className?: string;
}
export function Skeleton({ className }: SkeletonProps) {
return (
<div
className={cn(
"animate-pulse rounded-md bg-slate-200 dark:bg-slate-700",
className,
)}
/>
);
}
export function StatsCardSkeleton() {
return (
<div className="bg-white rounded-xl shadow-sm border border-slate-100 p-6">
<div className="flex items-center gap-4">
<Skeleton className="h-12 w-12 rounded-full" />
<div className="flex-1 space-y-2">
<Skeleton className="h-4 w-20" />
<Skeleton className="h-8 w-16" />
</div>
</div>
</div>
);
}
export function TableSkeleton({ rows = 5 }: { rows?: number }) {
return (
<div className="space-y-3">
{Array.from({ length: rows }).map((_, i) => (
<div key={i} className="flex gap-4">
<Skeleton className="h-12 flex-1" />
<Skeleton className="h-12 w-32" />
<Skeleton className="h-12 w-24" />
</div>
))}
</div>
);
}
export function CardSkeleton() {
return (
<div className="bg-white rounded-xl shadow-sm border border-slate-100 p-6 space-y-4">
<Skeleton className="h-6 w-40" />
<div className="space-y-2">
<Skeleton className="h-4 w-full" />
<Skeleton className="h-4 w-3/4" />
<Skeleton className="h-4 w-5/6" />
</div>
</div>
);
}

View File

@ -3,6 +3,7 @@
import { useState } from "react";
import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import { useUser } from "@clerk/nextjs";
import {
Dialog,
DialogContent,
@ -26,6 +27,9 @@ import {
type InvitationOptionsData,
} from "@/lib/validation/user-schemas";
import { ChevronLeft, ChevronRight, Check } from "lucide-react";
import { useGyms } from "@/hooks/use-api";
import { getInvitableRoles } from "@/lib/auth/permissions";
import type { UserRole } from "@fitai/shared";
interface CreateUserModalProps {
open: boolean;
@ -45,6 +49,15 @@ export function CreateUserModal({
const [basicInfo, setBasicInfo] = useState<BasicInfoData | null>(null);
const [clientInfo, setClientInfo] = useState<ClientInfoData | null>(null);
const [isSubmitting, setIsSubmitting] = useState(false);
const { data: gyms = [] } = useGyms();
const { user } = useUser();
// Get current user's role and gym from Clerk metadata
const currentUserRole = (user?.publicMetadata?.role as UserRole) || "client";
const currentUserGymId = user?.publicMetadata?.gymId as string | null;
// Determine which roles the current user can create
const invitableRoles = getInvitableRoles(currentUserRole);
// Step 1: Role Selection Form
const roleForm = useForm<RoleSelectionData>({
@ -136,10 +149,18 @@ export function CreateUserModal({
invitationData,
);
// Auto-assign gym for non-superAdmins
// SuperAdmins can select gym manually, others use their own gym
const finalPayload = {
...payload,
gymId:
currentUserRole === "superAdmin" ? payload.gymId : currentUserGymId,
};
const response = await fetch("/api/users/create", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify(payload),
body: JSON.stringify(finalPayload),
});
if (!response.ok) {
@ -153,6 +174,10 @@ export function CreateUserModal({
// Reset form
resetModal();
// Add delay to allow Clerk API to propagate invitation
await new Promise((resolve) => setTimeout(resolve, 1000));
onSuccess();
onOpenChange(false);
} catch (error) {
@ -259,7 +284,7 @@ export function CreateUserModal({
<div className="space-y-2">
<label className="text-sm font-medium">Role *</label>
<div className="grid grid-cols-3 gap-3">
{(["admin", "trainer", "client"] as const).map((role) => (
{invitableRoles.map((role) => (
<label
key={role}
className={`flex items-center justify-center p-4 border-2 rounded-lg cursor-pointer transition-colors ${
@ -461,6 +486,39 @@ export function CreateUserModal({
</p>
</div>
)}
{/* Gym Selector - Only for SuperAdmins */}
{currentUserRole === "superAdmin" && (
<div className="space-y-2">
<label className="text-sm font-medium">
Assign to Gym (Optional)
</label>
<select
{...invitationForm.register("gymId")}
className="w-full p-2 border border-gray-300 rounded-md"
>
<option value="">Select a gym...</option>
{gyms.map((gym) => (
<option key={gym.id} value={gym.id}>
{gym.name}
</option>
))}
</select>
<p className="text-xs text-gray-500">
If not selected, user will not be assigned to any gym
</p>
</div>
)}
{/* Info message for non-superAdmins */}
{currentUserRole !== "superAdmin" && currentUserGymId && (
<div className="p-4 bg-gray-50 border border-gray-200 rounded-md">
<p className="text-sm text-gray-700">
User will be automatically assigned to your gym
</p>
</div>
)}
<DialogFooter className="flex justify-between">
<Button type="button" variant="outline" onClick={handleBack}>
<ChevronLeft className="mr-2 h-4 w-4" /> Back

View File

@ -0,0 +1,140 @@
"use client";
import { useState } from "react";
import { formatDistanceToNow } from "date-fns";
import { Button } from "@/components/ui/button";
import { Card } from "@/components/ui/card";
import { toast } from "@/lib/toast";
import {
useRevokeInvitation,
useResendInvitation,
type Invitation,
} from "@/hooks/use-api";
import { Copy, RefreshCw, Trash2 } from "lucide-react";
interface InvitationsGridProps {
invitations: Invitation[];
onRefetch: () => void;
}
export function InvitationsGrid({
invitations,
onRefetch,
}: InvitationsGridProps) {
const revokeInvitation = useRevokeInvitation();
const resendInvitation = useResendInvitation();
const [actioningId, setActioningId] = useState<string | null>(null);
const handleCopyUrl = async (url: string) => {
try {
await navigator.clipboard.writeText(url);
toast.success("Invitation link copied to clipboard");
} catch (error) {
toast.error("Failed to copy link");
}
};
const handleRevoke = async (invitation: Invitation) => {
if (!confirm(`Cancel invitation for ${invitation.emailAddress}?`)) {
return;
}
setActioningId(invitation.id);
try {
await revokeInvitation.mutateAsync(invitation.id);
toast.success("Invitation cancelled");
onRefetch();
} catch (error) {
toast.error("Failed to cancel invitation");
} finally {
setActioningId(null);
}
};
const handleResend = async (invitation: Invitation) => {
setActioningId(invitation.id);
try {
await resendInvitation.mutateAsync(invitation.id);
toast.success("Invitation resent");
onRefetch();
} catch (error) {
toast.error("Failed to resend invitation");
} finally {
setActioningId(null);
}
};
if (invitations.length === 0) {
return (
<Card className="p-8 text-center text-muted-foreground">
No pending invitations
</Card>
);
}
return (
<div className="space-y-4">
{invitations.map((invitation) => {
const role = invitation.publicMetadata?.role || "unknown";
const createdDate = new Date(invitation.createdAt);
const isActioning = actioningId === invitation.id;
return (
<Card key={invitation.id} className="p-4">
<div className="flex items-center justify-between">
<div className="flex-1">
<div className="flex items-center gap-2">
<h3 className="font-medium">{invitation.emailAddress}</h3>
<span className="text-xs px-2 py-1 rounded bg-yellow-100 text-yellow-800 dark:bg-yellow-900 dark:text-yellow-200">
Pending
</span>
<span className="text-xs px-2 py-1 rounded bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200">
{role}
</span>
</div>
<p className="text-sm text-muted-foreground mt-1">
Sent {formatDistanceToNow(createdDate, { addSuffix: true })}
</p>
</div>
<div className="flex items-center gap-2">
{invitation.url && (
<Button
variant="outline"
size="sm"
onClick={() => handleCopyUrl(invitation.url!)}
>
<Copy className="h-4 w-4 mr-1" />
Copy Link
</Button>
)}
<Button
variant="outline"
size="sm"
onClick={() => handleResend(invitation)}
disabled={isActioning}
>
<RefreshCw
className={`h-4 w-4 mr-1 ${isActioning ? "animate-spin" : ""}`}
/>
Resend
</Button>
<Button
variant="destructive"
size="sm"
onClick={() => handleRevoke(invitation)}
disabled={isActioning}
>
<Trash2 className="h-4 w-4 mr-1" />
Cancel
</Button>
</div>
</div>
</Card>
);
})}
</div>
);
}

View File

@ -9,9 +9,10 @@ import { formatDate } from "@/lib/utils";
ModuleRegistry.registerModules([AllCommunityModule]);
function getTimeAgo(date: Date): string {
function getTimeAgo(date: Date | string): string {
const now = new Date();
const diffMs = now.getTime() - date.getTime();
const d = typeof date === "string" ? new Date(date) : date;
const diffMs = now.getTime() - d.getTime();
const diffMins = Math.floor(diffMs / 60000);
const diffHours = Math.floor(diffMins / 60);
const diffDays = Math.floor(diffHours / 24);
@ -22,7 +23,7 @@ function getTimeAgo(date: Date): string {
return `${diffDays}d ago`;
}
interface User {
export interface User {
id: string;
email: string;
firstName: string;
@ -31,15 +32,18 @@ interface User {
phone?: string;
gymId?: string;
gymName?: string | null;
createdAt: Date;
createdAt?: string | Date;
isCheckedIn?: boolean;
checkInTime?: Date;
checkInTime?: string | Date;
lastCheckInTime?: string | Date;
checkInsThisWeek?: number;
checkInsThisMonth?: number;
client?: {
id: string;
membershipType: string;
membershipStatus: string;
joinDate: Date;
lastVisit?: Date;
joinDate: string | Date;
lastVisit?: string | Date;
};
}
@ -256,7 +260,7 @@ export function UserGrid({
// },
{
headerName: "Last Visit",
valueGetter: (params) => params.data?.client?.lastVisit,
valueGetter: (params) => params.data?.lastCheckInTime,
filter: "agDateColumnFilter",
sortable: true,
valueFormatter: (params: any) =>

View File

@ -1,44 +1,33 @@
"use client";
import { useState, useEffect } from "react";
import { UserGrid } from "@/components/users/UserGrid";
// import { Button } from "@/components/ui/button";
import { useState } from "react";
import { UserGrid, type User } from "@/components/users/UserGrid";
import { InvitationsGrid } from "@/components/users/InvitationsGrid";
import { Card, CardHeader, CardContent } from "@/components/ui/card";
import { Button } from "@/components/ui/button";
import { useUser } from "@clerk/nextjs";
import { getGymIdFromUser } from "@/lib/error-helpers";
import log from "@/lib/logger";
import { toast } from "@/lib/toast";
import { CreateUserModal } from "./CreateUserModal";
import {
useUsers,
useGyms,
useUpdateUser,
useDeleteUser,
useSendInvitation,
useInvitations,
} from "@/hooks/use-api";
interface User {
id: string;
email: string;
firstName: string;
lastName: string;
role: string;
phone?: string;
interface UserManagementProps {
gymId?: string;
createdAt: Date;
isCheckedIn?: boolean;
checkInTime?: Date;
lastCheckInTime?: Date;
checkInsThisWeek?: number;
checkInsThisMonth?: number;
client?: {
id: string;
membershipType: string;
membershipStatus: string;
joinDate: Date;
lastVisit?: Date;
};
}
export function UserManagement() {
export function UserManagement({ gymId }: UserManagementProps) {
const { user } = useUser();
const [users, setUsers] = useState<User[]>([]);
const [loading, setLoading] = useState(true);
const [filter, setFilter] = useState<string>("all");
const [viewFilter, setViewFilter] = useState<"all" | "active" | "pending">(
"all",
);
const [selectedUser, setSelectedUser] = useState<User | null>(null);
const [isEditing, setIsEditing] = useState(false);
const [isDeleting, setIsDeleting] = useState(false);
@ -50,70 +39,33 @@ export function UserManagement() {
role: string;
phone: string;
gymId: string;
membershipType?: string;
membershipStatus?: string;
} | null>(null);
// Active gyms for dropdown
const [gyms, setGyms] = useState<Array<{ id: string; name: string }>>([]);
const {
data: users = [],
isLoading: usersLoading,
refetch: refetchUsers,
} = useUsers({
role: filter !== "all" ? filter : undefined,
gymId,
});
// Load gyms when modal opens or refreshes
useEffect(() => {
if (isEditing) {
(async () => {
try {
const res = await fetch("/api/gyms");
const data = await res.json();
if (Array.isArray(data)) {
// map down to id and name to avoid extra payload use here
setGyms(data.map((g: any) => ({ id: g.id, name: g.name })));
} else {
setGyms([]);
}
} catch {
setGyms([]);
}
})();
}
}, [isEditing]);
const {
data: invitations = [],
isLoading: invitationsLoading,
refetch: refetchInvitations,
} = useInvitations(gymId);
useEffect(() => {
fetchUsers();
}, [filter]);
const { data: gyms = [] } = useGyms();
const updateUser = useUpdateUser();
const deleteUser = useDeleteUser();
const sendInvitation = useSendInvitation();
const fetchUsers = async () => {
setLoading(true);
try {
const ts = Date.now();
const url =
filter === "all"
? `/api/users?ts=${ts}`
: `/api/users?role=${filter}&ts=${ts}`;
log.debug("Fetching users", { url });
const response = await fetch(url, { cache: "no-store" });
log.debug("Users fetch response", {
ok: response.ok,
status: response.status,
});
const responseData = await response.json();
const data = responseData.data || responseData; // Handle both old and new API response formats
log.debug("Received users data", {
count: Array.isArray(data.users) ? data.users.length : 0,
sample:
data.users && data.users[0]
? {
id: data.users[0].id,
gymId: data.users[0].gymId,
role: data.users[0].role,
}
: null,
});
setUsers(data.users || []);
} catch (error) {
log.error("Failed to fetch users", error);
} finally {
setLoading(false);
}
};
const isLoading =
viewFilter === "pending" ? invitationsLoading : usersLoading;
const refetch = viewFilter === "pending" ? refetchInvitations : refetchUsers;
const handleUserSelect = (user: User | null) => {
setSelectedUser(user);
@ -128,6 +80,8 @@ export function UserManagement() {
role: user.role,
phone: user.phone || "",
gymId: user.gymId || "",
membershipType: user.client?.membershipType || "basic",
membershipStatus: user.client?.membershipStatus || "active",
});
setIsEditing(true);
};
@ -144,17 +98,12 @@ export function UserManagement() {
return;
try {
const response = await fetch("/api/users", {
method: "DELETE",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ ids: users.map((u) => u.id) }),
});
if (response.ok) {
fetchUsers();
toast.success("Users deleted successfully");
} else {
toast.error("Error deleting users");
}
const deletePromises = users.map((u) =>
fetch(`/api/users?id=${u.id}`, { method: "DELETE" }),
);
await Promise.all(deletePromises);
refetch();
toast.success("Users deleted successfully");
} catch (error) {
log.error("Failed to delete users", error);
}
@ -196,7 +145,7 @@ export function UserManagement() {
};
const handleRefresh = () => {
fetchUsers();
refetch();
};
const handleSaveEdit = async () => {
@ -204,8 +153,7 @@ export function UserManagement() {
try {
if (selectedUser) {
// Update existing user
const payload = {
const payload: any = {
id: selectedUser.id,
email: editForm.email,
firstName: editForm.firstName,
@ -214,80 +162,27 @@ export function UserManagement() {
phone: editForm.phone,
gymId: editForm.gymId === "" ? null : editForm.gymId,
};
log.debug("Updating user", payload);
const response = await fetch("/api/users", {
method: "PUT",
headers: { "Content-Type": "application/json" },
body: JSON.stringify(payload),
});
log.debug("User update response", {
ok: response.ok,
status: response.status,
});
if (response.ok) {
// Optimistically update local state so grid reflects changes immediately
setUsers((prev) =>
prev.map((u) =>
u.id === selectedUser.id
? {
...u,
email: editForm.email,
firstName: editForm.firstName,
lastName: editForm.lastName,
role: editForm.role,
phone: editForm.phone || undefined,
gymId: editForm.gymId === "" ? undefined : editForm.gymId,
}
: u,
),
);
setSelectedUser((prev) =>
prev
? {
...prev,
email: editForm.email,
firstName: editForm.firstName,
lastName: editForm.lastName,
role: editForm.role,
phone: editForm.phone || undefined,
gymId: editForm.gymId === "" ? undefined : editForm.gymId,
}
: prev,
);
setIsEditing(false);
setEditForm(null);
// Still re-fetch from server to ensure consistency
log.debug("Re-fetching users after successful edit");
fetchUsers();
toast.success("User updated successfully");
} else {
const errText = await response.text().catch(() => "");
log.error("User update failed", new Error(errText), {
status: response.status,
});
toast.error("Error updating user");
}
} else {
// Create (Invite) new user
const response = await fetch("/api/invitations", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
inviteeEmail: editForm.email,
roleAssigned: editForm.role,
gymId: editForm.gymId || undefined,
}),
});
if (response.ok) {
setIsEditing(false);
setEditForm(null);
fetchUsers();
toast.success("Invitation sent successfully!");
} else {
const errorData = await response.json();
toast.error(`Error sending invitation: ${errorData.error}`);
// Include membership fields if user is a client
if (editForm.role === "client") {
payload.membershipType = editForm.membershipType;
payload.membershipStatus = editForm.membershipStatus;
}
await updateUser.mutateAsync(payload);
setIsEditing(false);
setEditForm(null);
refetch();
toast.success("User updated successfully");
} else {
await sendInvitation.mutateAsync({
email: editForm.email,
role: editForm.role,
});
setIsEditing(false);
setEditForm(null);
refetch();
toast.success("Invitation sent successfully!");
}
} catch (error) {
console.error(error);
@ -299,17 +194,11 @@ export function UserManagement() {
if (!selectedUser) return;
try {
const response = await fetch(`/api/users?id=${selectedUser.id}`, {
method: "DELETE",
});
if (response.ok) {
setIsDeleting(false);
setSelectedUser(null);
fetchUsers();
toast.success("User deleted successfully");
} else {
toast.error("Error deleting user");
}
await deleteUser.mutateAsync(selectedUser.id);
setIsDeleting(false);
setSelectedUser(null);
refetch();
toast.success("User deleted successfully");
} catch (error) {
log.error("Failed to delete user", error);
}
@ -320,6 +209,19 @@ export function UserManagement() {
<div className="flex justify-between items-center">
<h2 className="text-2xl font-bold">User Management</h2>
<div className="flex gap-2">
{/* View Filter: All/Active/Pending */}
<select
value={viewFilter}
onChange={(e) =>
setViewFilter(e.target.value as "all" | "active" | "pending")
}
className="px-3 py-2 border rounded-md bg-white dark:bg-gray-800"
>
<option value="all">All Users</option>
<option value="active">Active Users</option>
<option value="pending">Pending Invitations</option>
</select>
<Button
variant={filter === "all" ? "default" : "outline"}
onClick={() => setFilter("all")}
@ -347,7 +249,7 @@ export function UserManagement() {
variant={filter === "client" ? "default" : "outline"}
onClick={() => setFilter("client")}
>
Clientsa
Clients
</Button>
<Button
variant={filter === "trainer" ? "default" : "outline"}
@ -372,33 +274,48 @@ export function UserManagement() {
<div className="flex justify-between items-center">
<div className="text-sm text-gray-600">
Showing {users.length} users
{selectedUser && (
<span className="ml-4 text-blue-600">
Selected: {selectedUser.firstName} {selectedUser.lastName}
</span>
{viewFilter === "pending" ? (
<>Showing {invitations.length} pending invitations</>
) : (
<>
Showing {users.length} users
{selectedUser && (
<span className="ml-4 text-blue-600">
Selected: {selectedUser.firstName} {selectedUser.lastName}
</span>
)}
</>
)}
</div>
<div className="flex gap-2">
<Button variant="default" onClick={handleRefresh}>
Refresh
</Button>
<Button variant="default" onClick={handleExport}>
Export CSV
</Button>
{viewFilter !== "pending" && (
<Button variant="default" onClick={handleExport}>
Export CSV
</Button>
)}
</div>
</div>
<Card>
<CardContent className="p-0">
<UserGrid
users={users}
onUserSelect={(user) => handleUserSelect(user)}
onEditUser={handleEditUser}
onDeleteUser={handleDeleteUser}
onBulkDelete={handleBulkDelete}
loading={loading}
/>
{viewFilter === "pending" ? (
<InvitationsGrid
invitations={invitations}
onRefetch={refetchInvitations}
/>
) : (
<UserGrid
users={users}
onUserSelect={(user) => handleUserSelect(user)}
onEditUser={handleEditUser}
onDeleteUser={handleDeleteUser}
onBulkDelete={handleBulkDelete}
loading={isLoading}
/>
)}
</CardContent>
</Card>
@ -488,26 +405,78 @@ export function UserManagement() {
className="w-full border border-gray-300 rounded px-3 py-2"
/>
</div>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">Gym</label>
<select
value={editForm.gymId}
onChange={(e) =>
setEditForm({ ...editForm, gymId: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
>
<option value="">Proceed without gym</option>
{gyms.map((g) => (
<option key={g.id} value={g.id}>
{g.name}
</option>
))}
</select>
<p className="text-xs text-gray-500 mt-1">
Select an active gym or proceed without a gym.
</p>
</div>
{/* Only superAdmins can reassign gyms */}
{(() => {
const currentRole = user?.publicMetadata?.role;
console.log("Current user role for gym selector:", currentRole);
return currentRole === "superAdmin";
})() && (
<div className="mb-4">
<label className="block text-sm font-medium mb-1">Gym</label>
<select
value={editForm.gymId}
onChange={(e) =>
setEditForm({ ...editForm, gymId: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
>
<option value="">Proceed without gym</option>
{gyms.map((g) => (
<option key={g.id} value={g.id}>
{g.name}
</option>
))}
</select>
<p className="text-xs text-gray-500 mt-1">
Select an active gym or proceed without a gym.
</p>
</div>
)}
{/* Client-specific fields - only show when role is client */}
{editForm.role === "client" && (
<>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">
Membership Type
</label>
<select
value={editForm.membershipType || "basic"}
onChange={(e) =>
setEditForm({
...editForm,
membershipType: e.target.value,
})
}
className="w-full border border-gray-300 rounded px-3 py-2"
>
<option value="basic">Basic</option>
<option value="premium">Premium</option>
<option value="vip">VIP</option>
</select>
</div>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">
Membership Status
</label>
<select
value={editForm.membershipStatus || "active"}
onChange={(e) =>
setEditForm({
...editForm,
membershipStatus: e.target.value,
})
}
className="w-full border border-gray-300 rounded px-3 py-2"
>
<option value="active">Active</option>
<option value="inactive">Inactive</option>
<option value="suspended">Suspended</option>
</select>
</div>
</>
)}
<div className="flex justify-end gap-2">
<button
type="button"
@ -591,7 +560,9 @@ export function UserManagement() {
</p>
<p>
<span className="font-medium">Joined:</span>{" "}
{new Date(selectedUser.createdAt).toLocaleDateString()}
{selectedUser.createdAt
? new Date(selectedUser.createdAt).toLocaleDateString()
: "N/A"}
</p>
</div>
</div>
@ -616,9 +587,9 @@ export function UserManagement() {
</p>
<p>
<span className="font-medium">Last Visit:</span>{" "}
{selectedUser.client.lastVisit
{selectedUser.lastCheckInTime
? new Date(
selectedUser.client.lastVisit,
selectedUser.lastCheckInTime,
).toLocaleDateString()
: "Never"}
</p>
@ -653,7 +624,10 @@ export function UserManagement() {
<CreateUserModal
open={createModalOpen}
onOpenChange={setCreateModalOpen}
onSuccess={() => fetchUsers()}
onSuccess={() => {
refetchUsers();
refetchInvitations();
}}
/>
</div>
);

View File

@ -0,0 +1,452 @@
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
export interface DashboardStats {
totalUsers: number;
activeClients: number;
totalRevenue: number;
revenueGrowth: number;
}
export interface User {
id: string;
email: string;
firstName: string;
lastName: string;
role: string;
phone?: string;
gymId?: string;
gymName?: string | null;
createdAt?: string;
isCheckedIn?: boolean;
checkInTime?: string;
lastCheckInTime?: string;
checkInsThisWeek?: number;
checkInsThisMonth?: number;
client?: {
id: string;
membershipType: string;
membershipStatus: string;
joinDate: string;
lastVisit?: string;
};
}
export interface Recommendation {
id: string;
userId: string;
type: string;
title: string;
description: string;
content?: string;
recommendationText?: string;
activityPlan?: string;
dietPlan?: string;
status: "pending" | "approved" | "rejected";
createdAt: string;
user?: User;
}
export interface Gym {
id: string;
name: string;
address?: string;
}
export interface AttendanceRecord {
id: string;
userId: string;
userName?: string;
userEmail?: string;
checkInTime: Date;
checkOutTime?: Date;
date: string;
type?: string;
}
export interface Invitation {
id: string;
emailAddress: string;
publicMetadata: {
role?: string;
gymId?: string;
createdBy?: string;
} | null;
status: "pending" | "accepted" | "revoked" | "expired";
url?: string;
createdAt: number;
updatedAt: number;
revoked?: boolean;
}
async function fetchApi<T>(url: string, options?: RequestInit): Promise<T> {
const response = await fetch(url, {
...options,
headers: {
"Content-Type": "application/json",
...options?.headers,
},
});
if (!response.ok) {
const error = await response
.json()
.catch(() => ({ error: "Request failed" }));
throw new Error(error.error || `HTTP ${response.status}`);
}
return response.json();
}
export function useDashboardStats(gymId?: string) {
return useQuery({
queryKey: ["dashboard-stats", gymId],
queryFn: () => {
const url = gymId
? `/api/admin/stats?gymId=${gymId}`
: "/api/admin/stats";
return fetchApi<{ data: { stats: DashboardStats } }>(url).then(
(res) => res.data?.stats,
);
},
});
}
export function useUsers(filters?: {
role?: string;
gymId?: string;
search?: string;
}) {
const params = new URLSearchParams();
if (filters?.role) params.set("role", filters.role);
if (filters?.gymId) params.set("gymId", filters.gymId);
if (filters?.search) params.set("search", filters.search);
const queryString = params.toString();
const url = queryString ? `/api/users?${queryString}` : "/api/users";
return useQuery({
queryKey: ["users", filters],
queryFn: () =>
fetchApi<{ data: { users: User[] } }>(url).then(
(res) => res.data?.users ?? [],
),
});
}
export function useUser(userId: string) {
return useQuery({
queryKey: ["user", userId],
queryFn: () =>
fetchApi<{ data: { user: User } }>(`/api/users?id=${userId}`).then(
(res) => res.data?.user,
),
enabled: !!userId,
});
}
export function useRecommendations(filters?: {
userId?: string;
status?: string;
}) {
const params = new URLSearchParams();
if (filters?.userId) params.set("userId", filters.userId);
if (filters?.status) params.set("status", filters.status);
const queryString = params.toString();
const url = queryString
? `/api/recommendations?${queryString}`
: "/api/recommendations";
return useQuery({
queryKey: ["recommendations", filters],
queryFn: () =>
fetchApi<{ data: { recommendations: Recommendation[] } }>(url).then(
(res) => res.data?.recommendations ?? [],
),
});
}
export function usePendingRecommendationsCount() {
return useQuery({
queryKey: ["pending-recommendations-count"],
queryFn: () =>
fetchApi<{ data: { recommendations: Recommendation[] } }>(
"/api/recommendations",
).then(
(res) =>
(res.data?.recommendations ?? []).filter(
(r) => r.status === "pending",
).length,
),
refetchInterval: 30000,
});
}
export function useGyms() {
return useQuery({
queryKey: ["gyms"],
queryFn: () =>
fetchApi<{ data: { gyms: Gym[] } }>("/api/gyms").then(
(res) => res.data?.gyms ?? (Array.isArray(res) ? res : []),
),
});
}
export function useAttendance(filters?: { userId?: string; date?: string }) {
const params = new URLSearchParams();
if (filters?.userId) params.set("userId", filters.userId);
if (filters?.date) params.set("date", filters.date);
const queryString = params.toString();
const url = queryString
? `/api/admin/attendance?${queryString}`
: "/api/admin/attendance";
return useQuery({
queryKey: ["attendance", filters],
queryFn: () =>
fetchApi<{ data: { records: AttendanceRecord[] } }>(url).then(
(res) => res.data?.records ?? [],
),
});
}
export function useCreateUser() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (data: {
email: string;
firstName: string;
lastName: string;
role: string;
phone?: string;
gymId?: string;
}) =>
fetchApi<{ data: { userId: string } }>("/api/users/create", {
method: "POST",
body: JSON.stringify(data),
}).then((res) => res.data),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["users"] });
queryClient.invalidateQueries({ queryKey: ["dashboard-stats"] });
},
});
}
export function useUpdateUser() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (data: { id: string; [key: string]: unknown }) =>
fetchApi<{ data: { success: boolean } }>(`/api/users?id=${data.id}`, {
method: "PUT",
body: JSON.stringify(data),
}),
onSuccess: (_, variables) => {
queryClient.invalidateQueries({ queryKey: ["users"] });
queryClient.invalidateQueries({ queryKey: ["user", variables.id] });
},
});
}
export function useDeleteUser() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (userId: string) =>
fetchApi<{ data: { success: boolean } }>(`/api/users?id=${userId}`, {
method: "DELETE",
}),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["users"] });
queryClient.invalidateQueries({ queryKey: ["dashboard-stats"] });
},
});
}
export function useGenerateRecommendations() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (userId?: string) =>
fetchApi<{
data: { success: boolean; recommendations?: Recommendation[] };
}>("/api/recommendations/generate", {
method: "POST",
body: userId ? JSON.stringify({ userId }) : undefined,
}),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["recommendations"] });
queryClient.invalidateQueries({
queryKey: ["pending-recommendations-count"],
});
},
});
}
export function useApproveRecommendation() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (data: { recommendationId: string; approved: boolean }) =>
fetchApi<{ data: { success: boolean } }>("/api/recommendations/approve", {
method: "POST",
body: JSON.stringify(data),
}),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["recommendations"] });
queryClient.invalidateQueries({
queryKey: ["pending-recommendations-count"],
});
},
});
}
export function useUpdateRecommendation() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (data: {
id: string;
content?: string;
activityPlan?: string;
dietPlan?: string;
}) =>
fetchApi<{ data: { success: boolean } }>("/api/recommendations", {
method: "PUT",
body: JSON.stringify(data),
}),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["recommendations"] });
},
});
}
export function useCheckIn() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (userId: string) =>
fetchApi<{ data: { success: boolean; record: AttendanceRecord } }>(
"/api/attendance/check-in",
{
method: "POST",
body: JSON.stringify({ userId }),
},
),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["attendance"] });
},
});
}
export function useCheckOut() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (userId: string) =>
fetchApi<{ data: { success: boolean } }>("/api/attendance/check-out", {
method: "POST",
body: JSON.stringify({ userId }),
}),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["attendance"] });
},
});
}
export function useInvitations(gymId?: string) {
return useQuery({
queryKey: ["invitations", gymId],
queryFn: () => {
const url = gymId
? `/api/invitations?gymId=${gymId}`
: "/api/invitations";
return fetchApi<{ data: { invitations: Invitation[] } }>(url).then(
(res) => res.data?.invitations ?? [],
);
},
refetchInterval: (query) => {
const data = query.state.data;
const hasData = data && data.length > 0;
const fetchCount = query.state.dataUpdateCount;
// Poll every 2 seconds if:
// 1. No invitations returned yet
// 2. Haven't exceeded 5 attempts (10 seconds total)
if (!hasData && fetchCount < 5) {
return 2000;
}
// Stop polling after 10 seconds or when data is present
return false;
},
});
}
export function useSendInvitation() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (data: { email: string; role: string }) =>
fetchApi<{ data: { success: boolean } }>("/api/invitations", {
method: "POST",
body: JSON.stringify(data),
}),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["invitations"] });
},
});
}
export function useRevokeInvitation() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (invitationId: string) =>
fetchApi<{ data: { success: boolean } }>(
`/api/invitations/${invitationId}`,
{ method: "DELETE" },
),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["invitations"] });
queryClient.invalidateQueries({ queryKey: ["users"] });
},
});
}
export function useResendInvitation() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (invitationId: string) =>
fetchApi<{ data: { invitation: any } }>(
`/api/invitations/${invitationId}/resend`,
{ method: "POST" },
),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["invitations"] });
},
});
}
export interface AnalyticsData {
userGrowth: { label: string; value: number }[];
membershipDistribution: { label: string; value: number; color: string }[];
revenue: { label: string; value: number; color: string }[];
}
export function useAnalytics(months: number = 6, gymId?: string) {
return useQuery({
queryKey: ["analytics", months, gymId],
queryFn: () => {
const url = gymId
? `/api/admin/analytics?months=${months}&gymId=${gymId}`
: `/api/admin/analytics?months=${months}`;
return fetchApi<{ data: { analytics: AnalyticsData } }>(url).then(
(res) => res.data?.analytics,
);
},
});
}

View File

@ -0,0 +1,274 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { USER_ROLES, type UserRole } from "@fitai/shared";
import { getDatabase } from "../database";
export interface AuthContext {
userId: string;
role: UserRole;
gymId: string | null;
}
export interface AuthResult extends AuthContext {}
/**
* Get the full authentication context including gymId
* Combines Clerk session claims with database lookup for accurate gym assignment
*
* @returns AuthContext with userId, role, and gymId
*
* @example
* const { userId, role, gymId } = await getAuthContext();
* // userId: "user_abc123"
* // role: "admin"
* // gymId: "gym_xyz"
*/
export async function getAuthContext(): Promise<AuthContext> {
const { userId, sessionClaims } = await auth();
if (!userId) {
throw new Error("Unauthorized: No authenticated user");
}
// Try to get role from Clerk metadata (could be in publicMetadata or metadata)
let role = (sessionClaims?.publicMetadata as { role?: UserRole })?.role;
// Fallback: Try the metadata field if publicMetadata doesn't have it
if (!role) {
role = (sessionClaims?.metadata as { role?: UserRole })?.role;
}
// If still no role, try database lookup as last resort
if (!role || !USER_ROLES.includes(role)) {
console.log("Role not found in session claims, fetching from database", {
userId,
roleFromPublicMetadata: (
sessionClaims?.publicMetadata as { role?: UserRole }
)?.role,
roleFromMetadata: (sessionClaims?.metadata as { role?: UserRole })?.role,
});
try {
const db = await getDatabase();
const user = await db.getUserById(userId);
if (!user) {
console.error("User not found in database", { userId });
throw new Error(`Forbidden: User not found in database - ${userId}`);
}
if (user.role && USER_ROLES.includes(user.role as UserRole)) {
console.log("Role found in database", { userId, role: user.role });
role = user.role as UserRole;
} else {
console.error("Invalid role in database", { userId, role: user.role });
throw new Error(`Forbidden: Invalid role in database - ${user.role}`);
}
} catch (error) {
console.error("Failed to get role from database:", error);
throw new Error(
`Forbidden: Could not determine user role - ${error instanceof Error ? error.message : String(error)}`,
);
}
}
// Get gymId from Clerk metadata or database
let gymId: string | null = null;
// First try Clerk publicMetadata (faster path)
const clerkGymId = (sessionClaims?.publicMetadata as { gymId?: string })
?.gymId;
if (clerkGymId) {
gymId = clerkGymId;
} else {
// Try metadata field
const metadataGymId = (sessionClaims?.metadata as { gymId?: string })
?.gymId;
if (metadataGymId) {
gymId = metadataGymId;
} else {
// Fallback to database lookup
try {
const db = await getDatabase();
const user = await db.getUserById(userId);
gymId = user?.gymId ?? null;
} catch (error) {
console.error("Failed to get gymId from database:", error);
gymId = null;
}
}
}
return { userId, role, gymId };
}
/**
* Middleware to require authentication and optionally check user role
* Enhanced version that also retrieves gymId
*
* @param allowedRoles - Array of roles allowed to access the endpoint (optional)
* @returns Authentication result with userId, role, and gymId, or NextResponse error
*
* @example
* export async function DELETE(request: NextRequest) {
* const authResult = await requireAuth(["admin", "superAdmin"]);
* if (authResult instanceof NextResponse) return authResult;
* const { userId, role, gymId } = authResult;
* // ... proceed with authorized logic
* }
*/
export async function requireAuth(
allowedRoles?: UserRole[],
): Promise<AuthResult | NextResponse> {
try {
const context = await getAuthContext();
// Check if user has required role
if (allowedRoles && allowedRoles.length > 0) {
if (!allowedRoles.includes(context.role)) {
return NextResponse.json(
{
error: `Forbidden - Requires one of: ${allowedRoles.join(", ")}`,
requiredRoles: allowedRoles,
userRole: context.role,
},
{ status: 403 },
);
}
}
return context;
} catch (error) {
const message = error instanceof Error ? error.message : "Unknown error";
if (message.includes("Unauthorized")) {
return NextResponse.json(
{ error: "Unauthorized - Authentication required" },
{ status: 401 },
);
}
if (message.includes("Forbidden")) {
return NextResponse.json({ error: message }, { status: 403 });
}
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
/**
* Require authentication with gym access validation
* Use this for endpoints that need to verify the user can access a specific gym
*
* @param allowedRoles - Roles allowed to access this endpoint
* @param requiredGymId - Gym ID that must be accessed (optional - uses user's gym if not provided)
* @returns AuthResult with full context, or NextResponse error
*
* @example
* // Admin accessing their own gym
* const result = await requireGymAccess(["admin"]);
*
* // SuperAdmin accessing specific gym
* const result = await requireGymAccess(["superAdmin"], "gym_123");
*/
export async function requireGymAccess(
allowedRoles: UserRole[],
requiredGymId?: string,
): Promise<AuthResult | NextResponse> {
const authResult = await requireAuth(allowedRoles);
if (authResult instanceof NextResponse) {
return authResult;
}
const { role, gymId: userGymId } = authResult;
// Determine target gym
const targetGymId = requiredGymId ?? userGymId;
// SuperAdmin can access any gym
if (role === "superAdmin") {
return { ...authResult, gymId: targetGymId };
}
// Other roles must have matching gym
if (!targetGymId) {
return NextResponse.json(
{ error: "Forbidden - User is not assigned to a gym" },
{ status: 403 },
);
}
if (userGymId !== targetGymId) {
return NextResponse.json(
{ error: "Forbidden - Cannot access other gym's data" },
{ status: 403 },
);
}
return { ...authResult, gymId: targetGymId };
}
/**
* Get the target gym ID from request for superAdmin context switching
* SuperAdmins can pass ?gymId to view specific gym data
*
* @param request - NextRequest with query params
* @param userRole - Current user's role
* @param userGymId - Current user's gym ID
* @returns Target gym ID or undefined for superAdmin viewing all
*
* @example
* const targetGymId = getTargetGymId(request, role, gymId);
* // For superAdmin with ?gymId=abc -> "abc"
* // For superAdmin without param -> undefined (all gyms)
* // For admin -> always returns their gymId
*/
export function getTargetGymId(
request: NextRequest,
userRole: UserRole,
userGymId: string | null,
): string | undefined {
// For superAdmin, allow gymId query param for context switching
if (userRole === "superAdmin") {
const queryGymId = request.nextUrl.searchParams.get("gymId");
return queryGymId ?? undefined;
}
// For other roles, they can only access their own gym
return userGymId ?? undefined;
}
/**
* Check if the authenticated user is trying to modify their own account
* Useful for preventing self-deletion or self-demotion
*/
export function isSelfModification(
userId: string,
targetUserId: string,
): boolean {
return userId === targetUserId;
}
/**
* Validate that an admin isn't demoting themselves or deleting their own account
*/
export function preventSelfModification(
userId: string,
targetUserId: string,
action: string,
): NextResponse | undefined {
if (isSelfModification(userId, targetUserId)) {
return NextResponse.json(
{
error: `Cannot ${action} your own account`,
hint: "Ask another administrator to perform this action",
},
{ status: 403 },
);
}
return undefined;
}

View File

@ -0,0 +1,249 @@
import type { UserRole } from "@fitai/shared";
import { NextResponse } from "next/server";
/**
* Check if a user can access data from a specific gym
*
* @param userRole - The role of the requesting user
* @param userGymId - The gym ID of the requesting user (null for superAdmin)
* @param targetGymId - The gym ID being accessed (undefined means all gyms)
* @returns true if access is allowed
*
* @example
* // SuperAdmin accessing any gym
* canAccessGym("superAdmin", null, "gym_123") // true
*
* // Admin accessing their own gym
* canAccessGym("admin", "gym_abc", "gym_abc") // true
*
* // Admin trying to access other gym
* canAccessGym("admin", "gym_abc", "gym_xyz") // false
*/
export function canAccessGym(
userRole: UserRole,
userGymId: string | null,
targetGymId: string | undefined,
): boolean {
// SuperAdmin can access any gym
if (userRole === "superAdmin") {
return true;
}
// If no target gym specified, allow access to own gym
if (targetGymId === undefined) {
return true;
}
// Non-superAdmins must have a gymId
if (!userGymId) {
return false;
}
// Must match the gym
return userGymId === targetGymId;
}
/**
* Check if a user can manage (create/update/delete) another user
* Based on role hierarchy and gym membership
*
* @param requesterRole - Role of the user making the request
* @param requesterGymId - Gym ID of the requesting user
* @param targetRole - Role of the user being managed
* @param targetGymId - Gym ID of the user being managed
* @returns true if management is allowed
*
* @example
* // SuperAdmin can manage anyone
* canManageUser("superAdmin", null, "admin", "gym_abc") // true
*
* // Admin can manage trainers and clients in their gym
* canManageUser("admin", "gym_abc", "trainer", "gym_abc") // true
* canManageUser("admin", "gym_abc", "client", "gym_abc") // true
*
* // Admin cannot manage users in other gyms
* canManageUser("admin", "gym_abc", "trainer", "gym_xyz") // false
*
* // Trainer can manage clients in their gym
* canManageUser("trainer", "gym_abc", "client", "gym_abc") // true
*/
export function canManageUser(
requesterRole: UserRole,
requesterGymId: string | null,
targetRole: UserRole,
targetGymId: string | null,
): boolean {
// SuperAdmin can manage anyone
if (requesterRole === "superAdmin") {
return true;
}
// Role hierarchy: admin > trainer > client
const roleHierarchy: Record<UserRole, number> = {
superAdmin: 4,
admin: 3,
trainer: 2,
client: 1,
};
const requesterLevel = roleHierarchy[requesterRole];
const targetLevel = roleHierarchy[targetRole];
// Cannot manage users at same or higher level
if (requesterLevel <= targetLevel) {
return false;
}
// Must have gym access
if (!requesterGymId) {
return false;
}
// For admin/trainer, target must be in same gym
if (requesterRole === "admin" || requesterRole === "trainer") {
return requesterGymId === targetGymId;
}
return false;
}
/**
* Check if a trainer can access a specific client's data
* Trainer must be assigned to that client via trainerClients table
*
* @param trainerGymId - Gym ID of the trainer
* @param clientGymId - Gym ID of the client
* @returns true if in same gym
*
* @note In a more complex system, this would query trainerClients table
* For now, we use gym-based access (trainer and client in same gym)
*/
export function canTrainerAccessClient(
trainerGymId: string | null,
clientGymId: string | null,
): boolean {
if (!trainerGymId || !clientGymId) {
return false;
}
return trainerGymId === clientGymId;
}
/**
* Get the roles that a specific role can invite
* Based on role hierarchy
*
* @param role - The role wanting to invite
* @returns Array of roles that can be invited
*
* @example
* getInvitableRoles("superAdmin") // ["admin", "trainer", "client"]
* getInvitableRoles("admin") // ["trainer", "client"]
* getInvitableRoles("trainer") // ["client"]
* getInvitableRoles("client") // []
*/
export function getInvitableRoles(role: UserRole): UserRole[] {
const invitationRules: Record<UserRole, UserRole[]> = {
superAdmin: ["admin", "trainer", "client"],
admin: ["trainer", "client"],
trainer: ["client"],
client: [],
};
return invitationRules[role] ?? [];
}
/**
* Validate gym access and return error response if denied
* Use this in API routes to enforce gym-based access control
*
* @param request - NextRequest for getting query params
* @param userRole - Current user's role
* @param userGymId - Current user's gym ID
* @param targetGymId - Gym being accessed (optional)
* @returns NextResponse if denied, undefined if allowed
*
* @example
* export async function GET(request: NextRequest) {
* const error = await validateGymAccess(request, role, gymId);
* if (error) return error;
* // ... proceed
* }
*/
export function validateGymAccess(
userRole: UserRole,
userGymId: string | null,
targetGymId?: string,
): NextResponse | undefined {
if (!canAccessGym(userRole, userGymId, targetGymId)) {
return NextResponse.json(
{ error: "Forbidden - Cannot access this gym's data" },
{ status: 403 },
);
}
return undefined;
}
/**
* Get filter condition for gym-scoped queries
* Returns appropriate filter based on user role
*
* @param userRole - Current user's role
* @param userGymId - Current user's gym ID
* @returns Filter object or undefined (no filter = all gyms)
*
* @example
* // For superAdmin: returns undefined (no filter)
* getGymFilter("superAdmin", null) // undefined
*
* // For admin: returns { gymId: "gym_abc" }
* getGymFilter("admin", "gym_abc") // { gymId: "gym_abc" }
*/
export function getGymFilter(
userRole: UserRole,
userGymId: string | null,
): { gymId: string } | undefined {
// SuperAdmin sees all gyms
if (userRole === "superAdmin") {
return undefined;
}
// Other roles filtered by their gym
if (userGymId) {
return { gymId: userGymId };
}
// No gym assigned - return filter that matches nothing
return { gymId: "" };
}
/**
* Get all gym IDs a user can access
* SuperAdmin gets all gyms, others get only their own
*
* @param userRole - Current user's role
* @param userGymId - Current user's gym ID
* @param allGymIds - List of all gym IDs in system (for superAdmin)
* @returns Array of accessible gym IDs
*
* @example
* getAccessibleGymIds("admin", "gym_abc", ["gym_abc", "gym_xyz"])
* // Returns: ["gym_abc"]
*
* getAccessibleGymIds("superAdmin", null, ["gym_abc", "gym_xyz"])
* // Returns: ["gym_abc", "gym_xyz"]
*/
export function getAccessibleGymIds(
userRole: UserRole,
userGymId: string | null,
allGymIds: string[],
): string[] {
if (userRole === "superAdmin") {
return allGymIds;
}
if (userGymId) {
return [userGymId];
}
return [];
}

View File

@ -1,11 +1,7 @@
import { clerkClient } from "@clerk/nextjs/server";
import { type UserRole } from "@fitai/shared";
import log from "./logger";
/**
* User roles available in the application
*/
export type UserRole = "admin" | "trainer" | "client";
/**
* Set a user's role in Clerk public metadata
* This will trigger a webhook that syncs the role to the database
@ -71,7 +67,8 @@ export async function hasRole(
* const isAdmin = await isAdmin('user_abc123');
*/
export async function isAdmin(userId: string): Promise<boolean> {
return hasRole(userId, "admin");
const role = await getUserRole(userId);
return role === "admin" || role === "superAdmin";
}
/**
@ -161,6 +158,7 @@ export async function getUserCountByRole(): Promise<Record<UserRole, number>> {
const { data: users } = await client.users.getUserList();
const counts: Record<UserRole, number> = {
superAdmin: 0,
admin: 0,
trainer: 0,
client: 0,

View File

@ -7,6 +7,11 @@ import {
Recommendation,
FitnessGoal,
Notification,
DailyNutrition,
DailyHydration,
MealEntry,
FitnessProfileHistory,
TrainerClientAssignment,
DatabaseConfig,
} from "./types";
import {
@ -18,6 +23,11 @@ import {
recommendations,
fitnessGoals,
notifications,
dailyNutrition,
dailyHydration,
mealEntries,
fitnessProfileHistory,
trainerClientAssignments,
eq,
and,
desc,
@ -1318,9 +1328,14 @@ export class DrizzleDatabase implements IDatabase {
membershipStatus: String(
row.membershipStatus,
) as Client["membershipStatus"],
joinDate: new Date(row.joinDate as number | Date),
joinDate:
typeof row.joinDate === "number"
? new Date(row.joinDate * 1000)
: new Date(row.joinDate as Date),
lastVisit: row.lastVisit
? new Date(row.lastVisit as number | Date)
? typeof row.lastVisit === "number"
? new Date(row.lastVisit * 1000)
: new Date(row.lastVisit as Date)
: undefined,
emergencyContact: row.emergencyContactName
? {
@ -1363,12 +1378,20 @@ export class DrizzleDatabase implements IDatabase {
id: String(row.id),
userId: String(row.userId),
type: String(row.type) as Attendance["type"],
checkInTime: new Date(row.checkInTime as number | Date),
checkInTime:
typeof row.checkInTime === "number"
? new Date(row.checkInTime * 1000)
: new Date(row.checkInTime as Date),
checkOutTime: row.checkOutTime
? new Date(row.checkOutTime as number | Date)
? typeof row.checkOutTime === "number"
? new Date(row.checkOutTime * 1000)
: new Date(row.checkOutTime as Date)
: undefined,
notes: row.notes ? String(row.notes) : undefined,
createdAt: new Date(row.createdAt as number | Date),
createdAt:
typeof row.createdAt === "number"
? new Date(row.createdAt * 1000)
: new Date(row.createdAt as Date),
};
}
@ -1511,4 +1534,661 @@ export class DrizzleDatabase implements IDatabase {
createdAt,
};
}
// ==================== DAILY NUTRITION OPERATIONS ====================
async createDailyNutrition(
nutrition: Omit<DailyNutrition, "id" | "createdAt" | "updatedAt">,
): Promise<DailyNutrition> {
const id = `nutrition_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
const now = new Date();
const newNutrition = {
id,
...nutrition,
createdAt: now,
updatedAt: now,
};
await this.db.insert(dailyNutrition).values(newNutrition as any);
return this.mapDailyNutrition(newNutrition);
}
async getDailyNutrition(
userId: string,
date: string,
): Promise<DailyNutrition | null> {
const results = await this.db
.select()
.from(dailyNutrition)
.where(
and(eq(dailyNutrition.userId, userId), eq(dailyNutrition.date, date)),
)
.limit(1)
.all();
return results.length > 0 ? this.mapDailyNutrition(results[0]) : null;
}
async getDailyNutritionById(id: string): Promise<DailyNutrition | null> {
const results = await this.db
.select()
.from(dailyNutrition)
.where(eq(dailyNutrition.id, id))
.limit(1)
.all();
return results.length > 0 ? this.mapDailyNutrition(results[0]) : null;
}
async getDailyNutritionRange(
userId: string,
startDate: string,
endDate: string,
): Promise<DailyNutrition[]> {
const results = await this.db
.select()
.from(dailyNutrition)
.where(
and(
eq(dailyNutrition.userId, userId),
gte(dailyNutrition.date, startDate),
lte(dailyNutrition.date, endDate),
),
)
.orderBy(dailyNutrition.date)
.all();
return results.map((row) => this.mapDailyNutrition(row));
}
async updateDailyNutrition(
id: string,
updates: Partial<DailyNutrition>,
): Promise<DailyNutrition | null> {
const { id: _, ...updateData } = updates as any;
if (Object.keys(updateData).length === 0) {
const existing = await this.db
.select()
.from(dailyNutrition)
.where(eq(dailyNutrition.id, id))
.limit(1)
.all();
return existing.length > 0 ? this.mapDailyNutrition(existing[0]) : null;
}
await this.db
.update(dailyNutrition)
.set({ ...updateData, updatedAt: new Date() })
.where(eq(dailyNutrition.id, id))
.run();
const results = await this.db
.select()
.from(dailyNutrition)
.where(eq(dailyNutrition.id, id))
.limit(1)
.all();
return results.length > 0 ? this.mapDailyNutrition(results[0]) : null;
}
async deleteDailyNutrition(id: string): Promise<boolean> {
const result = await this.db
.delete(dailyNutrition)
.where(eq(dailyNutrition.id, id))
.run();
return result.changes > 0;
}
private mapDailyNutrition(row: Record<string, unknown>): DailyNutrition {
const createdAtValue = row.createdAt as number | Date;
const updatedAtValue = row.updatedAt as number | Date;
return {
id: String(row.id),
userId: String(row.userId),
date: String(row.date),
totalCalories: Number(row.totalCalories || 0),
calorieGoal: Number(row.calorieGoal || 2000),
meals: row.meals ? (JSON.parse(String(row.meals)) as any) : undefined,
createdAt:
typeof createdAtValue === "number"
? new Date(createdAtValue * 1000)
: new Date(createdAtValue),
updatedAt:
typeof updatedAtValue === "number"
? new Date(updatedAtValue * 1000)
: new Date(updatedAtValue),
};
}
// ==================== MEAL ENTRY OPERATIONS ====================
async createMealEntry(
meal: Omit<MealEntry, "id" | "createdAt">,
): Promise<MealEntry> {
const id = `meal_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
const now = new Date();
const newMeal = {
id,
...meal,
createdAt: now,
};
await this.db.insert(mealEntries).values(newMeal as any);
return this.mapMealEntry(newMeal);
}
async getMealEntriesByDate(
userId: string,
date: string,
): Promise<MealEntry[]> {
// Parse date string to get start and end of day in seconds
const startOfDay = new Date(date);
startOfDay.setHours(0, 0, 0, 0);
const endOfDay = new Date(date);
endOfDay.setHours(23, 59, 59, 999);
const startTimestamp = Math.floor(startOfDay.getTime() / 1000);
const endTimestamp = Math.floor(endOfDay.getTime() / 1000);
const results = await this.db
.select()
.from(mealEntries)
.where(
and(
eq(mealEntries.userId, userId),
sql`${mealEntries.timestamp} >= ${startTimestamp}`,
sql`${mealEntries.timestamp} <= ${endTimestamp}`,
),
)
.orderBy(mealEntries.timestamp)
.all();
return results.map((row) => this.mapMealEntry(row));
}
async getMealEntryById(id: string): Promise<MealEntry | null> {
const results = await this.db
.select()
.from(mealEntries)
.where(eq(mealEntries.id, id))
.limit(1)
.all();
return results.length > 0 ? this.mapMealEntry(results[0]) : null;
}
async deleteMealEntry(id: string): Promise<boolean> {
const result = await this.db
.delete(mealEntries)
.where(eq(mealEntries.id, id))
.run();
return result.changes > 0;
}
private mapMealEntry(row: Record<string, unknown>): MealEntry {
const timestampValue = row.timestamp as number | Date;
const createdAtValue = row.createdAt as number | Date;
return {
id: String(row.id),
userId: String(row.userId),
dailyNutritionId: row.dailyNutritionId
? String(row.dailyNutritionId)
: undefined,
mealType: String(row.mealType) as MealEntry["mealType"],
foodName: String(row.foodName),
calories: Number(row.calories),
protein: row.protein ? Number(row.protein) : undefined,
carbs: row.carbs ? Number(row.carbs) : undefined,
fats: row.fats ? Number(row.fats) : undefined,
timestamp:
typeof timestampValue === "number"
? new Date(timestampValue * 1000)
: new Date(timestampValue),
createdAt:
typeof createdAtValue === "number"
? new Date(createdAtValue * 1000)
: new Date(createdAtValue),
};
}
// ==================== DAILY HYDRATION OPERATIONS ====================
async createDailyHydration(
hydration: Omit<DailyHydration, "id" | "createdAt" | "updatedAt">,
): Promise<DailyHydration> {
const id = `hydration_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
const now = new Date();
const newHydration = {
id,
...hydration,
createdAt: now,
updatedAt: now,
};
await this.db.insert(dailyHydration).values(newHydration as any);
return this.mapDailyHydration(newHydration);
}
async getDailyHydration(
userId: string,
date: string,
): Promise<DailyHydration | null> {
const results = await this.db
.select()
.from(dailyHydration)
.where(
and(eq(dailyHydration.userId, userId), eq(dailyHydration.date, date)),
)
.limit(1)
.all();
return results.length > 0 ? this.mapDailyHydration(results[0]) : null;
}
async getDailyHydrationById(id: string): Promise<DailyHydration | null> {
const results = await this.db
.select()
.from(dailyHydration)
.where(eq(dailyHydration.id, id))
.limit(1)
.all();
return results.length > 0 ? this.mapDailyHydration(results[0]) : null;
}
async getDailyHydrationRange(
userId: string,
startDate: string,
endDate: string,
): Promise<DailyHydration[]> {
const results = await this.db
.select()
.from(dailyHydration)
.where(
and(
eq(dailyHydration.userId, userId),
gte(dailyHydration.date, startDate),
lte(dailyHydration.date, endDate),
),
)
.orderBy(dailyHydration.date)
.all();
return results.map((row) => this.mapDailyHydration(row));
}
async updateDailyHydration(
id: string,
updates: Partial<DailyHydration>,
): Promise<DailyHydration | null> {
const { id: _, ...updateData } = updates as any;
if (Object.keys(updateData).length === 0) {
const existing = await this.db
.select()
.from(dailyHydration)
.where(eq(dailyHydration.id, id))
.limit(1)
.all();
return existing.length > 0 ? this.mapDailyHydration(existing[0]) : null;
}
await this.db
.update(dailyHydration)
.set({ ...updateData, updatedAt: new Date() })
.where(eq(dailyHydration.id, id))
.run();
const results = await this.db
.select()
.from(dailyHydration)
.where(eq(dailyHydration.id, id))
.limit(1)
.all();
return results.length > 0 ? this.mapDailyHydration(results[0]) : null;
}
async deleteDailyHydration(id: string): Promise<boolean> {
const result = await this.db
.delete(dailyHydration)
.where(eq(dailyHydration.id, id))
.run();
return result.changes > 0;
}
private mapDailyHydration(row: Record<string, unknown>): DailyHydration {
const createdAtValue = row.createdAt as number | Date;
const updatedAtValue = row.updatedAt as number | Date;
return {
id: String(row.id),
userId: String(row.userId),
date: String(row.date),
totalWater: Number(row.totalWater || 0),
waterGoal: Number(row.waterGoal || 2000),
entries: row.entries
? (JSON.parse(String(row.entries)) as any)
: undefined,
createdAt:
typeof createdAtValue === "number"
? new Date(createdAtValue * 1000)
: new Date(createdAtValue),
updatedAt:
typeof updatedAtValue === "number"
? new Date(updatedAtValue * 1000)
: new Date(updatedAtValue),
};
}
// ==================== FITNESS PROFILE HISTORY OPERATIONS ====================
async createFitnessProfileHistory(
history: Omit<FitnessProfileHistory, "id" | "createdAt">,
): Promise<FitnessProfileHistory> {
const id = `profile_history_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
const now = new Date();
const newHistory = {
id,
...history,
createdAt: now,
};
await this.db.insert(fitnessProfileHistory).values(newHistory as any);
return this.mapFitnessProfileHistory(newHistory);
}
async getFitnessProfileHistory(
userId: string,
startDate?: Date,
endDate?: Date,
): Promise<FitnessProfileHistory[]> {
const conditions = [eq(fitnessProfileHistory.userId, userId)];
if (startDate) {
const startTimestamp = Math.floor(startDate.getTime() / 1000);
conditions.push(
sql`${fitnessProfileHistory.changedAt} >= ${startTimestamp}`,
);
}
if (endDate) {
const endTimestamp = Math.floor(endDate.getTime() / 1000);
conditions.push(
sql`${fitnessProfileHistory.changedAt} <= ${endTimestamp}`,
);
}
const results = await this.db
.select()
.from(fitnessProfileHistory)
.where(and(...conditions))
.orderBy(desc(fitnessProfileHistory.changedAt))
.all();
return results.map((row) => this.mapFitnessProfileHistory(row));
}
async getWeightHistory(
userId: string,
startDate: Date,
endDate: Date,
): Promise<FitnessProfileHistory[]> {
const startTimestamp = Math.floor(startDate.getTime() / 1000);
const endTimestamp = Math.floor(endDate.getTime() / 1000);
const results = await this.db
.select()
.from(fitnessProfileHistory)
.where(
and(
eq(fitnessProfileHistory.userId, userId),
eq(fitnessProfileHistory.changeType, "weight"),
sql`${fitnessProfileHistory.changedAt} >= ${startTimestamp}`,
sql`${fitnessProfileHistory.changedAt} <= ${endTimestamp}`,
),
)
.orderBy(fitnessProfileHistory.changedAt)
.all();
return results.map((row) => this.mapFitnessProfileHistory(row));
}
private mapFitnessProfileHistory(
row: Record<string, unknown>,
): FitnessProfileHistory {
const changedAtValue = row.changedAt as number | Date;
const createdAtValue = row.createdAt as number | Date;
return {
id: String(row.id),
userId: String(row.userId),
fitnessProfileId: String(row.fitnessProfileId),
changeType: String(row.changeType) as FitnessProfileHistory["changeType"],
fieldName: String(row.fieldName),
previousValue: row.previousValue ? String(row.previousValue) : undefined,
newValue: row.newValue ? String(row.newValue) : undefined,
changedAt:
typeof changedAtValue === "number"
? new Date(changedAtValue * 1000)
: new Date(changedAtValue),
createdAt:
typeof createdAtValue === "number"
? new Date(createdAtValue * 1000)
: new Date(createdAtValue),
};
}
// ==================== TRAINER-CLIENT ASSIGNMENT OPERATIONS ====================
async createTrainerClientAssignment(
assignment: Omit<TrainerClientAssignment, "id" | "createdAt" | "updatedAt">,
): Promise<TrainerClientAssignment> {
const id = `assignment_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
const now = new Date();
const newAssignment = {
id,
...assignment,
createdAt: now,
updatedAt: now,
};
await this.db.insert(trainerClientAssignments).values(newAssignment as any);
return this.mapTrainerClientAssignment(newAssignment);
}
async getTrainerClientAssignments(
trainerId: string,
): Promise<TrainerClientAssignment[]> {
const results = await this.db
.select()
.from(trainerClientAssignments)
.where(
and(
eq(trainerClientAssignments.trainerId, trainerId),
eq(trainerClientAssignments.isActive, true),
),
)
.orderBy(desc(trainerClientAssignments.assignedAt))
.all();
return results.map((row) => this.mapTrainerClientAssignment(row));
}
async getAllTrainerClientAssignments(): Promise<TrainerClientAssignment[]> {
const results = await this.db
.select()
.from(trainerClientAssignments)
.orderBy(desc(trainerClientAssignments.assignedAt))
.all();
return results.map((row) => this.mapTrainerClientAssignment(row));
}
async getClientTrainerAssignment(
clientId: string,
): Promise<TrainerClientAssignment | null> {
const results = await this.db
.select()
.from(trainerClientAssignments)
.where(
and(
eq(trainerClientAssignments.clientId, clientId),
eq(trainerClientAssignments.isActive, true),
),
)
.limit(1)
.all();
return results.length > 0
? this.mapTrainerClientAssignment(results[0])
: null;
}
async deleteTrainerClientAssignment(id: string): Promise<boolean> {
const result = await this.db
.delete(trainerClientAssignments)
.where(eq(trainerClientAssignments.id, id))
.run();
return result.changes > 0;
}
async deactivateTrainerClientAssignment(
id: string,
): Promise<TrainerClientAssignment | null> {
await this.db
.update(trainerClientAssignments)
.set({ isActive: false, updatedAt: new Date() })
.where(eq(trainerClientAssignments.id, id))
.run();
const results = await this.db
.select()
.from(trainerClientAssignments)
.where(eq(trainerClientAssignments.id, id))
.limit(1)
.all();
return results.length > 0
? this.mapTrainerClientAssignment(results[0])
: null;
}
private mapTrainerClientAssignment(
row: Record<string, unknown>,
): TrainerClientAssignment {
const assignedAtValue = row.assignedAt as number | Date;
const createdAtValue = row.createdAt as number | Date;
const updatedAtValue = row.updatedAt as number | Date;
return {
id: String(row.id),
trainerId: String(row.trainerId),
clientId: String(row.clientId),
assignedAt:
typeof assignedAtValue === "number"
? new Date(assignedAtValue * 1000)
: new Date(assignedAtValue),
assignedBy: row.assignedBy ? String(row.assignedBy) : undefined,
isActive: Boolean(row.isActive),
createdAt:
typeof createdAtValue === "number"
? new Date(createdAtValue * 1000)
: new Date(createdAtValue),
updatedAt:
typeof updatedAtValue === "number"
? new Date(updatedAtValue * 1000)
: new Date(updatedAtValue),
};
}
// ==================== ENHANCED ATTENDANCE OPERATIONS FOR REPORTS ====================
async getAttendanceByWeek(
userId: string,
weekStart: Date,
): Promise<Attendance[]> {
// Calculate week end (Sunday at 23:59:59)
const weekEnd = new Date(weekStart);
weekEnd.setDate(weekEnd.getDate() + 6);
weekEnd.setHours(23, 59, 59, 999);
const startTimestamp = Math.floor(weekStart.getTime() / 1000);
const endTimestamp = Math.floor(weekEnd.getTime() / 1000);
const results = await this.db
.select()
.from(attendance)
.where(
and(
eq(attendance.userId, userId),
sql`${attendance.checkInTime} >= ${startTimestamp}`,
sql`${attendance.checkInTime} <= ${endTimestamp}`,
),
)
.orderBy(attendance.checkInTime)
.all();
return results.map((row) => this.mapAttendance(row));
}
async calculateWeeklyCheckInStats(
userId: string,
weekStart: Date,
): Promise<{
totalCheckIns: number;
totalTimeSpent: number;
avgSessionDuration: number;
byType: { gym: number; class: number; personal_training: number };
}> {
const attendanceRecords = await this.getAttendanceByWeek(userId, weekStart);
const stats = {
totalCheckIns: attendanceRecords.length,
totalTimeSpent: 0,
avgSessionDuration: 0,
byType: {
gym: 0,
class: 0,
personal_training: 0,
},
};
let completedSessions = 0;
for (const record of attendanceRecords) {
// Count by type
if (record.type === "gym") stats.byType.gym++;
else if (record.type === "class") stats.byType.class++;
else if (record.type === "personal_training")
stats.byType.personal_training++;
// Calculate time spent (only for completed sessions with check-out)
if (record.checkOutTime) {
const duration =
(record.checkOutTime.getTime() - record.checkInTime.getTime()) /
(1000 * 60); // in minutes
stats.totalTimeSpent += duration;
completedSessions++;
}
}
// Calculate average session duration
if (completedSessions > 0) {
stats.avgSessionDuration = stats.totalTimeSpent / completedSessions;
}
return stats;
}
}

View File

@ -6,6 +6,11 @@ import {
Recommendation,
FitnessGoal,
Notification,
DailyNutrition,
DailyHydration,
MealEntry,
FitnessProfileHistory,
TrainerClientAssignment,
} from "@fitai/shared";
import type { SortConfig, FilterCondition } from "../filtering";
@ -23,6 +28,11 @@ export type {
Recommendation,
FitnessGoal,
Notification,
DailyNutrition,
DailyHydration,
MealEntry,
FitnessProfileHistory,
TrainerClientAssignment,
};
// Database Interface - allows us to swap implementations
@ -144,6 +154,7 @@ export interface IDatabase {
getRecommendationsByUserId(userId: string): Promise<Recommendation[]>;
getRecommendationsByUserIds(userIds: string[]): Promise<Recommendation[]>;
getAllRecommendations(): Promise<Recommendation[]>;
getRecommendationById(id: string): Promise<Recommendation | null>;
updateRecommendation(
id: string,
updates: Partial<Recommendation>,
@ -188,6 +199,101 @@ export interface IDatabase {
totalRevenue: number;
revenueGrowth: number; // Percentage vs last month
}>;
// Daily Nutrition operations
createDailyNutrition(
nutrition: Omit<DailyNutrition, "id" | "createdAt" | "updatedAt">,
): Promise<DailyNutrition>;
getDailyNutrition(
userId: string,
date: string,
): Promise<DailyNutrition | null>;
getDailyNutritionById(id: string): Promise<DailyNutrition | null>;
getDailyNutritionRange(
userId: string,
startDate: string,
endDate: string,
): Promise<DailyNutrition[]>;
updateDailyNutrition(
id: string,
updates: Partial<DailyNutrition>,
): Promise<DailyNutrition | null>;
deleteDailyNutrition(id: string): Promise<boolean>;
// Meal Entry operations
createMealEntry(
meal: Omit<MealEntry, "id" | "createdAt">,
): Promise<MealEntry>;
getMealEntriesByDate(userId: string, date: string): Promise<MealEntry[]>;
getMealEntryById(id: string): Promise<MealEntry | null>;
deleteMealEntry(id: string): Promise<boolean>;
// Daily Hydration operations
createDailyHydration(
hydration: Omit<DailyHydration, "id" | "createdAt" | "updatedAt">,
): Promise<DailyHydration>;
getDailyHydration(
userId: string,
date: string,
): Promise<DailyHydration | null>;
getDailyHydrationById(id: string): Promise<DailyHydration | null>;
getDailyHydrationRange(
userId: string,
startDate: string,
endDate: string,
): Promise<DailyHydration[]>;
updateDailyHydration(
id: string,
updates: Partial<DailyHydration>,
): Promise<DailyHydration | null>;
deleteDailyHydration(id: string): Promise<boolean>;
// Fitness Profile History operations
createFitnessProfileHistory(
history: Omit<FitnessProfileHistory, "id" | "createdAt">,
): Promise<FitnessProfileHistory>;
getFitnessProfileHistory(
userId: string,
startDate?: Date,
endDate?: Date,
): Promise<FitnessProfileHistory[]>;
getWeightHistory(
userId: string,
startDate: Date,
endDate: Date,
): Promise<FitnessProfileHistory[]>;
// Trainer-Client Assignment operations
createTrainerClientAssignment(
assignment: Omit<TrainerClientAssignment, "id" | "createdAt" | "updatedAt">,
): Promise<TrainerClientAssignment>;
getTrainerClientAssignments(
trainerId: string,
): Promise<TrainerClientAssignment[]>;
getAllTrainerClientAssignments(): Promise<TrainerClientAssignment[]>;
getClientTrainerAssignment(
clientId: string,
): Promise<TrainerClientAssignment | null>;
deleteTrainerClientAssignment(id: string): Promise<boolean>;
deactivateTrainerClientAssignment(
id: string,
): Promise<TrainerClientAssignment | null>;
// Enhanced Attendance operations for reports
getAttendanceByWeek(userId: string, weekStart: Date): Promise<Attendance[]>;
calculateWeeklyCheckInStats(
userId: string,
weekStart: Date,
): Promise<{
totalCheckIns: number;
totalTimeSpent: number; // in minutes
avgSessionDuration: number; // in minutes
byType: {
gym: number;
class: number;
personal_training: number;
};
}>;
}
// Database configuration

View File

@ -0,0 +1,277 @@
import { db, eq, sql, users } from "@fitai/database";
export const DEFAULT_GEOFENCE_RADIUS_METERS = 30;
export const MAX_LOCATION_ACCURACY_METERS = 50;
export const MAX_FALLBACK_ACCURACY_MARGIN_METERS = 120;
export interface UserLocation {
latitude: number;
longitude: number;
accuracy: number;
}
export async function ensureGymsGeofenceColumns(): Promise<void> {
const rows = await db.all(sql`PRAGMA table_info('gyms')`);
const columns = new Set(
(rows as Array<{ name?: string }>).map((row) => row.name).filter(Boolean),
);
if (!columns.has("latitude")) {
await db.run(sql`ALTER TABLE gyms ADD COLUMN latitude REAL`);
}
if (!columns.has("longitude")) {
await db.run(sql`ALTER TABLE gyms ADD COLUMN longitude REAL`);
}
if (!columns.has("geofence_radius_meters")) {
await db.run(
sql`ALTER TABLE gyms ADD COLUMN geofence_radius_meters REAL NOT NULL DEFAULT 30`,
);
}
if (!columns.has("geofence_enabled")) {
await db.run(
sql`ALTER TABLE gyms ADD COLUMN geofence_enabled INTEGER NOT NULL DEFAULT 1`,
);
}
}
interface GymGeofenceConfig {
id: string;
name: string;
latitude: number | null;
longitude: number | null;
geofenceRadiusMeters: number | null;
geofenceEnabled: boolean | null;
}
export async function getUserGymGeofence(
userId: string,
): Promise<GymGeofenceConfig | null> {
await ensureGymsGeofenceColumns();
const user = await db.select().from(users).where(eq(users.id, userId)).get();
if (!user?.gymId) {
return null;
}
const rows = await db.all(sql`
SELECT
id,
name,
latitude,
longitude,
geofence_radius_meters as geofenceRadiusMeters,
geofence_enabled as geofenceEnabled
FROM gyms
WHERE id = ${user.gymId}
LIMIT 1
`);
const gym = rows?.[0] as
| {
id: string;
name: string;
latitude: number | null;
longitude: number | null;
geofenceRadiusMeters: number | null;
geofenceEnabled: number | boolean | null;
}
| undefined;
if (!gym) {
return null;
}
return {
id: gym.id,
name: gym.name,
latitude: gym.latitude,
longitude: gym.longitude,
geofenceRadiusMeters: gym.geofenceRadiusMeters,
geofenceEnabled:
typeof gym.geofenceEnabled === "boolean"
? gym.geofenceEnabled
: gym.geofenceEnabled === null
? null
: Boolean(gym.geofenceEnabled),
};
}
export function parseUserLocation(payload: unknown): UserLocation | null {
if (!payload || typeof payload !== "object") {
return null;
}
const raw = payload as Record<string, unknown>;
const latitude = Number(raw.latitude);
const longitude = Number(raw.longitude);
const accuracy = Number(raw.accuracy);
if (
!Number.isFinite(latitude) ||
!Number.isFinite(longitude) ||
!Number.isFinite(accuracy)
) {
return null;
}
return { latitude, longitude, accuracy };
}
export function validateGeofence(
gym: GymGeofenceConfig,
location: UserLocation | null,
): { ok: true } | { ok: false; status: number; error: string } {
const geofenceEnabled = gym.geofenceEnabled ?? true;
if (!geofenceEnabled) {
return { ok: true };
}
if (!location) {
return {
ok: false,
status: 400,
error: "Location is required for gym check-in/check-out",
};
}
if (location.accuracy > MAX_LOCATION_ACCURACY_METERS) {
return {
ok: false,
status: 400,
error: `Location accuracy too low (${Math.round(location.accuracy)}m). Move to an open area and try again.`,
};
}
if (gym.latitude === null || gym.longitude === null) {
return {
ok: false,
status: 400,
error: "Gym geofence is enabled but gym coordinates are not configured",
};
}
const radius = gym.geofenceRadiusMeters ?? DEFAULT_GEOFENCE_RADIUS_METERS;
const distanceMeters = haversineDistanceMeters(
gym.latitude,
gym.longitude,
location.latitude,
location.longitude,
);
if (distanceMeters > radius) {
return {
ok: false,
status: 403,
error: `You are outside the gym geofence (${Math.round(distanceMeters)}m away, allowed ${Math.round(radius)}m).`,
};
}
return { ok: true };
}
export function validateGeofenceWithFallback(
gym: GymGeofenceConfig,
location: UserLocation | null,
fallbackRequested: boolean,
): { ok: true } | { ok: false; status: number; error: string } {
const geofenceEnabled = gym.geofenceEnabled ?? true;
if (!geofenceEnabled) {
return { ok: true };
}
if (!location) {
return {
ok: false,
status: 400,
error: "Location is required for gym check-in/check-out",
};
}
if (gym.latitude === null || gym.longitude === null) {
return {
ok: false,
status: 400,
error: "Gym geofence is enabled but gym coordinates are not configured",
};
}
const radius = gym.geofenceRadiusMeters ?? DEFAULT_GEOFENCE_RADIUS_METERS;
const distanceMeters = haversineDistanceMeters(
gym.latitude,
gym.longitude,
location.latitude,
location.longitude,
);
if (location.accuracy <= MAX_LOCATION_ACCURACY_METERS) {
if (distanceMeters > radius) {
return {
ok: false,
status: 403,
error: `You are outside the gym geofence (${Math.round(distanceMeters)}m away, allowed ${Math.round(radius)}m).`,
};
}
return { ok: true };
}
if (!fallbackRequested) {
return {
ok: false,
status: 400,
error: `Location accuracy too low (${Math.round(location.accuracy)}m). Move to an open area and try again.`,
};
}
const fallbackMargin = Math.min(
location.accuracy,
MAX_FALLBACK_ACCURACY_MARGIN_METERS,
);
const fallbackAllowedDistance = radius + fallbackMargin;
if (distanceMeters > fallbackAllowedDistance) {
return {
ok: false,
status: 403,
error: `You are outside the gym geofence (${Math.round(distanceMeters)}m away, fallback allowed ${Math.round(fallbackAllowedDistance)}m).`,
};
}
return { ok: true };
}
export function validateCheckInGeofence(
gym: GymGeofenceConfig,
location: UserLocation | null,
fallbackRequested: boolean,
): { ok: true } | { ok: false; status: number; error: string } {
return validateGeofenceWithFallback(gym, location, fallbackRequested);
}
function haversineDistanceMeters(
latitude1: number,
longitude1: number,
latitude2: number,
longitude2: number,
): number {
const earthRadiusMeters = 6371000;
const dLat = toRadians(latitude2 - latitude1);
const dLng = toRadians(longitude2 - longitude1);
const lat1Rad = toRadians(latitude1);
const lat2Rad = toRadians(latitude2);
const a =
Math.sin(dLat / 2) * Math.sin(dLat / 2) +
Math.sin(dLng / 2) *
Math.sin(dLng / 2) *
Math.cos(lat1Rad) *
Math.cos(lat2Rad);
const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
return earthRadiusMeters * c;
}
function toRadians(degrees: number): number {
return (degrees * Math.PI) / 180;
}

View File

@ -0,0 +1,149 @@
import { getDatabase } from "@/lib/database";
/**
* Get users filtered by gym ID
*
* @param gymId - Gym ID to filter by (undefined returns all users for superAdmin)
* @returns Array of users in the gym
*/
export async function getUsersByGym(gymId?: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
if (!gymId) {
return allUsers;
}
return allUsers.filter((u) => u.gymId === gymId);
}
/**
* Get clients filtered by gym ID
*
* @param gymId - Gym ID to filter by
* @returns Array of clients in the gym
*/
export async function getClientsByGym(gymId: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
const allClients = await db.getAllClients();
const gymClientUserIds = allUsers
.filter((u) => u.gymId === gymId && u.role === "client")
.map((u) => u.id);
if (gymClientUserIds.length === 0) {
return [];
}
return allClients.filter((c) => gymClientUserIds.includes(c.userId));
}
/**
* Get trainers filtered by gym ID
*
* @param gymId - Gym ID to filter by
* @returns Array of trainers in the gym
*/
export async function getTrainersByGym(gymId: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
return allUsers.filter((u) => u.gymId === gymId && u.role === "trainer");
}
/**
* Get attendance filtered by gym ID
*
* @param gymId - Gym ID to filter by
* @returns Array of attendance records in the gym
*/
export async function getAttendanceByGym(gymId: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
const allAttendance = await db.getAllAttendance();
const gymUserIds = allUsers.filter((u) => u.gymId === gymId).map((u) => u.id);
if (gymUserIds.length === 0) {
return [];
}
return allAttendance.filter((a) => gymUserIds.includes(a.userId));
}
/**
* Get recommendations filtered by gym ID
*
* @param gymId - Gym ID to filter by
* @returns Array of recommendations in the gym
*/
export async function getRecommendationsByGym(gymId: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
const allRecommendations = await db.getAllRecommendations();
const gymUserIds = allUsers.filter((u) => u.gymId === gymId).map((u) => u.id);
if (gymUserIds.length === 0) {
return [];
}
return allRecommendations.filter((r) => gymUserIds.includes(r.userId));
}
/**
* Get fitness profiles filtered by gym ID
*
* @param gymId - Gym ID to filter by
* @returns Array of fitness profiles in the gym
*/
export async function getFitnessProfilesByGym(gymId: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
const allProfiles = await db.getAllFitnessProfiles();
const gymUserIds = allUsers.filter((u) => u.gymId === gymId).map((u) => u.id);
if (gymUserIds.length === 0) {
return [];
}
return allProfiles.filter((p) => gymUserIds.includes(p.userId));
}
/**
* Get fitness goals filtered by gym ID
*
* @param gymId - Gym ID to filter by
* @returns Array of fitness goals in the gym
*/
export async function getFitnessGoalsByGym(gymId: string) {
const db = await getDatabase();
const allUsers = await db.getAllUsers();
const gymUserIds = allUsers.filter((u) => u.gymId === gymId).map((u) => u.id);
if (gymUserIds.length === 0) {
return [];
}
const allGoals: any[] = [];
for (const userId of gymUserIds) {
const goals = await db.getFitnessGoalsByUserId(userId);
allGoals.push(...goals);
}
return allGoals.sort(
(a, b) => new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime(),
);
}
/**
* Get user by ID and check if they belong to a specific gym
*/
export async function getUserGymId(userId: string): Promise<string | null> {
const db = await getDatabase();
const user = await db.getUserById(userId);
return user?.gymId ?? null;
}

View File

@ -0,0 +1,26 @@
import { getDatabase } from "@/lib/database";
import { getMembershipFeatures } from "./features";
export async function getUserMembershipContext(userId: string): Promise<{
membershipType: "basic" | "premium" | "vip";
features: ReturnType<typeof getMembershipFeatures>;
}> {
const db = await getDatabase();
const user = await db.getUserById(userId);
if (!user || user.role !== "client") {
const membershipType = "vip" as const;
return {
membershipType,
features: getMembershipFeatures(membershipType),
};
}
const client = await db.getClientByUserId(userId);
const membershipType = client?.membershipType ?? "basic";
return {
membershipType,
features: getMembershipFeatures(membershipType),
};
}

View File

@ -0,0 +1,35 @@
import type { MembershipType } from "@/lib/validation/schemas";
export interface MembershipFeatures {
recommendationsPerMonth: number;
hydrationTracking: boolean;
nutritionTracking: boolean;
advancedStatistics: boolean;
}
export const MEMBERSHIP_FEATURES: Record<MembershipType, MembershipFeatures> = {
basic: {
recommendationsPerMonth: 1,
hydrationTracking: false,
nutritionTracking: false,
advancedStatistics: false,
},
premium: {
recommendationsPerMonth: -1,
hydrationTracking: true,
nutritionTracking: true,
advancedStatistics: true,
},
vip: {
recommendationsPerMonth: -1,
hydrationTracking: true,
nutritionTracking: true,
advancedStatistics: true,
},
};
export function getMembershipFeatures(
membershipType: MembershipType,
): MembershipFeatures {
return MEMBERSHIP_FEATURES[membershipType];
}

View File

@ -0,0 +1,232 @@
/**
* Migration Script: Create new tables for report generation
*
* This script:
* 1. Creates the following tables:
* - daily_nutrition - Daily calorie tracking
* - meal_entries - Individual meal details
* - daily_hydration - Daily water intake tracking
* - fitness_profile_history - Profile change history
* - trainer_client_assignments - Trainer-client relationships
*
* 2. Fixes gym assignments for users without gymId:
* - Assigns superAdmin to their first gym
* - Assigns other users to gym of their trainer
*
* Run with: node apps/admin/src/lib/migrations/create-report-tables.js
*
* Note: Run this AFTER setting up the base database
*/
const Database = require("better-sqlite3");
// Use absolute path to the database
const dbPath = "/home/echo/dev/prototype/apps/admin/data/fitai.db";
function createReportTables() {
console.log("Starting report tables migration...\n");
console.log(`Database path: ${dbPath}\n`);
const db = new Database(dbPath);
// 1. Create daily_nutrition table
console.log("Creating daily_nutrition table...");
db.exec(`
CREATE TABLE IF NOT EXISTS daily_nutrition (
id TEXT PRIMARY KEY,
user_id TEXT NOT NULL,
date TEXT NOT NULL,
total_calories INTEGER DEFAULT 0,
calorie_goal INTEGER DEFAULT 2000,
meals TEXT DEFAULT '[]',
created_at INTEGER NOT NULL,
updated_at INTEGER NOT NULL,
UNIQUE(user_id, date)
)
`);
console.log(" ✓ daily_nutrition table created");
// 2. Create meal_entries table
console.log("Creating meal_entries table...");
db.exec(`
CREATE TABLE IF NOT EXISTS meal_entries (
id TEXT PRIMARY KEY,
user_id TEXT NOT NULL,
daily_nutrition_id TEXT,
meal_type TEXT NOT NULL,
food_name TEXT NOT NULL,
calories INTEGER NOT NULL,
protein INTEGER,
carbs INTEGER,
fats INTEGER,
timestamp INTEGER NOT NULL,
created_at INTEGER NOT NULL
)
`);
console.log(" ✓ meal_entries table created");
// 3. Create daily_hydration table
console.log("Creating daily_hydration table...");
db.exec(`
CREATE TABLE IF NOT EXISTS daily_hydration (
id TEXT PRIMARY KEY,
user_id TEXT NOT NULL,
date TEXT NOT NULL,
total_water INTEGER DEFAULT 0,
water_goal INTEGER DEFAULT 2000,
entries TEXT DEFAULT '[]',
created_at INTEGER NOT NULL,
updated_at INTEGER NOT NULL,
UNIQUE(user_id, date)
)
`);
console.log(" ✓ daily_hydration table created");
// 4. Create fitness_profile_history table
console.log("Creating fitness_profile_history table...");
db.exec(`
CREATE TABLE IF NOT EXISTS fitness_profile_history (
id TEXT PRIMARY KEY,
user_id TEXT NOT NULL,
fitness_profile_id TEXT NOT NULL,
change_type TEXT NOT NULL,
field_name TEXT NOT NULL,
previous_value TEXT,
new_value TEXT,
changed_at INTEGER NOT NULL,
created_at INTEGER NOT NULL
)
`);
console.log(" ✓ fitness_profile_history table created");
// 5. Create trainer_client_assignments table
console.log("Creating trainer_client_assignments table...");
db.exec(`
CREATE TABLE IF NOT EXISTS trainer_client_assignments (
id TEXT PRIMARY KEY,
trainer_id TEXT NOT NULL,
client_id TEXT NOT NULL,
assigned_at INTEGER NOT NULL,
assigned_by TEXT,
is_active INTEGER DEFAULT 1,
created_at INTEGER NOT NULL,
updated_at INTEGER NOT NULL
)
`);
console.log(" ✓ trainer_client_assignments table created");
// Create indexes for better query performance
console.log("\nCreating indexes...");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_daily_nutrition_user_date
ON daily_nutrition(user_id, date)
`);
console.log(" ✓ Index: daily_nutrition.user_id + date");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_meal_entries_user_timestamp
ON meal_entries(user_id, timestamp)
`);
console.log(" ✓ Index: meal_entries.user_id + timestamp");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_daily_hydration_user_date
ON daily_hydration(user_id, date)
`);
console.log(" ✓ Index: daily_hydration.user_id + date");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_fitness_profile_history_user
ON fitness_profile_history(user_id)
`);
console.log(" ✓ Index: fitness_profile_history.user_id");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_trainer_client_assignments_trainer
ON trainer_client_assignments(trainer_id)
`);
console.log(" ✓ Index: trainer_client_assignments.trainer_id");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_trainer_client_assignments_client
ON trainer_client_assignments(client_id)
`);
console.log(" ✓ Index: trainer_client_assignments.client_id");
db.exec(`
CREATE INDEX IF NOT EXISTS idx_trainer_client_assignments_active
ON trainer_client_assignments(trainer_id, client_id, is_active)
`);
console.log(" ✓ Index: trainer_client_assignments (composite)");
db.close();
console.log("\n=== Migration Complete ===");
console.log("All report generation tables created successfully!");
console.log("\nTables created:");
console.log(" - daily_nutrition");
console.log(" - meal_entries");
console.log(" - daily_hydration");
console.log(" - fitness_profile_history");
console.log(" - trainer_client_assignments");
console.log("\nIndexes created: 7");
// Fix gym assignments for users without gymId
console.log("\n=== Fixing Gym Assignments ===");
const usersWithoutGym = db
.prepare("SELECT id, email, role FROM users WHERE gym_id IS NULL")
.all();
console.log(`Found ${usersWithoutGym.length} users without gymId`);
let fixedCount = 0;
for (const user of usersWithoutGym) {
if (user.role === "superAdmin") {
// Get first gym
const gym = db.prepare("SELECT id FROM gyms LIMIT 1").get();
if (gym) {
db.prepare("UPDATE users SET gym_id = ? WHERE id = ?").run(
gym.id,
user.id,
);
console.log(` ✓ Fixed ${user.email} (superAdmin) -> gym ${gym.id}`);
fixedCount++;
}
} else {
// Try to find gym from trainer_clients table
const trainerClient = db
.prepare(
"SELECT gym_id FROM trainer_clients WHERE trainer_user_id = ? OR client_user_id = ? LIMIT 1",
)
.get(user.id, user.id);
if (trainerClient) {
db.prepare("UPDATE users SET gym_id = ? WHERE id = ?").run(
trainerClient.gym_id,
user.id,
);
console.log(
` ✓ Fixed ${user.email} (${user.role}) -> gym ${trainerClient.gym_id}`,
);
fixedCount++;
} else {
console.log(
` ⚠ Could not fix ${user.email} (${user.role}) - no trainer_clients record`,
);
}
}
}
console.log(`\nFixed ${fixedCount} users without gymId`);
console.log("\nGym assignments update complete!");
}
// Run if called directly
if (require.main === module) {
createReportTables();
}
module.exports = { createReportTables };

View File

@ -0,0 +1,138 @@
/**
* Migration Script: Fix gym assignments for users without gymId
*
* This script:
* 1. Finds all users with null gymId
* 2. For admins: Assigns them to gym where they are adminUserId
* 3. For trainers: Gets gymId from their trainerClients records
* 4. For clients: Gets gymId from trainerClients or leaves as null for manual review
*
* Run with: node apps/admin/src/lib/migrations/fix-gym-assignments.js
*
* Note: Run this AFTER setting up the database, before starting the app
*/
const Database = require("better-sqlite3");
// Use absolute path to the database - this is where the app stores data
const dbPath = "/home/echo/dev/prototype/apps/admin/data/fitai.db";
function fixGymAssignments() {
console.log("Starting gym assignment migration...\n");
console.log(`Database path: ${dbPath}\n`);
const db = new Database(dbPath);
// Step 1: Find all users without gymId
const usersWithoutGym = db
.prepare(
`
SELECT id, email, role FROM users WHERE gym_id IS NULL
`,
)
.all();
console.log(`Found ${usersWithoutGym.length} users without gymId`);
let adminsFixed = 0;
let trainersFixed = 0;
let clientsFixed = 0;
const unableToFix = [];
for (const user of usersWithoutGym) {
try {
if (user.role === "admin") {
// Find gym where this user is admin
const gym = db
.prepare(
`
SELECT id, name FROM gyms WHERE admin_user_id = ?
`,
)
.get(user.id);
if (gym) {
db.prepare(
`
UPDATE users SET gym_id = ? WHERE id = ?
`,
).run(gym.id, user.id);
console.log(` ✓ Fixed admin ${user.email} -> gym ${gym.name}`);
adminsFixed++;
} else {
unableToFix.push(`Admin ${user.email} (no gym found)`);
}
} else if (user.role === "trainer") {
// Get gym from trainerClients
const tc = db
.prepare(
`
SELECT gym_id FROM trainer_clients WHERE trainer_user_id = ? LIMIT 1
`,
)
.get(user.id);
if (tc) {
db.prepare(
`
UPDATE users SET gym_id = ? WHERE id = ?
`,
).run(tc.gym_id, user.id);
console.log(` ✓ Fixed trainer ${user.email} -> gym ${tc.gym_id}`);
trainersFixed++;
} else {
unableToFix.push(`Trainer ${user.email} (no trainerClients record)`);
}
} else if (user.role === "client") {
// Get gym from trainerClients
const tc = db
.prepare(
`
SELECT gym_id FROM trainer_clients WHERE client_user_id = ? LIMIT 1
`,
)
.get(user.id);
if (tc) {
db.prepare(
`
UPDATE users SET gym_id = ? WHERE id = ?
`,
).run(tc.gym_id, user.id);
console.log(` ✓ Fixed client ${user.email} -> gym ${tc.gym_id}`);
clientsFixed++;
} else {
unableToFix.push(`Client ${user.email} (no trainer assignment)`);
}
}
} catch (error) {
console.error(` ✗ Error fixing user ${user.email}:`, error.message);
unableToFix.push(`User ${user.email} (error: ${error.message})`);
}
}
db.close();
console.log("\n=== Migration Summary ===");
console.log(`Admins fixed: ${adminsFixed}`);
console.log(`Trainers fixed: ${trainersFixed}`);
console.log(`Clients fixed: ${clientsFixed}`);
console.log(`Unable to fix: ${unableToFix.length}`);
if (unableToFix.length > 0) {
console.log("\nUsers requiring manual review:");
unableToFix.forEach((u) => console.log(` - ${u}`));
}
console.log("\nMigration complete!");
}
// Run if called directly
if (require.main === module) {
fixGymAssignments();
}
module.exports = { fixGymAssignments };

View File

@ -0,0 +1,236 @@
import type { UserReport, User, Client, FitnessProfile } from "@fitai/shared";
// Mock data for testing
const mockUser: User = {
id: "user_123",
email: "john.doe@example.com",
firstName: "John",
lastName: "Doe",
role: "client",
phone: "555-1234",
createdAt: new Date("2024-01-15"),
updatedAt: new Date("2024-01-15"),
};
const mockClient: Client = {
id: "client_123",
userId: "user_123",
membershipType: "premium",
membershipStatus: "active",
joinDate: new Date("2024-01-15"),
};
const mockProfile: FitnessProfile = {
id: "profile_123",
userId: "user_123",
height: 175,
weight: 70,
age: 30,
gender: "male",
activityLevel: "moderately_active",
createdAt: new Date("2024-01-15"),
updatedAt: new Date("2024-01-15"),
};
const mockReport: UserReport = {
userId: "user_123",
user: mockUser,
client: mockClient,
fitnessProfile: mockProfile,
reportPeriod: {
startDate: "2024-01-01",
endDate: "2024-01-31",
},
weeklyCheckIns: [
{
weekStart: "2024-01-01",
weekEnd: "2024-01-07",
totalCheckIns: 4,
totalTimeMinutes: 240,
averageDurationMinutes: 60,
checkInsByType: [
{ type: "gym", count: 3 },
{ type: "class", count: 1 },
],
},
{
weekStart: "2024-01-08",
weekEnd: "2024-01-14",
totalCheckIns: 5,
totalTimeMinutes: 300,
averageDurationMinutes: 60,
checkInsByType: [
{ type: "gym", count: 4 },
{ type: "class", count: 1 },
],
},
],
nutrition: {
dailySummaries: [
{
date: "2024-01-01",
totalCalories: 2000,
calorieGoal: 2200,
caloriesDelta: -200,
mealsCount: 3,
},
{
date: "2024-01-02",
totalCalories: 2300,
calorieGoal: 2200,
caloriesDelta: 100,
mealsCount: 4,
},
],
averageDailyCalories: 2150,
totalDays: 2,
daysMetGoal: 1,
},
hydration: {
dailySummaries: [
{
date: "2024-01-01",
totalWater: 2000,
waterGoal: 2500,
hydrationPercentage: 80,
},
{
date: "2024-01-02",
totalWater: 2600,
waterGoal: 2500,
hydrationPercentage: 104,
},
],
averageDailyWater: 2300,
totalDays: 2,
daysMetGoal: 1,
},
goals: {
active: [
{
id: "goal_1",
userId: "user_123",
goalType: "weight_target",
title: "Lose 5kg",
description: "Lose weight for summer",
targetValue: 65,
currentValue: 70,
unit: "kg",
startDate: new Date("2024-01-01"),
status: "active",
progress: 0,
priority: "high",
createdAt: new Date("2024-01-01"),
updatedAt: new Date("2024-01-01"),
},
],
completed: [
{
id: "goal_2",
userId: "user_123",
goalType: "strength_milestone",
title: "Bench press 100kg",
targetValue: 100,
currentValue: 100,
unit: "kg",
startDate: new Date("2023-11-01"),
completedDate: new Date("2023-12-15"),
status: "completed",
progress: 100,
priority: "medium",
createdAt: new Date("2023-11-01"),
updatedAt: new Date("2023-12-15"),
},
],
totalActive: 1,
totalCompleted: 1,
averageProgress: 0,
},
profileHistory: [
{
changeType: "weight",
fieldName: "weight",
previousValue: "72",
newValue: "70",
changedAt: new Date("2024-01-15"),
},
],
recommendations: {
accepted: [
{
id: "rec_1",
userId: "user_123",
fitnessProfileId: "profile_123",
recommendationText: "Increase protein intake to 150g per day",
activityPlan: "High protein diet",
dietPlan: "Protein focused",
status: "approved",
generatedAt: new Date("2024-01-10"),
approvedAt: new Date("2024-01-11"),
approvedBy: "user_trainer",
createdAt: new Date("2024-01-10"),
updatedAt: new Date("2024-01-11"),
},
],
rejected: [],
pending: [
{
id: "rec_2",
userId: "user_123",
fitnessProfileId: "profile_123",
recommendationText: "Try HIIT workouts 3 times per week",
activityPlan: "HIIT training",
dietPlan: "No change",
status: "pending",
generatedAt: new Date("2024-01-20"),
createdAt: new Date("2024-01-20"),
updatedAt: new Date("2024-01-20"),
},
],
totalAccepted: 1,
totalRejected: 0,
totalPending: 1,
},
generatedAt: new Date(),
};
// Test function
export async function testPDFGeneration() {
try {
const { generateReportPDFBase64 } = await import(
"@/lib/pdf/report-helpers"
);
console.log("Generating PDF from mock report...");
const pdfBase64 = generateReportPDFBase64(mockReport);
console.log(`PDF generated successfully!`);
console.log(`PDF size: ${Math.round(pdfBase64.length / 1024)} KB`);
// Verify it's a valid base64 string
if (pdfBase64.startsWith("JVBERi0")) {
console.log("✓ PDF is valid (starts with PDF header)");
} else {
console.log("✗ PDF is invalid");
}
return true;
} catch (error) {
console.error("✗ PDF generation failed:", error);
return false;
}
}
// Run if executed directly
if (typeof window === "undefined" && process.argv[1]?.includes("test-pdf")) {
testPDFGeneration()
.then((success) => {
process.exit(success ? 0 : 1);
})
.catch((err) => {
console.error(err);
process.exit(1);
});
}
export default testPDFGeneration;

View File

@ -0,0 +1,316 @@
import jsPDF from "jspdf";
export interface ChartData {
labels: string[];
datasets: {
label: string;
data: number[];
color?: string;
}[];
}
export interface ChartConfig {
title?: string;
width: number;
height: number;
}
/**
* Generate a bar chart image as a data URL
*/
export function generateBarChart(data: ChartData, config: ChartConfig): string {
const canvas = createCanvas(config.width, config.height);
const ctx = canvas.getContext("2d");
if (!ctx) {
throw new Error("Could not get canvas context");
}
const padding = 40;
const chartWidth = config.width - padding * 2;
const chartHeight = config.height - padding * 2;
const barWidth = chartWidth / data.labels.length - 10;
// Clear canvas
ctx.fillStyle = "#ffffff";
ctx.fillRect(0, 0, config.width, config.height);
// Add title if provided
if (config.title) {
ctx.fillStyle = "#000000";
ctx.font = "bold 14px Arial";
ctx.textAlign = "center";
ctx.fillText(config.title, config.width / 2, 20);
}
// Calculate max value
const maxValue = Math.max(
...data.datasets.flatMap((dataset) => dataset.data),
);
// Draw bars
data.datasets.forEach((dataset, datasetIndex) => {
const color = dataset.color || getDefaultColor(datasetIndex);
ctx.fillStyle = color;
dataset.data.forEach((value, index) => {
const x =
padding +
index * (barWidth + 10) +
(datasetIndex * barWidth) / data.datasets.length;
const barHeight = (value / maxValue) * chartHeight;
const y = config.height - padding - barHeight;
ctx.fillRect(x, y, barWidth / data.datasets.length - 2, barHeight);
// Draw value on top
ctx.fillStyle = "#000000";
ctx.font = "10px Arial";
ctx.textAlign = "center";
ctx.fillText(
value.toString(),
x + barWidth / data.datasets.length / 2,
y - 5,
);
ctx.fillStyle = color;
});
});
// Draw labels
ctx.fillStyle = "#000000";
ctx.font = "11px Arial";
ctx.textAlign = "center";
data.labels.forEach((label, index) => {
const x = padding + index * (barWidth + 10) + barWidth / 2;
ctx.fillText(label, x, config.height - padding + 15);
});
// Draw axes
ctx.strokeStyle = "#cccccc";
ctx.beginPath();
ctx.moveTo(padding, padding);
ctx.lineTo(padding, config.height - padding);
ctx.lineTo(config.width - padding, config.height - padding);
ctx.stroke();
return canvas.toDataURL("image/png");
}
/**
* Generate a line chart image as a data URL
*/
export function generateLineChart(
data: ChartData,
config: ChartConfig,
): string {
const canvas = createCanvas(config.width, config.height);
const ctx = canvas.getContext("2d");
if (!ctx) {
throw new Error("Could not get canvas context");
}
const padding = 40;
const chartWidth = config.width - padding * 2;
const chartHeight = config.height - padding * 2;
// Clear canvas
ctx.fillStyle = "#ffffff";
ctx.fillRect(0, 0, config.width, config.height);
// Add title if provided
if (config.title) {
ctx.fillStyle = "#000000";
ctx.font = "bold 14px Arial";
ctx.textAlign = "center";
ctx.fillText(config.title, config.width / 2, 20);
}
// Calculate max value
const maxValue = Math.max(
...data.datasets.flatMap((dataset) => dataset.data),
1,
);
// Draw grid lines
ctx.strokeStyle = "#f0f0f0";
ctx.lineWidth = 1;
for (let i = 0; i <= 5; i++) {
const y = padding + (i * chartHeight) / 5;
ctx.beginPath();
ctx.moveTo(padding, y);
ctx.lineTo(config.width - padding, y);
ctx.stroke();
// Draw value labels
ctx.fillStyle = "#666666";
ctx.font = "10px Arial";
ctx.textAlign = "right";
const value = Math.round(maxValue - (i * maxValue) / 5);
ctx.fillText(value.toString(), padding - 5, y + 3);
}
// Draw lines
data.datasets.forEach((dataset, datasetIndex) => {
const color = dataset.color || getDefaultColor(datasetIndex);
ctx.strokeStyle = color;
ctx.lineWidth = 2;
ctx.beginPath();
dataset.data.forEach((value, index) => {
const x = padding + (index * chartWidth) / (data.labels.length - 1);
const y = config.height - padding - (value / maxValue) * chartHeight;
if (index === 0) {
ctx.moveTo(x, y);
} else {
ctx.lineTo(x, y);
}
});
ctx.stroke();
// Draw points
ctx.fillStyle = color;
dataset.data.forEach((value, index) => {
const x = padding + (index * chartWidth) / (data.labels.length - 1);
const y = config.height - padding - (value / maxValue) * chartHeight;
ctx.beginPath();
ctx.arc(x, y, 4, 0, 2 * Math.PI);
ctx.fill();
});
});
// Draw labels
ctx.fillStyle = "#000000";
ctx.font = "11px Arial";
ctx.textAlign = "center";
data.labels.forEach((label, index) => {
const x = padding + (index * chartWidth) / (data.labels.length - 1);
ctx.fillText(label, x, config.height - padding + 15);
});
// Draw axes
ctx.strokeStyle = "#cccccc";
ctx.lineWidth = 1;
ctx.beginPath();
ctx.moveTo(padding, padding);
ctx.lineTo(padding, config.height - padding);
ctx.lineTo(config.width - padding, config.height - padding);
ctx.stroke();
return canvas.toDataURL("image/png");
}
/**
* Generate a pie chart image as a data URL
*/
export function generatePieChart(data: ChartData, config: ChartConfig): string {
const canvas = createCanvas(config.width, config.height);
const ctx = canvas.getContext("2d");
if (!ctx) {
throw new Error("Could not get canvas context");
}
const centerX = config.width / 2;
const centerY = config.height / 2;
const radius = Math.min(config.width, config.height) / 2 - 40;
// Clear canvas
ctx.fillStyle = "#ffffff";
ctx.fillRect(0, 0, config.width, config.height);
// Add title if provided
if (config.title) {
ctx.fillStyle = "#000000";
ctx.font = "bold 14px Arial";
ctx.textAlign = "center";
ctx.fillText(config.title, config.width / 2, 20);
}
// Calculate total
const total = data.datasets[0]?.data.reduce((sum, val) => sum + val, 0) || 0;
// Draw pie slices
let startAngle = 0;
data.datasets[0]?.data.forEach((value, index) => {
const sliceAngle = (value / total) * 2 * Math.PI;
const color = data.datasets[0].color || getDefaultColor(index);
ctx.fillStyle = color;
ctx.beginPath();
ctx.moveTo(centerX, centerY);
ctx.arc(centerX, centerY, radius, startAngle, startAngle + sliceAngle);
ctx.closePath();
ctx.fill();
// Draw label
const midAngle = startAngle + sliceAngle / 2;
const labelX = centerX + (radius + 20) * Math.cos(midAngle);
const labelY = centerY + (radius + 20) * Math.sin(midAngle);
ctx.fillStyle = "#000000";
ctx.font = "11px Arial";
ctx.textAlign = "center";
ctx.fillText(`${data.labels[index]}: ${value}`, labelX, labelY);
startAngle += sliceAngle;
});
return canvas.toDataURL("image/png");
}
/**
* Helper function to create a canvas element
*/
function createCanvas(width: number, height: number): HTMLCanvasElement {
if (typeof document !== "undefined") {
const canvas = document.createElement("canvas");
canvas.width = width;
canvas.height = height;
return canvas;
}
// Fallback for server-side rendering
const Canvas = require("canvas");
return new Canvas(width, height);
}
/**
* Get default color for a dataset index
*/
function getDefaultColor(index: number): string {
const colors = [
"#3c82e1", // primary
"#2ea643", // success
"#f0a500", // warning
"#d73232", // destructive
"#8b5cf6", // purple
"#06b6d4", // cyan
];
return colors[index % colors.length];
}
/**
* Add chart image to PDF
*/
export function addChartToPDF(
pdf: jsPDF,
imageData: string,
x: number,
y: number,
width?: number,
height?: number,
): void {
const imgWidth = width || 150;
const imgHeight = height || 80;
pdf.addImage(imageData, "PNG", x, y, imgWidth, imgHeight);
}
export default {
generateBarChart,
generateLineChart,
generatePieChart,
addChartToPDF,
};

View File

@ -0,0 +1,15 @@
export { default as PDFGenerator } from "./pdf-generator";
export {
default as ChartGenerator,
generateBarChart,
generateLineChart,
generatePieChart,
addChartToPDF,
type ChartData,
type ChartConfig,
} from "./chart-generator";
export {
generateReportPDF,
generateReportPDFBase64,
saveReportPDF,
} from "./report-helpers";

View File

@ -0,0 +1,821 @@
import jsPDF from "jspdf";
import autoTable from "jspdf-autotable";
import type { UserReport } from "@fitai/shared";
export class PDFGenerator {
private doc: jsPDF;
private pageWidth: number;
private pageHeight: number;
private margin: { top: number; right: number; bottom: number; left: number };
private currentY: number = 0;
// Theme colors (from globals.css)
private colors = {
primary: [60, 130, 225] as [number, number, number],
primaryDark: [40, 80, 180] as [number, number, number],
success: [46, 160, 67] as [number, number, number],
warning: [240, 185, 11] as [number, number, number],
destructive: [215, 50, 50] as [number, number, number],
text: [30, 30, 30] as [number, number, number],
textLight: [100, 100, 100] as [number, number, number],
background: [245, 245, 245] as [number, number, number],
white: [255, 255, 255] as [number, number, number],
};
constructor() {
this.doc = new jsPDF({
orientation: "portrait",
unit: "mm",
format: "a4",
});
this.pageWidth = this.doc.internal.pageSize.getWidth();
this.pageHeight = this.doc.internal.pageSize.getHeight();
this.margin = { top: 20, right: 20, bottom: 20, left: 20 };
}
/**
* Generate a complete user report PDF
*/
generateUserReport(report: UserReport): jsPDF {
this.addHeader(report);
this.addUserInfo(report);
this.addReportPeriod(report);
this.addWeeklyCheckIns(report.weeklyCheckIns);
this.addNutritionSummary(report.nutrition);
this.addHydrationSummary(report.hydration);
this.addGoalsSummary(report.goals);
this.addProfileHistory(report.profileHistory);
this.addRecommendations(report.recommendations);
this.addFooter();
return this.doc;
}
/**
* Save PDF to file
*/
save(filename: string): void {
this.doc.save(filename);
}
/**
* Get PDF as blob for API response
*/
toBlob(): Blob {
return this.doc.output("blob");
}
/**
* Get PDF as base64 for API response
*/
toBase64(): string {
return this.doc.output("datauristring").split(",")[1];
}
/**
* Add header with title and FitAI branding
*/
private addHeader(report: UserReport): void {
this.currentY = this.margin.top;
// Header background
this.doc.setFillColor(...this.colors.primary);
this.doc.rect(0, 0, this.pageWidth, 35, "F");
// Title
this.doc.setTextColor(...this.colors.white);
this.doc.setFontSize(24);
this.doc.setFont("helvetica", "bold");
this.doc.text("FitAI User Report", this.margin.left, 20);
// Subtitle
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "normal");
this.doc.text(
`Generated: ${new Date().toLocaleDateString("en-US", {
year: "numeric",
month: "long",
day: "numeric",
hour: "2-digit",
minute: "2-digit",
})}`,
this.margin.left,
28,
);
// FitAI logo text
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("FitAI", this.pageWidth - this.margin.right - 20, 20);
this.currentY = 45;
}
/**
* Add user information section
*/
private addUserInfo(report: UserReport): void {
this.checkPageBreak(30);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("User Information", this.margin.left, this.currentY);
this.currentY += 8;
const user = report.user;
const client = report.client;
const profile = report.fitnessProfile;
const userInfo = [
["Name", `${user.firstName} ${user.lastName}`],
["Email", user.email],
["Phone", user.phone || "N/A"],
["Role", user.role],
[
"Member Since",
user.createdAt ? new Date(user.createdAt).toLocaleDateString() : "N/A",
],
];
if (client) {
userInfo.push([
"Membership",
`${client.membershipType} - ${client.membershipStatus}`,
]);
userInfo.push([
"Join Date",
new Date(client.joinDate).toLocaleDateString(),
]);
}
if (profile) {
userInfo.push([
"Fitness Profile",
`Height: ${profile.height || "N/A"}cm, Weight: ${profile.weight || "N/A"}kg`,
]);
}
autoTable(this.doc, {
startY: this.currentY,
body: userInfo,
theme: "plain",
margin: { left: this.margin.left, right: this.margin.right },
columnStyles: {
0: { fontStyle: "bold", cellWidth: 40, fontSize: 10 },
1: { cellWidth: "auto", fontSize: 10 },
},
styles: {
cellPadding: 3,
textColor: this.colors.text,
},
tableWidth: "auto" as any,
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
}
/**
* Add report period section
*/
private addReportPeriod(report: UserReport): void {
this.checkPageBreak(25);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("Report Period", this.margin.left, this.currentY);
this.currentY += 8;
const periodInfo = [
["Start Date", report.reportPeriod.startDate],
["End Date", report.reportPeriod.endDate],
[
"Duration",
`${Math.ceil(
(new Date(report.reportPeriod.endDate).getTime() -
new Date(report.reportPeriod.startDate).getTime()) /
(1000 * 60 * 60 * 24),
)} days`,
],
];
autoTable(this.doc, {
startY: this.currentY,
body: periodInfo,
theme: "plain",
margin: { left: this.margin.left, right: this.margin.right },
columnStyles: {
0: { fontStyle: "bold", cellWidth: 40, fontSize: 10 },
1: { cellWidth: "auto", fontSize: 10 },
},
styles: {
cellPadding: 3,
textColor: this.colors.text,
},
tableWidth: "auto" as any,
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
}
/**
* Add weekly check-ins section
*/
private addWeeklyCheckIns(
weeklyCheckIns: UserReport["weeklyCheckIns"],
): void {
this.checkPageBreak(50);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("Weekly Check-ins", this.margin.left, this.currentY);
this.currentY += 8;
if (weeklyCheckIns.length === 0) {
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "italic");
this.doc.text(
"No check-in data available for this period.",
this.margin.left,
this.currentY,
);
this.currentY += 15;
return;
}
const tableData = weeklyCheckIns.map((week) => [
week.weekStart,
week.weekEnd,
week.totalCheckIns.toString(),
`${week.totalTimeMinutes} min`,
`${week.averageDurationMinutes} min`,
]);
autoTable(this.doc, {
startY: this.currentY,
head: [
["Week Start", "Week End", "Check-ins", "Total Time", "Avg Duration"],
],
body: tableData,
theme: "striped",
headStyles: {
fillColor: this.colors.primary,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 9,
},
bodyStyles: {
fontSize: 9,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
}
/**
* Add nutrition summary section
*/
private addNutritionSummary(nutrition: UserReport["nutrition"]): void {
this.checkPageBreak(60);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("Nutrition Summary", this.margin.left, this.currentY);
this.currentY += 8;
const summaryStats = [
["Total Days Tracked", nutrition.totalDays.toString()],
[
"Average Daily Calories",
nutrition.averageDailyCalories.toLocaleString(),
],
[
"Days Met Goal (±10%)",
`${nutrition.daysMetGoal} / ${nutrition.totalDays}`,
],
];
autoTable(this.doc, {
startY: this.currentY,
body: summaryStats,
theme: "plain",
margin: { left: this.margin.left, right: this.margin.right },
columnStyles: {
0: { fontStyle: "bold", cellWidth: 60, fontSize: 10 },
1: { cellWidth: "auto", fontSize: 10 },
},
styles: {
cellPadding: 3,
textColor: this.colors.text,
},
tableWidth: "auto" as any,
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 10;
// Add daily breakdown if available
if (
nutrition.dailySummaries.length > 0 &&
nutrition.dailySummaries.length <= 14
) {
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "italic");
this.doc.text("Daily Breakdown", this.margin.left, this.currentY);
this.currentY += 5;
const dailyData = nutrition.dailySummaries.map((day) => [
day.date,
day.totalCalories.toLocaleString(),
day.calorieGoal.toLocaleString(),
day.caloriesDelta > 0
? `+${day.caloriesDelta}`
: day.caloriesDelta.toString(),
day.mealsCount.toString(),
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Date", "Calories", "Goal", "Delta", "Meals"]],
body: dailyData,
theme: "striped",
headStyles: {
fillColor: this.colors.primary,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
} else if (nutrition.dailySummaries.length > 14) {
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(9);
this.doc.text(
`Daily breakdown (${nutrition.dailySummaries.length} days) - See dashboard for detailed view.`,
this.margin.left,
this.currentY,
);
this.currentY += 15;
}
}
/**
* Add hydration summary section
*/
private addHydrationSummary(hydration: UserReport["hydration"]): void {
this.checkPageBreak(50);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("Hydration Summary", this.margin.left, this.currentY);
this.currentY += 8;
const summaryStats = [
["Total Days Tracked", hydration.totalDays.toString()],
["Average Daily Water", `${hydration.averageDailyWater} ml`],
["Days Met Goal", `${hydration.daysMetGoal} / ${hydration.totalDays}`],
];
autoTable(this.doc, {
startY: this.currentY,
body: summaryStats,
theme: "plain",
margin: { left: this.margin.left, right: this.margin.right },
columnStyles: {
0: { fontStyle: "bold", cellWidth: 60, fontSize: 10 },
1: { cellWidth: "auto", fontSize: 10 },
},
styles: {
cellPadding: 3,
textColor: this.colors.text,
},
tableWidth: "auto" as any,
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 10;
// Add daily breakdown if available
if (
hydration.dailySummaries.length > 0 &&
hydration.dailySummaries.length <= 14
) {
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "italic");
this.doc.text("Daily Breakdown", this.margin.left, this.currentY);
this.currentY += 5;
const dailyData = hydration.dailySummaries.map((day) => [
day.date,
`${day.totalWater} ml`,
`${day.waterGoal} ml`,
`${day.hydrationPercentage}%`,
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Date", "Total Water", "Goal", "Achievement"]],
body: dailyData,
theme: "striped",
headStyles: {
fillColor: this.colors.primary,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
} else if (hydration.dailySummaries.length > 14) {
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(9);
this.doc.text(
`Daily breakdown (${hydration.dailySummaries.length} days) - See dashboard for detailed view.`,
this.margin.left,
this.currentY,
);
this.currentY += 15;
}
}
/**
* Add goals summary section
*/
private addGoalsSummary(goals: UserReport["goals"]): void {
this.checkPageBreak(60);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("Fitness Goals", this.margin.left, this.currentY);
this.currentY += 8;
const summaryStats = [
["Active Goals", goals.totalActive.toString()],
["Completed Goals", goals.totalCompleted.toString()],
["Average Progress", `${goals.averageProgress}%`],
];
autoTable(this.doc, {
startY: this.currentY,
body: summaryStats,
theme: "plain",
margin: { left: this.margin.left, right: this.margin.right },
columnStyles: {
0: { fontStyle: "bold", cellWidth: 60, fontSize: 10 },
1: { cellWidth: "auto", fontSize: 10 },
},
styles: {
cellPadding: 3,
textColor: this.colors.text,
},
tableWidth: "auto" as any,
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 10;
// Add active goals
if (goals.active.length > 0) {
this.doc.setTextColor(...this.colors.success);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "bold");
this.doc.text("Active Goals", this.margin.left, this.currentY);
this.currentY += 5;
const activeGoals = goals.active.map((goal) => [
goal.title,
goal.goalType,
`${goal.progress}%`,
goal.priority,
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Title", "Type", "Progress", "Priority"]],
body: activeGoals,
theme: "striped",
headStyles: {
fillColor: this.colors.success,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 10;
}
// Add completed goals
if (goals.completed.length > 0) {
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "bold");
this.doc.text("Completed Goals", this.margin.left, this.currentY);
this.currentY += 5;
const completedGoals = goals.completed.map((goal) => [
goal.title,
goal.goalType,
goal.completedDate
? new Date(goal.completedDate).toLocaleDateString()
: "N/A",
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Title", "Type", "Completed Date"]],
body: completedGoals,
theme: "striped",
headStyles: {
fillColor: this.colors.primary,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
}
}
/**
* Add profile history section
*/
private addProfileHistory(history: UserReport["profileHistory"]): void {
this.checkPageBreak(50);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("Fitness Profile Changes", this.margin.left, this.currentY);
this.currentY += 8;
if (history.length === 0) {
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "italic");
this.doc.text(
"No profile changes recorded during this period.",
this.margin.left,
this.currentY,
);
this.currentY += 15;
return;
}
const historyData = history.map((item) => [
item.fieldName,
item.changeType,
item.previousValue || "N/A",
item.newValue || "N/A",
new Date(item.changedAt).toLocaleDateString(),
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Field", "Change Type", "Previous", "New", "Date"]],
body: historyData,
theme: "striped",
headStyles: {
fillColor: this.colors.primary,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
}
/**
* Add recommendations section
*/
private addRecommendations(
recommendations: UserReport["recommendations"],
): void {
this.checkPageBreak(60);
this.doc.setTextColor(...this.colors.primary);
this.doc.setFontSize(14);
this.doc.setFont("helvetica", "bold");
this.doc.text("AI Recommendations", this.margin.left, this.currentY);
this.currentY += 8;
const summaryStats = [
["Accepted", recommendations.totalAccepted.toString()],
["Rejected", recommendations.totalRejected.toString()],
["Pending", recommendations.totalPending.toString()],
];
autoTable(this.doc, {
startY: this.currentY,
body: summaryStats,
theme: "plain",
margin: { left: this.margin.left, right: this.margin.right },
columnStyles: {
0: { fontStyle: "bold", cellWidth: 60, fontSize: 10 },
1: { cellWidth: "auto", fontSize: 10 },
},
styles: {
cellPadding: 3,
textColor: this.colors.text,
},
tableWidth: "auto" as any,
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 10;
// Add accepted recommendations
if (recommendations.accepted.length > 0) {
this.doc.setTextColor(...this.colors.success);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "bold");
this.doc.text(
"Accepted Recommendations",
this.margin.left,
this.currentY,
);
this.currentY += 5;
const acceptedRecs = recommendations.accepted.map((rec) => [
rec.recommendationText.substring(0, 80) +
(rec.recommendationText.length > 80 ? "..." : ""),
new Date(rec.generatedAt).toLocaleDateString(),
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Recommendation", "Generated"]],
body: acceptedRecs,
theme: "striped",
headStyles: {
fillColor: this.colors.success,
textColor: this.colors.white,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 10;
}
// Add pending recommendations
if (recommendations.pending.length > 0) {
this.doc.setTextColor(...this.colors.warning);
this.doc.setFontSize(10);
this.doc.setFont("helvetica", "bold");
this.doc.text("Pending Recommendations", this.margin.left, this.currentY);
this.currentY += 5;
const pendingRecs = recommendations.pending.map((rec) => [
rec.recommendationText.substring(0, 80) +
(rec.recommendationText.length > 80 ? "..." : ""),
new Date(rec.generatedAt).toLocaleDateString(),
]);
autoTable(this.doc, {
startY: this.currentY,
head: [["Recommendation", "Generated"]],
body: pendingRecs,
theme: "striped",
headStyles: {
fillColor: this.colors.warning,
textColor: this.colors.text,
fontStyle: "bold",
fontSize: 8,
},
bodyStyles: {
fontSize: 8,
textColor: this.colors.text,
},
alternateRowStyles: {
fillColor: [250, 250, 250],
},
margin: { left: this.margin.left, right: this.margin.right },
});
this.currentY = (this.doc as any).lastAutoTable.finalY + 15;
}
}
/**
* Add footer to each page
*/
private addFooter(): void {
const pageCount = this.doc.getNumberOfPages();
for (let i = 1; i <= pageCount; i++) {
this.doc.setPage(i);
// Footer line
this.doc.setDrawColor(...this.colors.primary);
this.doc.setLineWidth(0.5);
this.doc.line(
this.margin.left,
this.pageHeight - 15,
this.pageWidth - this.margin.right,
this.pageHeight - 15,
);
// Footer text
this.doc.setTextColor(...this.colors.textLight);
this.doc.setFontSize(8);
this.doc.setFont("helvetica", "normal");
this.doc.text(
`FitAI User Report - Page ${i} of ${pageCount}`,
this.margin.left,
this.pageHeight - 10,
);
this.doc.text(
"Generated by FitAI",
this.pageWidth - this.margin.right - 30,
this.pageHeight - 10,
);
}
}
/**
* Check if we need a page break
*/
private checkPageBreak(requiredSpace: number): void {
if (this.currentY + requiredSpace > this.pageHeight - this.margin.bottom) {
this.doc.addPage();
this.currentY = this.margin.top;
}
}
}
export default PDFGenerator;

View File

@ -0,0 +1,35 @@
import type { UserReport } from "@fitai/shared";
import PDFGenerator from "./pdf-generator";
/**
* Generate a PDF report from a UserReport object
*/
export function generateReportPDF(report: UserReport): Blob {
const generator = new PDFGenerator();
generator.generateUserReport(report);
return generator.toBlob();
}
/**
* Generate a PDF report and return as base64 string
*/
export function generateReportPDFBase64(report: UserReport): string {
const generator = new PDFGenerator();
generator.generateUserReport(report);
return generator.toBase64();
}
/**
* Generate a PDF report and save to file
*/
export function saveReportPDF(report: UserReport, filename: string): void {
const generator = new PDFGenerator();
generator.generateUserReport(report);
generator.save(filename);
}
export default {
generateReportPDF,
generateReportPDFBase64,
saveReportPDF,
};

View File

@ -0,0 +1,153 @@
/**
* ISO Week Calculation Utilities
*
* Helper functions for ISO 8601 week calculations
* Week starts on Monday and ends on Sunday
*/
/**
* Get the Monday of the ISO week for a given date
*/
export function getWeekStart(date: Date): Date {
const d = new Date(date);
const day = d.getDay();
const diff = d.getDate() - day + (day === 0 ? -6 : 1);
d.setDate(diff);
d.setHours(0, 0, 0, 0);
return d;
}
/**
* Get the Sunday of the ISO week for a given date
*/
export function getWeekEnd(date: Date): Date {
const monday = getWeekStart(date);
const sunday = new Date(monday);
sunday.setDate(sunday.getDate() + 6);
sunday.setHours(23, 59, 59, 999);
return sunday;
}
/**
* Get ISO week number (1-53)
*/
export function getISOWeek(date: Date): number {
const d = new Date(
Date.UTC(date.getFullYear(), date.getMonth(), date.getDate()),
);
const dayNum = d.getUTCDay() || 7;
d.setUTCDate(d.getUTCDate() + 4 - dayNum);
const yearStart = new Date(Date.UTC(d.getUTCFullYear(), 0, 1));
return Math.ceil(((d.getTime() - yearStart.getTime()) / 86400000 + 1) / 7);
}
/**
* Format date as YYYY-MM-DD
*/
export function formatDate(date: Date): string {
return date.toISOString().split("T")[0];
}
/**
* Get all weeks in a date range
*/
export function getWeeksInRange(
startDate: string,
endDate: string,
): Array<{
weekStart: string;
weekEnd: string;
weekNumber: number;
}> {
const weeks: Array<{
weekStart: string;
weekEnd: string;
weekNumber: number;
}> = [];
const start = new Date(startDate);
const end = new Date(endDate);
const current = getWeekStart(start);
while (current <= end) {
const weekStart = getWeekStart(current);
const weekEnd = getWeekEnd(current);
weeks.push({
weekStart: formatDate(weekStart),
weekEnd: formatDate(weekEnd),
weekNumber: getISOWeek(current),
});
current.setDate(current.getDate() + 7);
}
return weeks;
}
/**
* Test ISO week calculations
*/
export function testISOWeekCalculations(): void {
console.log("ISO Week Calculation Tests\n");
const testDates = [
{
date: "2024-01-01",
expected: { monday: "2024-01-01", sunday: "2024-01-07", week: 1 },
},
{
date: "2024-01-02",
expected: { monday: "2024-01-01", sunday: "2024-01-07", week: 1 },
},
{
date: "2024-01-07",
expected: { monday: "2024-01-01", sunday: "2024-01-07", week: 1 },
},
{
date: "2024-01-08",
expected: { monday: "2024-01-08", sunday: "2024-01-14", week: 2 },
},
{
date: "2024-12-31",
expected: { monday: "2024-12-30", sunday: "2025-01-05", week: 1 },
}, // Week 1 of 2025
];
testDates.forEach(({ date, expected }) => {
const d = new Date(date);
const monday = formatDate(getWeekStart(d));
const sunday = formatDate(getWeekEnd(d));
const week = getISOWeek(d);
const mondayMatch = monday === expected.monday;
const sundayMatch = sunday === expected.sunday;
const weekMatch = week === expected.week;
console.log(`Date: ${date}`);
console.log(
` Monday: ${monday} ${mondayMatch ? "✓" : "✗ (expected: " + expected.monday + ")"}`,
);
console.log(
` Sunday: ${sunday} ${sundayMatch ? "✓" : "✗ (expected: " + expected.sunday + ")"}`,
);
console.log(
` Week: ${week} ${weekMatch ? "✓" : "✗ (expected: " + expected.week + ")"}`,
);
console.log();
});
}
// Run tests if executed directly
if (typeof require !== "undefined" && require.main === module) {
testISOWeekCalculations();
}
export default {
getWeekStart,
getWeekEnd,
getISOWeek,
formatDate,
getWeeksInRange,
testISOWeekCalculations,
};

View File

@ -15,8 +15,13 @@ export const passwordSchema = z
export const phoneSchema = z
.string()
.regex(/^\+?[1-9]\d{1,14}$/, "Invalid phone number format")
.optional();
.optional()
.refine(
(val) => !val || val.trim() === "" || /^\+?[1-9]\d{1,14}$/.test(val),
{
message: "Invalid phone number format",
},
);
export const dateTimeSchema = z.string().datetime("Invalid datetime format");
@ -31,6 +36,14 @@ export const userRoleSchema = z.enum([
"superAdmin",
]);
export const membershipTypeSchema = z.enum(["basic", "premium", "vip"]);
export const membershipStatusSchema = z.enum([
"active",
"inactive",
"suspended",
]);
export const userSchema = z.object({
email: emailSchema,
password: passwordSchema,
@ -63,6 +76,8 @@ export const userUpdateWithIdSchema = z.object({
role: userRoleSchema.optional(),
phone: phoneSchema,
gymId: z.string().nullable().optional(),
membershipType: membershipTypeSchema.optional(),
membershipStatus: membershipStatusSchema.optional(),
});
// ============================================================================
@ -333,6 +348,8 @@ export type User = z.infer<typeof userSchema>;
export type UserUpdate = z.infer<typeof userUpdateSchema>;
export type UserLogin = z.infer<typeof userLoginSchema>;
export type UserRole = z.infer<typeof userRoleSchema>;
export type MembershipType = z.infer<typeof membershipTypeSchema>;
export type MembershipStatus = z.infer<typeof membershipStatusSchema>;
export type FitnessGoal = z.infer<typeof fitnessGoalSchema>;
export type FitnessGoalUpdate = z.infer<typeof fitnessGoalUpdateSchema>;

Some files were not shown because too many files have changed in this diff Show More