hazo_auth 1.4.2 → 1.6.1

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 (354) hide show
  1. package/README.md +65 -19
  2. package/SETUP_CHECKLIST.md +779 -0
  3. package/dist/app/api/hazo_auth/change_password/route.d.ts +8 -0
  4. package/dist/app/api/hazo_auth/change_password/route.d.ts.map +1 -0
  5. package/dist/app/api/hazo_auth/change_password/route.js +98 -0
  6. package/dist/app/api/hazo_auth/forgot_password/route.d.ts +8 -0
  7. package/dist/app/api/hazo_auth/forgot_password/route.d.ts.map +1 -0
  8. package/dist/app/api/hazo_auth/forgot_password/route.js +78 -0
  9. package/dist/app/api/hazo_auth/get_auth/route.d.ts +10 -0
  10. package/dist/app/api/hazo_auth/get_auth/route.d.ts.map +1 -0
  11. package/dist/app/api/hazo_auth/get_auth/route.js +63 -0
  12. package/dist/app/api/hazo_auth/invalidate_cache/route.d.ts +14 -0
  13. package/dist/app/api/hazo_auth/invalidate_cache/route.d.ts.map +1 -0
  14. package/dist/app/api/hazo_auth/invalidate_cache/route.js +96 -0
  15. package/dist/app/api/hazo_auth/library_photo/[category]/[filename]/route.d.ts +9 -0
  16. package/dist/app/api/hazo_auth/library_photo/[category]/[filename]/route.d.ts.map +1 -0
  17. package/dist/app/api/hazo_auth/library_photo/[category]/[filename]/route.js +82 -0
  18. package/dist/app/api/hazo_auth/library_photos/route.d.ts +22 -0
  19. package/dist/app/api/hazo_auth/library_photos/route.d.ts.map +1 -0
  20. package/dist/app/api/hazo_auth/library_photos/route.js +80 -0
  21. package/dist/app/api/hazo_auth/login/route.d.ts +12 -0
  22. package/dist/app/api/hazo_auth/login/route.d.ts.map +1 -0
  23. package/dist/app/api/hazo_auth/login/route.js +140 -0
  24. package/dist/app/api/hazo_auth/logout/route.d.ts +8 -0
  25. package/dist/app/api/hazo_auth/logout/route.d.ts.map +1 -0
  26. package/dist/app/api/hazo_auth/logout/route.js +71 -0
  27. package/dist/app/api/hazo_auth/me/route.d.ts +3 -0
  28. package/dist/app/api/hazo_auth/me/route.d.ts.map +1 -0
  29. package/dist/app/api/hazo_auth/me/route.js +34 -0
  30. package/dist/app/api/hazo_auth/profile_picture/[filename]/route.d.ts +7 -0
  31. package/dist/app/api/hazo_auth/profile_picture/[filename]/route.d.ts.map +1 -0
  32. package/dist/app/api/hazo_auth/profile_picture/[filename]/route.js +43 -0
  33. package/dist/app/api/hazo_auth/register/route.d.ts +9 -0
  34. package/dist/app/api/hazo_auth/register/route.d.ts.map +1 -0
  35. package/dist/app/api/hazo_auth/register/route.js +80 -0
  36. package/dist/app/api/hazo_auth/remove_profile_picture/route.d.ts +8 -0
  37. package/dist/app/api/hazo_auth/remove_profile_picture/route.d.ts.map +1 -0
  38. package/dist/app/api/hazo_auth/remove_profile_picture/route.js +64 -0
  39. package/dist/app/api/hazo_auth/resend_verification/route.d.ts +8 -0
  40. package/dist/app/api/hazo_auth/resend_verification/route.d.ts.map +1 -0
  41. package/dist/app/api/hazo_auth/resend_verification/route.js +79 -0
  42. package/dist/app/api/hazo_auth/reset_password/route.d.ts +8 -0
  43. package/dist/app/api/hazo_auth/reset_password/route.d.ts.map +1 -0
  44. package/dist/app/api/hazo_auth/reset_password/route.js +76 -0
  45. package/dist/app/api/hazo_auth/update_user/route.d.ts +9 -0
  46. package/dist/app/api/hazo_auth/update_user/route.d.ts.map +1 -0
  47. package/dist/app/api/hazo_auth/update_user/route.js +95 -0
  48. package/dist/app/api/hazo_auth/upload_profile_picture/route.d.ts +9 -0
  49. package/dist/app/api/hazo_auth/upload_profile_picture/route.d.ts.map +1 -0
  50. package/dist/app/api/hazo_auth/upload_profile_picture/route.js +204 -0
  51. package/dist/app/api/hazo_auth/validate_reset_token/route.d.ts +6 -0
  52. package/dist/app/api/hazo_auth/validate_reset_token/route.d.ts.map +1 -0
  53. package/dist/app/api/hazo_auth/validate_reset_token/route.js +58 -0
  54. package/dist/app/api/hazo_auth/verify_email/route.d.ts +11 -0
  55. package/dist/app/api/hazo_auth/verify_email/route.d.ts.map +1 -0
  56. package/dist/app/api/hazo_auth/verify_email/route.js +63 -0
  57. package/dist/cli/generate.d.ts +7 -0
  58. package/dist/cli/generate.d.ts.map +1 -0
  59. package/dist/cli/generate.js +184 -0
  60. package/dist/cli/index.d.ts +3 -0
  61. package/dist/cli/index.d.ts.map +1 -0
  62. package/dist/cli/index.js +173 -0
  63. package/dist/cli/init.d.ts +2 -0
  64. package/dist/cli/init.d.ts.map +1 -0
  65. package/dist/cli/init.js +201 -0
  66. package/dist/cli/validate.d.ts +15 -0
  67. package/dist/cli/validate.d.ts.map +1 -0
  68. package/dist/cli/validate.js +509 -0
  69. package/dist/components/ui/card.d.ts +9 -0
  70. package/dist/components/ui/card.d.ts.map +1 -0
  71. package/dist/components/ui/card.js +45 -0
  72. package/dist/hooks/use-mobile.d.ts.map +1 -1
  73. package/dist/hooks/use-mobile.js +17 -3
  74. package/dist/lib/services/profile_picture_service.d.ts +34 -2
  75. package/dist/lib/services/profile_picture_service.d.ts.map +1 -1
  76. package/dist/lib/services/profile_picture_service.js +157 -15
  77. package/dist/page_components/forgot_password.d.ts +19 -0
  78. package/dist/page_components/forgot_password.d.ts.map +1 -0
  79. package/dist/page_components/forgot_password.js +36 -0
  80. package/dist/page_components/index.d.ts +7 -0
  81. package/dist/page_components/index.d.ts.map +1 -0
  82. package/dist/page_components/index.js +9 -0
  83. package/dist/page_components/login.d.ts +26 -0
  84. package/dist/page_components/login.d.ts.map +1 -0
  85. package/dist/page_components/login.js +40 -0
  86. package/dist/page_components/my_settings.d.ts +64 -0
  87. package/dist/page_components/my_settings.d.ts.map +1 -0
  88. package/dist/page_components/my_settings.js +67 -0
  89. package/dist/page_components/register.d.ts +25 -0
  90. package/dist/page_components/register.d.ts.map +1 -0
  91. package/dist/page_components/register.js +43 -0
  92. package/dist/page_components/reset_password.d.ts +25 -0
  93. package/dist/page_components/reset_password.d.ts.map +1 -0
  94. package/dist/page_components/reset_password.js +43 -0
  95. package/dist/page_components/verify_email.d.ts +21 -0
  96. package/dist/page_components/verify_email.d.ts.map +1 -0
  97. package/dist/page_components/verify_email.js +36 -0
  98. package/dist/server/routes/change_password.d.ts +2 -0
  99. package/dist/server/routes/change_password.d.ts.map +1 -0
  100. package/dist/server/routes/change_password.js +2 -0
  101. package/dist/server/routes/forgot_password.d.ts +2 -0
  102. package/dist/server/routes/forgot_password.d.ts.map +1 -0
  103. package/dist/server/routes/forgot_password.js +2 -0
  104. package/dist/server/routes/get_auth.d.ts +2 -0
  105. package/dist/server/routes/get_auth.d.ts.map +1 -0
  106. package/dist/server/routes/get_auth.js +2 -0
  107. package/dist/server/routes/index.d.ts +19 -0
  108. package/dist/server/routes/index.d.ts.map +1 -0
  109. package/dist/server/routes/index.js +25 -0
  110. package/dist/server/routes/invalidate_cache.d.ts +2 -0
  111. package/dist/server/routes/invalidate_cache.d.ts.map +1 -0
  112. package/dist/server/routes/invalidate_cache.js +2 -0
  113. package/dist/server/routes/library_photo.d.ts +2 -0
  114. package/dist/server/routes/library_photo.d.ts.map +1 -0
  115. package/dist/server/routes/library_photo.js +3 -0
  116. package/dist/server/routes/library_photos.d.ts +2 -0
  117. package/dist/server/routes/library_photos.d.ts.map +1 -0
  118. package/dist/server/routes/library_photos.js +2 -0
  119. package/dist/server/routes/login.d.ts +2 -0
  120. package/dist/server/routes/login.d.ts.map +1 -0
  121. package/dist/server/routes/login.js +2 -0
  122. package/dist/server/routes/logout.d.ts +2 -0
  123. package/dist/server/routes/logout.d.ts.map +1 -0
  124. package/dist/server/routes/logout.js +2 -0
  125. package/dist/server/routes/me.d.ts +2 -0
  126. package/dist/server/routes/me.d.ts.map +1 -0
  127. package/dist/server/routes/me.js +2 -0
  128. package/dist/server/routes/profile_picture_filename.d.ts +2 -0
  129. package/dist/server/routes/profile_picture_filename.d.ts.map +1 -0
  130. package/dist/server/routes/profile_picture_filename.js +3 -0
  131. package/dist/server/routes/register.d.ts +2 -0
  132. package/dist/server/routes/register.d.ts.map +1 -0
  133. package/dist/server/routes/register.js +2 -0
  134. package/dist/server/routes/remove_profile_picture.d.ts +2 -0
  135. package/dist/server/routes/remove_profile_picture.d.ts.map +1 -0
  136. package/dist/server/routes/remove_profile_picture.js +2 -0
  137. package/dist/server/routes/resend_verification.d.ts +2 -0
  138. package/dist/server/routes/resend_verification.d.ts.map +1 -0
  139. package/dist/server/routes/resend_verification.js +2 -0
  140. package/dist/server/routes/reset_password.d.ts +2 -0
  141. package/dist/server/routes/reset_password.d.ts.map +1 -0
  142. package/dist/server/routes/reset_password.js +2 -0
  143. package/dist/server/routes/update_user.d.ts +2 -0
  144. package/dist/server/routes/update_user.d.ts.map +1 -0
  145. package/dist/server/routes/update_user.js +2 -0
  146. package/dist/server/routes/upload_profile_picture.d.ts +2 -0
  147. package/dist/server/routes/upload_profile_picture.d.ts.map +1 -0
  148. package/dist/server/routes/upload_profile_picture.js +2 -0
  149. package/dist/server/routes/validate_reset_token.d.ts +2 -0
  150. package/dist/server/routes/validate_reset_token.d.ts.map +1 -0
  151. package/dist/server/routes/validate_reset_token.js +2 -0
  152. package/dist/server/routes/verify_email.d.ts +2 -0
  153. package/dist/server/routes/verify_email.d.ts.map +1 -0
  154. package/dist/server/routes/verify_email.js +2 -0
  155. package/package.json +40 -17
  156. package/components.json +0 -22
  157. package/instrumentation.ts +0 -32
  158. package/migrations/001_add_token_type_to_refresh_tokens.sql +0 -14
  159. package/migrations/002_add_name_to_hazo_users.sql +0 -7
  160. package/migrations/003_add_url_on_logon_to_hazo_users.sql +0 -8
  161. package/next.config.mjs +0 -67
  162. package/postcss.config.mjs +0 -8
  163. package/public/file.svg +0 -1
  164. package/public/globe.svg +0 -1
  165. package/public/next.svg +0 -1
  166. package/public/vercel.svg +0 -1
  167. package/public/window.svg +0 -1
  168. package/scripts/apply_migration.ts +0 -118
  169. package/scripts/init_users.ts +0 -378
  170. package/src/app/api/hazo_auth/auth/upload_profile_picture/route.ts +0 -268
  171. package/src/app/api/hazo_auth/change_password/route.ts +0 -132
  172. package/src/app/api/hazo_auth/forgot_password/route.ts +0 -107
  173. package/src/app/api/hazo_auth/get_auth/route.ts +0 -89
  174. package/src/app/api/hazo_auth/invalidate_cache/route.ts +0 -139
  175. package/src/app/api/hazo_auth/library_photos/route.ts +0 -73
  176. package/src/app/api/hazo_auth/login/route.ts +0 -181
  177. package/src/app/api/hazo_auth/logout/route.ts +0 -89
  178. package/src/app/api/hazo_auth/me/route.ts +0 -47
  179. package/src/app/api/hazo_auth/profile_picture/[filename]/route.ts +0 -67
  180. package/src/app/api/hazo_auth/register/route.ts +0 -109
  181. package/src/app/api/hazo_auth/remove_profile_picture/route.ts +0 -86
  182. package/src/app/api/hazo_auth/resend_verification/route.ts +0 -108
  183. package/src/app/api/hazo_auth/reset_password/route.ts +0 -107
  184. package/src/app/api/hazo_auth/update_user/route.ts +0 -126
  185. package/src/app/api/hazo_auth/upload_profile_picture/route.ts +0 -268
  186. package/src/app/api/hazo_auth/user_management/permissions/route.ts +0 -367
  187. package/src/app/api/hazo_auth/user_management/roles/route.ts +0 -442
  188. package/src/app/api/hazo_auth/user_management/users/roles/route.ts +0 -367
  189. package/src/app/api/hazo_auth/user_management/users/route.ts +0 -239
  190. package/src/app/api/hazo_auth/validate_reset_token/route.ts +0 -83
  191. package/src/app/api/hazo_auth/verify_email/route.ts +0 -88
  192. package/src/app/api/migrations/apply/route.ts +0 -91
  193. package/src/app/favicon.ico +0 -0
  194. package/src/app/fonts/GeistMonoVF.woff +0 -0
  195. package/src/app/fonts/GeistVF.woff +0 -0
  196. package/src/app/globals.css +0 -89
  197. package/src/app/hazo_auth/forgot_password/forgot_password_page_client.tsx +0 -60
  198. package/src/app/hazo_auth/forgot_password/page.tsx +0 -24
  199. package/src/app/hazo_auth/login/login_page_client.tsx +0 -86
  200. package/src/app/hazo_auth/login/page.tsx +0 -38
  201. package/src/app/hazo_auth/my_settings/my_settings_page_client.tsx +0 -120
  202. package/src/app/hazo_auth/my_settings/page.tsx +0 -40
  203. package/src/app/hazo_auth/register/page.tsx +0 -36
  204. package/src/app/hazo_auth/register/register_page_client.tsx +0 -81
  205. package/src/app/hazo_auth/reset_password/page.tsx +0 -29
  206. package/src/app/hazo_auth/reset_password/reset_password_page_client.tsx +0 -81
  207. package/src/app/hazo_auth/user_management/page.tsx +0 -14
  208. package/src/app/hazo_auth/user_management/user_management_page_client.tsx +0 -16
  209. package/src/app/hazo_auth/verify_email/page.tsx +0 -24
  210. package/src/app/hazo_auth/verify_email/verify_email_page_client.tsx +0 -60
  211. package/src/app/hazo_connect/api/sqlite/data/route.ts +0 -203
  212. package/src/app/hazo_connect/api/sqlite/schema/route.ts +0 -45
  213. package/src/app/hazo_connect/api/sqlite/tables/route.ts +0 -36
  214. package/src/app/hazo_connect/sqlite_admin/page.tsx +0 -51
  215. package/src/app/hazo_connect/sqlite_admin/sqlite-admin-client.tsx +0 -984
  216. package/src/app/layout.tsx +0 -43
  217. package/src/app/page.tsx +0 -170
  218. package/src/components/index.ts +0 -7
  219. package/src/components/layouts/email_verification/config/email_verification_field_config.ts +0 -86
  220. package/src/components/layouts/email_verification/hooks/use_email_verification.ts +0 -297
  221. package/src/components/layouts/email_verification/index.tsx +0 -297
  222. package/src/components/layouts/forgot_password/config/forgot_password_field_config.ts +0 -58
  223. package/src/components/layouts/forgot_password/hooks/use_forgot_password_form.ts +0 -179
  224. package/src/components/layouts/forgot_password/index.tsx +0 -168
  225. package/src/components/layouts/index.ts +0 -26
  226. package/src/components/layouts/login/config/login_field_config.ts +0 -67
  227. package/src/components/layouts/login/hooks/use_login_form.ts +0 -286
  228. package/src/components/layouts/login/index.tsx +0 -252
  229. package/src/components/layouts/my_settings/components/editable_field.tsx +0 -177
  230. package/src/components/layouts/my_settings/components/password_change_dialog.tsx +0 -301
  231. package/src/components/layouts/my_settings/components/profile_picture_dialog.tsx +0 -385
  232. package/src/components/layouts/my_settings/components/profile_picture_display.tsx +0 -66
  233. package/src/components/layouts/my_settings/components/profile_picture_gravatar_tab.tsx +0 -143
  234. package/src/components/layouts/my_settings/components/profile_picture_library_tab.tsx +0 -311
  235. package/src/components/layouts/my_settings/components/profile_picture_upload_tab.tsx +0 -341
  236. package/src/components/layouts/my_settings/config/my_settings_field_config.ts +0 -61
  237. package/src/components/layouts/my_settings/hooks/use_my_settings.ts +0 -458
  238. package/src/components/layouts/my_settings/index.tsx +0 -351
  239. package/src/components/layouts/register/config/register_field_config.ts +0 -101
  240. package/src/components/layouts/register/hooks/use_register_form.ts +0 -275
  241. package/src/components/layouts/register/index.tsx +0 -226
  242. package/src/components/layouts/reset_password/config/reset_password_field_config.ts +0 -86
  243. package/src/components/layouts/reset_password/hooks/use_reset_password_form.ts +0 -276
  244. package/src/components/layouts/reset_password/index.tsx +0 -294
  245. package/src/components/layouts/shared/components/already_logged_in_guard.tsx +0 -95
  246. package/src/components/layouts/shared/components/auth_page_shell.tsx +0 -36
  247. package/src/components/layouts/shared/components/field_error_message.tsx +0 -29
  248. package/src/components/layouts/shared/components/form_action_buttons.tsx +0 -64
  249. package/src/components/layouts/shared/components/form_field_wrapper.tsx +0 -44
  250. package/src/components/layouts/shared/components/form_header.tsx +0 -36
  251. package/src/components/layouts/shared/components/logout_button.tsx +0 -76
  252. package/src/components/layouts/shared/components/password_field.tsx +0 -72
  253. package/src/components/layouts/shared/components/profile_pic_menu.tsx +0 -321
  254. package/src/components/layouts/shared/components/profile_pic_menu_wrapper.tsx +0 -40
  255. package/src/components/layouts/shared/components/sidebar_layout_wrapper.tsx +0 -214
  256. package/src/components/layouts/shared/components/standalone_layout_wrapper.tsx +0 -53
  257. package/src/components/layouts/shared/components/two_column_auth_layout.tsx +0 -44
  258. package/src/components/layouts/shared/components/unauthorized_guard.tsx +0 -78
  259. package/src/components/layouts/shared/components/visual_panel.tsx +0 -41
  260. package/src/components/layouts/shared/config/layout_customization.ts +0 -95
  261. package/src/components/layouts/shared/data/layout_data_client.ts +0 -19
  262. package/src/components/layouts/shared/hooks/use_auth_status.ts +0 -103
  263. package/src/components/layouts/shared/hooks/use_hazo_auth.ts +0 -158
  264. package/src/components/layouts/shared/index.ts +0 -34
  265. package/src/components/layouts/shared/utils/ip_address.ts +0 -37
  266. package/src/components/layouts/shared/utils/validation.ts +0 -66
  267. package/src/components/layouts/user_management/components/roles_matrix.tsx +0 -607
  268. package/src/components/layouts/user_management/index.tsx +0 -1295
  269. package/src/components/ui/alert-dialog.tsx +0 -141
  270. package/src/components/ui/avatar.tsx +0 -50
  271. package/src/components/ui/button.tsx +0 -57
  272. package/src/components/ui/checkbox.tsx +0 -30
  273. package/src/components/ui/dialog.tsx +0 -122
  274. package/src/components/ui/dropdown-menu.tsx +0 -201
  275. package/src/components/ui/hazo_ui_tooltip.tsx +0 -67
  276. package/src/components/ui/index.ts +0 -22
  277. package/src/components/ui/input.tsx +0 -22
  278. package/src/components/ui/label.tsx +0 -26
  279. package/src/components/ui/separator.tsx +0 -31
  280. package/src/components/ui/sheet.tsx +0 -139
  281. package/src/components/ui/sidebar.tsx +0 -773
  282. package/src/components/ui/skeleton.tsx +0 -15
  283. package/src/components/ui/sonner.tsx +0 -31
  284. package/src/components/ui/switch.tsx +0 -29
  285. package/src/components/ui/table.tsx +0 -120
  286. package/src/components/ui/tabs.tsx +0 -55
  287. package/src/components/ui/tooltip.tsx +0 -32
  288. package/src/components/ui/vertical-tabs.tsx +0 -59
  289. package/src/hooks/use-mobile.tsx +0 -19
  290. package/src/index.ts +0 -7
  291. package/src/lib/already_logged_in_config.server.ts +0 -46
  292. package/src/lib/app_logger.ts +0 -24
  293. package/src/lib/auth/auth_cache.ts +0 -220
  294. package/src/lib/auth/auth_rate_limiter.ts +0 -121
  295. package/src/lib/auth/auth_types.ts +0 -65
  296. package/src/lib/auth/auth_utils.server.ts +0 -196
  297. package/src/lib/auth/hazo_get_auth.server.ts +0 -333
  298. package/src/lib/auth/index.ts +0 -23
  299. package/src/lib/auth/server_auth.ts +0 -88
  300. package/src/lib/auth_utility_config.server.ts +0 -136
  301. package/src/lib/config/config_loader.server.ts +0 -164
  302. package/src/lib/email_verification_config.server.ts +0 -32
  303. package/src/lib/file_types_config.server.ts +0 -25
  304. package/src/lib/forgot_password_config.server.ts +0 -32
  305. package/src/lib/hazo_connect_instance.server.ts +0 -101
  306. package/src/lib/hazo_connect_setup.server.ts +0 -194
  307. package/src/lib/hazo_connect_setup.ts +0 -54
  308. package/src/lib/index.ts +0 -44
  309. package/src/lib/login_config.server.ts +0 -71
  310. package/src/lib/messages_config.server.ts +0 -45
  311. package/src/lib/migrations/apply_migration.ts +0 -105
  312. package/src/lib/my_settings_config.server.ts +0 -135
  313. package/src/lib/password_requirements_config.server.ts +0 -39
  314. package/src/lib/profile_pic_menu_config.server.ts +0 -138
  315. package/src/lib/profile_picture_config.server.ts +0 -56
  316. package/src/lib/register_config.server.ts +0 -73
  317. package/src/lib/reset_password_config.server.ts +0 -75
  318. package/src/lib/services/email_service.ts +0 -581
  319. package/src/lib/services/email_verification_service.ts +0 -270
  320. package/src/lib/services/index.ts +0 -15
  321. package/src/lib/services/login_service.ts +0 -134
  322. package/src/lib/services/password_change_service.ts +0 -154
  323. package/src/lib/services/password_reset_service.ts +0 -405
  324. package/src/lib/services/profile_picture_remove_service.ts +0 -120
  325. package/src/lib/services/profile_picture_service.ts +0 -215
  326. package/src/lib/services/profile_picture_source_mapper.ts +0 -62
  327. package/src/lib/services/registration_service.ts +0 -184
  328. package/src/lib/services/token_service.ts +0 -240
  329. package/src/lib/services/user_profiles_service.ts +0 -143
  330. package/src/lib/services/user_update_service.ts +0 -141
  331. package/src/lib/ui_shell_config.server.ts +0 -73
  332. package/src/lib/ui_sizes_config.server.ts +0 -37
  333. package/src/lib/user_fields_config.server.ts +0 -31
  334. package/src/lib/user_management_config.server.ts +0 -39
  335. package/src/lib/utils/api_route_helpers.ts +0 -60
  336. package/src/lib/utils/error_sanitizer.ts +0 -75
  337. package/src/lib/utils.ts +0 -11
  338. package/src/middleware.ts +0 -94
  339. package/src/routes/index.ts +0 -34
  340. package/src/server/config/config_loader.ts +0 -496
  341. package/src/server/index.ts +0 -38
  342. package/src/server/logging/logger_service.ts +0 -56
  343. package/src/server/routes/root_router.ts +0 -16
  344. package/src/server/server.ts +0 -28
  345. package/src/server/types/app_types.ts +0 -74
  346. package/src/server/types/express.d.ts +0 -16
  347. package/src/stories/email_verification_layout.stories.tsx +0 -137
  348. package/src/stories/forgot_password_layout.stories.tsx +0 -85
  349. package/src/stories/login_layout.stories.tsx +0 -85
  350. package/src/stories/project_overview.stories.tsx +0 -33
  351. package/src/stories/register_layout.stories.tsx +0 -107
  352. package/tailwind.config.ts +0 -77
  353. package/tsconfig.build.json +0 -36
  354. package/tsconfig.json +0 -28
