2022-02-25 18:37:15 +00:00
|
|
|
// Copyright 2021-2022 Signal Messenger, LLC
|
2021-05-06 00:09:29 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
// `ChallengeHandler` is responsible for:
|
|
|
|
// 1. tracking the messages that failed to send with 428 error and could be
|
|
|
|
// retried when user solves the challenge
|
|
|
|
// 2. presenting the challenge to user and sending the challenge response back
|
|
|
|
// to the server
|
|
|
|
//
|
|
|
|
// The tracked messages are persisted in the database, and are imported back
|
|
|
|
// to the `ChallengeHandler` on `.load()` call (from `ts/background.ts`). They
|
|
|
|
// are not immediately retried, however, until `.onOnline()` is called from
|
|
|
|
// when we are actually online.
|
|
|
|
|
|
|
|
import { assert } from './util/assert';
|
|
|
|
import { isOlderThan } from './util/timestamp';
|
|
|
|
import { parseRetryAfter } from './util/parseRetryAfter';
|
2022-02-25 18:37:15 +00:00
|
|
|
import { clearTimeoutIfNecessary } from './util/clearTimeoutIfNecessary';
|
2021-05-06 00:09:29 +00:00
|
|
|
import { getEnvironment, Environment } from './environment';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { StorageInterface } from './types/Storage.d';
|
2021-09-22 00:58:03 +00:00
|
|
|
import { HTTPError } from './textsecure/Errors';
|
2022-03-21 21:19:37 +00:00
|
|
|
import type { SendMessageChallengeData } from './textsecure/Errors';
|
2021-09-17 18:27:53 +00:00
|
|
|
import * as log from './logging/log';
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
export type ChallengeResponse = {
|
|
|
|
readonly captcha: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type IPCRequest = {
|
|
|
|
readonly seq: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type IPCResponse = {
|
|
|
|
readonly seq: number;
|
|
|
|
readonly data: ChallengeResponse;
|
|
|
|
};
|
|
|
|
|
|
|
|
type Handler = {
|
|
|
|
readonly token: string | undefined;
|
|
|
|
|
|
|
|
resolve(response: ChallengeResponse): void;
|
|
|
|
reject(error: Error): void;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type ChallengeData = {
|
|
|
|
readonly type: 'recaptcha';
|
|
|
|
readonly token: string;
|
|
|
|
readonly captcha: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type Options = {
|
2021-06-15 00:09:37 +00:00
|
|
|
readonly storage: Pick<StorageInterface, 'get' | 'put'>;
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
requestChallenge(request: IPCRequest): void;
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
startQueue(conversationId: string): void;
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
sendChallengeResponse(data: ChallengeData): Promise<void>;
|
|
|
|
|
|
|
|
setChallengeStatus(challengeStatus: 'idle' | 'required' | 'pending'): void;
|
|
|
|
|
|
|
|
onChallengeSolved(): void;
|
|
|
|
onChallengeFailed(retryAfter?: number): void;
|
|
|
|
|
|
|
|
expireAfter?: number;
|
|
|
|
};
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
export const STORAGE_KEY = 'challenge:conversations';
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
export type RegisteredChallengeType = Readonly<{
|
|
|
|
conversationId: string;
|
|
|
|
createdAt: number;
|
|
|
|
retryAt: number;
|
|
|
|
token?: string;
|
|
|
|
}>;
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
const DEFAULT_EXPIRE_AFTER = 24 * 3600 * 1000; // one day
|
|
|
|
const CAPTCHA_URL = 'https://signalcaptchas.org/challenge/generate.html';
|
|
|
|
const CAPTCHA_STAGING_URL =
|
|
|
|
'https://signalcaptchas.org/staging/challenge/generate.html';
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
function shouldStartQueue(registered: RegisteredChallengeType): boolean {
|
|
|
|
if (!registered.retryAt || registered.retryAt <= Date.now()) {
|
2021-05-06 00:09:29 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getChallengeURL(): string {
|
|
|
|
if (getEnvironment() === Environment.Staging) {
|
|
|
|
return CAPTCHA_STAGING_URL;
|
|
|
|
}
|
|
|
|
return CAPTCHA_URL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note that even though this is a class - only one instance of
|
|
|
|
// `ChallengeHandler` should be in memory at the same time because they could
|
|
|
|
// overwrite each others storage data.
|
|
|
|
export class ChallengeHandler {
|
2022-03-21 21:19:37 +00:00
|
|
|
private solving = 0;
|
|
|
|
|
2021-05-06 00:09:29 +00:00
|
|
|
private isLoaded = false;
|
|
|
|
|
|
|
|
private challengeToken: string | undefined;
|
|
|
|
|
|
|
|
private seq = 0;
|
|
|
|
|
|
|
|
private isOnline = false;
|
|
|
|
|
|
|
|
private readonly responseHandlers = new Map<number, Handler>();
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
private readonly registeredConversations = new Map<
|
|
|
|
string,
|
|
|
|
RegisteredChallengeType
|
|
|
|
>();
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
private readonly startTimers = new Map<string, NodeJS.Timeout>();
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
private readonly pendingStarts = new Set<string>();
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
constructor(private readonly options: Options) {}
|
|
|
|
|
|
|
|
public async load(): Promise<void> {
|
|
|
|
if (this.isLoaded) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.isLoaded = true;
|
2022-03-21 21:19:37 +00:00
|
|
|
const challenges: ReadonlyArray<RegisteredChallengeType> =
|
|
|
|
this.options.storage.get(STORAGE_KEY) || [];
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
log.info(`challenge: loading ${challenges.length} challenges`);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
await Promise.all(
|
2022-03-21 21:19:37 +00:00
|
|
|
challenges.map(async challenge => {
|
2021-05-06 00:09:29 +00:00
|
|
|
const expireAfter = this.options.expireAfter || DEFAULT_EXPIRE_AFTER;
|
2022-03-21 21:19:37 +00:00
|
|
|
if (isOlderThan(challenge.createdAt, expireAfter)) {
|
|
|
|
log.info(
|
|
|
|
`challenge: expired challenge for conversation ${challenge.conversationId}`
|
|
|
|
);
|
2021-05-06 00:09:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The initialization order is following:
|
|
|
|
//
|
|
|
|
// 1. `.load()` when the `window.storage` is ready
|
|
|
|
// 2. `.onOnline()` when we connected to the server
|
|
|
|
//
|
|
|
|
// Wait for `.onOnline()` to trigger the retries instead of triggering
|
|
|
|
// them here immediately (if the message is ready to be retried).
|
2022-03-21 21:19:37 +00:00
|
|
|
await this.register(challenge);
|
2021-05-06 00:09:29 +00:00
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public async onOffline(): Promise<void> {
|
|
|
|
this.isOnline = false;
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info('challenge: offline');
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public async onOnline(): Promise<void> {
|
|
|
|
this.isOnline = true;
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
const pending = Array.from(this.pendingStarts.values());
|
|
|
|
this.pendingStarts.clear();
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
log.info(`challenge: online, starting ${pending.length} queues`);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
// Start queues for challenges that matured while we were offline
|
|
|
|
await Promise.all(
|
|
|
|
pending.map(conversationId => this.startQueue(conversationId))
|
|
|
|
);
|
|
|
|
|
|
|
|
await this.startAllQueues();
|
|
|
|
}
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
public maybeSolve(conversationId: string): void {
|
|
|
|
const challenge = this.registeredConversations.get(conversationId);
|
|
|
|
if (!challenge) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.solving > 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (challenge.token) {
|
|
|
|
this.solve(challenge.token);
|
|
|
|
}
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public async register(
|
2022-03-21 21:19:37 +00:00
|
|
|
challenge: RegisteredChallengeType,
|
|
|
|
data?: SendMessageChallengeData
|
2021-05-06 00:09:29 +00:00
|
|
|
): Promise<void> {
|
2022-03-21 21:19:37 +00:00
|
|
|
const { conversationId } = challenge;
|
|
|
|
|
|
|
|
if (this.isRegistered(conversationId)) {
|
|
|
|
log.info(`challenge: conversation ${conversationId} already registered`);
|
2021-05-06 00:09:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
this.registeredConversations.set(conversationId, challenge);
|
2021-05-06 00:09:29 +00:00
|
|
|
await this.persist();
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
// Challenge is already retryable - start the queue
|
|
|
|
if (shouldStartQueue(challenge)) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2022-03-21 21:19:37 +00:00
|
|
|
`challenge: starting conversation ${conversationId} immediately`
|
2021-05-06 00:09:29 +00:00
|
|
|
);
|
2022-03-21 21:19:37 +00:00
|
|
|
await this.startQueue(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
const waitTime = Math.max(0, challenge.retryAt - Date.now());
|
|
|
|
const oldTimer = this.startTimers.get(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
if (oldTimer) {
|
2022-02-25 18:37:15 +00:00
|
|
|
clearTimeoutIfNecessary(oldTimer);
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
2022-03-21 21:19:37 +00:00
|
|
|
this.startTimers.set(
|
|
|
|
conversationId,
|
2021-05-06 00:09:29 +00:00
|
|
|
setTimeout(() => {
|
2022-03-21 21:19:37 +00:00
|
|
|
this.startTimers.delete(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
this.startQueue(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
}, waitTime)
|
|
|
|
);
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
log.info(`challenge: tracking ${conversationId} with waitTime=${waitTime}`);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
if (data && !data.options?.includes('recaptcha')) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(
|
2022-03-21 21:19:37 +00:00
|
|
|
`challenge: unexpected options ${JSON.stringify(data.options)}`
|
2021-05-06 00:09:29 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
if (!challenge.token) {
|
|
|
|
const dataString = JSON.stringify(data);
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(
|
2022-03-21 21:19:37 +00:00
|
|
|
`challenge: ${conversationId} is waiting; no token in data ${dataString}`
|
2021-05-06 00:09:29 +00:00
|
|
|
);
|
2022-03-21 21:19:37 +00:00
|
|
|
return;
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
2022-03-21 21:19:37 +00:00
|
|
|
|
|
|
|
this.solve(challenge.token);
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public onResponse(response: IPCResponse): void {
|
|
|
|
const handler = this.responseHandlers.get(response.seq);
|
|
|
|
if (!handler) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.responseHandlers.delete(response.seq);
|
|
|
|
handler.resolve(response.data);
|
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
public async unregister(conversationId: string): Promise<void> {
|
|
|
|
log.info(`challenge: unregistered conversation ${conversationId}`);
|
|
|
|
this.registeredConversations.delete(conversationId);
|
|
|
|
this.pendingStarts.delete(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
const timer = this.startTimers.get(conversationId);
|
|
|
|
this.startTimers.delete(conversationId);
|
2022-02-25 18:37:15 +00:00
|
|
|
clearTimeoutIfNecessary(timer);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
await this.persist();
|
|
|
|
}
|
|
|
|
|
2021-11-30 17:51:53 +00:00
|
|
|
public async requestCaptcha(token = ''): Promise<string> {
|
|
|
|
const request: IPCRequest = { seq: this.seq };
|
|
|
|
this.seq += 1;
|
|
|
|
|
|
|
|
this.options.requestChallenge(request);
|
|
|
|
|
|
|
|
const response = await new Promise<ChallengeResponse>((resolve, reject) => {
|
|
|
|
this.responseHandlers.set(request.seq, { token, resolve, reject });
|
|
|
|
});
|
|
|
|
|
|
|
|
return response.captcha;
|
|
|
|
}
|
|
|
|
|
2021-05-06 00:09:29 +00:00
|
|
|
private async persist(): Promise<void> {
|
|
|
|
assert(
|
|
|
|
this.isLoaded,
|
|
|
|
'ChallengeHandler has to be loaded before persisting new data'
|
|
|
|
);
|
|
|
|
await this.options.storage.put(
|
2022-03-21 21:19:37 +00:00
|
|
|
STORAGE_KEY,
|
|
|
|
Array.from(this.registeredConversations.values())
|
2021-05-06 00:09:29 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
public isRegistered(conversationId: string): boolean {
|
|
|
|
return this.registeredConversations.has(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
private startAllQueues({
|
|
|
|
force = false,
|
|
|
|
}: {
|
|
|
|
force?: boolean;
|
|
|
|
} = {}): void {
|
|
|
|
log.info(`challenge: startAllQueues force=${force}`);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
Array.from(this.registeredConversations.values())
|
|
|
|
.filter(challenge => force || shouldStartQueue(challenge))
|
|
|
|
.forEach(challenge => this.startQueue(challenge.conversationId));
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
private async startQueue(conversationId: string): Promise<void> {
|
2021-05-06 00:09:29 +00:00
|
|
|
if (!this.isOnline) {
|
2022-03-21 21:19:37 +00:00
|
|
|
this.pendingStarts.add(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
await this.unregister(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
if (this.registeredConversations.size === 0) {
|
|
|
|
this.options.setChallengeStatus('idle');
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
log.info(`startQueue: starting queue ${conversationId}`);
|
|
|
|
this.options.startQueue(conversationId);
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private async solve(token: string): Promise<void> {
|
2022-03-21 21:19:37 +00:00
|
|
|
this.solving += 1;
|
2021-05-06 00:09:29 +00:00
|
|
|
this.options.setChallengeStatus('required');
|
2021-11-30 17:51:53 +00:00
|
|
|
this.challengeToken = token;
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2021-11-30 17:51:53 +00:00
|
|
|
const captcha = await this.requestCaptcha(token);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
// Another `.solve()` has completed earlier than us
|
|
|
|
if (this.challengeToken === undefined) {
|
2022-03-21 21:19:37 +00:00
|
|
|
this.solving -= 1;
|
2021-05-06 00:09:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const lastToken = this.challengeToken;
|
|
|
|
this.challengeToken = undefined;
|
|
|
|
|
|
|
|
this.options.setChallengeStatus('pending');
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info('challenge: sending challenge to server');
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
await this.sendChallengeResponse({
|
|
|
|
type: 'recaptcha',
|
|
|
|
token: lastToken,
|
2021-11-30 17:51:53 +00:00
|
|
|
captcha,
|
2021-05-06 00:09:29 +00:00
|
|
|
});
|
|
|
|
} catch (error) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(`challenge: challenge failure, error: ${error && error.stack}`);
|
2021-05-06 00:09:29 +00:00
|
|
|
this.options.setChallengeStatus('required');
|
2022-03-21 21:19:37 +00:00
|
|
|
this.solving -= 1;
|
2021-05-06 00:09:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info('challenge: challenge success. force sending');
|
2021-05-06 00:09:29 +00:00
|
|
|
|
|
|
|
this.options.setChallengeStatus('idle');
|
|
|
|
|
2022-03-21 21:19:37 +00:00
|
|
|
this.startAllQueues({ force: true });
|
|
|
|
this.solving -= 1;
|
2021-05-06 00:09:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private async sendChallengeResponse(data: ChallengeData): Promise<void> {
|
|
|
|
try {
|
|
|
|
await this.options.sendChallengeResponse(data);
|
|
|
|
} catch (error) {
|
|
|
|
if (
|
2021-09-22 00:58:03 +00:00
|
|
|
!(error instanceof HTTPError) ||
|
2022-02-25 00:26:58 +00:00
|
|
|
!(error.code === 413 || error.code === 429) ||
|
2021-05-06 00:09:29 +00:00
|
|
|
!error.responseHeaders
|
|
|
|
) {
|
|
|
|
this.options.onChallengeFailed();
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
|
2021-09-02 22:31:21 +00:00
|
|
|
const retryAfter = parseRetryAfter(error.responseHeaders['retry-after']);
|
2021-05-06 00:09:29 +00:00
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`challenge: retry after ${retryAfter}ms`);
|
2021-05-06 00:09:29 +00:00
|
|
|
this.options.onChallengeFailed(retryAfter);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.options.onChallengeSolved();
|
|
|
|
}
|
|
|
|
}
|