2022-07-21 16:23:56 +02:00
|
|
|
import { IGroupStore, IStoreGroup } from '../types/stores/group-store';
|
|
|
|
import { Knex } from 'knex';
|
|
|
|
import NotFoundError from '../error/notfound-error';
|
|
|
|
import Group, {
|
|
|
|
IGroup,
|
|
|
|
IGroupModel,
|
|
|
|
IGroupProject,
|
|
|
|
IGroupRole,
|
|
|
|
IGroupUser,
|
|
|
|
IGroupUserModel,
|
|
|
|
} from '../types/group';
|
|
|
|
import Transaction = Knex.Transaction;
|
2023-01-30 09:02:44 +01:00
|
|
|
import { Db } from './db';
|
2022-07-21 16:23:56 +02:00
|
|
|
|
|
|
|
const T = {
|
|
|
|
GROUPS: 'groups',
|
|
|
|
GROUP_USER: 'group_user',
|
|
|
|
GROUP_ROLE: 'group_role',
|
|
|
|
USERS: 'users',
|
|
|
|
PROJECTS: 'projects',
|
2022-07-25 12:11:16 +02:00
|
|
|
ROLES: 'roles',
|
2022-07-21 16:23:56 +02:00
|
|
|
};
|
|
|
|
|
2022-10-13 12:34:47 +02:00
|
|
|
const GROUP_COLUMNS = [
|
|
|
|
'id',
|
|
|
|
'name',
|
|
|
|
'description',
|
|
|
|
'mappings_sso',
|
|
|
|
'created_at',
|
|
|
|
'created_by',
|
|
|
|
];
|
2022-07-21 16:23:56 +02:00
|
|
|
|
|
|
|
const rowToGroup = (row) => {
|
|
|
|
if (!row) {
|
|
|
|
throw new NotFoundError('No group found');
|
|
|
|
}
|
|
|
|
return new Group({
|
|
|
|
id: row.id,
|
|
|
|
name: row.name,
|
|
|
|
description: row.description,
|
2022-10-13 12:34:47 +02:00
|
|
|
mappingsSSO: row.mappings_sso,
|
2022-07-21 16:23:56 +02:00
|
|
|
createdAt: row.created_at,
|
|
|
|
createdBy: row.created_by,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const rowToGroupUser = (row) => {
|
|
|
|
if (!row) {
|
|
|
|
throw new NotFoundError('No group user found');
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
userId: row.user_id,
|
|
|
|
groupId: row.group_id,
|
|
|
|
joinedAt: row.created_at,
|
2022-10-20 09:47:19 +02:00
|
|
|
createdBy: row.created_by,
|
2022-07-21 16:23:56 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-10-13 12:34:47 +02:00
|
|
|
const groupToRow = (group: IStoreGroup) => ({
|
|
|
|
name: group.name,
|
|
|
|
description: group.description,
|
|
|
|
mappings_sso: JSON.stringify(group.mappingsSSO),
|
2022-07-21 16:23:56 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
export default class GroupStore implements IGroupStore {
|
2023-01-30 09:02:44 +01:00
|
|
|
private db: Db;
|
2022-07-21 16:23:56 +02:00
|
|
|
|
2023-01-30 09:02:44 +01:00
|
|
|
constructor(db: Db) {
|
2022-07-21 16:23:56 +02:00
|
|
|
this.db = db;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getAllWithId(ids: number[]): Promise<Group[]> {
|
|
|
|
const groups = await this.db
|
|
|
|
.select(GROUP_COLUMNS)
|
|
|
|
.from(T.GROUPS)
|
|
|
|
.whereIn('id', ids);
|
|
|
|
return groups.map(rowToGroup);
|
|
|
|
}
|
|
|
|
|
|
|
|
async update(group: IGroupModel): Promise<IGroup> {
|
|
|
|
const rows = await this.db(T.GROUPS)
|
|
|
|
.where({ id: group.id })
|
2022-10-13 12:34:47 +02:00
|
|
|
.update(groupToRow(group))
|
2022-07-21 16:23:56 +02:00
|
|
|
.returning(GROUP_COLUMNS);
|
|
|
|
|
|
|
|
return rowToGroup(rows[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getProjectGroupRoles(projectId: string): Promise<IGroupRole[]> {
|
|
|
|
const rows = await this.db
|
2022-07-25 12:11:16 +02:00
|
|
|
.select('gr.group_id', 'gr.role_id', 'gr.created_at', 'r.name')
|
|
|
|
.from(`${T.GROUP_ROLE} as gr`)
|
|
|
|
.innerJoin(`${T.ROLES} as r`, 'gr.role_id', 'r.id')
|
2022-07-21 16:23:56 +02:00
|
|
|
.where('project', projectId);
|
2022-07-25 12:11:16 +02:00
|
|
|
|
2022-07-21 16:23:56 +02:00
|
|
|
return rows.map((r) => {
|
|
|
|
return {
|
|
|
|
groupId: r.group_id,
|
|
|
|
roleId: r.role_id,
|
|
|
|
createdAt: r.created_at,
|
2022-07-25 12:11:16 +02:00
|
|
|
name: r.name,
|
2022-07-21 16:23:56 +02:00
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async getGroupProjects(groupIds: number[]): Promise<IGroupProject[]> {
|
|
|
|
const rows = await this.db
|
|
|
|
.select('group_id', 'project')
|
|
|
|
.from(T.GROUP_ROLE)
|
|
|
|
.whereIn('group_id', groupIds)
|
|
|
|
.distinct();
|
|
|
|
return rows.map((r) => {
|
|
|
|
return {
|
|
|
|
groupId: r.group_id,
|
|
|
|
project: r.project,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async getAllUsersByGroups(groupIds: number[]): Promise<IGroupUser[]> {
|
|
|
|
const rows = await this.db
|
2022-10-20 09:47:19 +02:00
|
|
|
.select(
|
|
|
|
'gu.group_id',
|
|
|
|
'u.id as user_id',
|
|
|
|
'gu.created_at',
|
|
|
|
'gu.created_by',
|
|
|
|
)
|
2022-07-21 16:23:56 +02:00
|
|
|
.from(`${T.GROUP_USER} AS gu`)
|
|
|
|
.join(`${T.USERS} AS u`, 'u.id', 'gu.user_id')
|
|
|
|
.whereIn('gu.group_id', groupIds);
|
|
|
|
return rows.map(rowToGroupUser);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getAll(): Promise<Group[]> {
|
|
|
|
const groups = await this.db.select(GROUP_COLUMNS).from(T.GROUPS);
|
|
|
|
return groups.map(rowToGroup);
|
|
|
|
}
|
|
|
|
|
|
|
|
async delete(id: number): Promise<void> {
|
|
|
|
return this.db(T.GROUPS).where({ id }).del();
|
|
|
|
}
|
|
|
|
|
|
|
|
async deleteAll(): Promise<void> {
|
|
|
|
await this.db(T.GROUPS).del();
|
|
|
|
}
|
|
|
|
|
|
|
|
destroy(): void {}
|
|
|
|
|
|
|
|
async exists(id: number): Promise<boolean> {
|
|
|
|
const result = await this.db.raw(
|
|
|
|
`SELECT EXISTS(SELECT 1 FROM ${T.GROUPS} WHERE id = ?) AS present`,
|
|
|
|
[id],
|
|
|
|
);
|
|
|
|
const { present } = result.rows[0];
|
|
|
|
return present;
|
|
|
|
}
|
|
|
|
|
|
|
|
async existsWithName(name: string): Promise<boolean> {
|
|
|
|
const result = await this.db.raw(
|
|
|
|
`SELECT EXISTS(SELECT 1 FROM ${T.GROUPS} WHERE name = ?) AS present`,
|
|
|
|
[name],
|
|
|
|
);
|
|
|
|
const { present } = result.rows[0];
|
|
|
|
return present;
|
|
|
|
}
|
|
|
|
|
|
|
|
async get(id: number): Promise<Group> {
|
|
|
|
const row = await this.db(T.GROUPS).where({ id }).first();
|
|
|
|
return rowToGroup(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
async create(group: IStoreGroup): Promise<Group> {
|
|
|
|
const row = await this.db(T.GROUPS)
|
|
|
|
.insert(groupToRow(group))
|
|
|
|
.returning('*');
|
|
|
|
return rowToGroup(row[0]);
|
|
|
|
}
|
|
|
|
|
2022-10-25 13:10:27 +02:00
|
|
|
async count(): Promise<number> {
|
|
|
|
return this.db(T.GROUPS)
|
|
|
|
.count('*')
|
|
|
|
.then((res) => Number(res[0].count));
|
|
|
|
}
|
|
|
|
|
2022-10-14 12:08:14 +02:00
|
|
|
async addUsersToGroup(
|
2022-07-21 16:23:56 +02:00
|
|
|
groupId: number,
|
|
|
|
users: IGroupUserModel[],
|
|
|
|
userName: string,
|
|
|
|
transaction?: Transaction,
|
|
|
|
): Promise<void> {
|
|
|
|
const rows = users.map((user) => {
|
|
|
|
return {
|
|
|
|
group_id: groupId,
|
|
|
|
user_id: user.user.id,
|
|
|
|
created_by: userName,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
return (transaction || this.db).batchInsert(T.GROUP_USER, rows);
|
|
|
|
}
|
|
|
|
|
2022-10-14 12:08:14 +02:00
|
|
|
async deleteUsersFromGroup(
|
2022-07-21 16:23:56 +02:00
|
|
|
deletableUsers: IGroupUser[],
|
|
|
|
transaction?: Transaction,
|
|
|
|
): Promise<void> {
|
|
|
|
return (transaction || this.db)(T.GROUP_USER)
|
|
|
|
.whereIn(
|
|
|
|
['group_id', 'user_id'],
|
|
|
|
deletableUsers.map((user) => [user.groupId, user.userId]),
|
|
|
|
)
|
|
|
|
.delete();
|
|
|
|
}
|
|
|
|
|
|
|
|
async updateGroupUsers(
|
|
|
|
groupId: number,
|
|
|
|
newUsers: IGroupUserModel[],
|
|
|
|
existingUsers: IGroupUserModel[],
|
|
|
|
deletableUsers: IGroupUser[],
|
|
|
|
userName: string,
|
|
|
|
): Promise<void> {
|
|
|
|
await this.db.transaction(async (tx) => {
|
2022-10-14 12:08:14 +02:00
|
|
|
await this.addUsersToGroup(groupId, newUsers, userName, tx);
|
|
|
|
await this.deleteUsersFromGroup(deletableUsers, tx);
|
2022-07-21 16:23:56 +02:00
|
|
|
});
|
|
|
|
}
|
2022-10-14 12:08:14 +02:00
|
|
|
|
|
|
|
async getNewGroupsForExternalUser(
|
|
|
|
userId: number,
|
|
|
|
externalGroups: string[],
|
|
|
|
): Promise<IGroup[]> {
|
|
|
|
const rows = await this.db(`${T.GROUPS} as g`)
|
|
|
|
.leftJoin(`${T.GROUP_USER} as gs`, function () {
|
|
|
|
this.on('g.id', 'gs.group_id').andOnVal(
|
|
|
|
'gs.user_id',
|
|
|
|
'=',
|
|
|
|
userId,
|
|
|
|
);
|
|
|
|
})
|
|
|
|
.where('gs.user_id', null)
|
|
|
|
.whereRaw('mappings_sso \\?| :groups', { groups: externalGroups });
|
|
|
|
return rows.map(rowToGroup);
|
|
|
|
}
|
|
|
|
|
|
|
|
async addUserToGroups(
|
|
|
|
userId: number,
|
|
|
|
groupIds: number[],
|
|
|
|
createdBy?: string,
|
|
|
|
): Promise<void> {
|
|
|
|
const rows = groupIds.map((groupId) => {
|
|
|
|
return {
|
|
|
|
group_id: groupId,
|
|
|
|
user_id: userId,
|
|
|
|
created_by: createdBy,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
return this.db.batchInsert(T.GROUP_USER, rows);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getOldGroupsForExternalUser(
|
|
|
|
userId: number,
|
|
|
|
externalGroups: string[],
|
|
|
|
): Promise<IGroupUser[]> {
|
|
|
|
const rows = await this.db(`${T.GROUP_USER} as gu`)
|
|
|
|
.leftJoin(`${T.GROUPS} as g`, 'g.id', 'gu.group_id')
|
|
|
|
.whereNotIn(
|
|
|
|
'g.id',
|
|
|
|
this.db(T.GROUPS)
|
|
|
|
.select('id')
|
|
|
|
.whereRaw('mappings_sso \\?| :groups', {
|
|
|
|
groups: externalGroups,
|
2022-10-17 10:44:36 +02:00
|
|
|
})
|
|
|
|
.orWhereRaw('jsonb_array_length(mappings_sso) = 0'),
|
2022-10-14 12:08:14 +02:00
|
|
|
)
|
|
|
|
.where('gu.user_id', userId);
|
2022-10-17 10:44:36 +02:00
|
|
|
|
2022-10-14 12:08:14 +02:00
|
|
|
return rows.map(rowToGroupUser);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getGroupsForUser(userId: number): Promise<Group[]> {
|
|
|
|
const rows = await this.db(T.GROUPS)
|
|
|
|
.leftJoin(T.GROUP_USER, 'groups.id', 'group_user.group_id')
|
|
|
|
.where('user_id', userId);
|
|
|
|
return rows.map(rowToGroup);
|
|
|
|
}
|
2022-07-21 16:23:56 +02:00
|
|
|
}
|