import { Injectable } from '@nestjs/common'; import { Insertable, Kysely, NotNull, sql } from 'kysely'; import { jsonObjectFrom } from 'kysely/helpers/postgres'; import { InjectKysely } from 'nestjs-kysely'; import { columns } from 'src/database'; import { DummyValue, GenerateSql } from 'src/decorators'; import { AssetVisibility } from 'src/enum'; import { DB } from 'src/schema'; import { ActivityTable } from 'src/schema/tables/activity.table'; import { asUuid } from 'src/utils/database'; export interface ActivitySearch { albumId?: string; assetId?: string | null; userId?: string; isLiked?: boolean; } @Injectable() export class ActivityRepository { constructor(@InjectKysely() private db: Kysely) {} @GenerateSql({ params: [{ albumId: DummyValue.UUID }] }) search(options: ActivitySearch) { const { userId, assetId, albumId, isLiked } = options; return this.db .selectFrom('activity') .selectAll('activity') .innerJoin('user as user2', (join) => join.onRef('user2.id', '=', 'activity.userId').on('user2.deletedAt', 'is', null), ) .innerJoinLateral( (eb) => eb .selectFrom(sql`(select 1)`.as('dummy')) .select(columns.userWithPrefix) .as('user'), (join) => join.onTrue(), ) .select((eb) => eb.fn.toJson('user').as('user')) .leftJoin('asset', 'asset.id', 'activity.assetId') .$if(!!userId, (qb) => qb.where('activity.userId', '=', userId!)) .$if(assetId === null, (qb) => qb.where('assetId', 'is', null)) .$if(!!assetId, (qb) => qb.where('activity.assetId', '=', assetId!)) .$if(!!albumId, (qb) => qb.where('activity.albumId', '=', albumId!)) .$if(isLiked !== undefined, (qb) => qb.where('activity.isLiked', '=', isLiked!)) .where('asset.deletedAt', 'is', null) .orderBy('activity.createdAt', 'asc') .execute(); } @GenerateSql({ params: [{ albumId: DummyValue.UUID, userId: DummyValue.UUID }] }) async create(activity: Insertable) { return this.db .insertInto('activity') .values(activity) .returningAll() .returning((eb) => jsonObjectFrom(eb.selectFrom('user').whereRef('user.id', '=', 'activity.userId').select(columns.user)).as( 'user', ), ) .$narrowType<{ user: NotNull }>() .executeTakeFirstOrThrow(); } @GenerateSql({ params: [DummyValue.UUID] }) async delete(id: string) { await this.db.deleteFrom('activity').where('id', '=', asUuid(id)).execute(); } @GenerateSql({ params: [{ albumId: DummyValue.UUID, assetId: DummyValue.UUID }] }) async getStatistics({ albumId, assetId, }: { albumId: string; assetId?: string; }): Promise<{ comments: number; likes: number }> { const result = await this.db .selectFrom('activity') .select((eb) => [ eb.fn.countAll().filterWhere('activity.isLiked', '=', false).as('comments'), eb.fn.countAll().filterWhere('activity.isLiked', '=', true).as('likes'), ]) .innerJoin('user', (join) => join.onRef('user.id', '=', 'activity.userId').on('user.deletedAt', 'is', null)) .leftJoin('asset', 'asset.id', 'activity.assetId') .$if(!!assetId, (qb) => qb.where('activity.assetId', '=', assetId!)) .where('activity.albumId', '=', albumId) .where(({ or, and, eb }) => or([ and([eb('asset.deletedAt', 'is', null), eb('asset.visibility', '!=', sql.lit(AssetVisibility.LOCKED))]), eb('asset.id', 'is', null), ]), ) .executeTakeFirstOrThrow(); return result; } }