2022-02-11 21:38:52 +00:00
|
|
|
// Copyright 2021-2022 Signal Messenger, LLC
|
2021-03-18 17:09:27 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { isNil, sortBy } from 'lodash';
|
2021-05-21 19:53:05 +00:00
|
|
|
import PQueue from 'p-queue';
|
2021-03-18 17:09:27 +00:00
|
|
|
|
|
|
|
import * as log from './logging/log';
|
|
|
|
import { assert } from './util/assert';
|
2022-07-08 20:46:25 +00:00
|
|
|
import { sleep } from './util/sleep';
|
2021-03-18 17:09:27 +00:00
|
|
|
import { missingCaseError } from './util/missingCaseError';
|
|
|
|
import { isNormalNumber } from './util/isNormalNumber';
|
2021-05-21 19:53:05 +00:00
|
|
|
import { take } from './util/iterables';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ConversationModel } from './models/conversations';
|
|
|
|
import type { StorageInterface } from './types/Storage.d';
|
2022-07-08 20:46:25 +00:00
|
|
|
import * as Errors from './types/errors';
|
2022-02-11 21:38:52 +00:00
|
|
|
import { getProfile } from './util/getProfile';
|
2022-07-08 20:46:25 +00:00
|
|
|
import { MINUTE, HOUR, DAY, MONTH } from './util/durations';
|
2021-03-18 17:09:27 +00:00
|
|
|
|
|
|
|
const STORAGE_KEY = 'lastAttemptedToRefreshProfilesAt';
|
2022-07-08 20:46:25 +00:00
|
|
|
const MAX_AGE_TO_BE_CONSIDERED_ACTIVE = MONTH;
|
|
|
|
const MAX_AGE_TO_BE_CONSIDERED_RECENTLY_REFRESHED = DAY;
|
2021-03-18 17:09:27 +00:00
|
|
|
const MAX_CONVERSATIONS_TO_REFRESH = 50;
|
2022-07-08 20:46:25 +00:00
|
|
|
const MIN_ELAPSED_DURATION_TO_REFRESH_AGAIN = 12 * HOUR;
|
|
|
|
const MIN_REFRESH_DELAY = MINUTE;
|
|
|
|
|
|
|
|
export class RoutineProfileRefresher {
|
|
|
|
private interval: NodeJS.Timeout | undefined;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
private readonly options: {
|
|
|
|
getAllConversations: () => ReadonlyArray<ConversationModel>;
|
|
|
|
getOurConversationId: () => string | undefined;
|
|
|
|
storage: Pick<StorageInterface, 'get' | 'put'>;
|
|
|
|
}
|
|
|
|
) {}
|
|
|
|
|
|
|
|
public async start(): Promise<void> {
|
|
|
|
if (this.interval !== undefined) {
|
|
|
|
clearInterval(this.interval);
|
|
|
|
}
|
|
|
|
|
|
|
|
const { storage, getAllConversations, getOurConversationId } = this.options;
|
|
|
|
|
|
|
|
// eslint-disable-next-line no-constant-condition
|
|
|
|
while (true) {
|
|
|
|
const refreshInMs = timeUntilNextRefresh(storage);
|
|
|
|
|
|
|
|
log.info(`routineProfileRefresh: waiting for ${refreshInMs}ms`);
|
|
|
|
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
await sleep(refreshInMs);
|
|
|
|
|
|
|
|
const ourConversationId = getOurConversationId();
|
|
|
|
if (!ourConversationId) {
|
|
|
|
log.warn('routineProfileRefresh: missing our conversation id');
|
|
|
|
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
await sleep(MIN_REFRESH_DELAY);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
await routineProfileRefresh({
|
|
|
|
allConversations: getAllConversations(),
|
|
|
|
ourConversationId,
|
|
|
|
storage,
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
log.error('routineProfileRefresh: failure', Errors.toLogFormat(error));
|
|
|
|
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
await sleep(MIN_REFRESH_DELAY);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-18 17:09:27 +00:00
|
|
|
|
|
|
|
export async function routineProfileRefresh({
|
|
|
|
allConversations,
|
|
|
|
ourConversationId,
|
|
|
|
storage,
|
2022-02-11 21:38:52 +00:00
|
|
|
|
|
|
|
// Only for tests
|
|
|
|
getProfileFn = getProfile,
|
2021-03-18 17:09:27 +00:00
|
|
|
}: {
|
2022-07-08 20:46:25 +00:00
|
|
|
allConversations: ReadonlyArray<ConversationModel>;
|
2021-03-18 17:09:27 +00:00
|
|
|
ourConversationId: string;
|
2021-06-15 00:09:37 +00:00
|
|
|
storage: Pick<StorageInterface, 'get' | 'put'>;
|
2022-02-11 21:38:52 +00:00
|
|
|
getProfileFn?: typeof getProfile;
|
2021-03-18 17:09:27 +00:00
|
|
|
}): Promise<void> {
|
|
|
|
log.info('routineProfileRefresh: starting');
|
|
|
|
|
2022-07-08 20:46:25 +00:00
|
|
|
const refreshInMs = timeUntilNextRefresh(storage);
|
|
|
|
if (refreshInMs > 0) {
|
2021-03-18 17:09:27 +00:00
|
|
|
log.info('routineProfileRefresh: too soon to refresh. Doing nothing');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
log.info('routineProfileRefresh: updating last refresh time');
|
|
|
|
await storage.put(STORAGE_KEY, Date.now());
|
|
|
|
|
|
|
|
const conversationsToRefresh = getConversationsToRefresh(
|
|
|
|
allConversations,
|
|
|
|
ourConversationId
|
|
|
|
);
|
|
|
|
|
|
|
|
log.info('routineProfileRefresh: starting to refresh conversations');
|
|
|
|
|
|
|
|
let totalCount = 0;
|
|
|
|
let successCount = 0;
|
2021-05-21 19:53:05 +00:00
|
|
|
|
|
|
|
async function refreshConversation(
|
|
|
|
conversation: ConversationModel
|
|
|
|
): Promise<void> {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2021-05-21 19:53:05 +00:00
|
|
|
`routineProfileRefresh: refreshing profile for ${conversation.idForLogging()}`
|
|
|
|
);
|
|
|
|
|
|
|
|
totalCount += 1;
|
|
|
|
try {
|
2022-02-11 21:38:52 +00:00
|
|
|
await getProfileFn(conversation.get('uuid'), conversation.get('e164'));
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2021-05-21 19:53:05 +00:00
|
|
|
`routineProfileRefresh: refreshed profile for ${conversation.idForLogging()}`
|
|
|
|
);
|
|
|
|
successCount += 1;
|
|
|
|
} catch (err) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(
|
2021-05-21 19:53:05 +00:00
|
|
|
`routineProfileRefresh: refreshed profile for ${conversation.idForLogging()}`,
|
|
|
|
err?.stack || err
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-23 22:01:03 +00:00
|
|
|
const refreshQueue = new PQueue({
|
|
|
|
concurrency: 5,
|
2022-06-27 16:46:43 +00:00
|
|
|
timeout: MINUTE * 30,
|
2021-11-23 22:01:03 +00:00
|
|
|
throwOnTimeout: true,
|
|
|
|
});
|
2021-05-21 19:53:05 +00:00
|
|
|
for (const conversation of conversationsToRefresh) {
|
|
|
|
refreshQueue.add(() => refreshConversation(conversation));
|
|
|
|
}
|
|
|
|
await refreshQueue.onIdle();
|
2021-03-18 17:09:27 +00:00
|
|
|
|
|
|
|
log.info(
|
|
|
|
`routineProfileRefresh: successfully refreshed ${successCount} out of ${totalCount} conversation(s)`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-07-08 20:46:25 +00:00
|
|
|
function timeUntilNextRefresh(storage: Pick<StorageInterface, 'get'>): number {
|
2021-03-18 17:09:27 +00:00
|
|
|
const storedValue = storage.get(STORAGE_KEY);
|
|
|
|
|
|
|
|
if (isNil(storedValue)) {
|
2022-07-08 20:46:25 +00:00
|
|
|
return 0;
|
2021-03-18 17:09:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isNormalNumber(storedValue)) {
|
2022-07-08 20:46:25 +00:00
|
|
|
const planned = storedValue + MIN_ELAPSED_DURATION_TO_REFRESH_AGAIN;
|
|
|
|
const now = Date.now();
|
|
|
|
return Math.max(0, planned - now);
|
2021-03-18 17:09:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(
|
|
|
|
false,
|
|
|
|
`An invalid value was stored in ${STORAGE_KEY}; treating it as nil`
|
|
|
|
);
|
2022-07-08 20:46:25 +00:00
|
|
|
return 0;
|
2021-03-18 17:09:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function getConversationsToRefresh(
|
|
|
|
conversations: ReadonlyArray<ConversationModel>,
|
|
|
|
ourConversationId: string
|
|
|
|
): Iterable<ConversationModel> {
|
|
|
|
const filteredConversations = getFilteredConversations(
|
|
|
|
conversations,
|
|
|
|
ourConversationId
|
|
|
|
);
|
|
|
|
return take(filteredConversations, MAX_CONVERSATIONS_TO_REFRESH);
|
|
|
|
}
|
|
|
|
|
|
|
|
function* getFilteredConversations(
|
|
|
|
conversations: ReadonlyArray<ConversationModel>,
|
|
|
|
ourConversationId: string
|
|
|
|
): Iterable<ConversationModel> {
|
|
|
|
const sorted = sortBy(conversations, c => c.get('active_at'));
|
|
|
|
|
|
|
|
const conversationIdsSeen = new Set<string>([ourConversationId]);
|
|
|
|
|
|
|
|
for (const conversation of sorted) {
|
|
|
|
const type = conversation.get('type');
|
|
|
|
switch (type) {
|
|
|
|
case 'private':
|
2022-07-08 20:46:25 +00:00
|
|
|
if (
|
|
|
|
conversation.hasProfileKeyCredentialExpired() &&
|
|
|
|
(conversation.id === ourConversationId ||
|
|
|
|
!conversationIdsSeen.has(conversation.id))
|
|
|
|
) {
|
|
|
|
conversation.set({
|
|
|
|
profileKeyCredential: null,
|
|
|
|
profileKeyCredentialExpiration: null,
|
|
|
|
});
|
|
|
|
conversationIdsSeen.add(conversation.id);
|
|
|
|
yield conversation;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-03-18 17:09:27 +00:00
|
|
|
if (
|
|
|
|
!conversationIdsSeen.has(conversation.id) &&
|
|
|
|
isConversationActive(conversation) &&
|
|
|
|
!hasRefreshedProfileRecently(conversation)
|
|
|
|
) {
|
|
|
|
conversationIdsSeen.add(conversation.id);
|
|
|
|
yield conversation;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'group':
|
|
|
|
for (const member of conversation.getMembers()) {
|
|
|
|
if (
|
|
|
|
!conversationIdsSeen.has(member.id) &&
|
|
|
|
!hasRefreshedProfileRecently(member)
|
|
|
|
) {
|
|
|
|
conversationIdsSeen.add(member.id);
|
|
|
|
yield member;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw missingCaseError(type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function isConversationActive(
|
|
|
|
conversation: Readonly<ConversationModel>
|
|
|
|
): boolean {
|
|
|
|
const activeAt = conversation.get('active_at');
|
|
|
|
return (
|
|
|
|
isNormalNumber(activeAt) &&
|
|
|
|
activeAt + MAX_AGE_TO_BE_CONSIDERED_ACTIVE > Date.now()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function hasRefreshedProfileRecently(
|
|
|
|
conversation: Readonly<ConversationModel>
|
|
|
|
): boolean {
|
|
|
|
const profileLastFetchedAt = conversation.get('profileLastFetchedAt');
|
|
|
|
return (
|
|
|
|
isNormalNumber(profileLastFetchedAt) &&
|
|
|
|
profileLastFetchedAt + MAX_AGE_TO_BE_CONSIDERED_RECENTLY_REFRESHED >
|
|
|
|
Date.now()
|
|
|
|
);
|
|
|
|
}
|