@riddance/service
Version:
Too much code slows you down, creates risks, increases maintainability burdens, confuses AI. So let's commit less of it.
212 lines • 29.4 kB
JavaScript
/* eslint-disable no-console */
import { createContext } from '@riddance/host/context';
import { setMeta } from '@riddance/host/registry';
import { mkdir, readdir, readFile, writeFile } from 'node:fs/promises';
import { EOL } from 'node:os';
import { basename, extname, join, relative, sep } from 'node:path';
import { performance } from 'node:perf_hooks';
import { pathToFileURL } from 'node:url';
function setup() {
setupTestContext();
before(async () => {
const { name, config } = await readConfig();
const dir = process.cwd();
const files = (await readdir('.')).filter(file => extname(file) === '.ts' && !file.endsWith('.d.ts'));
for (const file of files) {
const base = basename(file, '.ts');
setMeta(name, base, 'test-mock', config);
await import(pathToFileURL(join(dir, base + '.js')).toString());
}
});
}
setup();
async function readEnv() {
try {
const envText = await readFile('test/env.txt', 'utf-8');
return Object.fromEntries(envText
.split('\n')
.filter(l => l.length !== 0 && !l.startsWith('#'))
.map(line => {
const ix = line.indexOf('=');
return [line.slice(0, ix).trim(), line.slice(ix + 1).trim()];
}));
}
catch (e) {
if (e.code === 'ENOENT') {
return {};
}
throw e;
}
}
async function readConfig() {
const packageJson = JSON.parse(await readFile('package.json', 'utf-8'));
return packageJson;
}
let testContext;
function setupTestContext() {
beforeEach('Clear logged entries', async () => {
const env = await readEnv();
if (testContext) {
throw new Error('Context exists.');
}
testContext = new TestContext(env);
});
afterEach('Check log', async function () {
if (!testContext) {
throw new Error('Test context lost.');
}
const test = this.currentTest;
if (test) {
const title = test.fullTitle();
if (test.isFailed()) {
await testContext.log.dumpLog(title);
}
if (testContext.log.failed) {
if (!test.isFailed()) {
await testContext.log.dumpLog(title);
throw new Error(`"${title}" passed but subsequently failed because errors was logged during the test. Add using _ = allowErrorLogs() if the error log entries are expected.`);
}
}
}
testContext = undefined;
});
}
export function jsonRoundtrip(obj) {
if (obj === undefined) {
return undefined;
}
// eslint-disable-next-line unicorn/prefer-structured-clone
return JSON.parse(JSON.stringify(obj));
}
export function createMockContext(client, config, meta) {
const ctx = getTestContext();
return createContext(client, [ctx.log], {
sendEvent(topic, type, subject, data, messageId, signal) {
signal.throwIfAborted();
ctx.emitted.push({
topic,
type,
subject,
data: jsonRoundtrip(data),
messageId,
});
return Promise.resolve();
},
}, { default: 15 }, new AbortController(), config, meta, ctx.env, () => ctx.now());
}
export function getTestContext() {
if (!testContext) {
throw new Error('No test is running.');
}
return testContext;
}
class MockLogger {
#entries = [];
#startTime = Math.round(performance.now() * 10_000);
failOnErrorLogs = true;
failed = false;
getEntries() {
return [...this.#entries];
}
clear() {
this.#entries = [];
this.failOnErrorLogs = true;
this.failed = false;
}
sendEntries(entries) {
if (this.failOnErrorLogs && entries.some(e => e.level === 'error' || e.level === 'fatal')) {
this.failed = true;
}
this.#entries.push(...entries);
return undefined;
}
#msSinceStart(entry) {
return (Math.round(entry.timestamp * 10_000) - this.#startTime) / 10_000;
}
async dumpLog(testTitle) {
if (this.#entries.length !== 0) {
const p = this.writeLog();
const errors = this.#entries.filter(e => e.level === 'fatal' || e.level === 'error');
if (errors.length !== 0) {
console.error(testTitle + ' error log:');
errors.forEach(e => {
console.error(`@${this.#msSinceStart(e)}ms ${levelString(e.level)} ${e.message}`);
if (e.error) {
console.error(e.error);
}
});
}
const logFile = await p;
if (logFile) {
console.info(`Full log of "${testTitle}" saved to .${sep}${relative(process.env.PROJECT_DIRECTORY ?? process.cwd(), logFile)}`);
}
}
}
async writeLog() {
try {
const resultPath = join('test', 'results');
await mkdir(resultPath, { recursive: true });
const name = join(resultPath, 'log-' + new Date().toISOString().replaceAll(':', '') + '.json');
await writeFile(name, `[${this.#entries
.map(e => JSON.stringify({
timeOffset: this.#msSinceStart(e),
...JSON.parse(e.json),
}, undefined, ' '))
.join(',' + EOL)}]`);
return name;
}
catch (e) {
console.error(`Error saving log:`);
console.error(e);
console.log('Full log:');
this.#entries.forEach(entry => {
console.log(`@${this.#msSinceStart(entry)}ms ${levelString(entry.level)} ${entry.message}`);
if (entry.error) {
console.log(entry.error);
}
});
return undefined;
}
}
}
function levelString(level) {
switch (level) {
case 'trace':
return '[TRACE] ';
case 'debug':
return '[DEBUG] ';
case 'info':
return '[INFO] ';
case 'warning':
return '[WARNING]';
case 'error':
return '[ERROR] ';
case 'fatal':
return '[FATAL] ';
default:
return ' ';
}
}
class TestContext {
log;
get env() {
return this.environment;
}
environment;
emitted = [];
timeShift = 0;
constructor(env) {
this.environment = {
BEARER_PUBLIC_KEY: 'MHYwEAYHKoZIzj0CAQYFK4EEACIDYgAESKk7sgjLJNz4erSkGiuFRQCUZiVELR4VjqrWS01kKxZSthAKuX5A4ib8ODd2le/4m99vBIKpDKWP6CT/LvhzcXstSxz4VaOkbczfo3VUvKREi0yUZLasKB5oQP2AGAyr',
BEARER_PRIVATE_KEY: 'MIGkAgEBBDCuIjzsQ+q0iCuyEiLq9vFfZ6Lj6/vxlZDxLanGoO88yL9V0EsZbofwvpW4cb32++SgBwYFK4EEACKhZANiAARIqTuyCMsk3Ph6tKQaK4VFAJRmJUQtHhWOqtZLTWQrFlK2EAq5fkDiJvw4N3aV7/ib328EgqkMpY/oJP8u+HNxey1LHPhVo6RtzN+jdVS8pESLTJRktqwoHmhA/YAYDKs=',
...env,
};
this.log = new MockLogger();
}
now() {
const d = new Date();
d.setUTCSeconds(d.getUTCSeconds() + this.timeShift);
return d;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup.js","sourceRoot":"","sources":["setup.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport { ClientInfo, createContext, LogEntry, LogLevel, LogTransport } from '@riddance/host/context'\nimport { FullConfiguration, Metadata, setMeta } from '@riddance/host/registry'\nimport { mkdir, readdir, readFile, writeFile } from 'node:fs/promises'\nimport { EOL } from 'node:os'\nimport { basename, extname, join, relative, sep } from 'node:path'\nimport { performance } from 'node:perf_hooks'\nimport { pathToFileURL } from 'node:url'\nimport {\n    Environment,\n    JsonSafeObject,\n    type JsonObject,\n    type JsonSafe,\n    type Stringified,\n} from '../context.js'\n\nfunction setup() {\n    setupTestContext()\n    before(async () => {\n        const { name, config } = await readConfig()\n        const dir = process.cwd()\n        const files = (await readdir('.')).filter(\n            file => extname(file) === '.ts' && !file.endsWith('.d.ts'),\n        )\n        for (const file of files) {\n            const base = basename(file, '.ts')\n            setMeta(name, base, 'test-mock', config)\n            await import(pathToFileURL(join(dir, base + '.js')).toString())\n        }\n    })\n}\n\nsetup()\n\nasync function readEnv() {\n    try {\n        const envText = await readFile('test/env.txt', 'utf-8')\n        return Object.fromEntries(\n            envText\n                .split('\\n')\n                .filter(l => l.length !== 0 && !l.startsWith('#'))\n                .map(line => {\n                    const ix = line.indexOf('=')\n                    return [line.slice(0, ix).trim(), line.slice(ix + 1).trim()]\n                }),\n        )\n    } catch (e) {\n        if ((e as { code?: string }).code === 'ENOENT') {\n            return {}\n        }\n        throw e\n    }\n}\n\nasync function readConfig() {\n    const packageJson = JSON.parse(await readFile('package.json', 'utf-8')) as {\n        name: string\n        config?: object\n    }\n    return packageJson\n}\n\nlet testContext: TestContext | undefined\n\nfunction setupTestContext() {\n    beforeEach('Clear logged entries', async () => {\n        const env = await readEnv()\n        if (testContext) {\n            throw new Error('Context exists.')\n        }\n        testContext = new TestContext(env)\n    })\n\n    afterEach('Check log', async function () {\n        if (!testContext) {\n            throw new Error('Test context lost.')\n        }\n        const test = this.currentTest\n        if (test) {\n            const title = test.fullTitle()\n            if (test.isFailed()) {\n                await testContext.log.dumpLog(title)\n            }\n            if (testContext.log.failed) {\n                if (!test.isFailed()) {\n                    await testContext.log.dumpLog(title)\n                    throw new Error(\n                        `\"${title}\" passed but subsequently failed because errors was logged during the test. Add using _ = allowErrorLogs() if the error log entries are expected.`,\n                    )\n                }\n            }\n        }\n        testContext = undefined\n    })\n}\n\nexport function jsonRoundtrip<T extends JsonSafe>(obj: T | undefined): Stringified<T> | undefined {\n    if (obj === undefined) {\n        return undefined\n    }\n    // eslint-disable-next-line unicorn/prefer-structured-clone\n    return JSON.parse(JSON.stringify(obj)) as Stringified<T>\n}\n\nexport function createMockContext(client: ClientInfo, config?: FullConfiguration, meta?: Metadata) {\n    const ctx = getTestContext()\n    return createContext(\n        client,\n        [ctx.log],\n        {\n            sendEvent(\n                topic: string,\n                type: string,\n                subject: string,\n                data: JsonSafeObject | undefined,\n                messageId: string | undefined,\n                signal: AbortSignal,\n            ) {\n                signal.throwIfAborted()\n                ctx.emitted.push({\n                    topic,\n                    type,\n                    subject,\n                    data: jsonRoundtrip(data),\n                    messageId,\n                })\n                return Promise.resolve()\n            },\n        },\n        { default: 15 },\n        new AbortController(),\n        config,\n        meta,\n        ctx.env,\n        () => ctx.now(),\n    )\n}\n\nexport function getTestContext(): TestContext {\n    if (!testContext) {\n        throw new Error('No test is running.')\n    }\n    return testContext\n}\n\nclass MockLogger implements LogTransport {\n    #entries: LogEntry[] = []\n    readonly #startTime = Math.round(performance.now() * 10_000)\n    failOnErrorLogs = true\n    failed = false\n\n    getEntries() {\n        return [...this.#entries]\n    }\n\n    clear() {\n        this.#entries = []\n        this.failOnErrorLogs = true\n        this.failed = false\n    }\n\n    sendEntries(entries: LogEntry[]) {\n        if (this.failOnErrorLogs && entries.some(e => e.level === 'error' || e.level === 'fatal')) {\n            this.failed = true\n        }\n        this.#entries.push(...entries)\n        return undefined\n    }\n\n    #msSinceStart(entry: LogEntry) {\n        return (Math.round(entry.timestamp * 10_000) - this.#startTime) / 10_000\n    }\n\n    async dumpLog(testTitle: string) {\n        if (this.#entries.length !== 0) {\n            const p = this.writeLog()\n            const errors = this.#entries.filter(e => e.level === 'fatal' || e.level === 'error')\n            if (errors.length !== 0) {\n                console.error(testTitle + ' error log:')\n                errors.forEach(e => {\n                    console.error(\n                        `@${this.#msSinceStart(e)}ms ${levelString(e.level)} ${e.message}`,\n                    )\n                    if (e.error) {\n                        console.error(e.error)\n                    }\n                })\n            }\n            const logFile = await p\n            if (logFile) {\n                console.info(\n                    `Full log of \"${testTitle}\" saved to .${sep}${relative(process.env.PROJECT_DIRECTORY ?? process.cwd(), logFile)}`,\n                )\n            }\n        }\n    }\n\n    async writeLog() {\n        try {\n            const resultPath = join('test', 'results')\n            await mkdir(resultPath, { recursive: true })\n            const name = join(\n                resultPath,\n                'log-' + new Date().toISOString().replaceAll(':', '') + '.json',\n            )\n            await writeFile(\n                name,\n                `[${this.#entries\n                    .map(e =>\n                        JSON.stringify(\n                            {\n                                timeOffset: this.#msSinceStart(e),\n                                ...JSON.parse(e.json),\n                            },\n                            undefined,\n                            '  ',\n                        ),\n                    )\n                    .join(',' + EOL)}]`,\n            )\n            return name\n        } catch (e) {\n            console.error(`Error saving log:`)\n            console.error(e)\n            console.log('Full log:')\n            this.#entries.forEach(entry => {\n                console.log(\n                    `@${this.#msSinceStart(entry)}ms ${levelString(entry.level)} ${entry.message}`,\n                )\n                if (entry.error) {\n                    console.log(entry.error)\n                }\n            })\n            return undefined\n        }\n    }\n}\n\nfunction levelString(level: LogLevel) {\n    switch (level) {\n        case 'trace':\n            return '[TRACE]  '\n        case 'debug':\n            return '[DEBUG]  '\n        case 'info':\n            return '[INFO]   '\n        case 'warning':\n            return '[WARNING]'\n        case 'error':\n            return '[ERROR]  '\n        case 'fatal':\n            return '[FATAL]  '\n        default:\n            return '         '\n    }\n}\n\ntype Event = {\n    topic: string\n    type: string\n    subject: string\n    data?: JsonObject\n    messageId: string | undefined\n}\n\nclass TestContext {\n    readonly log: MockLogger\n\n    get env() {\n        return this.environment\n    }\n\n    environment: { [key: string]: string }\n    emitted: Event[] = []\n\n    timeShift = 0\n\n    constructor(env: Environment) {\n        this.environment = {\n            BEARER_PUBLIC_KEY:\n                'MHYwEAYHKoZIzj0CAQYFK4EEACIDYgAESKk7sgjLJNz4erSkGiuFRQCUZiVELR4VjqrWS01kKxZSthAKuX5A4ib8ODd2le/4m99vBIKpDKWP6CT/LvhzcXstSxz4VaOkbczfo3VUvKREi0yUZLasKB5oQP2AGAyr',\n            BEARER_PRIVATE_KEY:\n                'MIGkAgEBBDCuIjzsQ+q0iCuyEiLq9vFfZ6Lj6/vxlZDxLanGoO88yL9V0EsZbofwvpW4cb32++SgBwYFK4EEACKhZANiAARIqTuyCMsk3Ph6tKQaK4VFAJRmJUQtHhWOqtZLTWQrFlK2EAq5fkDiJvw4N3aV7/ib328EgqkMpY/oJP8u+HNxey1LHPhVo6RtzN+jdVS8pESLTJRktqwoHmhA/YAYDKs=',\n            ...env,\n        }\n        this.log = new MockLogger()\n    }\n\n    now(): Date {\n        const d = new Date()\n        d.setUTCSeconds(d.getUTCSeconds() + this.timeShift)\n        return d\n    }\n}\n"]}