1162 lines
35 KiB
TypeScript
1162 lines
35 KiB
TypeScript
// Copyright 2020 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { Database } from '@signalapp/better-sqlite3';
|
|
import type { ReadonlyDeep } from 'type-fest';
|
|
import type {
|
|
ConversationAttributesType,
|
|
MessageAttributesType,
|
|
SenderKeyInfoType,
|
|
} from '../model-types.d';
|
|
import type { StoredJob } from '../jobs/types';
|
|
import type { ReactionType, ReactionReadStatus } from '../types/Reactions';
|
|
import type { ConversationColorType, CustomColorType } from '../types/Colors';
|
|
import type { StorageAccessType } from '../types/Storage.d';
|
|
import type { BytesToStrings } from '../types/Util';
|
|
import type { QualifiedAddressStringType } from '../types/QualifiedAddress';
|
|
import type { StoryDistributionIdString } from '../types/StoryDistributionId';
|
|
import type { AciString, PniString, ServiceIdString } from '../types/ServiceId';
|
|
import type { BadgeType } from '../badges/types';
|
|
import type { ReadStatus } from '../messages/MessageReadStatus';
|
|
import type { RawBodyRange } from '../types/BodyRange';
|
|
import type {
|
|
GetMessagesBetweenOptions,
|
|
MaybeStaleCallHistory,
|
|
} from './Server';
|
|
import type { MessageTimestamps } from '../state/ducks/conversations';
|
|
import type {
|
|
CallHistoryDetails,
|
|
CallHistoryFilter,
|
|
CallHistoryGroup,
|
|
CallHistoryPagination,
|
|
CallLogEventTarget,
|
|
} from '../types/CallDisposition';
|
|
import type {
|
|
CallLinkRecord,
|
|
CallLinkStateType,
|
|
CallLinkType,
|
|
} from '../types/CallLink';
|
|
import type { AttachmentDownloadJobType } from '../types/AttachmentDownload';
|
|
import type {
|
|
GroupSendEndorsementsData,
|
|
GroupSendMemberEndorsementRecord,
|
|
} from '../types/GroupSendEndorsements';
|
|
import type { SyncTaskType } from '../util/syncTasks';
|
|
import type { AttachmentBackupJobType } from '../types/AttachmentBackup';
|
|
import type { SingleProtoJobQueue } from '../jobs/singleProtoJobQueue';
|
|
import type { DeleteCallLinkOptions } from './server/callLinks';
|
|
|
|
export type ReadableDB = Database & { __readable_db: never };
|
|
export type WritableDB = ReadableDB & { __writable_db: never };
|
|
|
|
export type AdjacentMessagesByConversationOptionsType = Readonly<{
|
|
conversationId: string;
|
|
messageId?: string;
|
|
includeStoryReplies: boolean;
|
|
limit?: number;
|
|
receivedAt?: number;
|
|
sentAt?: number;
|
|
storyId: string | undefined;
|
|
requireVisualMediaAttachments?: boolean;
|
|
requireFileAttachments?: boolean;
|
|
}>;
|
|
|
|
export type GetNearbyMessageFromDeletedSetOptionsType = Readonly<{
|
|
conversationId: string;
|
|
lastSelectedMessage: MessageTimestamps;
|
|
deletedMessageIds: ReadonlyArray<string>;
|
|
storyId: string | undefined;
|
|
includeStoryReplies: boolean;
|
|
}>;
|
|
|
|
export type MessageMetricsType = {
|
|
id: string;
|
|
received_at: number;
|
|
sent_at: number;
|
|
};
|
|
export type ConversationMetricsType = {
|
|
oldest?: MessageMetricsType;
|
|
newest?: MessageMetricsType;
|
|
oldestUnseen?: MessageMetricsType;
|
|
totalUnseen: number;
|
|
};
|
|
export type ConversationType = ConversationAttributesType;
|
|
export type EmojiType = {
|
|
shortName: string;
|
|
lastUsage: number;
|
|
};
|
|
|
|
export type IdentityKeyType = {
|
|
firstUse: boolean;
|
|
id: ServiceIdString | `conversation:${string}`;
|
|
nonblockingApproval: boolean;
|
|
publicKey: Uint8Array;
|
|
timestamp: number;
|
|
verified: number;
|
|
};
|
|
export type StoredIdentityKeyType = {
|
|
firstUse: boolean;
|
|
id: ServiceIdString | `conversation:${string}`;
|
|
nonblockingApproval: boolean;
|
|
publicKey: string;
|
|
timestamp: number;
|
|
verified: number;
|
|
};
|
|
export type IdentityKeyIdType = IdentityKeyType['id'];
|
|
|
|
export type ItemKeyType = keyof StorageAccessType;
|
|
export type AllItemsType = Partial<StorageAccessType>;
|
|
export type StoredAllItemsType = Partial<BytesToStrings<StorageAccessType>>;
|
|
export type ItemType<K extends ItemKeyType> = {
|
|
id: K;
|
|
value: StorageAccessType[K];
|
|
};
|
|
export type StoredItemType<K extends ItemKeyType> = {
|
|
id: K;
|
|
value: BytesToStrings<StorageAccessType[K]>;
|
|
};
|
|
export type MessageType = MessageAttributesType;
|
|
export type MessageTypeUnhydrated = {
|
|
json: string;
|
|
};
|
|
|
|
export type PreKeyIdType = `${ServiceIdString}:${number}`;
|
|
export type KyberPreKeyType = {
|
|
id: PreKeyIdType;
|
|
|
|
createdAt: number;
|
|
data: Uint8Array;
|
|
isConfirmed: boolean;
|
|
isLastResort: boolean;
|
|
keyId: number;
|
|
ourServiceId: ServiceIdString;
|
|
};
|
|
export type StoredKyberPreKeyType = KyberPreKeyType & {
|
|
data: string;
|
|
};
|
|
export type PreKeyType = {
|
|
id: PreKeyIdType;
|
|
|
|
createdAt: number;
|
|
keyId: number;
|
|
ourServiceId: ServiceIdString;
|
|
privateKey: Uint8Array;
|
|
publicKey: Uint8Array;
|
|
};
|
|
|
|
export type StoredPreKeyType = PreKeyType & {
|
|
privateKey: string;
|
|
publicKey: string;
|
|
};
|
|
export type ServerSearchResultMessageType = {
|
|
json: string;
|
|
|
|
// If the FTS matches text in message.body, snippet will be populated
|
|
ftsSnippet: string | null;
|
|
|
|
// Otherwise, a matching mention will be returned
|
|
mentionAci: string | null;
|
|
mentionStart: number | null;
|
|
mentionLength: number | null;
|
|
};
|
|
export type ClientSearchResultMessageType = MessageType & {
|
|
json: string;
|
|
bodyRanges: ReadonlyArray<RawBodyRange>;
|
|
snippet: string;
|
|
};
|
|
|
|
export type SentProtoType = {
|
|
contentHint: number;
|
|
proto: Uint8Array;
|
|
timestamp: number;
|
|
urgent: boolean;
|
|
hasPniSignatureMessage: boolean;
|
|
};
|
|
export type SentProtoWithMessageIdsType = SentProtoType & {
|
|
messageIds: Array<string>;
|
|
};
|
|
export type SentRecipientsType = Record<ServiceIdString, Array<number>>;
|
|
export type SentMessagesType = Array<string>;
|
|
|
|
// These two are for test only
|
|
export type SentRecipientsDBType = {
|
|
payloadId: number;
|
|
recipientServiceId: ServiceIdString;
|
|
deviceId: number;
|
|
};
|
|
export type SentMessageDBType = {
|
|
payloadId: number;
|
|
messageId: string;
|
|
};
|
|
|
|
export type SenderKeyType = {
|
|
// Primary key
|
|
id: `${QualifiedAddressStringType}--${string}`;
|
|
// These two are combined into one string to give us the final id
|
|
senderId: string;
|
|
distributionId: string;
|
|
// Raw data to serialize/deserialize into signal-client SenderKeyRecord
|
|
data: Uint8Array;
|
|
lastUpdatedDate: number;
|
|
};
|
|
export type SenderKeyIdType = SenderKeyType['id'];
|
|
export type SessionType = {
|
|
id: QualifiedAddressStringType;
|
|
ourServiceId: ServiceIdString;
|
|
serviceId: ServiceIdString;
|
|
conversationId: string;
|
|
deviceId: number;
|
|
record: string;
|
|
version?: number;
|
|
};
|
|
export type SessionIdType = SessionType['id'];
|
|
export type SignedPreKeyType = {
|
|
confirmed: boolean;
|
|
created_at: number;
|
|
ourServiceId: ServiceIdString;
|
|
id: `${ServiceIdString}:${number}`;
|
|
keyId: number;
|
|
privateKey: Uint8Array;
|
|
publicKey: Uint8Array;
|
|
};
|
|
export type StoredSignedPreKeyType = {
|
|
confirmed: boolean;
|
|
created_at: number;
|
|
ourServiceId: ServiceIdString;
|
|
id: `${ServiceIdString}:${number}`;
|
|
keyId: number;
|
|
privateKey: string;
|
|
publicKey: string;
|
|
};
|
|
export type SignedPreKeyIdType = SignedPreKeyType['id'];
|
|
|
|
export type StickerType = Readonly<{
|
|
id: number;
|
|
packId: string;
|
|
|
|
emoji?: string;
|
|
isCoverOnly: boolean;
|
|
lastUsed?: number;
|
|
path: string;
|
|
|
|
width: number;
|
|
height: number;
|
|
|
|
version: 1 | 2;
|
|
localKey?: string;
|
|
size?: number;
|
|
}>;
|
|
|
|
export const StickerPackStatuses = [
|
|
'known',
|
|
'ephemeral',
|
|
'downloaded',
|
|
'installed',
|
|
'pending',
|
|
'error',
|
|
] as const;
|
|
|
|
export type StickerPackStatusType = (typeof StickerPackStatuses)[number];
|
|
|
|
export type StorageServiceFieldsType = Readonly<{
|
|
storageID?: string;
|
|
storageVersion?: number;
|
|
storageUnknownFields?: Uint8Array | null;
|
|
storageNeedsSync: boolean;
|
|
}>;
|
|
|
|
export type InstalledStickerPackType = Readonly<{
|
|
id: string;
|
|
key: string;
|
|
|
|
uninstalledAt?: undefined;
|
|
position?: number | null;
|
|
}> &
|
|
StorageServiceFieldsType;
|
|
|
|
export type UninstalledStickerPackType = Readonly<{
|
|
id: string;
|
|
key?: undefined;
|
|
|
|
uninstalledAt: number;
|
|
position?: undefined;
|
|
}> &
|
|
StorageServiceFieldsType;
|
|
|
|
export type StickerPackInfoType =
|
|
| InstalledStickerPackType
|
|
| UninstalledStickerPackType;
|
|
|
|
export type StickerPackType = InstalledStickerPackType &
|
|
Readonly<{
|
|
attemptedStatus?: 'downloaded' | 'installed' | 'ephemeral';
|
|
author: string;
|
|
coverStickerId: number;
|
|
createdAt: number;
|
|
downloadAttempts: number;
|
|
installedAt?: number;
|
|
lastUsed?: number;
|
|
status: StickerPackStatusType;
|
|
stickerCount: number;
|
|
stickers: Record<string, StickerType>;
|
|
title: string;
|
|
}>;
|
|
|
|
export type UnprocessedType = {
|
|
id: string;
|
|
timestamp: number;
|
|
receivedAtCounter: number | null;
|
|
version: number;
|
|
attempts: number;
|
|
envelope?: string;
|
|
|
|
messageAgeSec?: number;
|
|
source?: string;
|
|
sourceServiceId?: ServiceIdString;
|
|
sourceDevice?: number;
|
|
destinationServiceId?: ServiceIdString;
|
|
updatedPni?: PniString;
|
|
serverGuid?: string;
|
|
serverTimestamp?: number;
|
|
decrypted?: string;
|
|
urgent?: boolean;
|
|
story?: boolean;
|
|
reportingToken?: string;
|
|
};
|
|
|
|
export type UnprocessedUpdateType = {
|
|
source?: string;
|
|
sourceServiceId?: ServiceIdString;
|
|
sourceDevice?: number;
|
|
serverGuid?: string;
|
|
serverTimestamp?: number;
|
|
decrypted?: string;
|
|
};
|
|
|
|
export type ConversationMessageStatsType = {
|
|
activity?: MessageType;
|
|
preview?: MessageType;
|
|
hasUserInitiatedMessages: boolean;
|
|
};
|
|
|
|
export type DeleteSentProtoRecipientOptionsType = Readonly<{
|
|
timestamp: number;
|
|
recipientServiceId: ServiceIdString;
|
|
deviceId: number;
|
|
}>;
|
|
|
|
export type DeleteSentProtoRecipientResultType = Readonly<{
|
|
successfulPhoneNumberShares: ReadonlyArray<ServiceIdString>;
|
|
}>;
|
|
|
|
export type StoryDistributionType = Readonly<{
|
|
id: StoryDistributionIdString;
|
|
name: string;
|
|
deletedAtTimestamp?: number;
|
|
allowsReplies: boolean;
|
|
isBlockList: boolean;
|
|
senderKeyInfo: SenderKeyInfoType | undefined;
|
|
}> &
|
|
StorageServiceFieldsType;
|
|
export type StoryDistributionMemberType = Readonly<{
|
|
listId: StoryDistributionIdString;
|
|
serviceId: ServiceIdString;
|
|
}>;
|
|
export type StoryDistributionWithMembersType = Readonly<
|
|
{
|
|
members: Array<ServiceIdString>;
|
|
} & StoryDistributionType
|
|
>;
|
|
|
|
export type StoryReadType = Readonly<{
|
|
authorId: ServiceIdString;
|
|
conversationId: string;
|
|
storyId: string;
|
|
storyReadDate: number;
|
|
}>;
|
|
|
|
export type ReactionResultType = Pick<
|
|
ReactionType,
|
|
'targetAuthorAci' | 'targetTimestamp' | 'messageId'
|
|
> & { rowid: number };
|
|
|
|
export type GetUnreadByConversationAndMarkReadResultType = Array<
|
|
{ originalReadStatus: ReadStatus | undefined } & Pick<
|
|
MessageType,
|
|
| 'id'
|
|
| 'source'
|
|
| 'sourceServiceId'
|
|
| 'sent_at'
|
|
| 'type'
|
|
| 'readStatus'
|
|
| 'seenStatus'
|
|
| 'expirationStartTimestamp'
|
|
>
|
|
>;
|
|
|
|
export type GetConversationRangeCenteredOnMessageResultType<Message> =
|
|
Readonly<{
|
|
older: Array<Message>;
|
|
newer: Array<Message>;
|
|
metrics: ConversationMetricsType;
|
|
}>;
|
|
|
|
export type MessageCursorType = Readonly<{
|
|
done: boolean;
|
|
runId: string;
|
|
count: number;
|
|
}>;
|
|
|
|
export type MessageAttachmentsCursorType = MessageCursorType &
|
|
Readonly<{
|
|
__message_attachments_cursor: never;
|
|
}>;
|
|
|
|
export type GetKnownMessageAttachmentsResultType = Readonly<{
|
|
cursor: MessageAttachmentsCursorType;
|
|
attachments: ReadonlyArray<string>;
|
|
downloads: ReadonlyArray<string>;
|
|
}>;
|
|
|
|
export type PageMessagesCursorType = MessageCursorType &
|
|
Readonly<{
|
|
__page_messages_cursor: never;
|
|
}>;
|
|
|
|
export type PageMessagesResultType = Readonly<{
|
|
cursor: PageMessagesCursorType;
|
|
messages: ReadonlyArray<MessageAttributesType>;
|
|
}>;
|
|
|
|
export type GetAllStoriesResultType = ReadonlyArray<
|
|
MessageType & {
|
|
hasReplies: boolean;
|
|
hasRepliesFromSelf: boolean;
|
|
}
|
|
>;
|
|
|
|
export type EditedMessageType = Readonly<{
|
|
conversationId: string;
|
|
messageId: string;
|
|
sentAt: number;
|
|
readStatus: MessageType['readStatus'];
|
|
}>;
|
|
|
|
export type BackupCdnMediaObjectType = {
|
|
mediaId: string;
|
|
cdnNumber: number;
|
|
sizeOnBackupCdn: number;
|
|
};
|
|
|
|
export type GetRecentStoryRepliesOptionsType = {
|
|
limit?: number;
|
|
messageId?: string;
|
|
receivedAt?: number;
|
|
sentAt?: number;
|
|
};
|
|
|
|
export enum AttachmentDownloadSource {
|
|
BACKUP_IMPORT = 'backup_import',
|
|
STANDARD = 'standard',
|
|
}
|
|
|
|
type ReadableInterface = {
|
|
close: () => void;
|
|
|
|
getSenderKeyById: (id: SenderKeyIdType) => SenderKeyType | undefined;
|
|
getAllSenderKeys: () => Array<SenderKeyType>;
|
|
|
|
getAllSentProtos: () => Array<SentProtoType>;
|
|
|
|
// Test-only
|
|
_getAllSentProtoRecipients: () => Array<SentRecipientsDBType>;
|
|
_getAllSentProtoMessageIds: () => Array<SentMessageDBType>;
|
|
|
|
getAllSessions: () => Array<SessionType>;
|
|
|
|
getConversationCount: () => number;
|
|
getConversationById: (id: string) => ConversationType | undefined;
|
|
|
|
getAllConversations: () => Array<ConversationType>;
|
|
getAllConversationIds: () => Array<string>;
|
|
getAllGroupsInvolvingServiceId: (
|
|
serviceId: ServiceIdString
|
|
) => Array<ConversationType>;
|
|
|
|
getGroupSendCombinedEndorsementExpiration: (groupId: string) => number | null;
|
|
getGroupSendEndorsementsData: (
|
|
groupId: string
|
|
) => GroupSendEndorsementsData | null;
|
|
getGroupSendMemberEndorsement: (
|
|
groupId: string,
|
|
memberAci: AciString
|
|
) => GroupSendMemberEndorsementRecord | null;
|
|
|
|
getMessageCount: (conversationId?: string) => number;
|
|
getStoryCount: (conversationId: string) => number;
|
|
|
|
pageMessages: (cursor?: PageMessagesCursorType) => PageMessagesResultType;
|
|
finishPageMessages: (cursor: PageMessagesCursorType) => void;
|
|
|
|
getTotalUnreadForConversation: (
|
|
conversationId: string,
|
|
options: {
|
|
storyId: string | undefined;
|
|
includeStoryReplies: boolean;
|
|
}
|
|
) => number;
|
|
getTotalUnreadMentionsOfMeForConversation: (
|
|
conversationId: string,
|
|
options: {
|
|
storyId?: string;
|
|
includeStoryReplies: boolean;
|
|
}
|
|
) => number;
|
|
getOldestUnreadMentionOfMeForConversation(
|
|
conversationId: string,
|
|
options: {
|
|
storyId?: string;
|
|
includeStoryReplies: boolean;
|
|
}
|
|
): MessageMetricsType | undefined;
|
|
|
|
getReactionByTimestamp: (
|
|
fromId: string,
|
|
timestamp: number
|
|
) => ReactionType | undefined;
|
|
_getAllReactions: () => Array<ReactionType>;
|
|
|
|
getMessageBySender: (options: {
|
|
source?: string;
|
|
sourceServiceId?: ServiceIdString;
|
|
sourceDevice?: number;
|
|
sent_at: number;
|
|
}) => MessageType | undefined;
|
|
getMessageById: (id: string) => MessageType | undefined;
|
|
getMessagesById: (messageIds: ReadonlyArray<string>) => Array<MessageType>;
|
|
_getAllMessages: () => Array<MessageType>;
|
|
_getAllEditedMessages: () => Array<{ messageId: string; sentAt: number }>;
|
|
getAllMessageIds: () => Array<string>;
|
|
getMessagesBySentAt: (sentAt: number) => Array<MessageType>;
|
|
getExpiredMessages: () => Array<MessageType>;
|
|
getMessagesUnexpectedlyMissingExpirationStartTimestamp: () => Array<MessageType>;
|
|
getSoonestMessageExpiry: () => undefined | number;
|
|
getNextTapToViewMessageTimestampToAgeOut: () => undefined | number;
|
|
getTapToViewMessagesNeedingErase: () => Array<MessageType>;
|
|
// getOlderMessagesByConversation is JSON on server, full message on Client
|
|
getAllStories: (options: {
|
|
conversationId?: string;
|
|
sourceServiceId?: ServiceIdString;
|
|
}) => GetAllStoriesResultType;
|
|
// getNewerMessagesByConversation is JSON on server, full message on Client
|
|
getMessageMetricsForConversation: (options: {
|
|
conversationId: string;
|
|
storyId?: string;
|
|
includeStoryReplies: boolean;
|
|
}) => ConversationMetricsType;
|
|
// getConversationRangeCenteredOnMessage is JSON on server, full message on client
|
|
getConversationMessageStats: (options: {
|
|
conversationId: string;
|
|
includeStoryReplies: boolean;
|
|
}) => ConversationMessageStatsType;
|
|
getLastConversationMessage(options: {
|
|
conversationId: string;
|
|
}): MessageType | undefined;
|
|
getAllCallHistory: () => ReadonlyArray<CallHistoryDetails>;
|
|
getCallHistoryUnreadCount(): number;
|
|
getCallHistoryMessageByCallId(options: {
|
|
conversationId: string;
|
|
callId: string;
|
|
}): MessageType | undefined;
|
|
getCallHistory(
|
|
callId: string,
|
|
peerId: ServiceIdString | string
|
|
): CallHistoryDetails | undefined;
|
|
getCallHistoryGroupsCount(filter: CallHistoryFilter): number;
|
|
getCallHistoryGroups(
|
|
filter: CallHistoryFilter,
|
|
pagination: CallHistoryPagination
|
|
): Array<CallHistoryGroup>;
|
|
hasGroupCallHistoryMessage: (
|
|
conversationId: string,
|
|
eraId: string
|
|
) => boolean;
|
|
callLinkExists(roomId: string): boolean;
|
|
getAllCallLinks: () => ReadonlyArray<CallLinkType>;
|
|
getCallLinkByRoomId: (roomId: string) => CallLinkType | undefined;
|
|
getCallLinkRecordByRoomId: (roomId: string) => CallLinkRecord | undefined;
|
|
getAllCallLinkRecordsWithAdminKey(): ReadonlyArray<CallLinkRecord>;
|
|
getAllMarkedDeletedCallLinkRoomIds(): ReadonlyArray<string>;
|
|
getMessagesBetween: (
|
|
conversationId: string,
|
|
options: GetMessagesBetweenOptions
|
|
) => Array<string>;
|
|
getNearbyMessageFromDeletedSet: (
|
|
options: GetNearbyMessageFromDeletedSetOptionsType
|
|
) => string | null;
|
|
getMostRecentAddressableMessages: (
|
|
conversationId: string,
|
|
limit?: number
|
|
) => Array<MessageType>;
|
|
getMostRecentAddressableNondisappearingMessages: (
|
|
conversationId: string,
|
|
limit?: number
|
|
) => Array<MessageType>;
|
|
|
|
getUnprocessedCount: () => number;
|
|
getUnprocessedById: (id: string) => UnprocessedType | undefined;
|
|
|
|
getAttachmentDownloadJob(
|
|
job: Pick<
|
|
AttachmentDownloadJobType,
|
|
'messageId' | 'attachmentType' | 'digest'
|
|
>
|
|
): AttachmentDownloadJobType;
|
|
|
|
getBackupCdnObjectMetadata: (
|
|
mediaId: string
|
|
) => BackupCdnMediaObjectType | undefined;
|
|
|
|
getStickerCount: () => number;
|
|
getAllStickerPacks: () => Array<StickerPackType>;
|
|
getInstalledStickerPacks: () => Array<StickerPackType>;
|
|
getUninstalledStickerPacks: () => Array<UninstalledStickerPackType>;
|
|
getStickerPackInfo: (packId: string) => StickerPackInfoType | undefined;
|
|
getAllStickers: () => Array<StickerType>;
|
|
getRecentStickers: (options?: { limit?: number }) => Array<StickerType>;
|
|
|
|
getRecentEmojis: (limit?: number) => Array<EmojiType>;
|
|
|
|
getAllBadges(): Array<BadgeType>;
|
|
|
|
_getAllStoryDistributions(): Array<StoryDistributionType>;
|
|
_getAllStoryDistributionMembers(): Array<StoryDistributionMemberType>;
|
|
getAllStoryDistributionsWithMembers(): Array<StoryDistributionWithMembersType>;
|
|
getStoryDistributionWithMembers(
|
|
id: string
|
|
): StoryDistributionWithMembersType | undefined;
|
|
|
|
_getAllStoryReads(): Array<StoryReadType>;
|
|
getLastStoryReadsForAuthor(options: {
|
|
authorId: ServiceIdString;
|
|
conversationId?: string;
|
|
limit?: number;
|
|
}): Array<StoryReadType>;
|
|
countStoryReadsByConversation(conversationId: string): number;
|
|
|
|
getMessagesNeedingUpgrade: (
|
|
limit: number,
|
|
options: { maxVersion: number }
|
|
) => Array<MessageType>;
|
|
getMessageServerGuidsForSpam: (conversationId: string) => Array<string>;
|
|
|
|
getJobsInQueue(queueType: string): Array<StoredJob>;
|
|
|
|
wasGroupCallRingPreviouslyCanceled(ringId: bigint): boolean;
|
|
|
|
getMaxMessageCounter(): number | undefined;
|
|
|
|
getStatisticsForLogging(): Record<string, string>;
|
|
getSizeOfPendingBackupAttachmentDownloadJobs(): number;
|
|
};
|
|
|
|
type WritableInterface = {
|
|
close: () => void;
|
|
|
|
removeIndexedDBFiles: () => void;
|
|
|
|
removeIdentityKeyById: (id: IdentityKeyIdType) => number;
|
|
removeAllIdentityKeys: () => number;
|
|
|
|
removeKyberPreKeyById: (id: PreKeyIdType | Array<PreKeyIdType>) => number;
|
|
removeKyberPreKeysByServiceId: (serviceId: ServiceIdString) => void;
|
|
removeAllKyberPreKeys: () => number;
|
|
|
|
removePreKeyById: (id: PreKeyIdType | Array<PreKeyIdType>) => number;
|
|
removePreKeysByServiceId: (serviceId: ServiceIdString) => void;
|
|
removeAllPreKeys: () => number;
|
|
|
|
removeSignedPreKeyById: (
|
|
id: SignedPreKeyIdType | Array<SignedPreKeyIdType>
|
|
) => number;
|
|
removeSignedPreKeysByServiceId: (serviceId: ServiceIdString) => void;
|
|
removeAllSignedPreKeys: () => number;
|
|
|
|
removeAllItems: () => number;
|
|
removeItemById: (id: ItemKeyType | Array<ItemKeyType>) => number;
|
|
|
|
createOrUpdateSenderKey: (key: SenderKeyType) => void;
|
|
removeAllSenderKeys: () => void;
|
|
removeSenderKeyById: (id: SenderKeyIdType) => void;
|
|
|
|
getSentProtoByRecipient: (options: {
|
|
now: number;
|
|
recipientServiceId: ServiceIdString;
|
|
timestamp: number;
|
|
}) => SentProtoWithMessageIdsType | undefined;
|
|
insertSentProto: (
|
|
proto: SentProtoType,
|
|
options: {
|
|
recipients: SentRecipientsType;
|
|
messageIds: SentMessagesType;
|
|
}
|
|
) => number;
|
|
deleteSentProtosOlderThan: (timestamp: number) => void;
|
|
deleteSentProtoByMessageId: (messageId: string) => void;
|
|
insertProtoRecipients: (options: {
|
|
id: number;
|
|
recipientServiceId: ServiceIdString;
|
|
deviceIds: Array<number>;
|
|
}) => void;
|
|
deleteSentProtoRecipient: (
|
|
options:
|
|
| DeleteSentProtoRecipientOptionsType
|
|
| ReadonlyArray<DeleteSentProtoRecipientOptionsType>
|
|
) => DeleteSentProtoRecipientResultType;
|
|
removeAllSentProtos: () => void;
|
|
|
|
createOrUpdateSession: (data: SessionType) => void;
|
|
createOrUpdateSessions: (array: Array<SessionType>) => void;
|
|
commitDecryptResult(options: {
|
|
senderKeys: Array<SenderKeyType>;
|
|
sessions: Array<SessionType>;
|
|
unprocessed: Array<UnprocessedType>;
|
|
}): void;
|
|
bulkAddSessions: (array: Array<SessionType>) => void;
|
|
removeSessionById: (id: SessionIdType) => number;
|
|
removeSessionsByConversation: (conversationId: string) => void;
|
|
removeSessionsByServiceId: (serviceId: ServiceIdString) => void;
|
|
removeAllSessions: () => number;
|
|
|
|
saveConversation: (data: ConversationType) => void;
|
|
saveConversations: (array: Array<ConversationType>) => void;
|
|
// updateConversation is a normal data method on Server, a sync batch-add on Client
|
|
updateConversations: (array: Array<ConversationType>) => void;
|
|
// removeConversation handles either one id or an array on Server, and one id on Client
|
|
_removeAllConversations: () => void;
|
|
updateAllConversationColors: (
|
|
conversationColor?: ConversationColorType,
|
|
customColorData?: {
|
|
id: string;
|
|
value: CustomColorType;
|
|
}
|
|
) => void;
|
|
removeAllProfileKeyCredentials: () => void;
|
|
|
|
replaceAllEndorsementsForGroup: (data: GroupSendEndorsementsData) => void;
|
|
deleteAllEndorsementsForGroup: (groupId: string) => void;
|
|
|
|
saveMessage: (
|
|
data: ReadonlyDeep<MessageType>,
|
|
options: {
|
|
jobToInsert?: StoredJob;
|
|
forceSave?: boolean;
|
|
ourAci: AciString;
|
|
}
|
|
) => string;
|
|
saveMessages: (
|
|
arrayOfMessages: ReadonlyArray<ReadonlyDeep<MessageType>>,
|
|
options: { forceSave?: boolean; ourAci: AciString }
|
|
) => Array<string>;
|
|
|
|
getUnreadByConversationAndMarkRead: (options: {
|
|
conversationId: string;
|
|
includeStoryReplies: boolean;
|
|
newestUnreadAt: number;
|
|
now?: number;
|
|
readAt?: number;
|
|
storyId?: string;
|
|
}) => GetUnreadByConversationAndMarkReadResultType;
|
|
getUnreadEditedMessagesAndMarkRead: (options: {
|
|
conversationId: string;
|
|
newestUnreadAt: number;
|
|
}) => GetUnreadByConversationAndMarkReadResultType;
|
|
getUnreadReactionsAndMarkRead: (options: {
|
|
conversationId: string;
|
|
newestUnreadAt: number;
|
|
storyId?: string;
|
|
}) => Array<ReactionResultType>;
|
|
markReactionAsRead: (
|
|
targetAuthorServiceId: ServiceIdString,
|
|
targetTimestamp: number
|
|
) => ReactionType | undefined;
|
|
removeReactionFromConversation: (reaction: {
|
|
emoji: string;
|
|
fromId: string;
|
|
targetAuthorServiceId: ServiceIdString;
|
|
targetTimestamp: number;
|
|
}) => void;
|
|
addReaction: (
|
|
reactionObj: ReactionType,
|
|
options: {
|
|
readStatus: ReactionReadStatus;
|
|
}
|
|
) => void;
|
|
_removeAllReactions: () => void;
|
|
_removeAllMessages: () => void;
|
|
incrementMessagesMigrationAttempts: (
|
|
messageIds: ReadonlyArray<string>
|
|
) => void;
|
|
|
|
clearCallHistory: (target: CallLogEventTarget) => ReadonlyArray<string>;
|
|
_removeAllCallHistory: () => void;
|
|
markCallHistoryDeleted: (callId: string) => void;
|
|
cleanupCallHistoryMessages: () => void;
|
|
markCallHistoryRead(callId: string): void;
|
|
markAllCallHistoryRead(target: CallLogEventTarget): number;
|
|
markAllCallHistoryReadInConversation(target: CallLogEventTarget): number;
|
|
saveCallHistory(callHistory: CallHistoryDetails): void;
|
|
markCallHistoryMissed(callIds: ReadonlyArray<string>): void;
|
|
getRecentStaleRingsAndMarkOlderMissed(): ReadonlyArray<MaybeStaleCallHistory>;
|
|
insertCallLink(callLink: CallLinkType): void;
|
|
updateCallLink(callLink: CallLinkType): void;
|
|
updateCallLinkAdminKeyByRoomId(roomId: string, adminKey: string): void;
|
|
updateCallLinkState(
|
|
roomId: string,
|
|
callLinkState: CallLinkStateType
|
|
): CallLinkType;
|
|
beginDeleteAllCallLinks(): void;
|
|
beginDeleteCallLink(roomId: string, options: DeleteCallLinkOptions): void;
|
|
finalizeDeleteCallLink(roomId: string): void;
|
|
_removeAllCallLinks(): void;
|
|
deleteCallLinkFromSync(roomId: string): void;
|
|
migrateConversationMessages: (obsoleteId: string, currentId: string) => void;
|
|
saveEditedMessage: (
|
|
mainMessage: ReadonlyDeep<MessageType>,
|
|
ourAci: AciString,
|
|
opts: ReadonlyDeep<EditedMessageType>
|
|
) => void;
|
|
saveEditedMessages: (
|
|
mainMessage: ReadonlyDeep<MessageType>,
|
|
ourAci: AciString,
|
|
history: ReadonlyArray<ReadonlyDeep<EditedMessageType>>
|
|
) => void;
|
|
|
|
removeSyncTaskById: (id: string) => void;
|
|
saveSyncTasks: (tasks: Array<SyncTaskType>) => void;
|
|
getAllSyncTasks: () => Array<SyncTaskType>;
|
|
|
|
getAllUnprocessedIds: () => Array<string>;
|
|
getUnprocessedByIdsAndIncrementAttempts: (
|
|
ids: ReadonlyArray<string>
|
|
) => Array<UnprocessedType>;
|
|
updateUnprocessedWithData: (id: string, data: UnprocessedUpdateType) => void;
|
|
updateUnprocessedsWithData: (
|
|
array: Array<{ id: string; data: UnprocessedUpdateType }>
|
|
) => void;
|
|
removeUnprocessed: (id: string | Array<string>) => void;
|
|
|
|
/** only for testing */
|
|
removeAllUnprocessed: () => void;
|
|
|
|
getNextAttachmentDownloadJobs: (options: {
|
|
limit: number;
|
|
prioritizeMessageIds?: Array<string>;
|
|
timestamp?: number;
|
|
}) => Array<AttachmentDownloadJobType>;
|
|
saveAttachmentDownloadJob: (job: AttachmentDownloadJobType) => void;
|
|
resetAttachmentDownloadActive: () => void;
|
|
removeAttachmentDownloadJob: (job: AttachmentDownloadJobType) => void;
|
|
removeAllBackupAttachmentDownloadJobs: () => void;
|
|
|
|
getNextAttachmentBackupJobs: (options: {
|
|
limit: number;
|
|
timestamp?: number;
|
|
}) => Array<AttachmentBackupJobType>;
|
|
saveAttachmentBackupJob: (job: AttachmentBackupJobType) => void;
|
|
markAllAttachmentBackupJobsInactive: () => void;
|
|
removeAttachmentBackupJob: (job: AttachmentBackupJobType) => void;
|
|
clearAllAttachmentBackupJobs: () => void;
|
|
|
|
clearAllBackupCdnObjectMetadata: () => void;
|
|
saveBackupCdnObjectMetadata: (
|
|
mediaObjects: Array<BackupCdnMediaObjectType>
|
|
) => void;
|
|
|
|
createOrUpdateStickerPack: (pack: StickerPackType) => void;
|
|
updateStickerPackStatus: (
|
|
id: string,
|
|
status: StickerPackStatusType,
|
|
options?: { timestamp: number }
|
|
) => void;
|
|
updateStickerPackInfo: (info: StickerPackInfoType) => void;
|
|
createOrUpdateSticker: (sticker: StickerType) => void;
|
|
createOrUpdateStickers: (sticker: ReadonlyArray<StickerType>) => void;
|
|
updateStickerLastUsed: (
|
|
packId: string,
|
|
stickerId: number,
|
|
lastUsed: number
|
|
) => void;
|
|
addStickerPackReference: (messageId: string, packId: string) => void;
|
|
deleteStickerPackReference: (
|
|
messageId: string,
|
|
packId: string
|
|
) => ReadonlyArray<string> | undefined;
|
|
deleteStickerPack: (packId: string) => Array<string>;
|
|
addUninstalledStickerPack: (pack: UninstalledStickerPackType) => void;
|
|
removeUninstalledStickerPack: (packId: string) => void;
|
|
installStickerPack: (packId: string, timestamp: number) => void;
|
|
uninstallStickerPack: (packId: string, timestamp: number) => void;
|
|
clearAllErrorStickerPackAttempts: () => void;
|
|
|
|
updateEmojiUsage: (shortName: string, timeUsed?: number) => void;
|
|
|
|
updateOrCreateBadges(badges: ReadonlyArray<BadgeType>): void;
|
|
badgeImageFileDownloaded(url: string, localPath: string): void;
|
|
|
|
_deleteAllStoryDistributions(): void;
|
|
createNewStoryDistribution(
|
|
distribution: StoryDistributionWithMembersType
|
|
): void;
|
|
modifyStoryDistribution(distribution: StoryDistributionType): void;
|
|
modifyStoryDistributionMembers(
|
|
listId: string,
|
|
options: {
|
|
toAdd: Array<ServiceIdString>;
|
|
toRemove: Array<ServiceIdString>;
|
|
}
|
|
): void;
|
|
modifyStoryDistributionWithMembers(
|
|
distribution: StoryDistributionType,
|
|
options: {
|
|
toAdd: Array<ServiceIdString>;
|
|
toRemove: Array<ServiceIdString>;
|
|
}
|
|
): void;
|
|
deleteStoryDistribution(id: StoryDistributionIdString): void;
|
|
|
|
_deleteAllStoryReads(): void;
|
|
addNewStoryRead(read: StoryReadType): void;
|
|
|
|
removeAll: () => void;
|
|
removeAllConfiguration: () => void;
|
|
eraseStorageServiceState: () => void;
|
|
|
|
insertJob(job: Readonly<StoredJob>): void;
|
|
deleteJob(id: string): void;
|
|
|
|
processGroupCallRingCancellation(ringId: bigint): void;
|
|
cleanExpiredGroupCallRingCancellations(): void;
|
|
};
|
|
|
|
// Adds a database argument
|
|
type AddReadonlyDB<I> = {
|
|
[Key in keyof I]: I[Key] extends (...args: infer Args) => infer R
|
|
? (db: ReadableDB, ...args: Args) => R
|
|
: never;
|
|
};
|
|
|
|
export type ServerReadableDirectInterface = ReadableInterface & {
|
|
// Differing signature on client/server
|
|
searchMessages: ({
|
|
query,
|
|
conversationId,
|
|
options,
|
|
contactServiceIdsMatchingQuery,
|
|
}: {
|
|
query: string;
|
|
conversationId?: string;
|
|
options?: { limit?: number };
|
|
contactServiceIdsMatchingQuery?: Array<ServiceIdString>;
|
|
}) => Array<ServerSearchResultMessageType>;
|
|
|
|
getRecentStoryReplies(
|
|
storyId: string,
|
|
options?: GetRecentStoryRepliesOptionsType
|
|
): Array<MessageTypeUnhydrated>;
|
|
getOlderMessagesByConversation: (
|
|
options: AdjacentMessagesByConversationOptionsType
|
|
) => Array<MessageTypeUnhydrated>;
|
|
getNewerMessagesByConversation: (
|
|
options: AdjacentMessagesByConversationOptionsType
|
|
) => Array<MessageTypeUnhydrated>;
|
|
getConversationRangeCenteredOnMessage: (
|
|
options: AdjacentMessagesByConversationOptionsType
|
|
) => GetConversationRangeCenteredOnMessageResultType<MessageTypeUnhydrated>;
|
|
|
|
getIdentityKeyById: (
|
|
id: IdentityKeyIdType
|
|
) => StoredIdentityKeyType | undefined;
|
|
getAllIdentityKeys: () => Array<StoredIdentityKeyType>;
|
|
|
|
getKyberPreKeyById: (id: PreKeyIdType) => StoredKyberPreKeyType | undefined;
|
|
getAllKyberPreKeys: () => Array<StoredKyberPreKeyType>;
|
|
|
|
getPreKeyById: (id: PreKeyIdType) => StoredPreKeyType | undefined;
|
|
getAllPreKeys: () => Array<StoredPreKeyType>;
|
|
|
|
getSignedPreKeyById: (
|
|
id: SignedPreKeyIdType
|
|
) => StoredSignedPreKeyType | undefined;
|
|
getAllSignedPreKeys: () => Array<StoredSignedPreKeyType>;
|
|
|
|
getItemById<K extends ItemKeyType>(id: K): StoredItemType<K> | undefined;
|
|
getAllItems: () => StoredAllItemsType;
|
|
|
|
// Server-only
|
|
|
|
getKnownMessageAttachments: (
|
|
cursor?: MessageAttachmentsCursorType
|
|
) => GetKnownMessageAttachmentsResultType;
|
|
finishGetKnownMessageAttachments: (
|
|
cursor: MessageAttachmentsCursorType
|
|
) => void;
|
|
getKnownDownloads: () => Array<string>;
|
|
getKnownConversationAttachments: () => Array<string>;
|
|
|
|
getAllBadgeImageFileLocalPaths: () => Set<string>;
|
|
};
|
|
export type ServerReadableInterface =
|
|
AddReadonlyDB<ServerReadableDirectInterface>;
|
|
|
|
// Adds a database argument
|
|
type AddWritableDB<I> = {
|
|
[Key in keyof I]: I[Key] extends (...args: infer Args) => infer R
|
|
? (db: WritableDB, ...args: Args) => R
|
|
: never;
|
|
};
|
|
|
|
export type ServerWritableDirectInterface = WritableInterface & {
|
|
// Differing signature on client/server
|
|
|
|
updateConversation: (data: ConversationType) => void;
|
|
removeConversation: (id: Array<string> | string) => void;
|
|
|
|
removeMessage: (id: string) => void;
|
|
removeMessages: (ids: ReadonlyArray<string>) => void;
|
|
|
|
createOrUpdateIdentityKey: (data: StoredIdentityKeyType) => void;
|
|
bulkAddIdentityKeys: (array: Array<StoredIdentityKeyType>) => void;
|
|
|
|
createOrUpdateKyberPreKey: (data: StoredKyberPreKeyType) => void;
|
|
bulkAddKyberPreKeys: (array: Array<StoredKyberPreKeyType>) => void;
|
|
|
|
createOrUpdatePreKey: (data: StoredPreKeyType) => void;
|
|
bulkAddPreKeys: (array: Array<StoredPreKeyType>) => void;
|
|
|
|
createOrUpdateSignedPreKey: (data: StoredSignedPreKeyType) => void;
|
|
bulkAddSignedPreKeys: (array: Array<StoredSignedPreKeyType>) => void;
|
|
|
|
createOrUpdateItem<K extends ItemKeyType>(data: StoredItemType<K>): void;
|
|
|
|
// Server-only
|
|
|
|
removeKnownStickers: (allStickers: ReadonlyArray<string>) => Array<string>;
|
|
removeKnownDraftAttachments: (
|
|
allStickers: ReadonlyArray<string>
|
|
) => Array<string>;
|
|
|
|
runCorruptionChecks: () => void;
|
|
};
|
|
|
|
export type ServerWritableInterface =
|
|
AddWritableDB<ServerWritableDirectInterface>;
|
|
|
|
// Makes sync calls - async
|
|
export type ClientInterfaceWrap<I> = {
|
|
[Key in keyof I]: I[Key] extends (...args: infer Args) => infer R
|
|
? (...args: Args) => Promise<R>
|
|
: never;
|
|
};
|
|
|
|
export type ClientOnlyReadableInterface = ClientInterfaceWrap<{
|
|
// Differing signature on client/server
|
|
searchMessages: ({
|
|
query,
|
|
conversationId,
|
|
options,
|
|
contactServiceIdsMatchingQuery,
|
|
}: {
|
|
query: string;
|
|
conversationId?: string;
|
|
options?: { limit?: number };
|
|
contactServiceIdsMatchingQuery?: Array<ServiceIdString>;
|
|
}) => Array<ClientSearchResultMessageType>;
|
|
|
|
getRecentStoryReplies(
|
|
storyId: string,
|
|
options?: GetRecentStoryRepliesOptionsType
|
|
): Array<MessageType>;
|
|
getOlderMessagesByConversation: (
|
|
options: AdjacentMessagesByConversationOptionsType
|
|
) => Array<MessageType>;
|
|
getNewerMessagesByConversation: (
|
|
options: AdjacentMessagesByConversationOptionsType
|
|
) => Array<MessageType>;
|
|
getConversationRangeCenteredOnMessage: (
|
|
options: AdjacentMessagesByConversationOptionsType
|
|
) => GetConversationRangeCenteredOnMessageResultType<MessageType>;
|
|
|
|
getIdentityKeyById: (id: IdentityKeyIdType) => IdentityKeyType | undefined;
|
|
getAllIdentityKeys: () => Array<IdentityKeyType>;
|
|
|
|
getKyberPreKeyById: (id: PreKeyIdType) => KyberPreKeyType | undefined;
|
|
getAllKyberPreKeys: () => Array<KyberPreKeyType>;
|
|
|
|
getPreKeyById: (id: PreKeyIdType) => PreKeyType | undefined;
|
|
getAllPreKeys: () => Array<PreKeyType>;
|
|
|
|
getSignedPreKeyById: (id: SignedPreKeyIdType) => SignedPreKeyType | undefined;
|
|
getAllSignedPreKeys: () => Array<SignedPreKeyType>;
|
|
|
|
getItemById<K extends ItemKeyType>(id: K): ItemType<K> | undefined;
|
|
getAllItems: () => AllItemsType;
|
|
}>;
|
|
|
|
export type ClientOnlyWritableInterface = ClientInterfaceWrap<{
|
|
// Differing signature on client/server
|
|
updateConversation: (data: ConversationType) => void;
|
|
removeConversation: (id: string) => void;
|
|
flushUpdateConversationBatcher: () => void;
|
|
|
|
removeMessage: (
|
|
id: string,
|
|
options: {
|
|
fromSync?: boolean;
|
|
singleProtoJobQueue: SingleProtoJobQueue;
|
|
}
|
|
) => void;
|
|
removeMessages: (
|
|
ids: ReadonlyArray<string>,
|
|
options: {
|
|
fromSync?: boolean;
|
|
singleProtoJobQueue: SingleProtoJobQueue;
|
|
}
|
|
) => void;
|
|
|
|
createOrUpdateIdentityKey: (data: IdentityKeyType) => void;
|
|
bulkAddIdentityKeys: (array: Array<IdentityKeyType>) => void;
|
|
|
|
createOrUpdateKyberPreKey: (data: KyberPreKeyType) => void;
|
|
bulkAddKyberPreKeys: (array: Array<KyberPreKeyType>) => void;
|
|
|
|
createOrUpdatePreKey: (data: PreKeyType) => void;
|
|
bulkAddPreKeys: (array: Array<PreKeyType>) => void;
|
|
|
|
createOrUpdateSignedPreKey: (data: SignedPreKeyType) => void;
|
|
bulkAddSignedPreKeys: (array: Array<SignedPreKeyType>) => void;
|
|
|
|
createOrUpdateItem<K extends ItemKeyType>(data: ItemType<K>): void;
|
|
|
|
// Client-side only
|
|
|
|
shutdown: () => void;
|
|
removeDB: () => void;
|
|
removeMessagesInConversation: (
|
|
conversationId: string,
|
|
options: {
|
|
fromSync?: boolean;
|
|
logId: string;
|
|
receivedAt?: number;
|
|
singleProtoJobQueue: SingleProtoJobQueue;
|
|
}
|
|
) => void;
|
|
removeOtherData: () => void;
|
|
cleanupOrphanedAttachments: () => void;
|
|
ensureFilePermissions: () => void;
|
|
}>;
|
|
|
|
export type ClientReadableInterface = ClientInterfaceWrap<ReadableInterface> &
|
|
ClientOnlyReadableInterface;
|
|
export type ClientWritableInterface = ClientInterfaceWrap<WritableInterface> &
|
|
ClientOnlyWritableInterface;
|