2021-04-05 15:18:19 -07:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { parentPort } from 'worker_threads';
|
|
|
|
|
2021-09-16 14:54:06 -07:00
|
|
|
import type { LoggerType } from '../types/Logging';
|
|
|
|
import type {
|
|
|
|
WrappedWorkerRequest,
|
|
|
|
WrappedWorkerResponse,
|
|
|
|
WrappedWorkerLogEntry,
|
|
|
|
} from './main';
|
2024-07-22 11:16:33 -07:00
|
|
|
import type { WritableDB } from './Interface';
|
2024-08-12 12:54:24 -07:00
|
|
|
import { initialize, DataReader, DataWriter, removeDB } from './Server';
|
2023-10-11 01:19:11 +02:00
|
|
|
import { SqliteErrorKind, parseSqliteError } from './errors';
|
2021-04-05 15:18:19 -07:00
|
|
|
|
|
|
|
if (!parentPort) {
|
|
|
|
throw new Error('Must run as a worker thread');
|
|
|
|
}
|
|
|
|
|
|
|
|
const port = parentPort;
|
|
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
2025-02-20 20:05:15 -07:00
|
|
|
function respond(seq: number, response?: any) {
|
2021-04-05 15:18:19 -07:00
|
|
|
const wrappedResponse: WrappedWorkerResponse = {
|
2021-09-16 14:54:06 -07:00
|
|
|
type: 'response',
|
2021-04-05 15:18:19 -07:00
|
|
|
seq,
|
2025-02-20 20:05:15 -07:00
|
|
|
error: undefined,
|
|
|
|
errorKind: undefined,
|
2021-04-05 15:18:19 -07:00
|
|
|
response,
|
|
|
|
};
|
|
|
|
port.postMessage(wrappedResponse);
|
|
|
|
}
|
|
|
|
|
2021-09-16 14:54:06 -07:00
|
|
|
const log = (
|
|
|
|
level: WrappedWorkerLogEntry['level'],
|
|
|
|
args: Array<unknown>
|
|
|
|
): void => {
|
|
|
|
const wrappedResponse: WrappedWorkerResponse = {
|
|
|
|
type: 'log',
|
|
|
|
level,
|
|
|
|
args,
|
|
|
|
};
|
|
|
|
port.postMessage(wrappedResponse);
|
|
|
|
};
|
|
|
|
|
|
|
|
const logger: LoggerType = {
|
|
|
|
fatal(...args: Array<unknown>) {
|
|
|
|
log('fatal', args);
|
|
|
|
},
|
|
|
|
error(...args: Array<unknown>) {
|
|
|
|
log('error', args);
|
|
|
|
},
|
|
|
|
warn(...args: Array<unknown>) {
|
|
|
|
log('warn', args);
|
|
|
|
},
|
|
|
|
info(...args: Array<unknown>) {
|
|
|
|
log('info', args);
|
|
|
|
},
|
|
|
|
debug(...args: Array<unknown>) {
|
|
|
|
log('debug', args);
|
|
|
|
},
|
|
|
|
trace(...args: Array<unknown>) {
|
|
|
|
log('trace', args);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2024-07-22 11:16:33 -07:00
|
|
|
let db: WritableDB | undefined;
|
|
|
|
let isPrimary = false;
|
|
|
|
let isRemoved = false;
|
|
|
|
|
2025-02-20 20:05:15 -07:00
|
|
|
const onMessage = (
|
|
|
|
{ seq, request }: WrappedWorkerRequest,
|
|
|
|
isRetrying = false
|
|
|
|
): void => {
|
2021-04-05 15:18:19 -07:00
|
|
|
try {
|
|
|
|
if (request.type === 'init') {
|
2024-07-22 11:16:33 -07:00
|
|
|
isPrimary = request.isPrimary;
|
|
|
|
isRemoved = false;
|
|
|
|
db = initialize({
|
2021-09-16 14:54:06 -07:00
|
|
|
...request.options,
|
2024-07-22 11:16:33 -07:00
|
|
|
isPrimary,
|
2021-09-16 14:54:06 -07:00
|
|
|
logger,
|
|
|
|
});
|
2021-04-05 15:18:19 -07:00
|
|
|
|
2025-02-20 20:05:15 -07:00
|
|
|
respond(seq, undefined);
|
2021-04-05 15:18:19 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-22 11:16:33 -07:00
|
|
|
// 'close' is sent on shutdown, but we already removed the database.
|
|
|
|
if (isRemoved && request.type === 'close') {
|
2025-02-20 20:05:15 -07:00
|
|
|
respond(seq, undefined);
|
2024-07-22 11:16:33 -07:00
|
|
|
process.exit(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-12 12:54:24 -07:00
|
|
|
// Removing database does not require active connection.
|
|
|
|
if (request.type === 'removeDB') {
|
|
|
|
try {
|
|
|
|
if (db) {
|
|
|
|
if (isPrimary) {
|
|
|
|
DataWriter.close(db);
|
|
|
|
} else {
|
|
|
|
DataReader.close(db);
|
|
|
|
}
|
|
|
|
db = undefined;
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
logger.error('Failed to close database before removal');
|
|
|
|
}
|
2024-07-22 11:16:33 -07:00
|
|
|
|
|
|
|
if (isPrimary) {
|
2024-08-12 12:54:24 -07:00
|
|
|
removeDB();
|
2024-07-22 11:16:33 -07:00
|
|
|
}
|
2024-08-12 12:54:24 -07:00
|
|
|
|
|
|
|
isRemoved = true;
|
2021-04-05 15:18:19 -07:00
|
|
|
|
2025-02-20 20:05:15 -07:00
|
|
|
respond(seq, undefined);
|
2021-04-05 15:18:19 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-12 12:54:24 -07:00
|
|
|
if (!db) {
|
|
|
|
throw new Error('Not initialized');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request.type === 'close') {
|
2024-07-22 11:16:33 -07:00
|
|
|
if (isPrimary) {
|
2024-08-12 12:54:24 -07:00
|
|
|
DataWriter.close(db);
|
2024-07-22 11:16:33 -07:00
|
|
|
} else {
|
|
|
|
DataReader.close(db);
|
|
|
|
}
|
|
|
|
db = undefined;
|
2021-07-09 17:43:36 -07:00
|
|
|
|
2025-02-20 20:05:15 -07:00
|
|
|
respond(seq, undefined);
|
2024-08-12 12:54:24 -07:00
|
|
|
process.exit(0);
|
2021-07-09 17:43:36 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-22 11:16:33 -07:00
|
|
|
if (request.type === 'sqlCall:read' || request.type === 'sqlCall:write') {
|
|
|
|
const DataInterface =
|
|
|
|
request.type === 'sqlCall:read' ? DataReader : DataWriter;
|
2021-04-05 15:18:19 -07:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
2024-07-22 11:16:33 -07:00
|
|
|
const method = (DataInterface as any)[request.method];
|
2021-04-05 15:18:19 -07:00
|
|
|
if (typeof method !== 'function') {
|
2024-09-06 10:52:19 -07:00
|
|
|
throw new Error(`Invalid sql method: ${request.method} ${method}`);
|
2021-04-05 15:18:19 -07:00
|
|
|
}
|
|
|
|
|
2023-07-14 20:52:20 -04:00
|
|
|
const start = performance.now();
|
2024-07-22 11:16:33 -07:00
|
|
|
const result = method(db, ...request.args);
|
2023-07-14 20:52:20 -04:00
|
|
|
const end = performance.now();
|
2021-04-26 15:01:22 -07:00
|
|
|
|
2025-02-20 20:05:15 -07:00
|
|
|
respond(seq, { result, duration: end - start });
|
2021-04-05 15:18:19 -07:00
|
|
|
} else {
|
|
|
|
throw new Error('Unexpected request type');
|
|
|
|
}
|
|
|
|
} catch (error) {
|
2025-02-20 20:05:15 -07:00
|
|
|
const errorKind = parseSqliteError(error);
|
|
|
|
|
2025-05-13 12:01:56 -07:00
|
|
|
if (
|
|
|
|
(errorKind === SqliteErrorKind.Corrupted ||
|
|
|
|
errorKind === SqliteErrorKind.Logic) &&
|
|
|
|
db != null
|
|
|
|
) {
|
2025-02-20 20:05:15 -07:00
|
|
|
const wasRecovered = DataWriter.runCorruptionChecks(db);
|
|
|
|
if (
|
|
|
|
wasRecovered &&
|
|
|
|
!isRetrying &&
|
|
|
|
// Don't retry 'init'/'close'/'removeDB' automatically and notify user
|
|
|
|
// about the database error (even on successful recovery).
|
|
|
|
(request.type === 'sqlCall:read' || request.type === 'sqlCall:write')
|
|
|
|
) {
|
|
|
|
logger.error(`Retrying request: ${request.type}`);
|
|
|
|
return onMessage({ seq, request }, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const wrappedResponse: WrappedWorkerResponse = {
|
|
|
|
type: 'response',
|
|
|
|
seq,
|
|
|
|
error: {
|
|
|
|
name: error.name,
|
|
|
|
message: error.message,
|
|
|
|
stack: error.stack,
|
|
|
|
},
|
|
|
|
errorKind,
|
|
|
|
response: undefined,
|
|
|
|
};
|
|
|
|
port.postMessage(wrappedResponse);
|
2021-04-05 15:18:19 -07:00
|
|
|
}
|
2025-02-20 20:05:15 -07:00
|
|
|
};
|
|
|
|
port.on('message', (message: WrappedWorkerRequest) => onMessage(message));
|