@@ -1,276 +0,0 @@
1
- // file_description: encapsulate reset password form state, validation, and data interactions
2
- // section: imports
3
- import { useCallback, useMemo, useState, useEffect } from "react";
4
- import { useSearchParams, useRouter } from "next/navigation";
5
- import { toast } from "sonner";
6
- import type { LayoutDataClient } from "../../shared/data/layout_data_client";
7
- import type { PasswordRequirementOptions } from "../../shared/config/layout_customization";
8
- import { RESET_PASSWORD_FIELD_IDS, type ResetPasswordFieldId } from "../config/reset_password_field_config";
9
- import { validatePassword } from "../../shared/utils/validation";
10
-
11
- // section: constants
12
- const PASSWORD_FIELDS: Array<ResetPasswordFieldId> = [
13
- RESET_PASSWORD_FIELD_IDS.PASSWORD,
14
- RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD,
15
- ];
16
-
17
- // section: types
18
- export type ResetPasswordFormValues = Record<ResetPasswordFieldId, string>;
19
- export type ResetPasswordFormErrors = Partial<Record<ResetPasswordFieldId, string | string[]>>;
20
- export type PasswordVisibilityState = Record<
21
- Extract<ResetPasswordFieldId, "password" | "confirm_password">,
22
- boolean
23
- >;
24
-
25
- export type UseResetPasswordFormParams<TClient = unknown> = {
26
- passwordRequirements: PasswordRequirementOptions;
27
- dataClient: LayoutDataClient<TClient>;
28
- loginPath?: string;
29
- };
30
-
31
- export type UseResetPasswordFormResult = {
32
- values: ResetPasswordFormValues;
33
- errors: ResetPasswordFormErrors;
34
- passwordVisibility: PasswordVisibilityState;
35
- isSubmitDisabled: boolean;
36
- isSubmitting: boolean;
37
- isSuccess: boolean;
38
- token: string | null;
39
- isValidatingToken: boolean;
40
- tokenError: string | null;
41
- handleFieldChange: (fieldId: ResetPasswordFieldId, value: string) => void;
42
- togglePasswordVisibility: (fieldId: "password" | "confirm_password") => void;
43
- handleSubmit: (event: React.FormEvent<HTMLFormElement>) => void;
44
- handleCancel: () => void;
45
- };
46
-
47
- // section: helpers
48
- const buildInitialValues = (): ResetPasswordFormValues => ({
49
- [RESET_PASSWORD_FIELD_IDS.PASSWORD]: "",
50
- [RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: "",
51
- });
52
-
53
- // section: hook
54
- export const use_reset_password_form = <TClient,>({
55
- passwordRequirements,
56
- dataClient,
57
- loginPath = "/hazo_auth/login",
58
- }: UseResetPasswordFormParams<TClient>): UseResetPasswordFormResult => {
59
- const router = useRouter();
60
- const searchParams = useSearchParams();
61
- const tokenParam = searchParams.get("token");
62
-
63
- const [values, setValues] = useState<ResetPasswordFormValues>(buildInitialValues);
64
- const [errors, setErrors] = useState<ResetPasswordFormErrors>({});
65
- const [passwordVisibility, setPasswordVisibility] = useState<PasswordVisibilityState>({
66
- password: false,
67
- confirm_password: false,
68
- });
69
- const [isSubmitting, setIsSubmitting] = useState<boolean>(false);
70
- const [isSuccess, setIsSuccess] = useState<boolean>(false);
71
- const [token, setToken] = useState<string | null>(tokenParam);
72
- const [isValidatingToken, setIsValidatingToken] = useState<boolean>(false);
73
- const [tokenError, setTokenError] = useState<string | null>(null);
74
-
75
- // Validate token on mount
76
- useEffect(() => {
77
- if (!tokenParam) {
78
- setTokenError("Reset password link invalid or has expired. Please go to Reset Password page to get a new link.");
79
- return;
80
- }
81
-
82
- // Validate token by calling validation API
83
- const validateToken = async () => {
84
- setIsValidatingToken(true);
85
- setTokenError(null);
86
-
87
- try {
88
- const response = await fetch(`/api/hazo_auth/validate_reset_token?token=${encodeURIComponent(tokenParam)}`, {
89
- method: "GET",
90
- });
91
-
92
- const data = await response.json();
93
-
94
- if (!response.ok || !data.success) {
95
- const errorMessage = data.error || "Reset password link invalid or has expired. Please go to Reset Password page to get a new link.";
96
- setTokenError(errorMessage);
97
- setToken(null);
98
- } else {
99
- // Token is valid
100
- setToken(tokenParam);
101
- setTokenError(null);
102
- }
103
- } catch (error) {
104
- const errorMessage = error instanceof Error ? error.message : "An error occurred while validating the token";
105
- setTokenError("Reset password link invalid or has expired. Please go to Reset Password page to get a new link.");
106
- setToken(null);
107
- } finally {
108
- setIsValidatingToken(false);
109
- }
110
- };
111
-
112
- void validateToken();
113
- }, [tokenParam]);
114
-
115
- const isSubmitDisabled = useMemo(() => {
116
- if (isSubmitting || isSuccess || !token || tokenError) {
117
- return true;
118
- }
119
-
120
- const hasEmptyField = Object.values(values).some((value) => value.trim() === "");
121
- if (hasEmptyField) {
122
- return true;
123
- }
124
-
125
- return Object.keys(errors).length > 0;
126
- }, [isSubmitting, isSuccess, token, tokenError, values, errors]);
127
-
128
- const handleFieldChange = useCallback((fieldId: ResetPasswordFieldId, value: string) => {
129
- setValues((prev) => ({ ...prev, [fieldId]: value }));
130
-
131
- // Clear error for this field when user starts typing
132
- if (errors[fieldId]) {
133
- setErrors((prev) => {
134
- const next = { ...prev };
135
- delete next[fieldId];
136
- return next;
137
- });
138
- }
139
-
140
- // Validate password fields in real-time
141
- if (fieldId === RESET_PASSWORD_FIELD_IDS.PASSWORD) {
142
- const passwordError = validatePassword(value, passwordRequirements);
143
- if (passwordError) {
144
- setErrors((prev) => ({
145
- ...prev,
146
- [RESET_PASSWORD_FIELD_IDS.PASSWORD]: passwordError,
147
- }));
148
- } else {
149
- setErrors((prev) => {
150
- const next = { ...prev };
151
- delete next[RESET_PASSWORD_FIELD_IDS.PASSWORD];
152
- return next;
153
- });
154
- }
155
-
156
- // Also validate confirm password if it has a value
157
- if (values[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]) {
158
- if (value !== values[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]) {
159
- setErrors((prev) => ({
160
- ...prev,
161
- [RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: "Passwords do not match",
162
- }));
163
- } else {
164
- setErrors((prev) => {
165
- const next = { ...prev };
166
- delete next[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD];
167
- return next;
168
- });
169
- }
170
- }
171
- } else if (fieldId === RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD) {
172
- if (value !== values[RESET_PASSWORD_FIELD_IDS.PASSWORD]) {
173
- setErrors((prev) => ({
174
- ...prev,
175
- [RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: "Passwords do not match",
176
- }));
177
- } else {
178
- setErrors((prev) => {
179
- const next = { ...prev };
180
- delete next[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD];
181
- return next;
182
- });
183
- }
184
- }
185
- }, [errors, passwordRequirements, values]);
186
-
187
- const togglePasswordVisibility = useCallback((fieldId: "password" | "confirm_password") => {
188
- setPasswordVisibility((prev) => ({
189
- ...prev,
190
- [fieldId]: !prev[fieldId],
191
- }));
192
- }, []);
193
-
194
- const handleSubmit = useCallback(async (event: React.FormEvent<HTMLFormElement>) => {
195
- event.preventDefault();
196
-
197
- if (!token) {
198
- toast.error("Reset token is missing");
199
- return;
200
- }
201
-
202
- // Validate all fields
203
- const passwordError = validatePassword(values[RESET_PASSWORD_FIELD_IDS.PASSWORD], passwordRequirements);
204
- const confirmPasswordError =
205
- values[RESET_PASSWORD_FIELD_IDS.PASSWORD] !== values[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]
206
- ? "Passwords do not match"
207
- : undefined;
208
-
209
- if (passwordError || confirmPasswordError) {
210
- setErrors({
211
- ...(passwordError ? { [RESET_PASSWORD_FIELD_IDS.PASSWORD]: passwordError } : {}),
212
- ...(confirmPasswordError ? { [RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: confirmPasswordError } : {}),
213
- });
214
- return;
215
- }
216
-
217
- setIsSubmitting(true);
218
-
219
- try {
220
- const response = await fetch("/api/hazo_auth/reset_password", {
221
- method: "POST",
222
- headers: {
223
- "Content-Type": "application/json",
224
- },
225
- body: JSON.stringify({
226
- token,
227
- new_password: values[RESET_PASSWORD_FIELD_IDS.PASSWORD],
228
- }),
229
- });
230
-
231
- const data = await response.json();
232
-
233
- if (!response.ok || !data.success) {
234
- const errorMessage = data.error || "Failed to reset password";
235
- toast.error(errorMessage);
236
- setTokenError(errorMessage);
237
- setIsSubmitting(false);
238
- return;
239
- }
240
-
241
- toast.success(data.message || "Password reset successfully");
242
- setIsSuccess(true);
243
-
244
- // Redirect to login after a short delay
245
- setTimeout(() => {
246
- router.push(loginPath);
247
- }, 2000);
248
- } catch (error) {
249
- const errorMessage = error instanceof Error ? error.message : "An error occurred";
250
- toast.error(errorMessage);
251
- setTokenError(errorMessage);
252
- setIsSubmitting(false);
253
- }
254
- }, [token, values, passwordRequirements, router, loginPath]);
255
-
256
- const handleCancel = useCallback(() => {
257
- router.push(loginPath);
258
- }, [router, loginPath]);
259
-
260
- return {
261
- values,
262
- errors,
263
- passwordVisibility,
264
- isSubmitDisabled,
265
- isSubmitting,
266
- isSuccess,
267
- token,
268
- isValidatingToken,
269
- tokenError,
270
- handleFieldChange,
271
- togglePasswordVisibility,
272
- handleSubmit,
273
- handleCancel,
274
- };
275
- };
276
-
@@ -1,294 +0,0 @@
1
- // file_description: reset password layout component built atop shared layout utilities
2
- // section: client_directive
3
- "use client";
4
-
5
- // section: imports
6
- import { PasswordField } from "../shared/components/password_field";
7
- import { FormFieldWrapper } from "../shared/components/form_field_wrapper";
8
- import { FormHeader } from "../shared/components/form_header";
9
- import { FormActionButtons } from "../shared/components/form_action_buttons";
10
- import { TwoColumnAuthLayout } from "../shared/components/two_column_auth_layout";
11
- import { CheckCircle, XCircle, Loader2 } from "lucide-react";
12
- import { AlreadyLoggedInGuard } from "../shared/components/already_logged_in_guard";
13
- import {
14
- type ButtonPaletteOverrides,
15
- type LayoutFieldMapOverrides,
16
- type LayoutLabelOverrides,
17
- type PasswordRequirementOverrides,
18
- } from "../shared/config/layout_customization";
19
- import {
20
- RESET_PASSWORD_FIELD_IDS,
21
- createResetPasswordFieldDefinitions,
22
- resolveResetPasswordButtonPalette,
23
- resolveResetPasswordLabels,
24
- resolveResetPasswordPasswordRequirements,
25
- } from "./config/reset_password_field_config";
26
- import {
27
- use_reset_password_form,
28
- type UseResetPasswordFormResult,
29
- } from "./hooks/use_reset_password_form";
30
- import { type LayoutDataClient } from "../shared/data/layout_data_client";
31
- import Link from "next/link";
32
-
33
- // section: types
34
- export type ResetPasswordLayoutProps<TClient = unknown> = {
35
- image_src: string;
36
- image_alt: string;
37
- image_background_color?: string;
38
- field_overrides?: LayoutFieldMapOverrides;
39
- labels?: LayoutLabelOverrides;
40
- button_colors?: ButtonPaletteOverrides;
41
- password_requirements?: PasswordRequirementOverrides;
42
- data_client: LayoutDataClient<TClient>;
43
- alreadyLoggedInMessage?: string;
44
- showLogoutButton?: boolean;
45
- showReturnHomeButton?: boolean;
46
- returnHomeButtonLabel?: string;
47
- returnHomePath?: string;
48
- errorMessage?: string;
49
- successMessage?: string;
50
- loginPath?: string;
51
- forgotPasswordPath?: string;
52
- };
53
-
54
- const ORDERED_FIELDS: ResetPasswordFieldId[] = [
55
- RESET_PASSWORD_FIELD_IDS.PASSWORD,
56
- RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD,
57
- ];
58
-
59
- type ResetPasswordFieldId = (typeof RESET_PASSWORD_FIELD_IDS)[keyof typeof RESET_PASSWORD_FIELD_IDS];
60
-
61
- // section: component
62
- export default function reset_password_layout<TClient>({
63
- image_src,
64
- image_alt,
65
- image_background_color = "#f1f5f9",
66
- field_overrides,
67
- labels,
68
- button_colors,
69
- password_requirements,
70
- data_client,
71
- alreadyLoggedInMessage,
72
- showLogoutButton = true,
73
- showReturnHomeButton = false,
74
- returnHomeButtonLabel = "Return home",
75
- returnHomePath = "/",
76
- errorMessage = "Reset password link invalid or has expired. Please go to Reset Password page to get a new link.",
77
- successMessage = "Password reset successfully. Redirecting to login...",
78
- loginPath = "/hazo_auth/login",
79
- forgotPasswordPath = "/hazo_auth/forgot_password",
80
- }: ResetPasswordLayoutProps<TClient>) {
81
- const fieldDefinitions = createResetPasswordFieldDefinitions(field_overrides);
82
- const resolvedLabels = resolveResetPasswordLabels(labels);
83
- const resolvedButtonPalette = resolveResetPasswordButtonPalette(button_colors);
84
- const resolvedPasswordRequirements = resolveResetPasswordPasswordRequirements(
85
- password_requirements,
86
- );
87
-
88
- const form = use_reset_password_form({
89
- passwordRequirements: resolvedPasswordRequirements,
90
- dataClient: data_client,
91
- loginPath,
92
- });
93
-
94
- const renderFields = (formState: UseResetPasswordFormResult) => {
95
- return ORDERED_FIELDS.map((fieldId) => {
96
- const fieldDefinition = fieldDefinitions[fieldId];
97
- const fieldValue = formState.values[fieldId];
98
- const fieldError = formState.errors[fieldId];
99
-
100
- const inputElement = (
101
- <PasswordField
102
- inputId={fieldDefinition.id}
103
- ariaLabel={fieldDefinition.ariaLabel}
104
- value={fieldValue}
105
- placeholder={fieldDefinition.placeholder}
106
- autoComplete={fieldDefinition.autoComplete}
107
- isVisible={formState.passwordVisibility[fieldDefinition.id as "password" | "confirm_password"]}
108
- onChange={(nextValue) => formState.handleFieldChange(fieldId, nextValue)}
109
- onToggleVisibility={() =>
110
- formState.togglePasswordVisibility(fieldDefinition.id as "password" | "confirm_password")
111
- }
112
- errorMessage={fieldError as string | string[] | undefined}
113
- />
114
- );
115
-
116
- return (
117
- <FormFieldWrapper
118
- key={fieldId}
119
- fieldId={fieldDefinition.id}
120
- label={fieldDefinition.label}
121
- input={inputElement}
122
- errorMessage={fieldError as string | string[] | undefined}
123
- />
124
- );
125
- });
126
- };
127
-
128
- // Show success message if password reset was successful
129
- if (form.isSuccess) {
130
- return (
131
- <AlreadyLoggedInGuard
132
- image_src={image_src}
133
- image_alt={image_alt}
134
- image_background_color={image_background_color}
135
- message={alreadyLoggedInMessage}
136
- showLogoutButton={showLogoutButton}
137
- showReturnHomeButton={showReturnHomeButton}
138
- returnHomeButtonLabel={returnHomeButtonLabel}
139
- returnHomePath={returnHomePath}
140
- >
141
- <TwoColumnAuthLayout
142
- imageSrc={image_src}
143
- imageAlt={image_alt}
144
- imageBackgroundColor={image_background_color}
145
- formContent={
146
- <>
147
- <FormHeader
148
- heading={resolvedLabels.heading}
149
- subHeading={resolvedLabels.subHeading}
150
- />
151
- <div className="cls_reset_password_layout_success flex flex-col items-center justify-center gap-4 p-8 text-center">
152
- <CheckCircle
153
- className="cls_reset_password_layout_success_icon h-16 w-16 text-green-600"
154
- aria-hidden="true"
155
- />
156
- <p className="cls_reset_password_layout_success_message text-lg font-medium text-slate-900">
157
- {successMessage}
158
- </p>
159
- </div>
160
- </>
161
- }
162
- />
163
- </AlreadyLoggedInGuard>
164
- );
165
- }
166
-
167
- // Show loading state while validating token
168
- if (form.isValidatingToken) {
169
- return (
170
- <AlreadyLoggedInGuard
171
- image_src={image_src}
172
- image_alt={image_alt}
173
- image_background_color={image_background_color}
174
- message={alreadyLoggedInMessage}
175
- showLogoutButton={showLogoutButton}
176
- showReturnHomeButton={showReturnHomeButton}
177
- returnHomeButtonLabel={returnHomeButtonLabel}
178
- returnHomePath={returnHomePath}
179
- >
180
- <TwoColumnAuthLayout
181
- imageSrc={image_src}
182
- imageAlt={image_alt}
183
- imageBackgroundColor={image_background_color}
184
- formContent={
185
- <div className="cls_reset_password_layout_validating flex flex-col items-center justify-center gap-4 py-8">
186
- <Loader2 className="h-12 w-12 animate-spin text-slate-600" aria-hidden="true" />
187
- <div className="cls_reset_password_layout_validating_text text-center">
188
- <h1 className="cls_reset_password_layout_validating_heading text-2xl font-semibold text-slate-900">
189
- {resolvedLabels.heading}
190
- </h1>
191
- <p className="cls_reset_password_layout_validating_subheading mt-2 text-sm text-slate-600">
192
- Validating reset token...
193
- </p>
194
- </div>
195
- </div>
196
- }
197
- />
198
- </AlreadyLoggedInGuard>
199
- );
200
- }
201
-
202
- // Show error message if token is invalid or missing
203
- if (form.tokenError || !form.token) {
204
- return (
205
- <AlreadyLoggedInGuard
206
- image_src={image_src}
207
- image_alt={image_alt}
208
- image_background_color={image_background_color}
209
- message={alreadyLoggedInMessage}
210
- showLogoutButton={showLogoutButton}
211
- showReturnHomeButton={showReturnHomeButton}
212
- returnHomeButtonLabel={returnHomeButtonLabel}
213
- returnHomePath={returnHomePath}
214
- >
215
- <TwoColumnAuthLayout
216
- imageSrc={image_src}
217
- imageAlt={image_alt}
218
- imageBackgroundColor={image_background_color}
219
- formContent={
220
- <div className="cls_reset_password_layout_error flex flex-col items-center justify-center gap-4 p-8 text-center">
221
- <XCircle
222
- className="cls_reset_password_layout_error_icon h-16 w-16 text-red-600"
223
- aria-hidden="true"
224
- />
225
- <div className="cls_reset_password_layout_error_text">
226
- <h1 className="cls_reset_password_layout_error_heading text-2xl font-semibold text-slate-900">
227
- Invalid Reset Link
228
- </h1>
229
- <p className="cls_reset_password_layout_error_message mt-2 text-sm text-slate-600">
230
- {form.tokenError || errorMessage}
231
- </p>
232
- </div>
233
- <Link
234
- href={forgotPasswordPath}
235
- className="cls_reset_password_layout_forgot_password_link mt-4 text-sm text-blue-600 hover:text-blue-800 underline"
236
- >
237
- Go to Reset Password page
238
- </Link>
239
- </div>
240
- }
241
- />
242
- </AlreadyLoggedInGuard>
243
- );
244
- }
245
-
246
- return (
247
- <AlreadyLoggedInGuard
248
- image_src={image_src}
249
- image_alt={image_alt}
250
- image_background_color={image_background_color}
251
- message={alreadyLoggedInMessage}
252
- showLogoutButton={showLogoutButton}
253
- showReturnHomeButton={showReturnHomeButton}
254
- returnHomeButtonLabel={returnHomeButtonLabel}
255
- returnHomePath={returnHomePath}
256
- >
257
- <TwoColumnAuthLayout
258
- imageSrc={image_src}
259
- imageAlt={image_alt}
260
- imageBackgroundColor={image_background_color}
261
- formContent={
262
- <>
263
- <FormHeader
264
- heading={resolvedLabels.heading}
265
- subHeading={resolvedLabels.subHeading}
266
- />
267
- <form
268
- className="cls_reset_password_layout_form_fields flex flex-col gap-5"
269
- onSubmit={form.handleSubmit}
270
- aria-label="Reset password form"
271
- >
272
- {renderFields(form)}
273
- <FormActionButtons
274
- submitLabel={resolvedLabels.submitButton}
275
- cancelLabel={resolvedLabels.cancelButton}
276
- buttonPalette={resolvedButtonPalette}
277
- isSubmitDisabled={form.isSubmitDisabled}
278
- onCancel={form.handleCancel}
279
- submitAriaLabel="Submit reset password form"
280
- cancelAriaLabel="Cancel reset password form"
281
- />
282
- {form.isSubmitting && (
283
- <div className="cls_reset_password_layout_submitting_indicator text-sm text-slate-600 text-center">
284
- Resetting password...
285
- </div>
286
- )}
287
- </form>
288
- </>
289
- }
290
- />
291
- </AlreadyLoggedInGuard>
292
- );
293
- }
294
-
@@ -1,95 +0,0 @@
1
- // file_description: reusable component to show "already logged in" message when user is authenticated
2
- // section: client_directive
3
- "use client";
4
-
5
- // section: imports
6
- import { use_auth_status } from "../hooks/use_auth_status";
7
- import { LogoutButton } from "./logout_button";
8
- import { Button } from "../../../ui/button";
9
- import { TwoColumnAuthLayout } from "./two_column_auth_layout";
10
- import { useRouter } from "next/navigation";
11
- import { Home } from "lucide-react";
12
-
13
- // section: types
14
- export type AlreadyLoggedInGuardProps = {
15
- image_src: string;
16
- image_alt: string;
17
- image_background_color?: string;
18
- message?: string;
19
- showLogoutButton?: boolean;
20
- showReturnHomeButton?: boolean;
21
- returnHomeButtonLabel?: string;
22
- returnHomePath?: string;
23
- requireEmailVerified?: boolean;
24
- children: React.ReactNode;
25
- };
26
-
27
- // section: component
28
- /**
29
- * Guard component that shows "already logged in" message if user is authenticated
30
- * Otherwise renders children
31
- * @param props - Component props including layout config and message customization
32
- * @returns Either the "already logged in" UI or the children
33
- */
34
- export function AlreadyLoggedInGuard({
35
- image_src,
36
- image_alt,
37
- image_background_color = "#f1f5f9",
38
- message = "You're already logged in.",
39
- showLogoutButton = true,
40
- showReturnHomeButton = false,
41
- returnHomeButtonLabel = "Return home",
42
- returnHomePath = "/",
43
- requireEmailVerified = false,
44
- children,
45
- }: AlreadyLoggedInGuardProps) {
46
- const router = useRouter();
47
- const authStatus = use_auth_status();
48
-
49
- // Check if user should see "already logged in" message
50
- // If requireEmailVerified is true, only show if email is verified
51
- // If requireEmailVerified is false, show if authenticated
52
- const shouldShowAlreadyLoggedIn =
53
- authStatus.authenticated &&
54
- !authStatus.loading &&
55
- (!requireEmailVerified || authStatus.email_verified === true);
56
-
57
- if (shouldShowAlreadyLoggedIn) {
58
- return (
59
- <TwoColumnAuthLayout
60
- imageSrc={image_src}
61
- imageAlt={image_alt}
62
- imageBackgroundColor={image_background_color}
63
- formContent={
64
- <div className="cls_already_logged_in_guard flex flex-col items-center justify-center gap-4 p-8 text-center">
65
- <p className="cls_already_logged_in_guard_message text-lg font-medium text-slate-900">
66
- {message}
67
- </p>
68
- <div className="cls_already_logged_in_guard_actions flex flex-col gap-3 items-center mt-4">
69
- {showLogoutButton && (
70
- <LogoutButton
71
- className="cls_already_logged_in_guard_logout_button"
72
- variant="default"
73
- />
74
- )}
75
- {showReturnHomeButton && (
76
- <Button
77
- onClick={() => router.push(returnHomePath)}
78
- variant="outline"
79
- className="cls_already_logged_in_guard_return_home_button"
80
- aria-label={returnHomeButtonLabel}
81
- >
82
- <Home className="h-4 w-4 mr-2" aria-hidden="true" />
83
- {returnHomeButtonLabel}
84
- </Button>
85
- )}
86
- </div>
87
- </div>
88
- }
89
- />
90
- );
91
- }
92
-
93
- return <>{children}</>;
94
- }
95
-
@@ -1,36 +0,0 @@
1
- // file_description: server component that chooses between sidebar shell and standalone shell
2
- // section: imports
3
- import type { ReactNode } from "react";
4
- import { SidebarLayoutWrapper } from "./sidebar_layout_wrapper";
5
- import { StandaloneLayoutWrapper } from "./standalone_layout_wrapper";
6
- import { get_ui_shell_config } from "../../../../lib/ui_shell_config.server";
7
-
8
- // section: types
9
- type AuthPageShellProps = {
10
- children: ReactNode;
11
- };
12
-
13
- // section: component
14
- export function AuthPageShell({ children }: AuthPageShellProps) {
15
- const uiShellConfig = get_ui_shell_config();
16
-
17
- if (uiShellConfig.layout_mode === "standalone") {
18
- return (
19
- <StandaloneLayoutWrapper
20
- heading={uiShellConfig.standalone_heading}
21
- description={uiShellConfig.standalone_description}
22
- wrapperClassName={uiShellConfig.standalone_wrapper_class}
23
- contentClassName={uiShellConfig.standalone_content_class}
24
- showHeading={uiShellConfig.standalone_show_heading}
25
- showDescription={uiShellConfig.standalone_show_description}
26
- >
27
- {children}
28
- </StandaloneLayoutWrapper>
29
- );
30
- }
31
-
32
- return <SidebarLayoutWrapper>{children}</SidebarLayoutWrapper>;
33
- }
34
-
35
-
36
-