hazo_auth 0.1.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 (162) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +48 -0
  3. package/components.json +22 -0
  4. package/hazo_auth_config.example.ini +414 -0
  5. package/hazo_notify_config.example.ini +159 -0
  6. package/instrumentation.ts +32 -0
  7. package/migrations/001_add_token_type_to_refresh_tokens.sql +14 -0
  8. package/migrations/002_add_name_to_hazo_users.sql +7 -0
  9. package/next.config.mjs +55 -0
  10. package/package.json +114 -0
  11. package/postcss.config.mjs +8 -0
  12. package/public/file.svg +1 -0
  13. package/public/globe.svg +1 -0
  14. package/public/next.svg +1 -0
  15. package/public/vercel.svg +1 -0
  16. package/public/window.svg +1 -0
  17. package/scripts/apply_migration.ts +118 -0
  18. package/src/app/api/auth/change_password/route.ts +109 -0
  19. package/src/app/api/auth/forgot_password/route.ts +107 -0
  20. package/src/app/api/auth/library_photos/route.ts +70 -0
  21. package/src/app/api/auth/login/route.ts +155 -0
  22. package/src/app/api/auth/logout/route.ts +62 -0
  23. package/src/app/api/auth/me/route.ts +47 -0
  24. package/src/app/api/auth/profile_picture/[filename]/route.ts +67 -0
  25. package/src/app/api/auth/register/route.ts +106 -0
  26. package/src/app/api/auth/remove_profile_picture/route.ts +86 -0
  27. package/src/app/api/auth/resend_verification/route.ts +107 -0
  28. package/src/app/api/auth/reset_password/route.ts +107 -0
  29. package/src/app/api/auth/update_user/route.ts +126 -0
  30. package/src/app/api/auth/upload_profile_picture/route.ts +268 -0
  31. package/src/app/api/auth/validate_reset_token/route.ts +80 -0
  32. package/src/app/api/auth/verify_email/route.ts +85 -0
  33. package/src/app/api/migrations/apply/route.ts +91 -0
  34. package/src/app/favicon.ico +0 -0
  35. package/src/app/fonts/GeistMonoVF.woff +0 -0
  36. package/src/app/fonts/GeistVF.woff +0 -0
  37. package/src/app/forgot_password/forgot_password_page_client.tsx +60 -0
  38. package/src/app/forgot_password/page.tsx +24 -0
  39. package/src/app/globals.css +89 -0
  40. package/src/app/hazo_connect/api/sqlite/data/route.ts +197 -0
  41. package/src/app/hazo_connect/api/sqlite/schema/route.ts +35 -0
  42. package/src/app/hazo_connect/api/sqlite/tables/route.ts +26 -0
  43. package/src/app/hazo_connect/sqlite_admin/page.tsx +51 -0
  44. package/src/app/hazo_connect/sqlite_admin/sqlite-admin-client.tsx +947 -0
  45. package/src/app/layout.tsx +43 -0
  46. package/src/app/login/login_page_client.tsx +71 -0
  47. package/src/app/login/page.tsx +26 -0
  48. package/src/app/my_settings/my_settings_page_client.tsx +120 -0
  49. package/src/app/my_settings/page.tsx +40 -0
  50. package/src/app/page.tsx +170 -0
  51. package/src/app/register/page.tsx +26 -0
  52. package/src/app/register/register_page_client.tsx +72 -0
  53. package/src/app/reset_password/page.tsx +29 -0
  54. package/src/app/reset_password/reset_password_page_client.tsx +81 -0
  55. package/src/app/verify_email/page.tsx +24 -0
  56. package/src/app/verify_email/verify_email_page_client.tsx +60 -0
  57. package/src/components/layouts/email_verification/config/email_verification_field_config.ts +86 -0
  58. package/src/components/layouts/email_verification/hooks/use_email_verification.ts +291 -0
  59. package/src/components/layouts/email_verification/index.tsx +297 -0
  60. package/src/components/layouts/forgot_password/config/forgot_password_field_config.ts +58 -0
  61. package/src/components/layouts/forgot_password/hooks/use_forgot_password_form.ts +179 -0
  62. package/src/components/layouts/forgot_password/index.tsx +168 -0
  63. package/src/components/layouts/login/config/login_field_config.ts +67 -0
  64. package/src/components/layouts/login/hooks/use_login_form.ts +281 -0
  65. package/src/components/layouts/login/index.tsx +224 -0
  66. package/src/components/layouts/my_settings/components/editable_field.tsx +177 -0
  67. package/src/components/layouts/my_settings/components/password_change_dialog.tsx +301 -0
  68. package/src/components/layouts/my_settings/components/profile_picture_dialog.tsx +385 -0
  69. package/src/components/layouts/my_settings/components/profile_picture_display.tsx +66 -0
  70. package/src/components/layouts/my_settings/components/profile_picture_gravatar_tab.tsx +143 -0
  71. package/src/components/layouts/my_settings/components/profile_picture_library_tab.tsx +282 -0
  72. package/src/components/layouts/my_settings/components/profile_picture_upload_tab.tsx +341 -0
  73. package/src/components/layouts/my_settings/config/my_settings_field_config.ts +61 -0
  74. package/src/components/layouts/my_settings/hooks/use_my_settings.ts +458 -0
  75. package/src/components/layouts/my_settings/index.tsx +351 -0
  76. package/src/components/layouts/register/config/register_field_config.ts +101 -0
  77. package/src/components/layouts/register/hooks/use_register_form.ts +272 -0
  78. package/src/components/layouts/register/index.tsx +208 -0
  79. package/src/components/layouts/reset_password/config/reset_password_field_config.ts +86 -0
  80. package/src/components/layouts/reset_password/hooks/use_reset_password_form.ts +276 -0
  81. package/src/components/layouts/reset_password/index.tsx +294 -0
  82. package/src/components/layouts/shared/components/already_logged_in_guard.tsx +95 -0
  83. package/src/components/layouts/shared/components/field_error_message.tsx +29 -0
  84. package/src/components/layouts/shared/components/form_action_buttons.tsx +64 -0
  85. package/src/components/layouts/shared/components/form_field_wrapper.tsx +44 -0
  86. package/src/components/layouts/shared/components/form_header.tsx +36 -0
  87. package/src/components/layouts/shared/components/logout_button.tsx +76 -0
  88. package/src/components/layouts/shared/components/password_field.tsx +72 -0
  89. package/src/components/layouts/shared/components/sidebar_layout_wrapper.tsx +264 -0
  90. package/src/components/layouts/shared/components/two_column_auth_layout.tsx +44 -0
  91. package/src/components/layouts/shared/components/unauthorized_guard.tsx +78 -0
  92. package/src/components/layouts/shared/components/visual_panel.tsx +41 -0
  93. package/src/components/layouts/shared/config/layout_customization.ts +95 -0
  94. package/src/components/layouts/shared/data/layout_data_client.ts +19 -0
  95. package/src/components/layouts/shared/hooks/use_auth_status.ts +103 -0
  96. package/src/components/layouts/shared/utils/ip_address.ts +37 -0
  97. package/src/components/layouts/shared/utils/validation.ts +66 -0
  98. package/src/components/ui/avatar.tsx +50 -0
  99. package/src/components/ui/button.tsx +57 -0
  100. package/src/components/ui/dialog.tsx +122 -0
  101. package/src/components/ui/hazo_ui_tooltip.tsx +67 -0
  102. package/src/components/ui/input.tsx +22 -0
  103. package/src/components/ui/label.tsx +26 -0
  104. package/src/components/ui/separator.tsx +31 -0
  105. package/src/components/ui/sheet.tsx +139 -0
  106. package/src/components/ui/sidebar.tsx +773 -0
  107. package/src/components/ui/skeleton.tsx +15 -0
  108. package/src/components/ui/sonner.tsx +31 -0
  109. package/src/components/ui/switch.tsx +29 -0
  110. package/src/components/ui/tabs.tsx +55 -0
  111. package/src/components/ui/tooltip.tsx +32 -0
  112. package/src/components/ui/vertical-tabs.tsx +59 -0
  113. package/src/hooks/use-mobile.tsx +19 -0
  114. package/src/lib/already_logged_in_config.server.ts +46 -0
  115. package/src/lib/app_logger.ts +24 -0
  116. package/src/lib/auth/auth_utils.server.ts +196 -0
  117. package/src/lib/auth/server_auth.ts +88 -0
  118. package/src/lib/config/config_loader.server.ts +149 -0
  119. package/src/lib/email_verification_config.server.ts +32 -0
  120. package/src/lib/file_types_config.server.ts +25 -0
  121. package/src/lib/forgot_password_config.server.ts +32 -0
  122. package/src/lib/hazo_connect_instance.server.ts +77 -0
  123. package/src/lib/hazo_connect_setup.server.ts +181 -0
  124. package/src/lib/hazo_connect_setup.ts +54 -0
  125. package/src/lib/login_config.server.ts +46 -0
  126. package/src/lib/messages_config.server.ts +45 -0
  127. package/src/lib/migrations/apply_migration.ts +105 -0
  128. package/src/lib/my_settings_config.server.ts +135 -0
  129. package/src/lib/password_requirements_config.server.ts +39 -0
  130. package/src/lib/profile_picture_config.server.ts +56 -0
  131. package/src/lib/register_config.server.ts +57 -0
  132. package/src/lib/reset_password_config.server.ts +75 -0
  133. package/src/lib/services/email_service.ts +581 -0
  134. package/src/lib/services/email_verification_service.ts +264 -0
  135. package/src/lib/services/login_service.ts +118 -0
  136. package/src/lib/services/password_change_service.ts +154 -0
  137. package/src/lib/services/password_reset_service.ts +405 -0
  138. package/src/lib/services/profile_picture_remove_service.ts +120 -0
  139. package/src/lib/services/profile_picture_service.ts +215 -0
  140. package/src/lib/services/profile_picture_source_mapper.ts +62 -0
  141. package/src/lib/services/registration_service.ts +163 -0
  142. package/src/lib/services/token_service.ts +240 -0
  143. package/src/lib/services/user_update_service.ts +128 -0
  144. package/src/lib/ui_sizes_config.server.ts +37 -0
  145. package/src/lib/user_fields_config.server.ts +31 -0
  146. package/src/lib/utils/api_route_helpers.ts +60 -0
  147. package/src/lib/utils.ts +11 -0
  148. package/src/middleware.ts +91 -0
  149. package/src/server/config/config_loader.ts +496 -0
  150. package/src/server/index.ts +38 -0
  151. package/src/server/logging/logger_service.ts +56 -0
  152. package/src/server/routes/root_router.ts +16 -0
  153. package/src/server/server.ts +28 -0
  154. package/src/server/types/app_types.ts +74 -0
  155. package/src/server/types/express.d.ts +15 -0
  156. package/src/stories/email_verification_layout.stories.tsx +137 -0
  157. package/src/stories/forgot_password_layout.stories.tsx +85 -0
  158. package/src/stories/login_layout.stories.tsx +85 -0
  159. package/src/stories/project_overview.stories.tsx +33 -0
  160. package/src/stories/register_layout.stories.tsx +107 -0
  161. package/tailwind.config.ts +77 -0
  162. package/tsconfig.json +27 -0
