| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2020 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2020-10-30 15:34:04 -05:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-24 14:53:21 -07:00
										 |  |  | /* eslint-disable camelcase */ | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-29 13:05:51 +01:00
										 |  |  | import { mkdirSync } from 'fs'; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | import { join } from 'path'; | 
					
						
							|  |  |  | import rimraf from 'rimraf'; | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  | import { randomBytes } from 'crypto'; | 
					
						
							| 
									
										
										
										
											2022-12-14 12:48:36 -08:00
										 |  |  | import type { Database, Statement } from '@signalapp/better-sqlite3'; | 
					
						
							|  |  |  | import SQL from '@signalapp/better-sqlite3'; | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  | import pProps from 'p-props'; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { Dictionary } from 'lodash'; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | import { | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   forEach, | 
					
						
							|  |  |  |   fromPairs, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   groupBy, | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |   isBoolean, | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |   isNil, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   isNumber, | 
					
						
							|  |  |  |   isString, | 
					
						
							|  |  |  |   last, | 
					
						
							|  |  |  |   map, | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |   mapValues, | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |   omit, | 
					
						
							| 
									
										
										
										
											2021-05-07 14:50:14 -07:00
										 |  |  |   pick, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | } from 'lodash'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 09:36:06 -07:00
										 |  |  | import * as Errors from '../types/errors'; | 
					
						
							| 
									
										
										
										
											2021-08-12 13:15:55 -05:00
										 |  |  | import { ReadStatus } from '../messages/MessageReadStatus'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { GroupV2MemberType } from '../model-types.d'; | 
					
						
							|  |  |  | import type { ReactionType } from '../types/Reactions'; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  | import { STORAGE_UI_KEYS } from '../types/StorageUIKeys'; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | import { UUID } from '../types/UUID'; | 
					
						
							|  |  |  | import type { UUIDStringType } from '../types/UUID'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { StoredJob } from '../jobs/types'; | 
					
						
							| 
									
										
										
										
											2022-09-15 12:17:15 -07:00
										 |  |  | import { assertDev, assertSync, strictAssert } from '../util/assert'; | 
					
						
							| 
									
										
										
										
											2020-09-24 14:53:21 -07:00
										 |  |  | import { combineNames } from '../util/combineNames'; | 
					
						
							| 
									
										
										
										
											2021-10-21 12:49:53 -07:00
										 |  |  | import { consoleLogger } from '../util/consoleLogger'; | 
					
						
							| 
									
										
										
										
											2021-07-09 12:36:10 -07:00
										 |  |  | import { dropNull } from '../util/dropNull'; | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  | import { isNormalNumber } from '../util/isNormalNumber'; | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  | import { isNotNil } from '../util/isNotNil'; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  | import { missingCaseError } from '../util/missingCaseError'; | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | import { parseIntOrThrow } from '../util/parseIntOrThrow'; | 
					
						
							| 
									
										
										
										
											2021-08-26 09:10:58 -05:00
										 |  |  | import * as durations from '../util/durations'; | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  | import { formatCountForLogging } from '../logging/formatCountForLogging'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ConversationColorType, CustomColorType } from '../types/Colors'; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  | import { RemoveAllConfiguration } from '../types/RemoveAllConfiguration'; | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  | import type { BadgeType, BadgeImageType } from '../badges/types'; | 
					
						
							|  |  |  | import { parseBadgeCategory } from '../badges/BadgeCategory'; | 
					
						
							|  |  |  | import { parseBadgeImageTheme } from '../badges/BadgeImageTheme'; | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  | import type { LoggerType } from '../types/Logging'; | 
					
						
							| 
									
										
										
										
											2021-09-17 14:27:53 -04:00
										 |  |  | import * as log from '../logging/log'; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | import type { EmptyQuery, ArrayQuery, Query, JSONRows } from './util'; | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   jsonToObject, | 
					
						
							|  |  |  |   objectToJSON, | 
					
						
							|  |  |  |   batchMultiVarQuery, | 
					
						
							|  |  |  |   getCountFromTable, | 
					
						
							|  |  |  |   removeById, | 
					
						
							|  |  |  |   removeAllFromTable, | 
					
						
							|  |  |  |   getAllFromTable, | 
					
						
							|  |  |  |   getById, | 
					
						
							|  |  |  |   bulkAdd, | 
					
						
							|  |  |  |   createOrUpdate, | 
					
						
							|  |  |  |   setUserVersion, | 
					
						
							|  |  |  |   getUserVersion, | 
					
						
							|  |  |  |   getSchemaVersion, | 
					
						
							|  |  |  | } from './util'; | 
					
						
							|  |  |  | import { updateSchema } from './migrations'; | 
					
						
							| 
									
										
										
										
											2020-09-24 14:53:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   StoredAllItemsType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   AttachmentDownloadJobType, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   ConversationMetricsType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   ConversationType, | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |   DeleteSentProtoRecipientOptionsType, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |   DeleteSentProtoRecipientResultType, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   EmojiType, | 
					
						
							| 
									
										
										
										
											2023-02-09 13:13:08 -08:00
										 |  |  |   FTSOptimizationStateType, | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   GetAllStoriesResultType, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   GetConversationRangeCenteredOnMessageResultType, | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   GetKnownMessageAttachmentsResultType, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   GetUnreadByConversationAndMarkReadResultType, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   IdentityKeyIdType, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   StoredIdentityKeyType, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   InstalledStickerPackType, | 
					
						
							| 
									
										
										
										
											2021-06-14 17:09:37 -07:00
										 |  |  |   ItemKeyType, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   StoredItemType, | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  |   ConversationMessageStatsType, | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   MessageAttachmentsCursorType, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   MessageMetricsType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   MessageType, | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   MessageTypeUnhydrated, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   PreKeyIdType, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   ReactionResultType, | 
					
						
							|  |  |  |   StoredPreKeyType, | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   ServerSearchResultMessageType, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   SenderKeyIdType, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   SenderKeyType, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   SentMessageDBType, | 
					
						
							|  |  |  |   SentMessagesType, | 
					
						
							|  |  |  |   SentProtoType, | 
					
						
							|  |  |  |   SentProtoWithMessageIdsType, | 
					
						
							|  |  |  |   SentRecipientsDBType, | 
					
						
							|  |  |  |   SentRecipientsType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   ServerInterface, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   SessionIdType, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   SessionType, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   SignedPreKeyIdType, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   StoredSignedPreKeyType, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   StickerPackInfoType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   StickerPackStatusType, | 
					
						
							|  |  |  |   StickerPackType, | 
					
						
							|  |  |  |   StickerType, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   StoryDistributionMemberType, | 
					
						
							|  |  |  |   StoryDistributionType, | 
					
						
							|  |  |  |   StoryDistributionWithMembersType, | 
					
						
							|  |  |  |   StoryReadType, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   UninstalledStickerPackType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   UnprocessedType, | 
					
						
							| 
									
										
										
										
											2021-04-16 16:13:13 -07:00
										 |  |  |   UnprocessedUpdateType, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | } from './Interface'; | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  | import { SeenStatus } from '../MessageSeenStatus'; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | type ConversationRow = Readonly<{ | 
					
						
							|  |  |  |   json: string; | 
					
						
							|  |  |  |   profileLastFetchedAt: null | number; | 
					
						
							|  |  |  | }>; | 
					
						
							|  |  |  | type ConversationRows = Array<ConversationRow>; | 
					
						
							| 
									
										
										
										
											2021-04-07 13:00:22 -07:00
										 |  |  | type StickerRow = Readonly<{ | 
					
						
							|  |  |  |   id: number; | 
					
						
							|  |  |  |   packId: string; | 
					
						
							|  |  |  |   emoji: string | null; | 
					
						
							|  |  |  |   height: number; | 
					
						
							|  |  |  |   isCoverOnly: number; | 
					
						
							|  |  |  |   lastUsed: number; | 
					
						
							|  |  |  |   path: string; | 
					
						
							|  |  |  |   width: number; | 
					
						
							|  |  |  | }>; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | // Because we can't force this module to conform to an interface, we narrow our exports
 | 
					
						
							|  |  |  | //   to this one default export, which does conform to the interface.
 | 
					
						
							|  |  |  | // Note: In Javascript, you need to access the .default property when requiring it
 | 
					
						
							|  |  |  | // https://github.com/microsoft/TypeScript/issues/420
 | 
					
						
							|  |  |  | const dataInterface: ServerInterface = { | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   close, | 
					
						
							|  |  |  |   removeDB, | 
					
						
							| 
									
										
										
										
											2018-11-01 16:40:19 -07:00
										 |  |  |   removeIndexedDBFiles, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   createOrUpdateIdentityKey, | 
					
						
							|  |  |  |   getIdentityKeyById, | 
					
						
							|  |  |  |   bulkAddIdentityKeys, | 
					
						
							|  |  |  |   removeIdentityKeyById, | 
					
						
							|  |  |  |   removeAllIdentityKeys, | 
					
						
							| 
									
										
										
										
											2019-02-04 15:54:37 -08:00
										 |  |  |   getAllIdentityKeys, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   createOrUpdatePreKey, | 
					
						
							|  |  |  |   getPreKeyById, | 
					
						
							|  |  |  |   bulkAddPreKeys, | 
					
						
							|  |  |  |   removePreKeyById, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   removePreKeysByUuid, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   removeAllPreKeys, | 
					
						
							| 
									
										
										
										
											2019-02-04 15:54:37 -08:00
										 |  |  |   getAllPreKeys, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   createOrUpdateSignedPreKey, | 
					
						
							|  |  |  |   getSignedPreKeyById, | 
					
						
							|  |  |  |   bulkAddSignedPreKeys, | 
					
						
							|  |  |  |   removeSignedPreKeyById, | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   removeSignedPreKeysByUuid, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   removeAllSignedPreKeys, | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   getAllSignedPreKeys, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   createOrUpdateItem, | 
					
						
							|  |  |  |   getItemById, | 
					
						
							|  |  |  |   removeItemById, | 
					
						
							|  |  |  |   removeAllItems, | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   getAllItems, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  |   createOrUpdateSenderKey, | 
					
						
							|  |  |  |   getSenderKeyById, | 
					
						
							|  |  |  |   removeAllSenderKeys, | 
					
						
							|  |  |  |   getAllSenderKeys, | 
					
						
							| 
									
										
										
										
											2021-05-25 15:40:04 -07:00
										 |  |  |   removeSenderKeyById, | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   insertSentProto, | 
					
						
							|  |  |  |   deleteSentProtosOlderThan, | 
					
						
							|  |  |  |   deleteSentProtoByMessageId, | 
					
						
							|  |  |  |   insertProtoRecipients, | 
					
						
							|  |  |  |   deleteSentProtoRecipient, | 
					
						
							|  |  |  |   getSentProtoByRecipient, | 
					
						
							|  |  |  |   removeAllSentProtos, | 
					
						
							|  |  |  |   getAllSentProtos, | 
					
						
							|  |  |  |   _getAllSentProtoRecipients, | 
					
						
							|  |  |  |   _getAllSentProtoMessageIds, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   createOrUpdateSession, | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  |   createOrUpdateSessions, | 
					
						
							| 
									
										
										
										
											2022-01-07 18:12:13 -08:00
										 |  |  |   commitDecryptResult, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   bulkAddSessions, | 
					
						
							|  |  |  |   removeSessionById, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   removeSessionsByConversation, | 
					
						
							| 
									
										
										
										
											2022-12-12 14:06:16 -08:00
										 |  |  |   removeSessionsByUUID, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   removeAllSessions, | 
					
						
							| 
									
										
										
										
											2019-02-04 15:54:37 -08:00
										 |  |  |   getAllSessions, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   eraseStorageServiceStateFromConversations, | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   getConversationCount, | 
					
						
							|  |  |  |   saveConversation, | 
					
						
							|  |  |  |   saveConversations, | 
					
						
							|  |  |  |   getConversationById, | 
					
						
							|  |  |  |   updateConversation, | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  |   updateConversations, | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   removeConversation, | 
					
						
							| 
									
										
										
										
											2022-08-09 14:39:00 -07:00
										 |  |  |   _removeAllConversations, | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   updateAllConversationColors, | 
					
						
							| 
									
										
										
										
											2022-07-08 13:46:25 -07:00
										 |  |  |   removeAllProfileKeyCredentials, | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   getAllConversations, | 
					
						
							|  |  |  |   getAllConversationIds, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   getAllGroupsInvolvingUuid, | 
					
						
							| 
									
										
										
										
											2019-01-14 13:47:19 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   searchMessages, | 
					
						
							|  |  |  |   searchMessagesInConversation, | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  |   getMessageCount, | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  |   getStoryCount, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   saveMessage, | 
					
						
							|  |  |  |   saveMessages, | 
					
						
							|  |  |  |   removeMessage, | 
					
						
							| 
									
										
										
										
											2021-01-12 16:42:15 -08:00
										 |  |  |   removeMessages, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |   getUnreadByConversationAndMarkRead, | 
					
						
							|  |  |  |   getUnreadReactionsAndMarkRead, | 
					
						
							|  |  |  |   markReactionAsRead, | 
					
						
							|  |  |  |   addReaction, | 
					
						
							|  |  |  |   removeReactionFromConversation, | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  |   _getAllReactions, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   _removeAllReactions, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   getMessageBySender, | 
					
						
							|  |  |  |   getMessageById, | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |   getMessagesById, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   _getAllMessages, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   _removeAllMessages, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   getAllMessageIds, | 
					
						
							|  |  |  |   getMessagesBySentAt, | 
					
						
							|  |  |  |   getExpiredMessages, | 
					
						
							| 
									
										
										
										
											2021-06-18 14:12:04 -05:00
										 |  |  |   getMessagesUnexpectedlyMissingExpirationStartTimestamp, | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |   getSoonestMessageExpiry, | 
					
						
							| 
									
										
										
										
											2021-05-19 11:17:51 -05:00
										 |  |  |   getNextTapToViewMessageTimestampToAgeOut, | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  |   getTapToViewMessagesNeedingErase, | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   getOlderMessagesByConversation, | 
					
						
							| 
									
										
										
										
											2022-10-22 02:26:16 -04:00
										 |  |  |   getAllStories, | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   getNewerMessagesByConversation, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   getTotalUnreadForConversation, | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   getMessageMetricsForConversation, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   getConversationRangeCenteredOnMessage, | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  |   getConversationMessageStats, | 
					
						
							|  |  |  |   getLastConversationMessage, | 
					
						
							| 
									
										
										
										
											2023-01-09 16:52:01 -08:00
										 |  |  |   getCallHistoryMessageByCallId, | 
					
						
							| 
									
										
										
										
											2020-12-07 14:43:19 -06:00
										 |  |  |   hasGroupCallHistoryMessage, | 
					
						
							| 
									
										
										
										
											2020-07-10 11:28:49 -07:00
										 |  |  |   migrateConversationMessages, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-28 15:51:26 -07:00
										 |  |  |   getUnprocessedCount, | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |   getUnprocessedByIdsAndIncrementAttempts, | 
					
						
							|  |  |  |   getAllUnprocessedIds, | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  |   updateUnprocessedWithData, | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  |   updateUnprocessedsWithData, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   getUnprocessedById, | 
					
						
							|  |  |  |   removeUnprocessed, | 
					
						
							|  |  |  |   removeAllUnprocessed, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 16:07:41 -07:00
										 |  |  |   getAttachmentDownloadJobById, | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  |   getNextAttachmentDownloadJobs, | 
					
						
							|  |  |  |   saveAttachmentDownloadJob, | 
					
						
							|  |  |  |   resetAttachmentDownloadPending, | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   setAttachmentDownloadJobPending, | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  |   removeAttachmentDownloadJob, | 
					
						
							|  |  |  |   removeAllAttachmentDownloadJobs, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   createOrUpdateStickerPack, | 
					
						
							|  |  |  |   updateStickerPackStatus, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   updateStickerPackInfo, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   createOrUpdateSticker, | 
					
						
							|  |  |  |   updateStickerLastUsed, | 
					
						
							|  |  |  |   addStickerPackReference, | 
					
						
							|  |  |  |   deleteStickerPackReference, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   getStickerCount, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   deleteStickerPack, | 
					
						
							|  |  |  |   getAllStickerPacks, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   addUninstalledStickerPack, | 
					
						
							|  |  |  |   removeUninstalledStickerPack, | 
					
						
							|  |  |  |   getInstalledStickerPacks, | 
					
						
							|  |  |  |   getUninstalledStickerPacks, | 
					
						
							|  |  |  |   installStickerPack, | 
					
						
							|  |  |  |   uninstallStickerPack, | 
					
						
							|  |  |  |   getStickerPackInfo, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   getAllStickers, | 
					
						
							|  |  |  |   getRecentStickers, | 
					
						
							| 
									
										
										
										
											2021-01-27 14:39:45 -08:00
										 |  |  |   clearAllErrorStickerPackAttempts, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 16:58:27 -07:00
										 |  |  |   updateEmojiUsage, | 
					
						
							|  |  |  |   getRecentEmojis, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |   getAllBadges, | 
					
						
							|  |  |  |   updateOrCreateBadges, | 
					
						
							|  |  |  |   badgeImageFileDownloaded, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   _getAllStoryDistributions, | 
					
						
							|  |  |  |   _getAllStoryDistributionMembers, | 
					
						
							|  |  |  |   _deleteAllStoryDistributions, | 
					
						
							|  |  |  |   createNewStoryDistribution, | 
					
						
							|  |  |  |   getAllStoryDistributionsWithMembers, | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |   getStoryDistributionWithMembers, | 
					
						
							| 
									
										
										
										
											2021-12-09 18:15:59 -08:00
										 |  |  |   modifyStoryDistribution, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   modifyStoryDistributionMembers, | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |   modifyStoryDistributionWithMembers, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   deleteStoryDistribution, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _getAllStoryReads, | 
					
						
							|  |  |  |   _deleteAllStoryReads, | 
					
						
							|  |  |  |   addNewStoryRead, | 
					
						
							|  |  |  |   getLastStoryReadsForAuthor, | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  |   countStoryReadsByConversation, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   removeAll, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   removeAllConfiguration, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   getMessagesNeedingUpgrade, | 
					
						
							|  |  |  |   getMessagesWithVisualMediaAttachments, | 
					
						
							|  |  |  |   getMessagesWithFileAttachments, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |   getMessageServerGuidsForSpam, | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  |   getJobsInQueue, | 
					
						
							|  |  |  |   insertJob, | 
					
						
							|  |  |  |   deleteJob, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  |   wasGroupCallRingPreviouslyCanceled, | 
					
						
							|  |  |  |   processGroupCallRingCancellation, | 
					
						
							|  |  |  |   cleanExpiredGroupCallRingCancellations, | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 11:45:22 -07:00
										 |  |  |   getMaxMessageCounter, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |   getStatisticsForLogging, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 13:13:08 -08:00
										 |  |  |   optimizeFTS, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   // Server-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   initialize, | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |   initializeRenderer, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   getKnownMessageAttachments, | 
					
						
							|  |  |  |   finishGetKnownMessageAttachments, | 
					
						
							|  |  |  |   getKnownConversationAttachments, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   removeKnownStickers, | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |   removeKnownDraftAttachments, | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |   getAllBadgeImageFileLocalPaths, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | export default dataInterface; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | type DatabaseQueryCache = Map<string, Statement<Array<unknown>>>; | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | const statementCache = new WeakMap<Database, DatabaseQueryCache>(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-14 11:35:37 -08:00
										 |  |  | function prepare<T extends Array<unknown> | Record<string, unknown>>( | 
					
						
							|  |  |  |   db: Database, | 
					
						
							| 
									
										
										
										
											2023-01-25 13:22:13 -08:00
										 |  |  |   query: string, | 
					
						
							|  |  |  |   { pluck = false }: { pluck?: boolean } = {} | 
					
						
							| 
									
										
										
										
											2022-11-14 11:35:37 -08:00
										 |  |  | ): Statement<T> { | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |   let dbCache = statementCache.get(db); | 
					
						
							|  |  |  |   if (!dbCache) { | 
					
						
							|  |  |  |     dbCache = new Map(); | 
					
						
							|  |  |  |     statementCache.set(db, dbCache); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-25 13:22:13 -08:00
										 |  |  |   const cacheKey = `${pluck}:${query}`; | 
					
						
							|  |  |  |   let result = dbCache.get(cacheKey) as Statement<T>; | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |   if (!result) { | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |     result = db.prepare<T>(query); | 
					
						
							| 
									
										
										
										
											2023-01-25 13:22:13 -08:00
										 |  |  |     if (pluck === true) { | 
					
						
							|  |  |  |       result.pluck(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     dbCache.set(cacheKey, result); | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | function rowToConversation(row: ConversationRow): ConversationType { | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |   const parsedJson = JSON.parse(row.json); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let profileLastFetchedAt: undefined | number; | 
					
						
							|  |  |  |   if (isNormalNumber(row.profileLastFetchedAt)) { | 
					
						
							|  |  |  |     profileLastFetchedAt = row.profileLastFetchedAt; | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2022-09-15 12:17:15 -07:00
										 |  |  |     assertDev( | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |       isNil(row.profileLastFetchedAt), | 
					
						
							|  |  |  |       'profileLastFetchedAt contained invalid data; defaulting to undefined' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     profileLastFetchedAt = undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     ...parsedJson, | 
					
						
							|  |  |  |     profileLastFetchedAt, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-07 13:00:22 -07:00
										 |  |  | function rowToSticker(row: StickerRow): StickerType { | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     ...row, | 
					
						
							|  |  |  |     isCoverOnly: Boolean(row.isCoverOnly), | 
					
						
							| 
									
										
										
										
											2021-07-09 12:36:10 -07:00
										 |  |  |     emoji: dropNull(row.emoji), | 
					
						
							| 
									
										
										
										
											2021-04-07 13:00:22 -07:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-15 11:43:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function isRenderer() { | 
					
						
							|  |  |  |   if (typeof process === 'undefined' || !process) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return process.type === 'renderer'; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function keyDatabase(db: Database, key: string): void { | 
					
						
							|  |  |  |   // https://www.zetetic.net/sqlcipher/sqlcipher-api/#key
 | 
					
						
							|  |  |  |   db.pragma(`key = "x'${key}'"`); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function switchToWAL(db: Database): void { | 
					
						
							|  |  |  |   // https://sqlite.org/wal.html
 | 
					
						
							|  |  |  |   db.pragma('journal_mode = WAL'); | 
					
						
							|  |  |  |   db.pragma('synchronous = FULL'); | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function migrateSchemaVersion(db: Database): void { | 
					
						
							|  |  |  |   const userVersion = getUserVersion(db); | 
					
						
							|  |  |  |   if (userVersion > 0) { | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const schemaVersion = getSchemaVersion(db); | 
					
						
							|  |  |  |   const newUserVersion = schemaVersion > 18 ? 16 : schemaVersion; | 
					
						
							|  |  |  |   logger.info( | 
					
						
							|  |  |  |     'migrateSchemaVersion: Migrating from schema_version ' + | 
					
						
							|  |  |  |       `${schemaVersion} to user_version ${newUserVersion}` | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   setUserVersion(db, newUserVersion); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function openAndMigrateDatabase(filePath: string, key: string) { | 
					
						
							|  |  |  |   let db: Database | undefined; | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   // First, we try to open the database without any cipher changes
 | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     db = new SQL(filePath); | 
					
						
							|  |  |  |     keyDatabase(db, key); | 
					
						
							|  |  |  |     switchToWAL(db); | 
					
						
							|  |  |  |     migrateSchemaVersion(db); | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     return db; | 
					
						
							|  |  |  |   } catch (error) { | 
					
						
							|  |  |  |     if (db) { | 
					
						
							|  |  |  |       db.close(); | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     logger.info('migrateDatabase: Migration without cipher change failed'); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   // If that fails, we try to open the database with 3.x compatibility to extract the
 | 
					
						
							|  |  |  |   //   user_version (previously stored in schema_version, blown away by cipher_migrate).
 | 
					
						
							|  |  |  |   db = new SQL(filePath); | 
					
						
							|  |  |  |   keyDatabase(db, key); | 
					
						
							| 
									
										
										
										
											2019-08-20 06:24:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   // https://www.zetetic.net/blog/2018/11/30/sqlcipher-400-release/#compatability-sqlcipher-4-0-0
 | 
					
						
							|  |  |  |   db.pragma('cipher_compatibility = 3'); | 
					
						
							|  |  |  |   migrateSchemaVersion(db); | 
					
						
							|  |  |  |   db.close(); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   // After migrating user_version -> schema_version, we reopen database, because we can't
 | 
					
						
							|  |  |  |   //   migrate to the latest ciphers after we've modified the defaults.
 | 
					
						
							|  |  |  |   db = new SQL(filePath); | 
					
						
							|  |  |  |   keyDatabase(db, key); | 
					
						
							| 
									
										
										
										
											2020-03-31 12:22:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   db.pragma('cipher_migrate'); | 
					
						
							|  |  |  |   switchToWAL(db); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return db; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | const INVALID_KEY = /[^0-9A-Fa-f]/; | 
					
						
							|  |  |  | function openAndSetUpSQLCipher(filePath: string, { key }: { key: string }) { | 
					
						
							|  |  |  |   const match = INVALID_KEY.exec(key); | 
					
						
							|  |  |  |   if (match) { | 
					
						
							|  |  |  |     throw new Error(`setupSQLCipher: key '${key}' is not valid`); | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const db = openAndMigrateDatabase(filePath, key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Because foreign key support is not enabled by default!
 | 
					
						
							|  |  |  |   db.pragma('foreign_keys = ON'); | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return db; | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | let globalInstance: Database | undefined; | 
					
						
							| 
									
										
										
										
											2021-10-21 12:49:53 -07:00
										 |  |  | let logger = consoleLogger; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | let globalInstanceRenderer: Database | undefined; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | let databaseFilePath: string | undefined; | 
					
						
							|  |  |  | let indexedDBPath: string | undefined; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-18 14:12:33 -08:00
										 |  |  | SQL.setLogHandler((code, value) => { | 
					
						
							|  |  |  |   logger.warn(`Database log code=${code}: ${value}`); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function initialize({ | 
					
						
							|  |  |  |   configDir, | 
					
						
							|  |  |  |   key, | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger: suppliedLogger, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | }: { | 
					
						
							|  |  |  |   configDir: string; | 
					
						
							|  |  |  |   key: string; | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger: LoggerType; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | }): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   if (globalInstance) { | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     throw new Error('Cannot initialize more than once!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!isString(configDir)) { | 
					
						
							|  |  |  |     throw new Error('initialize: configDir is required!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!isString(key)) { | 
					
						
							| 
									
										
										
										
											2019-02-19 17:32:44 -08:00
										 |  |  |     throw new Error('initialize: key is required!'); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger = suppliedLogger; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   indexedDBPath = join(configDir, 'IndexedDB'); | 
					
						
							| 
									
										
										
										
											2018-11-01 16:40:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   const dbDir = join(configDir, 'sql'); | 
					
						
							| 
									
										
										
										
											2021-11-29 13:05:51 +01:00
										 |  |  |   mkdirSync(dbDir, { recursive: true }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   databaseFilePath = join(dbDir, 'db.sqlite'); | 
					
						
							| 
									
										
										
										
											2018-11-01 16:40:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   let db: Database | undefined; | 
					
						
							| 
									
										
										
										
											2019-08-19 15:26:45 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-27 10:17:22 -08:00
										 |  |  |   try { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     db = openAndSetUpSQLCipher(databaseFilePath, { key }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // For profiling use:
 | 
					
						
							|  |  |  |     // db.pragma('cipher_profile=\'sqlcipher.log\'');
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     updateSchema(db, logger); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-19 15:26:45 -07:00
										 |  |  |     // At this point we can allow general access to the database
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     globalInstance = db; | 
					
						
							| 
									
										
										
										
											2019-08-19 15:26:45 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // test database
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     getMessageCountSync(); | 
					
						
							| 
									
										
										
										
											2019-02-19 17:32:44 -08:00
										 |  |  |   } catch (error) { | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |     logger.error('Database startup error:', error.stack); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     if (db) { | 
					
						
							|  |  |  |       db.close(); | 
					
						
							| 
									
										
										
										
											2019-02-19 17:32:44 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |     throw error; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-02-19 17:32:44 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  | async function initializeRenderer({ | 
					
						
							|  |  |  |   configDir, | 
					
						
							|  |  |  |   key, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   configDir: string; | 
					
						
							|  |  |  |   key: string; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | }): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |   if (!isRenderer()) { | 
					
						
							|  |  |  |     throw new Error('Cannot call from main process.'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (globalInstanceRenderer) { | 
					
						
							|  |  |  |     throw new Error('Cannot initialize more than once!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!isString(configDir)) { | 
					
						
							|  |  |  |     throw new Error('initialize: configDir is required!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!isString(key)) { | 
					
						
							|  |  |  |     throw new Error('initialize: key is required!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!indexedDBPath) { | 
					
						
							|  |  |  |     indexedDBPath = join(configDir, 'IndexedDB'); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |   const dbDir = join(configDir, 'sql'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!databaseFilePath) { | 
					
						
							|  |  |  |     databaseFilePath = join(dbDir, 'db.sqlite'); | 
					
						
							| 
									
										
										
										
											2019-02-19 17:32:44 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   let promisified: Database | undefined; | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   try { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     promisified = openAndSetUpSQLCipher(databaseFilePath, { key }); | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // At this point we can allow general access to the database
 | 
					
						
							|  |  |  |     globalInstanceRenderer = promisified; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // test database
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     getMessageCountSync(); | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |   } catch (error) { | 
					
						
							| 
									
										
										
										
											2021-09-17 14:27:53 -04:00
										 |  |  |     log.error('Database startup error:', error.stack); | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |     throw error; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function close(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-08-18 09:52:48 -07:00
										 |  |  |   for (const dbRef of [globalInstanceRenderer, globalInstance]) { | 
					
						
							| 
									
										
										
										
											2021-08-17 14:55:34 -07:00
										 |  |  |     // SQLLite documentation suggests that we run `PRAGMA optimize` right
 | 
					
						
							|  |  |  |     // before closing the database connection.
 | 
					
						
							| 
									
										
										
										
											2021-08-18 09:52:48 -07:00
										 |  |  |     dbRef?.pragma('optimize'); | 
					
						
							| 
									
										
										
										
											2021-08-17 14:55:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-18 09:52:48 -07:00
										 |  |  |     dbRef?.close(); | 
					
						
							| 
									
										
										
										
											2021-08-17 14:55:34 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-06 10:29:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-18 09:52:48 -07:00
										 |  |  |   globalInstance = undefined; | 
					
						
							|  |  |  |   globalInstanceRenderer = undefined; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeDB(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   if (globalInstance) { | 
					
						
							| 
									
										
										
										
											2021-09-08 13:39:14 -07:00
										 |  |  |     try { | 
					
						
							|  |  |  |       globalInstance.close(); | 
					
						
							|  |  |  |     } catch (error) { | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |       logger.error('removeDB: Failed to close database:', error.stack); | 
					
						
							| 
									
										
										
										
											2021-09-08 13:39:14 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     globalInstance = undefined; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   if (!databaseFilePath) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'removeDB: Cannot erase database without a databaseFilePath!' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 12:06:47 -07:00
										 |  |  |   logger.warn('removeDB: Removing all database files'); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   rimraf.sync(databaseFilePath); | 
					
						
							| 
									
										
										
										
											2021-03-24 14:35:06 -07:00
										 |  |  |   rimraf.sync(`${databaseFilePath}-shm`); | 
					
						
							|  |  |  |   rimraf.sync(`${databaseFilePath}-wal`); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeIndexedDBFiles(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2018-11-01 16:40:19 -07:00
										 |  |  |   if (!indexedDBPath) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'removeIndexedDBFiles: Need to initialize and set indexedDBPath first!' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   const pattern = join(indexedDBPath, '*.leveldb'); | 
					
						
							| 
									
										
										
										
											2018-11-01 16:40:19 -07:00
										 |  |  |   rimraf.sync(pattern); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   indexedDBPath = undefined; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | function getInstance(): Database { | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  |   if (isRenderer()) { | 
					
						
							|  |  |  |     if (!globalInstanceRenderer) { | 
					
						
							|  |  |  |       throw new Error('getInstance: globalInstanceRenderer not set!'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return globalInstanceRenderer; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   if (!globalInstance) { | 
					
						
							|  |  |  |     throw new Error('getInstance: globalInstance not set!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return globalInstance; | 
					
						
							| 
									
										
										
										
											2018-11-01 16:40:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | const IDENTITY_KEYS_TABLE = 'identityKeys'; | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function createOrUpdateIdentityKey( | 
					
						
							|  |  |  |   data: StoredIdentityKeyType | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return createOrUpdate(getInstance(), IDENTITY_KEYS_TABLE, data); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function getIdentityKeyById( | 
					
						
							|  |  |  |   id: IdentityKeyIdType | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | ): Promise<StoredIdentityKeyType | undefined> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getById(getInstance(), IDENTITY_KEYS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function bulkAddIdentityKeys( | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   array: Array<StoredIdentityKeyType> | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return bulkAdd(getInstance(), IDENTITY_KEYS_TABLE, array); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function removeIdentityKeyById(id: IdentityKeyIdType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return removeById(getInstance(), IDENTITY_KEYS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function removeAllIdentityKeys(): Promise<void> { | 
					
						
							|  |  |  |   return removeAllFromTable(getInstance(), IDENTITY_KEYS_TABLE); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function getAllIdentityKeys(): Promise<Array<StoredIdentityKeyType>> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getAllFromTable(getInstance(), IDENTITY_KEYS_TABLE); | 
					
						
							| 
									
										
										
										
											2019-02-04 15:54:37 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | const PRE_KEYS_TABLE = 'preKeys'; | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function createOrUpdatePreKey(data: StoredPreKeyType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return createOrUpdate(getInstance(), PRE_KEYS_TABLE, data); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function getPreKeyById( | 
					
						
							|  |  |  |   id: PreKeyIdType | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | ): Promise<StoredPreKeyType | undefined> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getById(getInstance(), PRE_KEYS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function bulkAddPreKeys(array: Array<StoredPreKeyType>): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return bulkAdd(getInstance(), PRE_KEYS_TABLE, array); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function removePreKeyById(id: PreKeyIdType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return removeById(getInstance(), PRE_KEYS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function removePreKeysByUuid(uuid: UUIDStringType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<Query>('DELETE FROM preKeys WHERE ourUuid IS $uuid;').run({ | 
					
						
							|  |  |  |     uuid, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function removeAllPreKeys(): Promise<void> { | 
					
						
							|  |  |  |   return removeAllFromTable(getInstance(), PRE_KEYS_TABLE); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function getAllPreKeys(): Promise<Array<StoredPreKeyType>> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getAllFromTable(getInstance(), PRE_KEYS_TABLE); | 
					
						
							| 
									
										
										
										
											2019-02-04 15:54:37 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | const SIGNED_PRE_KEYS_TABLE = 'signedPreKeys'; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function createOrUpdateSignedPreKey( | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   data: StoredSignedPreKeyType | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return createOrUpdate(getInstance(), SIGNED_PRE_KEYS_TABLE, data); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function getSignedPreKeyById( | 
					
						
							|  |  |  |   id: SignedPreKeyIdType | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | ): Promise<StoredSignedPreKeyType | undefined> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getById(getInstance(), SIGNED_PRE_KEYS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function bulkAddSignedPreKeys( | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   array: Array<StoredSignedPreKeyType> | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return bulkAdd(getInstance(), SIGNED_PRE_KEYS_TABLE, array); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function removeSignedPreKeyById(id: SignedPreKeyIdType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return removeById(getInstance(), SIGNED_PRE_KEYS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function removeSignedPreKeysByUuid(uuid: UUIDStringType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<Query>('DELETE FROM signedPreKeys WHERE ourUuid IS $uuid;').run({ | 
					
						
							|  |  |  |     uuid, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function removeAllSignedPreKeys(): Promise<void> { | 
					
						
							|  |  |  |   return removeAllFromTable(getInstance(), SIGNED_PRE_KEYS_TABLE); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function getAllSignedPreKeys(): Promise<Array<StoredSignedPreKeyType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT json | 
					
						
							|  |  |  |       FROM signedPreKeys | 
					
						
							|  |  |  |       ORDER BY id ASC; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | const ITEMS_TABLE = 'items'; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function createOrUpdateItem<K extends ItemKeyType>( | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   data: StoredItemType<K> | 
					
						
							| 
									
										
										
										
											2021-06-14 17:09:37 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return createOrUpdate(getInstance(), ITEMS_TABLE, data); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function getItemById<K extends ItemKeyType>( | 
					
						
							| 
									
										
										
										
											2021-06-14 17:09:37 -07:00
										 |  |  |   id: K | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | ): Promise<StoredItemType<K> | undefined> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getById(getInstance(), ITEMS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | async function getAllItems(): Promise<StoredAllItemsType> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>('SELECT json FROM items ORDER BY id ASC;') | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   type RawItemType = { id: ItemKeyType; value: unknown }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const items = rows.map(row => jsonToObject<RawItemType>(row.json)); | 
					
						
							| 
									
										
										
										
											2021-06-14 17:09:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const result: Record<ItemKeyType, unknown> = Object.create(null); | 
					
						
							| 
									
										
										
										
											2021-06-14 17:09:37 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (const { id, value } of items) { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     result[id] = value; | 
					
						
							| 
									
										
										
										
											2021-06-14 17:09:37 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  |   return result as unknown as StoredAllItemsType; | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function removeItemById(id: ItemKeyType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return removeById(getInstance(), ITEMS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function removeAllItems(): Promise<void> { | 
					
						
							|  |  |  |   return removeAllFromTable(getInstance(), ITEMS_TABLE); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  | async function createOrUpdateSenderKey(key: SenderKeyType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2022-01-07 18:12:13 -08:00
										 |  |  |   createOrUpdateSenderKeySync(key); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function createOrUpdateSenderKeySync(key: SenderKeyType): void { | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO senderKeys ( | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       senderId, | 
					
						
							|  |  |  |       distributionId, | 
					
						
							|  |  |  |       data, | 
					
						
							|  |  |  |       lastUpdatedDate | 
					
						
							|  |  |  |     ) values ( | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $senderId, | 
					
						
							|  |  |  |       $distributionId, | 
					
						
							|  |  |  |       $data, | 
					
						
							|  |  |  |       $lastUpdatedDate | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(key); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function getSenderKeyById( | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   id: SenderKeyIdType | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  | ): Promise<SenderKeyType | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const row = prepare(db, 'SELECT * FROM senderKeys WHERE id = $id').get({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return row; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function removeAllSenderKeys(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   prepare<EmptyQuery>(db, 'DELETE FROM senderKeys').run(); | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  | } | 
					
						
							|  |  |  | async function getAllSenderKeys(): Promise<Array<SenderKeyType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   const rows = prepare<EmptyQuery>(db, 'SELECT * FROM senderKeys').all(); | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return rows; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function removeSenderKeyById(id: SenderKeyIdType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-05-25 15:40:04 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   prepare(db, 'DELETE FROM senderKeys WHERE id = $id').run({ id }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | async function insertSentProto( | 
					
						
							|  |  |  |   proto: SentProtoType, | 
					
						
							|  |  |  |   options: { | 
					
						
							|  |  |  |     recipients: SentRecipientsType; | 
					
						
							|  |  |  |     messageIds: SentMessagesType; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | ): Promise<number> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const { recipients, messageIds } = options; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Note: we use `pluck` in this function to fetch only the first column of returned row.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     // 1. Insert the payload, fetching its primary key id
 | 
					
						
							|  |  |  |     const info = prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       INSERT INTO sendLogPayloads ( | 
					
						
							|  |  |  |         contentHint, | 
					
						
							|  |  |  |         proto, | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |         timestamp, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |         urgent, | 
					
						
							|  |  |  |         hasPniSignatureMessage | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |       ) VALUES ( | 
					
						
							|  |  |  |         $contentHint, | 
					
						
							|  |  |  |         $proto, | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |         $timestamp, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |         $urgent, | 
					
						
							|  |  |  |         $hasPniSignatureMessage | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |       ); | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |     ).run({ | 
					
						
							|  |  |  |       ...proto, | 
					
						
							|  |  |  |       urgent: proto.urgent ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |       hasPniSignatureMessage: proto.hasPniSignatureMessage ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |     const id = parseIntOrThrow( | 
					
						
							|  |  |  |       info.lastInsertRowid, | 
					
						
							|  |  |  |       'insertSentProto/lastInsertRowid' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Insert a record for each recipient device.
 | 
					
						
							|  |  |  |     const recipientStatement = prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       INSERT INTO sendLogRecipients ( | 
					
						
							|  |  |  |         payloadId, | 
					
						
							|  |  |  |         recipientUuid, | 
					
						
							|  |  |  |         deviceId | 
					
						
							|  |  |  |       ) VALUES ( | 
					
						
							|  |  |  |         $id, | 
					
						
							|  |  |  |         $recipientUuid, | 
					
						
							|  |  |  |         $deviceId | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const recipientUuids = Object.keys(recipients); | 
					
						
							|  |  |  |     for (const recipientUuid of recipientUuids) { | 
					
						
							|  |  |  |       const deviceIds = recipients[recipientUuid]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (const deviceId of deviceIds) { | 
					
						
							|  |  |  |         recipientStatement.run({ | 
					
						
							|  |  |  |           id, | 
					
						
							|  |  |  |           recipientUuid, | 
					
						
							|  |  |  |           deviceId, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Insert a record for each message referenced by this payload.
 | 
					
						
							|  |  |  |     const messageStatement = prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       INSERT INTO sendLogMessageIds ( | 
					
						
							|  |  |  |         payloadId, | 
					
						
							|  |  |  |         messageId | 
					
						
							|  |  |  |       ) VALUES ( | 
					
						
							|  |  |  |         $id, | 
					
						
							|  |  |  |         $messageId | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |     for (const messageId of new Set(messageIds)) { | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |       messageStatement.run({ | 
					
						
							|  |  |  |         id, | 
					
						
							|  |  |  |         messageId, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return id; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function deleteSentProtosOlderThan(timestamp: number): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM sendLogPayloads | 
					
						
							|  |  |  |     WHERE | 
					
						
							|  |  |  |       timestamp IS NULL OR | 
					
						
							|  |  |  |       timestamp < $timestamp; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     timestamp, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function deleteSentProtoByMessageId(messageId: string): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM sendLogPayloads WHERE id IN ( | 
					
						
							|  |  |  |       SELECT payloadId FROM sendLogMessageIds | 
					
						
							|  |  |  |       WHERE messageId = $messageId | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     messageId, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function insertProtoRecipients({ | 
					
						
							|  |  |  |   id, | 
					
						
							|  |  |  |   recipientUuid, | 
					
						
							|  |  |  |   deviceIds, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   id: number; | 
					
						
							|  |  |  |   recipientUuid: string; | 
					
						
							|  |  |  |   deviceIds: Array<number>; | 
					
						
							|  |  |  | }): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     const statement = prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       INSERT INTO sendLogRecipients ( | 
					
						
							|  |  |  |         payloadId, | 
					
						
							|  |  |  |         recipientUuid, | 
					
						
							|  |  |  |         deviceId | 
					
						
							|  |  |  |       ) VALUES ( | 
					
						
							|  |  |  |         $id, | 
					
						
							|  |  |  |         $recipientUuid, | 
					
						
							|  |  |  |         $deviceId | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const deviceId of deviceIds) { | 
					
						
							|  |  |  |       statement.run({ | 
					
						
							|  |  |  |         id, | 
					
						
							|  |  |  |         recipientUuid, | 
					
						
							|  |  |  |         deviceId, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  | async function deleteSentProtoRecipient( | 
					
						
							|  |  |  |   options: | 
					
						
							|  |  |  |     | DeleteSentProtoRecipientOptionsType | 
					
						
							|  |  |  |     | ReadonlyArray<DeleteSentProtoRecipientOptionsType> | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  | ): Promise<DeleteSentProtoRecipientResultType> { | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |   const items = Array.isArray(options) ? options : [options]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Note: we use `pluck` in this function to fetch only the first column of
 | 
					
						
							|  |  |  |   // returned row.
 | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     const successfulPhoneNumberShares = new Array<string>(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |     for (const item of items) { | 
					
						
							|  |  |  |       const { timestamp, recipientUuid, deviceId } = item; | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |       // 1. Figure out what payload we're talking about.
 | 
					
						
							|  |  |  |       const rows = prepare( | 
					
						
							|  |  |  |         db, | 
					
						
							|  |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |         SELECT sendLogPayloads.id, sendLogPayloads.hasPniSignatureMessage | 
					
						
							|  |  |  |         FROM sendLogPayloads | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |         INNER JOIN sendLogRecipients | 
					
						
							|  |  |  |           ON sendLogRecipients.payloadId = sendLogPayloads.id | 
					
						
							|  |  |  |         WHERE | 
					
						
							|  |  |  |           sendLogPayloads.timestamp = $timestamp AND | 
					
						
							|  |  |  |           sendLogRecipients.recipientUuid = $recipientUuid AND | 
					
						
							|  |  |  |           sendLogRecipients.deviceId = $deviceId; | 
					
						
							|  |  |  |        `
 | 
					
						
							|  |  |  |       ).all({ timestamp, recipientUuid, deviceId }); | 
					
						
							|  |  |  |       if (!rows.length) { | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (rows.length > 1) { | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |         logger.warn( | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |           'deleteSentProtoRecipient: More than one payload matches ' + | 
					
						
							|  |  |  |             `recipient and timestamp ${timestamp}. Using the first.` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |       const { id, hasPniSignatureMessage } = rows[0]; | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |       // 2. Delete the recipient/device combination in question.
 | 
					
						
							|  |  |  |       prepare( | 
					
						
							|  |  |  |         db, | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         DELETE FROM sendLogRecipients | 
					
						
							|  |  |  |         WHERE | 
					
						
							|  |  |  |           payloadId = $id AND | 
					
						
							|  |  |  |           recipientUuid = $recipientUuid AND | 
					
						
							|  |  |  |           deviceId = $deviceId; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ id, recipientUuid, deviceId }); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |       // 3. See how many more recipient devices there were for this payload.
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |       const remainingDevices = prepare( | 
					
						
							|  |  |  |         db, | 
					
						
							|  |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         SELECT count(1) FROM sendLogRecipients | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |         WHERE payloadId = $id AND recipientUuid = $recipientUuid; | 
					
						
							| 
									
										
										
										
											2023-01-25 13:22:13 -08:00
										 |  |  |         `,
 | 
					
						
							|  |  |  |         { pluck: true } | 
					
						
							|  |  |  |       ).get({ id, recipientUuid }); | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |       // 4. If there are no remaining devices for this recipient and we included
 | 
					
						
							|  |  |  |       //    the pni signature in the proto - return the recipient to the caller.
 | 
					
						
							|  |  |  |       if (remainingDevices === 0 && hasPniSignatureMessage) { | 
					
						
							|  |  |  |         logger.info( | 
					
						
							|  |  |  |           'deleteSentProtoRecipient: ' + | 
					
						
							|  |  |  |             `Successfully shared phone number with ${recipientUuid} ` + | 
					
						
							|  |  |  |             `through message ${timestamp}` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         successfulPhoneNumberShares.push(recipientUuid); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       strictAssert( | 
					
						
							|  |  |  |         isNumber(remainingDevices), | 
					
						
							|  |  |  |         'deleteSentProtoRecipient: select count() returned non-number!' | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // 5. See how many more recipients there were for this payload.
 | 
					
						
							|  |  |  |       const remainingTotal = prepare( | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |         db, | 
					
						
							| 
									
										
										
										
											2023-01-25 13:22:13 -08:00
										 |  |  |         'SELECT count(1) FROM sendLogRecipients WHERE payloadId = $id;', | 
					
						
							|  |  |  |         { pluck: true } | 
					
						
							|  |  |  |       ).get({ id }); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |       strictAssert( | 
					
						
							|  |  |  |         isNumber(remainingTotal), | 
					
						
							|  |  |  |         'deleteSentProtoRecipient: select count() returned non-number!' | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |       if (remainingTotal > 0) { | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |       // 6. Delete the entire payload if there are no more recipients left.
 | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |       logger.info( | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |         'deleteSentProtoRecipient: ' + | 
					
						
							|  |  |  |           `Deleting proto payload for timestamp ${timestamp}` | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 14:35:01 -07:00
										 |  |  |       prepare(db, 'DELETE FROM sendLogPayloads WHERE id = $id;').run({ | 
					
						
							|  |  |  |         id, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return { successfulPhoneNumberShares }; | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function getSentProtoByRecipient({ | 
					
						
							|  |  |  |   now, | 
					
						
							|  |  |  |   recipientUuid, | 
					
						
							|  |  |  |   timestamp, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   now: number; | 
					
						
							|  |  |  |   recipientUuid: string; | 
					
						
							|  |  |  |   timestamp: number; | 
					
						
							|  |  |  | }): Promise<SentProtoWithMessageIdsType | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const HOUR = 1000 * 60 * 60; | 
					
						
							|  |  |  |   const oneDayAgo = now - HOUR * 24; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   await deleteSentProtosOlderThan(oneDayAgo); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const row = prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     SELECT | 
					
						
							|  |  |  |       sendLogPayloads.*, | 
					
						
							|  |  |  |       GROUP_CONCAT(DISTINCT sendLogMessageIds.messageId) AS messageIds | 
					
						
							|  |  |  |     FROM sendLogPayloads | 
					
						
							|  |  |  |     INNER JOIN sendLogRecipients ON sendLogRecipients.payloadId = sendLogPayloads.id | 
					
						
							|  |  |  |     LEFT JOIN sendLogMessageIds ON sendLogMessageIds.payloadId = sendLogPayloads.id | 
					
						
							|  |  |  |     WHERE | 
					
						
							|  |  |  |       sendLogPayloads.timestamp = $timestamp AND | 
					
						
							|  |  |  |       sendLogRecipients.recipientUuid = $recipientUuid | 
					
						
							|  |  |  |     GROUP BY sendLogPayloads.id; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).get({ | 
					
						
							|  |  |  |     timestamp, | 
					
						
							|  |  |  |     recipientUuid, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const { messageIds } = row; | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     ...row, | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |     urgent: isNumber(row.urgent) ? Boolean(row.urgent) : true, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |     hasPniSignatureMessage: isNumber(row.hasPniSignatureMessage) | 
					
						
							|  |  |  |       ? Boolean(row.hasPniSignatureMessage) | 
					
						
							|  |  |  |       : true, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |     messageIds: messageIds ? messageIds.split(',') : [], | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function removeAllSentProtos(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   prepare<EmptyQuery>(db, 'DELETE FROM sendLogPayloads;').run(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function getAllSentProtos(): Promise<Array<SentProtoType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const rows = prepare<EmptyQuery>(db, 'SELECT * FROM sendLogPayloads;').all(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |   return rows.map(row => ({ | 
					
						
							|  |  |  |     ...row, | 
					
						
							|  |  |  |     urgent: isNumber(row.urgent) ? Boolean(row.urgent) : true, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:53:33 -07:00
										 |  |  |     hasPniSignatureMessage: isNumber(row.hasPniSignatureMessage) | 
					
						
							|  |  |  |       ? Boolean(row.hasPniSignatureMessage) | 
					
						
							|  |  |  |       : true, | 
					
						
							| 
									
										
										
										
											2022-07-01 09:55:13 -07:00
										 |  |  |   })); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | } | 
					
						
							|  |  |  | async function _getAllSentProtoRecipients(): Promise< | 
					
						
							|  |  |  |   Array<SentRecipientsDBType> | 
					
						
							|  |  |  | > { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const rows = prepare<EmptyQuery>( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     'SELECT * FROM sendLogRecipients;' | 
					
						
							|  |  |  |   ).all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return rows; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function _getAllSentProtoMessageIds(): Promise<Array<SentMessageDBType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const rows = prepare<EmptyQuery>( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     'SELECT * FROM sendLogMessageIds;' | 
					
						
							|  |  |  |   ).all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return rows; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | const SESSIONS_TABLE = 'sessions'; | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | function createOrUpdateSessionSync(data: SessionType): void { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |   const { id, conversationId, ourUuid, uuid } = data; | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   if (!id) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'createOrUpdateSession: Provided data did not have a truthy id' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |   if (!conversationId) { | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |     throw new Error( | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |       'createOrUpdateSession: Provided data did not have a truthy conversationId' | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO sessions ( | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |       id, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |       ourUuid, | 
					
						
							|  |  |  |       uuid, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |       json | 
					
						
							|  |  |  |     ) values ( | 
					
						
							|  |  |  |       $id, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |       $conversationId, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |       $ourUuid, | 
					
						
							|  |  |  |       $uuid, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |       $json | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     ) | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  |     ourUuid, | 
					
						
							|  |  |  |     uuid, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     json: objectToJSON(data), | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | async function createOrUpdateSession(data: SessionType): Promise<void> { | 
					
						
							|  |  |  |   return createOrUpdateSessionSync(data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function createOrUpdateSessions( | 
					
						
							|  |  |  |   array: Array<SessionType> | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     for (const item of array) { | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       assertSync(createOrUpdateSessionSync(item)); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-07 18:12:13 -08:00
										 |  |  | async function commitDecryptResult({ | 
					
						
							|  |  |  |   senderKeys, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |   sessions, | 
					
						
							|  |  |  |   unprocessed, | 
					
						
							|  |  |  | }: { | 
					
						
							| 
									
										
										
										
											2022-01-07 18:12:13 -08:00
										 |  |  |   senderKeys: Array<SenderKeyType>; | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |   sessions: Array<SessionType>; | 
					
						
							|  |  |  |   unprocessed: Array<UnprocessedType>; | 
					
						
							|  |  |  | }): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2022-01-07 18:12:13 -08:00
										 |  |  |     for (const item of senderKeys) { | 
					
						
							|  |  |  |       assertSync(createOrUpdateSenderKeySync(item)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     for (const item of sessions) { | 
					
						
							| 
									
										
										
										
											2021-05-24 14:30:56 -07:00
										 |  |  |       assertSync(createOrUpdateSessionSync(item)); | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const item of unprocessed) { | 
					
						
							| 
									
										
										
										
											2021-05-24 14:30:56 -07:00
										 |  |  |       assertSync(saveUnprocessedSync(item)); | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function bulkAddSessions(array: Array<SessionType>): Promise<void> { | 
					
						
							|  |  |  |   return bulkAdd(getInstance(), SESSIONS_TABLE, array); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-09 19:38:11 -07:00
										 |  |  | async function removeSessionById(id: SessionIdType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return removeById(getInstance(), SESSIONS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeSessionsByConversation( | 
					
						
							|  |  |  |   conversationId: string | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM sessions | 
					
						
							|  |  |  |     WHERE conversationId = $conversationId; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-12-12 14:06:16 -08:00
										 |  |  | async function removeSessionsByUUID(uuid: UUIDStringType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM sessions | 
					
						
							|  |  |  |     WHERE uuid = $uuid; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     uuid, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function removeAllSessions(): Promise<void> { | 
					
						
							|  |  |  |   return removeAllFromTable(getInstance(), SESSIONS_TABLE); | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function getAllSessions(): Promise<Array<SessionType>> { | 
					
						
							|  |  |  |   return getAllFromTable(getInstance(), SESSIONS_TABLE); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | // Conversations
 | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function getConversationCount(): Promise<number> { | 
					
						
							|  |  |  |   return getCountFromTable(getInstance(), 'conversations'); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function getConversationMembersList({ members, membersV2 }: ConversationType) { | 
					
						
							|  |  |  |   if (membersV2) { | 
					
						
							|  |  |  |     return membersV2.map((item: GroupV2MemberType) => item.uuid).join(' '); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   if (members) { | 
					
						
							|  |  |  |     return members.join(' '); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return null; | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | function saveConversationSync( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   data: ConversationType, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db = getInstance() | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | ): void { | 
					
						
							| 
									
										
										
										
											2020-01-13 14:28:28 -08:00
										 |  |  |   const { | 
					
						
							|  |  |  |     active_at, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |     e164, | 
					
						
							|  |  |  |     groupId, | 
					
						
							|  |  |  |     id, | 
					
						
							| 
									
										
										
										
											2020-01-13 14:28:28 -08:00
										 |  |  |     name, | 
					
						
							|  |  |  |     profileFamilyName, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |     profileName, | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |     profileLastFetchedAt, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |     type, | 
					
						
							|  |  |  |     uuid, | 
					
						
							| 
									
										
										
										
											2020-01-13 14:28:28 -08:00
										 |  |  |   } = data; | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const membersList = getConversationMembersList(data); | 
					
						
							| 
									
										
										
										
											2020-09-08 19:25:05 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT INTO conversations ( | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       json, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       e164, | 
					
						
							|  |  |  |       uuid, | 
					
						
							|  |  |  |       groupId, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       active_at, | 
					
						
							|  |  |  |       type, | 
					
						
							|  |  |  |       members, | 
					
						
							|  |  |  |       name, | 
					
						
							|  |  |  |       profileName, | 
					
						
							|  |  |  |       profileFamilyName, | 
					
						
							|  |  |  |       profileFullName, | 
					
						
							|  |  |  |       profileLastFetchedAt | 
					
						
							|  |  |  |     ) values ( | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $json, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       $e164, | 
					
						
							|  |  |  |       $uuid, | 
					
						
							|  |  |  |       $groupId, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       $active_at, | 
					
						
							|  |  |  |       $type, | 
					
						
							|  |  |  |       $members, | 
					
						
							|  |  |  |       $name, | 
					
						
							|  |  |  |       $profileName, | 
					
						
							|  |  |  |       $profileFamilyName, | 
					
						
							|  |  |  |       $profileFullName, | 
					
						
							|  |  |  |       $profileLastFetchedAt | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |     id, | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |     json: objectToJSON( | 
					
						
							|  |  |  |       omit(data, ['profileLastFetchedAt', 'unblurredAvatarPath']) | 
					
						
							|  |  |  |     ), | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     e164: e164 || null, | 
					
						
							|  |  |  |     uuid: uuid || null, | 
					
						
							|  |  |  |     groupId: groupId || null, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     active_at: active_at || null, | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |     type, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     members: membersList, | 
					
						
							|  |  |  |     name: name || null, | 
					
						
							|  |  |  |     profileName: profileName || null, | 
					
						
							|  |  |  |     profileFamilyName: profileFamilyName || null, | 
					
						
							|  |  |  |     profileFullName: combineNames(profileName, profileFamilyName) || null, | 
					
						
							|  |  |  |     profileLastFetchedAt: profileLastFetchedAt || null, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | async function saveConversation( | 
					
						
							|  |  |  |   data: ConversationType, | 
					
						
							|  |  |  |   db = getInstance() | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return saveConversationSync(data, db); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function saveConversations( | 
					
						
							|  |  |  |   arrayOfConversations: Array<ConversationType> | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     for (const conversation of arrayOfConversations) { | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       assertSync(saveConversationSync(conversation)); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function updateConversationSync( | 
					
						
							|  |  |  |   data: ConversationType, | 
					
						
							|  |  |  |   db = getInstance() | 
					
						
							|  |  |  | ): void { | 
					
						
							| 
									
										
										
										
											2020-01-13 14:28:28 -08:00
										 |  |  |   const { | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     active_at, | 
					
						
							|  |  |  |     type, | 
					
						
							|  |  |  |     name, | 
					
						
							|  |  |  |     profileName, | 
					
						
							|  |  |  |     profileFamilyName, | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |     profileLastFetchedAt, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |     e164, | 
					
						
							|  |  |  |     uuid, | 
					
						
							| 
									
										
										
										
											2020-01-13 14:28:28 -08:00
										 |  |  |   } = data; | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const membersList = getConversationMembersList(data); | 
					
						
							| 
									
										
										
										
											2020-09-08 19:25:05 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   db.prepare( | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE conversations SET | 
					
						
							| 
									
										
										
										
											2019-05-23 18:27:42 -07:00
										 |  |  |       json = $json, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |       e164 = $e164, | 
					
						
							|  |  |  |       uuid = $uuid, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 18:27:42 -07:00
										 |  |  |       active_at = $active_at, | 
					
						
							|  |  |  |       type = $type, | 
					
						
							|  |  |  |       members = $members, | 
					
						
							|  |  |  |       name = $name, | 
					
						
							| 
									
										
										
										
											2020-01-13 14:28:28 -08:00
										 |  |  |       profileName = $profileName, | 
					
						
							|  |  |  |       profileFamilyName = $profileFamilyName, | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  |       profileFullName = $profileFullName, | 
					
						
							|  |  |  |       profileLastFetchedAt = $profileLastFetchedAt | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     WHERE id = $id; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |     json: objectToJSON( | 
					
						
							|  |  |  |       omit(data, ['profileLastFetchedAt', 'unblurredAvatarPath']) | 
					
						
							|  |  |  |     ), | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     e164: e164 || null, | 
					
						
							|  |  |  |     uuid: uuid || null, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     active_at: active_at || null, | 
					
						
							|  |  |  |     type, | 
					
						
							|  |  |  |     members: membersList, | 
					
						
							|  |  |  |     name: name || null, | 
					
						
							|  |  |  |     profileName: profileName || null, | 
					
						
							|  |  |  |     profileFamilyName: profileFamilyName || null, | 
					
						
							|  |  |  |     profileFullName: combineNames(profileName, profileFamilyName) || null, | 
					
						
							|  |  |  |     profileLastFetchedAt: profileLastFetchedAt || null, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-04 16:44:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | async function updateConversation(data: ConversationType): Promise<void> { | 
					
						
							|  |  |  |   return updateConversationSync(data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function updateConversations( | 
					
						
							|  |  |  |   array: Array<ConversationType> | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     for (const item of array) { | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       assertSync(updateConversationSync(item)); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  | function removeConversationsSync(ids: ReadonlyArray<string>): void { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-06-22 11:44:51 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Our node interface doesn't seem to allow you to replace one single ? with an array
 | 
					
						
							|  |  |  |   db.prepare<ArrayQuery>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM conversations | 
					
						
							|  |  |  |     WHERE id IN ( ${ids.map(() => '?').join(', ')} ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(ids); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function removeConversation(id: Array<string> | string): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   if (!Array.isArray(id)) { | 
					
						
							| 
									
										
										
										
											2021-04-06 11:15:17 -07:00
										 |  |  |     db.prepare<Query>('DELETE FROM conversations WHERE id = $id;').run({ | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       id, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!id.length) { | 
					
						
							|  |  |  |     throw new Error('removeConversation: No ids to delete!'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   batchMultiVarQuery(db, id, removeConversationsSync); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-09 14:39:00 -07:00
										 |  |  | async function _removeAllConversations(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<EmptyQuery>('DELETE from conversations;').run(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getConversationById( | 
					
						
							|  |  |  |   id: string | 
					
						
							|  |  |  | ): Promise<ConversationType | undefined> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const row: { json: string } = db | 
					
						
							|  |  |  |     .prepare<Query>('SELECT json FROM conversations WHERE id = $id;') | 
					
						
							|  |  |  |     .get({ id }); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return jsonToObject(row.json); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function eraseStorageServiceStateFromConversations(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-07-24 16:32:08 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<EmptyQuery>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE conversations | 
					
						
							|  |  |  |     SET | 
					
						
							| 
									
										
										
										
											2020-09-29 19:29:11 -04:00
										 |  |  |       json = json_remove(json, '$.storageID', '$.needsStorageServiceSync', '$.unknownFields', '$.storageProfileKey'); | 
					
						
							| 
									
										
										
										
											2020-07-24 16:32:08 -07:00
										 |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   ).run(); | 
					
						
							| 
									
										
										
										
											2020-07-24 16:32:08 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function getAllConversationsSync(db = getInstance()): Array<ConversationType> { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows: ConversationRows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT json, profileLastFetchedAt | 
					
						
							|  |  |  |       FROM conversations | 
					
						
							|  |  |  |       ORDER BY id ASC; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return rows.map(row => rowToConversation(row)); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function getAllConversations(): Promise<Array<ConversationType>> { | 
					
						
							|  |  |  |   return getAllConversationsSync(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getAllConversationIds(): Promise<Array<string>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows: Array<{ id: string }> = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT id FROM conversations ORDER BY id ASC; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return rows.map(row => row.id); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function getAllGroupsInvolvingUuid( | 
					
						
							|  |  |  |   uuid: UUIDStringType | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<ConversationType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows: ConversationRows = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT json, profileLastFetchedAt | 
					
						
							|  |  |  |       FROM conversations WHERE | 
					
						
							|  |  |  |         type = 'group' AND | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |         members LIKE $uuid | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       ORDER BY id ASC; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |       uuid: `%${uuid}%`, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return rows.map(row => rowToConversation(row)); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function searchMessages( | 
					
						
							|  |  |  |   query: string, | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |   params: { limit?: number; conversationId?: string } = {} | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  | ): Promise<Array<ServerSearchResultMessageType>> { | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |   const { limit = 500, conversationId } = params; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-26 15:01:22 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // sqlite queries with a join on a virtual table (like FTS5) are de-optimized
 | 
					
						
							|  |  |  |   // and can't use indices for ordering results. Instead an in-memory index of
 | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |   // the join rows is sorted on the fly, and this becomes substantially
 | 
					
						
							|  |  |  |   // slower when there are large columns in it (like `messages.json`).
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // Thus here we take an indirect approach and store `rowid`s in a temporary
 | 
					
						
							|  |  |  |   // table for all messages that match the FTS query. Then we create another
 | 
					
						
							|  |  |  |   // table to sort and limit the results, and finally join on it when fetch
 | 
					
						
							|  |  |  |   // the snippets and json. The benefit of this is that the `ORDER BY` and
 | 
					
						
							|  |  |  |   // `LIMIT` happen without virtual table and are thus covered by
 | 
					
						
							|  |  |  |   // `messages_searchOrder` index.
 | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     db.exec( | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |       CREATE TEMP TABLE tmp_results(rowid INTEGER PRIMARY KEY ASC); | 
					
						
							|  |  |  |       CREATE TEMP TABLE tmp_filtered_results(rowid INTEGER PRIMARY KEY ASC); | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |         INSERT INTO tmp_results (rowid) | 
					
						
							|  |  |  |         SELECT | 
					
						
							|  |  |  |           rowid | 
					
						
							|  |  |  |         FROM | 
					
						
							|  |  |  |           messages_fts | 
					
						
							| 
									
										
										
										
											2021-04-26 15:01:22 -07:00
										 |  |  |         WHERE | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |           messages_fts.body MATCH $query; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |     ).run({ query }); | 
					
						
							| 
									
										
										
										
											2019-01-14 13:47:19 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |     if (conversationId === undefined) { | 
					
						
							|  |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |           INSERT INTO tmp_filtered_results (rowid) | 
					
						
							|  |  |  |           SELECT | 
					
						
							|  |  |  |             tmp_results.rowid | 
					
						
							|  |  |  |           FROM | 
					
						
							|  |  |  |             tmp_results | 
					
						
							|  |  |  |           INNER JOIN | 
					
						
							|  |  |  |             messages ON messages.rowid = tmp_results.rowid | 
					
						
							|  |  |  |           ORDER BY messages.received_at DESC, messages.sent_at DESC | 
					
						
							|  |  |  |           LIMIT $limit; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ limit }); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |           INSERT INTO tmp_filtered_results (rowid) | 
					
						
							|  |  |  |           SELECT | 
					
						
							|  |  |  |             tmp_results.rowid | 
					
						
							|  |  |  |           FROM | 
					
						
							|  |  |  |             tmp_results | 
					
						
							|  |  |  |           INNER JOIN | 
					
						
							|  |  |  |             messages ON messages.rowid = tmp_results.rowid | 
					
						
							|  |  |  |           WHERE | 
					
						
							|  |  |  |             messages.conversationId = $conversationId | 
					
						
							|  |  |  |           ORDER BY messages.received_at DESC, messages.sent_at DESC | 
					
						
							|  |  |  |           LIMIT $limit; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ conversationId, limit }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // The `MATCH` is necessary in order to for `snippet()` helper function to
 | 
					
						
							|  |  |  |     // give us the right results. We can't call `snippet()` in the query above
 | 
					
						
							|  |  |  |     // because it would bloat the temporary table with text data and we want
 | 
					
						
							|  |  |  |     // to keep its size minimal for `ORDER BY` + `LIMIT` to be fast.
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |     const result = db | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |           messages.json, | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |           snippet(messages_fts, -1, '<<left>>', '<<right>>', '...', 10) | 
					
						
							|  |  |  |             AS snippet | 
					
						
							|  |  |  |         FROM tmp_filtered_results | 
					
						
							|  |  |  |         INNER JOIN messages_fts | 
					
						
							|  |  |  |           ON messages_fts.rowid = tmp_filtered_results.rowid | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |         INNER JOIN messages | 
					
						
							|  |  |  |           ON messages.rowid = tmp_filtered_results.rowid | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |         WHERE | 
					
						
							|  |  |  |           messages_fts.body MATCH $query | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |         ORDER BY messages.received_at DESC, messages.sent_at DESC; | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .all({ query }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     db.exec( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       DROP TABLE tmp_results; | 
					
						
							|  |  |  |       DROP TABLE tmp_filtered_results; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2019-01-14 13:47:19 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function searchMessagesInConversation( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   query: string, | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |   { limit = 100 }: { limit?: number } = {} | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  | ): Promise<Array<ServerSearchResultMessageType>> { | 
					
						
							| 
									
										
										
										
											2021-04-27 13:24:57 -07:00
										 |  |  |   return searchMessages(query, { conversationId, limit }); | 
					
						
							| 
									
										
										
										
											2019-01-14 13:47:19 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | function getMessageCountSync( | 
					
						
							|  |  |  |   conversationId?: string, | 
					
						
							|  |  |  |   db = getInstance() | 
					
						
							|  |  |  | ): number { | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |   if (conversationId === undefined) { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     return getCountFromTable(db, 'messages'); | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const count = db | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         SELECT count(1) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         FROM messages | 
					
						
							|  |  |  |         WHERE conversationId = $conversationId; | 
					
						
							|  |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     .pluck() | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |     .get({ conversationId }); | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return count; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  | async function getStoryCount(conversationId: string): Promise<number> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         SELECT count(1) | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  |         FROM messages | 
					
						
							|  |  |  |         WHERE conversationId = $conversationId AND isStory = 1; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck() | 
					
						
							|  |  |  |     .get({ conversationId }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function getMessageCount(conversationId?: string): Promise<number> { | 
					
						
							|  |  |  |   return getMessageCountSync(conversationId); | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  | // Note: we really only use this in 1:1 conversations, where story replies are always
 | 
					
						
							|  |  |  | //   shown, so this has no need to be story-aware.
 | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | function hasUserInitiatedMessages(conversationId: string): boolean { | 
					
						
							| 
									
										
										
										
											2021-06-02 13:55:10 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   const exists: number = db | 
					
						
							| 
									
										
										
										
											2021-06-02 13:55:10 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |       SELECT EXISTS( | 
					
						
							|  |  |  |         SELECT 1 FROM messages | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         INDEXED BY message_user_initiated | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |         WHERE | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |           conversationId IS $conversationId AND | 
					
						
							|  |  |  |           isUserInitiatedMessage IS 1 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-06-02 13:55:10 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |     .pluck() | 
					
						
							| 
									
										
										
										
											2021-06-02 13:55:10 -07:00
										 |  |  |     .get({ conversationId }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   return exists !== 0; | 
					
						
							| 
									
										
										
										
											2021-06-02 13:55:10 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | function saveMessageSync( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   data: MessageType, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   options: { | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |     alreadyInTransaction?: boolean; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     db?: Database; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |     forceSave?: boolean; | 
					
						
							|  |  |  |     jobToInsert?: StoredJob; | 
					
						
							|  |  |  |     ourUuid: UUIDStringType; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | ): string { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const { | 
					
						
							|  |  |  |     alreadyInTransaction, | 
					
						
							|  |  |  |     db = getInstance(), | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |     forceSave, | 
					
						
							|  |  |  |     jobToInsert, | 
					
						
							|  |  |  |     ourUuid, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   } = options; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!alreadyInTransaction) { | 
					
						
							|  |  |  |     return db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       return assertSync( | 
					
						
							|  |  |  |         saveMessageSync(data, { | 
					
						
							|  |  |  |           ...options, | 
					
						
							|  |  |  |           alreadyInTransaction: true, | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     })(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   const { | 
					
						
							| 
									
										
										
										
											2019-01-14 13:47:19 -08:00
										 |  |  |     body, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |     groupV2Change, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     hasAttachments, | 
					
						
							|  |  |  |     hasFileAttachments, | 
					
						
							|  |  |  |     hasVisualMediaAttachments, | 
					
						
							|  |  |  |     id, | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  |     isErased, | 
					
						
							| 
									
										
										
										
											2019-08-05 13:53:15 -07:00
										 |  |  |     isViewOnce, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     received_at, | 
					
						
							|  |  |  |     schemaVersion, | 
					
						
							|  |  |  |     sent_at, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |     serverGuid, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     source, | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  |     sourceUuid, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     storyId, | 
					
						
							| 
									
										
										
										
											2018-08-07 12:33:56 -07:00
										 |  |  |     type, | 
					
						
							| 
									
										
										
										
											2021-08-12 13:15:55 -05:00
										 |  |  |     readStatus, | 
					
						
							| 
									
										
										
										
											2018-08-07 12:33:56 -07:00
										 |  |  |     expireTimer, | 
					
						
							|  |  |  |     expirationStartTimestamp, | 
					
						
							| 
									
										
										
										
											2022-07-06 15:53:25 -07:00
										 |  |  |     attachments, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   } = data; | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |   let { seenStatus } = data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 15:53:25 -07:00
										 |  |  |   if (attachments) { | 
					
						
							|  |  |  |     strictAssert( | 
					
						
							|  |  |  |       attachments.every(attachment => !attachment.data), | 
					
						
							|  |  |  |       'Attempting to save a hydrated message' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |   if (readStatus === ReadStatus.Unread && seenStatus !== SeenStatus.Unseen) { | 
					
						
							|  |  |  |     log.warn( | 
					
						
							|  |  |  |       `saveMessage: Message ${id}/${type} is unread but had seenStatus=${seenStatus}. Forcing to UnseenStatus.Unseen.` | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // eslint-disable-next-line no-param-reassign
 | 
					
						
							|  |  |  |     data = { | 
					
						
							|  |  |  |       ...data, | 
					
						
							|  |  |  |       seenStatus: SeenStatus.Unseen, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     seenStatus = SeenStatus.Unseen; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-07 12:33:56 -07:00
										 |  |  |   const payload = { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     id, | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |     json: objectToJSON(data), | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     body: body || null, | 
					
						
							|  |  |  |     conversationId, | 
					
						
							|  |  |  |     expirationStartTimestamp: expirationStartTimestamp || null, | 
					
						
							|  |  |  |     expireTimer: expireTimer || null, | 
					
						
							|  |  |  |     hasAttachments: hasAttachments ? 1 : 0, | 
					
						
							|  |  |  |     hasFileAttachments: hasFileAttachments ? 1 : 0, | 
					
						
							|  |  |  |     hasVisualMediaAttachments: hasVisualMediaAttachments ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |     isChangeCreatedByUs: groupV2Change?.from === ourUuid ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     isErased: isErased ? 1 : 0, | 
					
						
							|  |  |  |     isViewOnce: isViewOnce ? 1 : 0, | 
					
						
							|  |  |  |     received_at: received_at || null, | 
					
						
							| 
									
										
										
										
											2021-06-17 10:15:10 -07:00
										 |  |  |     schemaVersion: schemaVersion || 0, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |     serverGuid: serverGuid || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     sent_at: sent_at || null, | 
					
						
							|  |  |  |     source: source || null, | 
					
						
							|  |  |  |     sourceUuid: sourceUuid || null, | 
					
						
							|  |  |  |     sourceDevice: sourceDevice || null, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     type: type || null, | 
					
						
							| 
									
										
										
										
											2021-08-12 13:15:55 -05:00
										 |  |  |     readStatus: readStatus ?? null, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |     seenStatus: seenStatus ?? SeenStatus.NotApplicable, | 
					
						
							| 
									
										
										
										
											2018-08-07 12:33:56 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   if (id && !forceSave) { | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |     prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |       UPDATE messages SET | 
					
						
							|  |  |  |         id = $id, | 
					
						
							|  |  |  |         json = $json, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         body = $body, | 
					
						
							|  |  |  |         conversationId = $conversationId, | 
					
						
							|  |  |  |         expirationStartTimestamp = $expirationStartTimestamp, | 
					
						
							|  |  |  |         expireTimer = $expireTimer, | 
					
						
							|  |  |  |         hasAttachments = $hasAttachments, | 
					
						
							|  |  |  |         hasFileAttachments = $hasFileAttachments, | 
					
						
							|  |  |  |         hasVisualMediaAttachments = $hasVisualMediaAttachments, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |         isChangeCreatedByUs = $isChangeCreatedByUs, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         isErased = $isErased, | 
					
						
							|  |  |  |         isViewOnce = $isViewOnce, | 
					
						
							|  |  |  |         received_at = $received_at, | 
					
						
							|  |  |  |         schemaVersion = $schemaVersion, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |         serverGuid = $serverGuid, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         sent_at = $sent_at, | 
					
						
							|  |  |  |         source = $source, | 
					
						
							|  |  |  |         sourceUuid = $sourceUuid, | 
					
						
							|  |  |  |         sourceDevice = $sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         storyId = $storyId, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         type = $type, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |         readStatus = $readStatus, | 
					
						
							|  |  |  |         seenStatus = $seenStatus | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       WHERE id = $id; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ).run(payload); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |     if (jobToInsert) { | 
					
						
							|  |  |  |       insertJobSync(db, jobToInsert); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |     return id; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const toCreate = { | 
					
						
							|  |  |  |     ...data, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     id: id || UUID.generate().toString(), | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT INTO messages ( | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       json, | 
					
						
							| 
									
										
										
										
											2021-03-22 18:09:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       body, | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       expirationStartTimestamp, | 
					
						
							|  |  |  |       expireTimer, | 
					
						
							|  |  |  |       hasAttachments, | 
					
						
							|  |  |  |       hasFileAttachments, | 
					
						
							|  |  |  |       hasVisualMediaAttachments, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |       isChangeCreatedByUs, | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       isErased, | 
					
						
							|  |  |  |       isViewOnce, | 
					
						
							|  |  |  |       received_at, | 
					
						
							|  |  |  |       schemaVersion, | 
					
						
							|  |  |  |       serverGuid, | 
					
						
							|  |  |  |       sent_at, | 
					
						
							|  |  |  |       source, | 
					
						
							|  |  |  |       sourceUuid, | 
					
						
							|  |  |  |       sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       storyId, | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       type, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |       readStatus, | 
					
						
							|  |  |  |       seenStatus | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |     ) values ( | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $json, | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       $body, | 
					
						
							|  |  |  |       $conversationId, | 
					
						
							|  |  |  |       $expirationStartTimestamp, | 
					
						
							|  |  |  |       $expireTimer, | 
					
						
							|  |  |  |       $hasAttachments, | 
					
						
							|  |  |  |       $hasFileAttachments, | 
					
						
							|  |  |  |       $hasVisualMediaAttachments, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |       $isChangeCreatedByUs, | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       $isErased, | 
					
						
							|  |  |  |       $isViewOnce, | 
					
						
							|  |  |  |       $received_at, | 
					
						
							|  |  |  |       $schemaVersion, | 
					
						
							|  |  |  |       $serverGuid, | 
					
						
							|  |  |  |       $sent_at, | 
					
						
							|  |  |  |       $source, | 
					
						
							|  |  |  |       $sourceUuid, | 
					
						
							|  |  |  |       $sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       $storyId, | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       $type, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |       $readStatus, | 
					
						
							|  |  |  |       $seenStatus | 
					
						
							| 
									
										
										
										
											2021-07-09 16:38:51 -05:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     ...payload, | 
					
						
							|  |  |  |     id: toCreate.id, | 
					
						
							|  |  |  |     json: objectToJSON(toCreate), | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |   if (jobToInsert) { | 
					
						
							|  |  |  |     insertJobSync(db, jobToInsert); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return toCreate.id; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | async function saveMessage( | 
					
						
							|  |  |  |   data: MessageType, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |   options: { | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |     jobToInsert?: StoredJob; | 
					
						
							|  |  |  |     forceSave?: boolean; | 
					
						
							|  |  |  |     alreadyInTransaction?: boolean; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |     ourUuid: UUIDStringType; | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | ): Promise<string> { | 
					
						
							|  |  |  |   return saveMessageSync(data, options); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function saveMessages( | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  |   arrayOfMessages: ReadonlyArray<MessageType>, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |   options: { forceSave?: boolean; ourUuid: UUIDStringType } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2018-07-31 19:29:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     for (const message of arrayOfMessages) { | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       assertSync( | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |         saveMessageSync(message, { ...options, alreadyInTransaction: true }) | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeMessage(id: string): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-02-04 12:46:55 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-06 11:15:17 -07:00
										 |  |  |   db.prepare<Query>('DELETE FROM messages WHERE id = $id;').run({ id }); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:42:15 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  | function removeMessagesSync(ids: ReadonlyArray<string>): void { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:42:15 -08:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-02-04 12:46:55 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-06 11:15:17 -07:00
										 |  |  |   db.prepare<ArrayQuery>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM messages | 
					
						
							|  |  |  |     WHERE id IN ( ${ids.map(() => '?').join(', ')} ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(ids); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  | async function removeMessages(ids: ReadonlyArray<string>): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   batchMultiVarQuery(getInstance(), ids, removeMessagesSync); | 
					
						
							| 
									
										
										
										
											2021-06-22 11:44:51 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getMessageById(id: string): Promise<MessageType | undefined> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2022-02-16 10:36:21 -08:00
										 |  |  |   return getMessageByIdSync(db, id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function getMessageByIdSync( | 
					
						
							|  |  |  |   db: Database, | 
					
						
							|  |  |  |   id: string | 
					
						
							|  |  |  | ): MessageType | undefined { | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>('SELECT json FROM messages WHERE id = $id;') | 
					
						
							| 
									
										
										
										
											2021-05-17 18:15:09 -05:00
										 |  |  |     .get({ | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return jsonToObject(row.json); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  | async function getMessagesById( | 
					
						
							|  |  |  |   messageIds: Array<string> | 
					
						
							|  |  |  | ): Promise<Array<MessageType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return batchMultiVarQuery( | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     db, | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |     messageIds, | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  |     (batch: ReadonlyArray<string>): Array<MessageType> => { | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  |       const query = db.prepare<ArrayQuery>( | 
					
						
							|  |  |  |         `SELECT json FROM messages WHERE id IN (${Array(batch.length) | 
					
						
							|  |  |  |           .fill('?') | 
					
						
							|  |  |  |           .join(',')});`
 | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |       const rows: JSONRows = query.all(batch); | 
					
						
							|  |  |  |       return rows.map(row => jsonToObject(row.json)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function _getAllMessages(): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>('SELECT json FROM messages ORDER BY id ASC;') | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | async function _removeAllMessages(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2023-01-26 15:53:22 -08:00
										 |  |  |   db.exec(`
 | 
					
						
							|  |  |  |     DELETE FROM messages; | 
					
						
							|  |  |  |     INSERT INTO messages_fts(messages_fts) VALUES('optimize'); | 
					
						
							|  |  |  |   `);
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getAllMessageIds(): Promise<Array<string>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows: Array<{ id: string }> = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>('SELECT id FROM messages ORDER BY id ASC;') | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return rows.map(row => row.id); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 13:14:58 -08:00
										 |  |  | async function getMessageBySender({ | 
					
						
							|  |  |  |   source, | 
					
						
							|  |  |  |   sourceUuid, | 
					
						
							|  |  |  |   sourceDevice, | 
					
						
							|  |  |  |   sent_at, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | }: { | 
					
						
							| 
									
										
										
										
											2023-01-05 13:58:13 -08:00
										 |  |  |   source?: string; | 
					
						
							|  |  |  |   sourceUuid?: UUIDStringType; | 
					
						
							|  |  |  |   sourceDevice?: number; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   sent_at: number; | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  | }): Promise<MessageType | undefined> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = prepare( | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |     SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |       (source = $source OR sourceUuid = $sourceUuid) AND | 
					
						
							|  |  |  |       sourceDevice = $sourceDevice AND | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |       sent_at = $sent_at | 
					
						
							|  |  |  |     LIMIT 2; | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |   ).all({ | 
					
						
							| 
									
										
										
										
											2023-01-05 13:58:13 -08:00
										 |  |  |     source: source || null, | 
					
						
							|  |  |  |     sourceUuid: sourceUuid || null, | 
					
						
							|  |  |  |     sourceDevice: sourceDevice || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |     sent_at, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-10 14:51:54 -08:00
										 |  |  |   if (rows.length > 1) { | 
					
						
							|  |  |  |     log.warn('getMessageBySender: More than one message found for', { | 
					
						
							|  |  |  |       sent_at, | 
					
						
							|  |  |  |       source, | 
					
						
							|  |  |  |       sourceUuid, | 
					
						
							|  |  |  |       sourceDevice, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (rows.length < 1) { | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return jsonToObject(rows[0].json); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 19:33:38 -04:00
										 |  |  | export function _storyIdPredicate( | 
					
						
							|  |  |  |   storyId: string | undefined, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies: boolean | 
					
						
							| 
									
										
										
										
											2022-04-20 19:33:38 -04:00
										 |  |  | ): string { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   // This is unintuitive, but 'including story replies' means that we need replies to
 | 
					
						
							|  |  |  |   //   lots of different stories. So, we remove the storyId check with a clause that will
 | 
					
						
							|  |  |  |   //   always be true. We don't just return TRUE because we want to use our passed-in
 | 
					
						
							|  |  |  |   //   $storyId parameter.
 | 
					
						
							|  |  |  |   if (includeStoryReplies && storyId === undefined) { | 
					
						
							| 
									
										
										
										
											2022-03-21 14:21:35 -07:00
										 |  |  |     return '$storyId IS NULL'; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-01 13:41:40 +02:00
										 |  |  |   // In contrast to: replies to a specific story
 | 
					
						
							| 
									
										
										
										
											2022-03-21 14:21:35 -07:00
										 |  |  |   return 'storyId IS $storyId'; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | async function getUnreadByConversationAndMarkRead({ | 
					
						
							|  |  |  |   conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   newestUnreadAt, | 
					
						
							|  |  |  |   storyId, | 
					
						
							|  |  |  |   readAt, | 
					
						
							| 
									
										
										
										
											2022-09-22 16:49:06 -07:00
										 |  |  |   now = Date.now(), | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   newestUnreadAt: number; | 
					
						
							|  |  |  |   storyId?: UUIDStringType; | 
					
						
							|  |  |  |   readAt?: number; | 
					
						
							| 
									
										
										
										
											2022-09-22 16:49:06 -07:00
										 |  |  |   now?: number; | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | }): Promise<GetUnreadByConversationAndMarkReadResultType> { | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2022-09-22 16:49:06 -07:00
										 |  |  |     const expirationStartTimestamp = Math.min(now, readAt ?? Infinity); | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       UPDATE messages | 
					
						
							| 
									
										
										
										
											2021-06-29 09:45:31 -07:00
										 |  |  |       INDEXED BY expiring_message_by_conversation_and_received_at | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |       SET | 
					
						
							|  |  |  |         expirationStartTimestamp = $expirationStartTimestamp, | 
					
						
							|  |  |  |         json = json_patch(json, $jsonPatch) | 
					
						
							|  |  |  |       WHERE | 
					
						
							| 
									
										
										
										
											2022-06-02 18:09:13 -07:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) AND | 
					
						
							| 
									
										
										
										
											2022-06-02 18:09:13 -07:00
										 |  |  |         isStory IS 0 AND | 
					
						
							|  |  |  |         type IS 'incoming' AND | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |         ( | 
					
						
							|  |  |  |           expirationStartTimestamp IS NULL OR | 
					
						
							|  |  |  |           expirationStartTimestamp > $expirationStartTimestamp | 
					
						
							|  |  |  |         ) AND | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |         expireTimer > 0 AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         received_at <= $newestUnreadAt; | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ).run({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       expirationStartTimestamp, | 
					
						
							|  |  |  |       jsonPatch: JSON.stringify({ expirationStartTimestamp }), | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       newestUnreadAt, | 
					
						
							|  |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |     const rows = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2021-06-29 09:45:31 -07:00
										 |  |  |         SELECT id, json FROM messages | 
					
						
							|  |  |  |         WHERE | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |           conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2022-04-29 16:42:47 -07:00
										 |  |  |           seenStatus = ${SeenStatus.Unseen} AND | 
					
						
							|  |  |  |           isStory = 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |           (${_storyIdPredicate(storyId, includeStoryReplies)}) AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |           received_at <= $newestUnreadAt | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |         ORDER BY received_at DESC, sent_at DESC; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .all({ | 
					
						
							|  |  |  |         conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         newestUnreadAt, | 
					
						
							|  |  |  |         storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-07 14:50:14 -07:00
										 |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |         UPDATE messages | 
					
						
							|  |  |  |         SET | 
					
						
							| 
									
										
										
										
											2021-08-12 13:15:55 -05:00
										 |  |  |           readStatus = ${ReadStatus.Read}, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |           seenStatus = ${SeenStatus.Seen}, | 
					
						
							| 
									
										
										
										
											2021-05-07 14:50:14 -07:00
										 |  |  |           json = json_patch(json, $jsonPatch) | 
					
						
							|  |  |  |         WHERE | 
					
						
							|  |  |  |           conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |           seenStatus = ${SeenStatus.Unseen} AND | 
					
						
							|  |  |  |           isStory = 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |           (${_storyIdPredicate(storyId, includeStoryReplies)}) AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |           received_at <= $newestUnreadAt; | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2021-05-07 14:50:14 -07:00
										 |  |  |     ).run({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |       jsonPatch: JSON.stringify({ | 
					
						
							|  |  |  |         readStatus: ReadStatus.Read, | 
					
						
							|  |  |  |         seenStatus: SeenStatus.Seen, | 
					
						
							|  |  |  |       }), | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       newestUnreadAt, | 
					
						
							|  |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-05-07 14:50:14 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return rows.map(row => { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |       const json = jsonToObject<MessageType>(row.json); | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       return { | 
					
						
							| 
									
										
										
										
											2022-04-29 16:42:47 -07:00
										 |  |  |         originalReadStatus: json.readStatus, | 
					
						
							| 
									
										
										
										
											2021-08-12 13:15:55 -05:00
										 |  |  |         readStatus: ReadStatus.Read, | 
					
						
							| 
									
										
										
										
											2022-04-29 16:42:47 -07:00
										 |  |  |         seenStatus: SeenStatus.Seen, | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |         ...pick(json, [ | 
					
						
							|  |  |  |           'expirationStartTimestamp', | 
					
						
							|  |  |  |           'id', | 
					
						
							|  |  |  |           'sent_at', | 
					
						
							|  |  |  |           'source', | 
					
						
							|  |  |  |           'sourceUuid', | 
					
						
							|  |  |  |           'type', | 
					
						
							|  |  |  |         ]), | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       }; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | async function getUnreadReactionsAndMarkRead({ | 
					
						
							|  |  |  |   conversationId, | 
					
						
							|  |  |  |   newestUnreadAt, | 
					
						
							|  |  |  |   storyId, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							|  |  |  |   newestUnreadAt: number; | 
					
						
							|  |  |  |   storyId?: UUIDStringType; | 
					
						
							|  |  |  | }): Promise<Array<ReactionResultType>> { | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |   return db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     const unreadMessages: Array<ReactionResultType> = db | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2021-12-21 21:01:09 +01:00
										 |  |  |         SELECT reactions.rowid, targetAuthorUuid, targetTimestamp, messageId | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         FROM reactions | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         INDEXED BY reactions_unread | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         JOIN messages on messages.id IS reactions.messageId | 
					
						
							|  |  |  |         WHERE | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |           reactions.conversationId IS $conversationId AND | 
					
						
							|  |  |  |           reactions.unread > 0 AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |           messages.received_at <= $newestUnreadAt AND | 
					
						
							|  |  |  |           messages.storyId IS $storyId | 
					
						
							|  |  |  |         ORDER BY messageReceivedAt DESC; | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .all({ | 
					
						
							|  |  |  |         conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         newestUnreadAt, | 
					
						
							|  |  |  |         storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     const idsToUpdate = unreadMessages.map(item => item.rowid); | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  |     batchMultiVarQuery(db, idsToUpdate, (ids: ReadonlyArray<number>): void => { | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       db.prepare<ArrayQuery>( | 
					
						
							|  |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         UPDATE reactions | 
					
						
							|  |  |  |         SET unread = 0 | 
					
						
							|  |  |  |         WHERE rowid IN ( ${ids.map(() => '?').join(', ')} ); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         `
 | 
					
						
							| 
									
										
										
										
											2021-12-20 16:15:36 +01:00
										 |  |  |       ).run(ids); | 
					
						
							| 
									
										
										
										
											2021-05-17 12:52:09 -04:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return unreadMessages; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function markReactionAsRead( | 
					
						
							|  |  |  |   targetAuthorUuid: string, | 
					
						
							|  |  |  |   targetTimestamp: number | 
					
						
							|  |  |  | ): Promise<ReactionType | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     const readReaction = db | 
					
						
							|  |  |  |       .prepare( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |           SELECT * | 
					
						
							|  |  |  |           FROM reactions | 
					
						
							|  |  |  |           WHERE | 
					
						
							|  |  |  |             targetAuthorUuid = $targetAuthorUuid AND | 
					
						
							|  |  |  |             targetTimestamp = $targetTimestamp AND | 
					
						
							|  |  |  |             unread = 1 | 
					
						
							|  |  |  |           ORDER BY rowId DESC | 
					
						
							|  |  |  |           LIMIT 1; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .get({ | 
					
						
							|  |  |  |         targetAuthorUuid, | 
					
						
							|  |  |  |         targetTimestamp, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     db.prepare( | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |         UPDATE reactions SET | 
					
						
							|  |  |  |         unread = 0 WHERE | 
					
						
							| 
									
										
										
										
											2021-05-17 12:52:09 -04:00
										 |  |  |         targetAuthorUuid = $targetAuthorUuid AND | 
					
						
							|  |  |  |         targetTimestamp = $targetTimestamp; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |     ).run({ | 
					
						
							|  |  |  |       targetAuthorUuid, | 
					
						
							|  |  |  |       targetTimestamp, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return readReaction; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function addReaction({ | 
					
						
							|  |  |  |   conversationId, | 
					
						
							|  |  |  |   emoji, | 
					
						
							|  |  |  |   fromId, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |   messageId, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |   messageReceivedAt, | 
					
						
							|  |  |  |   targetAuthorUuid, | 
					
						
							|  |  |  |   targetTimestamp, | 
					
						
							|  |  |  | }: ReactionType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   await db | 
					
						
							|  |  |  |     .prepare( | 
					
						
							|  |  |  |       `INSERT INTO reactions (
 | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       emoji, | 
					
						
							|  |  |  |       fromId, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |       messageId, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       messageReceivedAt, | 
					
						
							|  |  |  |       targetAuthorUuid, | 
					
						
							|  |  |  |       targetTimestamp, | 
					
						
							|  |  |  |       unread | 
					
						
							|  |  |  |     ) VALUES ( | 
					
						
							|  |  |  |       $conversationId, | 
					
						
							|  |  |  |       $emoji, | 
					
						
							|  |  |  |       $fromId, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |       $messageId, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       $messageReceivedAt, | 
					
						
							|  |  |  |       $targetAuthorUuid, | 
					
						
							|  |  |  |       $targetTimestamp, | 
					
						
							|  |  |  |       $unread | 
					
						
							|  |  |  |     );`
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |     .run({ | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       emoji, | 
					
						
							|  |  |  |       fromId, | 
					
						
							| 
									
										
										
										
											2021-07-15 16:48:09 -07:00
										 |  |  |       messageId, | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  |       messageReceivedAt, | 
					
						
							|  |  |  |       targetAuthorUuid, | 
					
						
							|  |  |  |       targetTimestamp, | 
					
						
							|  |  |  |       unread: 1, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-06 18:15:25 -07:00
										 |  |  | async function removeReactionFromConversation({ | 
					
						
							|  |  |  |   emoji, | 
					
						
							|  |  |  |   fromId, | 
					
						
							|  |  |  |   targetAuthorUuid, | 
					
						
							|  |  |  |   targetTimestamp, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   emoji: string; | 
					
						
							|  |  |  |   fromId: string; | 
					
						
							|  |  |  |   targetAuthorUuid: string; | 
					
						
							|  |  |  |   targetTimestamp: number; | 
					
						
							|  |  |  | }): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   await db | 
					
						
							|  |  |  |     .prepare( | 
					
						
							|  |  |  |       `DELETE FROM reactions WHERE
 | 
					
						
							|  |  |  |       emoji = $emoji AND | 
					
						
							|  |  |  |       fromId = $fromId AND | 
					
						
							|  |  |  |       targetAuthorUuid = $targetAuthorUuid AND | 
					
						
							|  |  |  |       targetTimestamp = $targetTimestamp;`
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .run({ | 
					
						
							|  |  |  |       emoji, | 
					
						
							|  |  |  |       fromId, | 
					
						
							|  |  |  |       targetAuthorUuid, | 
					
						
							|  |  |  |       targetTimestamp, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | async function _getAllReactions(): Promise<Array<ReactionType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db.prepare<EmptyQuery>('SELECT * from reactions;').all(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | async function _removeAllReactions(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<EmptyQuery>('DELETE from reactions;').run(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-15 15:54:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | async function getOlderMessagesByConversation( | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |   options: { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |     limit?: number; | 
					
						
							|  |  |  |     messageId?: string; | 
					
						
							|  |  |  |     receivedAt?: number; | 
					
						
							|  |  |  |     sentAt?: number; | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |     storyId: string | undefined; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | ): Promise<Array<MessageTypeUnhydrated>> { | 
					
						
							|  |  |  |   return getOlderMessagesByConversationSync(conversationId, options); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | function getOlderMessagesByConversationSync( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   conversationId: string, | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |     limit = 100, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     messageId, | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |     receivedAt = Number.MAX_VALUE, | 
					
						
							| 
									
										
										
										
											2021-01-13 08:32:18 -08:00
										 |  |  |     sentAt = Number.MAX_VALUE, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     storyId, | 
					
						
							| 
									
										
										
										
											2020-09-24 13:57:54 -07:00
										 |  |  |   }: { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2020-09-24 13:57:54 -07:00
										 |  |  |     limit?: number; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     messageId?: string; | 
					
						
							| 
									
										
										
										
											2020-09-24 13:57:54 -07:00
										 |  |  |     receivedAt?: number; | 
					
						
							| 
									
										
										
										
											2021-01-13 08:32:18 -08:00
										 |  |  |     sentAt?: number; | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |     storyId: string | undefined; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | ): Array<MessageTypeUnhydrated> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:06:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         ($messageId IS NULL OR id IS NOT $messageId) AND | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) AND | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         ( | 
					
						
							|  |  |  |           (received_at = $received_at AND sent_at < $sent_at) OR | 
					
						
							|  |  |  |           received_at < $received_at | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       limit, | 
					
						
							|  |  |  |       messageId: messageId || null, | 
					
						
							|  |  |  |       received_at: receivedAt, | 
					
						
							|  |  |  |       sent_at: sentAt, | 
					
						
							|  |  |  |       storyId: storyId || null, | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     .reverse(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-22 02:26:16 -04:00
										 |  |  | async function getAllStories({ | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   conversationId, | 
					
						
							|  |  |  |   sourceUuid, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   conversationId?: string; | 
					
						
							| 
									
										
										
										
											2022-07-08 13:46:25 -07:00
										 |  |  |   sourceUuid?: UUIDStringType; | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  | }): Promise<GetAllStoriesResultType> { | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   const rows: ReadonlyArray<{ | 
					
						
							|  |  |  |     json: string; | 
					
						
							|  |  |  |     hasReplies: number; | 
					
						
							|  |  |  |     hasRepliesFromSelf: number; | 
					
						
							|  |  |  |   }> = db | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |       SELECT | 
					
						
							|  |  |  |         json, | 
					
						
							|  |  |  |         (SELECT EXISTS( | 
					
						
							|  |  |  |           SELECT 1 | 
					
						
							|  |  |  |           FROM messages as replies | 
					
						
							|  |  |  |           WHERE replies.storyId IS messages.id | 
					
						
							|  |  |  |         )) as hasReplies, | 
					
						
							|  |  |  |         (SELECT EXISTS( | 
					
						
							|  |  |  |           SELECT 1 | 
					
						
							|  |  |  |           FROM messages AS selfReplies | 
					
						
							|  |  |  |           WHERE | 
					
						
							|  |  |  |             selfReplies.storyId IS messages.id AND | 
					
						
							|  |  |  |             selfReplies.type IS 'outgoing' | 
					
						
							|  |  |  |         )) as hasRepliesFromSelf | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       FROM messages | 
					
						
							|  |  |  |       WHERE | 
					
						
							|  |  |  |         type IS 'story' AND | 
					
						
							|  |  |  |         ($conversationId IS NULL OR conversationId IS $conversationId) AND | 
					
						
							| 
									
										
										
										
											2022-10-22 02:26:16 -04:00
										 |  |  |         ($sourceUuid IS NULL OR sourceUuid IS $sourceUuid) | 
					
						
							|  |  |  |       ORDER BY received_at ASC, sent_at ASC; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       conversationId: conversationId || null, | 
					
						
							|  |  |  |       sourceUuid: sourceUuid || null, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   return rows.map(row => ({ | 
					
						
							|  |  |  |     ...jsonToObject(row.json), | 
					
						
							|  |  |  |     hasReplies: row.hasReplies !== 0, | 
					
						
							|  |  |  |     hasRepliesFromSelf: row.hasRepliesFromSelf !== 0, | 
					
						
							|  |  |  |   })); | 
					
						
							| 
									
										
										
										
											2022-10-22 02:26:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | async function getNewerMessagesByConversation( | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |   options: { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |     limit?: number; | 
					
						
							|  |  |  |     receivedAt?: number; | 
					
						
							|  |  |  |     sentAt?: number; | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |     storyId: UUIDStringType | undefined; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | ): Promise<Array<MessageTypeUnhydrated>> { | 
					
						
							|  |  |  |   return getNewerMessagesByConversationSync(conversationId, options); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | function getNewerMessagesByConversationSync( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2021-01-13 08:32:18 -08:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-01-13 08:32:18 -08:00
										 |  |  |     limit = 100, | 
					
						
							|  |  |  |     receivedAt = 0, | 
					
						
							|  |  |  |     sentAt = 0, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     storyId, | 
					
						
							|  |  |  |   }: { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     limit?: number; | 
					
						
							|  |  |  |     receivedAt?: number; | 
					
						
							|  |  |  |     sentAt?: number; | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |     storyId: UUIDStringType | undefined; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | ): Array<MessageTypeUnhydrated> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) AND | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         ( | 
					
						
							|  |  |  |           (received_at = $received_at AND sent_at > $sent_at) OR | 
					
						
							|  |  |  |           received_at > $received_at | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |       ORDER BY received_at ASC, sent_at ASC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       limit, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       received_at: receivedAt, | 
					
						
							|  |  |  |       sent_at: sentAt, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-07-09 16:38:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | function getOldestMessageForConversation( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   { | 
					
						
							|  |  |  |     storyId, | 
					
						
							|  |  |  |     includeStoryReplies, | 
					
						
							|  |  |  |   }: { | 
					
						
							|  |  |  |     storyId?: UUIDStringType; | 
					
						
							|  |  |  |     includeStoryReplies: boolean; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): MessageMetricsType | undefined { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |       SELECT received_at, sent_at, id FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       ORDER BY received_at ASC, sent_at ASC | 
					
						
							|  |  |  |       LIMIT 1; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return row; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | function getNewestMessageForConversation( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   { | 
					
						
							|  |  |  |     storyId, | 
					
						
							|  |  |  |     includeStoryReplies, | 
					
						
							|  |  |  |   }: { | 
					
						
							|  |  |  |     storyId?: UUIDStringType; | 
					
						
							|  |  |  |     includeStoryReplies: boolean; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): MessageMetricsType | undefined { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |       SELECT received_at, sent_at, id FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT 1; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return row; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | function getLastConversationActivity({ | 
					
						
							| 
									
										
										
										
											2021-01-20 09:31:44 -08:00
										 |  |  |   conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   ourUuid, | 
					
						
							| 
									
										
										
										
											2021-01-20 09:31:44 -08:00
										 |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   ourUuid: UUIDStringType; | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | }): MessageType | undefined { | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const row = prepare( | 
					
						
							| 
									
										
										
										
											2021-04-08 12:16:10 -07:00
										 |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |       INDEXED BY messages_activity | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       WHERE | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         conversationId IS $conversationId AND | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |         shouldAffectActivity IS 1 AND | 
					
						
							|  |  |  |         isTimerChangeFromSync IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         ${includeStoryReplies ? '' : 'storyId IS NULL AND'} | 
					
						
							| 
									
										
										
										
											2021-12-20 13:04:02 -08:00
										 |  |  |         isGroupLeaveEventFromOther IS 0 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT 1; | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-04-08 12:16:10 -07:00
										 |  |  |   ).get({ | 
					
						
							|  |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     ourUuid, | 
					
						
							| 
									
										
										
										
											2021-04-08 12:16:10 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return jsonToObject(row.json); | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | function getLastConversationPreview({ | 
					
						
							| 
									
										
										
										
											2021-01-20 09:31:44 -08:00
										 |  |  |   conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-01-20 09:31:44 -08:00
										 |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | }): MessageType | undefined { | 
					
						
							| 
									
										
										
										
											2023-01-17 16:44:22 -08:00
										 |  |  |   type Row = Readonly<{ | 
					
						
							|  |  |  |     json: string; | 
					
						
							|  |  |  |   }>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2023-01-17 16:44:22 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |   const index = includeStoryReplies | 
					
						
							|  |  |  |     ? 'messages_preview' | 
					
						
							|  |  |  |     : 'messages_preview_without_story'; | 
					
						
							| 
									
										
										
										
											2023-01-17 16:44:22 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const row: Row | undefined = prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |       SELECT json FROM ( | 
					
						
							|  |  |  |         SELECT json, expiresAt FROM messages | 
					
						
							|  |  |  |         INDEXED BY ${index} | 
					
						
							|  |  |  |         WHERE | 
					
						
							|  |  |  |           conversationId IS $conversationId AND | 
					
						
							|  |  |  |           shouldAffectPreview IS 1 AND | 
					
						
							|  |  |  |           isGroupLeaveEventFromOther IS 0 | 
					
						
							|  |  |  |           ${includeStoryReplies ? '' : 'AND storyId IS NULL'} | 
					
						
							|  |  |  |         ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       WHERE likely(expiresAt > $now) | 
					
						
							|  |  |  |       LIMIT 1 | 
					
						
							| 
									
										
										
										
											2023-01-17 16:44:22 -08:00
										 |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2021-04-08 12:16:10 -07:00
										 |  |  |   ).get({ | 
					
						
							|  |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2021-09-20 12:57:59 -05:00
										 |  |  |     now: Date.now(), | 
					
						
							| 
									
										
										
										
											2021-04-08 12:16:10 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-17 16:44:22 -08:00
										 |  |  |   return row ? jsonToObject(row.json) : undefined; | 
					
						
							| 
									
										
										
										
											2020-08-06 17:50:54 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  | async function getConversationMessageStats({ | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  |   conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   ourUuid, | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   ourUuid: UUIDStringType; | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  | }): Promise<ConversationMessageStatsType> { | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       activity: getLastConversationActivity({ | 
					
						
							|  |  |  |         conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |         ourUuid, | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  |       }), | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |       preview: getLastConversationPreview({ | 
					
						
							|  |  |  |         conversationId, | 
					
						
							|  |  |  |         includeStoryReplies, | 
					
						
							|  |  |  |       }), | 
					
						
							| 
									
										
										
										
											2021-08-16 09:56:27 -07:00
										 |  |  |       hasUserInitiatedMessages: hasUserInitiatedMessages(conversationId), | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  | async function getLastConversationMessage({ | 
					
						
							|  |  |  |   conversationId, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							|  |  |  | }): Promise<MessageType | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |       SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2022-03-15 17:11:28 -07:00
										 |  |  |         conversationId = $conversationId | 
					
						
							|  |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT 1; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return jsonToObject(row.json); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  | function getOldestUnseenMessageForConversation( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   { | 
					
						
							|  |  |  |     storyId, | 
					
						
							|  |  |  |     includeStoryReplies, | 
					
						
							|  |  |  |   }: { | 
					
						
							|  |  |  |     storyId?: UUIDStringType; | 
					
						
							|  |  |  |     includeStoryReplies: boolean; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): MessageMetricsType | undefined { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |       SELECT received_at, sent_at, id FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |         seenStatus = ${SeenStatus.Unseen} AND | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       ORDER BY received_at ASC, sent_at ASC | 
					
						
							|  |  |  |       LIMIT 1; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return row; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | async function getTotalUnreadForConversation( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-04-27 10:41:24 -07:00
										 |  |  |   options: { | 
					
						
							|  |  |  |     storyId: UUIDStringType | undefined; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2022-04-27 10:41:24 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | ): Promise<number> { | 
					
						
							| 
									
										
										
										
											2022-04-27 10:41:24 -07:00
										 |  |  |   return getTotalUnreadForConversationSync(conversationId, options); | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | } | 
					
						
							|  |  |  | function getTotalUnreadForConversationSync( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-04-27 10:41:24 -07:00
										 |  |  |   { | 
					
						
							|  |  |  |     storyId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2022-04-27 10:41:24 -07:00
										 |  |  |   }: { | 
					
						
							|  |  |  |     storyId: UUIDStringType | undefined; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2022-04-27 10:41:24 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | ): number { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |       SELECT count(1) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       FROM messages | 
					
						
							|  |  |  |       WHERE | 
					
						
							|  |  |  |         conversationId = $conversationId AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         readStatus = ${ReadStatus.Unread} AND | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |     .pluck() | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       storyId: storyId || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |   return row; | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  | function getTotalUnseenForConversationSync( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   { | 
					
						
							|  |  |  |     storyId, | 
					
						
							|  |  |  |     includeStoryReplies, | 
					
						
							|  |  |  |   }: { | 
					
						
							|  |  |  |     storyId?: UUIDStringType; | 
					
						
							|  |  |  |     includeStoryReplies: boolean; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  | ): number { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |       SELECT count(1) | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |       FROM messages | 
					
						
							|  |  |  |       WHERE | 
					
						
							|  |  |  |         conversationId = $conversationId AND | 
					
						
							|  |  |  |         seenStatus = ${SeenStatus.Unseen} AND | 
					
						
							|  |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         (${_storyIdPredicate(storyId, includeStoryReplies)}) | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |     .pluck() | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       storyId: storyId || null, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |   return row; | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getMessageMetricsForConversation( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   options: { | 
					
						
							|  |  |  |     storyId?: UUIDStringType; | 
					
						
							|  |  |  |     includeStoryReplies: boolean; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<ConversationMetricsType> { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   return getMessageMetricsForConversationSync(conversationId, options); | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | } | 
					
						
							|  |  |  | function getMessageMetricsForConversationSync( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   options: { | 
					
						
							|  |  |  |     storyId?: UUIDStringType; | 
					
						
							|  |  |  |     includeStoryReplies: boolean; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | ): ConversationMetricsType { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   const oldest = getOldestMessageForConversation(conversationId, options); | 
					
						
							|  |  |  |   const newest = getNewestMessageForConversation(conversationId, options); | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |   const oldestUnseen = getOldestUnseenMessageForConversation( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     options | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |   const totalUnseen = getTotalUnseenForConversationSync( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |     options | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   ); | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     oldest: oldest ? pick(oldest, ['received_at', 'sent_at', 'id']) : undefined, | 
					
						
							|  |  |  |     newest: newest ? pick(newest, ['received_at', 'sent_at', 'id']) : undefined, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |     oldestUnseen: oldestUnseen | 
					
						
							|  |  |  |       ? pick(oldestUnseen, ['received_at', 'sent_at', 'id']) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       : undefined, | 
					
						
							| 
									
										
										
										
											2022-04-22 11:35:14 -07:00
										 |  |  |     totalUnseen, | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  | async function getConversationRangeCenteredOnMessage({ | 
					
						
							|  |  |  |   conversationId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   limit, | 
					
						
							|  |  |  |   messageId, | 
					
						
							|  |  |  |   receivedAt, | 
					
						
							|  |  |  |   sentAt, | 
					
						
							|  |  |  |   storyId, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   conversationId: string; | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |   includeStoryReplies: boolean; | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   limit?: number; | 
					
						
							|  |  |  |   messageId: string; | 
					
						
							|  |  |  |   receivedAt: number; | 
					
						
							|  |  |  |   sentAt?: number; | 
					
						
							| 
									
										
										
										
											2022-05-11 15:20:47 -07:00
										 |  |  |   storyId: UUIDStringType | undefined; | 
					
						
							| 
									
										
										
										
											2022-07-28 09:35:29 -07:00
										 |  |  | }): Promise< | 
					
						
							|  |  |  |   GetConversationRangeCenteredOnMessageResultType<MessageTypeUnhydrated> | 
					
						
							|  |  |  | > { | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       older: getOlderMessagesByConversationSync(conversationId, { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |         limit, | 
					
						
							|  |  |  |         messageId, | 
					
						
							|  |  |  |         receivedAt, | 
					
						
							|  |  |  |         sentAt, | 
					
						
							|  |  |  |         storyId, | 
					
						
							|  |  |  |       }), | 
					
						
							|  |  |  |       newer: getNewerMessagesByConversationSync(conversationId, { | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         includeStoryReplies, | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |         limit, | 
					
						
							|  |  |  |         receivedAt, | 
					
						
							|  |  |  |         sentAt, | 
					
						
							|  |  |  |         storyId, | 
					
						
							|  |  |  |       }), | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |       metrics: getMessageMetricsForConversationSync(conversationId, { | 
					
						
							| 
									
										
										
										
											2022-05-11 19:47:19 -04:00
										 |  |  |         storyId, | 
					
						
							| 
									
										
										
										
											2022-09-29 20:57:11 -04:00
										 |  |  |         includeStoryReplies, | 
					
						
							|  |  |  |       }), | 
					
						
							| 
									
										
										
										
											2021-12-20 13:05:13 -08:00
										 |  |  |     }; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 16:52:01 -08:00
										 |  |  | async function getCallHistoryMessageByCallId( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							|  |  |  |   callId: string | 
					
						
							|  |  |  | ): Promise<string | void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const id: string | void = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT id | 
					
						
							|  |  |  |       FROM messages | 
					
						
							|  |  |  |       WHERE conversationId = $conversationId | 
					
						
							|  |  |  |         AND type = 'call-history' | 
					
						
							|  |  |  |         AND callMode = 'Direct' | 
					
						
							|  |  |  |         AND callId = $callId | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck() | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       callId, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 14:43:19 -06:00
										 |  |  | async function hasGroupCallHistoryMessage( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							|  |  |  |   eraId: string | 
					
						
							|  |  |  | ): Promise<boolean> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   const exists: number = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |       SELECT EXISTS( | 
					
						
							|  |  |  |         SELECT 1 FROM messages | 
					
						
							|  |  |  |         WHERE conversationId = $conversationId | 
					
						
							|  |  |  |         AND type = 'call-history' | 
					
						
							|  |  |  |         AND json_extract(json, '$.callHistoryDetails.callMode') = 'Group' | 
					
						
							|  |  |  |         AND json_extract(json, '$.callHistoryDetails.eraId') = $eraId | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |     .pluck() | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .get({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       eraId, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-12-07 14:43:19 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-28 09:19:48 -08:00
										 |  |  |   return exists !== 0; | 
					
						
							| 
									
										
										
										
											2020-12-07 14:43:19 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-10 11:28:49 -07:00
										 |  |  | async function migrateConversationMessages( | 
					
						
							|  |  |  |   obsoleteId: string, | 
					
						
							|  |  |  |   currentId: string | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-07-10 11:28:49 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE messages SET | 
					
						
							| 
									
										
										
										
											2020-07-10 11:28:49 -07:00
										 |  |  |       conversationId = $currentId, | 
					
						
							|  |  |  |       json = json_set(json, '$.conversationId', $currentId) | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     WHERE conversationId = $obsoleteId; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     obsoleteId, | 
					
						
							|  |  |  |     currentId, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-07-10 11:28:49 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getMessagesBySentAt( | 
					
						
							|  |  |  |   sentAt: number | 
					
						
							|  |  |  | ): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       WHERE sent_at = $sent_at | 
					
						
							|  |  |  |       ORDER BY received_at DESC, sent_at DESC; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       sent_at: sentAt, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getExpiredMessages(): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   const now = Date.now(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |         expiresAt <= $now | 
					
						
							|  |  |  |       ORDER BY expiresAt ASC; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |     .all({ now }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 14:12:04 -05:00
										 |  |  | async function getMessagesUnexpectedlyMissingExpirationStartTimestamp(): Promise< | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |   Array<MessageType> | 
					
						
							|  |  |  | > { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages | 
					
						
							|  |  |  |       INDEXED BY messages_unexpectedly_missing_expiration_start_timestamp | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       WHERE | 
					
						
							|  |  |  |         expireTimer > 0 AND | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |         expirationStartTimestamp IS NULL AND | 
					
						
							| 
									
										
										
										
											2021-06-18 14:12:04 -05:00
										 |  |  |         ( | 
					
						
							|  |  |  |           type IS 'outgoing' OR | 
					
						
							|  |  |  |           (type IS 'incoming' AND ( | 
					
						
							| 
									
										
										
										
											2021-08-12 13:15:55 -05:00
										 |  |  |             readStatus = ${ReadStatus.Read} OR | 
					
						
							|  |  |  |             readStatus = ${ReadStatus.Viewed} OR | 
					
						
							|  |  |  |             readStatus IS NULL | 
					
						
							| 
									
										
										
										
											2021-06-18 14:12:04 -05:00
										 |  |  |           )) | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2018-08-07 12:33:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-08-07 12:33:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  | async function getSoonestMessageExpiry(): Promise<undefined | number> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |   // Note: we use `pluck` to only get the first column.
 | 
					
						
							|  |  |  |   const result: null | number = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |       SELECT MIN(expiresAt) | 
					
						
							|  |  |  |       FROM messages; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |     .pluck(true) | 
					
						
							|  |  |  |     .get(); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-27 09:47:24 -08:00
										 |  |  |   if (result != null && result >= Number.MAX_SAFE_INTEGER) { | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-16 17:20:17 -05:00
										 |  |  |   return result || undefined; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 11:17:51 -05:00
										 |  |  | async function getNextTapToViewMessageTimestampToAgeOut(): Promise< | 
					
						
							|  |  |  |   undefined | number | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | > { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-05-19 11:17:51 -05:00
										 |  |  |   const row = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT json FROM messages | 
					
						
							|  |  |  |       WHERE | 
					
						
							|  |  |  |         isViewOnce = 1 | 
					
						
							|  |  |  |         AND (isErased IS NULL OR isErased != 1) | 
					
						
							|  |  |  |       ORDER BY received_at ASC, sent_at ASC | 
					
						
							|  |  |  |       LIMIT 1; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2021-05-19 11:17:51 -05:00
										 |  |  |     .get(); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 11:17:51 -05:00
										 |  |  |   if (!row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     return undefined; | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const data = jsonToObject<MessageType>(row.json); | 
					
						
							| 
									
										
										
										
											2021-05-19 11:17:51 -05:00
										 |  |  |   const result = data.received_at_ms || data.received_at; | 
					
						
							|  |  |  |   return isNormalNumber(result) ? result : undefined; | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getTapToViewMessagesNeedingErase(): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  |   const THIRTY_DAYS_AGO = Date.now() - 30 * 24 * 60 * 60 * 1000; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json | 
					
						
							|  |  |  |       FROM messages | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       WHERE | 
					
						
							|  |  |  |         isViewOnce = 1 | 
					
						
							|  |  |  |         AND (isErased IS NULL OR isErased != 1) | 
					
						
							|  |  |  |         AND received_at <= $THIRTY_DAYS_AGO | 
					
						
							|  |  |  |       ORDER BY received_at ASC, sent_at ASC; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       THIRTY_DAYS_AGO, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:33:13 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-17 16:11:24 -07:00
										 |  |  | const MAX_UNPROCESSED_ATTEMPTS = 3; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  | function saveUnprocessedSync(data: UnprocessedType): string { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |   const { | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     timestamp, | 
					
						
							| 
									
										
										
										
											2022-03-24 14:28:56 -07:00
										 |  |  |     receivedAtCounter, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     version, | 
					
						
							|  |  |  |     attempts, | 
					
						
							|  |  |  |     envelope, | 
					
						
							|  |  |  |     source, | 
					
						
							|  |  |  |     sourceUuid, | 
					
						
							|  |  |  |     sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |     serverGuid, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     serverTimestamp, | 
					
						
							|  |  |  |     decrypted, | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |     urgent, | 
					
						
							| 
									
										
										
										
											2022-10-07 10:02:08 -07:00
										 |  |  |     story, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |   } = data; | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  |   if (!id) { | 
					
						
							| 
									
										
										
										
											2021-05-27 11:45:45 -04:00
										 |  |  |     throw new Error('saveUnprocessedSync: id was falsey'); | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-21 16:06:50 -08:00
										 |  |  |   if (attempts > MAX_UNPROCESSED_ATTEMPTS) { | 
					
						
							|  |  |  |     logger.warn( | 
					
						
							|  |  |  |       `saveUnprocessedSync: not saving ${id} due to exhausted attempts` | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-09-17 16:11:24 -07:00
										 |  |  |     removeUnprocessedSync(id); | 
					
						
							|  |  |  |     return id; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO unprocessed ( | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       id, | 
					
						
							|  |  |  |       timestamp, | 
					
						
							| 
									
										
										
										
											2022-03-24 14:28:56 -07:00
										 |  |  |       receivedAtCounter, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       version, | 
					
						
							|  |  |  |       attempts, | 
					
						
							|  |  |  |       envelope, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |       source, | 
					
						
							|  |  |  |       sourceUuid, | 
					
						
							|  |  |  |       sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |       serverGuid, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |       serverTimestamp, | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |       decrypted, | 
					
						
							| 
									
										
										
										
											2022-10-07 10:02:08 -07:00
										 |  |  |       urgent, | 
					
						
							|  |  |  |       story | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     ) values ( | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $timestamp, | 
					
						
							| 
									
										
										
										
											2022-03-24 14:28:56 -07:00
										 |  |  |       $receivedAtCounter, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |       $version, | 
					
						
							|  |  |  |       $attempts, | 
					
						
							|  |  |  |       $envelope, | 
					
						
							|  |  |  |       $source, | 
					
						
							|  |  |  |       $sourceUuid, | 
					
						
							|  |  |  |       $sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |       $serverGuid, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |       $serverTimestamp, | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |       $decrypted, | 
					
						
							| 
									
										
										
										
											2022-10-07 10:02:08 -07:00
										 |  |  |       $urgent, | 
					
						
							|  |  |  |       $story | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     timestamp, | 
					
						
							| 
									
										
										
										
											2022-03-24 14:28:56 -07:00
										 |  |  |     receivedAtCounter: receivedAtCounter ?? null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     version, | 
					
						
							|  |  |  |     attempts, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     envelope: envelope || null, | 
					
						
							|  |  |  |     source: source || null, | 
					
						
							|  |  |  |     sourceUuid: sourceUuid || null, | 
					
						
							|  |  |  |     sourceDevice: sourceDevice || null, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |     serverGuid: serverGuid || null, | 
					
						
							| 
									
										
										
										
											2021-05-17 11:03:42 -07:00
										 |  |  |     serverTimestamp: serverTimestamp || null, | 
					
						
							|  |  |  |     decrypted: decrypted || null, | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |     urgent: urgent || !isBoolean(urgent) ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2022-10-07 10:02:08 -07:00
										 |  |  |     story: story ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | function updateUnprocessedWithDataSync( | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   id: string, | 
					
						
							| 
									
										
										
										
											2021-04-16 16:13:13 -07:00
										 |  |  |   data: UnprocessedUpdateType | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | ): void { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |   const { | 
					
						
							|  |  |  |     source, | 
					
						
							|  |  |  |     sourceUuid, | 
					
						
							|  |  |  |     sourceDevice, | 
					
						
							|  |  |  |     serverGuid, | 
					
						
							|  |  |  |     serverTimestamp, | 
					
						
							|  |  |  |     decrypted, | 
					
						
							|  |  |  |   } = data; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 16:13:21 -07:00
										 |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE unprocessed SET | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  |       source = $source, | 
					
						
							| 
									
										
										
										
											2021-02-04 11:43:10 -08:00
										 |  |  |       sourceUuid = $sourceUuid, | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  |       sourceDevice = $sourceDevice, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |       serverGuid = $serverGuid, | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  |       serverTimestamp = $serverTimestamp, | 
					
						
							|  |  |  |       decrypted = $decrypted | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     WHERE id = $id; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     source: source || null, | 
					
						
							|  |  |  |     sourceUuid: sourceUuid || null, | 
					
						
							|  |  |  |     sourceDevice: sourceDevice || null, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  |     serverGuid: serverGuid || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     serverTimestamp: serverTimestamp || null, | 
					
						
							|  |  |  |     decrypted: decrypted || null, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | async function updateUnprocessedWithData( | 
					
						
							|  |  |  |   id: string, | 
					
						
							|  |  |  |   data: UnprocessedUpdateType | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return updateUnprocessedWithDataSync(id, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function updateUnprocessedsWithData( | 
					
						
							| 
									
										
										
										
											2021-04-16 16:13:13 -07:00
										 |  |  |   arrayOfUnprocessed: Array<{ id: string; data: UnprocessedUpdateType }> | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     for (const { id, data } of arrayOfUnprocessed) { | 
					
						
							| 
									
										
										
										
											2021-05-14 10:52:47 -07:00
										 |  |  |       assertSync(updateUnprocessedWithDataSync(id, data)); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2019-09-26 12:56:31 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getUnprocessedById( | 
					
						
							|  |  |  |   id: string | 
					
						
							|  |  |  | ): Promise<UnprocessedType | undefined> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const row = db | 
					
						
							|  |  |  |     .prepare<Query>('SELECT * FROM unprocessed WHERE id = $id;') | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-02-04 17:23:50 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |   return { | 
					
						
							|  |  |  |     ...row, | 
					
						
							|  |  |  |     urgent: isNumber(row.urgent) ? Boolean(row.urgent) : true, | 
					
						
							| 
									
										
										
										
											2022-10-07 10:02:08 -07:00
										 |  |  |     story: Boolean(row.story), | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getUnprocessedCount(): Promise<number> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getCountFromTable(getInstance(), 'unprocessed'); | 
					
						
							| 
									
										
										
										
											2018-09-28 15:51:26 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  | async function getAllUnprocessedIds(): Promise<Array<string>> { | 
					
						
							|  |  |  |   log.info('getAllUnprocessedIds'); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-12-14 02:25:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |   return db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |     // cleanup first
 | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |     const { changes: deletedStaleCount } = db | 
					
						
							|  |  |  |       .prepare<Query>('DELETE FROM unprocessed WHERE timestamp < $monthAgo') | 
					
						
							|  |  |  |       .run({ | 
					
						
							|  |  |  |         monthAgo: Date.now() - durations.MONTH, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2021-12-14 02:25:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |     if (deletedStaleCount !== 0) { | 
					
						
							|  |  |  |       logger.warn( | 
					
						
							|  |  |  |         'getAllUnprocessedAndIncrementAttempts: ' + | 
					
						
							|  |  |  |           `deleting ${deletedStaleCount} old unprocessed envelopes` | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-14 02:25:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |     const { changes: deletedInvalidCount } = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |           DELETE FROM unprocessed | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |           WHERE attempts >= $MAX_UNPROCESSED_ATTEMPTS | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .run({ MAX_UNPROCESSED_ATTEMPTS }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (deletedInvalidCount !== 0) { | 
					
						
							|  |  |  |       logger.warn( | 
					
						
							|  |  |  |         'getAllUnprocessedAndIncrementAttempts: ' + | 
					
						
							|  |  |  |           `deleting ${deletedInvalidCount} invalid unprocessed envelopes` | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return db | 
					
						
							|  |  |  |       .prepare<EmptyQuery>( | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |         `
 | 
					
						
							|  |  |  |           SELECT id  | 
					
						
							|  |  |  |           FROM unprocessed | 
					
						
							|  |  |  |           ORDER BY receivedAtCounter ASC | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .pluck() | 
					
						
							|  |  |  |       .all(); | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function getUnprocessedByIdsAndIncrementAttempts( | 
					
						
							|  |  |  |   ids: ReadonlyArray<string> | 
					
						
							|  |  |  | ): Promise<Array<UnprocessedType>> { | 
					
						
							|  |  |  |   log.info('getUnprocessedByIdsAndIncrementAttempts', { totalIds: ids.length }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   batchMultiVarQuery(db, ids, batch => { | 
					
						
							|  |  |  |     return db | 
					
						
							|  |  |  |       .prepare<ArrayQuery>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |           UPDATE unprocessed | 
					
						
							|  |  |  |           SET attempts = attempts + 1 | 
					
						
							|  |  |  |           WHERE id IN (${batch.map(() => '?').join(', ')}) | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .run(batch); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return batchMultiVarQuery(db, ids, batch => { | 
					
						
							|  |  |  |     return db | 
					
						
							|  |  |  |       .prepare<ArrayQuery>( | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |         `
 | 
					
						
							|  |  |  |           SELECT * | 
					
						
							|  |  |  |           FROM unprocessed | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |           WHERE id IN (${batch.map(() => '?').join(', ')}) | 
					
						
							| 
									
										
										
										
											2022-06-10 09:09:21 -07:00
										 |  |  |           ORDER BY receivedAtCounter ASC; | 
					
						
							| 
									
										
										
										
											2022-04-28 15:28:30 -07:00
										 |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |       .all(batch) | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |       .map(row => ({ | 
					
						
							|  |  |  |         ...row, | 
					
						
							|  |  |  |         urgent: isNumber(row.urgent) ? Boolean(row.urgent) : true, | 
					
						
							| 
									
										
										
										
											2022-10-07 10:02:08 -07:00
										 |  |  |         story: Boolean(row.story), | 
					
						
							| 
									
										
										
										
											2022-07-05 15:20:30 -07:00
										 |  |  |       })); | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  | function removeUnprocessedsSync(ids: ReadonlyArray<string>): void { | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |   log.info('removeUnprocessedsSync', { totalIds: ids.length }); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-22 11:44:51 -07:00
										 |  |  |   db.prepare<ArrayQuery>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     DELETE FROM unprocessed | 
					
						
							|  |  |  |     WHERE id IN ( ${ids.map(() => '?').join(', ')} ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(ids); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-17 16:11:24 -07:00
										 |  |  | function removeUnprocessedSync(id: string | Array<string>): void { | 
					
						
							| 
									
										
										
										
											2023-02-02 12:39:07 -07:00
										 |  |  |   log.info('removeUnprocessedSync', { id }); | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-06-22 11:44:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   if (!Array.isArray(id)) { | 
					
						
							| 
									
										
										
										
											2021-04-06 11:15:17 -07:00
										 |  |  |     prepare(db, 'DELETE FROM unprocessed WHERE id = $id;').run({ id }); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-12 11:16:58 -07:00
										 |  |  |   // This can happen normally due to flushing of `cacheRemoveBatcher` in
 | 
					
						
							|  |  |  |   // MessageReceiver.
 | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   if (!id.length) { | 
					
						
							| 
									
										
										
										
											2022-04-12 11:16:58 -07:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   assertSync(batchMultiVarQuery(db, id, removeUnprocessedsSync)); | 
					
						
							| 
									
										
										
										
											2021-09-17 16:11:24 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function removeUnprocessed(id: string | Array<string>): Promise<void> { | 
					
						
							|  |  |  |   removeUnprocessedSync(id); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeAllUnprocessed(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<EmptyQuery>('DELETE FROM unprocessed;').run(); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | // Attachment Downloads
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | const ATTACHMENT_DOWNLOADS_TABLE = 'attachment_downloads'; | 
					
						
							| 
									
										
										
										
											2022-05-23 16:07:41 -07:00
										 |  |  | async function getAttachmentDownloadJobById( | 
					
						
							|  |  |  |   id: string | 
					
						
							|  |  |  | ): Promise<AttachmentDownloadJobType | undefined> { | 
					
						
							|  |  |  |   return getById(getInstance(), ATTACHMENT_DOWNLOADS_TABLE, id); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function getNextAttachmentDownloadJobs( | 
					
						
							|  |  |  |   limit?: number, | 
					
						
							|  |  |  |   options: { timestamp?: number } = {} | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<AttachmentDownloadJobType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const timestamp = | 
					
						
							|  |  |  |     options && options.timestamp ? options.timestamp : Date.now(); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 09:36:06 -07:00
										 |  |  |   const rows: Array<{ json: string; id: string }> = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-09-07 09:36:06 -07:00
										 |  |  |       SELECT id, json | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       FROM attachment_downloads | 
					
						
							| 
									
										
										
										
											2021-06-01 10:13:10 -07:00
										 |  |  |       WHERE pending = 0 AND timestamp <= $timestamp | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       ORDER BY timestamp DESC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       limit: limit || 3, | 
					
						
							|  |  |  |       timestamp, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 09:36:06 -07:00
										 |  |  |   const INNER_ERROR = 'jsonToObject error'; | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     return rows.map(row => { | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         return jsonToObject(row.json); | 
					
						
							|  |  |  |       } catch (error) { | 
					
						
							|  |  |  |         logger.error( | 
					
						
							|  |  |  |           `getNextAttachmentDownloadJobs: Error with job '${row.id}', deleting. ` + | 
					
						
							|  |  |  |             `JSON: '${row.json}' ` + | 
					
						
							|  |  |  |             `Error: ${Errors.toLogFormat(error)}` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         removeAttachmentDownloadJobSync(row.id); | 
					
						
							|  |  |  |         throw new Error(INNER_ERROR); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } catch (error) { | 
					
						
							|  |  |  |     if ('message' in error && error.message === INNER_ERROR) { | 
					
						
							|  |  |  |       return getNextAttachmentDownloadJobs(limit, { timestamp }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw error; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function saveAttachmentDownloadJob( | 
					
						
							|  |  |  |   job: AttachmentDownloadJobType | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  |   const { id, pending, timestamp } = job; | 
					
						
							|  |  |  |   if (!id) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'saveAttachmentDownloadJob: Provided job did not have a truthy id' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO attachment_downloads ( | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  |       id, | 
					
						
							|  |  |  |       pending, | 
					
						
							|  |  |  |       timestamp, | 
					
						
							|  |  |  |       json | 
					
						
							|  |  |  |     ) values ( | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $pending, | 
					
						
							|  |  |  |       $timestamp, | 
					
						
							|  |  |  |       $json | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     ) | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     pending, | 
					
						
							|  |  |  |     timestamp, | 
					
						
							|  |  |  |     json: objectToJSON(job), | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function setAttachmentDownloadJobPending( | 
					
						
							|  |  |  |   id: string, | 
					
						
							|  |  |  |   pending: boolean | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE attachment_downloads | 
					
						
							|  |  |  |     SET pending = $pending | 
					
						
							|  |  |  |     WHERE id = $id; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     pending: pending ? 1 : 0, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function resetAttachmentDownloadPending(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<EmptyQuery>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE attachment_downloads | 
					
						
							|  |  |  |     SET pending = 0 | 
					
						
							|  |  |  |     WHERE pending != 0; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-07 09:36:06 -07:00
										 |  |  | function removeAttachmentDownloadJobSync(id: string): void { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return removeById(getInstance(), ATTACHMENT_DOWNLOADS_TABLE, id); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-07 09:36:06 -07:00
										 |  |  | async function removeAttachmentDownloadJob(id: string): Promise<void> { | 
					
						
							|  |  |  |   return removeAttachmentDownloadJobSync(id); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  | async function removeAllAttachmentDownloadJobs(): Promise<void> { | 
					
						
							|  |  |  |   return removeAllFromTable(getInstance(), ATTACHMENT_DOWNLOADS_TABLE); | 
					
						
							| 
									
										
										
										
											2019-01-30 12:15:07 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | // Stickers
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function createOrUpdateStickerPack(pack: StickerPackType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   const { | 
					
						
							|  |  |  |     attemptedStatus, | 
					
						
							|  |  |  |     author, | 
					
						
							|  |  |  |     coverStickerId, | 
					
						
							|  |  |  |     createdAt, | 
					
						
							|  |  |  |     downloadAttempts, | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     installedAt, | 
					
						
							|  |  |  |     key, | 
					
						
							|  |  |  |     lastUsed, | 
					
						
							|  |  |  |     status, | 
					
						
							|  |  |  |     stickerCount, | 
					
						
							|  |  |  |     title, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |     storageID, | 
					
						
							|  |  |  |     storageVersion, | 
					
						
							|  |  |  |     storageUnknownFields, | 
					
						
							|  |  |  |     storageNeedsSync, | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   } = pack; | 
					
						
							|  |  |  |   if (!id) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'createOrUpdateStickerPack: Provided data did not have a truthy id' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   let { position } = pack; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Assign default position
 | 
					
						
							|  |  |  |   if (!isNumber(position)) { | 
					
						
							|  |  |  |     position = db | 
					
						
							|  |  |  |       .prepare<EmptyQuery>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT IFNULL(MAX(position) + 1, 0) | 
					
						
							|  |  |  |         FROM sticker_packs | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .pluck() | 
					
						
							|  |  |  |       .get(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const row = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT id | 
					
						
							|  |  |  |       FROM sticker_packs | 
					
						
							|  |  |  |       WHERE id = $id; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |     .get({ id }); | 
					
						
							| 
									
										
										
										
											2019-05-23 18:27:42 -07:00
										 |  |  |   const payload = { | 
					
						
							| 
									
										
										
										
											2021-07-09 12:36:10 -07:00
										 |  |  |     attemptedStatus: attemptedStatus ?? null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     author, | 
					
						
							|  |  |  |     coverStickerId, | 
					
						
							|  |  |  |     createdAt: createdAt || Date.now(), | 
					
						
							|  |  |  |     downloadAttempts: downloadAttempts || 1, | 
					
						
							|  |  |  |     id, | 
					
						
							| 
									
										
										
										
											2021-07-09 12:36:10 -07:00
										 |  |  |     installedAt: installedAt ?? null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     key, | 
					
						
							|  |  |  |     lastUsed: lastUsed || null, | 
					
						
							|  |  |  |     status, | 
					
						
							|  |  |  |     stickerCount, | 
					
						
							|  |  |  |     title, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |     position: position ?? 0, | 
					
						
							|  |  |  |     storageID: storageID ?? null, | 
					
						
							|  |  |  |     storageVersion: storageVersion ?? null, | 
					
						
							|  |  |  |     storageUnknownFields: storageUnknownFields ?? null, | 
					
						
							|  |  |  |     storageNeedsSync: storageNeedsSync ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2019-05-23 18:27:42 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |   if (row) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       UPDATE sticker_packs SET | 
					
						
							| 
									
										
										
										
											2019-05-23 18:27:42 -07:00
										 |  |  |         attemptedStatus = $attemptedStatus, | 
					
						
							|  |  |  |         author = $author, | 
					
						
							|  |  |  |         coverStickerId = $coverStickerId, | 
					
						
							|  |  |  |         createdAt = $createdAt, | 
					
						
							|  |  |  |         downloadAttempts = $downloadAttempts, | 
					
						
							|  |  |  |         installedAt = $installedAt, | 
					
						
							|  |  |  |         key = $key, | 
					
						
							|  |  |  |         lastUsed = $lastUsed, | 
					
						
							|  |  |  |         status = $status, | 
					
						
							|  |  |  |         stickerCount = $stickerCount, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |         title = $title, | 
					
						
							|  |  |  |         position = $position, | 
					
						
							|  |  |  |         storageID = $storageID, | 
					
						
							|  |  |  |         storageVersion = $storageVersion, | 
					
						
							|  |  |  |         storageUnknownFields = $storageUnknownFields, | 
					
						
							|  |  |  |         storageNeedsSync = $storageNeedsSync | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       WHERE id = $id; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ).run(payload); | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 18:27:42 -07:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT INTO sticker_packs ( | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       attemptedStatus, | 
					
						
							|  |  |  |       author, | 
					
						
							|  |  |  |       coverStickerId, | 
					
						
							|  |  |  |       createdAt, | 
					
						
							|  |  |  |       downloadAttempts, | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       installedAt, | 
					
						
							|  |  |  |       key, | 
					
						
							|  |  |  |       lastUsed, | 
					
						
							|  |  |  |       status, | 
					
						
							|  |  |  |       stickerCount, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |       title, | 
					
						
							|  |  |  |       position, | 
					
						
							|  |  |  |       storageID, | 
					
						
							|  |  |  |       storageVersion, | 
					
						
							|  |  |  |       storageUnknownFields, | 
					
						
							|  |  |  |       storageNeedsSync | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |     ) values ( | 
					
						
							|  |  |  |       $attemptedStatus, | 
					
						
							|  |  |  |       $author, | 
					
						
							|  |  |  |       $coverStickerId, | 
					
						
							|  |  |  |       $createdAt, | 
					
						
							|  |  |  |       $downloadAttempts, | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $installedAt, | 
					
						
							|  |  |  |       $key, | 
					
						
							|  |  |  |       $lastUsed, | 
					
						
							|  |  |  |       $status, | 
					
						
							|  |  |  |       $stickerCount, | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |       $title, | 
					
						
							|  |  |  |       $position, | 
					
						
							|  |  |  |       $storageID, | 
					
						
							|  |  |  |       $storageVersion, | 
					
						
							|  |  |  |       $storageUnknownFields, | 
					
						
							|  |  |  |       $storageNeedsSync | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     ) | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(payload); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  | function updateStickerPackStatusSync( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   id: string, | 
					
						
							|  |  |  |   status: StickerPackStatusType, | 
					
						
							|  |  |  |   options?: { timestamp: number } | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  | ): void { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const timestamp = options ? options.timestamp || Date.now() : Date.now(); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   const installedAt = status === 'installed' ? timestamp : null; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE sticker_packs | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |     SET status = $status, installedAt = $installedAt | 
					
						
							|  |  |  |     WHERE id = $id; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     status, | 
					
						
							|  |  |  |     installedAt, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  | async function updateStickerPackStatus( | 
					
						
							|  |  |  |   id: string, | 
					
						
							|  |  |  |   status: StickerPackStatusType, | 
					
						
							|  |  |  |   options?: { timestamp: number } | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return updateStickerPackStatusSync(id, status, options); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function updateStickerPackInfo({ | 
					
						
							|  |  |  |   id, | 
					
						
							|  |  |  |   storageID, | 
					
						
							|  |  |  |   storageVersion, | 
					
						
							|  |  |  |   storageUnknownFields, | 
					
						
							|  |  |  |   storageNeedsSync, | 
					
						
							|  |  |  |   uninstalledAt, | 
					
						
							|  |  |  | }: StickerPackInfoType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (uninstalledAt) { | 
					
						
							|  |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       UPDATE uninstalled_sticker_packs | 
					
						
							|  |  |  |       SET | 
					
						
							|  |  |  |         storageID = $storageID, | 
					
						
							|  |  |  |         storageVersion = $storageVersion, | 
					
						
							|  |  |  |         storageUnknownFields = $storageUnknownFields, | 
					
						
							|  |  |  |         storageNeedsSync = $storageNeedsSync | 
					
						
							|  |  |  |       WHERE id = $id; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ).run({ | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       storageID: storageID ?? null, | 
					
						
							|  |  |  |       storageVersion: storageVersion ?? null, | 
					
						
							|  |  |  |       storageUnknownFields: storageUnknownFields ?? null, | 
					
						
							|  |  |  |       storageNeedsSync: storageNeedsSync ? 1 : 0, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       UPDATE sticker_packs | 
					
						
							|  |  |  |       SET | 
					
						
							|  |  |  |         storageID = $storageID, | 
					
						
							|  |  |  |         storageVersion = $storageVersion, | 
					
						
							|  |  |  |         storageUnknownFields = $storageUnknownFields, | 
					
						
							|  |  |  |         storageNeedsSync = $storageNeedsSync | 
					
						
							|  |  |  |       WHERE id = $id; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ).run({ | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       storageID: storageID ?? null, | 
					
						
							|  |  |  |       storageVersion: storageVersion ?? null, | 
					
						
							|  |  |  |       storageUnknownFields: storageUnknownFields ?? null, | 
					
						
							|  |  |  |       storageNeedsSync: storageNeedsSync ? 1 : 0, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-27 14:39:45 -08:00
										 |  |  | async function clearAllErrorStickerPackAttempts(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<EmptyQuery>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE sticker_packs | 
					
						
							|  |  |  |     SET downloadAttempts = 0 | 
					
						
							|  |  |  |     WHERE status = 'error'; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(); | 
					
						
							| 
									
										
										
										
											2021-01-27 14:39:45 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function createOrUpdateSticker(sticker: StickerType): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-11-11 16:43:05 -06:00
										 |  |  |   const { emoji, height, id, isCoverOnly, lastUsed, packId, path, width } = | 
					
						
							|  |  |  |     sticker; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   if (!isNumber(id)) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'createOrUpdateSticker: Provided data did not have a numeric id' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!packId) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'createOrUpdateSticker: Provided data did not have a truthy id' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO stickers ( | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       emoji, | 
					
						
							|  |  |  |       height, | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       isCoverOnly, | 
					
						
							|  |  |  |       lastUsed, | 
					
						
							|  |  |  |       packId, | 
					
						
							|  |  |  |       path, | 
					
						
							|  |  |  |       width | 
					
						
							|  |  |  |     ) values ( | 
					
						
							|  |  |  |       $emoji, | 
					
						
							|  |  |  |       $height, | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $isCoverOnly, | 
					
						
							|  |  |  |       $lastUsed, | 
					
						
							|  |  |  |       $packId, | 
					
						
							|  |  |  |       $path, | 
					
						
							|  |  |  |       $width | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     ) | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							| 
									
										
										
										
											2021-07-09 12:36:10 -07:00
										 |  |  |     emoji: emoji ?? null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     height, | 
					
						
							|  |  |  |     id, | 
					
						
							| 
									
										
										
										
											2021-04-07 13:00:22 -07:00
										 |  |  |     isCoverOnly: isCoverOnly ? 1 : 0, | 
					
						
							|  |  |  |     lastUsed: lastUsed || null, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     packId, | 
					
						
							|  |  |  |     path, | 
					
						
							|  |  |  |     width, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function updateStickerLastUsed( | 
					
						
							|  |  |  |   packId: string, | 
					
						
							|  |  |  |   stickerId: number, | 
					
						
							|  |  |  |   lastUsed: number | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE stickers | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |     SET lastUsed = $lastUsed | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     WHERE id = $id AND packId = $packId; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id: stickerId, | 
					
						
							|  |  |  |     packId, | 
					
						
							|  |  |  |     lastUsed, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE sticker_packs | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |     SET lastUsed = $lastUsed | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     WHERE id = $id; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id: packId, | 
					
						
							|  |  |  |     lastUsed, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function addStickerPackReference( | 
					
						
							|  |  |  |   messageId: string, | 
					
						
							|  |  |  |   packId: string | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   if (!messageId) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'addStickerPackReference: Provided data did not have a truthy messageId' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!packId) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'addStickerPackReference: Provided data did not have a truthy packId' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO sticker_references ( | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       messageId, | 
					
						
							|  |  |  |       packId | 
					
						
							|  |  |  |     ) values ( | 
					
						
							|  |  |  |       $messageId, | 
					
						
							|  |  |  |       $packId | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     ) | 
					
						
							|  |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2021-04-08 17:50:25 -07:00
										 |  |  |   ).run({ | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     messageId, | 
					
						
							|  |  |  |     packId, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function deleteStickerPackReference( | 
					
						
							|  |  |  |   messageId: string, | 
					
						
							|  |  |  |   packId: string | 
					
						
							| 
									
										
										
										
											2021-07-29 11:59:26 -07:00
										 |  |  | ): Promise<ReadonlyArray<string> | undefined> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   if (!messageId) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'addStickerPackReference: Provided data did not have a truthy messageId' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!packId) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'addStickerPackReference: Provided data did not have a truthy packId' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return db | 
					
						
							|  |  |  |     .transaction(() => { | 
					
						
							|  |  |  |       // We use an immediate transaction here to immediately acquire an exclusive lock,
 | 
					
						
							|  |  |  |       //   which would normally only happen when we did our first write.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // We need this to ensure that our five queries are all atomic, with no
 | 
					
						
							|  |  |  |       // other changes happening while we do it:
 | 
					
						
							|  |  |  |       // 1. Delete our target messageId/packId references
 | 
					
						
							|  |  |  |       // 2. Check the number of references still pointing at packId
 | 
					
						
							|  |  |  |       // 3. If that number is zero, get pack from sticker_packs database
 | 
					
						
							|  |  |  |       // 4. If it's not installed, then grab all of its sticker paths
 | 
					
						
							|  |  |  |       // 5. If it's not installed, then sticker pack (which cascades to all
 | 
					
						
							|  |  |  |       //    stickers and references)
 | 
					
						
							|  |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         DELETE FROM sticker_references | 
					
						
							|  |  |  |         WHERE messageId = $messageId AND packId = $packId; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ | 
					
						
							|  |  |  |         messageId, | 
					
						
							|  |  |  |         packId, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |       const count = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         .prepare<Query>( | 
					
						
							|  |  |  |           `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |           SELECT count(1) FROM sticker_references | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |           WHERE packId = $packId; | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |         .pluck() | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         .get({ packId }); | 
					
						
							|  |  |  |       if (count > 0) { | 
					
						
							| 
									
										
										
										
											2021-07-29 11:59:26 -07:00
										 |  |  |         return undefined; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       const packRow: { status: StickerPackStatusType } = db | 
					
						
							|  |  |  |         .prepare<Query>( | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |           SELECT status FROM sticker_packs | 
					
						
							|  |  |  |           WHERE id = $packId; | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         .get({ packId }); | 
					
						
							|  |  |  |       if (!packRow) { | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |         logger.warn('deleteStickerPackReference: did not find referenced pack'); | 
					
						
							| 
									
										
										
										
											2021-07-29 11:59:26 -07:00
										 |  |  |         return undefined; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       } | 
					
						
							|  |  |  |       const { status } = packRow; | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       if (status === 'installed') { | 
					
						
							| 
									
										
										
										
											2021-07-29 11:59:26 -07:00
										 |  |  |         return undefined; | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       const stickerPathRows: Array<{ path: string }> = db | 
					
						
							|  |  |  |         .prepare<Query>( | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |           SELECT path FROM stickers | 
					
						
							|  |  |  |           WHERE packId = $packId; | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         .all({ | 
					
						
							|  |  |  |           packId, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         DELETE FROM sticker_packs | 
					
						
							|  |  |  |         WHERE id = $packId; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ | 
					
						
							|  |  |  |         packId, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       return (stickerPathRows || []).map(row => row.path); | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     .immediate(); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-03 13:10:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function deleteStickerPack(packId: string): Promise<Array<string>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   if (!packId) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       'deleteStickerPack: Provided data did not have a truthy packId' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   return db | 
					
						
							|  |  |  |     .transaction(() => { | 
					
						
							|  |  |  |       // We use an immediate transaction here to immediately acquire an exclusive lock,
 | 
					
						
							|  |  |  |       //   which would normally only happen when we did our first write.
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       // We need this to ensure that our two queries are atomic, with no other changes
 | 
					
						
							|  |  |  |       //   happening while we do it:
 | 
					
						
							|  |  |  |       // 1. Grab all of target pack's sticker paths
 | 
					
						
							|  |  |  |       // 2. Delete sticker pack (which cascades to all stickers and references)
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       const stickerPathRows: Array<{ path: string }> = db | 
					
						
							|  |  |  |         .prepare<Query>( | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |           SELECT path FROM stickers | 
					
						
							|  |  |  |           WHERE packId = $packId; | 
					
						
							|  |  |  |           `
 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         .all({ | 
					
						
							|  |  |  |           packId, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         DELETE FROM sticker_packs | 
					
						
							|  |  |  |         WHERE id = $packId; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ packId }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return (stickerPathRows || []).map(row => row.path); | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     .immediate(); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-03 13:10:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getStickerCount(): Promise<number> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   return getCountFromTable(getInstance(), 'stickers'); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getAllStickerPacks(): Promise<Array<StickerPackType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT * FROM sticker_packs | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |       ORDER BY position ASC, id ASC | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return rows || []; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | function addUninstalledStickerPackSync(pack: UninstalledStickerPackType): void { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |         INSERT OR REPLACE INTO uninstalled_sticker_packs | 
					
						
							|  |  |  |         ( | 
					
						
							|  |  |  |           id, uninstalledAt, storageID, storageVersion, storageUnknownFields, | 
					
						
							|  |  |  |           storageNeedsSync | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         VALUES | 
					
						
							|  |  |  |         ( | 
					
						
							|  |  |  |           $id, $uninstalledAt, $storageID, $storageVersion, $unknownFields, | 
					
						
							|  |  |  |           $storageNeedsSync | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id: pack.id, | 
					
						
							|  |  |  |     uninstalledAt: pack.uninstalledAt, | 
					
						
							|  |  |  |     storageID: pack.storageID ?? null, | 
					
						
							|  |  |  |     storageVersion: pack.storageVersion ?? null, | 
					
						
							|  |  |  |     unknownFields: pack.storageUnknownFields ?? null, | 
					
						
							|  |  |  |     storageNeedsSync: pack.storageNeedsSync ? 1 : 0, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function addUninstalledStickerPack( | 
					
						
							|  |  |  |   pack: UninstalledStickerPackType | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   return addUninstalledStickerPackSync(pack); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | function removeUninstalledStickerPackSync(packId: string): void { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     'DELETE FROM uninstalled_sticker_packs WHERE id IS $id' | 
					
						
							|  |  |  |   ).run({ id: packId }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function removeUninstalledStickerPack(packId: string): Promise<void> { | 
					
						
							|  |  |  |   return removeUninstalledStickerPackSync(packId); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function getUninstalledStickerPacks(): Promise< | 
					
						
							|  |  |  |   Array<UninstalledStickerPackType> | 
					
						
							|  |  |  | > { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const rows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       'SELECT * FROM uninstalled_sticker_packs ORDER BY id ASC' | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return rows || []; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function getInstalledStickerPacks(): Promise<Array<StickerPackType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // If sticker pack has a storageID - it is being downloaded and about to be
 | 
					
						
							|  |  |  |   // installed so we better sync it back to storage service if asked.
 | 
					
						
							|  |  |  |   const rows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT * | 
					
						
							|  |  |  |       FROM sticker_packs | 
					
						
							|  |  |  |       WHERE | 
					
						
							| 
									
										
										
										
											2023-01-18 14:12:33 -08:00
										 |  |  |         status IS 'installed' OR | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  |         storageID IS NOT NULL | 
					
						
							|  |  |  |       ORDER BY id ASC | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return rows || []; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-03 10:10:49 -07:00
										 |  |  | async function getStickerPackInfo( | 
					
						
							|  |  |  |   packId: string | 
					
						
							|  |  |  | ): Promise<StickerPackInfoType | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     const uninstalled = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT * FROM uninstalled_sticker_packs | 
					
						
							|  |  |  |         WHERE id IS $packId | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .get({ packId }); | 
					
						
							|  |  |  |     if (uninstalled) { | 
					
						
							|  |  |  |       return uninstalled as UninstalledStickerPackType; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const installed = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT | 
					
						
							|  |  |  |           id, key, position, storageID, storageVersion, storageUnknownFields | 
					
						
							|  |  |  |         FROM sticker_packs | 
					
						
							|  |  |  |         WHERE id IS $packId | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .get({ packId }); | 
					
						
							|  |  |  |     if (installed) { | 
					
						
							|  |  |  |       return installed as InstalledStickerPackType; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function installStickerPack( | 
					
						
							|  |  |  |   packId: string, | 
					
						
							|  |  |  |   timestamp: number | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     const status = 'installed'; | 
					
						
							|  |  |  |     updateStickerPackStatusSync(packId, status, { timestamp }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     removeUninstalledStickerPackSync(packId); | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function uninstallStickerPack( | 
					
						
							|  |  |  |   packId: string, | 
					
						
							|  |  |  |   timestamp: number | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     const status = 'downloaded'; | 
					
						
							|  |  |  |     updateStickerPackStatusSync(packId, status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     db.prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       UPDATE sticker_packs SET | 
					
						
							|  |  |  |         storageID = NULL, | 
					
						
							|  |  |  |         storageVersion = NULL, | 
					
						
							|  |  |  |         storageUnknownFields = NULL, | 
					
						
							|  |  |  |         storageNeedsSync = 0 | 
					
						
							|  |  |  |       WHERE id = $packId; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ).run({ packId }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     addUninstalledStickerPackSync({ | 
					
						
							|  |  |  |       id: packId, | 
					
						
							|  |  |  |       uninstalledAt: timestamp, | 
					
						
							|  |  |  |       storageNeedsSync: true, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getAllStickers(): Promise<Array<StickerType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT * FROM stickers | 
					
						
							|  |  |  |       ORDER BY packId ASC, id ASC | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all(); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 13:00:22 -07:00
										 |  |  |   return (rows || []).map(row => rowToSticker(row)); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getRecentStickers({ limit }: { limit?: number } = {}): Promise< | 
					
						
							|  |  |  |   Array<StickerType> | 
					
						
							|  |  |  | > { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   // Note: we avoid 'IS NOT NULL' here because it does seem to bypass our index
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT stickers.* FROM stickers | 
					
						
							|  |  |  |       JOIN sticker_packs on stickers.packId = sticker_packs.id | 
					
						
							|  |  |  |       WHERE stickers.lastUsed > 0 AND sticker_packs.status = 'installed' | 
					
						
							|  |  |  |       ORDER BY stickers.lastUsed DESC | 
					
						
							|  |  |  |       LIMIT $limit | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       limit: limit || 24, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 13:00:22 -07:00
										 |  |  |   return (rows || []).map(row => rowToSticker(row)); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 16:58:27 -07:00
										 |  |  | // Emojis
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function updateEmojiUsage( | 
					
						
							|  |  |  |   shortName: string, | 
					
						
							|  |  |  |   timeUsed: number = Date.now() | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2019-05-24 16:58:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     const rows = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT * FROM emojis | 
					
						
							|  |  |  |         WHERE shortName = $shortName; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .get({ | 
					
						
							|  |  |  |         shortName, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2019-06-24 11:43:45 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (rows) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         UPDATE emojis | 
					
						
							|  |  |  |         SET lastUsage = $timeUsed | 
					
						
							|  |  |  |         WHERE shortName = $shortName; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ shortName, timeUsed }); | 
					
						
							| 
									
										
										
										
											2019-06-24 11:43:45 -07:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       db.prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         INSERT INTO emojis(shortName, lastUsage) | 
					
						
							|  |  |  |         VALUES ($shortName, $timeUsed); | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ).run({ shortName, timeUsed }); | 
					
						
							| 
									
										
										
										
											2019-05-24 16:58:27 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2019-05-24 16:58:27 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function getRecentEmojis(limit = 32): Promise<Array<EmojiType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   const rows = db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT * | 
					
						
							|  |  |  |       FROM emojis | 
					
						
							|  |  |  |       ORDER BY lastUsage DESC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ limit }); | 
					
						
							| 
									
										
										
										
											2019-05-24 16:58:27 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return rows || []; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  | async function getAllBadges(): Promise<Array<BadgeType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const [badgeRows, badgeImageFileRows] = db.transaction(() => [ | 
					
						
							|  |  |  |     db.prepare<EmptyQuery>('SELECT * FROM badges').all(), | 
					
						
							|  |  |  |     db.prepare<EmptyQuery>('SELECT * FROM badgeImageFiles').all(), | 
					
						
							|  |  |  |   ])(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const badgeImagesByBadge = new Map< | 
					
						
							|  |  |  |     string, | 
					
						
							|  |  |  |     Array<undefined | BadgeImageType> | 
					
						
							|  |  |  |   >(); | 
					
						
							|  |  |  |   for (const badgeImageFileRow of badgeImageFileRows) { | 
					
						
							|  |  |  |     const { badgeId, order, localPath, url, theme } = badgeImageFileRow; | 
					
						
							|  |  |  |     const badgeImages = badgeImagesByBadge.get(badgeId) || []; | 
					
						
							|  |  |  |     badgeImages[order] = { | 
					
						
							|  |  |  |       ...(badgeImages[order] || {}), | 
					
						
							|  |  |  |       [parseBadgeImageTheme(theme)]: { | 
					
						
							|  |  |  |         localPath: dropNull(localPath), | 
					
						
							|  |  |  |         url, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     badgeImagesByBadge.set(badgeId, badgeImages); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return badgeRows.map(badgeRow => ({ | 
					
						
							|  |  |  |     id: badgeRow.id, | 
					
						
							|  |  |  |     category: parseBadgeCategory(badgeRow.category), | 
					
						
							|  |  |  |     name: badgeRow.name, | 
					
						
							|  |  |  |     descriptionTemplate: badgeRow.descriptionTemplate, | 
					
						
							|  |  |  |     images: (badgeImagesByBadge.get(badgeRow.id) || []).filter(isNotNil), | 
					
						
							|  |  |  |   })); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This should match the logic in the badges Redux reducer.
 | 
					
						
							|  |  |  | async function updateOrCreateBadges( | 
					
						
							|  |  |  |   badges: ReadonlyArray<BadgeType> | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const insertBadge = prepare<Query>( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO badges ( | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       category, | 
					
						
							|  |  |  |       name, | 
					
						
							|  |  |  |       descriptionTemplate | 
					
						
							|  |  |  |     ) VALUES ( | 
					
						
							|  |  |  |       $id, | 
					
						
							|  |  |  |       $category, | 
					
						
							|  |  |  |       $name, | 
					
						
							|  |  |  |       $descriptionTemplate | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  |   const getImageFilesForBadge = prepare<Query>( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     'SELECT url, localPath FROM badgeImageFiles WHERE badgeId = $badgeId' | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  |   const insertBadgeImageFile = prepare<Query>( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT INTO badgeImageFiles ( | 
					
						
							|  |  |  |       badgeId, | 
					
						
							|  |  |  |       'order', | 
					
						
							|  |  |  |       url, | 
					
						
							|  |  |  |       localPath, | 
					
						
							|  |  |  |       theme | 
					
						
							|  |  |  |     ) VALUES ( | 
					
						
							|  |  |  |       $badgeId, | 
					
						
							|  |  |  |       $order, | 
					
						
							|  |  |  |       $url, | 
					
						
							|  |  |  |       $localPath, | 
					
						
							|  |  |  |       $theme | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     badges.forEach(badge => { | 
					
						
							|  |  |  |       const { id: badgeId } = badge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const oldLocalPaths = new Map<string, string>(); | 
					
						
							|  |  |  |       for (const { url, localPath } of getImageFilesForBadge.all({ badgeId })) { | 
					
						
							|  |  |  |         if (localPath) { | 
					
						
							|  |  |  |           oldLocalPaths.set(url, localPath); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       insertBadge.run({ | 
					
						
							|  |  |  |         id: badgeId, | 
					
						
							|  |  |  |         category: badge.category, | 
					
						
							|  |  |  |         name: badge.name, | 
					
						
							|  |  |  |         descriptionTemplate: badge.descriptionTemplate, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (const [order, image] of badge.images.entries()) { | 
					
						
							|  |  |  |         for (const [theme, imageFile] of Object.entries(image)) { | 
					
						
							|  |  |  |           insertBadgeImageFile.run({ | 
					
						
							|  |  |  |             badgeId, | 
					
						
							|  |  |  |             localPath: | 
					
						
							|  |  |  |               imageFile.localPath || oldLocalPaths.get(imageFile.url) || null, | 
					
						
							|  |  |  |             order, | 
					
						
							|  |  |  |             theme, | 
					
						
							|  |  |  |             url: imageFile.url, | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function badgeImageFileDownloaded( | 
					
						
							|  |  |  |   url: string, | 
					
						
							|  |  |  |   localPath: string | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   prepare<Query>( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     'UPDATE badgeImageFiles SET localPath = $localPath WHERE url = $url' | 
					
						
							|  |  |  |   ).run({ url, localPath }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function getAllBadgeImageFileLocalPaths(): Promise<Set<string>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const localPaths = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       'SELECT localPath FROM badgeImageFiles WHERE localPath IS NOT NULL' | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck() | 
					
						
							|  |  |  |     .all(); | 
					
						
							|  |  |  |   return new Set(localPaths); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | type StoryDistributionForDatabase = Readonly< | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     allowsReplies: 0 | 1; | 
					
						
							|  |  |  |     deletedAtTimestamp: number | null; | 
					
						
							|  |  |  |     isBlockList: 0 | 1; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     senderKeyInfoJson: string | null; | 
					
						
							| 
									
										
										
										
											2022-07-20 20:07:09 -04:00
										 |  |  |     storageID: string | null; | 
					
						
							|  |  |  |     storageVersion: number | null; | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     storageNeedsSync: 0 | 1; | 
					
						
							|  |  |  |   } & Omit< | 
					
						
							|  |  |  |     StoryDistributionType, | 
					
						
							|  |  |  |     | 'allowsReplies' | 
					
						
							|  |  |  |     | 'deletedAtTimestamp' | 
					
						
							|  |  |  |     | 'isBlockList' | 
					
						
							|  |  |  |     | 'senderKeyInfo' | 
					
						
							| 
									
										
										
										
											2022-07-20 20:07:09 -04:00
										 |  |  |     | 'storageID' | 
					
						
							|  |  |  |     | 'storageVersion' | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     | 'storageNeedsSync' | 
					
						
							|  |  |  |   > | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | >; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function hydrateStoryDistribution( | 
					
						
							|  |  |  |   fromDatabase: StoryDistributionForDatabase | 
					
						
							|  |  |  | ): StoryDistributionType { | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     ...omit(fromDatabase, 'senderKeyInfoJson'), | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     allowsReplies: Boolean(fromDatabase.allowsReplies), | 
					
						
							|  |  |  |     deletedAtTimestamp: fromDatabase.deletedAtTimestamp || undefined, | 
					
						
							|  |  |  |     isBlockList: Boolean(fromDatabase.isBlockList), | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     senderKeyInfo: fromDatabase.senderKeyInfoJson | 
					
						
							|  |  |  |       ? JSON.parse(fromDatabase.senderKeyInfoJson) | 
					
						
							|  |  |  |       : undefined, | 
					
						
							| 
									
										
										
										
											2022-07-20 20:07:09 -04:00
										 |  |  |     storageID: fromDatabase.storageID || undefined, | 
					
						
							|  |  |  |     storageVersion: fromDatabase.storageVersion || undefined, | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     storageNeedsSync: Boolean(fromDatabase.storageNeedsSync), | 
					
						
							|  |  |  |     storageUnknownFields: fromDatabase.storageUnknownFields || undefined, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | function freezeStoryDistribution( | 
					
						
							|  |  |  |   story: StoryDistributionType | 
					
						
							|  |  |  | ): StoryDistributionForDatabase { | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     ...omit(story, 'senderKeyInfo'), | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     allowsReplies: story.allowsReplies ? 1 : 0, | 
					
						
							|  |  |  |     deletedAtTimestamp: story.deletedAtTimestamp || null, | 
					
						
							|  |  |  |     isBlockList: story.isBlockList ? 1 : 0, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |     senderKeyInfoJson: story.senderKeyInfo | 
					
						
							|  |  |  |       ? JSON.stringify(story.senderKeyInfo) | 
					
						
							|  |  |  |       : null, | 
					
						
							| 
									
										
										
										
											2022-07-20 20:07:09 -04:00
										 |  |  |     storageID: story.storageID || null, | 
					
						
							|  |  |  |     storageVersion: story.storageVersion || null, | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     storageNeedsSync: story.storageNeedsSync ? 1 : 0, | 
					
						
							|  |  |  |     storageUnknownFields: story.storageUnknownFields || null, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function _getAllStoryDistributions(): Promise< | 
					
						
							|  |  |  |   Array<StoryDistributionType> | 
					
						
							|  |  |  | > { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const storyDistributions = db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>('SELECT * FROM storyDistributions;') | 
					
						
							|  |  |  |     .all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return storyDistributions.map(hydrateStoryDistribution); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function _getAllStoryDistributionMembers(): Promise< | 
					
						
							|  |  |  |   Array<StoryDistributionMemberType> | 
					
						
							|  |  |  | > { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>('SELECT * FROM storyDistributionMembers;') | 
					
						
							|  |  |  |     .all(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function _deleteAllStoryDistributions(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<EmptyQuery>('DELETE FROM storyDistributions;').run(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function createNewStoryDistribution( | 
					
						
							| 
									
										
										
										
											2021-12-09 18:15:59 -08:00
										 |  |  |   distribution: StoryDistributionWithMembersType | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   strictAssert( | 
					
						
							|  |  |  |     distribution.name, | 
					
						
							|  |  |  |     'Distribution list does not have a valid name' | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2021-12-09 18:15:59 -08:00
										 |  |  |     const payload = freezeStoryDistribution(distribution); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       INSERT INTO storyDistributions( | 
					
						
							|  |  |  |         id, | 
					
						
							|  |  |  |         name, | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |         deletedAtTimestamp, | 
					
						
							|  |  |  |         allowsReplies, | 
					
						
							|  |  |  |         isBlockList, | 
					
						
							|  |  |  |         senderKeyInfoJson, | 
					
						
							|  |  |  |         storageID, | 
					
						
							|  |  |  |         storageVersion, | 
					
						
							|  |  |  |         storageUnknownFields, | 
					
						
							|  |  |  |         storageNeedsSync | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       ) VALUES ( | 
					
						
							|  |  |  |         $id, | 
					
						
							|  |  |  |         $name, | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |         $deletedAtTimestamp, | 
					
						
							|  |  |  |         $allowsReplies, | 
					
						
							|  |  |  |         $isBlockList, | 
					
						
							|  |  |  |         $senderKeyInfoJson, | 
					
						
							|  |  |  |         $storageID, | 
					
						
							|  |  |  |         $storageVersion, | 
					
						
							|  |  |  |         $storageUnknownFields, | 
					
						
							|  |  |  |         $storageNeedsSync | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       ); | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ).run(payload); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 18:15:59 -08:00
										 |  |  |     const { id: listId, members } = distribution; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const memberInsertStatement = prepare( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       INSERT OR REPLACE INTO storyDistributionMembers ( | 
					
						
							|  |  |  |         listId, | 
					
						
							|  |  |  |         uuid | 
					
						
							|  |  |  |       ) VALUES ( | 
					
						
							|  |  |  |         $listId, | 
					
						
							|  |  |  |         $uuid | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const uuid of members) { | 
					
						
							|  |  |  |       memberInsertStatement.run({ | 
					
						
							|  |  |  |         listId, | 
					
						
							|  |  |  |         uuid, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function getAllStoryDistributionsWithMembers(): Promise< | 
					
						
							|  |  |  |   Array<StoryDistributionWithMembersType> | 
					
						
							|  |  |  | > { | 
					
						
							|  |  |  |   const allDistributions = await _getAllStoryDistributions(); | 
					
						
							|  |  |  |   const allMembers = await _getAllStoryDistributionMembers(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const byListId = groupBy(allMembers, member => member.listId); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return allDistributions.map(list => ({ | 
					
						
							|  |  |  |     ...list, | 
					
						
							|  |  |  |     members: (byListId[list.id] || []).map(member => member.uuid), | 
					
						
							|  |  |  |   })); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  | async function getStoryDistributionWithMembers( | 
					
						
							|  |  |  |   id: string | 
					
						
							|  |  |  | ): Promise<StoryDistributionWithMembersType | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2022-09-30 09:59:36 -07:00
										 |  |  |   const storyDistribution: StoryDistributionForDatabase | undefined = prepare( | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |     db, | 
					
						
							|  |  |  |     'SELECT * FROM storyDistributions WHERE id = $id;' | 
					
						
							|  |  |  |   ).get({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!storyDistribution) { | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const members = prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     'SELECT * FROM storyDistributionMembers WHERE listId = $id;' | 
					
						
							|  |  |  |   ).all({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							| 
									
										
										
										
											2022-09-30 09:59:36 -07:00
										 |  |  |     ...hydrateStoryDistribution(storyDistribution), | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |     members: members.map(({ uuid }) => uuid), | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  | function modifyStoryDistributionSync( | 
					
						
							|  |  |  |   db: Database, | 
					
						
							|  |  |  |   payload: StoryDistributionForDatabase | 
					
						
							|  |  |  | ): void { | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   if (payload.deletedAtTimestamp) { | 
					
						
							|  |  |  |     strictAssert( | 
					
						
							|  |  |  |       !payload.name, | 
					
						
							|  |  |  |       'Attempt to delete distribution list but still has a name' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     strictAssert( | 
					
						
							|  |  |  |       payload.name, | 
					
						
							|  |  |  |       'Cannot clear distribution list name without deletedAtTimestamp set' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 18:15:59 -08:00
										 |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE storyDistributions | 
					
						
							|  |  |  |     SET | 
					
						
							|  |  |  |       name = $name, | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |       deletedAtTimestamp = $deletedAtTimestamp, | 
					
						
							|  |  |  |       allowsReplies = $allowsReplies, | 
					
						
							|  |  |  |       isBlockList = $isBlockList, | 
					
						
							|  |  |  |       senderKeyInfoJson = $senderKeyInfoJson, | 
					
						
							|  |  |  |       storageID = $storageID, | 
					
						
							|  |  |  |       storageVersion = $storageVersion, | 
					
						
							|  |  |  |       storageUnknownFields = $storageUnknownFields, | 
					
						
							|  |  |  |       storageNeedsSync = $storageNeedsSync | 
					
						
							| 
									
										
										
										
											2021-12-09 18:15:59 -08:00
										 |  |  |     WHERE id = $id | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(payload); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  | function modifyStoryDistributionMembersSync( | 
					
						
							|  |  |  |   db: Database, | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   listId: string, | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     toAdd, | 
					
						
							|  |  |  |     toRemove, | 
					
						
							|  |  |  |   }: { toAdd: Array<UUIDStringType>; toRemove: Array<UUIDStringType> } | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  | ) { | 
					
						
							|  |  |  |   const memberInsertStatement = prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO storyDistributionMembers ( | 
					
						
							|  |  |  |       listId, | 
					
						
							|  |  |  |       uuid | 
					
						
							|  |  |  |     ) VALUES ( | 
					
						
							|  |  |  |       $listId, | 
					
						
							|  |  |  |       $uuid | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |   for (const uuid of toAdd) { | 
					
						
							|  |  |  |     memberInsertStatement.run({ | 
					
						
							|  |  |  |       listId, | 
					
						
							|  |  |  |       uuid, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  |   batchMultiVarQuery(db, toRemove, (uuids: ReadonlyArray<UUIDStringType>) => { | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     db.prepare<ArrayQuery>( | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |       DELETE FROM storyDistributionMembers | 
					
						
							|  |  |  |       WHERE listId = ? AND uuid IN ( ${uuids.map(() => '?').join(', ')} ); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |     ).run([listId, ...uuids]); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function modifyStoryDistributionWithMembers( | 
					
						
							|  |  |  |   distribution: StoryDistributionType, | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     toAdd, | 
					
						
							|  |  |  |     toRemove, | 
					
						
							|  |  |  |   }: { toAdd: Array<UUIDStringType>; toRemove: Array<UUIDStringType> } | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const payload = freezeStoryDistribution(distribution); | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |   if (toAdd.length || toRemove.length) { | 
					
						
							|  |  |  |     db.transaction(() => { | 
					
						
							|  |  |  |       modifyStoryDistributionSync(db, payload); | 
					
						
							|  |  |  |       modifyStoryDistributionMembersSync(db, payload.id, { toAdd, toRemove }); | 
					
						
							|  |  |  |     })(); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     modifyStoryDistributionSync(db, payload); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function modifyStoryDistribution( | 
					
						
							|  |  |  |   distribution: StoryDistributionType | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const payload = freezeStoryDistribution(distribution); | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   modifyStoryDistributionSync(db, payload); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function modifyStoryDistributionMembers( | 
					
						
							|  |  |  |   listId: string, | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     toAdd, | 
					
						
							|  |  |  |     toRemove, | 
					
						
							|  |  |  |   }: { toAdd: Array<UUIDStringType>; toRemove: Array<UUIDStringType> } | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-30 20:52:03 -04:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     modifyStoryDistributionMembersSync(db, listId, { toAdd, toRemove }); | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function deleteStoryDistribution(id: UUIDStringType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<Query>('DELETE FROM storyDistributions WHERE id = $id;').run({ | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function _getAllStoryReads(): Promise<Array<StoryReadType>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db.prepare<EmptyQuery>('SELECT * FROM storyReads;').all(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function _deleteAllStoryReads(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   db.prepare<EmptyQuery>('DELETE FROM storyReads;').run(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function addNewStoryRead(read: StoryReadType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     INSERT OR REPLACE INTO storyReads( | 
					
						
							|  |  |  |       authorId, | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       storyId, | 
					
						
							|  |  |  |       storyReadDate | 
					
						
							|  |  |  |     ) VALUES ( | 
					
						
							|  |  |  |       $authorId, | 
					
						
							|  |  |  |       $conversationId, | 
					
						
							|  |  |  |       $storyId, | 
					
						
							|  |  |  |       $storyReadDate | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run(read); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | async function getLastStoryReadsForAuthor({ | 
					
						
							|  |  |  |   authorId, | 
					
						
							|  |  |  |   conversationId, | 
					
						
							|  |  |  |   limit: initialLimit, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   authorId: UUIDStringType; | 
					
						
							|  |  |  |   conversationId?: UUIDStringType; | 
					
						
							|  |  |  |   limit?: number; | 
					
						
							|  |  |  | }): Promise<Array<StoryReadType>> { | 
					
						
							|  |  |  |   const limit = initialLimit || 5; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT * FROM storyReads | 
					
						
							|  |  |  |       WHERE | 
					
						
							|  |  |  |         authorId = $authorId AND | 
					
						
							|  |  |  |         ($conversationId IS NULL OR conversationId = $conversationId) | 
					
						
							|  |  |  |       ORDER BY storyReadDate DESC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       authorId, | 
					
						
							|  |  |  |       conversationId: conversationId || null, | 
					
						
							|  |  |  |       limit, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  | async function countStoryReadsByConversation( | 
					
						
							|  |  |  |   conversationId: string | 
					
						
							|  |  |  | ): Promise<number> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2023-01-17 13:07:21 -08:00
										 |  |  |       SELECT count(1) FROM storyReads | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  |       WHERE conversationId = $conversationId; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck() | 
					
						
							|  |  |  |     .get({ conversationId }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | // All data in database
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeAll(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2018-07-31 19:29:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							|  |  |  |     db.exec(`
 | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM attachment_downloads; | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |       DELETE FROM badgeImageFiles; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM badges; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM conversations; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM emojis; | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  |       DELETE FROM groupCallRingCancellations; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM identityKeys; | 
					
						
							|  |  |  |       DELETE FROM items; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM jobs; | 
					
						
							|  |  |  |       DELETE FROM messages_fts; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM messages; | 
					
						
							|  |  |  |       DELETE FROM preKeys; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM reactions; | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  |       DELETE FROM senderKeys; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM sendLogMessageIds; | 
					
						
							|  |  |  |       DELETE FROM sendLogPayloads; | 
					
						
							|  |  |  |       DELETE FROM sendLogRecipients; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM sessions; | 
					
						
							|  |  |  |       DELETE FROM signedPreKeys; | 
					
						
							|  |  |  |       DELETE FROM sticker_packs; | 
					
						
							|  |  |  |       DELETE FROM sticker_references; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM stickers; | 
					
						
							|  |  |  |       DELETE FROM storyDistributionMembers; | 
					
						
							|  |  |  |       DELETE FROM storyDistributions; | 
					
						
							|  |  |  |       DELETE FROM storyReads; | 
					
						
							|  |  |  |       DELETE FROM unprocessed; | 
					
						
							| 
									
										
										
										
											2022-10-20 12:16:37 -07:00
										 |  |  |       DELETE FROM uninstalled_sticker_packs; | 
					
						
							| 
									
										
										
										
											2023-01-26 15:53:22 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       INSERT INTO messages_fts(messages_fts) VALUES('optimize'); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     `);
 | 
					
						
							|  |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Anything that isn't user-visible data
 | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  | async function removeAllConfiguration( | 
					
						
							|  |  |  |   mode = RemoveAllConfiguration.Full | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   db.transaction(() => { | 
					
						
							| 
									
										
										
										
											2021-05-27 13:47:39 -07:00
										 |  |  |     db.exec( | 
					
						
							| 
									
										
										
										
											2021-05-25 15:40:04 -07:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM identityKeys; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM jobs; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM preKeys; | 
					
						
							| 
									
										
										
										
											2021-05-13 18:18:43 -07:00
										 |  |  |       DELETE FROM senderKeys; | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |       DELETE FROM sendLogMessageIds; | 
					
						
							|  |  |  |       DELETE FROM sendLogPayloads; | 
					
						
							|  |  |  |       DELETE FROM sendLogRecipients; | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       DELETE FROM sessions; | 
					
						
							|  |  |  |       DELETE FROM signedPreKeys; | 
					
						
							|  |  |  |       DELETE FROM unprocessed; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-05-27 13:47:39 -07:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (mode === RemoveAllConfiguration.Full) { | 
					
						
							|  |  |  |       db.exec( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         DELETE FROM items; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } else if (mode === RemoveAllConfiguration.Soft) { | 
					
						
							|  |  |  |       const itemIds: ReadonlyArray<string> = db | 
					
						
							|  |  |  |         .prepare<EmptyQuery>('SELECT id FROM items') | 
					
						
							|  |  |  |         .pluck(true) | 
					
						
							|  |  |  |         .all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const allowedSet = new Set<string>(STORAGE_UI_KEYS); | 
					
						
							|  |  |  |       for (const id of itemIds) { | 
					
						
							|  |  |  |         if (!allowedSet.has(id)) { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |           removeById(db, 'items', id); | 
					
						
							| 
									
										
										
										
											2021-08-30 14:39:57 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       throw missingCaseError(mode); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-11 13:11:31 -07:00
										 |  |  |     db.exec( | 
					
						
							|  |  |  |       "UPDATE conversations SET json = json_remove(json, '$.senderKeyInfo');" | 
					
						
							| 
									
										
										
										
											2021-05-27 13:47:39 -07:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   })(); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-20 14:18:23 -07:00
										 |  |  | const MAX_MESSAGE_MIGRATION_ATTEMPTS = 5; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function getMessagesNeedingUpgrade( | 
					
						
							|  |  |  |   limit: number, | 
					
						
							|  |  |  |   { maxVersion }: { maxVersion: number } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2022-06-20 14:18:23 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json | 
					
						
							|  |  |  |       FROM messages | 
					
						
							| 
									
										
										
										
											2022-06-20 14:18:23 -07:00
										 |  |  |       WHERE | 
					
						
							|  |  |  |         (schemaVersion IS NULL OR schemaVersion < $maxVersion) AND | 
					
						
							|  |  |  |         IFNULL( | 
					
						
							|  |  |  |           json_extract(json, '$.schemaMigrationAttempts'), | 
					
						
							|  |  |  |           0 | 
					
						
							|  |  |  |         ) < $maxAttempts | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       maxVersion, | 
					
						
							| 
									
										
										
										
											2022-06-20 14:18:23 -07:00
										 |  |  |       maxAttempts: MAX_MESSAGE_MIGRATION_ATTEMPTS, | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       limit, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function getMessagesWithVisualMediaAttachments( | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   conversationId: string, | 
					
						
							|  |  |  |   { limit }: { limit: number } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows: JSONRows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         storyId IS NULL AND | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							|  |  |  |         hasVisualMediaAttachments = 1 | 
					
						
							|  |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       limit, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return rows.map(row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  | async function getMessagesWithFileAttachments( | 
					
						
							|  |  |  |   conversationId: string, | 
					
						
							|  |  |  |   { limit }: { limit: number } | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<MessageType>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   const rows = db | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |       SELECT json FROM messages WHERE | 
					
						
							| 
									
										
										
										
											2021-12-15 00:17:14 -08:00
										 |  |  |         isStory IS 0 AND | 
					
						
							| 
									
										
										
										
											2021-12-08 11:52:46 -08:00
										 |  |  |         storyId IS NULL AND | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |         conversationId = $conversationId AND | 
					
						
							|  |  |  |         hasFileAttachments = 1 | 
					
						
							|  |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT $limit; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ | 
					
						
							|  |  |  |       conversationId, | 
					
						
							|  |  |  |       limit, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 16:51:45 -07:00
										 |  |  |   return map(rows, row => jsonToObject(row.json)); | 
					
						
							| 
									
										
										
										
											2018-07-26 18:13:56 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 16:17:05 -04:00
										 |  |  | async function getMessageServerGuidsForSpam( | 
					
						
							|  |  |  |   conversationId: string | 
					
						
							|  |  |  | ): Promise<Array<string>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // The server's maximum is 3, which is why you see `LIMIT 3` in this query. Note that we
 | 
					
						
							|  |  |  |   //   use `pluck` here to only get the first column!
 | 
					
						
							|  |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT serverGuid | 
					
						
							|  |  |  |       FROM messages | 
					
						
							|  |  |  |       WHERE conversationId = $conversationId | 
					
						
							|  |  |  |       AND type = 'incoming' | 
					
						
							|  |  |  |       AND serverGuid IS NOT NULL | 
					
						
							|  |  |  |       ORDER BY received_at DESC, sent_at DESC | 
					
						
							|  |  |  |       LIMIT 3; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck(true) | 
					
						
							|  |  |  |     .all({ conversationId }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | function getExternalFilesForMessage(message: MessageType): Array<string> { | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   const { attachments, contact, quote, preview, sticker } = message; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const files: Array<string> = []; | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   forEach(attachments, attachment => { | 
					
						
							|  |  |  |     const { path: file, thumbnail, screenshot } = attachment; | 
					
						
							|  |  |  |     if (file) { | 
					
						
							|  |  |  |       files.push(file); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (thumbnail && thumbnail.path) { | 
					
						
							|  |  |  |       files.push(thumbnail.path); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (screenshot && screenshot.path) { | 
					
						
							|  |  |  |       files.push(screenshot.path); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (quote && quote.attachments && quote.attachments.length) { | 
					
						
							|  |  |  |     forEach(quote.attachments, attachment => { | 
					
						
							|  |  |  |       const { thumbnail } = attachment; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (thumbnail && thumbnail.path) { | 
					
						
							|  |  |  |         files.push(thumbnail.path); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (contact && contact.length) { | 
					
						
							|  |  |  |     forEach(contact, item => { | 
					
						
							|  |  |  |       const { avatar } = item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (avatar && avatar.avatar && avatar.avatar.path) { | 
					
						
							|  |  |  |         files.push(avatar.avatar.path); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-15 19:03:56 -08:00
										 |  |  |   if (preview && preview.length) { | 
					
						
							|  |  |  |     forEach(preview, item => { | 
					
						
							|  |  |  |       const { image } = item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (image && image.path) { | 
					
						
							|  |  |  |         files.push(image.path); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   if (sticker && sticker.data && sticker.data.path) { | 
					
						
							|  |  |  |     files.push(sticker.data.path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sticker.data.thumbnail && sticker.data.thumbnail.path) { | 
					
						
							|  |  |  |       files.push(sticker.data.thumbnail.path); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  |   return files; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  | function getExternalFilesForConversation( | 
					
						
							|  |  |  |   conversation: Pick<ConversationType, 'avatar' | 'profileAvatar'> | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Array<string> { | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   const { avatar, profileAvatar } = conversation; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const files: Array<string> = []; | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (avatar && avatar.path) { | 
					
						
							|  |  |  |     files.push(avatar.path); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (profileAvatar && profileAvatar.path) { | 
					
						
							|  |  |  |     files.push(profileAvatar.path); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return files; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-18 12:09:27 -05:00
										 |  |  | function getExternalDraftFilesForConversation( | 
					
						
							|  |  |  |   conversation: Pick<ConversationType, 'draftAttachments'> | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Array<string> { | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |   const draftAttachments = conversation.draftAttachments || []; | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const files: Array<string> = []; | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   forEach(draftAttachments, attachment => { | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  |     if (attachment.pending) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |     const { path: file, screenshotPath } = attachment; | 
					
						
							|  |  |  |     if (file) { | 
					
						
							|  |  |  |       files.push(file); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (screenshotPath) { | 
					
						
							|  |  |  |       files.push(screenshotPath); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return files; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  | async function getKnownMessageAttachments( | 
					
						
							|  |  |  |   cursor?: MessageAttachmentsCursorType | 
					
						
							|  |  |  | ): Promise<GetKnownMessageAttachmentsResultType> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   const result = new Set<string>(); | 
					
						
							|  |  |  |   const chunkSize = 1000; | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   return db.transaction(() => { | 
					
						
							|  |  |  |     let count = cursor?.count ?? 0; | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |     strictAssert( | 
					
						
							|  |  |  |       !cursor?.done, | 
					
						
							|  |  |  |       'getKnownMessageAttachments: iteration cannot be restarted' | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |     let runId: string; | 
					
						
							|  |  |  |     if (cursor === undefined) { | 
					
						
							|  |  |  |       runId = randomBytes(8).toString('hex'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const total = getMessageCountSync(); | 
					
						
							|  |  |  |       logger.info( | 
					
						
							|  |  |  |         `getKnownMessageAttachments(${runId}): ` + | 
					
						
							|  |  |  |           `Starting iteration through ${total} messages` | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       db.exec( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         CREATE TEMP TABLE tmp_${runId}_updated_messages | 
					
						
							|  |  |  |           (rowid INTEGER PRIMARY KEY ASC); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         INSERT INTO tmp_${runId}_updated_messages (rowid) | 
					
						
							|  |  |  |         SELECT rowid FROM messages; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         CREATE TEMP TRIGGER tmp_${runId}_message_updates | 
					
						
							|  |  |  |         UPDATE OF json ON messages | 
					
						
							|  |  |  |         BEGIN | 
					
						
							|  |  |  |           INSERT OR IGNORE INTO tmp_${runId}_updated_messages (rowid) | 
					
						
							|  |  |  |           VALUES (NEW.rowid); | 
					
						
							|  |  |  |         END; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         CREATE TEMP TRIGGER tmp_${runId}_message_inserts | 
					
						
							|  |  |  |         AFTER INSERT ON messages | 
					
						
							|  |  |  |         BEGIN | 
					
						
							|  |  |  |           INSERT OR IGNORE INTO tmp_${runId}_updated_messages (rowid) | 
					
						
							|  |  |  |           VALUES (NEW.rowid); | 
					
						
							|  |  |  |         END; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       ({ runId } = cursor); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const rowids: Array<number> = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       DELETE FROM tmp_${runId}_updated_messages | 
					
						
							|  |  |  |       RETURNING rowid | 
					
						
							|  |  |  |       LIMIT $chunkSize; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .pluck() | 
					
						
							|  |  |  |       .all({ chunkSize }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const messages = batchMultiVarQuery( | 
					
						
							|  |  |  |       db, | 
					
						
							|  |  |  |       rowids, | 
					
						
							| 
									
										
										
										
											2022-12-21 16:07:02 -08:00
										 |  |  |       (batch: ReadonlyArray<number>): Array<MessageType> => { | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |         const query = db.prepare<ArrayQuery>( | 
					
						
							|  |  |  |           `SELECT json FROM messages WHERE rowid IN (${Array(batch.length) | 
					
						
							|  |  |  |             .fill('?') | 
					
						
							|  |  |  |             .join(',')});`
 | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         const rows: JSONRows = query.all(batch); | 
					
						
							|  |  |  |         return rows.map(row => jsonToObject(row.json)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const message of messages) { | 
					
						
							|  |  |  |       const externalFiles = getExternalFilesForMessage(message); | 
					
						
							|  |  |  |       forEach(externalFiles, file => result.add(file)); | 
					
						
							|  |  |  |       count += 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 12:06:19 -08:00
										 |  |  |     const done = rowids.length < chunkSize; | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |     return { | 
					
						
							|  |  |  |       attachments: Array.from(result), | 
					
						
							|  |  |  |       cursor: { runId, count, done }, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   })(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function finishGetKnownMessageAttachments({ | 
					
						
							|  |  |  |   runId, | 
					
						
							|  |  |  |   count, | 
					
						
							|  |  |  |   done, | 
					
						
							|  |  |  | }: MessageAttachmentsCursorType): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const logId = `finishGetKnownMessageAttachments(${runId})`; | 
					
						
							|  |  |  |   if (!done) { | 
					
						
							|  |  |  |     logger.warn(`${logId}: iteration not finished`); | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   logger.info(`${logId}: reached the end after processing ${count} messages`); | 
					
						
							|  |  |  |   db.exec(`
 | 
					
						
							|  |  |  |     DROP TABLE tmp_${runId}_updated_messages; | 
					
						
							|  |  |  |     DROP TRIGGER tmp_${runId}_message_updates; | 
					
						
							|  |  |  |     DROP TRIGGER tmp_${runId}_message_inserts; | 
					
						
							|  |  |  |   `);
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function getKnownConversationAttachments(): Promise<Array<string>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const result = new Set<string>(); | 
					
						
							|  |  |  |   const chunkSize = 500; | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   let complete = false; | 
					
						
							|  |  |  |   let id = ''; | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 08:50:45 -07:00
										 |  |  |   const conversationTotal = await getConversationCount(); | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger.info( | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |     'getKnownConversationAttachments: About to iterate through ' + | 
					
						
							|  |  |  |       `${conversationTotal}` | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-05 09:36:07 -07:00
										 |  |  |   const fetchConversations = db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |       SELECT json FROM conversations | 
					
						
							|  |  |  |       WHERE id > $id | 
					
						
							|  |  |  |       ORDER BY id ASC | 
					
						
							|  |  |  |       LIMIT $chunkSize; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |   while (!complete) { | 
					
						
							| 
									
										
										
										
											2021-10-05 09:36:07 -07:00
										 |  |  |     const rows = fetchConversations.all({ | 
					
						
							|  |  |  |       id, | 
					
						
							|  |  |  |       chunkSize, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |     const conversations: Array<ConversationType> = map(rows, row => | 
					
						
							|  |  |  |       jsonToObject(row.json) | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     conversations.forEach(conversation => { | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |       const externalFiles = getExternalFilesForConversation(conversation); | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |       externalFiles.forEach(file => result.add(file)); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |     const lastMessage: ConversationType | undefined = last(conversations); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  |     if (lastMessage) { | 
					
						
							|  |  |  |       ({ id } = lastMessage); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     complete = conversations.length < chunkSize; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   logger.info('getKnownConversationAttachments: Done processing'); | 
					
						
							| 
									
										
										
										
											2018-09-20 18:47:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   return Array.from(result); | 
					
						
							| 
									
										
										
										
											2018-08-06 16:18:58 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeKnownStickers( | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   allStickers: ReadonlyArray<string> | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<string>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const lookup: Dictionary<boolean> = fromPairs( | 
					
						
							|  |  |  |     map(allStickers, file => [file, true]) | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   const chunkSize = 50; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 09:12:26 -07:00
										 |  |  |   const total = await getStickerCount(); | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger.info( | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |     `removeKnownStickers: About to iterate through ${total} stickers` | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let count = 0; | 
					
						
							|  |  |  |   let complete = false; | 
					
						
							|  |  |  |   let rowid = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (!complete) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     const rows: Array<{ rowid: number; path: string }> = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT rowid, path FROM stickers | 
					
						
							|  |  |  |         WHERE rowid > $rowid | 
					
						
							|  |  |  |         ORDER BY rowid ASC | 
					
						
							|  |  |  |         LIMIT $chunkSize; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .all({ | 
					
						
							|  |  |  |         rowid, | 
					
						
							|  |  |  |         chunkSize, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     const files: Array<string> = rows.map(row => row.path); | 
					
						
							|  |  |  |     files.forEach(file => { | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       delete lookup[file]; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     const lastSticker = last(rows); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |     if (lastSticker) { | 
					
						
							|  |  |  |       ({ rowid } = lastSticker); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     complete = rows.length < chunkSize; | 
					
						
							|  |  |  |     count += rows.length; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger.info(`removeKnownStickers: Done processing ${count} stickers`); | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return Object.keys(lookup); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | async function removeKnownDraftAttachments( | 
					
						
							| 
									
										
										
										
											2022-11-16 16:29:15 -08:00
										 |  |  |   allStickers: ReadonlyArray<string> | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  | ): Promise<Array<string>> { | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const lookup: Dictionary<boolean> = fromPairs( | 
					
						
							|  |  |  |     map(allStickers, file => [file, true]) | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |   const chunkSize = 50; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 08:50:45 -07:00
										 |  |  |   const total = await getConversationCount(); | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger.info( | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |     `removeKnownDraftAttachments: About to iterate through ${total} conversations` | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let complete = false; | 
					
						
							|  |  |  |   let count = 0; | 
					
						
							|  |  |  |   // Though conversations.id is a string, this ensures that, when coerced, this
 | 
					
						
							|  |  |  |   //   value is still a string but it's smaller than every other string.
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |   let id: number | string = 0; | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while (!complete) { | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     const rows: JSONRows = db | 
					
						
							|  |  |  |       .prepare<Query>( | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |         SELECT json FROM conversations | 
					
						
							|  |  |  |         WHERE id > $id | 
					
						
							|  |  |  |         ORDER BY id ASC | 
					
						
							|  |  |  |         LIMIT $chunkSize; | 
					
						
							|  |  |  |         `
 | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |       .all({ | 
					
						
							|  |  |  |         id, | 
					
						
							|  |  |  |         chunkSize, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     const conversations: Array<ConversationType> = rows.map(row => | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |       jsonToObject(row.json) | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |     conversations.forEach(conversation => { | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |       const externalFiles = getExternalDraftFilesForConversation(conversation); | 
					
						
							| 
									
										
										
										
											2021-04-05 15:18:19 -07:00
										 |  |  |       externalFiles.forEach(file => { | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |         delete lookup[file]; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:59:11 -07:00
										 |  |  |     const lastMessage: ConversationType | undefined = last(conversations); | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |     if (lastMessage) { | 
					
						
							|  |  |  |       ({ id } = lastMessage); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     complete = conversations.length < chunkSize; | 
					
						
							|  |  |  |     count += conversations.length; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-16 14:54:06 -07:00
										 |  |  |   logger.info( | 
					
						
							| 
									
										
										
										
											2019-08-06 17:40:25 -07:00
										 |  |  |     `removeKnownDraftAttachments: Done processing ${count} conversations` | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return Object.keys(lookup); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 13:13:08 -08:00
										 |  |  | // Default value of 'automerge'.
 | 
					
						
							|  |  |  | // See: https://www.sqlite.org/fts5.html#the_automerge_configuration_option
 | 
					
						
							|  |  |  | const OPTIMIZE_FTS_PAGE_COUNT = 4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This query is incremental. It gets the `state` from the return value of
 | 
					
						
							|  |  |  | // previous `optimizeFTS` call. When `state.done` is `true` - optimization is
 | 
					
						
							|  |  |  | // complete.
 | 
					
						
							|  |  |  | async function optimizeFTS( | 
					
						
							|  |  |  |   state?: FTSOptimizationStateType | 
					
						
							|  |  |  | ): Promise<FTSOptimizationStateType | undefined> { | 
					
						
							|  |  |  |   // See https://www.sqlite.org/fts5.html#the_merge_command
 | 
					
						
							|  |  |  |   let pageCount = OPTIMIZE_FTS_PAGE_COUNT; | 
					
						
							|  |  |  |   if (state === undefined) { | 
					
						
							|  |  |  |     pageCount = -pageCount; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   const { changes } = prepare( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |       INSERT INTO messages_fts(messages_fts, rank) VALUES ('merge', $pageCount); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ pageCount }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (state === undefined) { | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       changes, | 
					
						
							|  |  |  |       steps: 1, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const { changes: prevChanges, steps } = state; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (Math.abs(changes - prevChanges) < 2) { | 
					
						
							|  |  |  |     return { changes, steps, done: true }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // More work is needed.
 | 
					
						
							|  |  |  |   return { changes, steps: steps + 1 }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  | async function getJobsInQueue(queueType: string): Promise<Array<StoredJob>> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2022-02-16 10:36:21 -08:00
										 |  |  |   return getJobsInQueueSync(db, queueType); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 10:36:21 -08:00
										 |  |  | export function getJobsInQueueSync( | 
					
						
							|  |  |  |   db: Database, | 
					
						
							|  |  |  |   queueType: string | 
					
						
							|  |  |  | ): Array<StoredJob> { | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT id, timestamp, data | 
					
						
							|  |  |  |       FROM jobs | 
					
						
							|  |  |  |       WHERE queueType = $queueType | 
					
						
							|  |  |  |       ORDER BY timestamp; | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .all({ queueType }) | 
					
						
							|  |  |  |     .map(row => ({ | 
					
						
							|  |  |  |       id: row.id, | 
					
						
							|  |  |  |       queueType, | 
					
						
							|  |  |  |       timestamp: row.timestamp, | 
					
						
							|  |  |  |       data: isNotNil(row.data) ? JSON.parse(row.data) : undefined, | 
					
						
							|  |  |  |     })); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 10:36:21 -08:00
										 |  |  | export function insertJobSync(db: Database, job: Readonly<StoredJob>): void { | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |       INSERT INTO jobs | 
					
						
							|  |  |  |       (id, queueType, timestamp, data) | 
					
						
							|  |  |  |       VALUES | 
					
						
							|  |  |  |       ($id, $queueType, $timestamp, $data); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     id: job.id, | 
					
						
							|  |  |  |     queueType: job.queueType, | 
					
						
							|  |  |  |     timestamp: job.timestamp, | 
					
						
							|  |  |  |     data: isNotNil(job.data) ? JSON.stringify(job.data) : null, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-31 15:58:39 -05:00
										 |  |  | async function insertJob(job: Readonly<StoredJob>): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  |   return insertJobSync(db, job); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 18:02:27 -05:00
										 |  |  | async function deleteJob(id: string): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.prepare<Query>('DELETE FROM jobs WHERE id = $id').run({ id }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-28 12:15:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  | async function wasGroupCallRingPreviouslyCanceled( | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |   ringId: bigint | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  | ): Promise<boolean> { | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  |   return db | 
					
						
							|  |  |  |     .prepare<Query>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |       SELECT EXISTS ( | 
					
						
							|  |  |  |         SELECT 1 FROM groupCallRingCancellations | 
					
						
							|  |  |  |         WHERE ringId = $ringId | 
					
						
							|  |  |  |         AND createdAt >= $ringsOlderThanThisAreIgnored | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |       ); | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  |       `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck() | 
					
						
							|  |  |  |     .get({ | 
					
						
							|  |  |  |       ringId, | 
					
						
							|  |  |  |       ringsOlderThanThisAreIgnored: Date.now() - MAX_GROUP_CALL_RING_AGE, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  | async function processGroupCallRingCancellation(ringId: bigint): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  |     INSERT INTO groupCallRingCancellations (ringId, createdAt) | 
					
						
							|  |  |  |     VALUES ($ringId, $createdAt) | 
					
						
							|  |  |  |     ON CONFLICT (ringId) DO NOTHING; | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ ringId, createdAt: Date.now() }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This age, in milliseconds, should be longer than any group call ring duration. Beyond
 | 
					
						
							|  |  |  | //   that, it doesn't really matter what the value is.
 | 
					
						
							| 
									
										
										
										
											2021-08-26 09:10:58 -05:00
										 |  |  | const MAX_GROUP_CALL_RING_AGE = 30 * durations.MINUTE; | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  | async function cleanExpiredGroupCallRingCancellations(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							| 
									
										
										
										
											2022-11-16 20:52:04 -06:00
										 |  |  |     DELETE FROM groupCallRingCancellations | 
					
						
							| 
									
										
										
										
											2021-08-20 11:06:15 -05:00
										 |  |  |     WHERE createdAt < $expiredRingTime; | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     expiredRingTime: Date.now() - MAX_GROUP_CALL_RING_AGE, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 11:45:22 -07:00
										 |  |  | async function getMaxMessageCounter(): Promise<number | undefined> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return db | 
					
						
							|  |  |  |     .prepare<EmptyQuery>( | 
					
						
							|  |  |  |       `
 | 
					
						
							|  |  |  |     SELECT MAX(counter) | 
					
						
							|  |  |  |     FROM | 
					
						
							|  |  |  |       ( | 
					
						
							|  |  |  |         SELECT MAX(received_at) AS counter FROM messages | 
					
						
							|  |  |  |         UNION | 
					
						
							|  |  |  |         SELECT MAX(timestamp) AS counter FROM unprocessed | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     .pluck() | 
					
						
							|  |  |  |     .get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  | async function getStatisticsForLogging(): Promise<Record<string, string>> { | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |   const db = getInstance(); | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |   const counts = await pProps({ | 
					
						
							|  |  |  |     messageCount: getMessageCount(), | 
					
						
							|  |  |  |     conversationCount: getConversationCount(), | 
					
						
							| 
									
										
										
										
											2021-10-26 15:59:08 -07:00
										 |  |  |     sessionCount: getCountFromTable(db, 'sessions'), | 
					
						
							|  |  |  |     senderKeyCount: getCountFromTable(db, 'senderKeys'), | 
					
						
							| 
									
										
										
										
											2021-07-30 11:43:16 -05:00
										 |  |  |   }); | 
					
						
							|  |  |  |   return mapValues(counts, formatCountForLogging); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-28 12:15:17 -04:00
										 |  |  | async function updateAllConversationColors( | 
					
						
							|  |  |  |   conversationColor?: ConversationColorType, | 
					
						
							|  |  |  |   customColorData?: { | 
					
						
							|  |  |  |     id: string; | 
					
						
							|  |  |  |     value: CustomColorType; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | ): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.prepare<Query>( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE conversations | 
					
						
							|  |  |  |     SET json = JSON_PATCH(json, $patch); | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ).run({ | 
					
						
							|  |  |  |     patch: JSON.stringify({ | 
					
						
							|  |  |  |       conversationColor: conversationColor || null, | 
					
						
							|  |  |  |       customColor: customColorData?.value || null, | 
					
						
							|  |  |  |       customColorId: customColorData?.id || null, | 
					
						
							|  |  |  |     }), | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-07-08 13:46:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | async function removeAllProfileKeyCredentials(): Promise<void> { | 
					
						
							|  |  |  |   const db = getInstance(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   db.exec( | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |     UPDATE conversations | 
					
						
							|  |  |  |     SET | 
					
						
							|  |  |  |       json = json_remove(json, '$.profileKeyCredential') | 
					
						
							|  |  |  |     `
 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } |