@payez/next-mvp 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (654) hide show
  1. package/README.md +782 -0
  2. package/dist/api/auth-handler.d.ts +67 -0
  3. package/dist/api/auth-handler.js +397 -0
  4. package/dist/api/index.d.ts +10 -0
  5. package/dist/api/index.js +19 -0
  6. package/dist/api-handlers/account/change-password.d.ts +9 -0
  7. package/dist/api-handlers/account/change-password.js +112 -0
  8. package/dist/api-handlers/account/masked-info.d.ts +2 -0
  9. package/dist/api-handlers/account/masked-info.js +41 -0
  10. package/dist/api-handlers/account/profile.d.ts +3 -0
  11. package/dist/api-handlers/account/profile.js +63 -0
  12. package/dist/api-handlers/account/recovery/initiate.d.ts +2 -0
  13. package/dist/api-handlers/account/recovery/initiate.js +26 -0
  14. package/dist/api-handlers/account/recovery/send-code.d.ts +2 -0
  15. package/dist/api-handlers/account/recovery/send-code.js +28 -0
  16. package/dist/api-handlers/account/recovery/verify-code.d.ts +2 -0
  17. package/dist/api-handlers/account/recovery/verify-code.js +28 -0
  18. package/dist/api-handlers/account/reset-password.d.ts +2 -0
  19. package/dist/api-handlers/account/reset-password.js +26 -0
  20. package/dist/api-handlers/account/send-code.d.ts +24 -0
  21. package/dist/api-handlers/account/send-code.js +60 -0
  22. package/dist/api-handlers/account/update-phone.d.ts +27 -0
  23. package/dist/api-handlers/account/update-phone.js +64 -0
  24. package/dist/api-handlers/account/validate-password.d.ts +17 -0
  25. package/dist/api-handlers/account/validate-password.js +81 -0
  26. package/dist/api-handlers/account/verify-email.d.ts +26 -0
  27. package/dist/api-handlers/account/verify-email.js +106 -0
  28. package/dist/api-handlers/account/verify-sms.d.ts +26 -0
  29. package/dist/api-handlers/account/verify-sms.js +106 -0
  30. package/dist/api-handlers/admin/analytics.d.ts +20 -0
  31. package/dist/api-handlers/admin/analytics.js +379 -0
  32. package/dist/api-handlers/admin/audit.d.ts +20 -0
  33. package/dist/api-handlers/admin/audit.js +214 -0
  34. package/dist/api-handlers/admin/index.d.ts +21 -0
  35. package/dist/api-handlers/admin/index.js +41 -0
  36. package/dist/api-handlers/admin/redis-sessions.d.ts +36 -0
  37. package/dist/api-handlers/admin/redis-sessions.js +204 -0
  38. package/dist/api-handlers/admin/sessions.d.ts +21 -0
  39. package/dist/api-handlers/admin/sessions.js +284 -0
  40. package/dist/api-handlers/admin/site-logs.d.ts +46 -0
  41. package/dist/api-handlers/admin/site-logs.js +318 -0
  42. package/dist/api-handlers/admin/users.d.ts +20 -0
  43. package/dist/api-handlers/admin/users.js +222 -0
  44. package/dist/api-handlers/admin/vibe-data.d.ts +80 -0
  45. package/dist/api-handlers/admin/vibe-data.js +268 -0
  46. package/dist/api-handlers/anon/preferences.d.ts +37 -0
  47. package/dist/api-handlers/anon/preferences.js +96 -0
  48. package/dist/api-handlers/auth/jwks.d.ts +2 -0
  49. package/dist/api-handlers/auth/jwks.js +24 -0
  50. package/dist/api-handlers/auth/login.d.ts +42 -0
  51. package/dist/api-handlers/auth/login.js +178 -0
  52. package/dist/api-handlers/auth/refresh.d.ts +74 -0
  53. package/dist/api-handlers/auth/refresh.js +635 -0
  54. package/dist/api-handlers/auth/signout.d.ts +37 -0
  55. package/dist/api-handlers/auth/signout.js +187 -0
  56. package/dist/api-handlers/auth/status.d.ts +8 -0
  57. package/dist/api-handlers/auth/status.js +26 -0
  58. package/dist/api-handlers/auth/update-session.d.ts +37 -0
  59. package/dist/api-handlers/auth/update-session.js +95 -0
  60. package/dist/api-handlers/auth/validate.d.ts +6 -0
  61. package/dist/api-handlers/auth/validate.js +43 -0
  62. package/dist/api-handlers/auth/verify-code.d.ts +43 -0
  63. package/dist/api-handlers/auth/verify-code.js +94 -0
  64. package/dist/api-handlers/session/refresh-viability.d.ts +14 -0
  65. package/dist/api-handlers/session/refresh-viability.js +39 -0
  66. package/dist/api-handlers/session/viability.d.ts +13 -0
  67. package/dist/api-handlers/session/viability.js +146 -0
  68. package/dist/api-handlers/test/force-expire.d.ts +23 -0
  69. package/dist/api-handlers/test/force-expire.js +65 -0
  70. package/dist/auth/auth-decision.d.ts +39 -0
  71. package/dist/auth/auth-decision.js +182 -0
  72. package/dist/auth/auth-options.d.ts +57 -0
  73. package/dist/auth/auth-options.js +213 -0
  74. package/dist/auth/callbacks/index.d.ts +6 -0
  75. package/dist/auth/callbacks/index.js +12 -0
  76. package/dist/auth/callbacks/jwt.d.ts +45 -0
  77. package/dist/auth/callbacks/jwt.js +305 -0
  78. package/dist/auth/callbacks/session.d.ts +60 -0
  79. package/dist/auth/callbacks/session.js +170 -0
  80. package/dist/auth/callbacks/signin.d.ts +23 -0
  81. package/dist/auth/callbacks/signin.js +44 -0
  82. package/dist/auth/events/index.d.ts +4 -0
  83. package/dist/auth/events/index.js +8 -0
  84. package/dist/auth/events/signout.d.ts +17 -0
  85. package/dist/auth/events/signout.js +32 -0
  86. package/dist/auth/providers/credentials.d.ts +32 -0
  87. package/dist/auth/providers/credentials.js +223 -0
  88. package/dist/auth/providers/index.d.ts +5 -0
  89. package/dist/auth/providers/index.js +21 -0
  90. package/dist/auth/providers/oauth.d.ts +26 -0
  91. package/dist/auth/providers/oauth.js +105 -0
  92. package/dist/auth/route-config.d.ts +66 -0
  93. package/dist/auth/route-config.js +190 -0
  94. package/dist/auth/types/auth-types.d.ts +417 -0
  95. package/dist/auth/types/auth-types.js +53 -0
  96. package/dist/auth/types/index.d.ts +6 -0
  97. package/dist/auth/types/index.js +22 -0
  98. package/dist/auth/unauthenticated-routes.d.ts +1 -0
  99. package/dist/auth/unauthenticated-routes.js +19 -0
  100. package/dist/auth/utils/idp-client.d.ts +94 -0
  101. package/dist/auth/utils/idp-client.js +383 -0
  102. package/dist/auth/utils/index.d.ts +5 -0
  103. package/dist/auth/utils/index.js +21 -0
  104. package/dist/auth/utils/token-utils.d.ts +84 -0
  105. package/dist/auth/utils/token-utils.js +219 -0
  106. package/dist/client/AuthContext.d.ts +19 -0
  107. package/dist/client/AuthContext.js +112 -0
  108. package/dist/client/fetch-with-auth.d.ts +11 -0
  109. package/dist/client/fetch-with-auth.js +44 -0
  110. package/dist/client/fetchWithSession.d.ts +3 -0
  111. package/dist/client/fetchWithSession.js +24 -0
  112. package/dist/client/index.d.ts +9 -0
  113. package/dist/client/index.js +20 -0
  114. package/dist/client/useAnonSession.d.ts +36 -0
  115. package/dist/client/useAnonSession.js +99 -0
  116. package/dist/components/SessionSync.d.ts +13 -0
  117. package/dist/components/SessionSync.js +119 -0
  118. package/dist/components/SignalRHealthCheck.d.ts +10 -0
  119. package/dist/components/SignalRHealthCheck.js +97 -0
  120. package/dist/components/account/UserAvatarMenu.d.ts +20 -0
  121. package/dist/components/account/UserAvatarMenu.js +80 -0
  122. package/dist/components/account/index.d.ts +7 -0
  123. package/dist/components/account/index.js +10 -0
  124. package/dist/components/admin/AlertSettingsTab.d.ts +48 -0
  125. package/dist/components/admin/AlertSettingsTab.js +351 -0
  126. package/dist/components/admin/AnalyticsTab.d.ts +22 -0
  127. package/dist/components/admin/AnalyticsTab.js +167 -0
  128. package/dist/components/admin/DataBrowserTab.d.ts +19 -0
  129. package/dist/components/admin/DataBrowserTab.js +252 -0
  130. package/dist/components/admin/LoggingSettingsTab.d.ts +73 -0
  131. package/dist/components/admin/LoggingSettingsTab.js +339 -0
  132. package/dist/components/admin/SessionsTab.d.ts +37 -0
  133. package/dist/components/admin/SessionsTab.js +165 -0
  134. package/dist/components/admin/StatsTab.d.ts +53 -0
  135. package/dist/components/admin/StatsTab.js +161 -0
  136. package/dist/components/admin/VibeAdminContext.d.ts +32 -0
  137. package/dist/components/admin/VibeAdminContext.js +38 -0
  138. package/dist/components/admin/VibeAdminLayout.d.ts +11 -0
  139. package/dist/components/admin/VibeAdminLayout.js +69 -0
  140. package/dist/components/admin/index.d.ts +29 -0
  141. package/dist/components/admin/index.js +44 -0
  142. package/dist/components/auth/FederatedAuthSection.d.ts +8 -0
  143. package/dist/components/auth/FederatedAuthSection.js +45 -0
  144. package/dist/components/auth/ModeAwareLoginPage.d.ts +10 -0
  145. package/dist/components/auth/ModeAwareLoginPage.js +42 -0
  146. package/dist/components/auth/ModeAwareSignupPage.d.ts +9 -0
  147. package/dist/components/auth/ModeAwareSignupPage.js +78 -0
  148. package/dist/components/auth/TraditionalAuthSection.d.ts +14 -0
  149. package/dist/components/auth/TraditionalAuthSection.js +20 -0
  150. package/dist/components/recovery/CompleteStep.d.ts +5 -0
  151. package/dist/components/recovery/CompleteStep.js +8 -0
  152. package/dist/components/recovery/InitiateRecoveryStep.d.ts +8 -0
  153. package/dist/components/recovery/InitiateRecoveryStep.js +20 -0
  154. package/dist/components/recovery/SelectMethodStep.d.ts +8 -0
  155. package/dist/components/recovery/SelectMethodStep.js +8 -0
  156. package/dist/components/recovery/SetPasswordStep.d.ts +6 -0
  157. package/dist/components/recovery/SetPasswordStep.js +20 -0
  158. package/dist/components/recovery/VerifyCodeStep.d.ts +10 -0
  159. package/dist/components/recovery/VerifyCodeStep.js +24 -0
  160. package/dist/components/reserved/ReservedRecoveryWarning.d.ts +38 -0
  161. package/dist/components/reserved/ReservedRecoveryWarning.js +92 -0
  162. package/dist/components/reserved/ReservedStatusBox.d.ts +30 -0
  163. package/dist/components/reserved/ReservedStatusBox.js +71 -0
  164. package/dist/components/ui/BetaBadge.d.ts +29 -0
  165. package/dist/components/ui/BetaBadge.js +38 -0
  166. package/dist/components/ui/Footer.d.ts +37 -0
  167. package/dist/components/ui/Footer.js +41 -0
  168. package/dist/config/env.d.ts +66 -0
  169. package/dist/config/env.js +57 -0
  170. package/dist/config/logger.d.ts +57 -0
  171. package/dist/config/logger.js +73 -0
  172. package/dist/config/logging-config.d.ts +30 -0
  173. package/dist/config/logging-config.js +122 -0
  174. package/dist/config/unauthenticated-routes.d.ts +17 -0
  175. package/dist/config/unauthenticated-routes.js +24 -0
  176. package/dist/config/vibe-log-transport.d.ts +79 -0
  177. package/dist/config/vibe-log-transport.js +203 -0
  178. package/dist/edge/internal-api-url.d.ts +53 -0
  179. package/dist/edge/internal-api-url.js +63 -0
  180. package/dist/edge/middleware.d.ts +14 -0
  181. package/dist/edge/middleware.js +32 -0
  182. package/dist/hooks/useAuth.d.ts +23 -0
  183. package/dist/hooks/useAuth.js +81 -0
  184. package/dist/hooks/useAuthSettings.d.ts +59 -0
  185. package/dist/hooks/useAuthSettings.js +93 -0
  186. package/dist/hooks/useAvailableProviders.d.ts +45 -0
  187. package/dist/hooks/useAvailableProviders.js +108 -0
  188. package/dist/hooks/usePasswordValidation.d.ts +27 -0
  189. package/dist/hooks/usePasswordValidation.js +102 -0
  190. package/dist/hooks/useProfile.d.ts +15 -0
  191. package/dist/hooks/useProfile.js +59 -0
  192. package/dist/hooks/usePublicAuthSettings.d.ts +56 -0
  193. package/dist/hooks/usePublicAuthSettings.js +131 -0
  194. package/dist/hooks/useSessionExpiration.d.ts +57 -0
  195. package/dist/hooks/useSessionExpiration.js +72 -0
  196. package/dist/hooks/useViabilitySession.d.ts +75 -0
  197. package/dist/hooks/useViabilitySession.js +268 -0
  198. package/dist/index.d.ts +12 -0
  199. package/dist/index.js +54 -0
  200. package/dist/lib/anon-session.d.ts +74 -0
  201. package/dist/lib/anon-session.js +169 -0
  202. package/dist/lib/api-handler.d.ts +123 -0
  203. package/dist/lib/api-handler.js +478 -0
  204. package/dist/lib/app-slug.d.ts +95 -0
  205. package/dist/lib/app-slug.js +172 -0
  206. package/dist/lib/demo-mode.d.ts +6 -0
  207. package/dist/lib/demo-mode.js +16 -0
  208. package/dist/lib/geolocation.d.ts +64 -0
  209. package/dist/lib/geolocation.js +235 -0
  210. package/dist/lib/idp-client-config.d.ts +75 -0
  211. package/dist/lib/idp-client-config.js +351 -0
  212. package/dist/lib/idp-fetch.d.ts +14 -0
  213. package/dist/lib/idp-fetch.js +91 -0
  214. package/dist/lib/internal-api.d.ts +87 -0
  215. package/dist/lib/internal-api.js +122 -0
  216. package/dist/lib/jwt-decode-client.d.ts +10 -0
  217. package/dist/lib/jwt-decode-client.js +46 -0
  218. package/dist/lib/jwt-decode.d.ts +48 -0
  219. package/dist/lib/jwt-decode.js +57 -0
  220. package/dist/lib/nextauth-secret.d.ts +10 -0
  221. package/dist/lib/nextauth-secret.js +104 -0
  222. package/dist/lib/rate-limit-service.d.ts +23 -0
  223. package/dist/lib/rate-limit-service.js +6 -0
  224. package/dist/lib/redis.d.ts +5 -0
  225. package/dist/lib/redis.js +28 -0
  226. package/dist/lib/refresh-token-validator.d.ts +13 -0
  227. package/dist/lib/refresh-token-validator.js +117 -0
  228. package/dist/lib/roles.d.ts +145 -0
  229. package/dist/lib/roles.js +168 -0
  230. package/dist/lib/secret-validation.d.ts +4 -0
  231. package/dist/lib/secret-validation.js +14 -0
  232. package/dist/lib/session-store.d.ts +166 -0
  233. package/dist/lib/session-store.js +537 -0
  234. package/dist/lib/session.d.ts +21 -0
  235. package/dist/lib/session.js +26 -0
  236. package/dist/lib/site-logger.d.ts +214 -0
  237. package/dist/lib/site-logger.js +210 -0
  238. package/dist/lib/standardized-client-api.d.ts +161 -0
  239. package/dist/lib/standardized-client-api.js +786 -0
  240. package/dist/lib/startup-init.d.ts +40 -0
  241. package/dist/lib/startup-init.js +261 -0
  242. package/dist/lib/test-aware-get-token.d.ts +2 -0
  243. package/dist/lib/test-aware-get-token.js +81 -0
  244. package/dist/lib/token-expiry.d.ts +14 -0
  245. package/dist/lib/token-expiry.js +39 -0
  246. package/dist/lib/token-lifecycle.d.ts +52 -0
  247. package/dist/lib/token-lifecycle.js +398 -0
  248. package/dist/lib/types/api-responses.d.ts +128 -0
  249. package/dist/lib/types/api-responses.js +171 -0
  250. package/dist/lib/user-agent-parser.d.ts +50 -0
  251. package/dist/lib/user-agent-parser.js +220 -0
  252. package/dist/logging/api/admin-analytics.d.ts +3 -0
  253. package/dist/logging/api/admin-analytics.js +45 -0
  254. package/dist/logging/api/audit-log.d.ts +3 -0
  255. package/dist/logging/api/audit-log.js +52 -0
  256. package/dist/logging/components/AdminAnalyticsLayout.d.ts +10 -0
  257. package/dist/logging/components/AdminAnalyticsLayout.js +11 -0
  258. package/dist/logging/components/AuditLogViewer.d.ts +7 -0
  259. package/dist/logging/components/AuditLogViewer.js +51 -0
  260. package/dist/logging/components/ErrorMetricsCard.d.ts +7 -0
  261. package/dist/logging/components/ErrorMetricsCard.js +16 -0
  262. package/dist/logging/components/HealthMetricsCard.d.ts +7 -0
  263. package/dist/logging/components/HealthMetricsCard.js +19 -0
  264. package/dist/logging/hooks/useAdminAnalytics.d.ts +24 -0
  265. package/dist/logging/hooks/useAdminAnalytics.js +22 -0
  266. package/dist/logging/hooks/useAuditLog.d.ts +6 -0
  267. package/dist/logging/hooks/useAuditLog.js +25 -0
  268. package/dist/logging/hooks/useErrorMetrics.d.ts +6 -0
  269. package/dist/logging/hooks/useErrorMetrics.js +38 -0
  270. package/dist/logging/hooks/useHealthMetrics.d.ts +6 -0
  271. package/dist/logging/hooks/useHealthMetrics.js +41 -0
  272. package/dist/logging/index.d.ts +11 -0
  273. package/dist/logging/index.js +40 -0
  274. package/dist/logging/types/analytics.d.ts +68 -0
  275. package/dist/logging/types/analytics.js +3 -0
  276. package/dist/logging/types/audit.d.ts +29 -0
  277. package/dist/logging/types/audit.js +2 -0
  278. package/dist/logging/types/index.d.ts +2 -0
  279. package/dist/logging/types/index.js +19 -0
  280. package/dist/middleware/auth-decision.d.ts +33 -0
  281. package/dist/middleware/auth-decision.js +65 -0
  282. package/dist/middleware/create-middleware.d.ts +100 -0
  283. package/dist/middleware/create-middleware.js +445 -0
  284. package/dist/middleware/rbac-check.d.ts +44 -0
  285. package/dist/middleware/rbac-check.js +191 -0
  286. package/dist/middleware/twofa-presets.d.ts +134 -0
  287. package/dist/middleware/twofa-presets.js +175 -0
  288. package/dist/models/DecodedAccessToken.d.ts +17 -0
  289. package/dist/models/DecodedAccessToken.js +2 -0
  290. package/dist/models/SessionModel.d.ts +122 -0
  291. package/dist/models/SessionModel.js +136 -0
  292. package/dist/pages/admin-login/page.d.ts +31 -0
  293. package/dist/pages/admin-login/page.js +83 -0
  294. package/dist/pages/admin-roles/RolesAdminPage.d.ts +15 -0
  295. package/dist/pages/admin-roles/RolesAdminPage.js +78 -0
  296. package/dist/pages/admin-roles/index.d.ts +8 -0
  297. package/dist/pages/admin-roles/index.js +15 -0
  298. package/dist/pages/admin-roles/modals.d.ts +72 -0
  299. package/dist/pages/admin-roles/modals.js +154 -0
  300. package/dist/pages/client-admin/ClientSiteAdminPage.d.ts +79 -0
  301. package/dist/pages/client-admin/ClientSiteAdminPage.js +177 -0
  302. package/dist/pages/client-admin/index.d.ts +32 -0
  303. package/dist/pages/client-admin/index.js +37 -0
  304. package/dist/pages/login/page.d.ts +22 -0
  305. package/dist/pages/login/page.js +239 -0
  306. package/dist/pages/profile/EnhancedProfilePage.d.ts +13 -0
  307. package/dist/pages/profile/EnhancedProfilePage.js +150 -0
  308. package/dist/pages/profile/index.d.ts +8 -0
  309. package/dist/pages/profile/index.js +16 -0
  310. package/dist/pages/profile/page.d.ts +19 -0
  311. package/dist/pages/profile/page.js +47 -0
  312. package/dist/pages/profile/profile-patch.d.ts +1 -0
  313. package/dist/pages/profile/profile-patch.js +281 -0
  314. package/dist/pages/recovery/page.d.ts +1 -0
  315. package/dist/pages/recovery/page.js +142 -0
  316. package/dist/pages/roles/MyRolesPage.d.ts +24 -0
  317. package/dist/pages/roles/MyRolesPage.js +71 -0
  318. package/dist/pages/roles/components.d.ts +63 -0
  319. package/dist/pages/roles/components.js +108 -0
  320. package/dist/pages/roles/index.d.ts +8 -0
  321. package/dist/pages/roles/index.js +19 -0
  322. package/dist/pages/security/EnhancedSecurityPage.d.ts +14 -0
  323. package/dist/pages/security/EnhancedSecurityPage.js +248 -0
  324. package/dist/pages/security/index.d.ts +8 -0
  325. package/dist/pages/security/index.js +16 -0
  326. package/dist/pages/security/page.d.ts +21 -0
  327. package/dist/pages/security/page.js +212 -0
  328. package/dist/pages/security/security-patch.d.ts +1 -0
  329. package/dist/pages/security/security-patch.js +302 -0
  330. package/dist/pages/settings/EnhancedSettingsPage.d.ts +46 -0
  331. package/dist/pages/settings/EnhancedSettingsPage.js +231 -0
  332. package/dist/pages/settings/index.d.ts +8 -0
  333. package/dist/pages/settings/index.js +16 -0
  334. package/dist/pages/settings/page.d.ts +7 -0
  335. package/dist/pages/settings/page.js +26 -0
  336. package/dist/pages/showcase/ShowcasePage.d.ts +13 -0
  337. package/dist/pages/showcase/ShowcasePage.js +140 -0
  338. package/dist/pages/showcase/index.d.ts +12 -0
  339. package/dist/pages/showcase/index.js +17 -0
  340. package/dist/pages/test-env/EmergencyLogoutPage.d.ts +14 -0
  341. package/dist/pages/test-env/EmergencyLogoutPage.js +98 -0
  342. package/dist/pages/test-env/JwtInspectPage.d.ts +14 -0
  343. package/dist/pages/test-env/JwtInspectPage.js +114 -0
  344. package/dist/pages/test-env/RefreshTokenPage.d.ts +15 -0
  345. package/dist/pages/test-env/RefreshTokenPage.js +91 -0
  346. package/dist/pages/test-env/TestEnvPage.d.ts +13 -0
  347. package/dist/pages/test-env/TestEnvPage.js +49 -0
  348. package/dist/pages/test-env/index.d.ts +24 -0
  349. package/dist/pages/test-env/index.js +32 -0
  350. package/dist/pages/verify-code/page.d.ts +30 -0
  351. package/dist/pages/verify-code/page.js +408 -0
  352. package/dist/routes/account/index.d.ts +28 -0
  353. package/dist/routes/account/index.js +71 -0
  354. package/dist/routes/account/masked-info.d.ts +33 -0
  355. package/dist/routes/account/masked-info.js +39 -0
  356. package/dist/routes/account/send-code.d.ts +37 -0
  357. package/dist/routes/account/send-code.js +42 -0
  358. package/dist/routes/account/update-phone.d.ts +13 -0
  359. package/dist/routes/account/update-phone.js +17 -0
  360. package/dist/routes/account/verify-email.d.ts +38 -0
  361. package/dist/routes/account/verify-email.js +43 -0
  362. package/dist/routes/account/verify-sms.d.ts +38 -0
  363. package/dist/routes/account/verify-sms.js +43 -0
  364. package/dist/routes/auth/index.d.ts +19 -0
  365. package/dist/routes/auth/index.js +64 -0
  366. package/dist/routes/auth/logout.d.ts +31 -0
  367. package/dist/routes/auth/logout.js +113 -0
  368. package/dist/routes/auth/nextauth.d.ts +19 -0
  369. package/dist/routes/auth/nextauth.js +72 -0
  370. package/dist/routes/auth/refresh.d.ts +30 -0
  371. package/dist/routes/auth/refresh.js +51 -0
  372. package/dist/routes/auth/session.d.ts +72 -0
  373. package/dist/routes/auth/session.js +180 -0
  374. package/dist/routes/auth/settings.d.ts +25 -0
  375. package/dist/routes/auth/settings.js +55 -0
  376. package/dist/routes/auth/viability.d.ts +52 -0
  377. package/dist/routes/auth/viability.js +201 -0
  378. package/dist/routes/index.d.ts +12 -0
  379. package/dist/routes/index.js +54 -0
  380. package/dist/routes/session/index.d.ts +6 -0
  381. package/dist/routes/session/index.js +10 -0
  382. package/dist/routes/session/refresh-viability.d.ts +16 -0
  383. package/dist/routes/session/refresh-viability.js +20 -0
  384. package/dist/services/signalrActivityService.d.ts +44 -0
  385. package/dist/services/signalrActivityService.js +257 -0
  386. package/dist/stores/authStore.d.ts +154 -0
  387. package/dist/stores/authStore.js +1531 -0
  388. package/dist/theme/ThemeProvider.d.ts +14 -0
  389. package/dist/theme/ThemeProvider.js +28 -0
  390. package/dist/theme/default.d.ts +8 -0
  391. package/dist/theme/default.js +33 -0
  392. package/dist/theme/index.d.ts +15 -0
  393. package/dist/theme/index.js +25 -0
  394. package/dist/theme/types.d.ts +56 -0
  395. package/dist/theme/types.js +8 -0
  396. package/dist/theme/useTheme.d.ts +60 -0
  397. package/dist/theme/useTheme.js +63 -0
  398. package/dist/theme/utils.d.ts +13 -0
  399. package/dist/theme/utils.js +39 -0
  400. package/dist/types/api.d.ts +134 -0
  401. package/dist/types/api.js +44 -0
  402. package/dist/types/auth.d.ts +19 -0
  403. package/dist/types/auth.js +2 -0
  404. package/dist/types/logging.d.ts +42 -0
  405. package/dist/types/logging.js +2 -0
  406. package/dist/types/recovery.d.ts +48 -0
  407. package/dist/types/recovery.js +2 -0
  408. package/dist/types/security.d.ts +1 -0
  409. package/dist/types/security.js +2 -0
  410. package/dist/utils/api.d.ts +85 -0
  411. package/dist/utils/api.js +287 -0
  412. package/dist/utils/circuitBreaker.d.ts +43 -0
  413. package/dist/utils/circuitBreaker.js +91 -0
  414. package/dist/utils/error-message.d.ts +1 -0
  415. package/dist/utils/error-message.js +103 -0
  416. package/dist/utils/layout/reservedSpace.d.ts +59 -0
  417. package/dist/utils/layout/reservedSpace.js +102 -0
  418. package/dist/utils/logout.d.ts +14 -0
  419. package/dist/utils/logout.js +32 -0
  420. package/dist/vibe/client.d.ts +261 -0
  421. package/dist/vibe/client.js +445 -0
  422. package/dist/vibe/errors.d.ts +83 -0
  423. package/dist/vibe/errors.js +146 -0
  424. package/dist/vibe/generic.d.ts +234 -0
  425. package/dist/vibe/generic.js +369 -0
  426. package/dist/vibe/hooks/index.d.ts +169 -0
  427. package/dist/vibe/hooks/index.js +252 -0
  428. package/dist/vibe/index.d.ts +23 -0
  429. package/dist/vibe/index.js +67 -0
  430. package/dist/vibe/sessions.d.ts +161 -0
  431. package/dist/vibe/sessions.js +391 -0
  432. package/dist/vibe/types.d.ts +353 -0
  433. package/dist/vibe/types.js +315 -0
  434. package/package.json +855 -0
  435. package/scripts/check-internal-url-usage.sh +73 -0
  436. package/scripts/dev-broker.ps1 +35 -0
  437. package/scripts/dev-local.ps1 +45 -0
  438. package/src/api/auth-handler.ts +550 -0
  439. package/src/api/index.ts +18 -0
  440. package/src/api-handlers/account/change-password.ts +145 -0
  441. package/src/api-handlers/account/masked-info.ts +45 -0
  442. package/src/api-handlers/account/profile.ts +80 -0
  443. package/src/api-handlers/account/recovery/initiate.ts +23 -0
  444. package/src/api-handlers/account/recovery/send-code.ts +25 -0
  445. package/src/api-handlers/account/recovery/verify-code.ts +25 -0
  446. package/src/api-handlers/account/reset-password.ts +23 -0
  447. package/src/api-handlers/account/send-code.ts +76 -0
  448. package/src/api-handlers/account/update-phone.ts +79 -0
  449. package/src/api-handlers/account/validate-password.ts +118 -0
  450. package/src/api-handlers/account/verify-email.ts +125 -0
  451. package/src/api-handlers/account/verify-sms.ts +125 -0
  452. package/src/api-handlers/admin/analytics.ts +445 -0
  453. package/src/api-handlers/admin/audit.ts +225 -0
  454. package/src/api-handlers/admin/index.ts +59 -0
  455. package/src/api-handlers/admin/redis-sessions.ts +253 -0
  456. package/src/api-handlers/admin/sessions.ts +320 -0
  457. package/src/api-handlers/admin/site-logs.ts +367 -0
  458. package/src/api-handlers/admin/users.ts +244 -0
  459. package/src/api-handlers/admin/vibe-data.ts +326 -0
  460. package/src/api-handlers/anon/preferences.ts +123 -0
  461. package/src/api-handlers/auth/jwks.ts +20 -0
  462. package/src/api-handlers/auth/login.ts +240 -0
  463. package/src/api-handlers/auth/refresh.ts +687 -0
  464. package/src/api-handlers/auth/signout.ts +212 -0
  465. package/src/api-handlers/auth/status.ts +23 -0
  466. package/src/api-handlers/auth/update-session.ts +125 -0
  467. package/src/api-handlers/auth/validate.ts +44 -0
  468. package/src/api-handlers/auth/verify-code.ts +129 -0
  469. package/src/api-handlers/session/refresh-viability.ts +36 -0
  470. package/src/api-handlers/session/viability.ts +166 -0
  471. package/src/api-handlers/test/force-expire.ts +67 -0
  472. package/src/auth/auth-decision.ts +230 -0
  473. package/src/auth/auth-options.ts +237 -0
  474. package/src/auth/callbacks/index.ts +7 -0
  475. package/src/auth/callbacks/jwt.ts +382 -0
  476. package/src/auth/callbacks/session.ts +243 -0
  477. package/src/auth/callbacks/signin.ts +56 -0
  478. package/src/auth/events/index.ts +5 -0
  479. package/src/auth/events/signout.ts +33 -0
  480. package/src/auth/providers/credentials.ts +256 -0
  481. package/src/auth/providers/index.ts +6 -0
  482. package/src/auth/providers/oauth.ts +114 -0
  483. package/src/auth/route-config.ts +220 -0
  484. package/src/auth/types/auth-types.ts +555 -0
  485. package/src/auth/types/index.ts +7 -0
  486. package/src/auth/unauthenticated-routes.ts +3 -0
  487. package/src/auth/utils/idp-client.ts +444 -0
  488. package/src/auth/utils/index.ts +6 -0
  489. package/src/auth/utils/token-utils.ts +244 -0
  490. package/src/client/AuthContext.tsx +140 -0
  491. package/src/client/fetch-with-auth.ts +48 -0
  492. package/src/client/fetchWithSession.ts +21 -0
  493. package/src/client/index.ts +13 -0
  494. package/src/client/useAnonSession.ts +131 -0
  495. package/src/components/SessionSync.tsx +137 -0
  496. package/src/components/SignalRHealthCheck.tsx +131 -0
  497. package/src/components/account/UserAvatarMenu.tsx +217 -0
  498. package/src/components/account/index.ts +8 -0
  499. package/src/components/admin/AlertSettingsTab.tsx +728 -0
  500. package/src/components/admin/AnalyticsTab.tsx +703 -0
  501. package/src/components/admin/DataBrowserTab.tsx +505 -0
  502. package/src/components/admin/LoggingSettingsTab.tsx +665 -0
  503. package/src/components/admin/SessionsTab.tsx +414 -0
  504. package/src/components/admin/StatsTab.tsx +379 -0
  505. package/src/components/admin/VibeAdminContext.tsx +87 -0
  506. package/src/components/admin/VibeAdminLayout.tsx +185 -0
  507. package/src/components/admin/index.ts +59 -0
  508. package/src/components/auth/FederatedAuthSection.tsx +95 -0
  509. package/src/components/auth/ModeAwareLoginPage.tsx +135 -0
  510. package/src/components/auth/ModeAwareSignupPage.tsx +267 -0
  511. package/src/components/auth/TraditionalAuthSection.tsx +99 -0
  512. package/src/components/recovery/CompleteStep.tsx +36 -0
  513. package/src/components/recovery/InitiateRecoveryStep.tsx +68 -0
  514. package/src/components/recovery/SelectMethodStep.tsx +73 -0
  515. package/src/components/recovery/SetPasswordStep.tsx +97 -0
  516. package/src/components/recovery/VerifyCodeStep.tsx +90 -0
  517. package/src/components/reserved/ReservedRecoveryWarning.tsx +160 -0
  518. package/src/components/reserved/ReservedStatusBox.tsx +118 -0
  519. package/src/components/ui/BetaBadge.tsx +58 -0
  520. package/src/components/ui/Footer.tsx +93 -0
  521. package/src/config/env.ts +57 -0
  522. package/src/config/logger.ts +62 -0
  523. package/src/config/logging-config.ts +82 -0
  524. package/src/config/unauthenticated-routes.ts +19 -0
  525. package/src/config/vibe-log-transport.ts +250 -0
  526. package/src/edge/internal-api-url.ts +65 -0
  527. package/src/edge/middleware.ts +42 -0
  528. package/src/hooks/useAuth.ts +115 -0
  529. package/src/hooks/useAuthSettings.ts +97 -0
  530. package/src/hooks/useAvailableProviders.ts +118 -0
  531. package/src/hooks/usePasswordValidation.ts +127 -0
  532. package/src/hooks/useProfile.ts +75 -0
  533. package/src/hooks/usePublicAuthSettings.ts +149 -0
  534. package/src/hooks/useSessionExpiration.ts +102 -0
  535. package/src/hooks/useViabilitySession.ts +335 -0
  536. package/src/index.ts +63 -0
  537. package/src/lib/anon-session.ts +213 -0
  538. package/src/lib/api-handler.ts +625 -0
  539. package/src/lib/app-slug.ts +178 -0
  540. package/src/lib/demo-mode.ts +13 -0
  541. package/src/lib/geolocation.ts +265 -0
  542. package/src/lib/idp-client-config.ts +442 -0
  543. package/src/lib/idp-fetch.ts +101 -0
  544. package/src/lib/internal-api.ts +171 -0
  545. package/src/lib/jwt-decode-client.ts +45 -0
  546. package/src/lib/jwt-decode.ts +83 -0
  547. package/src/lib/nextauth-secret.ts +126 -0
  548. package/src/lib/rate-limit-service.ts +9 -0
  549. package/src/lib/redis.ts +27 -0
  550. package/src/lib/refresh-token-validator.ts +64 -0
  551. package/src/lib/roles.ts +177 -0
  552. package/src/lib/secret-validation.ts +8 -0
  553. package/src/lib/session-store.ts +637 -0
  554. package/src/lib/session.ts +34 -0
  555. package/src/lib/site-logger.ts +245 -0
  556. package/src/lib/standardized-client-api.ts +896 -0
  557. package/src/lib/startup-init.ts +247 -0
  558. package/src/lib/test-aware-get-token.ts +30 -0
  559. package/src/lib/token-expiry.ts +40 -0
  560. package/src/lib/token-lifecycle.ts +477 -0
  561. package/src/lib/types/api-responses.ts +336 -0
  562. package/src/lib/user-agent-parser.ts +252 -0
  563. package/src/logging/api/admin-analytics.ts +51 -0
  564. package/src/logging/api/audit-log.ts +53 -0
  565. package/src/logging/components/AdminAnalyticsLayout.tsx +49 -0
  566. package/src/logging/components/AuditLogViewer.tsx +125 -0
  567. package/src/logging/components/ErrorMetricsCard.tsx +98 -0
  568. package/src/logging/components/HealthMetricsCard.tsx +70 -0
  569. package/src/logging/hooks/useAdminAnalytics.ts +22 -0
  570. package/src/logging/hooks/useAuditLog.ts +24 -0
  571. package/src/logging/hooks/useErrorMetrics.ts +40 -0
  572. package/src/logging/hooks/useHealthMetrics.ts +44 -0
  573. package/src/logging/index.ts +18 -0
  574. package/src/logging/types/analytics.ts +81 -0
  575. package/src/logging/types/audit.ts +31 -0
  576. package/src/logging/types/index.ts +3 -0
  577. package/src/middleware/auth-decision.ts +43 -0
  578. package/src/middleware/create-middleware.ts +626 -0
  579. package/src/middleware/rbac-check.ts +244 -0
  580. package/src/middleware/twofa-presets.ts +224 -0
  581. package/src/models/DecodedAccessToken.ts +17 -0
  582. package/src/models/SessionModel.ts +258 -0
  583. package/src/pages/admin-login/page.tsx +229 -0
  584. package/src/pages/admin-roles/RolesAdminPage.tsx +357 -0
  585. package/src/pages/admin-roles/index.ts +9 -0
  586. package/src/pages/admin-roles/modals.tsx +469 -0
  587. package/src/pages/client-admin/ClientSiteAdminPage.tsx +380 -0
  588. package/src/pages/client-admin/index.ts +33 -0
  589. package/src/pages/login/page.tsx +463 -0
  590. package/src/pages/profile/EnhancedProfilePage.tsx +479 -0
  591. package/src/pages/profile/index.ts +9 -0
  592. package/src/pages/profile/page.tsx +166 -0
  593. package/src/pages/recovery/page.tsx +234 -0
  594. package/src/pages/roles/MyRolesPage.tsx +211 -0
  595. package/src/pages/roles/components.tsx +294 -0
  596. package/src/pages/roles/index.ts +17 -0
  597. package/src/pages/security/EnhancedSecurityPage.tsx +574 -0
  598. package/src/pages/security/index.ts +9 -0
  599. package/src/pages/security/page.tsx +507 -0
  600. package/src/pages/settings/EnhancedSettingsPage.tsx +642 -0
  601. package/src/pages/settings/index.ts +9 -0
  602. package/src/pages/settings/page.tsx +47 -0
  603. package/src/pages/showcase/ShowcasePage.tsx +530 -0
  604. package/src/pages/showcase/index.ts +13 -0
  605. package/src/pages/test-env/EmergencyLogoutPage.tsx +179 -0
  606. package/src/pages/test-env/JwtInspectPage.tsx +418 -0
  607. package/src/pages/test-env/RefreshTokenPage.tsx +155 -0
  608. package/src/pages/test-env/TestEnvPage.tsx +116 -0
  609. package/src/pages/test-env/index.ts +25 -0
  610. package/src/pages/verify-code/page.tsx +648 -0
  611. package/src/routes/account/index.ts +32 -0
  612. package/src/routes/account/masked-info.ts +37 -0
  613. package/src/routes/account/send-code.ts +40 -0
  614. package/src/routes/account/update-phone.ts +13 -0
  615. package/src/routes/account/verify-email.ts +41 -0
  616. package/src/routes/account/verify-sms.ts +41 -0
  617. package/src/routes/auth/index.ts +23 -0
  618. package/src/routes/auth/logout.ts +127 -0
  619. package/src/routes/auth/nextauth.ts +71 -0
  620. package/src/routes/auth/refresh.ts +54 -0
  621. package/src/routes/auth/session.ts +193 -0
  622. package/src/routes/auth/settings.ts +75 -0
  623. package/src/routes/auth/viability.ts +220 -0
  624. package/src/routes/index.ts +18 -0
  625. package/src/routes/session/index.ts +7 -0
  626. package/src/routes/session/refresh-viability.ts +17 -0
  627. package/src/services/signalrActivityService.ts +258 -0
  628. package/src/stores/authStore.ts +1904 -0
  629. package/src/templates/instrumentation.ts +41 -0
  630. package/src/theme/ThemeProvider.tsx +39 -0
  631. package/src/theme/default.ts +33 -0
  632. package/src/theme/index.ts +31 -0
  633. package/src/theme/types.ts +69 -0
  634. package/src/theme/useTheme.ts +57 -0
  635. package/src/theme/utils.ts +40 -0
  636. package/src/types/api.ts +13 -0
  637. package/src/types/auth.d.ts +15 -0
  638. package/src/types/auth.ts +22 -0
  639. package/src/types/logging.ts +11 -0
  640. package/src/types/next-auth.d.ts +15 -0
  641. package/src/types/recovery.ts +54 -0
  642. package/src/types/security.ts +1 -0
  643. package/src/utils/api.ts +353 -0
  644. package/src/utils/circuitBreaker.ts +40 -0
  645. package/src/utils/error-message.ts +108 -0
  646. package/src/utils/layout/reservedSpace.ts +124 -0
  647. package/src/utils/logout.ts +30 -0
  648. package/src/vibe/client.ts +590 -0
  649. package/src/vibe/errors.ts +185 -0
  650. package/src/vibe/generic.ts +429 -0
  651. package/src/vibe/hooks/index.ts +367 -0
  652. package/src/vibe/index.ts +121 -0
  653. package/src/vibe/sessions.ts +551 -0
  654. package/src/vibe/types.ts +577 -0
@@ -0,0 +1,574 @@
1
+ /**
2
+ * Enhanced Security Page for @payez/next-mvp
3
+ *
4
+ * Implements BAPert's Member Self-Service spec with:
5
+ * - Password section with change form
6
+ * - Two-factor authentication management
7
+ * - Connected OAuth accounts
8
+ * - Active sessions list
9
+ * - Recent activity log
10
+ * - Danger zone (data export, account deletion)
11
+ *
12
+ * @see docs/specs/MEMBER_SELF_SERVICE_SPEC.md
13
+ */
14
+
15
+ 'use client';
16
+
17
+ import React, { useState, useEffect, useCallback } from 'react';
18
+ import { useProfile } from '../../hooks/useProfile';
19
+ import { usePasswordValidation } from '../../hooks/usePasswordValidation';
20
+ import { useLayout, useColors } from '../../theme/useTheme';
21
+
22
+ // Types matching BAPert's spec
23
+ interface SecurityData {
24
+ password?: {
25
+ last_changed?: string;
26
+ expires_at?: string;
27
+ days_until_expiry?: number;
28
+ requires_change?: boolean;
29
+ };
30
+ two_factor?: {
31
+ enabled: boolean;
32
+ method?: 'sms' | 'email' | 'authenticator';
33
+ method_display?: string;
34
+ last_updated?: string;
35
+ requires_reenrollment?: boolean;
36
+ backup_codes_remaining?: number;
37
+ };
38
+ connected_accounts?: Array<{
39
+ provider: string;
40
+ provider_key: string;
41
+ email?: string;
42
+ connected_at?: string;
43
+ can_disconnect?: boolean;
44
+ }>;
45
+ active_sessions?: Array<{
46
+ session_id: string;
47
+ device_type?: 'desktop' | 'mobile' | 'tablet';
48
+ device_name?: string;
49
+ ip_address?: string;
50
+ location?: string;
51
+ created_at?: string;
52
+ last_active_at?: string;
53
+ is_current?: boolean;
54
+ }>;
55
+ recent_activity?: Array<{
56
+ event_type: string;
57
+ description: string;
58
+ device?: string;
59
+ timestamp: string;
60
+ success: boolean;
61
+ }>;
62
+ }
63
+
64
+ interface PasswordData {
65
+ current_password: string;
66
+ new_password: string;
67
+ confirm_password: string;
68
+ }
69
+
70
+ // Password Strength Meter Component
71
+ function PasswordStrengthMeter({ score, failedRequirements, tip, isDark }: {
72
+ score: number;
73
+ failedRequirements: string[];
74
+ tip?: string;
75
+ isDark: boolean;
76
+ }) {
77
+ const getColor = (s: number) => {
78
+ if (s >= 4) return 'bg-green-500';
79
+ if (s >= 3) return 'bg-yellow-500';
80
+ if (s >= 2) return 'bg-orange-500';
81
+ return 'bg-red-500';
82
+ };
83
+ const getLabel = (s: number) => {
84
+ if (s >= 4) return 'Strong';
85
+ if (s >= 3) return 'Good';
86
+ if (s >= 2) return 'Fair';
87
+ if (s >= 1) return 'Weak';
88
+ return 'Very Weak';
89
+ };
90
+ const mutedText = isDark ? 'text-slate-400' : 'text-gray-500';
91
+ const barBg = isDark ? 'bg-slate-600' : 'bg-gray-200';
92
+
93
+ return (
94
+ <div className="space-y-2">
95
+ <div className="flex items-center gap-2">
96
+ <div className={`flex-1 h-2 rounded-full overflow-hidden ${barBg}`}>
97
+ <div className={`h-full transition-all duration-300 ${getColor(score)}`} style={{ width: `${Math.min(score * 20, 100)}%` }} />
98
+ </div>
99
+ <span className={`text-xs w-16 ${mutedText}`}>{getLabel(score)}</span>
100
+ </div>
101
+ {failedRequirements.length > 0 && (
102
+ <ul className={`text-xs space-y-1 ${mutedText}`}>
103
+ {failedRequirements.map((req, i) => (
104
+ <li key={i} className="flex items-center gap-1"><span className="text-red-400">x</span> {req}</li>
105
+ ))}
106
+ </ul>
107
+ )}
108
+ {tip && <p className="text-xs text-blue-400">{tip}</p>}
109
+ </div>
110
+ );
111
+ }
112
+
113
+ // Policy Checklist Component
114
+ function PolicyChecklist({ policy, password, isDark }: { policy: any; password: string; isDark: boolean }) {
115
+ const checks = React.useMemo(() => {
116
+ const list: { label: string; ok: boolean }[] = [];
117
+ const minLen = policy?.min_length || 8;
118
+ list.push({ label: `At least ${minLen} characters`, ok: (password?.length || 0) >= minLen });
119
+ if (policy?.require_uppercase) list.push({ label: 'One uppercase letter', ok: /[A-Z]/.test(password || '') });
120
+ if (policy?.require_lowercase) list.push({ label: 'One lowercase letter', ok: /[a-z]/.test(password || '') });
121
+ if (policy?.require_digit) list.push({ label: 'One digit', ok: /\d/.test(password || '') });
122
+ if (policy?.require_special) list.push({ label: 'One special character', ok: /[^A-Za-z0-9]/.test(password || '') });
123
+ return list;
124
+ }, [policy, password]);
125
+
126
+ const mutedText = isDark ? 'text-slate-400' : 'text-gray-500';
127
+ return (
128
+ <div className="space-y-1 text-xs">
129
+ {checks.map((c, i) => (
130
+ <div key={i} className="flex items-center gap-2">
131
+ {c.ok ? <span className="text-green-400">v</span> : <span className={mutedText}>o</span>}
132
+ <span className={c.ok ? 'text-green-400' : mutedText}>{c.label}</span>
133
+ </div>
134
+ ))}
135
+ </div>
136
+ );
137
+ }
138
+
139
+ // Section Card Component
140
+ function SecuritySection({ title, description, children, isDark }: {
141
+ title: string;
142
+ description?: string;
143
+ children: React.ReactNode;
144
+ isDark: boolean;
145
+ }) {
146
+ const cardBg = isDark ? 'bg-slate-800' : 'bg-white';
147
+ const borderColor = isDark ? 'border-slate-700' : 'border-gray-200';
148
+ const textPrimary = isDark ? 'text-white' : 'text-gray-900';
149
+ const textMuted = isDark ? 'text-slate-400' : 'text-gray-500';
150
+
151
+ return (
152
+ <div className={`rounded-lg border ${cardBg} ${borderColor}`}>
153
+ <div className={`px-6 py-4 border-b ${borderColor}`}>
154
+ <h2 className={`text-lg font-semibold ${textPrimary}`}>{title}</h2>
155
+ {description && <p className={`text-sm mt-1 ${textMuted}`}>{description}</p>}
156
+ </div>
157
+ <div className="px-6 py-4">{children}</div>
158
+ </div>
159
+ );
160
+ }
161
+
162
+ // Status Badge Component
163
+ function StatusBadge({ enabled, label, isDark }: { enabled: boolean; label?: string; isDark: boolean }) {
164
+ const bgClass = enabled
165
+ ? 'bg-green-100 text-green-800 dark:bg-green-900/30 dark:text-green-400'
166
+ : 'bg-yellow-100 text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-400';
167
+ return (
168
+ <span className={`inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium ${bgClass}`}>
169
+ {label || (enabled ? 'Enabled' : 'Not Active')}
170
+ </span>
171
+ );
172
+ }
173
+
174
+ function formatDate(dateString?: string): string {
175
+ if (!dateString) return '';
176
+ try {
177
+ return new Date(dateString).toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' });
178
+ } catch { return dateString; }
179
+ }
180
+
181
+ function formatRelativeTime(dateString?: string): string {
182
+ if (!dateString) return '';
183
+ try {
184
+ const date = new Date(dateString);
185
+ const now = new Date();
186
+ const diffMs = now.getTime() - date.getTime();
187
+ const diffMins = Math.floor(diffMs / 60000);
188
+ const diffHours = Math.floor(diffMs / 3600000);
189
+ const diffDays = Math.floor(diffMs / 86400000);
190
+ if (diffMins < 1) return 'Just now';
191
+ if (diffMins < 60) return `${diffMins}m ago`;
192
+ if (diffHours < 24) return `${diffHours}h ago`;
193
+ if (diffDays < 7) return `${diffDays}d ago`;
194
+ return formatDate(dateString);
195
+ } catch { return dateString; }
196
+ }
197
+
198
+ export default function EnhancedSecurityPage() {
199
+ const { data: profileData, isLoading: isProfileLoading } = useProfile();
200
+ const layout = useLayout();
201
+ const colors = useColors();
202
+
203
+ // Re-auth action types (moved for scope)
204
+ type ReAuthAction = 'enable_2fa' | 'disable_2fa' | 'export_data' | 'delete_account' | null;
205
+
206
+ // Security data state (would come from API in production)
207
+ const [securityData, setSecurityData] = useState<SecurityData | null>(null);
208
+ const [loadingSecurityData, setLoadingSecurityData] = useState(true);
209
+
210
+ // Determine dark mode
211
+ const isDark = colors?.background?.includes('slate-9') ||
212
+ colors?.background?.includes('gray-9') ||
213
+ colors?.card?.includes('slate-8');
214
+
215
+ // Password form state
216
+ const [formData, setFormData] = useState<PasswordData>({
217
+ current_password: '',
218
+ new_password: '',
219
+ confirm_password: '',
220
+ });
221
+ const [showPasswords, setShowPasswords] = useState({ current: false, new: false, confirm: false });
222
+
223
+ // Re-authentication modal state (Security Addendum)
224
+ const [reAuthModal, setReAuthModal] = useState<{
225
+ isOpen: boolean;
226
+ action: ReAuthAction;
227
+ }>({ isOpen: false, action: null });
228
+ const [submitting, setSubmitting] = useState(false);
229
+ const [error, setError] = useState('');
230
+ const [success, setSuccess] = useState('');
231
+
232
+ // Password validation
233
+ const { setPassword: validateNewPassword, isValid: newPasswordIsValid, score: newPasswordScore, failedRequirements, tip, policy } = usePasswordValidation({ debounceMs: 250 });
234
+
235
+ useEffect(() => { validateNewPassword(formData.new_password); }, [formData.new_password, validateNewPassword]);
236
+
237
+ // Fetch security data
238
+ const fetchSecurityData = useCallback(async () => {
239
+ try {
240
+ setLoadingSecurityData(true);
241
+ const res = await fetch('/api/account/security');
242
+ if (res.ok) {
243
+ const data = await res.json();
244
+ setSecurityData(data);
245
+ }
246
+ } catch (err) {
247
+ // Security data fetch is optional - page still works without it
248
+ } finally {
249
+ setLoadingSecurityData(false);
250
+ }
251
+ }, []);
252
+
253
+ useEffect(() => {
254
+ if (profileData) fetchSecurityData();
255
+ }, [profileData, fetchSecurityData]);
256
+
257
+ const passwordsMatch = formData.confirm_password.length > 0 && formData.new_password === formData.confirm_password;
258
+ const canSubmit = !submitting && newPasswordIsValid && passwordsMatch && formData.current_password.length > 0;
259
+
260
+ const handlePasswordSubmit = async (e: React.FormEvent) => {
261
+ e.preventDefault();
262
+ setError('');
263
+ setSuccess('');
264
+
265
+ if (!canSubmit) return;
266
+
267
+ try {
268
+ setSubmitting(true);
269
+ const response = await fetch('/api/account/change-password', {
270
+ method: 'POST',
271
+ headers: { 'Content-Type': 'application/json' },
272
+ body: JSON.stringify(formData),
273
+ });
274
+
275
+ const result = await response.json();
276
+ if (!response.ok || !result.success) {
277
+ let errorMsg = result.message || 'Failed to change password';
278
+ if (result.details?.value?.[0]?.message) errorMsg = result.details.value[0].message;
279
+ setError(errorMsg);
280
+ return;
281
+ }
282
+
283
+ setSuccess(result.message || 'Password changed successfully');
284
+ setFormData({ current_password: '', new_password: '', confirm_password: '' });
285
+ } catch (err: any) {
286
+ setError(err.message || 'Failed to change password');
287
+ } finally {
288
+ setSubmitting(false);
289
+ }
290
+ };
291
+
292
+ // Theme classes
293
+ const bgClass = isDark ? 'bg-slate-900' : 'bg-gray-50';
294
+ const textPrimary = isDark ? 'text-white' : 'text-gray-900';
295
+ const textSecondary = isDark ? 'text-slate-300' : 'text-gray-600';
296
+ const textMuted = isDark ? 'text-slate-400' : 'text-gray-500';
297
+ const cardBg = isDark ? 'bg-slate-800' : 'bg-white';
298
+ const borderColor = isDark ? 'border-slate-700' : 'border-gray-200';
299
+ const elevatedBg = isDark ? 'bg-slate-700' : 'bg-gray-100';
300
+ const inputBg = isDark ? 'bg-slate-800' : 'bg-white';
301
+ const inputBorder = isDark ? 'border-slate-600' : 'border-gray-300';
302
+ const inputText = isDark ? 'text-white placeholder-slate-400' : 'text-gray-900 placeholder-gray-400';
303
+
304
+ // Loading state
305
+ if (isProfileLoading) {
306
+ return (
307
+ <div className={`min-h-screen ${bgClass} flex items-center justify-center`}>
308
+ <div className="flex flex-col items-center space-y-4">
309
+ <div className="animate-spin rounded-full h-8 w-8 border-b-2 border-blue-500" />
310
+ <p className={textMuted}>Loading security settings...</p>
311
+ </div>
312
+ </div>
313
+ );
314
+ }
315
+
316
+ const twoFactorEnabled = profileData?.two_factor_enabled || securityData?.two_factor?.enabled || false;
317
+ const twoFactorMethod = securityData?.two_factor?.method_display || (twoFactorEnabled ? 'SMS' : undefined);
318
+
319
+ return (
320
+ <div className={`min-h-screen ${bgClass}`}>
321
+ <div className={`${layout?.maxWidth || 'max-w-3xl'} mx-auto ${layout?.padding || 'p-6'} space-y-6`}>
322
+ {/* Page Header */}
323
+ <div>
324
+ <h1 className={`text-2xl font-bold ${textPrimary}`}>Security</h1>
325
+ <p className={`mt-1 ${textMuted}`}>Manage your account security settings</p>
326
+ </div>
327
+
328
+ {/* Security Summary */}
329
+ <div className={`rounded-lg border ${cardBg} ${borderColor} p-6`}>
330
+ <h2 className={`text-lg font-semibold mb-4 ${textPrimary}`}>Security Summary</h2>
331
+ <div className="grid grid-cols-1 md:grid-cols-3 gap-4">
332
+ <div className={`p-4 rounded-lg ${elevatedBg}`}>
333
+ <p className={`text-sm font-medium mb-2 ${textSecondary}`}>2FA Status</p>
334
+ <StatusBadge enabled={twoFactorEnabled} isDark={isDark} />
335
+ {twoFactorMethod && <p className={`text-xs mt-2 ${textMuted}`}>Method: {twoFactorMethod}</p>}
336
+ </div>
337
+ <div className={`p-4 rounded-lg ${elevatedBg}`}>
338
+ <p className={`text-sm font-medium mb-2 ${textSecondary}`}>Email Status</p>
339
+ <StatusBadge enabled={profileData?.email_confirmed || false} label={profileData?.email_confirmed ? 'Verified' : 'Not Verified'} isDark={isDark} />
340
+ <p className={`text-xs mt-2 ${textMuted} truncate`}>{profileData?.email}</p>
341
+ </div>
342
+ <div className={`p-4 rounded-lg ${elevatedBg}`}>
343
+ <p className={`text-sm font-medium mb-2 ${textSecondary}`}>Phone Status</p>
344
+ <StatusBadge enabled={profileData?.phone_confirmed || false} label={profileData?.phone_confirmed ? 'Verified' : 'Not Verified'} isDark={isDark} />
345
+ {profileData?.phone_number && <p className={`text-xs mt-2 ${textMuted}`}>{profileData.phone_number}</p>}
346
+ </div>
347
+ </div>
348
+ </div>
349
+
350
+ {/* Password Section */}
351
+ <SecuritySection title="Password" description={securityData?.password?.last_changed ? `Last changed ${formatDate(securityData.password.last_changed)}` : undefined} isDark={isDark}>
352
+ <form onSubmit={handlePasswordSubmit} className="space-y-4">
353
+ <div>
354
+ <label className={`block text-sm font-medium mb-2 ${textSecondary}`}>Current Password</label>
355
+ <div className="relative">
356
+ <input
357
+ type={showPasswords.current ? 'text' : 'password'}
358
+ value={formData.current_password}
359
+ onChange={e => setFormData({ ...formData, current_password: e.target.value })}
360
+ className={`w-full px-3 py-2 rounded-md border focus:outline-none focus:ring-2 focus:ring-blue-500 ${inputBg} ${inputBorder} ${inputText}`}
361
+ placeholder="Enter current password"
362
+ disabled={submitting}
363
+ />
364
+ <button type="button" onClick={() => setShowPasswords(p => ({ ...p, current: !p.current }))} className={`absolute inset-y-0 right-2 flex items-center ${textMuted}`}>
365
+ {showPasswords.current ? 'Hide' : 'Show'}
366
+ </button>
367
+ </div>
368
+ </div>
369
+
370
+ <div>
371
+ <label className={`block text-sm font-medium mb-2 ${textSecondary}`}>New Password</label>
372
+ <div className="relative">
373
+ <input
374
+ type={showPasswords.new ? 'text' : 'password'}
375
+ value={formData.new_password}
376
+ onChange={e => setFormData({ ...formData, new_password: e.target.value })}
377
+ className={`w-full px-3 py-2 rounded-md border focus:outline-none focus:ring-2 focus:ring-blue-500 ${inputBg} ${inputBorder} ${inputText}`}
378
+ placeholder="Enter new password"
379
+ disabled={submitting}
380
+ />
381
+ <button type="button" onClick={() => setShowPasswords(p => ({ ...p, new: !p.new }))} className={`absolute inset-y-0 right-2 flex items-center ${textMuted}`}>
382
+ {showPasswords.new ? 'Hide' : 'Show'}
383
+ </button>
384
+ </div>
385
+ {formData.new_password && (
386
+ <div className="mt-2">
387
+ <PasswordStrengthMeter score={newPasswordScore} failedRequirements={failedRequirements} tip={tip} isDark={isDark} />
388
+ </div>
389
+ )}
390
+ {policy && formData.new_password && (
391
+ <div className="mt-3">
392
+ <PolicyChecklist policy={policy} password={formData.new_password} isDark={isDark} />
393
+ </div>
394
+ )}
395
+ </div>
396
+
397
+ <div>
398
+ <label className={`block text-sm font-medium mb-2 ${textSecondary}`}>Confirm New Password</label>
399
+ <div className="relative">
400
+ <input
401
+ type={showPasswords.confirm ? 'text' : 'password'}
402
+ value={formData.confirm_password}
403
+ onChange={e => setFormData({ ...formData, confirm_password: e.target.value })}
404
+ className={`w-full px-3 py-2 rounded-md border focus:outline-none focus:ring-2 focus:ring-blue-500 ${inputBg} ${inputBorder} ${inputText}`}
405
+ placeholder="Confirm new password"
406
+ disabled={submitting}
407
+ />
408
+ <button type="button" onClick={() => setShowPasswords(p => ({ ...p, confirm: !p.confirm }))} className={`absolute inset-y-0 right-2 flex items-center ${textMuted}`}>
409
+ {showPasswords.confirm ? 'Hide' : 'Show'}
410
+ </button>
411
+ </div>
412
+ {formData.confirm_password && !passwordsMatch && <p className="mt-1 text-xs text-red-400">Passwords do not match</p>}
413
+ {formData.confirm_password && passwordsMatch && <p className="mt-1 text-xs text-green-400">Passwords match</p>}
414
+ </div>
415
+
416
+ {error && <div className="bg-red-900/30 border border-red-600 rounded-lg p-3"><p className="text-red-400 text-sm">{error}</p></div>}
417
+ {success && <div className="bg-green-900/30 border border-green-600 rounded-lg p-3"><p className="text-green-400 text-sm">{success}</p></div>}
418
+
419
+ <button
420
+ type="submit"
421
+ disabled={!canSubmit}
422
+ className="px-4 py-2 bg-blue-600 text-white rounded-md hover:bg-blue-700 disabled:opacity-50 disabled:cursor-not-allowed"
423
+ >
424
+ {submitting ? 'Changing...' : 'Change Password'}
425
+ </button>
426
+ </form>
427
+ </SecuritySection>
428
+
429
+ {/* Two-Factor Authentication */}
430
+ <SecuritySection title="Two-Factor Authentication" description="Add an extra layer of security to your account" isDark={isDark}>
431
+ <div className="space-y-4">
432
+ <div className="flex items-center justify-between">
433
+ <div>
434
+ <p className={`font-medium ${textPrimary}`}>Status</p>
435
+ <p className={`text-sm ${textMuted}`}>{twoFactorEnabled ? `Enabled via ${twoFactorMethod || 'SMS'}` : 'Not enabled'}</p>
436
+ </div>
437
+ <StatusBadge enabled={twoFactorEnabled} isDark={isDark} />
438
+ </div>
439
+ {twoFactorEnabled ? (
440
+ <div className="flex gap-2">
441
+ <button className={`px-3 py-1.5 text-sm rounded border ${isDark ? 'border-slate-600 hover:bg-slate-700' : 'border-gray-300 hover:bg-gray-50'} ${textSecondary}`}>
442
+ Change Method
443
+ </button>
444
+ <button className="px-3 py-1.5 text-sm rounded border border-red-500 text-red-500 hover:bg-red-500/10">
445
+ Disable 2FA
446
+ </button>
447
+ </div>
448
+ ) : (
449
+ <button className="px-4 py-2 bg-blue-600 text-white rounded-md hover:bg-blue-700">
450
+ Enable 2FA
451
+ </button>
452
+ )}
453
+ {securityData?.two_factor?.backup_codes_remaining !== undefined && (
454
+ <div className={`mt-4 p-3 rounded-lg ${elevatedBg}`}>
455
+ <p className={`text-sm ${textSecondary}`}>Backup Codes</p>
456
+ <p className={textMuted}>You have {securityData.two_factor.backup_codes_remaining} of 10 backup codes remaining</p>
457
+ <button className={`mt-2 text-sm ${isDark ? 'text-blue-400' : 'text-blue-600'}`}>View Codes</button>
458
+ </div>
459
+ )}
460
+ </div>
461
+ </SecuritySection>
462
+
463
+ {/* Connected Accounts */}
464
+ <SecuritySection title="Connected Accounts" description="OAuth providers linked to your account" isDark={isDark}>
465
+ {securityData?.connected_accounts && securityData.connected_accounts.length > 0 ? (
466
+ <div className="space-y-3">
467
+ {securityData.connected_accounts.map((account, idx) => (
468
+ <div key={idx} className={`flex items-center justify-between p-3 rounded-lg ${elevatedBg}`}>
469
+ <div>
470
+ <p className={`font-medium ${textPrimary}`}>{account.provider}</p>
471
+ {account.email && <p className={`text-sm ${textMuted}`}>{account.email}</p>}
472
+ {account.connected_at && <p className={`text-xs ${textMuted}`}>Connected {formatDate(account.connected_at)}</p>}
473
+ </div>
474
+ {account.can_disconnect && (
475
+ <button className="text-sm text-red-500 hover:text-red-400">Disconnect</button>
476
+ )}
477
+ </div>
478
+ ))}
479
+ </div>
480
+ ) : (
481
+ <p className={textMuted}>No connected accounts</p>
482
+ )}
483
+ <button className={`mt-4 text-sm ${isDark ? 'text-blue-400' : 'text-blue-600'}`}>
484
+ Connect Another Account
485
+ </button>
486
+ </SecuritySection>
487
+
488
+ {/* Active Sessions */}
489
+ <SecuritySection title="Active Sessions" description="Devices currently logged into your account" isDark={isDark}>
490
+ {securityData?.active_sessions && securityData.active_sessions.length > 0 ? (
491
+ <div className="space-y-3">
492
+ {securityData.active_sessions.map((session, idx) => (
493
+ <div key={idx} className={`flex items-center justify-between p-3 rounded-lg ${elevatedBg}`}>
494
+ <div>
495
+ <div className="flex items-center gap-2">
496
+ <p className={`font-medium ${textPrimary}`}>{session.device_name || 'Unknown Device'}</p>
497
+ {session.is_current && <span className="px-2 py-0.5 rounded text-xs bg-blue-500 text-white">This Device</span>}
498
+ </div>
499
+ <p className={`text-sm ${textMuted}`}>{session.location || session.ip_address}</p>
500
+ <p className={`text-xs ${textMuted}`}>{session.is_current ? 'Active now' : `Last active ${formatRelativeTime(session.last_active_at)}`}</p>
501
+ </div>
502
+ {!session.is_current && (
503
+ <button className="text-sm text-red-500 hover:text-red-400">Revoke</button>
504
+ )}
505
+ </div>
506
+ ))}
507
+ </div>
508
+ ) : (
509
+ <p className={textMuted}>Session information not available</p>
510
+ )}
511
+ <button className="mt-4 text-sm text-red-500 hover:text-red-400">
512
+ Sign Out All Other Devices
513
+ </button>
514
+ </SecuritySection>
515
+
516
+ {/* Recent Activity */}
517
+ <SecuritySection title="Recent Activity" description="Security events on your account" isDark={isDark}>
518
+ {securityData?.recent_activity && securityData.recent_activity.length > 0 ? (
519
+ <div className="space-y-2">
520
+ {securityData.recent_activity.slice(0, 5).map((activity, idx) => (
521
+ <div key={idx} className={`flex items-center justify-between py-2 border-b last:border-b-0 ${borderColor}`}>
522
+ <div className="flex items-center gap-2">
523
+ <span className={activity.success ? 'text-green-400' : 'text-red-400'}>{activity.success ? 'v' : 'x'}</span>
524
+ <div>
525
+ <p className={`text-sm ${textPrimary}`}>{activity.description}</p>
526
+ {activity.device && <p className={`text-xs ${textMuted}`}>{activity.device}</p>}
527
+ </div>
528
+ </div>
529
+ <p className={`text-xs ${textMuted}`}>{formatRelativeTime(activity.timestamp)}</p>
530
+ </div>
531
+ ))}
532
+ </div>
533
+ ) : (
534
+ <p className={textMuted}>No recent activity</p>
535
+ )}
536
+ <button className={`mt-4 text-sm ${isDark ? 'text-blue-400' : 'text-blue-600'}`}>
537
+ View Full History
538
+ </button>
539
+ </SecuritySection>
540
+
541
+ {/* Danger Zone */}
542
+ <SecuritySection title="Danger Zone" description="Irreversible actions" isDark={isDark}>
543
+ <div className="space-y-4">
544
+ <div className={`flex items-center justify-between p-4 rounded-lg border border-yellow-500/50`}>
545
+ <div>
546
+ <p className={`font-medium ${textPrimary}`}>Download My Data</p>
547
+ <p className={`text-sm ${textMuted}`}>Get a copy of all your personal data</p>
548
+ </div>
549
+ <button className="px-4 py-2 text-sm rounded border border-yellow-500 text-yellow-500 hover:bg-yellow-500/10">
550
+ Request Export
551
+ </button>
552
+ </div>
553
+ <div className={`flex items-center justify-between p-4 rounded-lg border border-red-500/50`}>
554
+ <div>
555
+ <p className={`font-medium ${textPrimary}`}>Delete Account</p>
556
+ <p className={`text-sm ${textMuted}`}>Permanently delete your account and all data</p>
557
+ </div>
558
+ <button className="px-4 py-2 text-sm rounded border border-red-500 text-red-500 hover:bg-red-500/10">
559
+ Delete Account
560
+ </button>
561
+ </div>
562
+ </div>
563
+ </SecuritySection>
564
+
565
+ {/* Back to Profile link */}
566
+ <div className="text-center">
567
+ <a href="/account/profile" className={`text-sm hover:underline ${textMuted}`}>
568
+ Back to Profile
569
+ </a>
570
+ </div>
571
+ </div>
572
+ </div>
573
+ );
574
+ }
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Security Page exports
3
+ *
4
+ * - SecurityPage: Basic security display (legacy)
5
+ * - EnhancedSecurityPage: Full-featured security with password, 2FA, sessions, activity, danger zone
6
+ */
7
+
8
+ export { default as SecurityPage } from './page';
9
+ export { default as EnhancedSecurityPage } from './EnhancedSecurityPage';