diff --git a/api/controllers/AdminController.ts b/api/controllers/AdminController.ts index 6ab76108..56107c1b 100644 --- a/api/controllers/AdminController.ts +++ b/api/controllers/AdminController.ts @@ -12,7 +12,7 @@ import { CreateMilestoneResponse, CreateBonusResponse, UploadBannerResponse, - GetAllEmailsResponse, + GetAllNamesEmailsResponse, SubmitAttendanceForUsersResponse, ModifyUserAccessLevelResponse, GetAllUserAccessLevelsResponse, @@ -41,10 +41,10 @@ export class AdminController { } @Get('/email') - async getAllEmails(@AuthenticatedUser() user: UserModel): Promise { + async getAllNamesEmails(@AuthenticatedUser() user: UserModel): Promise { if (!PermissionsService.canSeeAllUserEmails(user)) throw new ForbiddenError(); - const emails = await this.userAccountService.getAllEmails(); - return { error: null, emails }; + const namesEmails = await this.userAccountService.getAllNamesEmails(); + return { error: null, namesEmails }; } @Post('/milestone') diff --git a/api/controllers/UserController.ts b/api/controllers/UserController.ts index e80f4347..f2ffd8ad 100644 --- a/api/controllers/UserController.ts +++ b/api/controllers/UserController.ts @@ -104,26 +104,26 @@ export class UserController { } @Post('/socialMedia') - async insertSocialMediaForUser(@Body() insertSocialMediaRequest: InsertSocialMediaRequest, + async insertSocialMediasForUser(@Body() insertSocialMediaRequests: InsertSocialMediaRequest[], @AuthenticatedUser() user: UserModel): Promise { const userSocialMedia = await this.userSocialMediaService - .insertSocialMediaForUser(user, insertSocialMediaRequest.socialMedia); + .insertSocialMediasForUser(user, insertSocialMediaRequests.map(request => request.socialMedia)); return { error: null, userSocialMedia: userSocialMedia.getPublicSocialMedia() }; } @Patch('/socialMedia/:uuid') - async updateSocialMediaForUser(@Params() params: UuidParam, - @Body() updateSocialMediaRequest: UpdateSocialMediaRequest, + async updateSocialMediasForUser(@Params() params: UuidParam[], + @Body() updateSocialMediaRequests: UpdateSocialMediaRequest[], @AuthenticatedUser() user: UserModel): Promise { const userSocialMedia = await this.userSocialMediaService - .updateSocialMediaByUuid(user, params.uuid, updateSocialMediaRequest.socialMedia); + .updateSocialMediasByUuid(user, params.map(param => param.uuid), updateSocialMediaRequests.map(request => request.socialMedia)); return { error: null, userSocialMedia: userSocialMedia.getPublicSocialMedia() }; } @Delete('/socialMedia/:uuid') - async deleteSocialMediaForUser(@Params() params: UuidParam, + async deleteSocialMediaForUser(@Params() params: UuidParam[], @AuthenticatedUser() user: UserModel): Promise { - await this.userSocialMediaService.deleteSocialMediaByUuid(user, params.uuid); + for (const uuid of params) await this.userSocialMediaService.deleteSocialMediasByUuid(user, params.map(param => param.uuid)); return { error: null }; } } diff --git a/repositories/UserRepository.ts b/repositories/UserRepository.ts index 17f32eba..fc9586d6 100644 --- a/repositories/UserRepository.ts +++ b/repositories/UserRepository.ts @@ -45,12 +45,12 @@ export class UserRepository extends BaseRepository { return this.repository.findOne({ accessCode }); } - public async getAllEmails(): Promise { - const emailsRaw = await this.repository + public async getAllNamesEmails(): Promise { + const namesEmailsRaw = await this.repository .createQueryBuilder() - .select('email') + .select(['email', 'UserModel.firstName', 'UserModel.lastName']) .getRawMany(); - return emailsRaw.map((emailRaw) => emailRaw.email); + return namesEmailsRaw.map((nameEmailRaw) => `${nameEmailRaw.UserModel_firstName} ${nameEmailRaw.UserModel_lastName} (${nameEmailRaw.email})`); } public static async generateHash(pass: string): Promise { diff --git a/repositories/UserSocialMediaRepository.ts b/repositories/UserSocialMediaRepository.ts index 4554ed10..ad12f971 100644 --- a/repositories/UserSocialMediaRepository.ts +++ b/repositories/UserSocialMediaRepository.ts @@ -17,12 +17,12 @@ export class UserSocialMediaRepository extends BaseRepository): Promise { - if (changes) userSocialMedia = UserSocialMediaModel.merge(userSocialMedia, changes); + changes?: Partial[]): Promise { + if (changes) for (const change of changes) userSocialMedia = UserSocialMediaModel.merge(userSocialMedia, change); return this.repository.save(userSocialMedia); } - public async deleteSocialMedia(userSocialMedia: UserSocialMediaModel): Promise { + public async deleteSocialMedia(userSocialMedia: UserSocialMediaModel[]): Promise { return this.repository.remove(userSocialMedia); } diff --git a/services/UserAccountService.ts b/services/UserAccountService.ts index 3a17d175..885a1a48 100644 --- a/services/UserAccountService.ts +++ b/services/UserAccountService.ts @@ -167,10 +167,10 @@ export default class UserAccountService { }); } - public async getAllEmails(): Promise { + public async getAllNamesEmails(): Promise { return this.transactions.readOnly(async (txn) => Repositories .user(txn) - .getAllEmails()); + .getAllNamesEmails()); } /** diff --git a/services/UserSocialMediaService.ts b/services/UserSocialMediaService.ts index 6a13cf20..62970a00 100644 --- a/services/UserSocialMediaService.ts +++ b/services/UserSocialMediaService.ts @@ -22,42 +22,53 @@ export default class UserSocialMediaService { return userSocialMedia; } - public async insertSocialMediaForUser(user: UserModel, socialMedia: SocialMedia) { - const addedSocialMedia = await this.transactions.readWrite(async (txn) => { + public async insertSocialMediasForUser(user: UserModel, socialMedias: SocialMedia[]) { + const addedSocialMedias = await this.transactions.readWrite(async (txn) => { const userSocialMediaRepository = Repositories.userSocialMedia(txn); - const isNewSocialMediaType = await userSocialMediaRepository.isNewSocialMediaTypeForUser(user, socialMedia.type); - if (!isNewSocialMediaType) { - throw new UserError('Social media URL of this type has already been created for this user'); + for (const socialMedia of socialMedias) { + const isNewSocialMediaType = await userSocialMediaRepository.isNewSocialMediaTypeForUser(user, socialMedia.type); + if (!isNewSocialMediaType) { + throw new UserError('Social media URL of this type has already been created for this user'); + } } - return userSocialMediaRepository.upsertSocialMedia(UserSocialMediaModel.create({ ...socialMedia, user })); + return userSocialMediaRepository.upsertSocialMedia(UserSocialMediaModel.create({ ...socialMedias, user })); }); - return addedSocialMedia; + return addedSocialMedias; } - public async updateSocialMediaByUuid(user: UserModel, - uuid: Uuid, - changes: Partial): Promise { + public async updateSocialMediasByUuid(user: UserModel, + uuids: Uuid[], + changes: Partial[]): Promise { const updatedSocialMedia = await this.transactions.readWrite(async (txn) => { const userSocialMediaRepository = Repositories.userSocialMedia(txn); - const socialMedia = await userSocialMediaRepository.findByUuid(uuid); - if (!socialMedia) throw new NotFoundError('Social media URL not found'); - if (user.uuid !== socialMedia.user.uuid) { - throw new ForbiddenError('User cannot update a social media URL of another user'); + const validSocials = []; + for (const uuid of uuids) { + const socialMedia = await userSocialMediaRepository.findByUuid(uuid); + if (!socialMedia) throw new NotFoundError('Social media URL not found'); + if (user.uuid !== socialMedia.user.uuid) { + throw new ForbiddenError('User cannot update a social media URL of another user'); + } + validSocials.push(socialMedia); } - return userSocialMediaRepository.upsertSocialMedia(socialMedia, changes); + return validSocials.map((socialMedia, index) => updatedSocialMedia.upsertSocialMedia(socialMedia, changes[index])); }); return updatedSocialMedia; } - public async deleteSocialMediaByUuid(user: UserModel, uuid: Uuid): Promise { + public async deleteSocialMediasByUuid(user: UserModel, uuids: Uuid[]): Promise { const updatedSocialMedia = await this.transactions.readWrite(async (txn) => { const userSocialMediaRepository = Repositories.userSocialMedia(txn); - const socialMedia = await userSocialMediaRepository.findByUuid(uuid); - if (!socialMedia) throw new NotFoundError('Social media URL not found'); - if (user.uuid !== socialMedia.user.uuid) { - throw new ForbiddenError('User cannot delete a social media URL of another user'); + const validSocials = []; + for (const uuid of uuids) { + const socialMedia = await userSocialMediaRepository.findByUuid(uuid); + if (!socialMedia) throw new NotFoundError('Social media URL not found'); + if (user.uuid !== socialMedia.user.uuid) { + throw new ForbiddenError('User cannot delete a social media URL of another user'); + } + validSocials.push(socialMedia); } - return userSocialMediaRepository.deleteSocialMedia(socialMedia); + for (const social of validSocials) userSocialMediaRepository.deleteSocialMedia(social); + return userSocialMediaRepository.getSocialMediaForUser(user); }); return updatedSocialMedia; } diff --git a/tests/admin.test.ts b/tests/admin.test.ts index 021f8c6d..9e362990 100644 --- a/tests/admin.test.ts +++ b/tests/admin.test.ts @@ -131,19 +131,20 @@ describe('retroactive attendance submission', () => { }); }); -describe('email retrieval', () => { +describe('names and emails retrieval', () => { test('gets all the emails of stored users', async () => { const conn = await DatabaseConnection.get(); const users = UserFactory.create(5); - const emails = users.map((user) => user.email.toLowerCase()); + const namesEmails = users.map((user) => `${user.firstName} ${user.lastName} (${user.email.toLowerCase()})`); const admin = UserFactory.fake({ accessType: UserAccessType.ADMIN }); await new PortalState() .createUsers(...users, admin) .write(); - const response = await ControllerFactory.admin(conn).getAllEmails(admin); - expect(expect.arrayContaining(response.emails)).toEqual([...emails, admin.email]); + const response = await ControllerFactory.admin(conn).getAllNamesEmails(admin); + expect(expect.arrayContaining(response.namesEmails)).toEqual([...namesEmails, + `${admin.firstName} ${admin.lastName} (${admin.email})`]); }); }); diff --git a/types/ApiResponses.ts b/types/ApiResponses.ts index cb93f046..e342c308 100644 --- a/types/ApiResponses.ts +++ b/types/ApiResponses.ts @@ -31,8 +31,8 @@ export interface UploadBannerResponse extends ApiResponse { banner: string; } -export interface GetAllEmailsResponse extends ApiResponse { - emails: string[]; +export interface GetAllNamesEmailsResponse extends ApiResponse { + namesEmails: string[]; } export interface SubmitAttendanceForUsersResponse extends ApiResponse {