@@ -0,0 +1,458 @@
1
+ // file_description: hook for managing my settings state and API calls
2
+ // section: client_directive
3
+ "use client";
4
+
5
+ // section: imports
6
+ import { useState, useEffect, useCallback } from "react";
7
+ import { use_auth_status, trigger_auth_status_refresh } from "@/components/layouts/shared/hooks/use_auth_status";
8
+ import { toast } from "sonner";
9
+ import type { PasswordRequirementOptions } from "@/components/layouts/shared/config/layout_customization";
10
+
11
+ // section: types
12
+ export type PasswordFields = {
13
+ currentPassword: string;
14
+ newPassword: string;
15
+ confirmPassword: string;
16
+ currentPasswordVisible: boolean;
17
+ newPasswordVisible: boolean;
18
+ confirmPasswordVisible: boolean;
19
+ errors: {
20
+ currentPassword?: string;
21
+ newPassword?: string | string[];
22
+ confirmPassword?: string;
23
+ };
24
+ };
25
+
26
+ export type UseMySettingsResult = {
27
+ // User data
28
+ name: string;
29
+ email: string;
30
+ profilePictureUrl?: string;
31
+ profileSource?: "upload" | "library" | "gravatar" | "custom";
32
+ lastLogon?: string;
33
+ loading: boolean;
34
+
35
+ // Password fields
36
+ passwordFields?: PasswordFields;
37
+ handlePasswordFieldChange: (field: "currentPassword" | "newPassword" | "confirmPassword", value: string) => void;
38
+ togglePasswordVisibility: (field: "currentPassword" | "newPassword" | "confirmPassword") => void;
39
+ handlePasswordSave: () => Promise<void>;
40
+ isPasswordSaveDisabled: boolean;
41
+
42
+ // Profile picture
43
+ profilePictureDialogOpen: boolean;
44
+ handleProfilePictureEdit: () => void;
45
+ handleProfilePictureDialogClose: () => void;
46
+ handleProfilePictureSave: (profilePictureUrl: string, profileSource: "upload" | "library" | "gravatar") => Promise<void>;
47
+ handleProfilePictureRemove: () => Promise<void>;
48
+
49
+ // Actions
50
+ handleNameSave: (value: string) => Promise<void>;
51
+ handleEmailSave: (value: string) => Promise<void>;
52
+
53
+ // Refresh
54
+ refreshUserData: () => Promise<void>;
55
+ };
56
+
57
+ export type UseMySettingsParams = {
58
+ passwordRequirements: PasswordRequirementOptions;
59
+ };
60
+
61
+ // section: helpers
62
+ /**
63
+ * Validates email format
64
+ */
65
+ const validateEmail = (email: string): string | null => {
66
+ if (!email || email.trim() === "") {
67
+ return "Email is required";
68
+ }
69
+ const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
70
+ if (!emailRegex.test(email)) {
71
+ return "Invalid email address format";
72
+ }
73
+ return null;
74
+ };
75
+
76
+ /**
77
+ * Validates name (optional, but if provided should not be empty)
78
+ */
79
+ const validateName = (name: string): string | null => {
80
+ if (name.trim() === "") {
81
+ return "Name cannot be empty";
82
+ }
83
+ return null;
84
+ };
85
+
86
+ // section: hook
87
+ /**
88
+ * Hook for managing my settings state and API calls
89
+ * Handles user data loading, field editing, and API calls for updates
90
+ * @param params - Hook parameters including password requirements
91
+ * @returns My settings hook result with state and actions
92
+ */
93
+ export function use_my_settings({
94
+ passwordRequirements,
95
+ }: UseMySettingsParams): UseMySettingsResult {
96
+ const authStatus = use_auth_status();
97
+
98
+ // Password fields state
99
+ const [passwordFields, setPasswordFields] = useState<PasswordFields>({
100
+ currentPassword: "",
101
+ newPassword: "",
102
+ confirmPassword: "",
103
+ currentPasswordVisible: false,
104
+ newPasswordVisible: false,
105
+ confirmPasswordVisible: false,
106
+ errors: {},
107
+ });
108
+
109
+ // Get user data from auth status
110
+ const name = authStatus.name || "";
111
+ const email = authStatus.email || "";
112
+ const profilePictureUrl = authStatus.profile_picture_url;
113
+ const profileSource = authStatus.profile_source;
114
+ const lastLogon = authStatus.last_logon;
115
+ const loading = authStatus.loading;
116
+
117
+ /**
118
+ * Refreshes user data by triggering auth status refresh
119
+ */
120
+ const refreshUserData = useCallback(async () => {
121
+ trigger_auth_status_refresh();
122
+ // Also call the refresh method directly
123
+ await authStatus.refresh();
124
+ }, [authStatus]);
125
+
126
+ /**
127
+ * Updates user name
128
+ */
129
+ const handleNameSave = useCallback(async (value: string) => {
130
+ const validationError = validateName(value);
131
+ if (validationError) {
132
+ toast.error(validationError);
133
+ throw new Error(validationError);
134
+ }
135
+
136
+ try {
137
+ const response = await fetch("/api/auth/update_user", {
138
+ method: "PATCH",
139
+ headers: {
140
+ "Content-Type": "application/json",
141
+ },
142
+ credentials: "include",
143
+ body: JSON.stringify({ name: value }),
144
+ });
145
+
146
+ const data = await response.json();
147
+
148
+ if (!response.ok || !data.success) {
149
+ const errorMessage = data.error || "Failed to update name";
150
+ toast.error(errorMessage);
151
+ throw new Error(errorMessage);
152
+ }
153
+
154
+ toast.success("Name updated successfully");
155
+ await refreshUserData();
156
+ } catch (error) {
157
+ const errorMessage = error instanceof Error ? error.message : "Failed to update name";
158
+ toast.error(errorMessage);
159
+ throw error;
160
+ }
161
+ }, [refreshUserData]);
162
+
163
+ /**
164
+ * Updates user email
165
+ */
166
+ const handleEmailSave = useCallback(async (value: string) => {
167
+ const validationError = validateEmail(value);
168
+ if (validationError) {
169
+ toast.error(validationError);
170
+ throw new Error(validationError);
171
+ }
172
+
173
+ try {
174
+ const response = await fetch("/api/auth/update_user", {
175
+ method: "PATCH",
176
+ headers: {
177
+ "Content-Type": "application/json",
178
+ },
179
+ credentials: "include",
180
+ body: JSON.stringify({ email: value }),
181
+ });
182
+
183
+ const data = await response.json();
184
+
185
+ if (!response.ok || !data.success) {
186
+ const errorMessage = data.error || "Failed to update email";
187
+ toast.error(errorMessage);
188
+ throw new Error(errorMessage);
189
+ }
190
+
191
+ if (data.email_changed) {
192
+ toast.success("Email updated successfully. Please verify your new email address.");
193
+ } else {
194
+ toast.success("Email updated successfully");
195
+ }
196
+
197
+ await refreshUserData();
198
+ } catch (error) {
199
+ const errorMessage = error instanceof Error ? error.message : "Failed to update email";
200
+ toast.error(errorMessage);
201
+ throw error;
202
+ }
203
+ }, [refreshUserData]);
204
+
205
+ /**
206
+ * Validates password requirements
207
+ */
208
+ const validatePassword = useCallback((password: string): string | null => {
209
+ if (!password || password.length < passwordRequirements.minimum_length) {
210
+ return `Password must be at least ${passwordRequirements.minimum_length} characters long`;
211
+ }
212
+
213
+ const errors: string[] = [];
214
+
215
+ if (passwordRequirements.require_uppercase && !/[A-Z]/.test(password)) {
216
+ errors.push("uppercase letter");
217
+ }
218
+
219
+ if (passwordRequirements.require_lowercase && !/[a-z]/.test(password)) {
220
+ errors.push("lowercase letter");
221
+ }
222
+
223
+ if (passwordRequirements.require_number && !/[0-9]/.test(password)) {
224
+ errors.push("number");
225
+ }
226
+
227
+ if (passwordRequirements.require_special && !/[^A-Za-z0-9]/.test(password)) {
228
+ errors.push("special character");
229
+ }
230
+
231
+ if (errors.length > 0) {
232
+ return `Password must contain at least one: ${errors.join(", ")}`;
233
+ }
234
+
235
+ return null;
236
+ }, [passwordRequirements]);
237
+
238
+ /**
239
+ * Handles password field change
240
+ */
241
+ const handlePasswordFieldChange = useCallback((field: "currentPassword" | "newPassword" | "confirmPassword", value: string) => {
242
+ setPasswordFields((prev) => {
243
+ const newFields = { ...prev, [field]: value };
244
+ // Clear errors for this field when user types
245
+ if (newFields.errors[field as keyof typeof newFields.errors]) {
246
+ newFields.errors = { ...newFields.errors, [field]: undefined };
247
+ }
248
+ return newFields;
249
+ });
250
+ }, []);
251
+
252
+ /**
253
+ * Toggles password visibility
254
+ */
255
+ const togglePasswordVisibility = useCallback((field: "currentPassword" | "newPassword" | "confirmPassword") => {
256
+ setPasswordFields((prev) => {
257
+ const fieldKey = `${field}Visible` as keyof PasswordFields;
258
+ const currentValue = prev[fieldKey] as boolean;
259
+ return {
260
+ ...prev,
261
+ [fieldKey]: !currentValue,
262
+ };
263
+ });
264
+ }, []);
265
+
266
+ /**
267
+ * Validates password form
268
+ */
269
+ const validatePasswordForm = useCallback((): boolean => {
270
+ const errors: PasswordFields["errors"] = {};
271
+
272
+ if (!passwordFields.currentPassword) {
273
+ errors.currentPassword = "Current password is required";
274
+ }
275
+
276
+ const newPasswordError = validatePassword(passwordFields.newPassword);
277
+ if (newPasswordError) {
278
+ errors.newPassword = newPasswordError;
279
+ }
280
+
281
+ if (!passwordFields.confirmPassword) {
282
+ errors.confirmPassword = "Please confirm your new password";
283
+ } else if (passwordFields.newPassword !== passwordFields.confirmPassword) {
284
+ errors.confirmPassword = "Passwords do not match";
285
+ }
286
+
287
+ setPasswordFields((prev) => ({ ...prev, errors }));
288
+ return Object.keys(errors).length === 0;
289
+ }, [passwordFields, validatePassword]);
290
+
291
+ /**
292
+ * Checks if password save should be disabled
293
+ */
294
+ const isPasswordSaveDisabled = useCallback((): boolean => {
295
+ return !passwordFields.currentPassword || !passwordFields.newPassword || !passwordFields.confirmPassword;
296
+ }, [passwordFields]);
297
+
298
+ /**
299
+ * Saves password changes
300
+ */
301
+ const handlePasswordSave = useCallback(async () => {
302
+ if (!validatePasswordForm()) {
303
+ return;
304
+ }
305
+
306
+ try {
307
+ const response = await fetch("/api/auth/change_password", {
308
+ method: "POST",
309
+ headers: {
310
+ "Content-Type": "application/json",
311
+ },
312
+ credentials: "include",
313
+ body: JSON.stringify({
314
+ current_password: passwordFields.currentPassword,
315
+ new_password: passwordFields.newPassword,
316
+ }),
317
+ });
318
+
319
+ const data = await response.json();
320
+
321
+ if (!response.ok || !data.success) {
322
+ const errorMessage = data.error || "Failed to change password";
323
+ setPasswordFields((prev) => ({
324
+ ...prev,
325
+ errors: { currentPassword: errorMessage },
326
+ }));
327
+ toast.error(errorMessage);
328
+ throw new Error(errorMessage);
329
+ }
330
+
331
+ toast.success("Password changed successfully");
332
+ // Reset password fields
333
+ setPasswordFields({
334
+ currentPassword: "",
335
+ newPassword: "",
336
+ confirmPassword: "",
337
+ currentPasswordVisible: false,
338
+ newPasswordVisible: false,
339
+ confirmPasswordVisible: false,
340
+ errors: {},
341
+ });
342
+ } catch (error) {
343
+ const errorMessage = error instanceof Error ? error.message : "Failed to change password";
344
+ toast.error(errorMessage);
345
+ throw error;
346
+ }
347
+ }, [passwordFields, validatePasswordForm]);
348
+
349
+ /**
350
+ * Profile picture dialog state
351
+ */
352
+ const [profilePictureDialogOpen, setProfilePictureDialogOpen] = useState(false);
353
+
354
+ /**
355
+ * Opens profile picture dialog
356
+ */
357
+ const handleProfilePictureEdit = useCallback(() => {
358
+ setProfilePictureDialogOpen(true);
359
+ }, []);
360
+
361
+ /**
362
+ * Closes profile picture dialog
363
+ */
364
+ const handleProfilePictureDialogClose = useCallback(() => {
365
+ setProfilePictureDialogOpen(false);
366
+ }, []);
367
+
368
+ /**
369
+ * Saves profile picture changes
370
+ * Note: profilePictureUrl is already a fully-formed URL (Gravatar URL is generated in the dialog)
371
+ */
372
+ const handleProfilePictureSave = useCallback(async (profilePictureUrl: string, profileSource: "upload" | "library" | "gravatar") => {
373
+ try {
374
+ const response = await fetch("/api/auth/update_user", {
375
+ method: "PATCH",
376
+ headers: {
377
+ "Content-Type": "application/json",
378
+ },
379
+ credentials: "include",
380
+ body: JSON.stringify({
381
+ profile_picture_url: profilePictureUrl,
382
+ profile_source: profileSource,
383
+ }),
384
+ });
385
+
386
+ const data = await response.json();
387
+
388
+ if (!response.ok || !data.success) {
389
+ const errorMessage = data.error || "Failed to update profile picture";
390
+ toast.error(errorMessage);
391
+ throw new Error(errorMessage);
392
+ }
393
+
394
+ toast.success("Profile picture updated successfully");
395
+ await refreshUserData();
396
+ setProfilePictureDialogOpen(false);
397
+ } catch (error) {
398
+ const errorMessage = error instanceof Error ? error.message : "Failed to update profile picture";
399
+ toast.error(errorMessage);
400
+ throw error;
401
+ }
402
+ }, [refreshUserData]);
403
+
404
+ /**
405
+ * Removes profile picture
406
+ * - If upload: deletes the file and clears profile_picture_url and profile_source
407
+ * - If gravatar/library: clears profile_picture_url and profile_source
408
+ */
409
+ const handleProfilePictureRemove = useCallback(async () => {
410
+ try {
411
+ const response = await fetch("/api/auth/remove_profile_picture", {
412
+ method: "DELETE",
413
+ headers: {
414
+ "Content-Type": "application/json",
415
+ },
416
+ credentials: "include",
417
+ });
418
+
419
+ const data = await response.json();
420
+
421
+ if (!response.ok || !data.success) {
422
+ const errorMessage = data.error || "Failed to remove profile picture";
423
+ toast.error(errorMessage);
424
+ throw new Error(errorMessage);
425
+ }
426
+
427
+ toast.success("Profile picture removed successfully");
428
+ await refreshUserData();
429
+ } catch (error) {
430
+ const errorMessage = error instanceof Error ? error.message : "Failed to remove profile picture";
431
+ toast.error(errorMessage);
432
+ throw error;
433
+ }
434
+ }, [refreshUserData]);
435
+
436
+ return {
437
+ name,
438
+ email,
439
+ profilePictureUrl,
440
+ profileSource,
441
+ lastLogon,
442
+ loading,
443
+ passwordFields,
444
+ handlePasswordFieldChange,
445
+ togglePasswordVisibility,
446
+ handlePasswordSave,
447
+ isPasswordSaveDisabled: isPasswordSaveDisabled(),
448
+ profilePictureDialogOpen,
449
+ handleProfilePictureEdit,
450
+ handleProfilePictureDialogClose,
451
+ handleProfilePictureSave,
452
+ handleProfilePictureRemove,
453
+ handleNameSave,
454
+ handleEmailSave,
455
+ refreshUserData,
456
+ };
457
+ }
458
+