/* * Copyright (C) 2026 Fluxer Contributors * * This file is part of Fluxer. * * Fluxer is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Fluxer is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with Fluxer. If not, see . */ import type {GuildID, UserID} from '~/BrandedTypes'; import {BatchBuilder, buildPatchFromData, executeVersionedUpdate, fetchMany, fetchOne} from '~/database/Cassandra'; import {EXPRESSION_PACK_COLUMNS, type ExpressionPackRow, type PackInstallationRow} from '~/database/CassandraTypes'; import {ExpressionPack} from '~/models/ExpressionPack'; import {ExpressionPacks, ExpressionPacksByCreator, PackInstallations} from '~/Tables'; export type PackType = ExpressionPack['type']; const FETCH_EXPRESSION_PACK_BY_ID_QUERY = ExpressionPacks.selectCql({ where: ExpressionPacks.where.eq('pack_id'), limit: 1, }); const FETCH_EXPRESSION_PACKS_BY_CREATOR_QUERY = ExpressionPacksByCreator.selectCql({ where: ExpressionPacksByCreator.where.eq('creator_id'), }); const FETCH_PACK_INSTALLATIONS_BY_USER_QUERY = PackInstallations.selectCql({ where: PackInstallations.where.eq('user_id'), }); const FETCH_PACK_INSTALLATION_QUERY = PackInstallations.selectCql({ where: [PackInstallations.where.eq('user_id'), PackInstallations.where.eq('pack_id')], limit: 1, }); export class PackRepository { async getPack(packId: GuildID): Promise { const row = await fetchOne(FETCH_EXPRESSION_PACK_BY_ID_QUERY, {pack_id: packId}); return row ? new ExpressionPack(row) : null; } async listPacksByCreator(creatorId: UserID, packType?: PackType): Promise> { const rows = await fetchMany(FETCH_EXPRESSION_PACKS_BY_CREATOR_QUERY, {creator_id: creatorId}); return rows.filter((row) => (packType ? row.pack_type === packType : true)).map((row) => new ExpressionPack(row)); } async countPacksByCreator(creatorId: UserID, packType: PackType): Promise { const rows = await fetchMany(FETCH_EXPRESSION_PACKS_BY_CREATOR_QUERY, {creator_id: creatorId}); return rows.filter((row) => row.pack_type === packType).length; } async upsertPack(data: ExpressionPackRow): Promise { const packId = data.pack_id; const previousPack = await this.getPack(packId); const result = await executeVersionedUpdate( async () => { const existing = await fetchOne(FETCH_EXPRESSION_PACK_BY_ID_QUERY, { pack_id: packId, }); return existing ?? null; }, (current) => ({ pk: {pack_id: packId}, patch: buildPatchFromData(data, current, EXPRESSION_PACK_COLUMNS, ['pack_id']), }), ExpressionPacks, {onFailure: 'log'}, ); const batch = new BatchBuilder(); if (previousPack && previousPack.creatorId !== data.creator_id) { batch.addPrepared( ExpressionPacksByCreator.deleteByPk({ creator_id: previousPack.creatorId, pack_id: packId, }), ); } const finalPack = await this.getPack(packId); if (finalPack) { const finalRow = finalPack.toRow(); finalRow.version = result.finalVersion ?? finalRow.version; batch.addPrepared(ExpressionPacksByCreator.insert(finalRow)); } await batch.execute(); return finalPack ?? new ExpressionPack({...data, version: result.finalVersion ?? 1}); } async deletePack(packId: GuildID): Promise { const pack = await this.getPack(packId); const batch = new BatchBuilder().addPrepared(ExpressionPacks.deleteByPk({pack_id: packId})); if (pack) { batch.addPrepared( ExpressionPacksByCreator.deleteByPk({ creator_id: pack.creatorId, pack_id: packId, }), ); } await batch.execute(); } async listInstallations(userId: UserID): Promise> { return await fetchMany(FETCH_PACK_INSTALLATIONS_BY_USER_QUERY, {user_id: userId}); } async addInstallation(data: PackInstallationRow): Promise { await fetchOne(PackInstallations.insert(data)); } async removeInstallation(userId: UserID, packId: GuildID): Promise { await PackInstallations.deleteByPk({ user_id: userId, pack_id: packId, }); } async hasInstallation(userId: UserID, packId: GuildID): Promise { const row = await fetchOne(FETCH_PACK_INSTALLATION_QUERY, { user_id: userId, pack_id: packId, }); return row !== null; } }