@sync-in/server 1.5.2 → 1.6.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/CHANGELOG.md +20 -0
- package/README.md +1 -0
- package/environment/environment.dist.min.yaml +1 -0
- package/environment/environment.dist.yaml +88 -30
- package/migrations/0002_sleepy_korath.sql +1 -0
- package/migrations/meta/0002_snapshot.json +2424 -0
- package/migrations/meta/_journal.json +7 -0
- package/package.json +6 -4
- package/server/app.bootstrap.js +1 -1
- package/server/app.bootstrap.js.map +1 -1
- package/server/applications/files/services/files-manager.service.js +1 -2
- package/server/applications/files/services/files-manager.service.js.map +1 -1
- package/server/applications/files/services/files-only-office-manager.service.js +5 -6
- package/server/applications/files/services/files-only-office-manager.service.js.map +1 -1
- package/server/applications/files/utils/files.js +6 -4
- package/server/applications/files/utils/files.js.map +1 -1
- package/server/applications/links/links.controller.js +2 -2
- package/server/applications/links/links.controller.js.map +1 -1
- package/server/applications/links/services/links-manager.service.js +2 -1
- package/server/applications/links/services/links-manager.service.js.map +1 -1
- package/server/applications/links/services/links-manager.service.spec.js +6 -3
- package/server/applications/links/services/links-manager.service.spec.js.map +1 -1
- package/server/applications/notifications/constants/notifications.js +9 -0
- package/server/applications/notifications/constants/notifications.js.map +1 -1
- package/server/applications/notifications/i18n/fr.js +10 -1
- package/server/applications/notifications/i18n/fr.js.map +1 -1
- package/server/applications/notifications/interfaces/notification-properties.interface.js.map +1 -1
- package/server/applications/notifications/mails/models.js +41 -3
- package/server/applications/notifications/mails/models.js.map +1 -1
- package/server/applications/notifications/mails/templates.js +1 -1
- package/server/applications/notifications/mails/templates.js.map +1 -1
- package/server/applications/notifications/schemas/notifications.schema.js +2 -1
- package/server/applications/notifications/schemas/notifications.schema.js.map +1 -1
- package/server/applications/notifications/services/notifications-manager.service.js +16 -13
- package/server/applications/notifications/services/notifications-manager.service.js.map +1 -1
- package/server/applications/notifications/services/notifications-manager.service.spec.js +9 -8
- package/server/applications/notifications/services/notifications-manager.service.spec.js.map +1 -1
- package/server/applications/notifications/services/notifications-queries.service.js +1 -1
- package/server/applications/notifications/services/notifications-queries.service.js.map +1 -1
- package/server/applications/shares/services/shares-manager.service.js +3 -2
- package/server/applications/shares/services/shares-manager.service.js.map +1 -1
- package/server/applications/sync/constants/auth.js +2 -2
- package/server/applications/sync/constants/auth.js.map +1 -1
- package/server/applications/sync/dtos/sync-client-registration.dto.js +5 -0
- package/server/applications/sync/dtos/sync-client-registration.dto.js.map +1 -1
- package/server/applications/sync/dtos/sync-operations.dto.js +1 -2
- package/server/applications/sync/dtos/sync-operations.dto.js.map +1 -1
- package/server/applications/sync/schemas/sync-clients.schema.js +2 -1
- package/server/applications/sync/schemas/sync-clients.schema.js.map +1 -1
- package/server/applications/sync/schemas/sync-paths.schema.js +2 -1
- package/server/applications/sync/schemas/sync-paths.schema.js.map +1 -1
- package/server/applications/sync/services/sync-clients-manager.service.js +28 -20
- package/server/applications/sync/services/sync-clients-manager.service.js.map +1 -1
- package/server/applications/sync/services/sync-clients-manager.service.spec.js +24 -18
- package/server/applications/sync/services/sync-clients-manager.service.spec.js.map +1 -1
- package/server/applications/sync/services/sync-queries.service.js +5 -5
- package/server/applications/sync/services/sync-queries.service.js.map +1 -1
- package/server/applications/users/admin-users.controller.js +48 -37
- package/server/applications/users/admin-users.controller.js.map +1 -1
- package/server/applications/users/admin-users.controller.spec.js +15 -0
- package/server/applications/users/admin-users.controller.spec.js.map +1 -1
- package/server/applications/users/constants/routes.js +5 -0
- package/server/applications/users/constants/routes.js.map +1 -1
- package/server/applications/users/constants/user.js +8 -0
- package/server/applications/users/constants/user.js.map +1 -1
- package/server/applications/users/dto/delete-user.dto.js +5 -23
- package/server/applications/users/dto/delete-user.dto.js.map +1 -1
- package/server/applications/users/dto/user-properties.dto.js +38 -3
- package/server/applications/users/dto/user-properties.dto.js.map +1 -1
- package/server/applications/users/interfaces/admin-user.interface.js.map +1 -1
- package/server/applications/users/interfaces/user-secrets.interface.js +10 -0
- package/server/applications/users/interfaces/user-secrets.interface.js.map +1 -0
- package/server/applications/users/models/user.model.js +84 -50
- package/server/applications/users/models/user.model.js.map +1 -1
- package/server/applications/users/schemas/user.interface.js.map +1 -1
- package/server/applications/users/schemas/users.schema.js +2 -0
- package/server/applications/users/schemas/users.schema.js.map +1 -1
- package/server/applications/users/services/admin-users-manager.service.js +7 -19
- package/server/applications/users/services/admin-users-manager.service.js.map +1 -1
- package/server/applications/users/services/admin-users-manager.service.spec.js +7 -26
- package/server/applications/users/services/admin-users-manager.service.spec.js.map +1 -1
- package/server/applications/users/services/admin-users-queries.service.js +1 -0
- package/server/applications/users/services/admin-users-queries.service.js.map +1 -1
- package/server/applications/users/services/users-manager.service.js +138 -28
- package/server/applications/users/services/users-manager.service.js.map +1 -1
- package/server/applications/users/services/users-manager.service.spec.js +11 -9
- package/server/applications/users/services/users-manager.service.spec.js.map +1 -1
- package/server/applications/users/services/users-queries.service.js +63 -57
- package/server/applications/users/services/users-queries.service.js.map +1 -1
- package/server/applications/users/users.controller.js +48 -1
- package/server/applications/users/users.controller.js.map +1 -1
- package/server/applications/users/users.controller.spec.js +8 -1
- package/server/applications/users/users.controller.spec.js.map +1 -1
- package/server/applications/users/users.e2e-spec.js +2 -1
- package/server/applications/users/users.e2e-spec.js.map +1 -1
- package/server/applications/users/utils/avatar.js +48 -0
- package/server/applications/users/utils/avatar.js.map +1 -0
- package/server/authentication/auth.config.js +85 -26
- package/server/authentication/auth.config.js.map +1 -1
- package/server/authentication/auth.controller.js +117 -9
- package/server/authentication/auth.controller.js.map +1 -1
- package/server/authentication/auth.controller.spec.js +16 -1
- package/server/authentication/auth.controller.spec.js.map +1 -1
- package/server/authentication/auth.e2e-spec.js +4 -3
- package/server/authentication/auth.e2e-spec.js.map +1 -1
- package/server/authentication/auth.module.js +4 -1
- package/server/authentication/auth.module.js.map +1 -1
- package/server/authentication/constants/auth.js +37 -4
- package/server/authentication/constants/auth.js.map +1 -1
- package/server/authentication/constants/routes.js +21 -0
- package/server/authentication/constants/routes.js.map +1 -1
- package/server/authentication/constants/scope.js +20 -0
- package/server/authentication/constants/scope.js.map +1 -0
- package/server/authentication/dto/login-response.dto.js +27 -4
- package/server/authentication/dto/login-response.dto.js.map +1 -1
- package/server/authentication/dto/token-response.dto.js +5 -0
- package/server/authentication/dto/token-response.dto.js.map +1 -1
- package/server/{applications/users/dto/user-password.dto.js → authentication/dto/two-fa-verify.dto.js} +27 -9
- package/server/authentication/dto/two-fa-verify.dto.js.map +1 -0
- package/server/authentication/guards/auth-basic.strategy.js +6 -5
- package/server/authentication/guards/auth-basic.strategy.js.map +1 -1
- package/server/authentication/guards/auth-token-access.strategy.js +3 -2
- package/server/authentication/guards/auth-token-access.strategy.js.map +1 -1
- package/server/authentication/guards/auth-token-refresh.strategy.js +3 -2
- package/server/authentication/guards/auth-token-refresh.strategy.js.map +1 -1
- package/server/authentication/guards/auth-two-fa-guard.js +81 -0
- package/server/authentication/guards/auth-two-fa-guard.js.map +1 -0
- package/server/authentication/interfaces/jwt-payload.interface.js +5 -0
- package/server/authentication/interfaces/jwt-payload.interface.js.map +1 -1
- package/server/authentication/interfaces/token.interface.js +2 -0
- package/server/authentication/interfaces/token.interface.js.map +1 -1
- package/server/authentication/interfaces/two-fa-setup.interface.js +10 -0
- package/server/authentication/interfaces/two-fa-setup.interface.js.map +1 -0
- package/server/authentication/models/auth-method.js.map +1 -1
- package/server/authentication/services/auth-manager.service.js +72 -49
- package/server/authentication/services/auth-manager.service.js.map +1 -1
- package/server/authentication/services/auth-methods/auth-method-database.service.js +3 -3
- package/server/authentication/services/auth-methods/auth-method-database.service.js.map +1 -1
- package/server/authentication/services/auth-methods/auth-method-database.service.spec.js +5 -0
- package/server/authentication/services/auth-methods/auth-method-database.service.spec.js.map +1 -1
- package/server/authentication/services/auth-methods/auth-method-ldap.service.js +100 -27
- package/server/authentication/services/auth-methods/auth-method-ldap.service.js.map +1 -1
- package/server/authentication/services/auth-methods/auth-method-ldap.service.spec.js +11 -12
- package/server/authentication/services/auth-methods/auth-method-ldap.service.spec.js.map +1 -1
- package/server/authentication/services/auth-methods/auth-method-two-fa.service.js +251 -0
- package/server/authentication/services/auth-methods/auth-method-two-fa.service.js.map +1 -0
- package/server/authentication/services/auth-methods/auth-method-two-fa.service.spec.js +41 -0
- package/server/authentication/services/auth-methods/auth-method-two-fa.service.spec.js.map +1 -0
- package/server/authentication/utils/crypt-secret.js +68 -0
- package/server/authentication/utils/crypt-secret.js.map +1 -0
- package/server/common/functions.js +18 -2
- package/server/common/functions.js.map +1 -1
- package/server/common/qrcode.js +34 -0
- package/server/common/qrcode.js.map +1 -0
- package/server/common/shared.js +18 -0
- package/server/common/shared.js.map +1 -1
- package/server/configuration/config.environment.js +23 -6
- package/server/configuration/config.environment.js.map +1 -1
- package/server/configuration/config.interfaces.js +10 -0
- package/server/configuration/config.interfaces.js.map +1 -0
- package/server/configuration/config.loader.js.map +1 -1
- package/server/configuration/config.validation.js +13 -13
- package/server/configuration/config.validation.js.map +1 -1
- package/server/infrastructure/cache/adapters/mysql-cache.adapter.js +6 -6
- package/server/infrastructure/cache/adapters/mysql-cache.adapter.js.map +1 -1
- package/server/infrastructure/cache/schemas/mysql-cache.schema.js +2 -1
- package/server/infrastructure/cache/schemas/mysql-cache.schema.js.map +1 -1
- package/server/infrastructure/cache/services/cache.service.js.map +1 -1
- package/server/infrastructure/database/columns.js +39 -0
- package/server/infrastructure/database/columns.js.map +1 -0
- package/server/infrastructure/database/database.config.js +0 -1
- package/server/infrastructure/database/database.config.js.map +1 -1
- package/server/infrastructure/mailer/interfaces/mail.interface.js.map +1 -1
- package/server/infrastructure/mailer/mailer.config.js +12 -0
- package/server/infrastructure/mailer/mailer.config.js.map +1 -1
- package/server/infrastructure/mailer/mailer.service.js +2 -1
- package/server/infrastructure/mailer/mailer.service.js.map +1 -1
- package/static/assets/mimes/text-x-c.svg +1 -0
- package/static/chunk-2TZUZMCM.js +4 -0
- package/static/chunk-2XJ5Z2GZ.js +1 -0
- package/static/{chunk-7VRUZRJG.js → chunk-5M4YJZUB.js} +2 -2
- package/static/{chunk-MRSWNAVB.js → chunk-5ZGQYTS2.js} +1 -1
- package/static/chunk-6BFNMDUD.js +1 -0
- package/static/chunk-6IRL673W.js +559 -0
- package/static/{chunk-2R6HHGUR.js → chunk-ABGR5AYC.js} +1 -1
- package/static/chunk-CN27VAGB.js +1 -0
- package/static/{chunk-MVO4WZLK.js → chunk-DNMO47SY.js} +1 -1
- package/static/{chunk-MGGT6MIJ.js → chunk-EI4PVI2W.js} +1 -1
- package/static/chunk-ET6QDNNM.js +1 -0
- package/static/{chunk-L6MU6S2V.js → chunk-G2TKYYWK.js} +1 -1
- package/static/chunk-G3FOG2QB.js +1 -0
- package/static/{chunk-MCLQFZ3S.js → chunk-GCUWGVYT.js} +1 -1
- package/static/{chunk-RSS6GYNE.js → chunk-HME7LAEY.js} +1 -1
- package/static/chunk-IEUANP3Q.js +1 -0
- package/static/{chunk-VJRTMDEJ.js → chunk-IIFHIIC6.js} +1 -1
- package/static/{chunk-YJMN3B4N.js → chunk-KPZ7FEMO.js} +1 -1
- package/static/{chunk-JYXLQRHG.js → chunk-M57NVD4V.js} +1 -1
- package/static/chunk-NN3VQOS7.js +1 -0
- package/static/chunk-NW3CTYUW.js +1 -0
- package/static/{chunk-6OJZWYRZ.js → chunk-O3ANXCPE.js} +1 -1
- package/static/{chunk-ZC5NIT55.js → chunk-QFOMEU3T.js} +1 -1
- package/static/{chunk-BIUNUYZ5.js → chunk-RKNTQYMU.js} +1 -1
- package/static/{chunk-VUI3KV7V.js → chunk-UQ4TRQCE.js} +1 -1
- package/static/{chunk-WI7FOANP.js → chunk-WINILGQN.js} +1 -1
- package/static/{chunk-NE4NDO45.js → chunk-X7MFVDBY.js} +1 -1
- package/static/chunk-XCBLEI2E.js +1 -0
- package/static/{chunk-CRQNEHTX.js → chunk-XLWCV4HI.js} +1 -1
- package/static/chunk-XPIYOZBX.js +4 -0
- package/static/{chunk-LLWSLOSX.js → chunk-YD74UCFG.js} +1 -1
- package/static/{chunk-IZL7JPTS.js → chunk-YDFVKH2D.js} +1 -1
- package/static/{chunk-SPTF6FSM.js → chunk-YVJDYSDE.js} +1 -1
- package/static/index.html +2 -2
- package/static/main-QNBKYA6L.js +9 -0
- package/static/{styles-FYUSO6OJ.css → styles-A5VYX3CE.css} +1 -1
- package/server/applications/users/dto/user-password.dto.js.map +0 -1
- package/static/chunk-4U5A2DEP.js +0 -4
- package/static/chunk-54EAZ2UD.js +0 -1
- package/static/chunk-7ZRXJONB.js +0 -1
- package/static/chunk-F2J2IIJE.js +0 -1
- package/static/chunk-FNFGUIQH.js +0 -4
- package/static/chunk-GGLK52CG.js +0 -1
- package/static/chunk-HW2H3ISM.js +0 -559
- package/static/chunk-HX6BBYVD.js +0 -1
- package/static/chunk-JF7S3UYQ.js +0 -1
- package/static/chunk-KSHPKI4G.js +0 -1
- package/static/chunk-VPJ2V27B.js +0 -1
- package/static/chunk-ZXS4V7J2.js +0 -1
- package/static/main-FFIWFD2F.js +0 -7
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../../../backend/src/applications/users/services/users-queries.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Inject, Injectable, Logger } from '@nestjs/common'\nimport { and, countDistinct, eq, inArray, like, lte, ne, notInArray, or, SelectedFields, SQL, sql } from 'drizzle-orm'\nimport { alias } from 'drizzle-orm/mysql-core'\nimport { MySql2PreparedQuery, MySqlQueryResult } from 'drizzle-orm/mysql2'\nimport { anonymizePassword, comparePassword, uniquePermissions } from '../../../common/functions'\nimport { CacheDecorator } from '../../../infrastructure/cache/cache.decorator'\nimport { Cache } from '../../../infrastructure/cache/services/cache.service'\nimport { DB_TOKEN_PROVIDER } from '../../../infrastructure/database/constants'\nimport { DBSchema } from '../../../infrastructure/database/interfaces/database.interface'\nimport {\n concatDistinctObjectsInArray,\n convertToSelect,\n dateTimeUTC,\n dbCheckAffectedRows,\n dbGetInsertedId\n} from '../../../infrastructure/database/utils'\nimport { GROUP_TYPE, GROUP_VISIBILITY } from '../constants/group'\nimport { MEMBER_TYPE } from '../constants/member'\nimport { USER_GROUP_ROLE, USER_ONLINE_STATUS, USER_PERMS_SEP, USER_ROLE } from '../constants/user'\nimport { UserCreateOrUpdateGroupDto } from '../dto/create-or-update-group.dto'\nimport { CreateUserDto } from '../dto/create-or-update-user.dto'\nimport { SearchMembersDto } from '../dto/search-members.dto'\nimport { GroupMember, GroupWithMembers } from '../interfaces/group-member'\nimport { GuestUser } from '../interfaces/guest-user.interface'\nimport { Member } from '../interfaces/member.interface'\nimport { UserOnline } from '../interfaces/websocket.interface'\nimport { UserModel } from '../models/user.model'\nimport { Group } from '../schemas/group.interface'\nimport { groups } from '../schemas/groups.schema'\nimport { UserGroup } from '../schemas/user-group.interface'\nimport { User } from '../schemas/user.interface'\nimport { usersGroups } from '../schemas/users-groups.schema'\nimport { usersGuests } from '../schemas/users-guests.schema'\nimport { userFullNameSQL, users } from '../schemas/users.schema'\n\n@Injectable()\nexport class UsersQueries {\n private readonly logger = new Logger(UsersQueries.name)\n private fromLoginOrEmailPermissionsQuery: MySql2PreparedQuery<any> = null\n private fromIdPermissionsQuery: MySql2PreparedQuery<any> = null\n\n constructor(\n @Inject(DB_TOKEN_PROVIDER) private readonly db: DBSchema,\n private readonly cache: Cache\n ) {}\n\n checkUserExists(login?: string, email?: string): Promise<{ login?: string; email?: string }> {\n if (!login && !email) {\n throw new Error('login or email must be specified')\n }\n const columns: { login?: boolean; email?: boolean } = {}\n const where: SQL[] = []\n if (login) {\n columns.login = true\n where.push(eq(users.login, login))\n }\n if (email) {\n columns.email = true\n where.push(eq(users.email, email))\n }\n const operator = login && email ? or : and\n return this.db.query.users.findFirst({\n columns: columns,\n where: operator(...where)\n })\n }\n\n setOnlineStatus(userId: number, onlineStatus: USER_ONLINE_STATUS): Promise<boolean> {\n return this.updateUserOrGuest(userId, { onlineStatus: onlineStatus })\n }\n\n getOnlineUsers(userIds: number[]): Promise<UserOnline[]> {\n return this.db\n .select({\n id: users.id,\n login: users.login,\n email: users.email,\n fullName: userFullNameSQL(users),\n onlineStatus: users.onlineStatus\n } satisfies UserOnline | SelectedFields<any, any>)\n .from(users)\n .where(inArray(users.id, userIds))\n }\n\n async checkGroupNameExists(groupName: string): Promise<boolean> {\n const [group] = await this.db.select({ name: groups.name }).from(groups).where(eq(groups.name, groupName)).limit(1)\n return !!group?.name\n }\n\n async compareUserPassword(userId: number, password: string): Promise<boolean> {\n const [hash] = (await this.selectUsers(['password'], [eq(users.id, userId)])) as { password: string }[]\n if (!hash) return false\n return comparePassword(password, hash.password)\n }\n\n async from(userId?: number, loginOrEmail?: string): Promise<User> {\n // retrieve user with application permissions\n let pQuery: MySql2PreparedQuery<any> = userId ? this.fromIdPermissionsQuery : this.fromLoginOrEmailPermissionsQuery\n if (!pQuery) {\n const where = userId\n ? eq(users.id, sql.placeholder('userId'))\n : or(eq(users.login, sql.placeholder('loginOrEmail')), eq(users.email, sql.placeholder('loginOrEmail')))\n pQuery = this.db\n .select({\n user: users,\n groupsPermissions: sql`GROUP_CONCAT(DISTINCT (${groups.permissions}) SEPARATOR ${USER_PERMS_SEP})`\n })\n .from(users)\n .leftJoin(usersGroups, eq(usersGroups.userId, users.id))\n .leftJoin(groups, and(eq(groups.id, usersGroups.groupId), ne(groups.permissions, '')))\n .where(where)\n .groupBy(users.id)\n .limit(1)\n .prepare()\n if (userId) {\n this.fromIdPermissionsQuery = pQuery\n } else {\n this.fromLoginOrEmailPermissionsQuery = pQuery\n }\n }\n const r = await pQuery.execute(userId ? { userId } : { loginOrEmail })\n if (!r.length) return null\n const [user, groupsPermissions] = [r[0].user, r[0].groupsPermissions]\n // merge user and groups permissions\n user.permissions = uniquePermissions(`${user.permissions},${groupsPermissions}`, USER_PERMS_SEP)\n return user\n }\n\n selectUsers(fields: Partial<keyof User>[] = ['id', 'login', 'email'], where: SQL[]): Promise<Partial<User>[]> {\n const select: Record<keyof User, any> = convertToSelect(users, fields)\n return this.db\n .select(select)\n .from(users)\n .where(and(...where))\n }\n\n async selectUserProperties(userId: number, fields: Partial<keyof User>[]): Promise<Record<string, any>> {\n const select: Record<keyof User, any> = convertToSelect(users, fields)\n const [r]: Record<string, any>[] = await this.db.select(select).from(users).where(eq(users.id, userId)).limit(1)\n return r\n }\n\n async createUserOrGuest(createUserDto: CreateUserDto, userRole: USER_ROLE): Promise<User['id']> {\n const userId: number = dbGetInsertedId(await this.db.insert(users).values({ ...createUserDto, role: userRole } as User))\n if (userRole === USER_ROLE.USER && createUserDto.groups?.length) {\n await this.db.insert(usersGroups).values(createUserDto.groups.map((gid: number) => ({ userId: userId, groupId: gid })))\n }\n if (userRole === USER_ROLE.GUEST && createUserDto.managers?.length) {\n await this.db.insert(usersGuests).values(createUserDto.managers.map((uid: number) => ({ guestId: userId, userId: uid })))\n }\n return userId\n }\n\n async updateUserOrGuest(userId: number, set: Partial<Record<keyof User, any>>, userRole?: USER_ROLE): Promise<boolean> {\n try {\n dbCheckAffectedRows(\n await this.db\n .update(users)\n .set({ ...set, ...(userRole && { role: userRole }) } as User)\n .where(eq(users.id, userId)),\n 1\n )\n this.logger.verbose(`${this.updateUserOrGuest.name} - user (${userId}) was updated : ${JSON.stringify(anonymizePassword(set))}`)\n return true\n } catch (e) {\n this.logger.error(`${this.updateUserOrGuest.name} - user (${userId}) was not updated : ${JSON.stringify(anonymizePassword(set))} : ${e}`)\n return false\n }\n }\n\n async deleteGuestLink(userId: number): Promise<void> {\n dbCheckAffectedRows(await this.db.delete(users).where(and(eq(users.id, userId), eq(users.role, USER_ROLE.LINK))), 1)\n }\n\n async searchUsersOrGroups(searchMembersDto: SearchMembersDto, userId?: number): Promise<Member[]> {\n const limit = searchMembersDto.onlyUsers || searchMembersDto.onlyGroups ? 6 : 3\n const members: Member[] = []\n if (!searchMembersDto.onlyGroups) {\n for (const u of await this.searchUsers(searchMembersDto, userId, limit)) {\n members.push({\n id: u.id,\n login: u.login,\n name: u.fullName,\n description: u.email,\n type: u.role === USER_ROLE.GUEST ? MEMBER_TYPE.GUEST : MEMBER_TYPE.USER,\n permissions: searchMembersDto.withPermissions ? u.permissions : undefined\n })\n }\n }\n if (!searchMembersDto.onlyUsers) {\n for (const g of await this.searchGroups(searchMembersDto, userId, limit)) {\n members.push({\n id: g.id,\n name: g.name,\n description: g.description,\n type: g.type === GROUP_TYPE.USER ? MEMBER_TYPE.GROUP : MEMBER_TYPE.PGROUP,\n permissions: searchMembersDto.withPermissions ? g.permissions : undefined\n })\n }\n }\n return members\n }\n\n async groupFromName(userId: number, name: string): Promise<Pick<Group, 'id' | 'name' | 'type'> & { role: UserGroup['role'] }> {\n const [group] = await this.db\n .select({\n id: groups.id,\n name: groups.name,\n type: groups.type,\n role: usersGroups.role\n } satisfies (Pick<Group, 'id' | 'name' | 'type'> & { role: UserGroup['role'] }) | SelectedFields<any, any>)\n .from(usersGroups)\n .innerJoin(groups, eq(groups.id, usersGroups.groupId))\n .where(and(eq(usersGroups.userId, userId), eq(groups.name, name)))\n .limit(1)\n return group\n }\n\n async browseRootGroups(userId: number): Promise<Member[]> {\n const members = alias(usersGroups, 'members')\n return this.db\n .select({\n id: groups.id,\n name: groups.name,\n description: groups.description,\n createdAt: groups.createdAt,\n modifiedAt: groups.modifiedAt,\n type: sql<MEMBER_TYPE>`IF(${groups.type} = ${GROUP_TYPE.USER}, ${MEMBER_TYPE.GROUP}, ${MEMBER_TYPE.PGROUP})`,\n groupRole: sql<USER_GROUP_ROLE>`${usersGroups.role}`,\n counts: { users: countDistinct(members.userId) }\n } satisfies Member | SelectedFields<any, any>)\n .from(usersGroups)\n .innerJoin(groups, and(eq(groups.id, usersGroups.groupId), eq(usersGroups.userId, userId)))\n .leftJoin(members, eq(members.groupId, groups.id))\n .groupBy(groups.id)\n }\n\n async browseGroupMembers(groupId: number): Promise<Member[]> {\n return this.db\n .select({\n id: users.id,\n login: users.login,\n name: userFullNameSQL(users).as('name'),\n description: users.email,\n createdAt: usersGroups.createdAt,\n type: sql<MEMBER_TYPE>`${MEMBER_TYPE.USER}`,\n groupRole: sql<USER_GROUP_ROLE>`${usersGroups.role}`\n } satisfies Member | SelectedFields<any, any>)\n .from(groups)\n .innerJoin(usersGroups, and(eq(usersGroups.groupId, groups.id), eq(usersGroups.groupId, groupId)))\n .leftJoin(users, eq(users.id, usersGroups.userId))\n .groupBy(users.id)\n }\n\n async canDeletePersonalGroup(userId: number, groupId: number): Promise<boolean> {\n const [group] = await this.db\n .select({ id: usersGroups.groupId })\n .from(usersGroups)\n .innerJoin(groups, and(eq(groups.id, usersGroups.groupId)))\n .where(\n and(\n eq(groups.type, GROUP_TYPE.PERSONAL),\n eq(usersGroups.userId, userId),\n eq(usersGroups.groupId, groupId),\n eq(usersGroups.role, USER_GROUP_ROLE.MANAGER)\n )\n )\n .limit(1)\n return !!group?.id\n }\n\n async getGroup(userId: number, groupId: number, asAdmin = false): Promise<GroupMember> {\n const [group] = await this.db\n .select({\n id: groups.id,\n name: groups.name,\n description: groups.description,\n createdAt: groups.createdAt,\n modifiedAt: groups.modifiedAt,\n type: sql<MEMBER_TYPE>`IF(${groups.type} = ${GROUP_TYPE.USER}, ${MEMBER_TYPE.GROUP}, ${MEMBER_TYPE.PGROUP})`\n })\n .from(usersGroups)\n .innerJoin(groups, and(eq(groups.id, usersGroups.groupId)))\n .where(\n and(\n eq(usersGroups.groupId, groupId),\n sql`IF(${+asAdmin} = 0, ${usersGroups.userId} = ${userId} AND ${usersGroups.role} = ${USER_GROUP_ROLE.MANAGER}, 1)`\n )\n )\n .limit(1)\n return group\n }\n\n async getGroupWithMembers(userId: number, groupId: number, asAdmin = false): Promise<GroupWithMembers> {\n const usersGroupsAlias: any = alias(usersGroups, 'usersFromGroups')\n const [group] = await this.db\n .select({\n id: groups.id,\n name: groups.name,\n description: groups.description,\n createdAt: groups.createdAt,\n modifiedAt: groups.modifiedAt,\n type: sql<MEMBER_TYPE>`IF(${groups.type} = ${GROUP_TYPE.USER}, ${sql.raw(`'${MEMBER_TYPE.GROUP}'`)}, ${sql.raw(`'${MEMBER_TYPE.PGROUP}'`)})`,\n members: concatDistinctObjectsInArray(users.id, {\n id: users.id,\n login: users.login,\n name: userFullNameSQL(users),\n description: users.email,\n type: sql.raw(`'${MEMBER_TYPE.USER}'`),\n groupRole: usersGroupsAlias.role,\n createdAt: dateTimeUTC(usersGroupsAlias.createdAt)\n } satisfies Record<keyof Pick<Member, 'id' | 'name' | 'login' | 'description' | 'type' | 'groupRole' | 'createdAt'>, any>)\n } satisfies GroupWithMembers | SelectedFields<any, any>)\n .from(usersGroups)\n .innerJoin(groups, eq(groups.id, usersGroups.groupId))\n .leftJoin(usersGroupsAlias, and(eq(usersGroupsAlias.groupId, groups.id)))\n .leftJoin(users, eq(users.id, usersGroupsAlias.userId))\n .where(\n and(\n eq(usersGroups.groupId, groupId),\n sql`IF(${+asAdmin} = 0, ${usersGroups.userId} = ${userId} AND ${usersGroups.role} = ${USER_GROUP_ROLE.MANAGER}, 1)`\n )\n )\n .groupBy(groups.id)\n .limit(1)\n return group\n }\n\n async deletePersonalGroup(groupId: number): Promise<boolean> {\n return dbCheckAffectedRows(\n await this.db\n .delete(groups)\n .where(and(eq(groups.id, groupId), eq(groups.type, GROUP_TYPE.PERSONAL)))\n .limit(1),\n 1,\n false\n )\n }\n\n async createPersonalGroup(managerId: number, userCreateOrUpdateGroupDto: UserCreateOrUpdateGroupDto): Promise<Group['id']> {\n const groupId: number = dbGetInsertedId(\n await this.db.insert(groups).values({\n ...userCreateOrUpdateGroupDto,\n type: GROUP_TYPE.PERSONAL,\n visibility: GROUP_VISIBILITY.PRIVATE\n } as Group)\n )\n await this.db.insert(usersGroups).values({ userId: managerId, groupId: groupId, role: USER_GROUP_ROLE.MANAGER } as UserGroup)\n return groupId\n }\n\n async updateGroup(groupId: number, set: Partial<Record<keyof Group, any>>) {\n if (Object.keys(set).length) {\n try {\n await this.db.update(groups).set(set).where(eq(groups.id, groupId))\n this.logger.log(`${this.updateGroup.name} - group (${groupId}) was updated : ${JSON.stringify(set)}`)\n } catch (e) {\n this.logger.error(`${this.updateGroup.name} - group (${groupId}) was not updated : ${JSON.stringify(set)} : ${e}`)\n throw new Error('Group was not updated')\n }\n }\n }\n\n async updateGroupMembers(\n groupId: number,\n members: {\n add?: Pick<Member, 'id' | 'groupRole'>[]\n remove?: UserGroup['userId'][]\n }\n ): Promise<void> {\n if (members?.add?.length) {\n try {\n await this.db.insert(usersGroups).values(members.add.map((m) => ({ userId: m.id, groupId: groupId, role: m.groupRole })))\n // clear cache\n this.clearWhiteListCaches(members.add.map((m) => m.id))\n this.logger.log(`${this.updateGroupMembers.name} - users ${JSON.stringify(members.add.map((m) => m.id))} was added to group (${groupId})`)\n } catch (e) {\n this.logger.error(\n `${this.updateGroupMembers.name} - users ${JSON.stringify(members.add.map((m) => m.id))} was not added to group (${groupId}) : ${e}`\n )\n throw new Error('Group members was not added')\n }\n }\n if (members?.remove?.length) {\n try {\n await this.db\n .delete(usersGroups)\n .where(and(eq(usersGroups.groupId, groupId), inArray(usersGroups.userId, members.remove)))\n .limit(members.remove.length)\n // clear cache\n this.clearWhiteListCaches(members.remove)\n this.logger.log(`${this.updateGroupMembers.name} - users ${JSON.stringify(members.remove)} was removed from group (${groupId})`)\n } catch (e) {\n this.logger.error(`${this.updateGroupMembers.name} - users ${JSON.stringify(members.remove)} was not removed from group (${groupId}) : ${e}`)\n throw new Error('Group members was not removed')\n }\n }\n }\n\n async listGuests(guestId: null, managerId?: number, asAdmin?: boolean): Promise<GuestUser[]>\n async listGuests(guestId: number, managerId?: number, asAdmin?: boolean): Promise<GuestUser>\n async listGuests(guestId: number | null, managerId?: number, asAdmin = false): Promise<GuestUser | GuestUser[]> {\n const where: SQL[] = [...(guestId ? [eq(usersGuests.guestId, guestId)] : []), ...(asAdmin ? [] : [eq(usersGuests.userId, managerId)])]\n const managersGuestAlias: any = alias(usersGuests, 'managersGuestAlias')\n const managersAlias: any = alias(users, 'managersAlias')\n const guests = await this.db\n .select({\n id: users.id,\n login: users.login,\n email: users.email,\n firstName: users.firstName,\n lastName: users.lastName,\n fullName: userFullNameSQL(users),\n role: users.role,\n isActive: users.isActive,\n passwordAttempts: users.passwordAttempts,\n language: users.language,\n notification: users.notification,\n currentAccess: users.currentAccess,\n lastAccess: users.lastAccess,\n currentIp: users.currentIp,\n lastIp: users.lastIp,\n createdAt: users.createdAt,\n managers: concatDistinctObjectsInArray(managersAlias.id, {\n id: managersAlias.id,\n login: managersAlias.login,\n name: userFullNameSQL(managersAlias),\n type: sql.raw(`'${MEMBER_TYPE.USER}'`),\n description: managersAlias.email,\n createdAt: dateTimeUTC(managersGuestAlias.createdAt)\n } satisfies Record<keyof Pick<Member, 'id' | 'name' | 'login' | 'description' | 'type' | 'createdAt'>, any>)\n } satisfies GuestUser | SelectedFields<any, any>)\n .from(usersGuests)\n .innerJoin(users, and(eq(users.id, usersGuests.guestId), eq(users.role, USER_ROLE.GUEST)))\n .leftJoin(managersGuestAlias, eq(managersGuestAlias.guestId, users.id))\n .leftJoin(managersAlias, eq(managersAlias.id, managersGuestAlias.userId))\n .where(and(...where))\n .groupBy(users.id)\n .limit(guestId ? 1 : undefined)\n return guestId ? guests[0] : guests\n }\n\n async isGuestManager(managerId: number, guestId: number): Promise<{ id: number; login: string } | undefined> {\n const [guest] = await this.db\n .select({ id: usersGuests.guestId, login: users.login })\n .from(usersGuests)\n .innerJoin(users, eq(users.id, usersGuests.guestId))\n .where(and(eq(usersGuests.userId, managerId), eq(usersGuests.guestId, guestId), eq(users.role, USER_ROLE.GUEST)))\n .limit(1)\n return guest\n }\n\n private async searchGroups(\n searchMembersDto: SearchMembersDto,\n userId?: number,\n limit = 3\n ): Promise<Pick<Group, 'id' | 'name' | 'description' | 'type' | 'permissions'>[]> {\n /* Search for groups */\n const where: SQL[] = [like(groups.name, `%${searchMembersDto.search}%`)]\n if (userId) {\n let idsWhitelist: number[] = await this.groupsWhitelist(userId)\n if (searchMembersDto.ignoreGroupIds?.length) {\n idsWhitelist = idsWhitelist.filter((id) => searchMembersDto.ignoreGroupIds.indexOf(id) === -1)\n }\n where.unshift(inArray(groups.id, idsWhitelist))\n } else if (searchMembersDto.ignoreGroupIds?.length) {\n where.unshift(notInArray(groups.id, searchMembersDto.ignoreGroupIds))\n }\n if (searchMembersDto.excludePersonalGroups) {\n where.unshift(eq(groups.type, GROUP_TYPE.USER))\n }\n return this.db\n .select({ id: groups.id, name: groups.name, description: groups.description, type: groups.type, permissions: groups.permissions })\n .from(groups)\n .where(and(...where))\n .limit(limit)\n }\n\n private async searchUsers(\n searchMembersDto: SearchMembersDto,\n userId?: number,\n limit = 3\n ): Promise<Pick<UserModel, 'id' | 'login' | 'email' | 'fullName' | 'role' | 'permissions'>[]> {\n /* Search for users */\n const where: SQL[] = [\n ne(users.role, USER_ROLE.LINK),\n or(like(sql`CONCAT_WS('-', ${users.login}, ${users.email}, ${users.firstName}, ${users.lastName})`, `%${searchMembersDto.search}%`))\n ]\n if (userId) {\n let idsWhitelist: number[] = await this.usersWhitelist(userId)\n if (searchMembersDto.ignoreUserIds?.length) {\n idsWhitelist = idsWhitelist.filter((id) => searchMembersDto.ignoreUserIds.indexOf(id) === -1)\n }\n where.unshift(inArray(users.id, idsWhitelist))\n } else {\n if (searchMembersDto.ignoreUserIds?.length) {\n where.unshift(notInArray(users.id, searchMembersDto.ignoreUserIds))\n }\n }\n if (typeof searchMembersDto.usersRole !== 'undefined') {\n if (searchMembersDto.usersRole === USER_ROLE.USER) {\n // allow admin users\n where.unshift(lte(users.role, searchMembersDto.usersRole))\n } else {\n where.unshift(eq(users.role, searchMembersDto.usersRole))\n }\n }\n return this.db\n .select({\n id: users.id,\n login: users.login,\n email: users.email,\n fullName: userFullNameSQL(users),\n role: users.role,\n permissions: users.permissions\n })\n .from(users)\n .where(and(...where))\n .limit(limit)\n }\n\n @CacheDecorator(900)\n async usersWhitelist(userId: number, lowerOrEqualUserRole: USER_ROLE = USER_ROLE.GUEST): Promise<number[]> {\n /* Get the list of user ids allowed to the current user\n - all users with no groups (except link users)\n - all users who are members of the current user's groups (excluding link users and members of isolated groups)\n - all guests managed by the current user\n - all managers who manage the current guest\n */\n const usersAlias: any = alias(users, 'usersAlias')\n const groupsAlias: any = alias(groups, 'groupsAlias')\n const userIds: any = sql`\n WITH RECURSIVE children (id, parentId) AS\n (SELECT ${groups.id},\n ${groups.parentId}\n FROM ${groups}\n WHERE (${groups.id} IN (SELECT ${usersGroups.groupId}\n FROM ${usersGroups}\n WHERE ${usersGroups.userId} = ${userId}\n AND ${groups.visibility} != ${GROUP_VISIBILITY.ISOLATED}))\n UNION\n SELECT ${groupsAlias.id},\n ${groupsAlias.parentId}\n FROM ${groups} AS groupsAlias\n INNER JOIN children cs ON ${groupsAlias.parentId} = cs.id AND ${groupsAlias.visibility} = ${sql.raw(`${GROUP_VISIBILITY.VISIBLE}`)})\n SELECT JSON_ARRAYAGG(id) AS ids\n FROM (\n -- Users from visible child groups\n SELECT ${users.id} AS id\n FROM children\n INNER JOIN ${usersGroups} ON ${usersGroups.groupId} = children.id\n INNER JOIN ${users} ON ${usersGroups.userId} = ${users.id} AND ${users.role} <= ${sql.raw(`${lowerOrEqualUserRole}`)}\n\n UNION\n -- Users visible but not assigned to groups\n SELECT ${usersAlias.id} AS id\n FROM ${users} AS usersAlias\n INNER JOIN ${users} ON ${users.id} = ${usersAlias.id} AND ${users.role} <= ${sql.raw(`${lowerOrEqualUserRole}`)}\n WHERE NOT EXISTS (SELECT ${usersGroups.userId} FROM ${usersGroups} WHERE ${usersGroups.userId} = ${usersAlias.id})\n UNION\n -- Users or guests that are manager/managed\n SELECT CASE\n WHEN ${usersGuests.userId} = ${userId} THEN ${usersGuests.guestId}\n WHEN ${usersGuests.guestId} = ${userId} THEN ${usersGuests.userId}\n END AS id\n FROM ${usersGuests}\n WHERE ${usersGuests.userId} = ${userId}\n OR ${usersGuests.guestId} = ${userId}) AS usersUnion\n `\n const [r] = await this.db.execute(userIds)\n return JSON.parse(r[0].ids) || []\n }\n\n @CacheDecorator(900)\n async groupsWhitelist(userId: number): Promise<number[]> {\n /* Get the list of groups ids allowed to the current user\n - all parent groups for which the user is a member (includes personal groups, excludes isolated groups)\n - all subgroups inherited from parent groups\n */\n const groupsAlias: any = alias(groups, 'groupsAlias')\n const groupIds: any = sql`\n WITH RECURSIVE children (id, parentId, type) AS\n (SELECT ${groups.id},\n ${groups.parentId},\n ${groups.type}\n FROM ${groups}\n WHERE (${groups.id} IN (SELECT ${usersGroups.groupId}\n FROM ${usersGroups}\n WHERE ${usersGroups.userId} = ${userId}\n AND ${groups.visibility} != ${GROUP_VISIBILITY.ISOLATED}))\n UNION\n SELECT ${groupsAlias.id},\n ${groupsAlias.parentId},\n ${groupsAlias.type}\n FROM ${groups} AS groupsAlias\n INNER JOIN children cs ON ${groupsAlias.parentId} = cs.id AND ${groupsAlias.visibility} = ${GROUP_VISIBILITY.VISIBLE})\n SELECT JSON_ARRAYAGG(children.id) as ids\n FROM children\n `\n const [r] = await this.db.execute(groupIds)\n return JSON.parse(r[0].ids) || []\n }\n\n clearWhiteListCaches(userIds: number[]) {\n this.cache\n .mdel([\n ...userIds.map((id) => this.cache.genSlugKey(this.constructor.name, this.usersWhitelist.name, id)),\n ...userIds.map((id) => this.cache.genSlugKey(this.constructor.name, this.groupsWhitelist.name, id))\n ])\n .catch((e: Error) => this.logger.error(`${this.clearWhiteListCaches.name} - ${e}`))\n }\n\n async allUserIdsFromGroupsAndSubGroups(groupIds: number[]): Promise<number[]> {\n if (!groupIds.length) return []\n const subGroup: any = alias(groups, 'subGroup')\n const withChildren: any = sql`\n WITH RECURSIVE child (id, parentId) AS\n (SELECT ${groups.id}, ${groups.parentId}\n FROM ${groups}\n WHERE ${inArray(groups.id, groupIds)}\n UNION\n SELECT ${subGroup.id},\n ${subGroup.parentId}\n FROM ${groups} AS subGroup\n INNER JOIN child AS cs ON ${subGroup.parentId} = cs.id)\n SELECT DISTINCT ${usersGroups.userId} as userId\n FROM child\n INNER JOIN ${usersGroups} ON child.id = ${usersGroups.groupId}\n `\n const [r]: { userId: number }[][] = (await this.db.execute(withChildren)) as MySqlQueryResult\n return r.length ? r.map((r) => r.userId) : []\n }\n}\n"],"names":["UsersQueries","checkUserExists","login","email","Error","columns","where","push","eq","users","operator","or","and","db","query","findFirst","setOnlineStatus","userId","onlineStatus","updateUserOrGuest","getOnlineUsers","userIds","select","id","fullName","userFullNameSQL","from","inArray","checkGroupNameExists","groupName","group","name","groups","limit","compareUserPassword","password","hash","selectUsers","comparePassword","loginOrEmail","pQuery","fromIdPermissionsQuery","fromLoginOrEmailPermissionsQuery","sql","placeholder","user","groupsPermissions","permissions","USER_PERMS_SEP","leftJoin","usersGroups","groupId","ne","groupBy","prepare","r","execute","length","uniquePermissions","fields","convertToSelect","selectUserProperties","createUserOrGuest","createUserDto","userRole","dbGetInsertedId","insert","values","role","USER_ROLE","USER","map","gid","GUEST","managers","usersGuests","uid","guestId","set","dbCheckAffectedRows","update","logger","verbose","JSON","stringify","anonymizePassword","e","error","deleteGuestLink","delete","LINK","searchUsersOrGroups","searchMembersDto","onlyUsers","onlyGroups","members","u","searchUsers","description","type","MEMBER_TYPE","withPermissions","undefined","g","searchGroups","GROUP_TYPE","GROUP","PGROUP","groupFromName","innerJoin","browseRootGroups","alias","createdAt","modifiedAt","groupRole","counts","countDistinct","browseGroupMembers","as","canDeletePersonalGroup","PERSONAL","USER_GROUP_ROLE","MANAGER","getGroup","asAdmin","getGroupWithMembers","usersGroupsAlias","raw","concatDistinctObjectsInArray","dateTimeUTC","deletePersonalGroup","createPersonalGroup","managerId","userCreateOrUpdateGroupDto","visibility","GROUP_VISIBILITY","PRIVATE","updateGroup","Object","keys","log","updateGroupMembers","add","m","clearWhiteListCaches","remove","listGuests","managersGuestAlias","managersAlias","guests","firstName","lastName","isActive","passwordAttempts","language","notification","currentAccess","lastAccess","currentIp","lastIp","isGuestManager","guest","like","search","idsWhitelist","groupsWhitelist","ignoreGroupIds","filter","indexOf","unshift","notInArray","excludePersonalGroups","usersWhitelist","ignoreUserIds","usersRole","lte","lowerOrEqualUserRole","usersAlias","groupsAlias","parentId","ISOLATED","VISIBLE","parse","ids","groupIds","cache","mdel","genSlugKey","catch","allUserIdsFromGroupsAndSubGroups","subGroup","withChildren","Logger"],"mappings":"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"}
|
|
1
|
+
{"version":3,"sources":["../../../../../backend/src/applications/users/services/users-queries.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Inject, Injectable, Logger } from '@nestjs/common'\nimport { and, countDistinct, eq, inArray, like, lte, ne, notInArray, or, SelectedFields, SQL, sql } from 'drizzle-orm'\nimport { alias } from 'drizzle-orm/mysql-core'\nimport { MySql2PreparedQuery, MySqlQueryResult } from 'drizzle-orm/mysql2'\nimport { anonymizePassword, comparePassword, uniquePermissions } from '../../../common/functions'\nimport { CacheDecorator } from '../../../infrastructure/cache/cache.decorator'\nimport { Cache } from '../../../infrastructure/cache/services/cache.service'\nimport { DB_TOKEN_PROVIDER } from '../../../infrastructure/database/constants'\nimport { DBSchema } from '../../../infrastructure/database/interfaces/database.interface'\nimport {\n concatDistinctObjectsInArray,\n convertToSelect,\n dateTimeUTC,\n dbCheckAffectedRows,\n dbGetInsertedId\n} from '../../../infrastructure/database/utils'\nimport { GROUP_TYPE, GROUP_VISIBILITY } from '../constants/group'\nimport { MEMBER_TYPE } from '../constants/member'\nimport { USER_GROUP_ROLE, USER_ONLINE_STATUS, USER_PERMS_SEP, USER_ROLE } from '../constants/user'\nimport { UserCreateOrUpdateGroupDto } from '../dto/create-or-update-group.dto'\nimport { CreateUserDto } from '../dto/create-or-update-user.dto'\nimport { SearchMembersDto } from '../dto/search-members.dto'\nimport { GroupMember, GroupWithMembers } from '../interfaces/group-member'\nimport { GuestUser } from '../interfaces/guest-user.interface'\nimport { Member } from '../interfaces/member.interface'\nimport { UserSecrets } from '../interfaces/user-secrets.interface'\nimport { UserOnline } from '../interfaces/websocket.interface'\nimport { UserModel } from '../models/user.model'\nimport { Group } from '../schemas/group.interface'\nimport { groups } from '../schemas/groups.schema'\nimport { UserGroup } from '../schemas/user-group.interface'\nimport { User } from '../schemas/user.interface'\nimport { usersGroups } from '../schemas/users-groups.schema'\nimport { usersGuests } from '../schemas/users-guests.schema'\nimport { userFullNameSQL, users } from '../schemas/users.schema'\n\n@Injectable()\nexport class UsersQueries {\n private readonly logger = new Logger(UsersQueries.name)\n private fromLoginOrEmailPermissionsQuery: MySql2PreparedQuery<any> = null\n private fromIdPermissionsQuery: MySql2PreparedQuery<any> = null\n\n constructor(\n @Inject(DB_TOKEN_PROVIDER) private readonly db: DBSchema,\n private readonly cache: Cache\n ) {}\n\n checkUserExists(login?: string, email?: string): Promise<{ login?: string; email?: string }> {\n if (!login && !email) {\n throw new Error('login or email must be specified')\n }\n const columns: { login?: boolean; email?: boolean } = {}\n const where: SQL[] = []\n if (login) {\n columns.login = true\n where.push(eq(users.login, login))\n }\n if (email) {\n columns.email = true\n where.push(eq(users.email, email))\n }\n const operator = login && email ? or : and\n return this.db.query.users.findFirst({\n columns: columns,\n where: operator(...where)\n })\n }\n\n setOnlineStatus(userId: number, onlineStatus: USER_ONLINE_STATUS): Promise<boolean> {\n return this.updateUserOrGuest(userId, { onlineStatus: onlineStatus })\n }\n\n getOnlineUsers(userIds: number[]): Promise<UserOnline[]> {\n return this.db\n .select({\n id: users.id,\n login: users.login,\n email: users.email,\n fullName: userFullNameSQL(users),\n onlineStatus: users.onlineStatus\n } satisfies UserOnline | SelectedFields<any, any>)\n .from(users)\n .where(inArray(users.id, userIds))\n }\n\n async checkGroupNameExists(groupName: string): Promise<boolean> {\n const [group] = await this.db.select({ name: groups.name }).from(groups).where(eq(groups.name, groupName)).limit(1)\n return !!group?.name\n }\n\n async compareUserPassword(userId: number, password: string): Promise<boolean> {\n const [hash] = (await this.selectUsers(['password'], [eq(users.id, userId)])) as { password: string }[]\n if (!hash) return false\n return comparePassword(password, hash.password)\n }\n\n async from(userId?: number, loginOrEmail?: string): Promise<User> {\n // retrieve user with application permissions\n let pQuery: MySql2PreparedQuery<any> = userId ? this.fromIdPermissionsQuery : this.fromLoginOrEmailPermissionsQuery\n if (!pQuery) {\n const where = userId\n ? eq(users.id, sql.placeholder('userId'))\n : or(eq(users.login, sql.placeholder('loginOrEmail')), eq(users.email, sql.placeholder('loginOrEmail')))\n pQuery = this.db\n .select({\n user: users,\n groupsPermissions: sql`GROUP_CONCAT(DISTINCT (${groups.permissions}) SEPARATOR ${USER_PERMS_SEP})`\n })\n .from(users)\n .leftJoin(usersGroups, eq(usersGroups.userId, users.id))\n .leftJoin(groups, and(eq(groups.id, usersGroups.groupId), ne(groups.permissions, '')))\n .where(where)\n .groupBy(users.id)\n .limit(1)\n .prepare()\n if (userId) {\n this.fromIdPermissionsQuery = pQuery\n } else {\n this.fromLoginOrEmailPermissionsQuery = pQuery\n }\n }\n const r = await pQuery.execute(userId ? { userId } : { loginOrEmail })\n if (!r.length) return null\n const [user, groupsPermissions] = [r[0].user, r[0].groupsPermissions]\n // merge user and groups permissions\n user.permissions = uniquePermissions(`${user.permissions},${groupsPermissions}`, USER_PERMS_SEP)\n return user\n }\n\n async getUserSecrets(userId: number): Promise<UserSecrets> {\n const [r]: { secrets: UserSecrets }[] = await this.db.select({ secrets: users.secrets }).from(users).where(eq(users.id, userId)).limit(1)\n return r.secrets || {}\n }\n\n selectUsers(fields: Partial<keyof User>[] = ['id', 'login', 'email'], where: SQL[]): Promise<Partial<User>[]> {\n const select: Record<keyof User, any> = convertToSelect(users, fields)\n return this.db\n .select(select)\n .from(users)\n .where(and(...where))\n }\n\n async selectUserProperties(userId: number, fields: Partial<keyof User>[]): Promise<Partial<User>> {\n const select: Record<keyof User, any> = convertToSelect(users, fields)\n const [r]: Record<string, any>[] = await this.db.select(select).from(users).where(eq(users.id, userId)).limit(1)\n return r\n }\n\n async createUserOrGuest(createUserDto: CreateUserDto, userRole: USER_ROLE): Promise<User['id']> {\n const userId: number = dbGetInsertedId(await this.db.insert(users).values({ ...createUserDto, role: userRole } as User))\n if (userRole === USER_ROLE.USER && createUserDto.groups?.length) {\n await this.db.insert(usersGroups).values(createUserDto.groups.map((gid: number) => ({ userId: userId, groupId: gid })))\n }\n if (userRole === USER_ROLE.GUEST && createUserDto.managers?.length) {\n await this.db.insert(usersGuests).values(createUserDto.managers.map((uid: number) => ({ guestId: userId, userId: uid })))\n }\n return userId\n }\n\n async updateUserOrGuest(userId: number, set: Partial<Record<keyof User, any>>, userRole?: USER_ROLE): Promise<boolean> {\n try {\n dbCheckAffectedRows(\n await this.db\n .update(users)\n .set({ ...set, ...(userRole && { role: userRole }) } as User)\n .where(eq(users.id, userId)),\n 1\n )\n this.logger.verbose(`${this.updateUserOrGuest.name} - user (${userId}) was updated : ${JSON.stringify(anonymizePassword(set))}`)\n return true\n } catch (e) {\n this.logger.error(`${this.updateUserOrGuest.name} - user (${userId}) was not updated : ${JSON.stringify(anonymizePassword(set))} : ${e}`)\n return false\n }\n }\n\n async deleteGuestLink(userId: number): Promise<void> {\n dbCheckAffectedRows(await this.db.delete(users).where(and(eq(users.id, userId), eq(users.role, USER_ROLE.LINK))), 1)\n }\n\n async searchUsersOrGroups(searchMembersDto: SearchMembersDto, userId?: number): Promise<Member[]> {\n const limit = searchMembersDto.onlyUsers || searchMembersDto.onlyGroups ? 6 : 3\n const members: Member[] = []\n if (!searchMembersDto.onlyGroups) {\n for (const u of await this.searchUsers(searchMembersDto, userId, limit)) {\n members.push({\n id: u.id,\n login: u.login,\n name: u.fullName,\n description: u.email,\n type: u.role === USER_ROLE.GUEST ? MEMBER_TYPE.GUEST : MEMBER_TYPE.USER,\n permissions: searchMembersDto.withPermissions ? u.permissions : undefined\n })\n }\n }\n if (!searchMembersDto.onlyUsers) {\n for (const g of await this.searchGroups(searchMembersDto, userId, limit)) {\n members.push({\n id: g.id,\n name: g.name,\n description: g.description,\n type: g.type === GROUP_TYPE.USER ? MEMBER_TYPE.GROUP : MEMBER_TYPE.PGROUP,\n permissions: searchMembersDto.withPermissions ? g.permissions : undefined\n })\n }\n }\n return members\n }\n\n async groupFromName(userId: number, name: string): Promise<Pick<Group, 'id' | 'name' | 'type'> & { role: UserGroup['role'] }> {\n const [group] = await this.db\n .select({\n id: groups.id,\n name: groups.name,\n type: groups.type,\n role: usersGroups.role\n } satisfies (Pick<Group, 'id' | 'name' | 'type'> & { role: UserGroup['role'] }) | SelectedFields<any, any>)\n .from(usersGroups)\n .innerJoin(groups, eq(groups.id, usersGroups.groupId))\n .where(and(eq(usersGroups.userId, userId), eq(groups.name, name)))\n .limit(1)\n return group\n }\n\n async browseRootGroups(userId: number): Promise<Member[]> {\n const members = alias(usersGroups, 'members')\n return this.db\n .select({\n id: groups.id,\n name: groups.name,\n description: groups.description,\n createdAt: groups.createdAt,\n modifiedAt: groups.modifiedAt,\n type: sql<MEMBER_TYPE>`IF(${groups.type} = ${GROUP_TYPE.USER}, ${MEMBER_TYPE.GROUP}, ${MEMBER_TYPE.PGROUP})`,\n groupRole: sql<USER_GROUP_ROLE>`${usersGroups.role}`,\n counts: { users: countDistinct(members.userId) }\n } satisfies Member | SelectedFields<any, any>)\n .from(usersGroups)\n .innerJoin(groups, and(eq(groups.id, usersGroups.groupId), eq(usersGroups.userId, userId)))\n .leftJoin(members, eq(members.groupId, groups.id))\n .groupBy(groups.id)\n }\n\n async browseGroupMembers(groupId: number): Promise<Member[]> {\n return this.db\n .select({\n id: users.id,\n login: users.login,\n name: userFullNameSQL(users).as('name'),\n description: users.email,\n createdAt: usersGroups.createdAt,\n type: sql<MEMBER_TYPE>`${MEMBER_TYPE.USER}`,\n groupRole: sql<USER_GROUP_ROLE>`${usersGroups.role}`\n } satisfies Member | SelectedFields<any, any>)\n .from(groups)\n .innerJoin(usersGroups, and(eq(usersGroups.groupId, groups.id), eq(usersGroups.groupId, groupId)))\n .leftJoin(users, eq(users.id, usersGroups.userId))\n .groupBy(users.id)\n }\n\n async canDeletePersonalGroup(userId: number, groupId: number): Promise<boolean> {\n const [group] = await this.db\n .select({ id: usersGroups.groupId })\n .from(usersGroups)\n .innerJoin(groups, and(eq(groups.id, usersGroups.groupId)))\n .where(\n and(\n eq(groups.type, GROUP_TYPE.PERSONAL),\n eq(usersGroups.userId, userId),\n eq(usersGroups.groupId, groupId),\n eq(usersGroups.role, USER_GROUP_ROLE.MANAGER)\n )\n )\n .limit(1)\n return !!group?.id\n }\n\n async getGroup(userId: number, groupId: number, asAdmin = false): Promise<GroupMember> {\n const [group] = await this.db\n .select({\n id: groups.id,\n name: groups.name,\n description: groups.description,\n createdAt: groups.createdAt,\n modifiedAt: groups.modifiedAt,\n type: sql<MEMBER_TYPE>`IF(${groups.type} = ${GROUP_TYPE.USER}, ${MEMBER_TYPE.GROUP}, ${MEMBER_TYPE.PGROUP})`\n })\n .from(usersGroups)\n .innerJoin(groups, and(eq(groups.id, usersGroups.groupId)))\n .where(\n and(\n eq(usersGroups.groupId, groupId),\n sql`IF(${+asAdmin} = 0, ${usersGroups.userId} = ${userId} AND ${usersGroups.role} = ${USER_GROUP_ROLE.MANAGER}, 1)`\n )\n )\n .limit(1)\n return group\n }\n\n async getGroupWithMembers(userId: number, groupId: number, asAdmin = false): Promise<GroupWithMembers> {\n const usersGroupsAlias: any = alias(usersGroups, 'usersFromGroups')\n const [group] = await this.db\n .select({\n id: groups.id,\n name: groups.name,\n description: groups.description,\n createdAt: groups.createdAt,\n modifiedAt: groups.modifiedAt,\n type: sql<MEMBER_TYPE>`IF(${groups.type} = ${GROUP_TYPE.USER}, ${sql.raw(`'${MEMBER_TYPE.GROUP}'`)}, ${sql.raw(`'${MEMBER_TYPE.PGROUP}'`)})`,\n members: concatDistinctObjectsInArray(users.id, {\n id: users.id,\n login: users.login,\n name: userFullNameSQL(users),\n description: users.email,\n type: sql.raw(`'${MEMBER_TYPE.USER}'`),\n groupRole: usersGroupsAlias.role,\n createdAt: dateTimeUTC(usersGroupsAlias.createdAt)\n } satisfies Record<keyof Pick<Member, 'id' | 'name' | 'login' | 'description' | 'type' | 'groupRole' | 'createdAt'>, any>)\n } satisfies GroupWithMembers | SelectedFields<any, any>)\n .from(usersGroups)\n .innerJoin(groups, eq(groups.id, usersGroups.groupId))\n .leftJoin(usersGroupsAlias, and(eq(usersGroupsAlias.groupId, groups.id)))\n .leftJoin(users, eq(users.id, usersGroupsAlias.userId))\n .where(\n and(\n eq(usersGroups.groupId, groupId),\n sql`IF(${+asAdmin} = 0, ${usersGroups.userId} = ${userId} AND ${usersGroups.role} = ${USER_GROUP_ROLE.MANAGER}, 1)`\n )\n )\n .groupBy(groups.id)\n .limit(1)\n return group\n }\n\n async deletePersonalGroup(groupId: number): Promise<boolean> {\n return dbCheckAffectedRows(\n await this.db\n .delete(groups)\n .where(and(eq(groups.id, groupId), eq(groups.type, GROUP_TYPE.PERSONAL)))\n .limit(1),\n 1,\n false\n )\n }\n\n async createPersonalGroup(managerId: number, userCreateOrUpdateGroupDto: UserCreateOrUpdateGroupDto): Promise<Group['id']> {\n const groupId: number = dbGetInsertedId(\n await this.db.insert(groups).values({\n ...userCreateOrUpdateGroupDto,\n type: GROUP_TYPE.PERSONAL,\n visibility: GROUP_VISIBILITY.PRIVATE\n } as Group)\n )\n await this.db.insert(usersGroups).values({ userId: managerId, groupId: groupId, role: USER_GROUP_ROLE.MANAGER } as UserGroup)\n return groupId\n }\n\n async updateGroup(groupId: number, set: Partial<Record<keyof Group, any>>) {\n if (Object.keys(set).length) {\n try {\n await this.db.update(groups).set(set).where(eq(groups.id, groupId))\n this.logger.log(`${this.updateGroup.name} - group (${groupId}) was updated : ${JSON.stringify(set)}`)\n } catch (e) {\n this.logger.error(`${this.updateGroup.name} - group (${groupId}) was not updated : ${JSON.stringify(set)} : ${e}`)\n throw new Error('Group was not updated')\n }\n }\n }\n\n async updateGroupMembers(\n groupId: number,\n members: {\n add?: Pick<Member, 'id' | 'groupRole'>[]\n remove?: UserGroup['userId'][]\n }\n ): Promise<void> {\n if (members?.add?.length) {\n try {\n await this.db.insert(usersGroups).values(members.add.map((m) => ({ userId: m.id, groupId: groupId, role: m.groupRole })))\n // clear cache\n this.clearWhiteListCaches(members.add.map((m) => m.id))\n this.logger.log(`${this.updateGroupMembers.name} - users ${JSON.stringify(members.add.map((m) => m.id))} was added to group (${groupId})`)\n } catch (e) {\n this.logger.error(\n `${this.updateGroupMembers.name} - users ${JSON.stringify(members.add.map((m) => m.id))} was not added to group (${groupId}) : ${e}`\n )\n throw new Error('Group members was not added')\n }\n }\n if (members?.remove?.length) {\n try {\n await this.db\n .delete(usersGroups)\n .where(and(eq(usersGroups.groupId, groupId), inArray(usersGroups.userId, members.remove)))\n .limit(members.remove.length)\n // clear cache\n this.clearWhiteListCaches(members.remove)\n this.logger.log(`${this.updateGroupMembers.name} - users ${JSON.stringify(members.remove)} was removed from group (${groupId})`)\n } catch (e) {\n this.logger.error(`${this.updateGroupMembers.name} - users ${JSON.stringify(members.remove)} was not removed from group (${groupId}) : ${e}`)\n throw new Error('Group members was not removed')\n }\n }\n }\n\n async listGuests(guestId: null, managerId?: number, asAdmin?: boolean): Promise<GuestUser[]>\n async listGuests(guestId: number, managerId?: number, asAdmin?: boolean): Promise<GuestUser>\n async listGuests(guestId: number | null, managerId?: number, asAdmin = false): Promise<GuestUser | GuestUser[]> {\n const where: SQL[] = [...(guestId ? [eq(usersGuests.guestId, guestId)] : []), ...(asAdmin ? [] : [eq(usersGuests.userId, managerId)])]\n const managersGuestAlias: any = alias(usersGuests, 'managersGuestAlias')\n const managersAlias: any = alias(users, 'managersAlias')\n const guests = await this.db\n .select({\n id: users.id,\n login: users.login,\n email: users.email,\n firstName: users.firstName,\n lastName: users.lastName,\n fullName: userFullNameSQL(users),\n role: users.role,\n isActive: users.isActive,\n passwordAttempts: users.passwordAttempts,\n language: users.language,\n notification: users.notification,\n currentAccess: users.currentAccess,\n lastAccess: users.lastAccess,\n currentIp: users.currentIp,\n lastIp: users.lastIp,\n createdAt: users.createdAt,\n managers: concatDistinctObjectsInArray(managersAlias.id, {\n id: managersAlias.id,\n login: managersAlias.login,\n name: userFullNameSQL(managersAlias),\n type: sql.raw(`'${MEMBER_TYPE.USER}'`),\n description: managersAlias.email,\n createdAt: dateTimeUTC(managersGuestAlias.createdAt)\n } satisfies Record<keyof Pick<Member, 'id' | 'name' | 'login' | 'description' | 'type' | 'createdAt'>, any>)\n } satisfies GuestUser | SelectedFields<any, any>)\n .from(usersGuests)\n .innerJoin(users, and(eq(users.id, usersGuests.guestId), eq(users.role, USER_ROLE.GUEST)))\n .leftJoin(managersGuestAlias, eq(managersGuestAlias.guestId, users.id))\n .leftJoin(managersAlias, eq(managersAlias.id, managersGuestAlias.userId))\n .where(and(...where))\n .groupBy(users.id)\n .limit(guestId ? 1 : undefined)\n return guestId ? guests[0] : guests\n }\n\n async isGuestManager(managerId: number, guestId: number): Promise<{ id: number; login: string } | undefined> {\n const [guest] = await this.db\n .select({ id: usersGuests.guestId, login: users.login })\n .from(usersGuests)\n .innerJoin(users, eq(users.id, usersGuests.guestId))\n .where(and(eq(usersGuests.userId, managerId), eq(usersGuests.guestId, guestId), eq(users.role, USER_ROLE.GUEST)))\n .limit(1)\n return guest\n }\n\n @CacheDecorator(900)\n async usersWhitelist(userId: number, lowerOrEqualUserRole: USER_ROLE = USER_ROLE.GUEST): Promise<number[]> {\n /* Get the list of user ids allowed to the current user\n - all users with no groups (except link users)\n - all users who are members of the current user's groups (excluding link users and members of isolated groups)\n - all guests managed by the current user\n - all managers who manage the current guest\n */\n const usersAlias: any = alias(users, 'usersAlias')\n const groupsAlias: any = alias(groups, 'groupsAlias')\n const userIds: any = sql`\n WITH RECURSIVE children (id, parentId) AS\n (SELECT ${groups.id},\n ${groups.parentId}\n FROM ${groups}\n WHERE (${groups.id} IN (SELECT ${usersGroups.groupId}\n FROM ${usersGroups}\n WHERE ${usersGroups.userId} = ${userId}\n AND ${groups.visibility} != ${GROUP_VISIBILITY.ISOLATED}))\n UNION\n SELECT ${groupsAlias.id},\n ${groupsAlias.parentId}\n FROM ${groups} AS groupsAlias\n INNER JOIN children cs ON ${groupsAlias.parentId} = cs.id AND ${groupsAlias.visibility} = ${sql.raw(`${GROUP_VISIBILITY.VISIBLE}`)})\n SELECT JSON_ARRAYAGG(id) AS ids\n FROM (\n -- Users from visible child groups\n SELECT ${users.id} AS id\n FROM children\n INNER JOIN ${usersGroups} ON ${usersGroups.groupId} = children.id\n INNER JOIN ${users} ON ${usersGroups.userId} = ${users.id} AND ${users.role} <= ${sql.raw(`${lowerOrEqualUserRole}`)}\n\n UNION\n -- Users visible but not assigned to groups\n SELECT ${usersAlias.id} AS id\n FROM ${users} AS usersAlias\n INNER JOIN ${users} ON ${users.id} = ${usersAlias.id} AND ${users.role} <= ${sql.raw(`${lowerOrEqualUserRole}`)}\n WHERE NOT EXISTS (SELECT ${usersGroups.userId} FROM ${usersGroups} WHERE ${usersGroups.userId} = ${usersAlias.id})\n UNION\n -- Users or guests that are manager/managed\n SELECT CASE\n WHEN ${usersGuests.userId} = ${userId} THEN ${usersGuests.guestId}\n WHEN ${usersGuests.guestId} = ${userId} THEN ${usersGuests.userId}\n END AS id\n FROM ${usersGuests}\n WHERE ${usersGuests.userId} = ${userId}\n OR ${usersGuests.guestId} = ${userId}) AS usersUnion\n `\n const [r] = await this.db.execute(userIds)\n return JSON.parse(r[0].ids) || []\n }\n\n @CacheDecorator(900)\n async groupsWhitelist(userId: number): Promise<number[]> {\n /* Get the list of groups ids allowed to the current user\n - all parent groups for which the user is a member (includes personal groups, excludes isolated groups)\n - all subgroups inherited from parent groups\n */\n const groupsAlias: any = alias(groups, 'groupsAlias')\n const groupIds: any = sql`\n WITH RECURSIVE children (id, parentId, type) AS\n (SELECT ${groups.id},\n ${groups.parentId},\n ${groups.type}\n FROM ${groups}\n WHERE (${groups.id} IN (SELECT ${usersGroups.groupId}\n FROM ${usersGroups}\n WHERE ${usersGroups.userId} = ${userId}\n AND ${groups.visibility} != ${GROUP_VISIBILITY.ISOLATED}))\n UNION\n SELECT ${groupsAlias.id},\n ${groupsAlias.parentId},\n ${groupsAlias.type}\n FROM ${groups} AS groupsAlias\n INNER JOIN children cs ON ${groupsAlias.parentId} = cs.id AND ${groupsAlias.visibility} = ${GROUP_VISIBILITY.VISIBLE})\n SELECT JSON_ARRAYAGG(children.id) as ids\n FROM children\n `\n const [r] = await this.db.execute(groupIds)\n return JSON.parse(r[0].ids) || []\n }\n\n clearWhiteListCaches(userIds: number[]) {\n this.cache\n .mdel([\n ...userIds.map((id) => this.cache.genSlugKey(this.constructor.name, this.usersWhitelist.name, id)),\n ...userIds.map((id) => this.cache.genSlugKey(this.constructor.name, this.groupsWhitelist.name, id))\n ])\n .catch((e: Error) => this.logger.error(`${this.clearWhiteListCaches.name} - ${e}`))\n }\n\n async allUserIdsFromGroupsAndSubGroups(groupIds: number[]): Promise<number[]> {\n if (!groupIds.length) return []\n const subGroup: any = alias(groups, 'subGroup')\n const withChildren: any = sql`\n WITH RECURSIVE child (id, parentId) AS\n (SELECT ${groups.id}, ${groups.parentId}\n FROM ${groups}\n WHERE ${inArray(groups.id, groupIds)}\n UNION\n SELECT ${subGroup.id},\n ${subGroup.parentId}\n FROM ${groups} AS subGroup\n INNER JOIN child AS cs ON ${subGroup.parentId} = cs.id)\n SELECT DISTINCT ${usersGroups.userId} as userId\n FROM child\n INNER JOIN ${usersGroups} ON child.id = ${usersGroups.groupId}\n `\n const [r]: { userId: number }[][] = (await this.db.execute(withChildren)) as MySqlQueryResult\n return r.length ? r.map((r) => r.userId) : []\n }\n\n private async searchGroups(\n searchMembersDto: SearchMembersDto,\n userId?: number,\n limit = 3\n ): Promise<Pick<Group, 'id' | 'name' | 'description' | 'type' | 'permissions'>[]> {\n /* Search for groups */\n const where: SQL[] = [like(groups.name, `%${searchMembersDto.search}%`)]\n if (userId) {\n let idsWhitelist: number[] = await this.groupsWhitelist(userId)\n if (searchMembersDto.ignoreGroupIds?.length) {\n idsWhitelist = idsWhitelist.filter((id) => searchMembersDto.ignoreGroupIds.indexOf(id) === -1)\n }\n where.unshift(inArray(groups.id, idsWhitelist))\n } else if (searchMembersDto.ignoreGroupIds?.length) {\n where.unshift(notInArray(groups.id, searchMembersDto.ignoreGroupIds))\n }\n if (searchMembersDto.excludePersonalGroups) {\n where.unshift(eq(groups.type, GROUP_TYPE.USER))\n }\n return this.db\n .select({ id: groups.id, name: groups.name, description: groups.description, type: groups.type, permissions: groups.permissions })\n .from(groups)\n .where(and(...where))\n .limit(limit)\n }\n\n private async searchUsers(\n searchMembersDto: SearchMembersDto,\n userId?: number,\n limit = 3\n ): Promise<Pick<UserModel, 'id' | 'login' | 'email' | 'fullName' | 'role' | 'permissions'>[]> {\n /* Search for users */\n const where: SQL[] = [\n ne(users.role, USER_ROLE.LINK),\n or(like(sql`CONCAT_WS('-', ${users.login}, ${users.email}, ${users.firstName}, ${users.lastName})`, `%${searchMembersDto.search}%`))\n ]\n if (userId) {\n let idsWhitelist: number[] = await this.usersWhitelist(userId)\n if (searchMembersDto.ignoreUserIds?.length) {\n idsWhitelist = idsWhitelist.filter((id) => searchMembersDto.ignoreUserIds.indexOf(id) === -1)\n }\n where.unshift(inArray(users.id, idsWhitelist))\n } else {\n if (searchMembersDto.ignoreUserIds?.length) {\n where.unshift(notInArray(users.id, searchMembersDto.ignoreUserIds))\n }\n }\n if (typeof searchMembersDto.usersRole !== 'undefined') {\n if (searchMembersDto.usersRole === USER_ROLE.USER) {\n // allow admin users\n where.unshift(lte(users.role, searchMembersDto.usersRole))\n } else {\n where.unshift(eq(users.role, searchMembersDto.usersRole))\n }\n }\n return this.db\n .select({\n id: users.id,\n login: users.login,\n email: users.email,\n fullName: userFullNameSQL(users),\n role: users.role,\n permissions: users.permissions\n })\n .from(users)\n .where(and(...where))\n .limit(limit)\n }\n}\n"],"names":["UsersQueries","checkUserExists","login","email","Error","columns","where","push","eq","users","operator","or","and","db","query","findFirst","setOnlineStatus","userId","onlineStatus","updateUserOrGuest","getOnlineUsers","userIds","select","id","fullName","userFullNameSQL","from","inArray","checkGroupNameExists","groupName","group","name","groups","limit","compareUserPassword","password","hash","selectUsers","comparePassword","loginOrEmail","pQuery","fromIdPermissionsQuery","fromLoginOrEmailPermissionsQuery","sql","placeholder","user","groupsPermissions","permissions","USER_PERMS_SEP","leftJoin","usersGroups","groupId","ne","groupBy","prepare","r","execute","length","uniquePermissions","getUserSecrets","secrets","fields","convertToSelect","selectUserProperties","createUserOrGuest","createUserDto","userRole","dbGetInsertedId","insert","values","role","USER_ROLE","USER","map","gid","GUEST","managers","usersGuests","uid","guestId","set","dbCheckAffectedRows","update","logger","verbose","JSON","stringify","anonymizePassword","e","error","deleteGuestLink","delete","LINK","searchUsersOrGroups","searchMembersDto","onlyUsers","onlyGroups","members","u","searchUsers","description","type","MEMBER_TYPE","withPermissions","undefined","g","searchGroups","GROUP_TYPE","GROUP","PGROUP","groupFromName","innerJoin","browseRootGroups","alias","createdAt","modifiedAt","groupRole","counts","countDistinct","browseGroupMembers","as","canDeletePersonalGroup","PERSONAL","USER_GROUP_ROLE","MANAGER","getGroup","asAdmin","getGroupWithMembers","usersGroupsAlias","raw","concatDistinctObjectsInArray","dateTimeUTC","deletePersonalGroup","createPersonalGroup","managerId","userCreateOrUpdateGroupDto","visibility","GROUP_VISIBILITY","PRIVATE","updateGroup","Object","keys","log","updateGroupMembers","add","m","clearWhiteListCaches","remove","listGuests","managersGuestAlias","managersAlias","guests","firstName","lastName","isActive","passwordAttempts","language","notification","currentAccess","lastAccess","currentIp","lastIp","isGuestManager","guest","usersWhitelist","lowerOrEqualUserRole","usersAlias","groupsAlias","parentId","ISOLATED","VISIBLE","parse","ids","groupsWhitelist","groupIds","cache","mdel","genSlugKey","catch","allUserIdsFromGroupsAndSubGroups","subGroup","withChildren","like","search","idsWhitelist","ignoreGroupIds","filter","indexOf","unshift","notInArray","excludePersonalGroups","ignoreUserIds","usersRole","lte","Logger"],"mappings":"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"}
|
|
@@ -14,6 +14,7 @@ Object.defineProperty(exports, "UsersController", {
|
|
|
14
14
|
});
|
|
15
15
|
const _common = require("@nestjs/common");
|
|
16
16
|
const _fs = require("fs");
|
|
17
|
+
const _authtwofaguard = require("../../authentication/guards/auth-two-fa-guard");
|
|
17
18
|
const _authrequestinterface = require("../../authentication/interfaces/auth-request.interface");
|
|
18
19
|
const _routes = require("./constants/routes");
|
|
19
20
|
const _user = require("./constants/user");
|
|
@@ -46,6 +47,15 @@ let UsersController = class UsersController {
|
|
|
46
47
|
me(user) {
|
|
47
48
|
return this.usersManager.me(user);
|
|
48
49
|
}
|
|
50
|
+
listAppPasswords(user) {
|
|
51
|
+
return this.usersManager.listAppPasswords(user);
|
|
52
|
+
}
|
|
53
|
+
generateAppPassword(user, userAppPasswordDto) {
|
|
54
|
+
return this.usersManager.generateAppPassword(user, userAppPasswordDto);
|
|
55
|
+
}
|
|
56
|
+
deleteAppPassword(user, name) {
|
|
57
|
+
return this.usersManager.deleteAppPassword(user, name);
|
|
58
|
+
}
|
|
49
59
|
updateLanguage(user, userLanguageDto) {
|
|
50
60
|
return this.usersManager.updateLanguage(user, userLanguageDto);
|
|
51
61
|
}
|
|
@@ -124,6 +134,42 @@ _ts_decorate([
|
|
|
124
134
|
]),
|
|
125
135
|
_ts_metadata("design:returntype", typeof Promise === "undefined" ? Object : Promise)
|
|
126
136
|
], UsersController.prototype, "me", null);
|
|
137
|
+
_ts_decorate([
|
|
138
|
+
(0, _common.Get)(`${_routes.USERS_ROUTE.ME}/${_routes.USERS_ROUTE.APP_PASSWORDS}`),
|
|
139
|
+
(0, _rolesdecorator.UserHaveRole)(_user.USER_ROLE.USER),
|
|
140
|
+
(0, _common.UseGuards)(_authtwofaguard.AuthTwoFaGuardWithoutPassword),
|
|
141
|
+
_ts_param(0, (0, _userdecorator.GetUser)()),
|
|
142
|
+
_ts_metadata("design:type", Function),
|
|
143
|
+
_ts_metadata("design:paramtypes", [
|
|
144
|
+
typeof _usermodel.UserModel === "undefined" ? Object : _usermodel.UserModel
|
|
145
|
+
]),
|
|
146
|
+
_ts_metadata("design:returntype", typeof Promise === "undefined" ? Object : Promise)
|
|
147
|
+
], UsersController.prototype, "listAppPasswords", null);
|
|
148
|
+
_ts_decorate([
|
|
149
|
+
(0, _common.Post)(`${_routes.USERS_ROUTE.ME}/${_routes.USERS_ROUTE.APP_PASSWORDS}`),
|
|
150
|
+
(0, _rolesdecorator.UserHaveRole)(_user.USER_ROLE.USER),
|
|
151
|
+
(0, _common.UseGuards)(_authtwofaguard.AuthTwoFaGuardWithoutPassword),
|
|
152
|
+
_ts_param(0, (0, _userdecorator.GetUser)()),
|
|
153
|
+
_ts_param(1, (0, _common.Body)()),
|
|
154
|
+
_ts_metadata("design:type", Function),
|
|
155
|
+
_ts_metadata("design:paramtypes", [
|
|
156
|
+
typeof _usermodel.UserModel === "undefined" ? Object : _usermodel.UserModel,
|
|
157
|
+
typeof _userpropertiesdto.UserAppPasswordDto === "undefined" ? Object : _userpropertiesdto.UserAppPasswordDto
|
|
158
|
+
]),
|
|
159
|
+
_ts_metadata("design:returntype", typeof Promise === "undefined" ? Object : Promise)
|
|
160
|
+
], UsersController.prototype, "generateAppPassword", null);
|
|
161
|
+
_ts_decorate([
|
|
162
|
+
(0, _common.Delete)(`${_routes.USERS_ROUTE.ME}/${_routes.USERS_ROUTE.APP_PASSWORDS}/:name`),
|
|
163
|
+
(0, _rolesdecorator.UserHaveRole)(_user.USER_ROLE.USER),
|
|
164
|
+
_ts_param(0, (0, _userdecorator.GetUser)()),
|
|
165
|
+
_ts_param(1, (0, _common.Param)('name')),
|
|
166
|
+
_ts_metadata("design:type", Function),
|
|
167
|
+
_ts_metadata("design:paramtypes", [
|
|
168
|
+
typeof _usermodel.UserModel === "undefined" ? Object : _usermodel.UserModel,
|
|
169
|
+
String
|
|
170
|
+
]),
|
|
171
|
+
_ts_metadata("design:returntype", typeof Promise === "undefined" ? Object : Promise)
|
|
172
|
+
], UsersController.prototype, "deleteAppPassword", null);
|
|
127
173
|
_ts_decorate([
|
|
128
174
|
(0, _common.Put)(`${_routes.USERS_ROUTE.ME}/${_routes.USERS_ROUTE.LANGUAGE}`),
|
|
129
175
|
(0, _rolesdecorator.UserHaveRole)(_user.USER_ROLE.GUEST),
|
|
@@ -138,12 +184,13 @@ _ts_decorate([
|
|
|
138
184
|
], UsersController.prototype, "updateLanguage", null);
|
|
139
185
|
_ts_decorate([
|
|
140
186
|
(0, _common.Put)(`${_routes.USERS_ROUTE.ME}/${_routes.USERS_ROUTE.PASSWORD}`),
|
|
187
|
+
(0, _common.UseGuards)(_authtwofaguard.AuthTwoFaGuardWithoutPassword),
|
|
141
188
|
_ts_param(0, (0, _userdecorator.GetUser)()),
|
|
142
189
|
_ts_param(1, (0, _common.Body)()),
|
|
143
190
|
_ts_metadata("design:type", Function),
|
|
144
191
|
_ts_metadata("design:paramtypes", [
|
|
145
192
|
typeof _usermodel.UserModel === "undefined" ? Object : _usermodel.UserModel,
|
|
146
|
-
typeof _userpropertiesdto.
|
|
193
|
+
typeof _userpropertiesdto.UserUpdatePasswordDto === "undefined" ? Object : _userpropertiesdto.UserUpdatePasswordDto
|
|
147
194
|
]),
|
|
148
195
|
_ts_metadata("design:returntype", void 0)
|
|
149
196
|
], UsersController.prototype, "updatePassword", null);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../../backend/src/applications/users/users.controller.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Body, Controller, Delete, Get, Header, Param, ParseIntPipe, Patch, Post, Put, Req, Search, StreamableFile, UseGuards } from '@nestjs/common'\nimport { createReadStream } from 'fs'\nimport { FastifyAuthenticatedRequest } from '../../authentication/interfaces/auth-request.interface'\nimport { USERS_ROUTE } from './constants/routes'\nimport { USER_PERMISSION, USER_ROLE } from './constants/user'\nimport { UserHavePermission } from './decorators/permissions.decorator'\nimport { UserHaveRole } from './decorators/roles.decorator'\nimport { GetUser } from './decorators/user.decorator'\nimport { UserCreateOrUpdateGroupDto } from './dto/create-or-update-group.dto'\nimport { CreateUserDto, UpdateUserDto, UpdateUserFromGroupDto } from './dto/create-or-update-user.dto'\nimport { SearchMembersDto } from './dto/search-members.dto'\nimport { UserLanguageDto, UserNotificationDto, UserPasswordDto } from './dto/user-properties.dto'\nimport { UserPermissionsGuard } from './guards/permissions.guard'\nimport { UserRolesGuard } from './guards/roles.guard'\nimport { GroupBrowse } from './interfaces/group-browse.interface'\nimport { GroupMember } from './interfaces/group-member'\nimport { GuestUser } from './interfaces/guest-user.interface'\nimport { Member } from './interfaces/member.interface'\nimport { UserModel } from './models/user.model'\nimport { UsersManager } from './services/users-manager.service'\n\n@Controller(USERS_ROUTE.BASE)\n@UseGuards(UserRolesGuard)\n@UserHaveRole(USER_ROLE.USER)\nexport class UsersController {\n constructor(private readonly usersManager: UsersManager) {}\n\n @Get(USERS_ROUTE.ME)\n @UserHaveRole(USER_ROLE.LINK)\n me(@GetUser() user: UserModel): Promise<{ user: Omit<UserModel, 'password'> }> {\n return this.usersManager.me(user)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.LANGUAGE}`)\n @UserHaveRole(USER_ROLE.GUEST)\n updateLanguage(@GetUser() user: UserModel, @Body() userLanguageDto: UserLanguageDto) {\n return this.usersManager.updateLanguage(user, userLanguageDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.PASSWORD}`)\n updatePassword(@GetUser() user: UserModel, @Body() userPasswordDto: UserPasswordDto) {\n return this.usersManager.updatePassword(user, userPasswordDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.NOTIFICATION}`)\n @UserHaveRole(USER_ROLE.GUEST)\n updateNotification(@GetUser() user: UserModel, @Body() userNotificationDto: UserNotificationDto) {\n return this.usersManager.updateNotification(user, userNotificationDto)\n }\n\n @Get(`${USERS_ROUTE.AVATAR}/:login`)\n @UserHaveRole(USER_ROLE.LINK)\n @Header('cache-control', 'public,max-age=86400')\n async avatar(@GetUser() user: UserModel, @Param('login') login: 'me' | string): Promise<StreamableFile> {\n const isMe: boolean = login === 'me'\n const [path, mime] = await this.usersManager.getAvatar(isMe ? user.login : login, false, isMe && user.role <= USER_ROLE.USER)\n return new StreamableFile(createReadStream(path), { type: mime })\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.AVATAR}`)\n updateAvatar(@Req() req: FastifyAuthenticatedRequest) {\n return this.usersManager.updateAvatar(req)\n }\n\n @Patch(`${USERS_ROUTE.ME}/${USERS_ROUTE.AVATAR}`)\n genAvatar(@GetUser() user: UserModel) {\n return this.usersManager.getAvatar(user.login, true)\n }\n\n @Search()\n searchMembers(@GetUser() user: UserModel, @Body() searchMembersDto: SearchMembersDto): Promise<Member[]> {\n return this.usersManager.searchMembers(user, searchMembersDto)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/${USERS_ROUTE.BROWSE}/:name?`)\n browseGroups(@GetUser() user: UserModel, @Param('name') name?: string): Promise<GroupBrowse> {\n return this.usersManager.browseGroups(user, name)\n }\n\n @Post(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}`)\n @UserHavePermission(USER_PERMISSION.PERSONAL_GROUPS_ADMIN)\n @UseGuards(UserPermissionsGuard)\n createPersonalGroup(@GetUser() user: UserModel, @Body() userCreateOrUpdateGroupDto: UserCreateOrUpdateGroupDto): Promise<GroupMember> {\n return this.usersManager.createPersonalGroup(user, userCreateOrUpdateGroupDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:id`)\n updatePersonalGroup(\n @GetUser() user: UserModel,\n @Param('id', ParseIntPipe) groupId: number,\n @Body() userCreateOrUpdateGroupDto: UserCreateOrUpdateGroupDto\n ): Promise<GroupMember> {\n return this.usersManager.updatePersonalGroup(user, groupId, userCreateOrUpdateGroupDto)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:id`)\n deletePersonalGroup(@GetUser() user: UserModel, @Param('id', ParseIntPipe) groupId: number): Promise<void> {\n return this.usersManager.deletePersonalGroup(user, groupId)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/${USERS_ROUTE.GROUPS_LEAVE}/:id`)\n leavePersonalGroup(@GetUser() user: UserModel, @Param('id', ParseIntPipe) groupId: number): Promise<void> {\n return this.usersManager.leavePersonalGroup(user, groupId)\n }\n\n @Patch(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:groupId/${USERS_ROUTE.USERS}`)\n addUsersToGroup(@GetUser() user: UserModel, @Param('groupId', ParseIntPipe) groupId: number, @Body() userIds: number[]): Promise<void> {\n return this.usersManager.addUsersToGroup(user, groupId, userIds)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:groupId/${USERS_ROUTE.USERS}/:userId`)\n removeUserFromGroup(\n @GetUser() user: UserModel,\n @Param('groupId', ParseIntPipe) groupId: number,\n @Param('userId', ParseIntPipe) userId: number\n ): Promise<void> {\n return this.usersManager.removeUserFromGroup(user, groupId, userId)\n }\n\n @Patch(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:groupId/${USERS_ROUTE.USERS}/:userId`)\n updateUserFromPersonalGroup(\n @GetUser() user: UserModel,\n @Param('groupId', ParseIntPipe) groupId: number,\n @Param('userId', ParseIntPipe) userId: number,\n @Body() updateUserFromGroupDto: UpdateUserFromGroupDto\n ): Promise<void> {\n return this.usersManager.updateUserFromPersonalGroup(user, groupId, userId, updateUserFromGroupDto)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}`)\n listGuests(@GetUser() user: UserModel): Promise<GuestUser[]> {\n return this.usersManager.listGuests(user)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}/:id`)\n getGuest(@GetUser() user: UserModel, @Param('id', ParseIntPipe) guestId: number): Promise<GuestUser> {\n return this.usersManager.getGuest(user, guestId)\n }\n\n @Post(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}`)\n @UserHavePermission(USER_PERMISSION.GUESTS_ADMIN)\n @UseGuards(UserPermissionsGuard)\n createGuest(@GetUser() user: UserModel, @Body() createGuestDto: CreateUserDto): Promise<GuestUser> {\n return this.usersManager.createGuest(user, createGuestDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}/:id`)\n updateGuest(@GetUser() user: UserModel, @Param('id', ParseIntPipe) guestId: number, @Body() updateGuestDto: UpdateUserDto): Promise<GuestUser> {\n return this.usersManager.updateGuest(user, guestId, updateGuestDto)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}/:id`)\n deleteGuest(@GetUser() user: UserModel, @Param('id', ParseIntPipe) guestId: number): Promise<void> {\n return this.usersManager.deleteGuest(user, guestId)\n }\n}\n"],"names":["UsersController","me","user","usersManager","updateLanguage","userLanguageDto","updatePassword","userPasswordDto","updateNotification","userNotificationDto","avatar","login","isMe","path","mime","getAvatar","role","USER_ROLE","USER","StreamableFile","createReadStream","type","updateAvatar","req","genAvatar","searchMembers","searchMembersDto","browseGroups","name","createPersonalGroup","userCreateOrUpdateGroupDto","updatePersonalGroup","groupId","deletePersonalGroup","leavePersonalGroup","addUsersToGroup","userIds","removeUserFromGroup","userId","updateUserFromPersonalGroup","updateUserFromGroupDto","listGuests","getGuest","guestId","createGuest","createGuestDto","updateGuest","updateGuestDto","deleteGuest","ME","LINK","USERS_ROUTE","LANGUAGE","GUEST","PASSWORD","NOTIFICATION","AVATAR","GROUPS","BROWSE","PERSONAL_GROUPS_ADMIN","GROUPS_LEAVE","USERS","GUESTS","GUESTS_ADMIN","BASE"],"mappings":"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"}
|
|
1
|
+
{"version":3,"sources":["../../../../backend/src/applications/users/users.controller.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { Body, Controller, Delete, Get, Header, Param, ParseIntPipe, Patch, Post, Put, Req, Search, StreamableFile, UseGuards } from '@nestjs/common'\nimport { createReadStream } from 'fs'\nimport { LoginResponseDto } from '../../authentication/dto/login-response.dto'\nimport { AuthTwoFaGuardWithoutPassword } from '../../authentication/guards/auth-two-fa-guard'\nimport { FastifyAuthenticatedRequest } from '../../authentication/interfaces/auth-request.interface'\nimport { USERS_ROUTE } from './constants/routes'\nimport { USER_PERMISSION, USER_ROLE } from './constants/user'\nimport { UserHavePermission } from './decorators/permissions.decorator'\nimport { UserHaveRole } from './decorators/roles.decorator'\nimport { GetUser } from './decorators/user.decorator'\nimport { UserCreateOrUpdateGroupDto } from './dto/create-or-update-group.dto'\nimport { CreateUserDto, UpdateUserDto, UpdateUserFromGroupDto } from './dto/create-or-update-user.dto'\nimport { SearchMembersDto } from './dto/search-members.dto'\nimport { UserAppPasswordDto, UserLanguageDto, UserNotificationDto, UserUpdatePasswordDto } from './dto/user-properties.dto'\nimport { UserPermissionsGuard } from './guards/permissions.guard'\nimport { UserRolesGuard } from './guards/roles.guard'\nimport { GroupBrowse } from './interfaces/group-browse.interface'\nimport { GroupMember } from './interfaces/group-member'\nimport { GuestUser } from './interfaces/guest-user.interface'\nimport { Member } from './interfaces/member.interface'\nimport { UserAppPassword } from './interfaces/user-secrets.interface'\nimport { UserModel } from './models/user.model'\nimport { UsersManager } from './services/users-manager.service'\n\n@Controller(USERS_ROUTE.BASE)\n@UseGuards(UserRolesGuard)\n@UserHaveRole(USER_ROLE.USER)\nexport class UsersController {\n constructor(private readonly usersManager: UsersManager) {}\n\n @Get(USERS_ROUTE.ME)\n @UserHaveRole(USER_ROLE.LINK)\n me(@GetUser() user: UserModel): Promise<Omit<LoginResponseDto, 'token'>> {\n return this.usersManager.me(user)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.APP_PASSWORDS}`)\n @UserHaveRole(USER_ROLE.USER)\n @UseGuards(AuthTwoFaGuardWithoutPassword)\n listAppPasswords(@GetUser() user: UserModel): Promise<Omit<UserAppPassword, 'password'>[]> {\n return this.usersManager.listAppPasswords(user)\n }\n\n @Post(`${USERS_ROUTE.ME}/${USERS_ROUTE.APP_PASSWORDS}`)\n @UserHaveRole(USER_ROLE.USER)\n @UseGuards(AuthTwoFaGuardWithoutPassword)\n generateAppPassword(@GetUser() user: UserModel, @Body() userAppPasswordDto: UserAppPasswordDto): Promise<UserAppPassword> {\n return this.usersManager.generateAppPassword(user, userAppPasswordDto)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.APP_PASSWORDS}/:name`)\n @UserHaveRole(USER_ROLE.USER)\n deleteAppPassword(@GetUser() user: UserModel, @Param('name') name: string): Promise<void> {\n return this.usersManager.deleteAppPassword(user, name)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.LANGUAGE}`)\n @UserHaveRole(USER_ROLE.GUEST)\n updateLanguage(@GetUser() user: UserModel, @Body() userLanguageDto: UserLanguageDto) {\n return this.usersManager.updateLanguage(user, userLanguageDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.PASSWORD}`)\n @UseGuards(AuthTwoFaGuardWithoutPassword)\n updatePassword(@GetUser() user: UserModel, @Body() userPasswordDto: UserUpdatePasswordDto) {\n return this.usersManager.updatePassword(user, userPasswordDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.NOTIFICATION}`)\n @UserHaveRole(USER_ROLE.GUEST)\n updateNotification(@GetUser() user: UserModel, @Body() userNotificationDto: UserNotificationDto) {\n return this.usersManager.updateNotification(user, userNotificationDto)\n }\n\n @Get(`${USERS_ROUTE.AVATAR}/:login`)\n @UserHaveRole(USER_ROLE.LINK)\n @Header('cache-control', 'public,max-age=86400')\n async avatar(@GetUser() user: UserModel, @Param('login') login: 'me' | string): Promise<StreamableFile> {\n const isMe: boolean = login === 'me'\n const [path, mime] = await this.usersManager.getAvatar(isMe ? user.login : login, false, isMe && user.role <= USER_ROLE.USER)\n return new StreamableFile(createReadStream(path), { type: mime })\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.AVATAR}`)\n updateAvatar(@Req() req: FastifyAuthenticatedRequest) {\n return this.usersManager.updateAvatar(req)\n }\n\n @Patch(`${USERS_ROUTE.ME}/${USERS_ROUTE.AVATAR}`)\n genAvatar(@GetUser() user: UserModel) {\n return this.usersManager.getAvatar(user.login, true)\n }\n\n @Search()\n searchMembers(@GetUser() user: UserModel, @Body() searchMembersDto: SearchMembersDto): Promise<Member[]> {\n return this.usersManager.searchMembers(user, searchMembersDto)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/${USERS_ROUTE.BROWSE}/:name?`)\n browseGroups(@GetUser() user: UserModel, @Param('name') name?: string): Promise<GroupBrowse> {\n return this.usersManager.browseGroups(user, name)\n }\n\n @Post(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}`)\n @UserHavePermission(USER_PERMISSION.PERSONAL_GROUPS_ADMIN)\n @UseGuards(UserPermissionsGuard)\n createPersonalGroup(@GetUser() user: UserModel, @Body() userCreateOrUpdateGroupDto: UserCreateOrUpdateGroupDto): Promise<GroupMember> {\n return this.usersManager.createPersonalGroup(user, userCreateOrUpdateGroupDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:id`)\n updatePersonalGroup(\n @GetUser() user: UserModel,\n @Param('id', ParseIntPipe) groupId: number,\n @Body() userCreateOrUpdateGroupDto: UserCreateOrUpdateGroupDto\n ): Promise<GroupMember> {\n return this.usersManager.updatePersonalGroup(user, groupId, userCreateOrUpdateGroupDto)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:id`)\n deletePersonalGroup(@GetUser() user: UserModel, @Param('id', ParseIntPipe) groupId: number): Promise<void> {\n return this.usersManager.deletePersonalGroup(user, groupId)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/${USERS_ROUTE.GROUPS_LEAVE}/:id`)\n leavePersonalGroup(@GetUser() user: UserModel, @Param('id', ParseIntPipe) groupId: number): Promise<void> {\n return this.usersManager.leavePersonalGroup(user, groupId)\n }\n\n @Patch(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:groupId/${USERS_ROUTE.USERS}`)\n addUsersToGroup(@GetUser() user: UserModel, @Param('groupId', ParseIntPipe) groupId: number, @Body() userIds: number[]): Promise<void> {\n return this.usersManager.addUsersToGroup(user, groupId, userIds)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:groupId/${USERS_ROUTE.USERS}/:userId`)\n removeUserFromGroup(\n @GetUser() user: UserModel,\n @Param('groupId', ParseIntPipe) groupId: number,\n @Param('userId', ParseIntPipe) userId: number\n ): Promise<void> {\n return this.usersManager.removeUserFromGroup(user, groupId, userId)\n }\n\n @Patch(`${USERS_ROUTE.ME}/${USERS_ROUTE.GROUPS}/:groupId/${USERS_ROUTE.USERS}/:userId`)\n updateUserFromPersonalGroup(\n @GetUser() user: UserModel,\n @Param('groupId', ParseIntPipe) groupId: number,\n @Param('userId', ParseIntPipe) userId: number,\n @Body() updateUserFromGroupDto: UpdateUserFromGroupDto\n ): Promise<void> {\n return this.usersManager.updateUserFromPersonalGroup(user, groupId, userId, updateUserFromGroupDto)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}`)\n listGuests(@GetUser() user: UserModel): Promise<GuestUser[]> {\n return this.usersManager.listGuests(user)\n }\n\n @Get(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}/:id`)\n getGuest(@GetUser() user: UserModel, @Param('id', ParseIntPipe) guestId: number): Promise<GuestUser> {\n return this.usersManager.getGuest(user, guestId)\n }\n\n @Post(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}`)\n @UserHavePermission(USER_PERMISSION.GUESTS_ADMIN)\n @UseGuards(UserPermissionsGuard)\n createGuest(@GetUser() user: UserModel, @Body() createGuestDto: CreateUserDto): Promise<GuestUser> {\n return this.usersManager.createGuest(user, createGuestDto)\n }\n\n @Put(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}/:id`)\n updateGuest(@GetUser() user: UserModel, @Param('id', ParseIntPipe) guestId: number, @Body() updateGuestDto: UpdateUserDto): Promise<GuestUser> {\n return this.usersManager.updateGuest(user, guestId, updateGuestDto)\n }\n\n @Delete(`${USERS_ROUTE.ME}/${USERS_ROUTE.GUESTS}/:id`)\n deleteGuest(@GetUser() user: UserModel, @Param('id', ParseIntPipe) guestId: number): Promise<void> {\n return this.usersManager.deleteGuest(user, guestId)\n }\n}\n"],"names":["UsersController","me","user","usersManager","listAppPasswords","generateAppPassword","userAppPasswordDto","deleteAppPassword","name","updateLanguage","userLanguageDto","updatePassword","userPasswordDto","updateNotification","userNotificationDto","avatar","login","isMe","path","mime","getAvatar","role","USER_ROLE","USER","StreamableFile","createReadStream","type","updateAvatar","req","genAvatar","searchMembers","searchMembersDto","browseGroups","createPersonalGroup","userCreateOrUpdateGroupDto","updatePersonalGroup","groupId","deletePersonalGroup","leavePersonalGroup","addUsersToGroup","userIds","removeUserFromGroup","userId","updateUserFromPersonalGroup","updateUserFromGroupDto","listGuests","getGuest","guestId","createGuest","createGuestDto","updateGuest","updateGuestDto","deleteGuest","ME","LINK","USERS_ROUTE","APP_PASSWORDS","LANGUAGE","GUEST","PASSWORD","NOTIFICATION","AVATAR","GROUPS","BROWSE","PERSONAL_GROUPS_ADMIN","GROUPS_LEAVE","USERS","GUESTS","GUESTS_ADMIN","BASE"],"mappings":"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"}
|
|
@@ -10,9 +10,11 @@ const _config = require("@nestjs/config");
|
|
|
10
10
|
const _jwt = require("@nestjs/jwt");
|
|
11
11
|
const _testing = require("@nestjs/testing");
|
|
12
12
|
const _authmanagerservice = require("../../authentication/services/auth-manager.service");
|
|
13
|
+
const _authmethodtwofaservice = require("../../authentication/services/auth-methods/auth-method-two-fa.service");
|
|
13
14
|
const _configenvironment = require("../../configuration/config.environment");
|
|
14
15
|
const _cacheservice = require("../../infrastructure/cache/services/cache.service");
|
|
15
16
|
const _constants = require("../../infrastructure/database/constants");
|
|
17
|
+
const _notificationsmanagerservice = require("../notifications/services/notifications-manager.service");
|
|
16
18
|
const _usermodel = require("./models/user.model");
|
|
17
19
|
const _adminusersmanagerservice = require("./services/admin-users-manager.service");
|
|
18
20
|
const _adminusersqueriesservice = require("./services/admin-users-queries.service");
|
|
@@ -53,7 +55,12 @@ describe(_userscontroller.UsersController.name, ()=>{
|
|
|
53
55
|
_adminusersmanagerservice.AdminUsersManager,
|
|
54
56
|
_adminusersqueriesservice.AdminUsersQueries,
|
|
55
57
|
_authmanagerservice.AuthManager,
|
|
56
|
-
_jwt.JwtService
|
|
58
|
+
_jwt.JwtService,
|
|
59
|
+
_authmethodtwofaservice.AuthMethod2FA,
|
|
60
|
+
{
|
|
61
|
+
provide: _notificationsmanagerservice.NotificationsManager,
|
|
62
|
+
useValue: {}
|
|
63
|
+
}
|
|
57
64
|
]
|
|
58
65
|
}).compile();
|
|
59
66
|
userController = module.get(_userscontroller.UsersController);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../../backend/src/applications/users/users.controller.spec.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { ConfigModule } from '@nestjs/config'\nimport { JwtService } from '@nestjs/jwt'\nimport { Test, TestingModule } from '@nestjs/testing'\nimport { AuthManager } from '../../authentication/services/auth-manager.service'\nimport { exportConfiguration } from '../../configuration/config.environment'\nimport { Cache } from '../../infrastructure/cache/services/cache.service'\nimport { DB_TOKEN_PROVIDER } from '../../infrastructure/database/constants'\nimport { UserModel } from './models/user.model'\nimport { AdminUsersManager } from './services/admin-users-manager.service'\nimport { AdminUsersQueries } from './services/admin-users-queries.service'\nimport { UsersManager } from './services/users-manager.service'\nimport { UsersQueries } from './services/users-queries.service'\nimport { UsersController } from './users.controller'\nimport { generateUserTest } from './utils/test'\n\ndescribe(UsersController.name, () => {\n let module: TestingModule\n let userController: UsersController\n let usersQueries: UsersQueries\n let adminUsersManager: AdminUsersManager\n let userTest: UserModel\n\n beforeAll(async () => {\n module = await Test.createTestingModule({\n imports: [await ConfigModule.forRoot({ load: [exportConfiguration], isGlobal: true })],\n controllers: [UsersController],\n providers: [\n { provide: DB_TOKEN_PROVIDER, useValue: {} },\n {\n provide: Cache,\n useValue: {}\n },\n UsersManager,\n UsersQueries,\n AdminUsersManager,\n AdminUsersQueries,\n AuthManager,\n JwtService\n ]\n }).compile()\n userController = module.get<UsersController>(UsersController)\n usersQueries = module.get<UsersQueries>(UsersQueries)\n adminUsersManager = module.get<AdminUsersManager>(AdminUsersManager)\n userTest = new UserModel(generateUserTest())\n })\n\n afterAll(async () => {\n await expect(adminUsersManager.deleteUserSpace(userTest.login)).resolves.not.toThrow()\n await module.close()\n })\n\n it('should be defined', () => {\n expect(userController).toBeDefined()\n expect(usersQueries).toBeDefined()\n expect(userTest).toBeDefined()\n })\n\n it('should get the user profile', async () => {\n usersQueries.from = jest.fn().mockReturnValue(userTest)\n const profile = await userController.me(userTest)\n expect(usersQueries.from).toHaveBeenCalled()\n expect(profile.user).toBeInstanceOf(UserModel)\n expect(profile.user.login).toBe(userTest.login)\n })\n\n it('should not generate the user avatar stream', async () => {\n usersQueries.from = jest.fn().mockReturnValueOnce(null)\n await expect(userController.genAvatar(userTest)).rejects.toThrow('does not exist')\n })\n})\n"],"names":["describe","UsersController","name","module","userController","usersQueries","adminUsersManager","userTest","beforeAll","Test","createTestingModule","imports","ConfigModule","forRoot","load","exportConfiguration","isGlobal","controllers","providers","provide","DB_TOKEN_PROVIDER","useValue","Cache","UsersManager","UsersQueries","AdminUsersManager","AdminUsersQueries","AuthManager","JwtService","compile","get","UserModel","generateUserTest","afterAll","expect","deleteUserSpace","login","resolves","not","toThrow","close","it","toBeDefined","from","jest","fn","mockReturnValue","profile","me","toHaveBeenCalled","user","toBeInstanceOf","toBe","mockReturnValueOnce","genAvatar","rejects"],"mappings":"AAAA;;;;CAIC;;;;wBAE4B;qBACF;yBACS;oCACR;
|
|
1
|
+
{"version":3,"sources":["../../../../backend/src/applications/users/users.controller.spec.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { ConfigModule } from '@nestjs/config'\nimport { JwtService } from '@nestjs/jwt'\nimport { Test, TestingModule } from '@nestjs/testing'\nimport { AuthManager } from '../../authentication/services/auth-manager.service'\nimport { AuthMethod2FA } from '../../authentication/services/auth-methods/auth-method-two-fa.service'\nimport { exportConfiguration } from '../../configuration/config.environment'\nimport { Cache } from '../../infrastructure/cache/services/cache.service'\nimport { DB_TOKEN_PROVIDER } from '../../infrastructure/database/constants'\nimport { NotificationsManager } from '../notifications/services/notifications-manager.service'\nimport { UserModel } from './models/user.model'\nimport { AdminUsersManager } from './services/admin-users-manager.service'\nimport { AdminUsersQueries } from './services/admin-users-queries.service'\nimport { UsersManager } from './services/users-manager.service'\nimport { UsersQueries } from './services/users-queries.service'\nimport { UsersController } from './users.controller'\nimport { generateUserTest } from './utils/test'\n\ndescribe(UsersController.name, () => {\n let module: TestingModule\n let userController: UsersController\n let usersQueries: UsersQueries\n let adminUsersManager: AdminUsersManager\n let userTest: UserModel\n\n beforeAll(async () => {\n module = await Test.createTestingModule({\n imports: [await ConfigModule.forRoot({ load: [exportConfiguration], isGlobal: true })],\n controllers: [UsersController],\n providers: [\n { provide: DB_TOKEN_PROVIDER, useValue: {} },\n {\n provide: Cache,\n useValue: {}\n },\n UsersManager,\n UsersQueries,\n AdminUsersManager,\n AdminUsersQueries,\n AuthManager,\n JwtService,\n AuthMethod2FA,\n { provide: NotificationsManager, useValue: {} }\n ]\n }).compile()\n userController = module.get<UsersController>(UsersController)\n usersQueries = module.get<UsersQueries>(UsersQueries)\n adminUsersManager = module.get<AdminUsersManager>(AdminUsersManager)\n userTest = new UserModel(generateUserTest())\n })\n\n afterAll(async () => {\n await expect(adminUsersManager.deleteUserSpace(userTest.login)).resolves.not.toThrow()\n await module.close()\n })\n\n it('should be defined', () => {\n expect(userController).toBeDefined()\n expect(usersQueries).toBeDefined()\n expect(userTest).toBeDefined()\n })\n\n it('should get the user profile', async () => {\n usersQueries.from = jest.fn().mockReturnValue(userTest)\n const profile = await userController.me(userTest)\n expect(usersQueries.from).toHaveBeenCalled()\n expect(profile.user).toBeInstanceOf(UserModel)\n expect(profile.user.login).toBe(userTest.login)\n })\n\n it('should not generate the user avatar stream', async () => {\n usersQueries.from = jest.fn().mockReturnValueOnce(null)\n await expect(userController.genAvatar(userTest)).rejects.toThrow('does not exist')\n })\n})\n"],"names":["describe","UsersController","name","module","userController","usersQueries","adminUsersManager","userTest","beforeAll","Test","createTestingModule","imports","ConfigModule","forRoot","load","exportConfiguration","isGlobal","controllers","providers","provide","DB_TOKEN_PROVIDER","useValue","Cache","UsersManager","UsersQueries","AdminUsersManager","AdminUsersQueries","AuthManager","JwtService","AuthMethod2FA","NotificationsManager","compile","get","UserModel","generateUserTest","afterAll","expect","deleteUserSpace","login","resolves","not","toThrow","close","it","toBeDefined","from","jest","fn","mockReturnValue","profile","me","toHaveBeenCalled","user","toBeInstanceOf","toBe","mockReturnValueOnce","genAvatar","rejects"],"mappings":"AAAA;;;;CAIC;;;;wBAE4B;qBACF;yBACS;oCACR;wCACE;mCACM;8BACd;2BACY;6CACG;2BACX;0CACQ;0CACA;qCACL;qCACA;iCACG;sBACC;AAEjCA,SAASC,gCAAe,CAACC,IAAI,EAAE;IAC7B,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IAEJC,UAAU;QACRL,SAAS,MAAMM,aAAI,CAACC,mBAAmB,CAAC;YACtCC,SAAS;gBAAC,MAAMC,oBAAY,CAACC,OAAO,CAAC;oBAAEC,MAAM;wBAACC,sCAAmB;qBAAC;oBAAEC,UAAU;gBAAK;aAAG;YACtFC,aAAa;gBAAChB,gCAAe;aAAC;YAC9BiB,WAAW;gBACT;oBAAEC,SAASC,4BAAiB;oBAAEC,UAAU,CAAC;gBAAE;gBAC3C;oBACEF,SAASG,mBAAK;oBACdD,UAAU,CAAC;gBACb;gBACAE,iCAAY;gBACZC,iCAAY;gBACZC,2CAAiB;gBACjBC,2CAAiB;gBACjBC,+BAAW;gBACXC,eAAU;gBACVC,qCAAa;gBACb;oBAAEV,SAASW,iDAAoB;oBAAET,UAAU,CAAC;gBAAE;aAC/C;QACH,GAAGU,OAAO;QACV3B,iBAAiBD,OAAO6B,GAAG,CAAkB/B,gCAAe;QAC5DI,eAAeF,OAAO6B,GAAG,CAAeR,iCAAY;QACpDlB,oBAAoBH,OAAO6B,GAAG,CAAoBP,2CAAiB;QACnElB,WAAW,IAAI0B,oBAAS,CAACC,IAAAA,sBAAgB;IAC3C;IAEAC,SAAS;QACP,MAAMC,OAAO9B,kBAAkB+B,eAAe,CAAC9B,SAAS+B,KAAK,GAAGC,QAAQ,CAACC,GAAG,CAACC,OAAO;QACpF,MAAMtC,OAAOuC,KAAK;IACpB;IAEAC,GAAG,qBAAqB;QACtBP,OAAOhC,gBAAgBwC,WAAW;QAClCR,OAAO/B,cAAcuC,WAAW;QAChCR,OAAO7B,UAAUqC,WAAW;IAC9B;IAEAD,GAAG,+BAA+B;QAChCtC,aAAawC,IAAI,GAAGC,KAAKC,EAAE,GAAGC,eAAe,CAACzC;QAC9C,MAAM0C,UAAU,MAAM7C,eAAe8C,EAAE,CAAC3C;QACxC6B,OAAO/B,aAAawC,IAAI,EAAEM,gBAAgB;QAC1Cf,OAAOa,QAAQG,IAAI,EAAEC,cAAc,CAACpB,oBAAS;QAC7CG,OAAOa,QAAQG,IAAI,CAACd,KAAK,EAAEgB,IAAI,CAAC/C,SAAS+B,KAAK;IAChD;IAEAK,GAAG,8CAA8C;QAC/CtC,aAAawC,IAAI,GAAGC,KAAKC,EAAE,GAAGQ,mBAAmB,CAAC;QAClD,MAAMnB,OAAOhC,eAAeoD,SAAS,CAACjD,WAAWkD,OAAO,CAAChB,OAAO,CAAC;IACnE;AACF"}
|
|
@@ -30,7 +30,8 @@ describe('Users (e2e)', ()=>{
|
|
|
30
30
|
});
|
|
31
31
|
afterAll(async ()=>{
|
|
32
32
|
await expect(adminUsersManager.deleteUserOrGuest(userTest.id, userTest.login, {
|
|
33
|
-
deleteSpace: true
|
|
33
|
+
deleteSpace: true,
|
|
34
|
+
isGuest: false
|
|
34
35
|
})).resolves.not.toThrow();
|
|
35
36
|
await (0, _utils.dbCloseConnection)(app);
|
|
36
37
|
await app.close();
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../../backend/src/applications/users/users.e2e-spec.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { NestFastifyApplication } from '@nestjs/platform-fastify'\nimport { appBootstrap } from '../../app.bootstrap'\nimport { TokenResponseDto } from '../../authentication/dto/token-response.dto'\nimport { AuthManager } from '../../authentication/services/auth-manager.service'\nimport { dbCheckConnection, dbCloseConnection } from '../../infrastructure/database/utils'\nimport { API_USERS_AVATAR, API_USERS_ME } from './constants/routes'\nimport { USER_ROLE } from './constants/user'\nimport { DeleteUserDto } from './dto/delete-user.dto'\nimport { UserModel } from './models/user.model'\nimport { AdminUsersManager } from './services/admin-users-manager.service'\nimport { generateUserTest } from './utils/test'\n\ndescribe('Users (e2e)', () => {\n let app: NestFastifyApplication\n let authManager: AuthManager\n let adminUsersManager: AdminUsersManager\n let userTest: UserModel\n let tokens: TokenResponseDto\n\n beforeAll(async () => {\n app = await appBootstrap()\n await app.init()\n await app.getHttpAdapter().getInstance().ready()\n authManager = app.get<AuthManager>(AuthManager)\n adminUsersManager = app.get<AdminUsersManager>(AdminUsersManager)\n userTest = new UserModel(generateUserTest(false), false)\n })\n\n afterAll(async () => {\n await expect(\n adminUsersManager.deleteUserOrGuest(userTest.id, userTest.login, { deleteSpace: true } satisfies DeleteUserDto)\n ).resolves.not.toThrow()\n await dbCloseConnection(app)\n await app.close()\n })\n\n it('should be defined', () => {\n expect(authManager).toBeDefined()\n expect(adminUsersManager).toBeDefined()\n expect(userTest).toBeDefined()\n })\n\n it('should get the database connection', async () => {\n expect(await dbCheckConnection(app)).toBe(true)\n })\n\n it(`GET ${API_USERS_ME} => 401`, async () => {\n const res = await app.inject({\n method: 'GET',\n url: API_USERS_ME\n })\n expect(res.statusCode).toEqual(401)\n })\n\n it(`GET ${API_USERS_ME} => 200`, async () => {\n userTest = await adminUsersManager.createUserOrGuest(userTest, USER_ROLE.USER)\n expect(userTest.id).toBeDefined()\n tokens = await authManager.getTokens(userTest)\n const res = await app.inject({\n method: 'GET',\n url: API_USERS_ME,\n headers: { authorization: `Bearer ${tokens.access}` }\n })\n expect(res.statusCode).toEqual(200)\n const content = res.json()\n expect(content.user).toBeDefined()\n expect(content.user.id).toBe(userTest.id)\n })\n\n it(`GET ${API_USERS_AVATAR} => 200`, async () => {\n const res1 = await app.inject({\n method: 'GET',\n url: `${API_USERS_AVATAR}/me`,\n headers: { authorization: `Bearer ${tokens.access}` }\n })\n const res2 = await app.inject({\n method: 'GET',\n url: `${API_USERS_AVATAR}/${userTest.login}`,\n headers: { authorization: `Bearer ${tokens.access}` }\n })\n for (const res of [res1, res2]) {\n expect(res.statusCode).toEqual(200)\n expect(res.rawPayload).toBeInstanceOf(Buffer)\n expect(res.rawPayload.byteLength).toBeGreaterThan(1)\n }\n expect((res1.raw.req as any).user).toBe((res2.raw.req as any).user)\n expect(res1.rawPayload.byteLength).toEqual(res2.rawPayload.byteLength)\n })\n})\n"],"names":["describe","app","authManager","adminUsersManager","userTest","tokens","beforeAll","appBootstrap","init","getHttpAdapter","getInstance","ready","get","AuthManager","AdminUsersManager","UserModel","generateUserTest","afterAll","expect","deleteUserOrGuest","id","login","deleteSpace","resolves","not","toThrow","dbCloseConnection","close","it","toBeDefined","dbCheckConnection","toBe","API_USERS_ME","res","inject","method","url","statusCode","toEqual","createUserOrGuest","USER_ROLE","USER","getTokens","headers","authorization","access","content","json","user","API_USERS_AVATAR","res1","res2","rawPayload","toBeInstanceOf","Buffer","byteLength","toBeGreaterThan","raw","req"],"mappings":"AAAA;;;;CAIC;;;;8BAG4B;oCAED;uBACyB;wBACN;sBACrB;2BAEA;0CACQ;sBACD;AAEjCA,SAAS,eAAe;IACtB,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IACJ,IAAIC;IAEJC,UAAU;QACRL,MAAM,MAAMM,IAAAA,0BAAY;QACxB,MAAMN,IAAIO,IAAI;QACd,MAAMP,IAAIQ,cAAc,GAAGC,WAAW,GAAGC,KAAK;QAC9CT,cAAcD,IAAIW,GAAG,CAAcC,+BAAW;QAC9CV,oBAAoBF,IAAIW,GAAG,CAAoBE,2CAAiB;QAChEV,WAAW,IAAIW,oBAAS,CAACC,IAAAA,sBAAgB,EAAC,QAAQ;IACpD;IAEAC,SAAS;QACP,MAAMC,OACJf,kBAAkBgB,iBAAiB,CAACf,SAASgB,EAAE,EAAEhB,SAASiB,KAAK,EAAE;YAAEC,aAAa;
|
|
1
|
+
{"version":3,"sources":["../../../../backend/src/applications/users/users.e2e-spec.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { NestFastifyApplication } from '@nestjs/platform-fastify'\nimport { appBootstrap } from '../../app.bootstrap'\nimport { TokenResponseDto } from '../../authentication/dto/token-response.dto'\nimport { AuthManager } from '../../authentication/services/auth-manager.service'\nimport { dbCheckConnection, dbCloseConnection } from '../../infrastructure/database/utils'\nimport { API_USERS_AVATAR, API_USERS_ME } from './constants/routes'\nimport { USER_ROLE } from './constants/user'\nimport { DeleteUserDto } from './dto/delete-user.dto'\nimport { UserModel } from './models/user.model'\nimport { AdminUsersManager } from './services/admin-users-manager.service'\nimport { generateUserTest } from './utils/test'\n\ndescribe('Users (e2e)', () => {\n let app: NestFastifyApplication\n let authManager: AuthManager\n let adminUsersManager: AdminUsersManager\n let userTest: UserModel\n let tokens: TokenResponseDto\n\n beforeAll(async () => {\n app = await appBootstrap()\n await app.init()\n await app.getHttpAdapter().getInstance().ready()\n authManager = app.get<AuthManager>(AuthManager)\n adminUsersManager = app.get<AdminUsersManager>(AdminUsersManager)\n userTest = new UserModel(generateUserTest(false), false)\n })\n\n afterAll(async () => {\n await expect(\n adminUsersManager.deleteUserOrGuest(userTest.id, userTest.login, { deleteSpace: true, isGuest: false } satisfies DeleteUserDto)\n ).resolves.not.toThrow()\n await dbCloseConnection(app)\n await app.close()\n })\n\n it('should be defined', () => {\n expect(authManager).toBeDefined()\n expect(adminUsersManager).toBeDefined()\n expect(userTest).toBeDefined()\n })\n\n it('should get the database connection', async () => {\n expect(await dbCheckConnection(app)).toBe(true)\n })\n\n it(`GET ${API_USERS_ME} => 401`, async () => {\n const res = await app.inject({\n method: 'GET',\n url: API_USERS_ME\n })\n expect(res.statusCode).toEqual(401)\n })\n\n it(`GET ${API_USERS_ME} => 200`, async () => {\n userTest = await adminUsersManager.createUserOrGuest(userTest, USER_ROLE.USER)\n expect(userTest.id).toBeDefined()\n tokens = await authManager.getTokens(userTest)\n const res = await app.inject({\n method: 'GET',\n url: API_USERS_ME,\n headers: { authorization: `Bearer ${tokens.access}` }\n })\n expect(res.statusCode).toEqual(200)\n const content = res.json()\n expect(content.user).toBeDefined()\n expect(content.user.id).toBe(userTest.id)\n })\n\n it(`GET ${API_USERS_AVATAR} => 200`, async () => {\n const res1 = await app.inject({\n method: 'GET',\n url: `${API_USERS_AVATAR}/me`,\n headers: { authorization: `Bearer ${tokens.access}` }\n })\n const res2 = await app.inject({\n method: 'GET',\n url: `${API_USERS_AVATAR}/${userTest.login}`,\n headers: { authorization: `Bearer ${tokens.access}` }\n })\n for (const res of [res1, res2]) {\n expect(res.statusCode).toEqual(200)\n expect(res.rawPayload).toBeInstanceOf(Buffer)\n expect(res.rawPayload.byteLength).toBeGreaterThan(1)\n }\n expect((res1.raw.req as any).user).toBe((res2.raw.req as any).user)\n expect(res1.rawPayload.byteLength).toEqual(res2.rawPayload.byteLength)\n })\n})\n"],"names":["describe","app","authManager","adminUsersManager","userTest","tokens","beforeAll","appBootstrap","init","getHttpAdapter","getInstance","ready","get","AuthManager","AdminUsersManager","UserModel","generateUserTest","afterAll","expect","deleteUserOrGuest","id","login","deleteSpace","isGuest","resolves","not","toThrow","dbCloseConnection","close","it","toBeDefined","dbCheckConnection","toBe","API_USERS_ME","res","inject","method","url","statusCode","toEqual","createUserOrGuest","USER_ROLE","USER","getTokens","headers","authorization","access","content","json","user","API_USERS_AVATAR","res1","res2","rawPayload","toBeInstanceOf","Buffer","byteLength","toBeGreaterThan","raw","req"],"mappings":"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"}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>
|
|
3
|
+
* This file is part of Sync-in | The open source file sync and share solution
|
|
4
|
+
* See the LICENSE file for licensing details
|
|
5
|
+
*/ "use strict";
|
|
6
|
+
Object.defineProperty(exports, "__esModule", {
|
|
7
|
+
value: true
|
|
8
|
+
});
|
|
9
|
+
function _export(target, all) {
|
|
10
|
+
for(var name in all)Object.defineProperty(target, name, {
|
|
11
|
+
enumerable: true,
|
|
12
|
+
get: Object.getOwnPropertyDescriptor(all, name).get
|
|
13
|
+
});
|
|
14
|
+
}
|
|
15
|
+
_export(exports, {
|
|
16
|
+
get USER_AVATAR_FILE_NAME () {
|
|
17
|
+
return USER_AVATAR_FILE_NAME;
|
|
18
|
+
},
|
|
19
|
+
get USER_AVATAR_MAX_UPLOAD_SIZE () {
|
|
20
|
+
return USER_AVATAR_MAX_UPLOAD_SIZE;
|
|
21
|
+
},
|
|
22
|
+
get USER_DEFAULT_AVATAR_FILE_PATH () {
|
|
23
|
+
return USER_DEFAULT_AVATAR_FILE_PATH;
|
|
24
|
+
},
|
|
25
|
+
get getAvatarBase64 () {
|
|
26
|
+
return getAvatarBase64;
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
const _nodepath = /*#__PURE__*/ _interop_require_default(require("node:path"));
|
|
30
|
+
const _image = require("../../../common/image");
|
|
31
|
+
const _configconstants = require("../../../configuration/config.constants");
|
|
32
|
+
const _files = require("../../files/utils/files");
|
|
33
|
+
const _usermodel = require("../models/user.model");
|
|
34
|
+
function _interop_require_default(obj) {
|
|
35
|
+
return obj && obj.__esModule ? obj : {
|
|
36
|
+
default: obj
|
|
37
|
+
};
|
|
38
|
+
}
|
|
39
|
+
const USER_DEFAULT_AVATAR_FILE_PATH = _nodepath.default.join(_configconstants.STATIC_ASSETS_PATH, 'avatar.svg');
|
|
40
|
+
const USER_AVATAR_FILE_NAME = 'avatar.png';
|
|
41
|
+
const USER_AVATAR_MAX_UPLOAD_SIZE = 1024 * 1024 * 5 // 5MB
|
|
42
|
+
;
|
|
43
|
+
async function getAvatarBase64(userLogin) {
|
|
44
|
+
const userAvatarPath = _nodepath.default.join(_usermodel.UserModel.getHomePath(userLogin), USER_AVATAR_FILE_NAME);
|
|
45
|
+
return (0, _image.convertImageToBase64)(await (0, _files.isPathExists)(userAvatarPath) ? userAvatarPath : USER_DEFAULT_AVATAR_FILE_PATH);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
//# sourceMappingURL=avatar.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../../backend/src/applications/users/utils/avatar.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport path from 'node:path'\nimport { convertImageToBase64 } from '../../../common/image'\nimport { STATIC_ASSETS_PATH } from '../../../configuration/config.constants'\nimport { isPathExists } from '../../files/utils/files'\nimport { UserModel } from '../models/user.model'\n\nexport const USER_DEFAULT_AVATAR_FILE_PATH = path.join(STATIC_ASSETS_PATH, 'avatar.svg')\nexport const USER_AVATAR_FILE_NAME = 'avatar.png'\nexport const USER_AVATAR_MAX_UPLOAD_SIZE = 1024 * 1024 * 5 // 5MB\n\nexport async function getAvatarBase64(userLogin: string): Promise<string> {\n const userAvatarPath = path.join(UserModel.getHomePath(userLogin), USER_AVATAR_FILE_NAME)\n return convertImageToBase64((await isPathExists(userAvatarPath)) ? userAvatarPath : USER_DEFAULT_AVATAR_FILE_PATH)\n}\n"],"names":["USER_AVATAR_FILE_NAME","USER_AVATAR_MAX_UPLOAD_SIZE","USER_DEFAULT_AVATAR_FILE_PATH","getAvatarBase64","path","join","STATIC_ASSETS_PATH","userLogin","userAvatarPath","UserModel","getHomePath","convertImageToBase64","isPathExists"],"mappings":"AAAA;;;;CAIC;;;;;;;;;;;QASYA;eAAAA;;QACAC;eAAAA;;QAFAC;eAAAA;;QAISC;eAAAA;;;iEAVL;uBACoB;iCACF;uBACN;2BACH;;;;;;AAEnB,MAAMD,gCAAgCE,iBAAI,CAACC,IAAI,CAACC,mCAAkB,EAAE;AACpE,MAAMN,wBAAwB;AAC9B,MAAMC,8BAA8B,OAAO,OAAO,EAAE,MAAM;;AAE1D,eAAeE,gBAAgBI,SAAiB;IACrD,MAAMC,iBAAiBJ,iBAAI,CAACC,IAAI,CAACI,oBAAS,CAACC,WAAW,CAACH,YAAYP;IACnE,OAAOW,IAAAA,2BAAoB,EAAC,AAAC,MAAMC,IAAAA,mBAAY,EAACJ,kBAAmBA,iBAAiBN;AACtF"}
|