2022-07-21 16:23:56 +02:00
|
|
|
import {
|
2023-07-24 11:05:55 +02:00
|
|
|
ICreateGroupModel,
|
2022-07-21 16:23:56 +02:00
|
|
|
IGroup,
|
|
|
|
IGroupModel,
|
|
|
|
IGroupModelWithProjectRole,
|
|
|
|
IGroupProject,
|
2022-07-25 12:11:16 +02:00
|
|
|
IGroupRole,
|
2022-07-21 16:23:56 +02:00
|
|
|
IGroupUser,
|
|
|
|
} from '../types/group';
|
|
|
|
import { IUnleashConfig, IUnleashStores } from '../types';
|
|
|
|
import { IGroupStore } from '../types/stores/group-store';
|
|
|
|
import { Logger } from '../logger';
|
|
|
|
import BadDataError from '../error/bad-data-error';
|
2023-09-22 12:04:46 +02:00
|
|
|
import { GROUP_CREATED, GROUP_DELETED, GROUP_UPDATED } from '../types/events';
|
2022-07-21 16:23:56 +02:00
|
|
|
import NameExistsError from '../error/name-exists-error';
|
2023-01-18 17:08:07 +01:00
|
|
|
import { IAccountStore } from '../types/stores/account-store';
|
2022-07-21 16:23:56 +02:00
|
|
|
import { IUser } from '../types/user';
|
2023-09-27 15:23:05 +02:00
|
|
|
import EventService from './event-service';
|
2022-07-21 16:23:56 +02:00
|
|
|
|
|
|
|
export class GroupService {
|
|
|
|
private groupStore: IGroupStore;
|
|
|
|
|
2023-09-27 15:23:05 +02:00
|
|
|
private eventService: EventService;
|
2022-07-21 16:23:56 +02:00
|
|
|
|
2023-01-18 17:08:07 +01:00
|
|
|
private accountStore: IAccountStore;
|
2022-07-21 16:23:56 +02:00
|
|
|
|
|
|
|
private logger: Logger;
|
|
|
|
|
|
|
|
constructor(
|
2023-09-27 15:23:05 +02:00
|
|
|
stores: Pick<IUnleashStores, 'groupStore' | 'accountStore'>,
|
2022-07-21 16:23:56 +02:00
|
|
|
{ getLogger }: Pick<IUnleashConfig, 'getLogger'>,
|
2023-09-27 15:23:05 +02:00
|
|
|
eventService: EventService,
|
2022-07-21 16:23:56 +02:00
|
|
|
) {
|
|
|
|
this.logger = getLogger('service/group-service.js');
|
|
|
|
this.groupStore = stores.groupStore;
|
2023-09-27 15:23:05 +02:00
|
|
|
this.eventService = eventService;
|
2023-01-18 17:08:07 +01:00
|
|
|
this.accountStore = stores.accountStore;
|
2022-07-21 16:23:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async getAll(): Promise<IGroupModel[]> {
|
|
|
|
const groups = await this.groupStore.getAll();
|
|
|
|
const allGroupUsers = await this.groupStore.getAllUsersByGroups(
|
|
|
|
groups.map((g) => g.id),
|
|
|
|
);
|
2023-01-18 17:08:07 +01:00
|
|
|
const users = await this.accountStore.getAllWithId(
|
2022-07-21 16:23:56 +02:00
|
|
|
allGroupUsers.map((u) => u.userId),
|
|
|
|
);
|
|
|
|
const groupProjects = await this.groupStore.getGroupProjects(
|
|
|
|
groups.map((g) => g.id),
|
|
|
|
);
|
|
|
|
|
|
|
|
return groups.map((group) => {
|
|
|
|
const mappedGroup = this.mapGroupWithUsers(
|
|
|
|
group,
|
|
|
|
allGroupUsers,
|
|
|
|
users,
|
|
|
|
);
|
|
|
|
return this.mapGroupWithProjects(groupProjects, mappedGroup);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
mapGroupWithProjects(
|
|
|
|
groupProjects: IGroupProject[],
|
|
|
|
group: IGroupModel,
|
|
|
|
): IGroupModel {
|
|
|
|
return {
|
|
|
|
...group,
|
|
|
|
projects: groupProjects
|
|
|
|
.filter((project) => project.groupId === group.id)
|
|
|
|
.map((project) => project.project),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
async getGroup(id: number): Promise<IGroupModel> {
|
|
|
|
const group = await this.groupStore.get(id);
|
|
|
|
const groupUsers = await this.groupStore.getAllUsersByGroups([id]);
|
2023-01-18 17:08:07 +01:00
|
|
|
const users = await this.accountStore.getAllWithId(
|
2022-07-21 16:23:56 +02:00
|
|
|
groupUsers.map((u) => u.userId),
|
|
|
|
);
|
|
|
|
return this.mapGroupWithUsers(group, groupUsers, users);
|
|
|
|
}
|
|
|
|
|
2023-07-24 11:05:55 +02:00
|
|
|
async createGroup(
|
|
|
|
group: ICreateGroupModel,
|
|
|
|
userName: string,
|
|
|
|
): Promise<IGroup> {
|
2022-07-21 16:23:56 +02:00
|
|
|
await this.validateGroup(group);
|
|
|
|
|
|
|
|
const newGroup = await this.groupStore.create(group);
|
|
|
|
|
2022-10-14 12:08:14 +02:00
|
|
|
await this.groupStore.addUsersToGroup(
|
2022-07-21 16:23:56 +02:00
|
|
|
newGroup.id,
|
|
|
|
group.users,
|
|
|
|
userName,
|
|
|
|
);
|
|
|
|
|
2023-09-27 15:23:05 +02:00
|
|
|
await this.eventService.storeEvent({
|
2022-07-21 16:23:56 +02:00
|
|
|
type: GROUP_CREATED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: group,
|
|
|
|
});
|
|
|
|
|
|
|
|
return newGroup;
|
|
|
|
}
|
|
|
|
|
2023-09-05 21:30:20 +02:00
|
|
|
async updateGroup(group: IGroupModel, userName: string): Promise<IGroup> {
|
2022-07-21 16:23:56 +02:00
|
|
|
const preData = await this.groupStore.get(group.id);
|
|
|
|
|
|
|
|
await this.validateGroup(group, preData);
|
|
|
|
|
|
|
|
const newGroup = await this.groupStore.update(group);
|
|
|
|
|
|
|
|
const existingUsers = await this.groupStore.getAllUsersByGroups([
|
|
|
|
group.id,
|
|
|
|
]);
|
|
|
|
const existingUserIds = existingUsers.map((g) => g.userId);
|
|
|
|
|
|
|
|
const deletableUsers = existingUsers.filter(
|
|
|
|
(existingUser) =>
|
|
|
|
!group.users.some(
|
|
|
|
(groupUser) => groupUser.user.id == existingUser.userId,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
|
|
|
await this.groupStore.updateGroupUsers(
|
|
|
|
newGroup.id,
|
|
|
|
group.users.filter(
|
|
|
|
(user) => !existingUserIds.includes(user.user.id),
|
|
|
|
),
|
|
|
|
deletableUsers,
|
|
|
|
userName,
|
|
|
|
);
|
|
|
|
|
2023-09-27 15:23:05 +02:00
|
|
|
await this.eventService.storeEvent({
|
2022-07-21 16:23:56 +02:00
|
|
|
type: GROUP_UPDATED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: newGroup,
|
|
|
|
preData,
|
|
|
|
});
|
|
|
|
|
|
|
|
return newGroup;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getProjectGroups(
|
2023-08-25 10:31:37 +02:00
|
|
|
projectId: string,
|
2022-07-21 16:23:56 +02:00
|
|
|
): Promise<IGroupModelWithProjectRole[]> {
|
2023-08-25 10:31:37 +02:00
|
|
|
const projectGroups = await this.groupStore.getProjectGroups(projectId);
|
|
|
|
|
|
|
|
if (projectGroups.length > 0) {
|
2022-07-21 16:23:56 +02:00
|
|
|
const groups = await this.groupStore.getAllWithId(
|
2023-09-05 21:30:20 +02:00
|
|
|
projectGroups.map((g) => g.id),
|
2022-07-21 16:23:56 +02:00
|
|
|
);
|
|
|
|
const groupUsers = await this.groupStore.getAllUsersByGroups(
|
2023-09-05 21:30:20 +02:00
|
|
|
groups.map((g) => g.id),
|
2022-07-21 16:23:56 +02:00
|
|
|
);
|
2023-01-18 17:08:07 +01:00
|
|
|
const users = await this.accountStore.getAllWithId(
|
2022-07-21 16:23:56 +02:00
|
|
|
groupUsers.map((u) => u.userId),
|
|
|
|
);
|
2023-08-25 10:31:37 +02:00
|
|
|
return groups.flatMap((group) => {
|
|
|
|
return projectGroups
|
|
|
|
.filter((gr) => gr.id === group.id)
|
|
|
|
.map((groupRole) => ({
|
|
|
|
...this.mapGroupWithUsers(group, groupUsers, users),
|
|
|
|
...groupRole,
|
|
|
|
}));
|
2022-07-21 16:23:56 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2023-09-22 13:57:09 +02:00
|
|
|
async deleteGroup(id: number, userName: string): Promise<void> {
|
2023-09-22 12:04:46 +02:00
|
|
|
const group = await this.groupStore.get(id);
|
|
|
|
|
|
|
|
await this.groupStore.delete(id);
|
|
|
|
|
2023-09-27 15:23:05 +02:00
|
|
|
await this.eventService.storeEvent({
|
2023-09-22 13:57:09 +02:00
|
|
|
type: GROUP_DELETED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: group,
|
|
|
|
});
|
2022-07-21 16:23:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async validateGroup(
|
2023-09-05 21:30:20 +02:00
|
|
|
group: IGroupModel | ICreateGroupModel,
|
2022-07-21 16:23:56 +02:00
|
|
|
existingGroup?: IGroup,
|
|
|
|
): Promise<void> {
|
2023-05-16 12:11:32 +02:00
|
|
|
if (!group.name) {
|
2022-07-21 16:23:56 +02:00
|
|
|
throw new BadDataError('Group name cannot be empty');
|
|
|
|
}
|
|
|
|
|
2023-05-16 12:11:32 +02:00
|
|
|
if (!existingGroup || existingGroup.name != group.name) {
|
|
|
|
if (await this.groupStore.existsWithName(group.name)) {
|
2022-07-21 16:23:56 +02:00
|
|
|
throw new NameExistsError('Group name already exists');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-25 12:11:16 +02:00
|
|
|
async getRolesForProject(projectId: string): Promise<IGroupRole[]> {
|
|
|
|
return this.groupStore.getProjectGroupRoles(projectId);
|
|
|
|
}
|
|
|
|
|
2022-07-21 16:23:56 +02:00
|
|
|
private mapGroupWithUsers(
|
|
|
|
group: IGroup,
|
|
|
|
allGroupUsers: IGroupUser[],
|
|
|
|
allUsers: IUser[],
|
|
|
|
): IGroupModel {
|
|
|
|
const groupUsers = allGroupUsers.filter(
|
|
|
|
(user) => user.groupId == group.id,
|
|
|
|
);
|
|
|
|
const groupUsersId = groupUsers.map((user) => user.userId);
|
|
|
|
const selectedUsers = allUsers.filter((user) =>
|
|
|
|
groupUsersId.includes(user.id),
|
|
|
|
);
|
|
|
|
const finalUsers = selectedUsers.map((user) => {
|
|
|
|
const roleUser = groupUsers.find((gu) => gu.userId == user.id);
|
|
|
|
return {
|
|
|
|
user: user,
|
|
|
|
joinedAt: roleUser.joinedAt,
|
2022-10-20 09:47:19 +02:00
|
|
|
createdBy: roleUser.createdBy,
|
2022-07-21 16:23:56 +02:00
|
|
|
};
|
|
|
|
});
|
|
|
|
return { ...group, users: finalUsers };
|
|
|
|
}
|
2022-10-14 12:08:14 +02:00
|
|
|
|
|
|
|
async syncExternalGroups(
|
|
|
|
userId: number,
|
|
|
|
externalGroups: string[],
|
2022-10-20 09:47:19 +02:00
|
|
|
createdBy?: string,
|
2022-10-14 12:08:14 +02:00
|
|
|
): Promise<void> {
|
2022-10-17 10:44:36 +02:00
|
|
|
if (Array.isArray(externalGroups)) {
|
|
|
|
let newGroups = await this.groupStore.getNewGroupsForExternalUser(
|
|
|
|
userId,
|
|
|
|
externalGroups,
|
|
|
|
);
|
|
|
|
await this.groupStore.addUserToGroups(
|
|
|
|
userId,
|
|
|
|
newGroups.map((g) => g.id),
|
2022-10-20 09:47:19 +02:00
|
|
|
createdBy,
|
2022-10-17 10:44:36 +02:00
|
|
|
);
|
|
|
|
let oldGroups = await this.groupStore.getOldGroupsForExternalUser(
|
|
|
|
userId,
|
|
|
|
externalGroups,
|
|
|
|
);
|
|
|
|
await this.groupStore.deleteUsersFromGroup(oldGroups);
|
|
|
|
}
|
2022-10-14 12:08:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async getGroupsForUser(userId: number): Promise<IGroup[]> {
|
|
|
|
return this.groupStore.getGroupsForUser(userId);
|
|
|
|
}
|
2022-07-21 16:23:56 +02:00
|
|
|
}
|