// Copyright 2023 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { Database } from '@signalapp/better-sqlite3'; import type { LoggerType } from '../../types/Logging'; import { isRecord } from '../../util/isRecord'; import { getJobsInQueueSync, getMessageByIdSync, insertJobSync, } from '../Server'; export default function updateToSchemaVersion78( currentVersion: number, db: Database, logger: LoggerType ): void { if (currentVersion >= 78) { return; } db.transaction(() => { const deleteJobsInQueue = db.prepare( 'DELETE FROM jobs WHERE queueType = $queueType' ); const queues = [ { queueType: 'delivery receipts', jobDataKey: 'deliveryReceipts', jobDataIsArray: true, newReceiptsType: 'deliveryReceipt', }, { queueType: 'read receipts', jobDataKey: 'readReceipts', jobDataIsArray: true, newReceiptsType: 'readReceipt', }, { queueType: 'viewed receipts', jobDataKey: 'viewedReceipt', jobDataIsArray: false, newReceiptsType: 'viewedReceipt', }, ]; for (const queue of queues) { const prevJobs = getJobsInQueueSync(db, queue.queueType); deleteJobsInQueue.run({ queueType: queue.queueType }); prevJobs.forEach(job => { const { data, id } = job; if (!isRecord(data)) { logger.warn( `updateToSchemaVersion78: ${queue.queueType} queue job ${id} was missing valid data` ); return; } const { messageId } = data; if (typeof messageId !== 'string') { logger.warn( `updateToSchemaVersion78: ${queue.queueType} queue job ${id} had a non-string messageId` ); return; } const message = getMessageByIdSync(db, messageId); if (!message) { logger.warn( `updateToSchemaVersion78: Unable to find message for ${queue.queueType} job ${id}` ); return; } const { conversationId } = message; if (typeof conversationId !== 'string') { logger.warn( `updateToSchemaVersion78: ${queue.queueType} queue job ${id} had a non-string conversationId` ); return; } const oldReceipts = queue.jobDataIsArray ? data[queue.jobDataKey] : [data[queue.jobDataKey]]; if (!Array.isArray(oldReceipts)) { logger.warn( `updateToSchemaVersion78: ${queue.queueType} queue job ${id} had a non-array ${queue.jobDataKey}` ); return; } const newReceipts = []; for (const receipt of oldReceipts) { if (!isRecord(receipt)) { logger.warn( `updateToSchemaVersion78: ${queue.queueType} queue job ${id} had a non-record receipt` ); continue; } newReceipts.push({ ...receipt, conversationId, }); } const newJob = { ...job, queueType: 'conversation', data: { type: 'Receipts', conversationId, receiptsType: queue.newReceiptsType, receipts: newReceipts, }, }; insertJobSync(db, newJob); }); } db.pragma('user_version = 78'); })(); logger.info('updateToSchemaVersion78: success!'); }