mirror of
https://github.com/docmost/docmost.git
synced 2026-05-07 06:23:06 +08:00
1ad53c2581
* feat(ee): public sharing controls * lint
193 lines
5.6 KiB
TypeScript
193 lines
5.6 KiB
TypeScript
import { Injectable } from '@nestjs/common';
|
|
import { InjectKysely } from 'nestjs-kysely';
|
|
import { KyselyDB, KyselyTransaction } from '@docmost/db/types/kysely.types';
|
|
import { dbOrTx } from '@docmost/db/utils';
|
|
import {
|
|
InsertableSpace,
|
|
Space,
|
|
UpdatableSpace,
|
|
} from '@docmost/db/types/entity.types';
|
|
import { ExpressionBuilder, sql } from 'kysely';
|
|
import { PaginationOptions } from '../../pagination/pagination-options';
|
|
import { executeWithCursorPagination } from '@docmost/db/pagination/cursor-pagination';
|
|
import { DB } from '@docmost/db/types/db';
|
|
import { validate as isValidUUID } from 'uuid';
|
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
|
import { EventName } from '../../../common/events/event.contants';
|
|
|
|
@Injectable()
|
|
export class SpaceRepo {
|
|
constructor(
|
|
@InjectKysely() private readonly db: KyselyDB,
|
|
private eventEmitter: EventEmitter2,
|
|
) {}
|
|
|
|
async findById(
|
|
spaceId: string,
|
|
workspaceId: string,
|
|
opts?: { includeMemberCount?: boolean; trx?: KyselyTransaction },
|
|
): Promise<Space> {
|
|
const db = dbOrTx(this.db, opts?.trx);
|
|
|
|
let query = db
|
|
.selectFrom('spaces')
|
|
.selectAll('spaces')
|
|
.$if(opts?.includeMemberCount, (qb) => qb.select(this.withMemberCount))
|
|
.where('workspaceId', '=', workspaceId);
|
|
|
|
if (isValidUUID(spaceId)) {
|
|
query = query.where('id', '=', spaceId);
|
|
} else {
|
|
query = query.where(sql`LOWER(slug)`, '=', sql`LOWER(${spaceId})`);
|
|
}
|
|
return query.executeTakeFirst();
|
|
}
|
|
|
|
async findBySlug(
|
|
slug: string,
|
|
workspaceId: string,
|
|
opts?: { includeMemberCount: boolean },
|
|
): Promise<Space> {
|
|
return await this.db
|
|
.selectFrom('spaces')
|
|
.selectAll('spaces')
|
|
.$if(opts?.includeMemberCount, (qb) => qb.select(this.withMemberCount))
|
|
.where(sql`LOWER(slug)`, '=', sql`LOWER(${slug})`)
|
|
.where('workspaceId', '=', workspaceId)
|
|
.executeTakeFirst();
|
|
}
|
|
|
|
async slugExists(
|
|
slug: string,
|
|
workspaceId: string,
|
|
trx?: KyselyTransaction,
|
|
): Promise<boolean> {
|
|
const db = dbOrTx(this.db, trx);
|
|
let { count } = await db
|
|
.selectFrom('spaces')
|
|
.select((eb) => eb.fn.count('id').as('count'))
|
|
.where(sql`LOWER(slug)`, '=', sql`LOWER(${slug})`)
|
|
.where('workspaceId', '=', workspaceId)
|
|
.executeTakeFirst();
|
|
count = count as number;
|
|
return count != 0;
|
|
}
|
|
|
|
async updateSpace(
|
|
updatableSpace: UpdatableSpace,
|
|
spaceId: string,
|
|
workspaceId: string,
|
|
trx?: KyselyTransaction,
|
|
) {
|
|
const db = dbOrTx(this.db, trx);
|
|
return db
|
|
.updateTable('spaces')
|
|
.set({ ...updatableSpace, updatedAt: new Date() })
|
|
.where('id', '=', spaceId)
|
|
.where('workspaceId', '=', workspaceId)
|
|
.returningAll()
|
|
.executeTakeFirst();
|
|
}
|
|
|
|
async updateSharingSettings(
|
|
spaceId: string,
|
|
workspaceId: string,
|
|
prefKey: string,
|
|
prefValue: string | boolean,
|
|
) {
|
|
return this.db
|
|
.updateTable('spaces')
|
|
.set({
|
|
settings: sql`COALESCE(settings, '{}'::jsonb)
|
|
|| jsonb_build_object('sharing', COALESCE(settings->'sharing', '{}'::jsonb)
|
|
|| jsonb_build_object('${sql.raw(prefKey)}', ${sql.lit(prefValue)}))`,
|
|
updatedAt: new Date(),
|
|
})
|
|
.where('id', '=', spaceId)
|
|
.where('workspaceId', '=', workspaceId)
|
|
.returningAll()
|
|
.executeTakeFirst();
|
|
}
|
|
|
|
async insertSpace(
|
|
insertableSpace: InsertableSpace,
|
|
trx?: KyselyTransaction,
|
|
): Promise<Space> {
|
|
const db = dbOrTx(this.db, trx);
|
|
return db
|
|
.insertInto('spaces')
|
|
.values(insertableSpace)
|
|
.returningAll()
|
|
.executeTakeFirst();
|
|
}
|
|
|
|
async getSpacesInWorkspace(
|
|
workspaceId: string,
|
|
pagination: PaginationOptions,
|
|
) {
|
|
// todo: show spaces user have access based on visibility and memberships
|
|
let query = this.db
|
|
.selectFrom('spaces')
|
|
.selectAll('spaces')
|
|
.select((eb) => [this.withMemberCount(eb)])
|
|
.where('workspaceId', '=', workspaceId);
|
|
|
|
if (pagination.query) {
|
|
query = query.where((eb) =>
|
|
eb(
|
|
sql`f_unaccent(name)`,
|
|
'ilike',
|
|
sql`f_unaccent(${'%' + pagination.query + '%'})`,
|
|
).or(
|
|
sql`f_unaccent(description)`,
|
|
'ilike',
|
|
sql`f_unaccent(${'%' + pagination.query + '%'})`,
|
|
),
|
|
);
|
|
}
|
|
|
|
return executeWithCursorPagination(query, {
|
|
perPage: pagination.limit,
|
|
cursor: pagination.cursor,
|
|
beforeCursor: pagination.beforeCursor,
|
|
fields: [{ expression: 'id', direction: 'asc' }],
|
|
parseCursor: (cursor) => ({ id: cursor.id }),
|
|
});
|
|
}
|
|
|
|
withMemberCount(eb: ExpressionBuilder<DB, 'spaces'>) {
|
|
const subquery = eb
|
|
.selectFrom('spaceMembers')
|
|
.select('spaceMembers.userId')
|
|
.where('spaceMembers.userId', 'is not', null)
|
|
.whereRef('spaceMembers.spaceId', '=', 'spaces.id')
|
|
.union(
|
|
eb
|
|
.selectFrom('spaceMembers')
|
|
.where('spaceMembers.groupId', 'is not', null)
|
|
.leftJoin('groups', 'groups.id', 'spaceMembers.groupId')
|
|
.leftJoin('groupUsers', 'groupUsers.groupId', 'groups.id')
|
|
.select('groupUsers.userId')
|
|
.whereRef('spaceMembers.spaceId', '=', 'spaces.id'),
|
|
)
|
|
.as('userId');
|
|
|
|
return eb
|
|
.selectFrom(subquery)
|
|
.select((eb) => eb.fn.count('userId').as('count'))
|
|
.as('memberCount');
|
|
}
|
|
|
|
async deleteSpace(spaceId: string, workspaceId: string): Promise<void> {
|
|
await this.db
|
|
.deleteFrom('spaces')
|
|
.where('id', '=', spaceId)
|
|
.where('workspaceId', '=', workspaceId)
|
|
.execute();
|
|
|
|
this.eventEmitter.emit(EventName.SPACE_DELETED, {
|
|
spaceId,
|
|
});
|
|
}
|
|
}
|