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.
- package/LICENSE +21 -0
- package/README.md +48 -0
- package/components.json +22 -0
- package/hazo_auth_config.example.ini +414 -0
- package/hazo_notify_config.example.ini +159 -0
- package/instrumentation.ts +32 -0
- package/migrations/001_add_token_type_to_refresh_tokens.sql +14 -0
- package/migrations/002_add_name_to_hazo_users.sql +7 -0
- package/next.config.mjs +55 -0
- package/package.json +114 -0
- package/postcss.config.mjs +8 -0
- package/public/file.svg +1 -0
- package/public/globe.svg +1 -0
- package/public/next.svg +1 -0
- package/public/vercel.svg +1 -0
- package/public/window.svg +1 -0
- package/scripts/apply_migration.ts +118 -0
- package/src/app/api/auth/change_password/route.ts +109 -0
- package/src/app/api/auth/forgot_password/route.ts +107 -0
- package/src/app/api/auth/library_photos/route.ts +70 -0
- package/src/app/api/auth/login/route.ts +155 -0
- package/src/app/api/auth/logout/route.ts +62 -0
- package/src/app/api/auth/me/route.ts +47 -0
- package/src/app/api/auth/profile_picture/[filename]/route.ts +67 -0
- package/src/app/api/auth/register/route.ts +106 -0
- package/src/app/api/auth/remove_profile_picture/route.ts +86 -0
- package/src/app/api/auth/resend_verification/route.ts +107 -0
- package/src/app/api/auth/reset_password/route.ts +107 -0
- package/src/app/api/auth/update_user/route.ts +126 -0
- package/src/app/api/auth/upload_profile_picture/route.ts +268 -0
- package/src/app/api/auth/validate_reset_token/route.ts +80 -0
- package/src/app/api/auth/verify_email/route.ts +85 -0
- package/src/app/api/migrations/apply/route.ts +91 -0
- package/src/app/favicon.ico +0 -0
- package/src/app/fonts/GeistMonoVF.woff +0 -0
- package/src/app/fonts/GeistVF.woff +0 -0
- package/src/app/forgot_password/forgot_password_page_client.tsx +60 -0
- package/src/app/forgot_password/page.tsx +24 -0
- package/src/app/globals.css +89 -0
- package/src/app/hazo_connect/api/sqlite/data/route.ts +197 -0
- package/src/app/hazo_connect/api/sqlite/schema/route.ts +35 -0
- package/src/app/hazo_connect/api/sqlite/tables/route.ts +26 -0
- package/src/app/hazo_connect/sqlite_admin/page.tsx +51 -0
- package/src/app/hazo_connect/sqlite_admin/sqlite-admin-client.tsx +947 -0
- package/src/app/layout.tsx +43 -0
- package/src/app/login/login_page_client.tsx +71 -0
- package/src/app/login/page.tsx +26 -0
- package/src/app/my_settings/my_settings_page_client.tsx +120 -0
- package/src/app/my_settings/page.tsx +40 -0
- package/src/app/page.tsx +170 -0
- package/src/app/register/page.tsx +26 -0
- package/src/app/register/register_page_client.tsx +72 -0
- package/src/app/reset_password/page.tsx +29 -0
- package/src/app/reset_password/reset_password_page_client.tsx +81 -0
- package/src/app/verify_email/page.tsx +24 -0
- package/src/app/verify_email/verify_email_page_client.tsx +60 -0
- package/src/components/layouts/email_verification/config/email_verification_field_config.ts +86 -0
- package/src/components/layouts/email_verification/hooks/use_email_verification.ts +291 -0
- package/src/components/layouts/email_verification/index.tsx +297 -0
- package/src/components/layouts/forgot_password/config/forgot_password_field_config.ts +58 -0
- package/src/components/layouts/forgot_password/hooks/use_forgot_password_form.ts +179 -0
- package/src/components/layouts/forgot_password/index.tsx +168 -0
- package/src/components/layouts/login/config/login_field_config.ts +67 -0
- package/src/components/layouts/login/hooks/use_login_form.ts +281 -0
- package/src/components/layouts/login/index.tsx +224 -0
- package/src/components/layouts/my_settings/components/editable_field.tsx +177 -0
- package/src/components/layouts/my_settings/components/password_change_dialog.tsx +301 -0
- package/src/components/layouts/my_settings/components/profile_picture_dialog.tsx +385 -0
- package/src/components/layouts/my_settings/components/profile_picture_display.tsx +66 -0
- package/src/components/layouts/my_settings/components/profile_picture_gravatar_tab.tsx +143 -0
- package/src/components/layouts/my_settings/components/profile_picture_library_tab.tsx +282 -0
- package/src/components/layouts/my_settings/components/profile_picture_upload_tab.tsx +341 -0
- package/src/components/layouts/my_settings/config/my_settings_field_config.ts +61 -0
- package/src/components/layouts/my_settings/hooks/use_my_settings.ts +458 -0
- package/src/components/layouts/my_settings/index.tsx +351 -0
- package/src/components/layouts/register/config/register_field_config.ts +101 -0
- package/src/components/layouts/register/hooks/use_register_form.ts +272 -0
- package/src/components/layouts/register/index.tsx +208 -0
- package/src/components/layouts/reset_password/config/reset_password_field_config.ts +86 -0
- package/src/components/layouts/reset_password/hooks/use_reset_password_form.ts +276 -0
- package/src/components/layouts/reset_password/index.tsx +294 -0
- package/src/components/layouts/shared/components/already_logged_in_guard.tsx +95 -0
- package/src/components/layouts/shared/components/field_error_message.tsx +29 -0
- package/src/components/layouts/shared/components/form_action_buttons.tsx +64 -0
- package/src/components/layouts/shared/components/form_field_wrapper.tsx +44 -0
- package/src/components/layouts/shared/components/form_header.tsx +36 -0
- package/src/components/layouts/shared/components/logout_button.tsx +76 -0
- package/src/components/layouts/shared/components/password_field.tsx +72 -0
- package/src/components/layouts/shared/components/sidebar_layout_wrapper.tsx +264 -0
- package/src/components/layouts/shared/components/two_column_auth_layout.tsx +44 -0
- package/src/components/layouts/shared/components/unauthorized_guard.tsx +78 -0
- package/src/components/layouts/shared/components/visual_panel.tsx +41 -0
- package/src/components/layouts/shared/config/layout_customization.ts +95 -0
- package/src/components/layouts/shared/data/layout_data_client.ts +19 -0
- package/src/components/layouts/shared/hooks/use_auth_status.ts +103 -0
- package/src/components/layouts/shared/utils/ip_address.ts +37 -0
- package/src/components/layouts/shared/utils/validation.ts +66 -0
- package/src/components/ui/avatar.tsx +50 -0
- package/src/components/ui/button.tsx +57 -0
- package/src/components/ui/dialog.tsx +122 -0
- package/src/components/ui/hazo_ui_tooltip.tsx +67 -0
- package/src/components/ui/input.tsx +22 -0
- package/src/components/ui/label.tsx +26 -0
- package/src/components/ui/separator.tsx +31 -0
- package/src/components/ui/sheet.tsx +139 -0
- package/src/components/ui/sidebar.tsx +773 -0
- package/src/components/ui/skeleton.tsx +15 -0
- package/src/components/ui/sonner.tsx +31 -0
- package/src/components/ui/switch.tsx +29 -0
- package/src/components/ui/tabs.tsx +55 -0
- package/src/components/ui/tooltip.tsx +32 -0
- package/src/components/ui/vertical-tabs.tsx +59 -0
- package/src/hooks/use-mobile.tsx +19 -0
- package/src/lib/already_logged_in_config.server.ts +46 -0
- package/src/lib/app_logger.ts +24 -0
- package/src/lib/auth/auth_utils.server.ts +196 -0
- package/src/lib/auth/server_auth.ts +88 -0
- package/src/lib/config/config_loader.server.ts +149 -0
- package/src/lib/email_verification_config.server.ts +32 -0
- package/src/lib/file_types_config.server.ts +25 -0
- package/src/lib/forgot_password_config.server.ts +32 -0
- package/src/lib/hazo_connect_instance.server.ts +77 -0
- package/src/lib/hazo_connect_setup.server.ts +181 -0
- package/src/lib/hazo_connect_setup.ts +54 -0
- package/src/lib/login_config.server.ts +46 -0
- package/src/lib/messages_config.server.ts +45 -0
- package/src/lib/migrations/apply_migration.ts +105 -0
- package/src/lib/my_settings_config.server.ts +135 -0
- package/src/lib/password_requirements_config.server.ts +39 -0
- package/src/lib/profile_picture_config.server.ts +56 -0
- package/src/lib/register_config.server.ts +57 -0
- package/src/lib/reset_password_config.server.ts +75 -0
- package/src/lib/services/email_service.ts +581 -0
- package/src/lib/services/email_verification_service.ts +264 -0
- package/src/lib/services/login_service.ts +118 -0
- package/src/lib/services/password_change_service.ts +154 -0
- package/src/lib/services/password_reset_service.ts +405 -0
- package/src/lib/services/profile_picture_remove_service.ts +120 -0
- package/src/lib/services/profile_picture_service.ts +215 -0
- package/src/lib/services/profile_picture_source_mapper.ts +62 -0
- package/src/lib/services/registration_service.ts +163 -0
- package/src/lib/services/token_service.ts +240 -0
- package/src/lib/services/user_update_service.ts +128 -0
- package/src/lib/ui_sizes_config.server.ts +37 -0
- package/src/lib/user_fields_config.server.ts +31 -0
- package/src/lib/utils/api_route_helpers.ts +60 -0
- package/src/lib/utils.ts +11 -0
- package/src/middleware.ts +91 -0
- package/src/server/config/config_loader.ts +496 -0
- package/src/server/index.ts +38 -0
- package/src/server/logging/logger_service.ts +56 -0
- package/src/server/routes/root_router.ts +16 -0
- package/src/server/server.ts +28 -0
- package/src/server/types/app_types.ts +74 -0
- package/src/server/types/express.d.ts +15 -0
- package/src/stories/email_verification_layout.stories.tsx +137 -0
- package/src/stories/forgot_password_layout.stories.tsx +85 -0
- package/src/stories/login_layout.stories.tsx +85 -0
- package/src/stories/project_overview.stories.tsx +33 -0
- package/src/stories/register_layout.stories.tsx +107 -0
- package/tailwind.config.ts +77 -0
- package/tsconfig.json +27 -0
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
// file_description: service for profile picture management including default photo logic, Gravatar, and library photos
|
|
2
|
+
// section: imports
|
|
3
|
+
import type { HazoConnectAdapter } from "hazo_connect";
|
|
4
|
+
import { createCrudService } from "hazo_connect/server";
|
|
5
|
+
import gravatarUrl from "gravatar-url";
|
|
6
|
+
import { get_profile_picture_config } from "../profile_picture_config.server";
|
|
7
|
+
import { get_ui_sizes_config } from "../ui_sizes_config.server";
|
|
8
|
+
import { get_file_types_config } from "../file_types_config.server";
|
|
9
|
+
import { create_app_logger } from "../app_logger";
|
|
10
|
+
import path from "path";
|
|
11
|
+
import fs from "fs";
|
|
12
|
+
import { map_ui_source_to_db, type ProfilePictureSourceUI } from "./profile_picture_source_mapper";
|
|
13
|
+
|
|
14
|
+
// section: types
|
|
15
|
+
export type ProfilePictureSource = ProfilePictureSourceUI;
|
|
16
|
+
|
|
17
|
+
export type DefaultProfilePictureResult = {
|
|
18
|
+
profile_picture_url: string;
|
|
19
|
+
profile_source: ProfilePictureSource;
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
// section: helpers
|
|
23
|
+
/**
|
|
24
|
+
* Generates Gravatar URL from email address
|
|
25
|
+
* @param email - User's email address
|
|
26
|
+
* @param size - Image size in pixels (defaults to config value)
|
|
27
|
+
* @returns Gravatar URL
|
|
28
|
+
*/
|
|
29
|
+
export function get_gravatar_url(email: string, size?: number): string {
|
|
30
|
+
const uiSizes = get_ui_sizes_config();
|
|
31
|
+
const gravatarSize = size || uiSizes.gravatar_size;
|
|
32
|
+
return gravatarUrl(email, {
|
|
33
|
+
size: gravatarSize,
|
|
34
|
+
default: "identicon",
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Gets library photo categories by reading subdirectories
|
|
40
|
+
* @returns Array of category names
|
|
41
|
+
*/
|
|
42
|
+
export function get_library_categories(): string[] {
|
|
43
|
+
const config = get_profile_picture_config();
|
|
44
|
+
const library_path = path.resolve(process.cwd(), "public", config.library_photo_path.replace(/^\//, ""));
|
|
45
|
+
|
|
46
|
+
if (!fs.existsSync(library_path)) {
|
|
47
|
+
return [];
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
try {
|
|
51
|
+
const entries = fs.readdirSync(library_path, { withFileTypes: true });
|
|
52
|
+
return entries
|
|
53
|
+
.filter((entry) => entry.isDirectory())
|
|
54
|
+
.map((entry) => entry.name)
|
|
55
|
+
.sort();
|
|
56
|
+
} catch (error) {
|
|
57
|
+
const logger = create_app_logger();
|
|
58
|
+
const error_message = error instanceof Error ? error.message : "Unknown error";
|
|
59
|
+
logger.warn("profile_picture_service_read_categories_failed", {
|
|
60
|
+
filename: "profile_picture_service.ts",
|
|
61
|
+
line_number: 0,
|
|
62
|
+
library_path,
|
|
63
|
+
error: error_message,
|
|
64
|
+
});
|
|
65
|
+
return [];
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Gets photos in a specific library category
|
|
71
|
+
* @param category - Category name
|
|
72
|
+
* @returns Array of photo URLs (relative to public directory)
|
|
73
|
+
*/
|
|
74
|
+
export function get_library_photos(category: string): string[] {
|
|
75
|
+
const config = get_profile_picture_config();
|
|
76
|
+
const category_path = path.resolve(process.cwd(), "public", config.library_photo_path.replace(/^\//, ""), category);
|
|
77
|
+
|
|
78
|
+
if (!fs.existsSync(category_path)) {
|
|
79
|
+
return [];
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
try {
|
|
83
|
+
const fileTypes = get_file_types_config();
|
|
84
|
+
const allowedExtensions = fileTypes.allowed_image_extensions.map(ext => `.${ext.toLowerCase()}`);
|
|
85
|
+
const entries = fs.readdirSync(category_path, { withFileTypes: true });
|
|
86
|
+
const photos = entries
|
|
87
|
+
.filter((entry) => {
|
|
88
|
+
if (!entry.isFile()) return false;
|
|
89
|
+
const ext = path.extname(entry.name).toLowerCase();
|
|
90
|
+
return allowedExtensions.includes(ext);
|
|
91
|
+
})
|
|
92
|
+
.map((entry) => `${config.library_photo_path}/${category}/${entry.name}`)
|
|
93
|
+
.sort();
|
|
94
|
+
return photos;
|
|
95
|
+
} catch (error) {
|
|
96
|
+
const logger = create_app_logger();
|
|
97
|
+
const error_message = error instanceof Error ? error.message : "Unknown error";
|
|
98
|
+
logger.warn("profile_picture_service_read_photos_failed", {
|
|
99
|
+
filename: "profile_picture_service.ts",
|
|
100
|
+
line_number: 0,
|
|
101
|
+
category,
|
|
102
|
+
category_path,
|
|
103
|
+
error: error_message,
|
|
104
|
+
});
|
|
105
|
+
return [];
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* Gets default profile picture based on configuration priority
|
|
111
|
+
* @param user_email - User's email address
|
|
112
|
+
* @param user_name - User's name (optional)
|
|
113
|
+
* @returns Default profile picture URL and source, or null if no default available
|
|
114
|
+
*/
|
|
115
|
+
export function get_default_profile_picture(
|
|
116
|
+
user_email: string,
|
|
117
|
+
user_name?: string,
|
|
118
|
+
): DefaultProfilePictureResult | null {
|
|
119
|
+
const config = get_profile_picture_config();
|
|
120
|
+
|
|
121
|
+
if (!config.user_photo_default) {
|
|
122
|
+
return null;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
const uiSizes = get_ui_sizes_config();
|
|
126
|
+
|
|
127
|
+
// Try priority 1
|
|
128
|
+
if (config.user_photo_default_priority1 === "gravatar") {
|
|
129
|
+
const gravatar_url = get_gravatar_url(user_email, uiSizes.gravatar_size);
|
|
130
|
+
// Note: We can't check if Gravatar actually exists without making a request
|
|
131
|
+
// For now, we'll always return Gravatar URL and let the browser handle 404
|
|
132
|
+
return {
|
|
133
|
+
profile_picture_url: gravatar_url,
|
|
134
|
+
profile_source: "gravatar",
|
|
135
|
+
};
|
|
136
|
+
} else if (config.user_photo_default_priority1 === "library") {
|
|
137
|
+
const categories = get_library_categories();
|
|
138
|
+
if (categories.length > 0) {
|
|
139
|
+
// Use first category, first photo as default
|
|
140
|
+
const photos = get_library_photos(categories[0]);
|
|
141
|
+
if (photos.length > 0) {
|
|
142
|
+
return {
|
|
143
|
+
profile_picture_url: photos[0],
|
|
144
|
+
profile_source: "library",
|
|
145
|
+
};
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// Try priority 2 if priority 1 didn't work (only if priority2 is different from priority1)
|
|
151
|
+
const priority1 = config.user_photo_default_priority1;
|
|
152
|
+
const priority2 = config.user_photo_default_priority2;
|
|
153
|
+
|
|
154
|
+
if (priority2 && priority2 !== priority1) {
|
|
155
|
+
if (priority2 === "gravatar") {
|
|
156
|
+
const gravatar_url = get_gravatar_url(user_email, uiSizes.gravatar_size);
|
|
157
|
+
return {
|
|
158
|
+
profile_picture_url: gravatar_url,
|
|
159
|
+
profile_source: "gravatar",
|
|
160
|
+
};
|
|
161
|
+
} else if (priority2 === "library") {
|
|
162
|
+
const categories = get_library_categories();
|
|
163
|
+
if (categories.length > 0) {
|
|
164
|
+
const photos = get_library_photos(categories[0]);
|
|
165
|
+
if (photos.length > 0) {
|
|
166
|
+
return {
|
|
167
|
+
profile_picture_url: photos[0],
|
|
168
|
+
profile_source: "library",
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
// No default photo available
|
|
176
|
+
return null;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Updates user profile picture in database
|
|
181
|
+
* @param adapter - The hazo_connect adapter instance
|
|
182
|
+
* @param user_id - User ID
|
|
183
|
+
* @param profile_picture_url - Profile picture URL
|
|
184
|
+
* @param profile_source - Profile picture source type
|
|
185
|
+
* @returns Success status
|
|
186
|
+
*/
|
|
187
|
+
export async function update_user_profile_picture(
|
|
188
|
+
adapter: HazoConnectAdapter,
|
|
189
|
+
user_id: string,
|
|
190
|
+
profile_picture_url: string,
|
|
191
|
+
profile_source: ProfilePictureSource,
|
|
192
|
+
): Promise<{ success: boolean; error?: string }> {
|
|
193
|
+
try {
|
|
194
|
+
const users_service = createCrudService(adapter, "hazo_users");
|
|
195
|
+
const now = new Date().toISOString();
|
|
196
|
+
|
|
197
|
+
// Map UI source value to database enum value
|
|
198
|
+
const db_profile_source = map_ui_source_to_db(profile_source);
|
|
199
|
+
|
|
200
|
+
await users_service.updateById(user_id, {
|
|
201
|
+
profile_picture_url,
|
|
202
|
+
profile_source: db_profile_source,
|
|
203
|
+
changed_at: now,
|
|
204
|
+
});
|
|
205
|
+
|
|
206
|
+
return { success: true };
|
|
207
|
+
} catch (error) {
|
|
208
|
+
const error_message = error instanceof Error ? error.message : "Unknown error";
|
|
209
|
+
return {
|
|
210
|
+
success: false,
|
|
211
|
+
error: error_message,
|
|
212
|
+
};
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
// file_description: helper to map between UI profile picture source values and database enum values
|
|
2
|
+
// section: types
|
|
3
|
+
/**
|
|
4
|
+
* UI representation of profile picture source
|
|
5
|
+
* Used in components and API interfaces
|
|
6
|
+
*/
|
|
7
|
+
export type ProfilePictureSourceUI = "upload" | "library" | "gravatar" | "custom";
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Database enum values for profile_source
|
|
11
|
+
* Must match the CHECK constraint in the database
|
|
12
|
+
*/
|
|
13
|
+
export type ProfilePictureSourceDB = "gravatar" | "custom" | "predefined";
|
|
14
|
+
|
|
15
|
+
// section: helpers
|
|
16
|
+
/**
|
|
17
|
+
* Maps UI profile picture source to database enum value
|
|
18
|
+
* @param uiSource - UI representation of source ("upload", "library", "gravatar", "custom")
|
|
19
|
+
* @returns Database enum value ("gravatar", "custom", "predefined")
|
|
20
|
+
*/
|
|
21
|
+
export function map_ui_source_to_db(uiSource: ProfilePictureSourceUI): ProfilePictureSourceDB {
|
|
22
|
+
switch (uiSource) {
|
|
23
|
+
case "upload":
|
|
24
|
+
return "custom"; // User uploaded their own photo
|
|
25
|
+
case "library":
|
|
26
|
+
return "predefined"; // User selected from predefined library
|
|
27
|
+
case "gravatar":
|
|
28
|
+
return "gravatar"; // User's Gravatar
|
|
29
|
+
case "custom":
|
|
30
|
+
return "custom"; // Already in database format
|
|
31
|
+
default:
|
|
32
|
+
// Fallback to custom for unknown values
|
|
33
|
+
return "custom";
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Maps database enum value to UI representation
|
|
39
|
+
* @param dbSource - Database enum value ("gravatar", "custom", "predefined")
|
|
40
|
+
* @returns UI representation ("upload", "library", "gravatar", "custom")
|
|
41
|
+
*/
|
|
42
|
+
export function map_db_source_to_ui(dbSource: ProfilePictureSourceDB | string | null | undefined): ProfilePictureSourceUI {
|
|
43
|
+
if (!dbSource) {
|
|
44
|
+
return "custom"; // Default fallback
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
switch (dbSource) {
|
|
48
|
+
case "gravatar":
|
|
49
|
+
return "gravatar";
|
|
50
|
+
case "custom":
|
|
51
|
+
return "upload"; // Map custom to upload in UI (user uploaded their own)
|
|
52
|
+
case "predefined":
|
|
53
|
+
return "library"; // Map predefined to library in UI (user selected from library)
|
|
54
|
+
default:
|
|
55
|
+
// For unknown values, try to return as-is if it matches UI format
|
|
56
|
+
if (dbSource === "upload" || dbSource === "library") {
|
|
57
|
+
return dbSource as ProfilePictureSourceUI;
|
|
58
|
+
}
|
|
59
|
+
return "custom"; // Fallback
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
// file_description: service for user registration operations using hazo_connect
|
|
2
|
+
// section: imports
|
|
3
|
+
import type { HazoConnectAdapter } from "hazo_connect";
|
|
4
|
+
import { createCrudService } from "hazo_connect/server";
|
|
5
|
+
import argon2 from "argon2";
|
|
6
|
+
import { randomUUID } from "crypto";
|
|
7
|
+
import { create_token } from "./token_service";
|
|
8
|
+
import { get_default_profile_picture } from "./profile_picture_service";
|
|
9
|
+
import { get_profile_picture_config } from "../profile_picture_config.server";
|
|
10
|
+
import { map_ui_source_to_db } from "./profile_picture_source_mapper";
|
|
11
|
+
import { create_app_logger } from "../app_logger";
|
|
12
|
+
import { send_template_email } from "./email_service";
|
|
13
|
+
|
|
14
|
+
// section: types
|
|
15
|
+
export type RegistrationData = {
|
|
16
|
+
email: string;
|
|
17
|
+
password: string;
|
|
18
|
+
name?: string;
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
export type RegistrationResult = {
|
|
22
|
+
success: boolean;
|
|
23
|
+
user_id?: string;
|
|
24
|
+
error?: string;
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
// section: helpers
|
|
28
|
+
/**
|
|
29
|
+
* Registers a new user in the database using hazo_connect
|
|
30
|
+
* @param adapter - The hazo_connect adapter instance
|
|
31
|
+
* @param data - Registration data (email, password, optional name)
|
|
32
|
+
* @returns Registration result with success status and user_id or error
|
|
33
|
+
*/
|
|
34
|
+
export async function register_user(
|
|
35
|
+
adapter: HazoConnectAdapter,
|
|
36
|
+
data: RegistrationData,
|
|
37
|
+
): Promise<RegistrationResult> {
|
|
38
|
+
try {
|
|
39
|
+
const { email, password, name } = data;
|
|
40
|
+
|
|
41
|
+
// Create CRUD service for hazo_users table
|
|
42
|
+
const users_service = createCrudService(adapter, "hazo_users");
|
|
43
|
+
|
|
44
|
+
// Check if user already exists
|
|
45
|
+
const existing_users = await users_service.findBy({
|
|
46
|
+
email_address: email,
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
if (Array.isArray(existing_users) && existing_users.length > 0) {
|
|
50
|
+
return {
|
|
51
|
+
success: false,
|
|
52
|
+
error: "Email address already registered",
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Hash password using argon2
|
|
57
|
+
const password_hash = await argon2.hash(password);
|
|
58
|
+
|
|
59
|
+
// Generate user ID
|
|
60
|
+
const user_id = randomUUID();
|
|
61
|
+
const now = new Date().toISOString();
|
|
62
|
+
|
|
63
|
+
// Insert user into database using CRUD service
|
|
64
|
+
const insert_data: Record<string, unknown> = {
|
|
65
|
+
id: user_id,
|
|
66
|
+
email_address: email,
|
|
67
|
+
password_hash: password_hash,
|
|
68
|
+
email_verified: false,
|
|
69
|
+
is_active: true,
|
|
70
|
+
login_attempts: 0,
|
|
71
|
+
created_at: now,
|
|
72
|
+
changed_at: now,
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
// Include name if provided
|
|
76
|
+
if (name) {
|
|
77
|
+
insert_data.name = name;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Set default profile picture if enabled
|
|
81
|
+
const profile_picture_config = get_profile_picture_config();
|
|
82
|
+
if (profile_picture_config.user_photo_default) {
|
|
83
|
+
const default_photo = get_default_profile_picture(email, name);
|
|
84
|
+
if (default_photo) {
|
|
85
|
+
insert_data.profile_picture_url = default_photo.profile_picture_url;
|
|
86
|
+
// Map UI source value to database enum value
|
|
87
|
+
insert_data.profile_source = map_ui_source_to_db(default_photo.profile_source);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
const inserted_users = await users_service.insert(insert_data);
|
|
92
|
+
|
|
93
|
+
// Verify insertion was successful
|
|
94
|
+
if (!Array.isArray(inserted_users) || inserted_users.length === 0) {
|
|
95
|
+
return {
|
|
96
|
+
success: false,
|
|
97
|
+
error: "Failed to create user account",
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// Create email verification token for the new user
|
|
102
|
+
const token_result = await create_token({
|
|
103
|
+
adapter,
|
|
104
|
+
user_id,
|
|
105
|
+
token_type: "email_verification",
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
if (!token_result.success) {
|
|
109
|
+
// Log error but don't fail registration - token can be resent later
|
|
110
|
+
const logger = create_app_logger();
|
|
111
|
+
const error_message = token_result.error || "Unknown error";
|
|
112
|
+
logger.error("registration_service_token_creation_failed", {
|
|
113
|
+
filename: "registration_service.ts",
|
|
114
|
+
line_number: 0,
|
|
115
|
+
user_id,
|
|
116
|
+
error: error_message,
|
|
117
|
+
note: "This may be due to missing token_type column in hazo_refresh_tokens table. Please ensure migration 001_add_token_type_to_refresh_tokens.sql has been applied.",
|
|
118
|
+
});
|
|
119
|
+
} else {
|
|
120
|
+
const logger = create_app_logger();
|
|
121
|
+
logger.info("registration_service_token_created", {
|
|
122
|
+
filename: "registration_service.ts",
|
|
123
|
+
line_number: 0,
|
|
124
|
+
user_id,
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// Send verification email if token was created successfully
|
|
129
|
+
if (token_result.success && token_result.raw_token) {
|
|
130
|
+
const email_result = await send_template_email("email_verification", email, {
|
|
131
|
+
token: token_result.raw_token,
|
|
132
|
+
user_email: email,
|
|
133
|
+
user_name: name,
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
if (!email_result.success) {
|
|
137
|
+
const logger = create_app_logger();
|
|
138
|
+
logger.error("registration_service_email_send_failed", {
|
|
139
|
+
filename: "registration_service.ts",
|
|
140
|
+
line_number: 0,
|
|
141
|
+
user_id,
|
|
142
|
+
email,
|
|
143
|
+
error: email_result.error,
|
|
144
|
+
note: "User registration succeeded but verification email failed to send",
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
return {
|
|
150
|
+
success: true,
|
|
151
|
+
user_id,
|
|
152
|
+
};
|
|
153
|
+
} catch (error) {
|
|
154
|
+
const error_message =
|
|
155
|
+
error instanceof Error ? error.message : "Unknown error";
|
|
156
|
+
|
|
157
|
+
return {
|
|
158
|
+
success: false,
|
|
159
|
+
error: error_message,
|
|
160
|
+
};
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
// file_description: shared service for creating and managing tokens in hazo_refresh_tokens table
|
|
2
|
+
// section: imports
|
|
3
|
+
import type { HazoConnectAdapter } from "hazo_connect";
|
|
4
|
+
import { createCrudService } from "hazo_connect/server";
|
|
5
|
+
import { randomBytes, randomUUID } from "crypto";
|
|
6
|
+
import argon2 from "argon2";
|
|
7
|
+
import { read_config_section } from "../config/config_loader.server";
|
|
8
|
+
import { create_app_logger } from "../app_logger";
|
|
9
|
+
|
|
10
|
+
// section: types
|
|
11
|
+
export type TokenType = "refresh" | "password_reset" | "email_verification";
|
|
12
|
+
|
|
13
|
+
export type CreateTokenParams = {
|
|
14
|
+
adapter: HazoConnectAdapter;
|
|
15
|
+
user_id: string;
|
|
16
|
+
token_type: TokenType;
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
export type CreateTokenResult = {
|
|
20
|
+
success: boolean;
|
|
21
|
+
raw_token?: string;
|
|
22
|
+
error?: string;
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
// section: helpers
|
|
26
|
+
/**
|
|
27
|
+
* Gets token expiry hours from hazo_auth_config.ini for a specific token type
|
|
28
|
+
* Falls back to defaults if config is not found
|
|
29
|
+
* @param token_type - The type of token (refresh, password_reset, email_verification)
|
|
30
|
+
* @returns Number of hours until token expires
|
|
31
|
+
*/
|
|
32
|
+
function get_token_expiry_hours(token_type: TokenType): number {
|
|
33
|
+
const default_expiries: Record<TokenType, number> = {
|
|
34
|
+
refresh: 720, // 30 days
|
|
35
|
+
password_reset: 0.167, // 10 minutes
|
|
36
|
+
email_verification: 48, // 48 hours
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
const logger = create_app_logger();
|
|
40
|
+
const token_config_section = read_config_section("hazo_auth__tokens");
|
|
41
|
+
|
|
42
|
+
// Get expiry from config or environment variable or default
|
|
43
|
+
const config_key = `${token_type}_expiry_hours`;
|
|
44
|
+
const env_key = `HAZO_AUTH_${token_type.toUpperCase()}_TOKEN_EXPIRY_HOURS`;
|
|
45
|
+
|
|
46
|
+
const expiry_hours =
|
|
47
|
+
token_config_section?.[config_key] ||
|
|
48
|
+
process.env[env_key] ||
|
|
49
|
+
default_expiries[token_type];
|
|
50
|
+
|
|
51
|
+
return parseFloat(String(expiry_hours)) || default_expiries[token_type];
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Creates a token for a user and stores it in hazo_refresh_tokens table
|
|
56
|
+
* Invalidates any existing tokens of the same type for the user before creating a new one
|
|
57
|
+
* @param params - Token creation parameters (adapter, user_id, token_type)
|
|
58
|
+
* @returns Token creation result with raw_token (for sending to user) or error
|
|
59
|
+
*/
|
|
60
|
+
export async function create_token(
|
|
61
|
+
params: CreateTokenParams,
|
|
62
|
+
): Promise<CreateTokenResult> {
|
|
63
|
+
try {
|
|
64
|
+
const { adapter, user_id, token_type } = params;
|
|
65
|
+
|
|
66
|
+
// Create CRUD service for hazo_refresh_tokens table
|
|
67
|
+
const tokens_service = createCrudService(adapter, "hazo_refresh_tokens");
|
|
68
|
+
|
|
69
|
+
// Invalidate any existing tokens of this type for this user
|
|
70
|
+
// If token_type column doesn't exist, this will fail - catch and continue
|
|
71
|
+
let existing_tokens: unknown[] = [];
|
|
72
|
+
try {
|
|
73
|
+
existing_tokens = (await tokens_service.findBy({
|
|
74
|
+
user_id: user_id,
|
|
75
|
+
token_type: token_type,
|
|
76
|
+
})) as unknown[];
|
|
77
|
+
} catch (error) {
|
|
78
|
+
// If token_type column doesn't exist, try without it
|
|
79
|
+
// This is a fallback for databases that haven't had the migration applied
|
|
80
|
+
const logger = create_app_logger();
|
|
81
|
+
const error_message = error instanceof Error ? error.message : "Unknown error";
|
|
82
|
+
logger.warn("token_service_token_type_column_missing", {
|
|
83
|
+
filename: "token_service.ts",
|
|
84
|
+
line_number: 0,
|
|
85
|
+
user_id,
|
|
86
|
+
token_type,
|
|
87
|
+
error: error_message,
|
|
88
|
+
note: "token_type column may not exist, trying without filter",
|
|
89
|
+
});
|
|
90
|
+
// Try to find tokens by user_id only (less precise but works without migration)
|
|
91
|
+
try {
|
|
92
|
+
existing_tokens = (await tokens_service.findBy({
|
|
93
|
+
user_id: user_id,
|
|
94
|
+
})) as unknown[];
|
|
95
|
+
} catch (fallbackError) {
|
|
96
|
+
// If that also fails, log and continue (will just create new token)
|
|
97
|
+
const fallback_error_message = fallbackError instanceof Error ? fallbackError.message : "Unknown error";
|
|
98
|
+
logger.warn("token_service_query_existing_tokens_failed", {
|
|
99
|
+
filename: "token_service.ts",
|
|
100
|
+
line_number: 0,
|
|
101
|
+
user_id,
|
|
102
|
+
error: fallback_error_message,
|
|
103
|
+
note: "Could not query existing tokens, will create new token anyway",
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
if (Array.isArray(existing_tokens) && existing_tokens.length > 0) {
|
|
109
|
+
// Delete existing tokens (of this type if token_type exists, or all for user if not)
|
|
110
|
+
for (const token of existing_tokens) {
|
|
111
|
+
try {
|
|
112
|
+
await tokens_service.deleteById((token as { id: unknown }).id);
|
|
113
|
+
} catch (deleteError) {
|
|
114
|
+
const logger = create_app_logger();
|
|
115
|
+
const error_message = deleteError instanceof Error ? deleteError.message : "Unknown error";
|
|
116
|
+
logger.warn("token_service_delete_existing_token_failed", {
|
|
117
|
+
filename: "token_service.ts",
|
|
118
|
+
line_number: 0,
|
|
119
|
+
user_id,
|
|
120
|
+
token_id: (token as { id: unknown }).id,
|
|
121
|
+
error: error_message,
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// Generate a secure random token
|
|
128
|
+
const raw_token = randomBytes(32).toString("hex");
|
|
129
|
+
|
|
130
|
+
// Hash the token before storing
|
|
131
|
+
const token_hash = await argon2.hash(raw_token);
|
|
132
|
+
|
|
133
|
+
// Get expiry hours from config
|
|
134
|
+
const expiry_hours = get_token_expiry_hours(token_type);
|
|
135
|
+
|
|
136
|
+
// Calculate expiration time (convert hours to milliseconds)
|
|
137
|
+
const expires_at = new Date();
|
|
138
|
+
expires_at.setTime(expires_at.getTime() + expiry_hours * 60 * 60 * 1000);
|
|
139
|
+
|
|
140
|
+
const now = new Date().toISOString();
|
|
141
|
+
|
|
142
|
+
// Insert the token into the database
|
|
143
|
+
// Try with token_type first, fallback to without if column doesn't exist
|
|
144
|
+
let inserted_tokens: unknown[];
|
|
145
|
+
try {
|
|
146
|
+
inserted_tokens = (await tokens_service.insert({
|
|
147
|
+
id: randomUUID(),
|
|
148
|
+
user_id: user_id,
|
|
149
|
+
token_hash: token_hash,
|
|
150
|
+
token_type: token_type,
|
|
151
|
+
expires_at: expires_at.toISOString(),
|
|
152
|
+
created_at: now,
|
|
153
|
+
})) as unknown[];
|
|
154
|
+
} catch (error) {
|
|
155
|
+
// If token_type column doesn't exist, try without it
|
|
156
|
+
const logger = create_app_logger();
|
|
157
|
+
const error_message = error instanceof Error ? error.message : "Unknown error";
|
|
158
|
+
logger.warn("token_service_insert_with_token_type_failed", {
|
|
159
|
+
filename: "token_service.ts",
|
|
160
|
+
line_number: 0,
|
|
161
|
+
user_id,
|
|
162
|
+
token_type,
|
|
163
|
+
error: error_message,
|
|
164
|
+
note: "token_type column may not exist, inserting without it",
|
|
165
|
+
});
|
|
166
|
+
// Fallback: insert without token_type (will use default if column exists with default)
|
|
167
|
+
inserted_tokens = (await tokens_service.insert({
|
|
168
|
+
id: randomUUID(),
|
|
169
|
+
user_id: user_id,
|
|
170
|
+
token_hash: token_hash,
|
|
171
|
+
expires_at: expires_at.toISOString(),
|
|
172
|
+
created_at: now,
|
|
173
|
+
})) as unknown[];
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
// Verify insertion was successful
|
|
177
|
+
if (!Array.isArray(inserted_tokens) || inserted_tokens.length === 0) {
|
|
178
|
+
const logger = create_app_logger();
|
|
179
|
+
const error_msg = `Failed to create ${token_type} token - no rows inserted`;
|
|
180
|
+
logger.error("token_service_insertion_failed", {
|
|
181
|
+
filename: "token_service.ts",
|
|
182
|
+
line_number: 0,
|
|
183
|
+
user_id,
|
|
184
|
+
token_type,
|
|
185
|
+
error: error_msg,
|
|
186
|
+
});
|
|
187
|
+
return {
|
|
188
|
+
success: false,
|
|
189
|
+
error: error_msg,
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
const logger = create_app_logger();
|
|
194
|
+
logger.info("token_service_token_created", {
|
|
195
|
+
filename: "token_service.ts",
|
|
196
|
+
line_number: 0,
|
|
197
|
+
user_id,
|
|
198
|
+
token_type,
|
|
199
|
+
});
|
|
200
|
+
|
|
201
|
+
// Log raw token and test URLs in debug mode (logger handles dev mode)
|
|
202
|
+
logger.debug("token_service_raw_token", {
|
|
203
|
+
filename: "token_service.ts",
|
|
204
|
+
line_number: 0,
|
|
205
|
+
user_id,
|
|
206
|
+
token_type,
|
|
207
|
+
raw_token,
|
|
208
|
+
test_url: token_type === "email_verification"
|
|
209
|
+
? `/verify_email?token=${raw_token}`
|
|
210
|
+
: token_type === "password_reset"
|
|
211
|
+
? `/reset_password?token=${raw_token}`
|
|
212
|
+
: undefined,
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
return {
|
|
216
|
+
success: true,
|
|
217
|
+
raw_token,
|
|
218
|
+
};
|
|
219
|
+
} catch (error) {
|
|
220
|
+
const logger = create_app_logger();
|
|
221
|
+
const error_message =
|
|
222
|
+
error instanceof Error ? error.message : "Unknown error";
|
|
223
|
+
const error_stack = error instanceof Error ? error.stack : undefined;
|
|
224
|
+
|
|
225
|
+
logger.error("token_service_create_token_error", {
|
|
226
|
+
filename: "token_service.ts",
|
|
227
|
+
line_number: 0,
|
|
228
|
+
user_id: params.user_id,
|
|
229
|
+
token_type: params.token_type,
|
|
230
|
+
error: error_message,
|
|
231
|
+
error_stack,
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
return {
|
|
235
|
+
success: false,
|
|
236
|
+
error: error_message,
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
|