From 7fca0d8da5e56f626a1e08c5a66843b2ce717236 Mon Sep 17 00:00:00 2001 From: Brian Austin <13002992+brianjaustin@users.noreply.github.com> Date: Sat, 11 Nov 2023 20:03:32 -0500 Subject: [PATCH] fix: replace first and last name with single field (#4915) --- cli/src/api/open-api/api.ts | 104 +++++------------- docs/docs/administration/server-commands.md | 3 +- mobile/assets/i18n/en-US.json | 2 +- .../activities/models/activity.model.dart | 3 +- .../activities/views/activities_page.dart | 2 +- .../album/views/album_options_part.dart | 4 +- .../models/authentication_state.model.dart | 20 ++-- .../providers/authentication.provider.dart | 9 +- .../login/ui/change_password_form.dart | 3 +- .../lib/modules/partner/ui/partner_list.dart | 2 +- .../partner/views/partner_detail_page.dart | 4 +- .../modules/partner/views/partner_page.dart | 4 +- mobile/lib/shared/models/album.dart | 7 +- mobile/lib/shared/models/user.dart | 18 +-- mobile/lib/shared/models/user.g.dart | Bin 50751 -> 45544 bytes .../app_bar_dialog/app_bar_profile_info.dart | 2 +- mobile/lib/shared/ui/user_avatar.dart | 5 +- mobile/lib/shared/ui/user_circle_avatar.dart | 2 +- mobile/openapi/doc/CreateUserDto.md | Bin 647 -> 610 bytes mobile/openapi/doc/LoginResponseDto.md | Bin 656 -> 619 bytes mobile/openapi/doc/PartnerResponseDto.md | Bin 959 -> 922 bytes mobile/openapi/doc/SignUpDto.md | Bin 501 -> 464 bytes mobile/openapi/doc/UpdateUserDto.md | Bin 810 -> 762 bytes mobile/openapi/doc/UsageByUserDto.md | Bin 563 -> 522 bytes mobile/openapi/doc/UserDto.md | Bin 533 -> 496 bytes mobile/openapi/doc/UserResponseDto.md | Bin 913 -> 876 bytes mobile/openapi/lib/model/create_user_dto.dart | Bin 5034 -> 4726 bytes .../openapi/lib/model/login_response_dto.dart | Bin 4806 -> 4498 bytes .../lib/model/partner_response_dto.dart | Bin 7973 -> 7665 bytes mobile/openapi/lib/model/sign_up_dto.dart | Bin 3408 -> 3100 bytes mobile/openapi/lib/model/update_user_dto.dart | Bin 9150 -> 8398 bytes .../openapi/lib/model/usage_by_user_dto.dart | Bin 4000 -> 3644 bytes mobile/openapi/lib/model/user_dto.dart | Bin 3644 -> 3336 bytes .../openapi/lib/model/user_response_dto.dart | Bin 7213 -> 6905 bytes mobile/openapi/test/create_user_dto_test.dart | Bin 1193 -> 1082 bytes .../openapi/test/login_response_dto_test.dart | Bin 1319 -> 1208 bytes .../test/partner_response_dto_test.dart | Bin 2050 -> 1939 bytes mobile/openapi/test/sign_up_dto_test.dart | Bin 850 -> 739 bytes mobile/openapi/test/update_user_dto_test.dart | Bin 1502 -> 1391 bytes .../openapi/test/usage_by_user_dto_test.dart | Bin 1062 -> 943 bytes mobile/openapi/test/user_dto_test.dart | Bin 949 -> 838 bytes .../openapi/test/user_response_dto_test.dart | Bin 1938 -> 1827 bytes mobile/test/sync_service_test.dart | 3 +- server/immich-openapi-specs.json | 78 ++++--------- server/package-lock.json | 5 - .../commands/reset-admin-password.command.ts | 4 +- server/src/domain/auth/auth.dto.ts | 13 +-- server/src/domain/auth/auth.service.spec.ts | 5 +- server/src/domain/auth/auth.service.ts | 7 +- .../domain/partner/partner.service.spec.ts | 6 +- .../domain/repositories/user.repository.ts | 3 +- .../src/domain/server-info/server-info.dto.ts | 4 +- .../server-info/server-info.service.spec.ts | 18 +-- .../domain/server-info/server-info.service.ts | 3 +- .../domain/user/dto/create-user.dto.spec.ts | 12 +- server/src/domain/user/dto/create-user.dto.ts | 15 +-- server/src/domain/user/dto/update-user.dto.ts | 7 +- .../user/response-dto/user-response.dto.ts | 6 +- server/src/domain/user/user.service.spec.ts | 9 +- server/src/infra/entities/user.entity.ts | 5 +- .../1699322864544-UserNameConsolidation.ts | 21 ++++ .../src/infra/repositories/user.repository.ts | 3 +- server/test/e2e/activity.e2e-spec.ts | 9 +- server/test/e2e/album.e2e-spec.ts | 6 +- server/test/e2e/asset.e2e-spec.ts | 6 +- server/test/e2e/auth.e2e-spec.ts | 20 ++-- server/test/e2e/library.e2e-spec.ts | 6 +- server/test/e2e/partner.e2e-spec.ts | 6 +- server/test/e2e/server-info.e2e-spec.ts | 5 +- server/test/e2e/shared-link.e2e-spec.ts | 3 +- server/test/e2e/system-config.e2e-spec.ts | 3 +- server/test/e2e/user.e2e-spec.ts | 24 ++-- server/test/fixtures/auth.stub.ts | 15 +-- server/test/fixtures/user.stub.ts | 21 ++-- web/src/api/open-api/api.ts | 104 +++++------------- .../admin-page/delete-confirm-dialoge.svelte | 2 +- .../admin-page/restore-dialoge.svelte | 2 +- .../server-stats/server-stats-panel.svelte | 2 +- .../components/album-page/album-card.svelte | 3 +- .../album-page/album-options.svelte | 4 +- .../album-page/share-info-modal.svelte | 8 +- .../album-page/user-selection-modal.svelte | 5 +- .../asset-viewer/activity-viewer.svelte | 9 +- .../asset-viewer/detail-panel.svelte | 3 +- .../forms/admin-registration-form.svelte | 15 +-- .../components/forms/create-user-form.svelte | 15 +-- .../components/forms/edit-user-form.svelte | 23 +--- .../navigation-bar/account-info-panel.svelte | 3 +- .../navigation-bar/navigation-bar.svelte | 2 +- .../shared-components/user-avatar.svelte | 7 +- .../partner-selection-modal.svelte | 3 +- .../partner-settings.svelte | 11 +- .../user-profile-settings.svelte | 17 +-- .../(user)/partners/[userId]/+page.svelte | 3 +- web/src/routes/(user)/sharing/+page.svelte | 3 +- .../routes/admin/user-management/+page.svelte | 8 +- .../routes/auth/change-password/+page.svelte | 3 +- web/src/test-data/factories/user-factory.ts | 3 +- 98 files changed, 261 insertions(+), 542 deletions(-) create mode 100644 server/src/infra/migrations/1699322864544-UserNameConsolidation.ts diff --git a/cli/src/api/open-api/api.ts b/cli/src/api/open-api/api.ts index 496fa5441d..8fb2c1b3d1 100644 --- a/cli/src/api/open-api/api.ts +++ b/cli/src/api/open-api/api.ts @@ -1341,24 +1341,18 @@ export interface CreateUserDto { * @memberof CreateUserDto */ 'externalPath'?: string | null; - /** - * - * @type {string} - * @memberof CreateUserDto - */ - 'firstName': string; - /** - * - * @type {string} - * @memberof CreateUserDto - */ - 'lastName': string; /** * * @type {boolean} * @memberof CreateUserDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof CreateUserDto + */ + 'name': string; /** * * @type {string} @@ -2137,12 +2131,6 @@ export interface LoginResponseDto { * @memberof LoginResponseDto */ 'accessToken': string; - /** - * - * @type {string} - * @memberof LoginResponseDto - */ - 'firstName': string; /** * * @type {boolean} @@ -2154,7 +2142,7 @@ export interface LoginResponseDto { * @type {string} * @memberof LoginResponseDto */ - 'lastName': string; + 'name': string; /** * * @type {string} @@ -2391,12 +2379,6 @@ export interface PartnerResponseDto { * @memberof PartnerResponseDto */ 'externalPath': string | null; - /** - * - * @type {string} - * @memberof PartnerResponseDto - */ - 'firstName': string; /** * * @type {string} @@ -2415,18 +2397,18 @@ export interface PartnerResponseDto { * @memberof PartnerResponseDto */ 'isAdmin': boolean; - /** - * - * @type {string} - * @memberof PartnerResponseDto - */ - 'lastName': string; /** * * @type {boolean} * @memberof PartnerResponseDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof PartnerResponseDto + */ + 'name': string; /** * * @type {string} @@ -3431,13 +3413,7 @@ export interface SignUpDto { * @type {string} * @memberof SignUpDto */ - 'firstName': string; - /** - * - * @type {string} - * @memberof SignUpDto - */ - 'lastName': string; + 'name': string; /** * * @type {string} @@ -4380,12 +4356,6 @@ export interface UpdateUserDto { * @memberof UpdateUserDto */ 'externalPath'?: string; - /** - * - * @type {string} - * @memberof UpdateUserDto - */ - 'firstName'?: string; /** * * @type {string} @@ -4398,18 +4368,18 @@ export interface UpdateUserDto { * @memberof UpdateUserDto */ 'isAdmin'?: boolean; - /** - * - * @type {string} - * @memberof UpdateUserDto - */ - 'lastName'?: string; /** * * @type {boolean} * @memberof UpdateUserDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof UpdateUserDto + */ + 'name'?: string; /** * * @type {string} @@ -4447,12 +4417,6 @@ export interface UsageByUserDto { * @memberof UsageByUserDto */ 'usage': number; - /** - * - * @type {string} - * @memberof UsageByUserDto - */ - 'userFirstName': string; /** * * @type {string} @@ -4464,7 +4428,7 @@ export interface UsageByUserDto { * @type {string} * @memberof UsageByUserDto */ - 'userLastName': string; + 'userName': string; /** * * @type {number} @@ -4484,12 +4448,6 @@ export interface UserDto { * @memberof UserDto */ 'email': string; - /** - * - * @type {string} - * @memberof UserDto - */ - 'firstName': string; /** * * @type {string} @@ -4501,7 +4459,7 @@ export interface UserDto { * @type {string} * @memberof UserDto */ - 'lastName': string; + 'name': string; /** * * @type {string} @@ -4539,12 +4497,6 @@ export interface UserResponseDto { * @memberof UserResponseDto */ 'externalPath': string | null; - /** - * - * @type {string} - * @memberof UserResponseDto - */ - 'firstName': string; /** * * @type {string} @@ -4557,18 +4509,18 @@ export interface UserResponseDto { * @memberof UserResponseDto */ 'isAdmin': boolean; - /** - * - * @type {string} - * @memberof UserResponseDto - */ - 'lastName': string; /** * * @type {boolean} * @memberof UserResponseDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof UserResponseDto + */ + 'name': string; /** * * @type {string} diff --git a/docs/docs/administration/server-commands.md b/docs/docs/administration/server-commands.md index d8cd2119d2..fc235f5a58 100644 --- a/docs/docs/administration/server-commands.md +++ b/docs/docs/administration/server-commands.md @@ -51,8 +51,7 @@ immich-admin list-users { id: 'e65e6f88-2a30-4dbe-8dd9-1885f4889b53', email: 'immich@example.com.com', - firstName: 'Immich', - lastName: 'Admin', + name: 'Immich Admin', storageLabel: 'admin', externalPath: null, profileImagePath: 'upload/profile/e65e6f88-2a30-4dbe-8dd9-1885f4889b53/e65e6f88-2a30-4dbe-8dd9-1885f4889b53.jpg', diff --git a/mobile/assets/i18n/en-US.json b/mobile/assets/i18n/en-US.json index ccb4195c32..5a176ab96c 100644 --- a/mobile/assets/i18n/en-US.json +++ b/mobile/assets/i18n/en-US.json @@ -114,7 +114,7 @@ "cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)", "cache_settings_title": "Caching Settings", "change_password_form_confirm_password": "Confirm Password", - "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.", + "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.", "change_password_form_new_password": "New Password", "change_password_form_password_mismatch": "Passwords do not match", "change_password_form_reenter_new_password": "Re-enter New Password", diff --git a/mobile/lib/modules/activities/models/activity.model.dart b/mobile/lib/modules/activities/models/activity.model.dart index 417ba4a863..2db626f54f 100644 --- a/mobile/lib/modules/activities/models/activity.model.dart +++ b/mobile/lib/modules/activities/models/activity.model.dart @@ -48,8 +48,7 @@ class Activity { : ActivityType.like, user = User( email: dto.user.email, - firstName: dto.user.firstName, - lastName: dto.user.lastName, + name: dto.user.name, profileImagePath: dto.user.profileImagePath, id: dto.user.id, // Placeholder values diff --git a/mobile/lib/modules/activities/views/activities_page.dart b/mobile/lib/modules/activities/views/activities_page.dart index b744fc4762..1cfd48b5bc 100644 --- a/mobile/lib/modules/activities/views/activities_page.dart +++ b/mobile/lib/modules/activities/views/activities_page.dart @@ -61,7 +61,7 @@ class ActivitiesPage extends HookConsumerWidget { mainAxisSize: leftAlign ? MainAxisSize.min : MainAxisSize.max, children: [ Text( - "${activity.user.firstName} ${activity.user.lastName}", + activity.user.name, style: textStyle, overflow: TextOverflow.ellipsis, ), diff --git a/mobile/lib/modules/album/views/album_options_part.dart b/mobile/lib/modules/album/views/album_options_part.dart index 88f0d7d3ca..8d45d36b4f 100644 --- a/mobile/lib/modules/album/views/album_options_part.dart +++ b/mobile/lib/modules/album/views/album_options_part.dart @@ -124,7 +124,7 @@ class AlbumOptionsPage extends HookConsumerWidget { ) : const SizedBox(), title: Text( - album.owner.value?.firstName ?? "", + album.owner.value?.name ?? "", style: const TextStyle( fontWeight: FontWeight.bold, ), @@ -155,7 +155,7 @@ class AlbumOptionsPage extends HookConsumerWidget { radius: 22, ), title: Text( - user.firstName, + user.name, style: const TextStyle( fontWeight: FontWeight.bold, ), diff --git a/mobile/lib/modules/login/models/authentication_state.model.dart b/mobile/lib/modules/login/models/authentication_state.model.dart index 648670ca63..9dcd320c81 100644 --- a/mobile/lib/modules/login/models/authentication_state.model.dart +++ b/mobile/lib/modules/login/models/authentication_state.model.dart @@ -3,8 +3,7 @@ class AuthenticationState { final String userId; final String userEmail; final bool isAuthenticated; - final String firstName; - final String lastName; + final String name; final bool isAdmin; final bool shouldChangePassword; final String profileImagePath; @@ -13,8 +12,7 @@ class AuthenticationState { required this.userId, required this.userEmail, required this.isAuthenticated, - required this.firstName, - required this.lastName, + required this.name, required this.isAdmin, required this.shouldChangePassword, required this.profileImagePath, @@ -25,8 +23,7 @@ class AuthenticationState { String? userId, String? userEmail, bool? isAuthenticated, - String? firstName, - String? lastName, + String? name, bool? isAdmin, bool? shouldChangePassword, String? profileImagePath, @@ -36,8 +33,7 @@ class AuthenticationState { userId: userId ?? this.userId, userEmail: userEmail ?? this.userEmail, isAuthenticated: isAuthenticated ?? this.isAuthenticated, - firstName: firstName ?? this.firstName, - lastName: lastName ?? this.lastName, + name: name ?? this.name, isAdmin: isAdmin ?? this.isAdmin, shouldChangePassword: shouldChangePassword ?? this.shouldChangePassword, profileImagePath: profileImagePath ?? this.profileImagePath, @@ -46,7 +42,7 @@ class AuthenticationState { @override String toString() { - return 'AuthenticationState(deviceId: $deviceId, userId: $userId, userEmail: $userEmail, isAuthenticated: $isAuthenticated, firstName: $firstName, lastName: $lastName, isAdmin: $isAdmin, shouldChangePassword: $shouldChangePassword, profileImagePath: $profileImagePath)'; + return 'AuthenticationState(deviceId: $deviceId, userId: $userId, userEmail: $userEmail, isAuthenticated: $isAuthenticated, name: $name, isAdmin: $isAdmin, shouldChangePassword: $shouldChangePassword, profileImagePath: $profileImagePath)'; } @override @@ -58,8 +54,7 @@ class AuthenticationState { other.userId == userId && other.userEmail == userEmail && other.isAuthenticated == isAuthenticated && - other.firstName == firstName && - other.lastName == lastName && + other.name == name && other.isAdmin == isAdmin && other.shouldChangePassword == shouldChangePassword && other.profileImagePath == profileImagePath; @@ -71,8 +66,7 @@ class AuthenticationState { userId.hashCode ^ userEmail.hashCode ^ isAuthenticated.hashCode ^ - firstName.hashCode ^ - lastName.hashCode ^ + name.hashCode ^ isAdmin.hashCode ^ shouldChangePassword.hashCode ^ profileImagePath.hashCode; diff --git a/mobile/lib/modules/login/providers/authentication.provider.dart b/mobile/lib/modules/login/providers/authentication.provider.dart index ad21037867..14a0949631 100644 --- a/mobile/lib/modules/login/providers/authentication.provider.dart +++ b/mobile/lib/modules/login/providers/authentication.provider.dart @@ -26,8 +26,7 @@ class AuthenticationNotifier extends StateNotifier { deviceId: "", userId: "", userEmail: "", - firstName: '', - lastName: '', + name: '', profileImagePath: '', isAdmin: false, shouldChangePassword: false, @@ -117,8 +116,7 @@ class AuthenticationNotifier extends StateNotifier { deviceId: "", userId: "", userEmail: "", - firstName: '', - lastName: '', + name: '', profileImagePath: '', isAdmin: false, shouldChangePassword: false, @@ -208,8 +206,7 @@ class AuthenticationNotifier extends StateNotifier { isAuthenticated: true, userId: user.id, userEmail: user.email, - firstName: user.firstName, - lastName: user.lastName, + name: user.name, profileImagePath: user.profileImagePath, isAdmin: user.isAdmin, shouldChangePassword: shouldChangePassword, diff --git a/mobile/lib/modules/login/ui/change_password_form.dart b/mobile/lib/modules/login/ui/change_password_form.dart index 884a26b6da..560967821a 100644 --- a/mobile/lib/modules/login/ui/change_password_form.dart +++ b/mobile/lib/modules/login/ui/change_password_form.dart @@ -46,8 +46,7 @@ class ChangePasswordForm extends HookConsumerWidget { child: Text( 'change_password_form_description'.tr( namedArgs: { - 'firstName': authState.firstName, - 'lastName': authState.lastName, + 'name': authState.name, }, ), style: TextStyle( diff --git a/mobile/lib/modules/partner/ui/partner_list.dart b/mobile/lib/modules/partner/ui/partner_list.dart index b1111fdcf6..58810fec5a 100644 --- a/mobile/lib/modules/partner/ui/partner_list.dart +++ b/mobile/lib/modules/partner/ui/partner_list.dart @@ -24,7 +24,7 @@ class PartnerList extends HookConsumerWidget { contentPadding: const EdgeInsets.symmetric(horizontal: 12.0), leading: userAvatar(context, p, radius: 30), title: Text( - "${p.firstName} ${p.lastName}'s photos", + "${p.name}'s photos", style: TextStyle( fontWeight: FontWeight.bold, fontSize: 14, diff --git a/mobile/lib/modules/partner/views/partner_detail_page.dart b/mobile/lib/modules/partner/views/partner_detail_page.dart index 995eebd3fd..2aecdb35cc 100644 --- a/mobile/lib/modules/partner/views/partner_detail_page.dart +++ b/mobile/lib/modules/partner/views/partner_detail_page.dart @@ -25,7 +25,7 @@ class PartnerDetailPage extends HookConsumerWidget { return Scaffold( appBar: AppBar( - title: Text("${partner.firstName} ${partner.lastName}"), + title: Text(partner.name), elevation: 0, centerTitle: false, ), @@ -34,7 +34,7 @@ class PartnerDetailPage extends HookConsumerWidget { ? Padding( padding: const EdgeInsets.all(16), child: Text( - "It seems ${partner.firstName} does not have any photos...\n" + "It seems ${partner.name} does not have any photos...\n" "Or your server version does not match the app version."), ) : ImmichAssetGrid( diff --git a/mobile/lib/modules/partner/views/partner_page.dart b/mobile/lib/modules/partner/views/partner_page.dart index 61c6397466..bb567b62bb 100644 --- a/mobile/lib/modules/partner/views/partner_page.dart +++ b/mobile/lib/modules/partner/views/partner_page.dart @@ -41,7 +41,7 @@ class PartnerPage extends HookConsumerWidget { padding: const EdgeInsets.only(right: 8), child: userAvatar(context, u), ), - Text("${u.firstName} ${u.lastName}"), + Text(u.name), ], ), ), @@ -71,7 +71,7 @@ class PartnerPage extends HookConsumerWidget { return ConfirmDialog( title: "partner_page_stop_sharing_title", content: - "partner_page_stop_sharing_content".tr(args: [u.firstName]), + "partner_page_stop_sharing_content".tr(args: [u.name]), onOk: () => ref.read(partnerServiceProvider).removePartner(u), ); }, diff --git a/mobile/lib/shared/models/album.dart b/mobile/lib/shared/models/album.dart index ceba607423..26b1ab16e8 100644 --- a/mobile/lib/shared/models/album.dart +++ b/mobile/lib/shared/models/album.dart @@ -68,11 +68,8 @@ class Album { } final name = []; - if (owner.value?.firstName != null) { - name.add(owner.value!.firstName); - } - if (owner.value?.lastName != null) { - name.add(owner.value!.lastName); + if (owner.value?.name != null) { + name.add(owner.value!.name); } return name.join(' '); diff --git a/mobile/lib/shared/models/user.dart b/mobile/lib/shared/models/user.dart index 59f01c7dc0..a247fb21e3 100644 --- a/mobile/lib/shared/models/user.dart +++ b/mobile/lib/shared/models/user.dart @@ -11,8 +11,7 @@ class User { required this.id, required this.updatedAt, required this.email, - required this.firstName, - required this.lastName, + required this.name, required this.isAdmin, this.isPartnerSharedBy = false, this.isPartnerSharedWith = false, @@ -27,8 +26,7 @@ class User { : id = dto.id, updatedAt = dto.updatedAt, email = dto.email, - firstName = dto.firstName, - lastName = dto.lastName, + name = dto.name, isPartnerSharedBy = false, isPartnerSharedWith = false, profileImagePath = dto.profileImagePath, @@ -39,8 +37,7 @@ class User { : id = dto.id, updatedAt = dto.updatedAt, email = dto.email, - firstName = dto.firstName, - lastName = dto.lastName, + name = dto.name, isPartnerSharedBy = false, isPartnerSharedWith = false, profileImagePath = dto.profileImagePath, @@ -52,8 +49,7 @@ class User { String id; DateTime updatedAt; String email; - String firstName; - String lastName; + String name; bool isPartnerSharedBy; bool isPartnerSharedWith; bool isAdmin; @@ -72,8 +68,7 @@ class User { return id == other.id && updatedAt.isAtSameMomentAs(other.updatedAt) && email == other.email && - firstName == other.firstName && - lastName == other.lastName && + name == other.name && isPartnerSharedBy == other.isPartnerSharedBy && isPartnerSharedWith == other.isPartnerSharedWith && profileImagePath == other.profileImagePath && @@ -88,8 +83,7 @@ class User { id.hashCode ^ updatedAt.hashCode ^ email.hashCode ^ - firstName.hashCode ^ - lastName.hashCode ^ + name.hashCode ^ isPartnerSharedBy.hashCode ^ isPartnerSharedWith.hashCode ^ profileImagePath.hashCode ^ diff --git a/mobile/lib/shared/models/user.g.dart b/mobile/lib/shared/models/user.g.dart index af3bdadf628befc4c1ab6c419017c722d65dadcd..f5167c2a822bc6f6b05f92156316de99adeccf48 100644 GIT binary patch delta 689 zcmZ9IT}YEr9L72B-TUh1`@TE-B5k@C>}%#qW~q}RiJA&&R!}0WXlpm4eH3CLm{$>z zf$`MnMp{UVg2CpYE;3rwO6Vd4LFg)>7)C*MX@1mkgAC8bbDsZ?hlle&xl!H1m$FQ? zfE$7n?E=N;3M0Nr+fWd8A|;sN)b~Nwn{iWr4p;OREbGHq&})dLMcfiya7y(Ei4^@( zD~UoAG<*^%cR$5OvJmpLsR z1oKt_H8uraE5&uIhOgEt`%}hOUGq+J&H8MZZyOY%=OdwsJ8rUpxN~1v{eNb;kc||R80#*SOxE*ZBT_;i zkH6oHv=gs3qwvXf^z`Bw*F%wY&H4hRf?p$dOC#0>O|0+o;X1aQkR@zQ${5Z}U?wSY zKZOVPPOs-4breZ=qMuLcL=T%w4hbxq3Y4>W<`VfA>jsbL_@Yps9K-me358?|i;qP% zKG~~dbI%{@aCRyy^zg3$eRh?%t-J^@^<}4yRlJVqup>XqnLigSJ+t?1J6p>8$X~h< B0_*?) delta 1347 zcmZ`&NlX)A7|paZ(*m98ObZ5@A{7#$SO|-(8XF}61;K#g5>lWbLch{ypwsh}rAyr>5gLonf@qQQU4EH&1iUf+A)^8We%FP{u^{lh7uVw>M$T6psR0WH%YCuo-aB2^ghqD5I+p$=L9SK8~}D9q~*pzB4YYlt$qP zYegZGi25=b%S=3#i*rmu@Sl_2VQ%w>bYEsgCP)OEm$0cQHHo;$<|E#ehGzuVm>Lb4 z8Pc375V~2XB?TRpRJ|D?qy>wCiqf`s@4{(rE0u*=E)@>mf;BD~r96jZJ_(n22Tt-X zJm;fO&0FQ=44VbEJfFwa(a_q|*psfTighl5%EvsPXDsk}Fl&j$K5HV`3Qg8GA;)DE zoKz+}!gezBu#iP%b|N4YAYHU!UC4Ii`rIzB*VXKA>u&Nyy2wvE4^nQ-^SMzd zW|L{x#X_n8Z^axrz1dbm!6|S!6>US8-Gu~^L%rRGX@alWSCiBaxpd^E3TJBj^aK-oJXBZ#i)lYY1O-ao6ET!t4XW}y zzGbwiMsLk475+5^uUlV)7k0NTgct6$Yr;m|km?z#*WfmPL&zfc^M+Lq5v05M&N$_k zk>m;@D`~e!&K-u9Q|t653TD;{)yRsAB-Z`9iR9I31+Dt#bxyaEpbAZLDu~2X5DBAO zNl2Z><0?T~s@yP07aFRPm}2<}qhjEwdJNv%B@p`Ah@cRB+tjEgmJe1)ie)hzmbf^osk^JzTG#)6X{s4<6^&9{I diff --git a/mobile/lib/shared/ui/app_bar_dialog/app_bar_profile_info.dart b/mobile/lib/shared/ui/app_bar_dialog/app_bar_profile_info.dart index 4f1f7db86d..c4951e3393 100644 --- a/mobile/lib/shared/ui/app_bar_dialog/app_bar_profile_info.dart +++ b/mobile/lib/shared/ui/app_bar_dialog/app_bar_profile_info.dart @@ -132,7 +132,7 @@ class AppBarProfileInfoBox extends HookConsumerWidget { ), ), title: Text( - "${authState.firstName} ${authState.lastName}", + authState.name, style: TextStyle( color: context.primaryColor, fontWeight: FontWeight.bold, diff --git a/mobile/lib/shared/ui/user_avatar.dart b/mobile/lib/shared/ui/user_avatar.dart index 382e44effe..95f76de43f 100644 --- a/mobile/lib/shared/ui/user_avatar.dart +++ b/mobile/lib/shared/ui/user_avatar.dart @@ -7,8 +7,7 @@ import 'package:immich_mobile/shared/models/user.dart'; Widget userAvatar(BuildContext context, User u, {double? radius}) { final url = "${Store.get(StoreKey.serverEndpoint)}/user/profile-image/${u.id}"; - final firstNameFirstLetter = u.firstName.isNotEmpty ? u.firstName[0] : ""; - final lastNameFirstLetter = u.lastName.isNotEmpty ? u.lastName[0] : ""; + final nameFirstLetter = u.name.isNotEmpty ? u.name[0] : ""; return CircleAvatar( radius: radius, backgroundColor: context.primaryColor.withAlpha(50), @@ -19,6 +18,6 @@ Widget userAvatar(BuildContext context, User u, {double? radius}) { ), // silence errors if user has no profile image, use initials as fallback onForegroundImageError: (exception, stackTrace) {}, - child: Text((firstNameFirstLetter + lastNameFirstLetter).toUpperCase()), + child: Text(nameFirstLetter.toUpperCase()), ); } diff --git a/mobile/lib/shared/ui/user_circle_avatar.dart b/mobile/lib/shared/ui/user_circle_avatar.dart index 98df36265c..5920758a7f 100644 --- a/mobile/lib/shared/ui/user_circle_avatar.dart +++ b/mobile/lib/shared/ui/user_circle_avatar.dart @@ -43,7 +43,7 @@ class UserCircleAvatar extends ConsumerWidget { '${Store.get(StoreKey.serverEndpoint)}/user/profile-image/${user.id}?d=${Random().nextInt(1024)}'; final textIcon = Text( - user.firstName[0].toUpperCase(), + user.name[0].toUpperCase(), style: TextStyle( fontWeight: FontWeight.bold, color: context.isDarkTheme ? Colors.black : Colors.white, diff --git a/mobile/openapi/doc/CreateUserDto.md b/mobile/openapi/doc/CreateUserDto.md index 0a25b4df4d8034d3413c2d27abf40f07e9cacf4e..08fa8665a1c5312bad00d6d5e2f753fe5d5b2865 100644 GIT binary patch delta 24 gcmZo?eZ;asg>iB!qXA1^Vs7eWMn=)elNi4O0AG*?+W-In delta 45 scmaFF($2a;g^@EYv#7YlFEKZDG9#lXH$^ZZW diff --git a/mobile/openapi/doc/LoginResponseDto.md b/mobile/openapi/doc/LoginResponseDto.md index e344ef124a52d65f95eb1d3a7c2b7deffd4c93cc..bd37dc420027e3d99302614e716c105a3c48ba47 100644 GIT binary patch delta 17 YcmbQh`kH0Jzsd59vW$6~9U1)@0Xtp=mjD0& delta 41 pcmaFOGJ$o%KhCtwqT&+2#N5=$8yLkV{*mX%Nks5Bn=+a)0sv#K4}t&y diff --git a/mobile/openapi/doc/PartnerResponseDto.md b/mobile/openapi/doc/PartnerResponseDto.md index 887664e89565a350d9905710362c086b13b301ee..f7133bbf7eb7e6ffd3d71163ea2998fd4bfaa2f1 100644 GIT binary patch delta 26 icmdnbK8t-r6XWE?j7gJKnG6{7CUY@KZw_V3Vgvwo+XuV= delta 47 ucmbQmzMp+V6C-C@W>Im8Ut(_Rs diff --git a/mobile/openapi/doc/SignUpDto.md b/mobile/openapi/doc/SignUpDto.md index 92294df026606c0b46fe72b19818197ff942b2de..9706c7c54cb82994255268aa5ec43ba2f0c34bec 100644 GIT binary patch delta 12 Tcmey$e1UnwN5;I#;*9qIB$x$7 delta 48 ycmcb>{FQmaN42!fqT&+2#N1RZErl8dEv?{^qRhN>FbBxz($dOF1j diff --git a/mobile/openapi/doc/UpdateUserDto.md b/mobile/openapi/doc/UpdateUserDto.md index ff6bc8d4299ef4bb536d7d6f9d518c3505a360d0..ffbe11253a744a833240a3cabe6c3034eb89d00d 100644 GIT binary patch delta 28 kcmZ3*_KS6cKI7zQMyJUq7!6qR5_3~0F)45U!Dz$?0Fh@2hyVZp delta 55 ycmeyxx{7UsJ|kyZW>Im8Ut(_RWJ5;L$-5Z&ChIdgapWYT2^KOMY(Bwg#0UTfx)Ia> diff --git a/mobile/openapi/doc/UsageByUserDto.md b/mobile/openapi/doc/UsageByUserDto.md index c1c5240b99a316cf54fd74e1a08981ebbbb9248b..f7b1b3593183622b954a8c14b6340dc5b967677d 100644 GIT binary patch delta 15 WcmdnY(#5hth;gzxquk_3##;a-D+IX! delta 30 icmeBT+03#*h>_JTv#7XavH_#aWIjeE7N0~g=N15l)(G$b diff --git a/mobile/openapi/doc/UserDto.md b/mobile/openapi/doc/UserDto.md index 617ceb9d3bb5a52753df07404e5ecfa2e287c737..c8b750a1d3750b61bb9d4defd2b80c4cad2b15d9 100644 GIT binary patch delta 16 YcmbQr@_~86`^f@~QjB?%^%%pBjOLTSGa4}FO}@h@&6b~7T9V;8c_E|ZW;Z5pMgRy)4B!9& delta 47 wcmaFEHj#ZpJtJpYW>Im8Ut(_Ri_@% diff --git a/mobile/openapi/lib/model/create_user_dto.dart b/mobile/openapi/lib/model/create_user_dto.dart index 1345ac995cfffb78eec6a58b9d3fd93435a6df01..b33a3043c938a5b638a7b038d7fb18620fb5a67f 100644 GIT binary patch delta 150 zcmZ3b{!L}WYR1V=7&RF4ChunypUlG)vbmF~j*%rVF*kMcZ|3KdwO9>#KpX{ITLrM7 z59^1?g=`KG74zBlPHtmY;sVLos({GN%h-(>C+l)95eD(pV-;)_N-{Ew^*}u9&4)Q> eF-^YB;|bB7$E(N=R;8)9S&BD?4d{p-VL1TJax-uM delta 448 zcmeySvPyl!YDVLt)WXutqSO?Hl8nq^y|m1t;u62a+*BPd1t36`%t?gEO}J-9;;xB79`dXcPeBR=jTNislyCFsDz1FPd4CKx%n4I4-*^2 yyvY~1RiqI{YbruQ7Zk!^Js^2*m~M4V#mP0?!JAomg4o0mnjl&rwoZ-^k^=xPK$t}U diff --git a/mobile/openapi/lib/model/login_response_dto.dart b/mobile/openapi/lib/model/login_response_dto.dart index 94da05a50c543de02e73b02d0354e8f4166030a5..0293d04dec198444264b4eb2b2bc3fd299fd63f4 100644 GIT binary patch delta 171 zcmX@6I!SrMM#jnK7^NBWCck4`A(@n)pQDgj?3j|9nP<(#rJxX8Qk0pOu8_C+1QQeE zqoOy}4sS38XKp{mIWyZahy83MCnt#d>+0f3i0)P2S061vH_aON9|=B)2@f093!a4$yuD1@*kisX~bWP4hP1 delta 349 zcmbQFd`xx2Mn<-@%%b8FzsYAA#V2oMlxEFIgz#1f1eX+L=A|n@8{?KFy*Hwdp&{EpxcItqPo}qmWtb zn39{BXR88b=s=7D8f&WpG4vHH6Qc}VhkC4ntwKpgX0aYzX!2b)GjW(>5b_Y+O6(0x tFyG(hR6%mxR8DuO>ww}=FEeq;v#Y`lR@dQD00MQW(_mb%!IKRI69GI6c((um diff --git a/mobile/openapi/lib/model/partner_response_dto.dart b/mobile/openapi/lib/model/partner_response_dto.dart index ae6e230982f5aca82c7b1d78eaedb4d28a22f5f2..5ccef68d8586784caf964fcdad072316d2722696 100644 GIT binary patch delta 183 zcmZ2#_tAO-H`8PdCX2~6Od5=NlLeW?Cof_8y?G0B7vtt2);dO(yu{qp%}3bU7$@g* z#83XnVaNj#QLwdD0Lut-USOR3k1Kg{HMavq?Mm(d#>xD=Mw3H%mAF7^Y*oN2OL?1_ zCMydZn#?D-L>MHZ9;;xhP?C{ZtOw#*Z(bsJoq6(Z(bbceh*dM-pw~ZR delta 399 zcmexpz0__4Hxox%W>Im8Ut;cLMJBPyJWS@3moxHl~lNmYkL|{5VGBEb!1P+JI zyEzUrvO&~NR^S$$T*H;j3RVE--RE-HY|3rNC;~UlRt3(ST*zZ01T)N51;(1Zm`7#v zWu8VRS-48|SOr^!l8nq^J-CqdWHV0Fl1MxX27Htzc`bfZ(X9fels2FUd$Pf|v%h38V(1wUwz4p;|8^u{gsy zKP6RDK^v@11ENTTIZYLAq^$~^sRJ6xDR$x<@M<$!kWAm<&HiTs>C7R-q&#vsf>0a)Ov5 iP-dRE;^ruc3Cxop`0U(s+)f>>M?IF<4I-Yge%&t z$agwjH^1fD!Yl??reJHU0Ow4e#OE#oGZ-WTV^2QC=df9o z-;hxbt`;PnSDKTfV6R}HUs$zcL6GBAy3s=&fvMY9APHnRxwGK#Jrd!fSG#s3IFgcJc&K ziOFmt3lU)l)F}=#9ib88N}xVS&;hk6PF^lDd-F!o3Cu{=o{~^R;+aa8Bg_R#Avg;p LLpN(lEnx!yA^M=A diff --git a/mobile/openapi/lib/model/usage_by_user_dto.dart b/mobile/openapi/lib/model/usage_by_user_dto.dart index 052ebaa9fcf46cc084347b8a21faec8658885c9c..27d8a26730fa9fad831fb4fc2e2072420ade27c3 100644 GIT binary patch delta 161 zcmZ1=zei@nVaCZ%7^NmNF$GS}VG^3$#B^+ODYHC}Ut(^mf~~DWX>n@Nk&BC0^{VhjAD#=lTR>uO;%^(1G21`R!nAR7U#@M%uQ9WwN=QQtj|1e@(X4W zpd2Gh4u58ftx9H!jsi&DRwZxpES4pcGg)Q%L45UC1zUxZjLc%ayv=i2PcTki%5DVI ec86VsF>f;~$46G~%oKGUE(IV^&zmg9rv?D#3n*v+ delta 343 zcmeB>+9R`J0wZTyW>Im8Ut(_RwD4l$e pO$EtGjcoQ%CjrGFtj*8ZKC&Wwrmh1F5(NeI%oG?4;-E)7Y5-)7cG~~| diff --git a/mobile/openapi/lib/model/user_response_dto.dart b/mobile/openapi/lib/model/user_response_dto.dart index 488c9aafd288549cdfaf75134bcbb96445e9238c..73b30881f82533d90a0ad71c88b337c63d556606 100644 GIT binary patch delta 199 zcmZ2$@zZnz6VqgQCgsTo`9vlcF^RC`CFZ71R%8;H+{W~K@-$|N&4-xl7{Q{OO<0>5 zC*NVSm~75&$OBTRU~8)YqBlpfUtpZbBffnPo60)2^8KgT?POVO+StR delta 386 zcmexqy4GR?6BAonW>Im8-()`~@ySd~%9F#H_&9PBfg*{ysgvcI^d_e={pN(I0f~e# ztMCVx6lLb6E5KB3E@p0K+|0&W&L{>~r(kQV0Ow53V{;dQsRqfw*prvBIc#QRKgbAk zgaN1M`I4x3#$tr#3}ukX4+Y7hR+d(~M9F6PaAcyK1v1-vMTrEB6bj@W72x k*HoPRU$6}7WS|rx^fH9PHh&Q+Wk(9WL@7n6%4VrD0B$sf8vp5lA_GKbcM9cqT&+2$pXwOoDgPWZtCQ}j3%r(iBNG4Fbg8AJ9#=& J#pE33a{zLT8Cd`T diff --git a/mobile/openapi/test/login_response_dto_test.dart b/mobile/openapi/test/login_response_dto_test.dart index 408b7fade6be8e2737a914891442df84fe7afac2..a8365ff062c0e8ada193854836e3cf492fb2fd5b 100644 GIT binary patch delta 27 jcmZ3^wS#kmJ=5f6Oc^YBiMgqhFR&;w=54lNe$5B~h?WU~ delta 64 zcmdnNxtwc*J=5d`OzfO#nMK7Veu=rMlM`7K*&w{hXPFEp+cRZwIm8Ut(_RyC^NZl7j~HM8 diff --git a/mobile/openapi/test/update_user_dto_test.dart b/mobile/openapi/test/update_user_dto_test.dart index 511de33af03a03f4360ba2a73e0b46d9334a13b6..039a1a2446551d9bb7bff3472e8352424374109c 100644 GIT binary patch delta 34 ncmcb|{hn*XJf_L4%$q06vQ)6-CFZ71He^u*Gd3r()G`79*&z#d delta 76 zcmaFQb&q?)JSNVx%%b8Fzr@_s$qSei*&w{h@0raf&tuv=S)PTRBPS6d&kE)Mr41&3 QU}Bvd#$3Evjm47@09G>_bpQYW diff --git a/mobile/openapi/test/usage_by_user_dto_test.dart b/mobile/openapi/test/usage_by_user_dto_test.dart index 685d49e9f907431c2288f66289136eabeb85e7ca..51becb06f0535a8b1f3775cfeb96472b0da4303e 100644 GIT binary patch delta 37 qcmZ3+v7UXyWTweX%!w?1iMgqh&oe7>loqEJfjR#f4JV&ujs*bp=?!)O delta 53 vcmZ3_zKmnTWF}U(%%b9w$zPZ>p)6Tu!^tn1xF=6$N@Vd#gh)XcvdohJ{|pk7 diff --git a/mobile/openapi/test/user_dto_test.dart b/mobile/openapi/test/user_dto_test.dart index fa1b7da8669a86636fba8fc4e2dcbcaf33084fde..e0866d12dc6c66132890729ceaed31d01b3421f8 100644 GIT binary patch delta 27 hcmdnWevEB{Jk#Vhrbx!T$*N2$K+>Pde6kC3Apmz(2k-y@ delta 59 ycmX@cwv~N@JQHVHW>Im8Ut(_RWJM-LHVAL>JSOwW@=TE&If)24Rxk%BwE_T5C=<*8 diff --git a/mobile/openapi/test/user_response_dto_test.dart b/mobile/openapi/test/user_response_dto_test.dart index 9e2095607b083b701f3264bb5a6c3a6735ea88bc..28830a73fd91c4a1f38eaed4de583ebbc82316ef 100644 GIT binary patch delta 46 ycmbQlznE`B8S~`3%$q0cv#?J7#Zt_YmzbM6`2njUgdxbLJ9z>N`{rC$RVDzDtPiCC delta 82 zcmZ3?H;I2k88c^EW>Im8Ut(_RcKT&6D$)**S6&5lRpoUMAhi=UG`N N>#-DX{>P%q1OTJ78~^|S diff --git a/mobile/test/sync_service_test.dart b/mobile/test/sync_service_test.dart index b2543c6635..9abb863522 100644 --- a/mobile/test/sync_service_test.dart +++ b/mobile/test/sync_service_test.dart @@ -62,8 +62,7 @@ void main() { id: "1", updatedAt: DateTime.now(), email: "a@b.c", - firstName: "first", - lastName: "last", + name: "first last", isAdmin: false, ); setUpAll(() async { diff --git a/server/immich-openapi-specs.json b/server/immich-openapi-specs.json index 782b69649a..b776e4b28c 100644 --- a/server/immich-openapi-specs.json +++ b/server/immich-openapi-specs.json @@ -6844,15 +6844,12 @@ "nullable": true, "type": "string" }, - "firstName": { - "type": "string" - }, - "lastName": { - "type": "string" - }, "memoriesEnabled": { "type": "boolean" }, + "name": { + "type": "string" + }, "password": { "type": "string" }, @@ -6864,8 +6861,7 @@ "required": [ "email", "password", - "firstName", - "lastName" + "name" ], "type": "object" }, @@ -7463,13 +7459,10 @@ "accessToken": { "type": "string" }, - "firstName": { - "type": "string" - }, "isAdmin": { "type": "boolean" }, - "lastName": { + "name": { "type": "string" }, "profileImagePath": { @@ -7489,8 +7482,7 @@ "accessToken", "userId", "userEmail", - "firstName", - "lastName", + "name", "profileImagePath", "isAdmin", "shouldChangePassword" @@ -7656,9 +7648,6 @@ "nullable": true, "type": "string" }, - "firstName": { - "type": "string" - }, "id": { "type": "string" }, @@ -7668,12 +7657,12 @@ "isAdmin": { "type": "boolean" }, - "lastName": { - "type": "string" - }, "memoriesEnabled": { "type": "boolean" }, + "name": { + "type": "string" + }, "oauthId": { "type": "string" }, @@ -7694,8 +7683,7 @@ }, "required": [ "id", - "firstName", - "lastName", + "name", "email", "profileImagePath", "storageLabel", @@ -8464,14 +8452,10 @@ "example": "testuser@email.com", "type": "string" }, - "firstName": { + "name": { "example": "Admin", "type": "string" }, - "lastName": { - "example": "Doe", - "type": "string" - }, "password": { "example": "password", "type": "string" @@ -8480,8 +8464,7 @@ "required": [ "email", "password", - "firstName", - "lastName" + "name" ], "type": "object" }, @@ -9163,9 +9146,6 @@ "externalPath": { "type": "string" }, - "firstName": { - "type": "string" - }, "id": { "format": "uuid", "type": "string" @@ -9173,12 +9153,12 @@ "isAdmin": { "type": "boolean" }, - "lastName": { - "type": "string" - }, "memoriesEnabled": { "type": "boolean" }, + "name": { + "type": "string" + }, "password": { "type": "string" }, @@ -9203,13 +9183,10 @@ "format": "int64", "type": "integer" }, - "userFirstName": { - "type": "string" - }, "userId": { "type": "string" }, - "userLastName": { + "userName": { "type": "string" }, "videos": { @@ -9218,8 +9195,7 @@ }, "required": [ "userId", - "userFirstName", - "userLastName", + "userName", "photos", "videos", "usage" @@ -9231,13 +9207,10 @@ "email": { "type": "string" }, - "firstName": { - "type": "string" - }, "id": { "type": "string" }, - "lastName": { + "name": { "type": "string" }, "profileImagePath": { @@ -9246,8 +9219,7 @@ }, "required": [ "id", - "firstName", - "lastName", + "name", "email", "profileImagePath" ], @@ -9271,21 +9243,18 @@ "nullable": true, "type": "string" }, - "firstName": { - "type": "string" - }, "id": { "type": "string" }, "isAdmin": { "type": "boolean" }, - "lastName": { - "type": "string" - }, "memoriesEnabled": { "type": "boolean" }, + "name": { + "type": "string" + }, "oauthId": { "type": "string" }, @@ -9306,8 +9275,7 @@ }, "required": [ "id", - "firstName", - "lastName", + "name", "email", "profileImagePath", "storageLabel", diff --git a/server/package-lock.json b/server/package-lock.json index 70c213b68f..218fc6edd6 100644 --- a/server/package-lock.json +++ b/server/package-lock.json @@ -16845,11 +16845,6 @@ "luxon": "^3.2.1" } }, - "cron-validator": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/cron-validator/-/cron-validator-1.3.1.tgz", - "integrity": "sha512-C1HsxuPCY/5opR55G5/WNzyEGDWFVG+6GLrA+fW/sCTcP6A6NTjUP2AK7B8n2PyFs90kDG2qzwm8LMheADku6A==" - }, "cross-spawn": { "version": "7.0.3", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", diff --git a/server/src/admin-cli/commands/reset-admin-password.command.ts b/server/src/admin-cli/commands/reset-admin-password.command.ts index f28780a7d9..af36c590c9 100644 --- a/server/src/admin-cli/commands/reset-admin-password.command.ts +++ b/server/src/admin-cli/commands/reset-admin-password.command.ts @@ -15,12 +15,12 @@ export class ResetAdminPasswordCommand extends CommandRunner { async run(): Promise { const ask = (admin: UserResponseDto) => { - const { id, oauthId, email, firstName, lastName } = admin; + const { id, oauthId, email, name } = admin; console.log(`Found Admin: - ID=${id} - OAuth ID=${oauthId} - Email=${email} -- Name=${firstName} ${lastName}`); +- Name=${name}`); return this.inquirer.ask<{ password: string }>('prompt-password', undefined).then(({ password }) => password); }; diff --git a/server/src/domain/auth/auth.dto.ts b/server/src/domain/auth/auth.dto.ts index c0703911d5..d5ee275ad2 100644 --- a/server/src/domain/auth/auth.dto.ts +++ b/server/src/domain/auth/auth.dto.ts @@ -33,8 +33,7 @@ export class LoginResponseDto { accessToken!: string; userId!: string; userEmail!: string; - firstName!: string; - lastName!: string; + name!: string; profileImagePath!: string; isAdmin!: boolean; shouldChangePassword!: boolean; @@ -45,8 +44,7 @@ export function mapLoginResponse(entity: UserEntity, accessToken: string): Login accessToken: accessToken, userId: entity.id, userEmail: entity.email, - firstName: entity.firstName, - lastName: entity.lastName, + name: entity.name, isAdmin: entity.isAdmin, profileImagePath: entity.profileImagePath, shouldChangePassword: entity.shouldChangePassword, @@ -62,12 +60,7 @@ export class SignUpDto extends LoginCredentialDto { @IsString() @IsNotEmpty() @ApiProperty({ example: 'Admin' }) - firstName!: string; - - @IsString() - @IsNotEmpty() - @ApiProperty({ example: 'Doe' }) - lastName!: string; + name!: string; } export class ChangePasswordDto { diff --git a/server/src/domain/auth/auth.service.spec.ts b/server/src/domain/auth/auth.service.spec.ts index 3fb5d4d328..f915883a5b 100644 --- a/server/src/domain/auth/auth.service.spec.ts +++ b/server/src/domain/auth/auth.service.spec.ts @@ -236,7 +236,7 @@ describe('AuthService', () => { }); describe('adminSignUp', () => { - const dto: SignUpDto = { email: 'test@immich.com', password: 'password', firstName: 'immich', lastName: 'admin' }; + const dto: SignUpDto = { email: 'test@immich.com', password: 'password', name: 'immich admin' }; it('should only allow one admin', async () => { userMock.getAdmin.mockResolvedValue({} as UserEntity); @@ -251,8 +251,7 @@ describe('AuthService', () => { id: 'admin', createdAt: new Date('2021-01-01'), email: 'test@immich.com', - firstName: 'immich', - lastName: 'admin', + name: 'immich admin', }); expect(userMock.getAdmin).toHaveBeenCalled(); expect(userMock.create).toHaveBeenCalled(); diff --git a/server/src/domain/auth/auth.service.ts b/server/src/domain/auth/auth.service.ts index 00aa41bfda..7fa3d69023 100644 --- a/server/src/domain/auth/auth.service.ts +++ b/server/src/domain/auth/auth.service.ts @@ -146,8 +146,7 @@ export class AuthService { const admin = await this.userCore.createUser({ isAdmin: true, email: dto.email, - firstName: dto.firstName, - lastName: dto.lastName, + name: dto.name, password: dto.password, storageLabel: 'admin', }); @@ -273,9 +272,9 @@ export class AuthService { storageLabel = null; } + const userName = profile.name ?? `${profile.given_name || ''} ${profile.family_name || ''}`; user = await this.userCore.createUser({ - firstName: profile.given_name || '', - lastName: profile.family_name || '', + name: userName, email: profile.email, oauthId: profile.sub, storageLabel, diff --git a/server/src/domain/partner/partner.service.spec.ts b/server/src/domain/partner/partner.service.spec.ts index 0bae95aa79..6eaa87ea05 100644 --- a/server/src/domain/partner/partner.service.spec.ts +++ b/server/src/domain/partner/partner.service.spec.ts @@ -7,10 +7,9 @@ import { PartnerService } from './partner.service'; const responseDto = { admin: { email: 'admin@test.com', - firstName: 'admin_first_name', + name: 'admin_name', id: 'admin_id', isAdmin: true, - lastName: 'admin_last_name', oauthId: '', profileImagePath: '', shouldChangePassword: false, @@ -24,10 +23,9 @@ const responseDto = { }, user1: { email: 'immich@test.com', - firstName: 'immich_first_name', + name: 'immich_name', id: 'user-id', isAdmin: false, - lastName: 'immich_last_name', oauthId: '', profileImagePath: '', shouldChangePassword: false, diff --git a/server/src/domain/repositories/user.repository.ts b/server/src/domain/repositories/user.repository.ts index c4647d5711..76060c3270 100644 --- a/server/src/domain/repositories/user.repository.ts +++ b/server/src/domain/repositories/user.repository.ts @@ -6,8 +6,7 @@ export interface UserListFilter { export interface UserStatsQueryResponse { userId: string; - userFirstName: string; - userLastName: string; + userName: string; photos: number; videos: number; usage: number; diff --git a/server/src/domain/server-info/server-info.dto.ts b/server/src/domain/server-info/server-info.dto.ts index 907ce26ea8..0223a58fa0 100644 --- a/server/src/domain/server-info/server-info.dto.ts +++ b/server/src/domain/server-info/server-info.dto.ts @@ -38,9 +38,7 @@ export class UsageByUserDto { @ApiProperty({ type: 'string' }) userId!: string; @ApiProperty({ type: 'string' }) - userFirstName!: string; - @ApiProperty({ type: 'string' }) - userLastName!: string; + userName!: string; @ApiProperty({ type: 'integer' }) photos!: number; @ApiProperty({ type: 'integer' }) diff --git a/server/src/domain/server-info/server-info.service.spec.ts b/server/src/domain/server-info/server-info.service.spec.ts index 204eb1bd1e..74aab12c7a 100644 --- a/server/src/domain/server-info/server-info.service.spec.ts +++ b/server/src/domain/server-info/server-info.service.spec.ts @@ -195,24 +195,21 @@ describe(ServerInfoService.name, () => { userMock.getUserStats.mockResolvedValue([ { userId: 'user1', - userFirstName: '1', - userLastName: 'User', + userName: '1 User', photos: 10, videos: 11, usage: 12345, }, { userId: 'user2', - userFirstName: '2', - userLastName: 'User', + userName: '2 User', photos: 10, videos: 20, usage: 123456, }, { userId: 'user3', - userFirstName: '3', - userLastName: 'User', + userName: '3 User', photos: 100, videos: 0, usage: 987654, @@ -227,25 +224,22 @@ describe(ServerInfoService.name, () => { { photos: 10, usage: 12345, - userFirstName: '1', + userName: '1 User', userId: 'user1', - userLastName: 'User', videos: 11, }, { photos: 10, usage: 123456, - userFirstName: '2', + userName: '2 User', userId: 'user2', - userLastName: 'User', videos: 20, }, { photos: 100, usage: 987654, - userFirstName: '3', + userName: '3 User', userId: 'user3', - userLastName: 'User', videos: 0, }, ], diff --git a/server/src/domain/server-info/server-info.service.ts b/server/src/domain/server-info/server-info.service.ts index 27005f1765..c215709eb6 100644 --- a/server/src/domain/server-info/server-info.service.ts +++ b/server/src/domain/server-info/server-info.service.ts @@ -98,8 +98,7 @@ export class ServerInfoService { for (const user of userStats) { const usage = new UsageByUserDto(); usage.userId = user.userId; - usage.userFirstName = user.userFirstName; - usage.userLastName = user.userLastName; + usage.userName = user.userName; usage.photos = user.photos; usage.videos = user.videos; usage.usage = user.usage; diff --git a/server/src/domain/user/dto/create-user.dto.spec.ts b/server/src/domain/user/dto/create-user.dto.spec.ts index 492d44c36f..4e571d38a4 100644 --- a/server/src/domain/user/dto/create-user.dto.spec.ts +++ b/server/src/domain/user/dto/create-user.dto.spec.ts @@ -7,8 +7,7 @@ describe('create user DTO', () => { const params: Partial = { email: undefined, password: 'password', - firstName: 'first name', - lastName: 'last name', + name: 'name', }; let dto: CreateUserDto = plainToInstance(CreateUserDto, params); let errors = await validate(dto); @@ -31,8 +30,7 @@ describe('create user DTO', () => { const dto = plainToInstance(CreateUserDto, { email: someEmail, password: 'some password', - firstName: 'some first name', - lastName: 'some last name', + name: 'some name', }); const errors = await validate(dto); expect(errors).toHaveLength(0); @@ -48,8 +46,7 @@ describe('create admin DTO', () => { isAdmin: true, email: someEmail, password: 'some password', - firstName: 'some first name', - lastName: 'some last name', + name: 'some name', }); const errors = await validate(dto); expect(errors).toHaveLength(0); @@ -64,8 +61,7 @@ describe('create user oauth DTO', () => { const dto = plainToInstance(CreateUserOAuthDto, { email: someEmail, oauthId: 'some oauth id', - firstName: 'some first name', - lastName: 'some last name', + name: 'some name', }); const errors = await validate(dto); expect(errors).toHaveLength(0); diff --git a/server/src/domain/user/dto/create-user.dto.ts b/server/src/domain/user/dto/create-user.dto.ts index 2a5f659efb..b1090d9c0e 100644 --- a/server/src/domain/user/dto/create-user.dto.ts +++ b/server/src/domain/user/dto/create-user.dto.ts @@ -13,11 +13,7 @@ export class CreateUserDto { @IsNotEmpty() @IsString() - firstName!: string; - - @IsNotEmpty() - @IsString() - lastName!: string; + name!: string; @Optional({ nullable: true }) @IsString() @@ -45,10 +41,7 @@ export class CreateAdminDto { password!: string; @IsNotEmpty() - firstName!: string; - - @IsNotEmpty() - lastName!: string; + name!: string; } export class CreateUserOAuthDto { @@ -59,7 +52,5 @@ export class CreateUserOAuthDto { @IsNotEmpty() oauthId!: string; - firstName?: string; - - lastName?: string; + name?: string; } diff --git a/server/src/domain/user/dto/update-user.dto.ts b/server/src/domain/user/dto/update-user.dto.ts index e993ae80ff..4f05498da9 100644 --- a/server/src/domain/user/dto/update-user.dto.ts +++ b/server/src/domain/user/dto/update-user.dto.ts @@ -17,12 +17,7 @@ export class UpdateUserDto { @Optional() @IsString() @IsNotEmpty() - firstName?: string; - - @Optional() - @IsString() - @IsNotEmpty() - lastName?: string; + name?: string; @Optional() @IsString() diff --git a/server/src/domain/user/response-dto/user-response.dto.ts b/server/src/domain/user/response-dto/user-response.dto.ts index b9f990378a..fd9a121678 100644 --- a/server/src/domain/user/response-dto/user-response.dto.ts +++ b/server/src/domain/user/response-dto/user-response.dto.ts @@ -2,8 +2,7 @@ import { UserEntity } from '@app/infra/entities'; export class UserDto { id!: string; - firstName!: string; - lastName!: string; + name!: string; email!: string; profileImagePath!: string; } @@ -24,8 +23,7 @@ export const mapSimpleUser = (entity: UserEntity): UserDto => { return { id: entity.id, email: entity.email, - firstName: entity.firstName, - lastName: entity.lastName, + name: entity.name, profileImagePath: entity.profileImagePath, }; }; diff --git a/server/src/domain/user/user.service.spec.ts b/server/src/domain/user/user.service.spec.ts index 1f9918fecb..94f919178f 100644 --- a/server/src/domain/user/user.service.spec.ts +++ b/server/src/domain/user/user.service.spec.ts @@ -289,8 +289,7 @@ describe(UserService.name, () => { await expect( sut.create({ email: 'john_smith@email.com', - firstName: 'John', - lastName: 'Smith', + name: 'John Smith', password: 'password', }), ).rejects.toBeInstanceOf(BadRequestException); @@ -303,8 +302,7 @@ describe(UserService.name, () => { await expect( sut.create({ email: userStub.user1.email, - firstName: userStub.user1.firstName, - lastName: userStub.user1.lastName, + name: userStub.user1.name, password: 'password', storageLabel: 'label', }), @@ -313,8 +311,7 @@ describe(UserService.name, () => { expect(userMock.getAdmin).toBeCalled(); expect(userMock.create).toBeCalledWith({ email: userStub.user1.email, - firstName: userStub.user1.firstName, - lastName: userStub.user1.lastName, + name: userStub.user1.name, storageLabel: 'label', password: expect.anything(), }); diff --git a/server/src/infra/entities/user.entity.ts b/server/src/infra/entities/user.entity.ts index e6555153ae..83f23bef60 100644 --- a/server/src/infra/entities/user.entity.ts +++ b/server/src/infra/entities/user.entity.ts @@ -16,10 +16,7 @@ export class UserEntity { id!: string; @Column({ default: '' }) - firstName!: string; - - @Column({ default: '' }) - lastName!: string; + name!: string; @Column({ default: false }) isAdmin!: boolean; diff --git a/server/src/infra/migrations/1699322864544-UserNameConsolidation.ts b/server/src/infra/migrations/1699322864544-UserNameConsolidation.ts new file mode 100644 index 0000000000..431a2a92fc --- /dev/null +++ b/server/src/infra/migrations/1699322864544-UserNameConsolidation.ts @@ -0,0 +1,21 @@ +import { MigrationInterface, QueryRunner } from "typeorm"; + +export class AddUsername1699322864544 implements MigrationInterface { + name = 'AddUsername1699322864544' + + public async up(queryRunner: QueryRunner): Promise { + await queryRunner.query(`ALTER TABLE "users" ADD "name" character varying NOT NULL DEFAULT ''`); + await queryRunner.query(`UPDATE "users" SET "name" = CONCAT(COALESCE("firstName", ''), ' ', COALESCE("lastName", ''))`); + await queryRunner.query(`ALTER TABLE "users" DROP COLUMN "firstName"`); + await queryRunner.query(`ALTER TABLE "users" DROP COLUMN "lastName"`); + } + + public async down(queryRunner: QueryRunner): Promise { + await queryRunner.query(`ALTER TABLE "users" DROP COLUMN "name"`); + await queryRunner.query(`ALTER TABLE "users" ADD "lastName" character varying NOT NULL DEFAULT ''`); + await queryRunner.query(`ALTER TABLE "users" ADD "firstName" character varying NOT NULL DEFAULT ''`); + await queryRunner.query(`UPDATE "users" SET "lastName" = COALESCE("email", '')`); + await queryRunner.query(`UPDATE "users" SET "firstName" = COALESCE("email", '')`); + } + +} diff --git a/server/src/infra/repositories/user.repository.ts b/server/src/infra/repositories/user.repository.ts index 9809ec75df..b84adb2fd7 100644 --- a/server/src/infra/repositories/user.repository.ts +++ b/server/src/infra/repositories/user.repository.ts @@ -80,8 +80,7 @@ export class UserRepository implements IUserRepository { const stats = await this.userRepository .createQueryBuilder('users') .select('users.id', 'userId') - .addSelect('users.firstName', 'userFirstName') - .addSelect('users.lastName', 'userLastName') + .addSelect('users.name', 'userName') .addSelect(`COUNT(assets.id) FILTER (WHERE assets.type = 'IMAGE' AND assets.isVisible)`, 'photos') .addSelect(`COUNT(assets.id) FILTER (WHERE assets.type = 'VIDEO' AND assets.isVisible)`, 'videos') .addSelect('COALESCE(SUM(exif.fileSizeInByte), 0)', 'usage') diff --git a/server/test/e2e/activity.e2e-spec.ts b/server/test/e2e/activity.e2e-spec.ts index 0bb8aa2c9a..34799b1634 100644 --- a/server/test/e2e/activity.e2e-spec.ts +++ b/server/test/e2e/activity.e2e-spec.ts @@ -350,8 +350,7 @@ describe(`${ActivityController.name} (e2e)`, () => { const { id: userId } = await api.userApi.create(server, admin.accessToken, { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }); await api.albumApi.addUsers(server, admin.accessToken, album.id, { sharedUserIds: [userId] }); const nonOwner = await api.authApi.login(server, { email: 'user1@immich.app', password: 'Password123' }); @@ -371,8 +370,7 @@ describe(`${ActivityController.name} (e2e)`, () => { const { id: userId } = await api.userApi.create(server, admin.accessToken, { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }); await api.albumApi.addUsers(server, admin.accessToken, album.id, { sharedUserIds: [userId] }); const nonOwner = await api.authApi.login(server, { email: 'user1@immich.app', password: 'Password123' }); @@ -393,8 +391,7 @@ describe(`${ActivityController.name} (e2e)`, () => { const { id: userId } = await api.userApi.create(server, admin.accessToken, { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }); await api.albumApi.addUsers(server, admin.accessToken, album.id, { sharedUserIds: [userId] }); const nonOwner = await api.authApi.login(server, { email: 'user1@immich.app', password: 'Password123' }); diff --git a/server/test/e2e/album.e2e-spec.ts b/server/test/e2e/album.e2e-spec.ts index 8348eff037..8c372a421a 100644 --- a/server/test/e2e/album.e2e-spec.ts +++ b/server/test/e2e/album.e2e-spec.ts @@ -41,14 +41,12 @@ describe(`${AlbumController.name} (e2e)`, () => { api.userApi.create(server, admin.accessToken, { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }), api.userApi.create(server, admin.accessToken, { email: 'user2@immich.app', password: 'Password123', - firstName: 'User 2', - lastName: 'Test', + name: 'User 2', }), ]); diff --git a/server/test/e2e/asset.e2e-spec.ts b/server/test/e2e/asset.e2e-spec.ts index 70be784c75..e15e27d5a4 100644 --- a/server/test/e2e/asset.e2e-spec.ts +++ b/server/test/e2e/asset.e2e-spec.ts @@ -22,15 +22,13 @@ import request from 'supertest'; const user1Dto = { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }; const user2Dto = { email: 'user2@immich.app', password: 'Password123', - firstName: 'User 2', - lastName: 'Test', + name: 'User 2', }; const makeUploadDto = (options?: { omit: string }): Record => { diff --git a/server/test/e2e/auth.e2e-spec.ts b/server/test/e2e/auth.e2e-spec.ts index 191add42fa..eb47a87255 100644 --- a/server/test/e2e/auth.e2e-spec.ts +++ b/server/test/e2e/auth.e2e-spec.ts @@ -13,15 +13,13 @@ import { import { testApp } from '@test/test-utils'; import request from 'supertest'; -const firstName = 'Immich'; -const lastName = 'Admin'; +const name = 'Immich Admin'; const password = 'Password123'; const email = 'admin@immich.app'; const adminSignupResponse = { id: expect.any(String), - firstName: 'Immich', - lastName: 'Admin', + name: 'Immich Admin', email: 'admin@immich.app', storageLabel: 'admin', externalPath: null, @@ -64,23 +62,19 @@ describe(`${AuthController.name} (e2e)`, () => { const invalid = [ { should: 'require an email address', - data: { firstName, lastName, password }, + data: { name, password }, }, { should: 'require a password', - data: { firstName, lastName, email }, + data: { name, email }, }, { - should: 'require a first name ', - data: { lastName, email, password }, - }, - { - should: 'require a last name ', - data: { firstName, email, password }, + should: 'require a name', + data: { email, password }, }, { should: 'require a valid email', - data: { firstName, lastName, email: 'immich', password }, + data: { name, email: 'immich', password }, }, ]; diff --git a/server/test/e2e/library.e2e-spec.ts b/server/test/e2e/library.e2e-spec.ts index 9cfbe89610..d8905a8c14 100644 --- a/server/test/e2e/library.e2e-spec.ts +++ b/server/test/e2e/library.e2e-spec.ts @@ -15,15 +15,13 @@ describe(`${LibraryController.name} (e2e)`, () => { const user1Dto = { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }; const user2Dto = { email: 'user2@immich.app', password: 'Password123', - firstName: 'User 2', - lastName: 'Test', + name: 'User 2', }; beforeAll(async () => { diff --git a/server/test/e2e/partner.e2e-spec.ts b/server/test/e2e/partner.e2e-spec.ts index f6b8e144fe..3eb48dd9e2 100644 --- a/server/test/e2e/partner.e2e-spec.ts +++ b/server/test/e2e/partner.e2e-spec.ts @@ -10,15 +10,13 @@ import request from 'supertest'; const user1Dto = { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }; const user2Dto = { email: 'user2@immich.app', password: 'Password123', - firstName: 'User 2', - lastName: 'Test', + name: 'User 2', }; describe(`${PartnerController.name} (e2e)`, () => { diff --git a/server/test/e2e/server-info.e2e-spec.ts b/server/test/e2e/server-info.e2e-spec.ts index 0b508a2ef5..be85e424f7 100644 --- a/server/test/e2e/server-info.e2e-spec.ts +++ b/server/test/e2e/server-info.e2e-spec.ts @@ -111,7 +111,7 @@ describe(`${ServerInfoController.name} (e2e)`, () => { it('should only work for admins', async () => { const loginDto = { email: 'test@immich.app', password: 'Immich123' }; - await api.userApi.create(server, accessToken, { ...loginDto, firstName: 'test', lastName: 'test' }); + await api.userApi.create(server, accessToken, { ...loginDto, name: 'test' }); const { accessToken: userAccessToken } = await api.authApi.login(server, loginDto); const { status, body } = await request(server) .get('/server-info/statistics') @@ -132,9 +132,8 @@ describe(`${ServerInfoController.name} (e2e)`, () => { { photos: 0, usage: 0, - userFirstName: 'Immich', + userName: 'Immich Admin', userId: loginResponse.userId, - userLastName: 'Admin', videos: 0, }, ], diff --git a/server/test/e2e/shared-link.e2e-spec.ts b/server/test/e2e/shared-link.e2e-spec.ts index 03eb9da7db..ededc01384 100644 --- a/server/test/e2e/shared-link.e2e-spec.ts +++ b/server/test/e2e/shared-link.e2e-spec.ts @@ -11,8 +11,7 @@ import request from 'supertest'; const user1Dto = { email: 'user1@immich.app', password: 'Password123', - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }; describe(`${PartnerController.name} (e2e)`, () => { diff --git a/server/test/e2e/system-config.e2e-spec.ts b/server/test/e2e/system-config.e2e-spec.ts index dd7a69439c..0e505ca11a 100644 --- a/server/test/e2e/system-config.e2e-spec.ts +++ b/server/test/e2e/system-config.e2e-spec.ts @@ -64,8 +64,7 @@ describe(`${SystemConfigController.name} (e2e)`, () => { const credentials = { email: 'user1@immich.app', password: 'Password123' }; await api.userApi.create(server, admin.accessToken, { ...credentials, - firstName: 'User 1', - lastName: 'Test', + name: 'User 1', }); const { accessToken } = await api.authApi.login(server, credentials); const { status, body } = await request(server) diff --git a/server/test/e2e/user.e2e-spec.ts b/server/test/e2e/user.e2e-spec.ts index 02b03e9bab..41ce7c116b 100644 --- a/server/test/e2e/user.e2e-spec.ts +++ b/server/test/e2e/user.e2e-spec.ts @@ -59,8 +59,7 @@ describe(`${UserController.name}`, () => { const user1 = await api.userApi.create(server, accessToken, { email: `user1@immich.app`, password: 'Password123', - firstName: `User 1`, - lastName: 'Test', + name: `User 1`, }); await api.userApi.delete(server, accessToken, user1.id); @@ -78,8 +77,7 @@ describe(`${UserController.name}`, () => { const user1 = await api.userApi.create(server, accessToken, { email: `user1@immich.app`, password: 'Password123', - firstName: `User 1`, - lastName: 'Test', + name: `User 1`, }); await api.userApi.delete(server, accessToken, user1.id); @@ -149,8 +147,7 @@ describe(`${UserController.name}`, () => { isAdmin: true, email: 'user1@immich.app', password: 'Password123', - firstName: 'Immich', - lastName: 'User', + name: 'Immich', }) .set('Authorization', `Bearer ${accessToken}`); expect(body).toMatchObject({ @@ -167,8 +164,7 @@ describe(`${UserController.name}`, () => { .send({ email: 'no-memories@immich.app', password: 'Password123', - firstName: 'No Memories', - lastName: 'User', + name: 'No Memories', memoriesEnabled: false, }) .set('Authorization', `Bearer ${accessToken}`); @@ -186,8 +182,7 @@ describe(`${UserController.name}`, () => { beforeEach(async () => { userToDelete = await api.userApi.create(server, accessToken, { email: userStub.user1.email, - firstName: userStub.user1.firstName, - lastName: userStub.user1.lastName, + name: userStub.user1.name, password: 'superSecurePassword', }); }); @@ -246,8 +241,7 @@ describe(`${UserController.name}`, () => { const user = await api.userApi.create(server, accessToken, { email: 'user1@immich.app', password: 'Password123', - firstName: 'Immich', - lastName: 'User', + name: 'Immich User', }); const { status, body } = await request(server) @@ -284,15 +278,13 @@ describe(`${UserController.name}`, () => { const before = await api.userApi.get(server, accessToken, loginResponse.userId); const after = await api.userApi.update(server, accessToken, { id: before.id, - firstName: 'First Name', - lastName: 'Last Name', + name: 'Name', }); expect(after).toEqual({ ...before, updatedAt: expect.any(String), - firstName: 'First Name', - lastName: 'Last Name', + name: 'Name', }); expect(before.updatedAt).not.toEqual(after.updatedAt); }); diff --git a/server/test/fixtures/auth.stub.ts b/server/test/fixtures/auth.stub.ts index 68ff9b717c..d4b6368960 100644 --- a/server/test/fixtures/auth.stub.ts +++ b/server/test/fixtures/auth.stub.ts @@ -1,8 +1,7 @@ import { AuthUserDto } from '@app/domain'; export const adminSignupStub = { - firstName: 'Immich', - lastName: 'Admin', + name: 'Immich Admin', email: 'admin@immich.app', password: 'Password123', }; @@ -103,9 +102,8 @@ export const loginResponseStub = { admin: { response: { accessToken: expect.any(String), - firstName: 'Immich', + name: 'Immich Admin', isAdmin: true, - lastName: 'Admin', profileImagePath: '', shouldChangePassword: true, userEmail: 'admin@immich.app', @@ -117,8 +115,7 @@ export const loginResponseStub = { accessToken: 'cmFuZG9tLWJ5dGVz', userId: 'user-id', userEmail: 'immich@test.com', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', profileImagePath: '', isAdmin: false, shouldChangePassword: false, @@ -133,8 +130,7 @@ export const loginResponseStub = { accessToken: 'cmFuZG9tLWJ5dGVz', userId: 'user-id', userEmail: 'immich@test.com', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', profileImagePath: '', isAdmin: false, shouldChangePassword: false, @@ -149,8 +145,7 @@ export const loginResponseStub = { accessToken: 'cmFuZG9tLWJ5dGVz', userId: 'user-id', userEmail: 'immich@test.com', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', profileImagePath: '', isAdmin: false, shouldChangePassword: false, diff --git a/server/test/fixtures/user.stub.ts b/server/test/fixtures/user.stub.ts index af5515bafc..b528a107f3 100644 --- a/server/test/fixtures/user.stub.ts +++ b/server/test/fixtures/user.stub.ts @@ -5,8 +5,7 @@ export const userStub = { admin: Object.freeze({ ...authStub.admin, password: 'admin_password', - firstName: 'admin_first_name', - lastName: 'admin_last_name', + name: 'admin_name', storageLabel: 'admin', externalPath: null, oauthId: '', @@ -22,8 +21,7 @@ export const userStub = { user1: Object.freeze({ ...authStub.user1, password: 'immich_password', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', storageLabel: null, externalPath: null, oauthId: '', @@ -39,8 +37,7 @@ export const userStub = { user2: Object.freeze({ ...authStub.user2, password: 'immich_password', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', storageLabel: null, externalPath: null, oauthId: '', @@ -56,8 +53,7 @@ export const userStub = { storageLabel: Object.freeze({ ...authStub.user1, password: 'immich_password', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', storageLabel: 'label-1', externalPath: null, oauthId: '', @@ -73,8 +69,7 @@ export const userStub = { externalPath1: Object.freeze({ ...authStub.user1, password: 'immich_password', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', storageLabel: 'label-1', externalPath: '/data/user1', oauthId: '', @@ -90,8 +85,7 @@ export const userStub = { externalPath2: Object.freeze({ ...authStub.user1, password: 'immich_password', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', storageLabel: 'label-1', externalPath: '/data/user2', oauthId: '', @@ -107,8 +101,7 @@ export const userStub = { profilePath: Object.freeze({ ...authStub.user1, password: 'immich_password', - firstName: 'immich_first_name', - lastName: 'immich_last_name', + name: 'immich_name', storageLabel: 'label-1', externalPath: null, oauthId: '', diff --git a/web/src/api/open-api/api.ts b/web/src/api/open-api/api.ts index 496fa5441d..8fb2c1b3d1 100644 --- a/web/src/api/open-api/api.ts +++ b/web/src/api/open-api/api.ts @@ -1341,24 +1341,18 @@ export interface CreateUserDto { * @memberof CreateUserDto */ 'externalPath'?: string | null; - /** - * - * @type {string} - * @memberof CreateUserDto - */ - 'firstName': string; - /** - * - * @type {string} - * @memberof CreateUserDto - */ - 'lastName': string; /** * * @type {boolean} * @memberof CreateUserDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof CreateUserDto + */ + 'name': string; /** * * @type {string} @@ -2137,12 +2131,6 @@ export interface LoginResponseDto { * @memberof LoginResponseDto */ 'accessToken': string; - /** - * - * @type {string} - * @memberof LoginResponseDto - */ - 'firstName': string; /** * * @type {boolean} @@ -2154,7 +2142,7 @@ export interface LoginResponseDto { * @type {string} * @memberof LoginResponseDto */ - 'lastName': string; + 'name': string; /** * * @type {string} @@ -2391,12 +2379,6 @@ export interface PartnerResponseDto { * @memberof PartnerResponseDto */ 'externalPath': string | null; - /** - * - * @type {string} - * @memberof PartnerResponseDto - */ - 'firstName': string; /** * * @type {string} @@ -2415,18 +2397,18 @@ export interface PartnerResponseDto { * @memberof PartnerResponseDto */ 'isAdmin': boolean; - /** - * - * @type {string} - * @memberof PartnerResponseDto - */ - 'lastName': string; /** * * @type {boolean} * @memberof PartnerResponseDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof PartnerResponseDto + */ + 'name': string; /** * * @type {string} @@ -3431,13 +3413,7 @@ export interface SignUpDto { * @type {string} * @memberof SignUpDto */ - 'firstName': string; - /** - * - * @type {string} - * @memberof SignUpDto - */ - 'lastName': string; + 'name': string; /** * * @type {string} @@ -4380,12 +4356,6 @@ export interface UpdateUserDto { * @memberof UpdateUserDto */ 'externalPath'?: string; - /** - * - * @type {string} - * @memberof UpdateUserDto - */ - 'firstName'?: string; /** * * @type {string} @@ -4398,18 +4368,18 @@ export interface UpdateUserDto { * @memberof UpdateUserDto */ 'isAdmin'?: boolean; - /** - * - * @type {string} - * @memberof UpdateUserDto - */ - 'lastName'?: string; /** * * @type {boolean} * @memberof UpdateUserDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof UpdateUserDto + */ + 'name'?: string; /** * * @type {string} @@ -4447,12 +4417,6 @@ export interface UsageByUserDto { * @memberof UsageByUserDto */ 'usage': number; - /** - * - * @type {string} - * @memberof UsageByUserDto - */ - 'userFirstName': string; /** * * @type {string} @@ -4464,7 +4428,7 @@ export interface UsageByUserDto { * @type {string} * @memberof UsageByUserDto */ - 'userLastName': string; + 'userName': string; /** * * @type {number} @@ -4484,12 +4448,6 @@ export interface UserDto { * @memberof UserDto */ 'email': string; - /** - * - * @type {string} - * @memberof UserDto - */ - 'firstName': string; /** * * @type {string} @@ -4501,7 +4459,7 @@ export interface UserDto { * @type {string} * @memberof UserDto */ - 'lastName': string; + 'name': string; /** * * @type {string} @@ -4539,12 +4497,6 @@ export interface UserResponseDto { * @memberof UserResponseDto */ 'externalPath': string | null; - /** - * - * @type {string} - * @memberof UserResponseDto - */ - 'firstName': string; /** * * @type {string} @@ -4557,18 +4509,18 @@ export interface UserResponseDto { * @memberof UserResponseDto */ 'isAdmin': boolean; - /** - * - * @type {string} - * @memberof UserResponseDto - */ - 'lastName': string; /** * * @type {boolean} * @memberof UserResponseDto */ 'memoriesEnabled'?: boolean; + /** + * + * @type {string} + * @memberof UserResponseDto + */ + 'name': string; /** * * @type {string} diff --git a/web/src/lib/components/admin-page/delete-confirm-dialoge.svelte b/web/src/lib/components/admin-page/delete-confirm-dialoge.svelte index 1b7261544b..5bad6c940e 100644 --- a/web/src/lib/components/admin-page/delete-confirm-dialoge.svelte +++ b/web/src/lib/components/admin-page/delete-confirm-dialoge.svelte @@ -27,7 +27,7 @@

- {user.firstName} {user.lastName}'s account and assets will be permanently deleted after 7 days. + {user.name}'s account and assets will be permanently deleted after 7 days.

Are you sure you want to continue?

diff --git a/web/src/lib/components/admin-page/restore-dialoge.svelte b/web/src/lib/components/admin-page/restore-dialoge.svelte index 4e4517045c..848dedaf65 100644 --- a/web/src/lib/components/admin-page/restore-dialoge.svelte +++ b/web/src/lib/components/admin-page/restore-dialoge.svelte @@ -16,6 +16,6 @@ -

{user.firstName} {user.lastName}'s account will be restored.

+

{user.name}'s account will be restored.

diff --git a/web/src/lib/components/admin-page/server-stats/server-stats-panel.svelte b/web/src/lib/components/admin-page/server-stats/server-stats-panel.svelte index 0dcddfb3b6..a4f5274454 100644 --- a/web/src/lib/components/admin-page/server-stats/server-stats-panel.svelte +++ b/web/src/lib/components/admin-page/server-stats/server-stats-panel.svelte @@ -96,7 +96,7 @@ - {user.userFirstName} {user.userLastName} + {user.userName} {user.photos.toLocaleString($locale)} {user.videos.toLocaleString($locale)} {asByteUnitString(user.usage, $locale)} diff --git a/web/src/lib/components/album-page/album-card.svelte b/web/src/lib/components/album-page/album-card.svelte index fdaf59395f..040d5faa4f 100644 --- a/web/src/lib/components/album-page/album-card.svelte +++ b/web/src/lib/components/album-page/album-card.svelte @@ -123,8 +123,7 @@

Owned

{:else}

- Shared by {albumOwner.firstName} - {albumOwner.lastName} + Shared by {albumOwner.name}

{/if} {/await} diff --git a/web/src/lib/components/album-page/album-options.svelte b/web/src/lib/components/album-page/album-options.svelte index 1a3e43892c..5fecc6431c 100644 --- a/web/src/lib/components/album-page/album-options.svelte +++ b/web/src/lib/components/album-page/album-options.svelte @@ -56,7 +56,7 @@
-
{`${user.firstName} ${user.lastName}`}
+
{user.name}
Owner
{#each album.sharedUsers as user (user.id)} @@ -64,7 +64,7 @@
-
{`${user.firstName} ${user.lastName}`}
+
{user.name}
{/each} diff --git a/web/src/lib/components/album-page/share-info-modal.svelte b/web/src/lib/components/album-page/share-info-modal.svelte index 53656de5a4..668946d837 100644 --- a/web/src/lib/components/album-page/share-info-modal.svelte +++ b/web/src/lib/components/album-page/share-info-modal.svelte @@ -56,7 +56,7 @@ try { await api.albumApi.removeUserFromAlbum({ id: album.id, userId }); dispatch('remove', userId); - const message = userId === 'me' ? `Left ${album.albumName}` : `Removed ${selectedRemoveUser.firstName}`; + const message = userId === 'me' ? `Left ${album.albumName}` : `Removed ${selectedRemoveUser.name}`; notificationController.show({ type: NotificationType.Info, message }); } catch (e) { handleError(e, 'Unable to remove user'); @@ -78,7 +78,7 @@
-

{album.owner.firstName} {album.owner.lastName}

+

{album.owner.name}

@@ -91,7 +91,7 @@ >
-

{user.firstName} {user.lastName}

+

{user.name}

@@ -138,7 +138,7 @@ {#if selectedRemoveUser && selectedRemoveUser?.id !== currentUser?.id} (selectedRemoveUser = null)} diff --git a/web/src/lib/components/album-page/user-selection-modal.svelte b/web/src/lib/components/album-page/user-selection-modal.svelte index 7a774a3f02..0b4818ad5f 100644 --- a/web/src/lib/components/album-page/user-selection-modal.svelte +++ b/web/src/lib/components/album-page/user-selection-modal.svelte @@ -72,7 +72,7 @@ class="flex place-items-center gap-1 rounded-full border border-gray-400 p-1 transition-colors hover:bg-gray-200 dark:hover:bg-gray-700" > -

{user.firstName} {user.lastName}

+

{user.name}

{/key} {/each} @@ -99,8 +99,7 @@

- {user.firstName} - {user.lastName} + {user.name}

{user.email} diff --git a/web/src/lib/components/asset-viewer/activity-viewer.svelte b/web/src/lib/components/asset-viewer/activity-viewer.svelte index f8663ecebc..0f1e4ffcdd 100644 --- a/web/src/lib/components/asset-viewer/activity-viewer.svelte +++ b/web/src/lib/components/asset-viewer/activity-viewer.svelte @@ -221,13 +221,8 @@

-
- {`${reaction.user.firstName} ${reaction.user.lastName} liked ${ - assetType ? `this ${getAssetType(assetType).toLowerCase()}` : 'it' - }`} +
+ {`${reaction.user.name} liked ${assetType ? `this ${getAssetType(assetType).toLowerCase()}` : 'it'}`}
{#if assetId === undefined && reaction.assetId}
diff --git a/web/src/lib/components/asset-viewer/detail-panel.svelte b/web/src/lib/components/asset-viewer/detail-panel.svelte index b64abdd5e2..5fff954afe 100644 --- a/web/src/lib/components/asset-viewer/detail-panel.svelte +++ b/web/src/lib/components/asset-viewer/detail-panel.svelte @@ -315,8 +315,7 @@

- {asset.owner.firstName} - {asset.owner.lastName} + {asset.owner.name}

diff --git a/web/src/lib/components/forms/admin-registration-form.svelte b/web/src/lib/components/forms/admin-registration-form.svelte index 9223f54e06..912c8f1da1 100644 --- a/web/src/lib/components/forms/admin-registration-form.svelte +++ b/web/src/lib/components/forms/admin-registration-form.svelte @@ -27,15 +27,13 @@ const email = form.get('email'); const password = form.get('password'); - const firstName = form.get('firstName'); - const lastName = form.get('lastName'); + const name = form.get('name'); const { status } = await api.authenticationApi.signUpAdmin({ signUpDto: { email: String(email), password: String(password), - firstName: String(firstName), - lastName: String(lastName), + name: String(name), }, }); @@ -83,13 +81,8 @@
- - -
- -
- - + +
{#if error} diff --git a/web/src/lib/components/forms/create-user-form.svelte b/web/src/lib/components/forms/create-user-form.svelte index aee097a226..50c20790ea 100644 --- a/web/src/lib/components/forms/create-user-form.svelte +++ b/web/src/lib/components/forms/create-user-form.svelte @@ -38,16 +38,14 @@ const email = form.get('email'); const password = form.get('password'); - const firstName = form.get('firstName'); - const lastName = form.get('lastName'); + const name = form.get('name'); try { const { status } = await api.userApi.createUser({ createUserDto: { email: String(email), password: String(password), - firstName: String(firstName), - lastName: String(lastName), + name: String(name), }, }); @@ -112,13 +110,8 @@
- - -
- -
- - + +
{#if error} diff --git a/web/src/lib/components/forms/edit-user-form.svelte b/web/src/lib/components/forms/edit-user-form.svelte index b4c426aca4..2ad1a955cc 100644 --- a/web/src/lib/components/forms/edit-user-form.svelte +++ b/web/src/lib/components/forms/edit-user-form.svelte @@ -20,13 +20,12 @@ const editUser = async () => { try { - const { id, email, firstName, lastName, storageLabel, externalPath } = user; + const { id, email, name, storageLabel, externalPath } = user; const { status } = await api.userApi.updateUser({ updateUserDto: { id, email, - firstName, - lastName, + name, storageLabel: storageLabel || '', externalPath: externalPath || '', }, @@ -84,20 +83,8 @@
- - -
- -
- - + +
@@ -161,7 +148,7 @@ >

- Are you sure you want to reset {user.firstName} {user.lastName}'s password? + Are you sure you want to reset {user.name}'s password?

diff --git a/web/src/lib/components/shared-components/navigation-bar/account-info-panel.svelte b/web/src/lib/components/shared-components/navigation-bar/account-info-panel.svelte index 4fa41150bc..6a135066f9 100644 --- a/web/src/lib/components/shared-components/navigation-bar/account-info-panel.svelte +++ b/web/src/lib/components/shared-components/navigation-bar/account-info-panel.svelte @@ -26,8 +26,7 @@

- {user.firstName} - {user.lastName} + {user.name}

{user.email}

diff --git a/web/src/lib/components/shared-components/navigation-bar/navigation-bar.svelte b/web/src/lib/components/shared-components/navigation-bar/navigation-bar.svelte index 36f9d2d656..c72803b54c 100644 --- a/web/src/lib/components/shared-components/navigation-bar/navigation-bar.svelte +++ b/web/src/lib/components/shared-components/navigation-bar/navigation-bar.svelte @@ -133,7 +133,7 @@ out:fade={{ delay: 200, duration: 150 }} class="absolute -bottom-12 right-5 rounded-md border bg-gray-500 p-2 text-[12px] text-gray-100 shadow-md dark:border-immich-dark-gray dark:bg-immich-dark-gray" > -

{user.firstName} {user.lastName}

+

{user.name}

{user.email}

{/if} diff --git a/web/src/lib/components/shared-components/user-avatar.svelte b/web/src/lib/components/shared-components/user-avatar.svelte index c7dbdad382..07e65d94aa 100644 --- a/web/src/lib/components/shared-components/user-avatar.svelte +++ b/web/src/lib/components/shared-components/user-avatar.svelte @@ -9,8 +9,7 @@ interface User { id: string; - firstName: string; - lastName: string; + name: string; email: string; profileImagePath: string; } @@ -51,7 +50,7 @@ $: colorClass = colorClasses[autoColor ? getUserColor() : color]; $: sizeClass = sizeClasses[size]; - $: title = `${user.firstName} ${user.lastName} (${user.email})`; + $: title = `${user.name} (${user.email})`; $: interactiveClass = interactive ? 'border-2 border-immich-primary hover:border-immich-dark-primary dark:hover:border-immich-primary dark:border-immich-dark-primary transition-colors' : ''; @@ -82,7 +81,7 @@ class:font-medium={!autoColor} class:font-semibold={autoColor} > - {((user.firstName[0] || '') + (user.lastName[0] || '')).toUpperCase()} + {(user.name[0] || '').toUpperCase()} {/if} diff --git a/web/src/lib/components/user-settings-page/partner-selection-modal.svelte b/web/src/lib/components/user-settings-page/partner-selection-modal.svelte index 755b942d0b..69b10ed6e3 100644 --- a/web/src/lib/components/user-settings-page/partner-selection-modal.svelte +++ b/web/src/lib/components/user-settings-page/partner-selection-modal.svelte @@ -61,8 +61,7 @@

- {user.firstName} - {user.lastName} + {user.name}

{user.email} diff --git a/web/src/lib/components/user-settings-page/partner-settings.svelte b/web/src/lib/components/user-settings-page/partner-settings.svelte index ee5fe8a091..374774aeca 100644 --- a/web/src/lib/components/user-settings-page/partner-settings.svelte +++ b/web/src/lib/components/user-settings-page/partner-settings.svelte @@ -116,8 +116,7 @@

- {partner.user.firstName} - {partner.user.lastName} + {partner.user.name}

{partner.user.email} @@ -139,8 +138,8 @@ {#if partner.sharedByMe}


-

SHARED WITH {partner.user.firstName.toUpperCase()}

-

{partner.user.firstName} can access

+

SHARED WITH {partner.user.name.toUpperCase()}

+

{partner.user.name} can access

  • All your photos and videos except those in Archived and Deleted @@ -154,7 +153,7 @@ {#if partner.sharedWithMe}
    -

    PHOTOS FROM {partner.user.firstName.toUpperCase()}

    +

    PHOTOS FROM {partner.user.name.toUpperCase()}

    (removePartner = null)} on:confirm={() => handleRemovePartner()} /> diff --git a/web/src/lib/components/user-settings-page/user-profile-settings.svelte b/web/src/lib/components/user-settings-page/user-profile-settings.svelte index d571adbbf6..89f4958ce8 100644 --- a/web/src/lib/components/user-settings-page/user-profile-settings.svelte +++ b/web/src/lib/components/user-settings-page/user-profile-settings.svelte @@ -17,8 +17,7 @@ updateUserDto: { id: user.id, email: user.email, - firstName: user.firstName, - lastName: user.lastName, + name: user.name, }, }); @@ -47,19 +46,7 @@ - - - + goto(AppRoute.SHARING)}>

    - {data.partner.firstName} - {data.partner.lastName}'s photos + {data.partner.name}'s photos

    diff --git a/web/src/routes/(user)/sharing/+page.svelte b/web/src/routes/(user)/sharing/+page.svelte index fd9d1c58fb..a8df23a0bd 100644 --- a/web/src/routes/(user)/sharing/+page.svelte +++ b/web/src/routes/(user)/sharing/+page.svelte @@ -72,8 +72,7 @@

    - {partner.firstName} - {partner.lastName} + {partner.name}

    {partner.email} diff --git a/web/src/routes/admin/user-management/+page.svelte b/web/src/routes/admin/user-management/+page.svelte index b800f5d94a..09b3db7f78 100644 --- a/web/src/routes/admin/user-management/+page.svelte +++ b/web/src/routes/admin/user-management/+page.svelte @@ -168,8 +168,7 @@ > Email - First name - Last name + Name Can import Action @@ -187,8 +186,7 @@ }`} > {user.email} - {user.firstName} - {user.lastName} + {user.name}

    {#if user.externalPath} @@ -253,7 +251,7 @@ : 'bg-immich-bg dark:bg-immich-dark-gray/50' }`} > - {user.firstName} {user.lastName} + {user.name} {user.email} {#if !isDeleted(user)} diff --git a/web/src/routes/auth/change-password/+page.svelte b/web/src/routes/auth/change-password/+page.svelte index f515e1d442..a4d0141d97 100644 --- a/web/src/routes/auth/change-password/+page.svelte +++ b/web/src/routes/auth/change-password/+page.svelte @@ -16,8 +16,7 @@

    - Hi {data.user.firstName} - {data.user.lastName} ({data.user.email}), + Hi {data.user.name} ({data.user.email}),

    This is either the first time you are signing into the system or a request has been made to change your password. Please diff --git a/web/src/test-data/factories/user-factory.ts b/web/src/test-data/factories/user-factory.ts index 507242fe8b..21b4e32396 100644 --- a/web/src/test-data/factories/user-factory.ts +++ b/web/src/test-data/factories/user-factory.ts @@ -5,8 +5,7 @@ import { Sync } from 'factory.ts'; export const userFactory = Sync.makeFactory({ id: Sync.each(() => faker.datatype.uuid()), email: Sync.each(() => faker.internet.email()), - firstName: Sync.each(() => faker.name.firstName()), - lastName: Sync.each(() => faker.name.lastName()), + name: Sync.each(() => faker.name.fullName()), storageLabel: Sync.each(() => faker.random.alphaNumeric()), externalPath: Sync.each(() => faker.random.alphaNumeric()), profileImagePath: '',