Skip to content

Commit

Permalink
BC-8572 room members are added as room-admins (#5400)
Browse files Browse the repository at this point in the history
- new room members are room admins by default. This will change as soon as roles can be changed.
- the endpoint for adding users no longer requires passing a role.
  • Loading branch information
Metauriel authored Dec 16, 2024
1 parent 0b4dacc commit b9b08d0
Show file tree
Hide file tree
Showing 6 changed files with 40 additions and 68 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -87,14 +87,12 @@ describe('RoomMembershipService', () => {
};
};

describe('when no user is provided', () => {
it('should throw an exception', async () => {
const { room } = setup();
it('should throw an exception', async () => {
const { room, user } = setup();

roomMembershipRepo.findByRoomId.mockResolvedValue(null);
roomMembershipRepo.findByRoomId.mockResolvedValue(null);

await expect(service.addMembersToRoom(room.id, [])).rejects.toThrow();
});
await expect(service.addMembersToRoom(room.id, [user.id])).rejects.toThrow();
});
});

Expand All @@ -120,20 +118,21 @@ describe('RoomMembershipService', () => {
};
};

it('should add user to existing roomMembership', async () => {
it('should add user as admin to existing roomMembership', async () => {
// TODO: in the future, once room roles can be changed, this should become ROOMVIEWER
const { user, room, group } = setup();

await service.addMembersToRoom(room.id, [{ userId: user.id, roleName: RoleName.ROOMEDITOR }]);
await service.addMembersToRoom(room.id, [user.id]);

expect(groupService.addUsersToGroup).toHaveBeenCalledWith(group.id, [
{ userId: user.id, roleName: RoleName.ROOMEDITOR },
{ userId: user.id, roleName: RoleName.ROOMADMIN },
]);
});

it('should add user to school', async () => {
const { user, room } = setup();

await service.addMembersToRoom(room.id, [{ userId: user.id, roleName: RoleName.ROOMEDITOR }]);
await service.addMembersToRoom(room.id, [user.id]);

expect(userService.addSecondarySchoolToUsers).toHaveBeenCalledWith([user.id], room.schoolId);
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -73,21 +73,17 @@ export class RoomMembershipService {
await this.roomMembershipRepo.delete(roomMembership);
}

public async addMembersToRoom(
roomId: EntityId,
userIdsAndRoles: Array<{
userId: EntityId;
roleName: RoleName.ROOMADMIN | RoleName.ROOMEDITOR | RoleName.ROOMVIEWER;
}>
): Promise<EntityId> {
public async addMembersToRoom(roomId: EntityId, userIds: Array<EntityId>): Promise<EntityId> {
const roomMembership = await this.roomMembershipRepo.findByRoomId(roomId);
if (roomMembership === null) {
throw new Error('Room membership not found');
}

const userIdsAndRoles = userIds.map((userId) => {
return { userId, roleName: RoleName.ROOMADMIN };
});
await this.groupService.addUsersToGroup(roomMembership.userGroupId, userIdsAndRoles);

const userIds = userIdsAndRoles.map((user) => user.userId);
await this.userService.addSecondarySchoolToUsers(userIds, roomMembership.schoolId);

return roomMembership.id;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,32 +1,12 @@
import { ApiProperty } from '@nestjs/swagger';
import { IsMongoId, IsString, ValidateNested } from 'class-validator';
import { Type } from 'class-transformer';
import { RoleName, RoomRoleArray } from '@shared/domain/interface';

class UserIdAndRole {
@ApiProperty({
description: 'The ID of the user',
required: true,
})
@IsMongoId()
userId!: string;

@ApiProperty({
description: 'The role of the user',
required: true,
enum: RoomRoleArray,
})
@IsString()
roleName!: RoleName.ROOMADMIN | RoleName.ROOMEDITOR | RoleName.ROOMVIEWER;
}
import { IsArray, IsMongoId } from 'class-validator';

export class AddRoomMembersBodyParams {
@ApiProperty({
description: 'Array of userIds and their roles inside of the room',
description: 'The IDs of the users',
required: true,
type: [UserIdAndRole],
})
@ValidateNested({ each: true })
@Type(() => UserIdAndRole)
userIdsAndRoles!: UserIdAndRole[];
@IsArray()
@IsMongoId({ each: true })
public userIds!: string[];
}
20 changes: 10 additions & 10 deletions apps/server/src/modules/room/api/room.controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.UNAUTHORIZED, type: UnauthorizedException })
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async getRooms(
public async getRooms(
@CurrentUser() currentUser: ICurrentUser,
@Query() pagination: RoomPaginationParams
): Promise<RoomListResponse> {
Expand All @@ -67,7 +67,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.UNAUTHORIZED, type: UnauthorizedException })
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async createRoom(
public async createRoom(
@CurrentUser() currentUser: ICurrentUser,
@Body() createRoomParams: CreateRoomBodyParams
): Promise<RoomItemResponse> {
Expand All @@ -86,7 +86,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: HttpStatus.NOT_FOUND, type: NotFoundException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async getRoomDetails(
public async getRoomDetails(
@CurrentUser() currentUser: ICurrentUser,
@Param() urlParams: RoomUrlParams
): Promise<RoomDetailsResponse> {
Expand All @@ -105,7 +105,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: HttpStatus.NOT_FOUND, type: NotFoundException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async getRoomBoards(
public async getRoomBoards(
@CurrentUser() currentUser: ICurrentUser,
@Param() urlParams: RoomUrlParams
): Promise<RoomBoardListResponse> {
Expand All @@ -124,7 +124,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: HttpStatus.NOT_FOUND, type: NotFoundException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async updateRoom(
public async updateRoom(
@CurrentUser() currentUser: ICurrentUser,
@Param() urlParams: RoomUrlParams,
@Body() updateRoomParams: UpdateRoomBodyParams
Expand All @@ -145,7 +145,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.NOT_FOUND, type: NotFoundException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
@HttpCode(204)
async deleteRoom(@CurrentUser() currentUser: ICurrentUser, @Param() urlParams: RoomUrlParams): Promise<void> {
public async deleteRoom(@CurrentUser() currentUser: ICurrentUser, @Param() urlParams: RoomUrlParams): Promise<void> {
await this.roomUc.deleteRoom(currentUser.userId, urlParams.roomId);
}

Expand All @@ -156,12 +156,12 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.UNAUTHORIZED, type: UnauthorizedException })
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async addMembers(
public async addMembers(
@CurrentUser() currentUser: ICurrentUser,
@Param() urlParams: RoomUrlParams,
@Body() bodyParams: AddRoomMembersBodyParams
): Promise<void> {
await this.roomUc.addMembersToRoom(currentUser.userId, urlParams.roomId, bodyParams.userIdsAndRoles);
await this.roomUc.addMembersToRoom(currentUser.userId, urlParams.roomId, bodyParams.userIds);
}

@Patch(':roomId/members/remove')
Expand All @@ -171,7 +171,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.UNAUTHORIZED, type: UnauthorizedException })
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async removeMembers(
public async removeMembers(
@CurrentUser() currentUser: ICurrentUser,
@Param() urlParams: RoomUrlParams,
@Body() bodyParams: RemoveRoomMembersBodyParams
Expand All @@ -190,7 +190,7 @@ export class RoomController {
@ApiResponse({ status: HttpStatus.UNAUTHORIZED, type: UnauthorizedException })
@ApiResponse({ status: HttpStatus.FORBIDDEN, type: ForbiddenException })
@ApiResponse({ status: '5XX', type: ErrorResponse })
async getMembers(
public async getMembers(
@CurrentUser() currentUser: ICurrentUser,
@Param() urlParams: RoomUrlParams
): Promise<RoomMemberListResponse> {
Expand Down
13 changes: 3 additions & 10 deletions apps/server/src/modules/room/api/room.uc.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import { Injectable } from '@nestjs/common';
import { ConfigService } from '@nestjs/config';
import { FeatureDisabledLoggableException } from '@shared/common/loggable-exception';
import { Page, UserDO } from '@shared/domain/domainobject';
import { IFindOptions, Permission, RoleName } from '@shared/domain/interface';
import { IFindOptions, Permission } from '@shared/domain/interface';
import { EntityId } from '@shared/domain/types';
import { BoardExternalReferenceType, ColumnBoard, ColumnBoardService } from '@modules/board';
import { Room, RoomService } from '../domain';
Expand Down Expand Up @@ -125,17 +125,10 @@ export class RoomUc {
return memberResponses;
}

public async addMembersToRoom(
currentUserId: EntityId,
roomId: EntityId,
userIdsAndRoles: Array<{
userId: EntityId;
roleName: RoleName.ROOMADMIN | RoleName.ROOMEDITOR | RoleName.ROOMVIEWER;
}>
): Promise<void> {
public async addMembersToRoom(currentUserId: EntityId, roomId: EntityId, userIds: Array<EntityId>): Promise<void> {
this.checkFeatureEnabled();
await this.checkRoomAuthorization(currentUserId, roomId, Action.write, [Permission.ROOM_MEMBERS_ADD]);
await this.roomMembershipService.addMembersToRoom(roomId, userIdsAndRoles);
await this.roomMembershipService.addMembersToRoom(roomId, userIds);
}

private mapToMember(member: UserWithRoomRoles, user: UserDO): RoomMemberResponse {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -119,21 +119,25 @@ describe('Room Controller (API)', () => {
};

it('should return forbidden error', async () => {
const { room } = await setupRoomWithMembers();
const { room, otherTeacherUser } = await setupRoomWithMembers();
const { loggedInClient } = await setupLoggedInUser();

const response = await loggedInClient.patch(`/${room.id}/members/add`);
const response = await loggedInClient.patch(`/${room.id}/members/add`, {
userIds: [otherTeacherUser.id],
});

expect(response.status).toBe(HttpStatus.FORBIDDEN);
});
});

describe('when the feature is disabled', () => {
it('should return a 403 error', async () => {
const { loggedInClient, room } = await setupRoomWithMembers();
const { loggedInClient, room, otherTeacherUser } = await setupRoomWithMembers();
config.FEATURE_ROOMS_ENABLED = false;

const response = await loggedInClient.patch(`/${room.id}/members/add`);
const response = await loggedInClient.patch(`/${room.id}/members/add`, {
userIds: [otherTeacherUser.id],
});

expect(response.status).toBe(HttpStatus.FORBIDDEN);
});
Expand All @@ -144,7 +148,7 @@ describe('Room Controller (API)', () => {
const { loggedInClient, room, otherTeacherUser } = await setupRoomWithMembers();

const response = await loggedInClient.patch(`/${room.id}/members/add`, {
userIdsAndRoles: [{ userId: otherTeacherUser.id, roleName: RoleName.ROOMEDITOR }],
userIds: [otherTeacherUser.id],
});

expect(response.status).toBe(HttpStatus.OK);
Expand Down

0 comments on commit b9b08d0

Please sign in to comment.