fluxer/fluxer_integration/src/globalSetup.tsx
2026-02-17 12:22:36 +00:00

157 lines
4.1 KiB
TypeScript

/*
* Copyright (C) 2026 Fluxer Contributors
*
* This file is part of Fluxer.
*
* Fluxer is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Fluxer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
*/
import {execSync} from 'node:child_process';
import path from 'node:path';
import {fileURLToPath} from 'node:url';
import WebSocket from 'ws';
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const composeFile = path.resolve(__dirname, '../docker/compose.yaml');
const PROJECT_NAME = 'fluxer-integration';
const API_URL = 'http://localhost:18088/api/v1';
const GATEWAY_URL = 'ws://localhost:18088/gateway';
const HEALTH_CHECK_URL = `${API_URL}/_health`;
const MAX_WAIT_SECONDS = 120;
const POLL_INTERVAL_MS = 2000;
const GATEWAY_OPCODE_HELLO = 10;
async function waitForApi(): Promise<void> {
const startTime = Date.now();
const maxWaitMs = MAX_WAIT_SECONDS * 1000;
while (Date.now() - startTime < maxWaitMs) {
try {
const response = await fetch(HEALTH_CHECK_URL, {
headers: {'X-Forwarded-For': '127.0.0.1'},
});
if (response.ok) {
console.log('API server is ready');
return;
}
} catch {}
await new Promise((resolve) => setTimeout(resolve, POLL_INTERVAL_MS));
}
throw new Error(`API server did not become ready within ${MAX_WAIT_SECONDS} seconds`);
}
async function probeGateway(): Promise<void> {
return new Promise((resolve, reject) => {
const url = `${GATEWAY_URL}?v=1&encoding=json`;
const ws = new WebSocket(url, {
headers: {
'User-Agent': 'FluxerIntegrationTests/1.0',
Origin: 'http://localhost:5173',
},
});
const timeout = setTimeout(() => {
ws.close();
reject(new Error('Gateway WebSocket probe timed out'));
}, 10000);
ws.on('message', (data: WebSocket.Data) => {
try {
const payload = JSON.parse(data.toString());
if (payload.op === GATEWAY_OPCODE_HELLO) {
clearTimeout(timeout);
ws.close(1000, 'Probe complete');
resolve();
}
} catch {
clearTimeout(timeout);
ws.close();
reject(new Error('Failed to parse gateway message'));
}
});
ws.on('error', (error) => {
clearTimeout(timeout);
reject(error);
});
ws.on('close', (code, reason) => {
clearTimeout(timeout);
if (code !== 1000) {
reject(new Error(`Gateway WebSocket closed unexpectedly: ${code} ${reason.toString()}`));
}
});
});
}
async function waitForGateway(): Promise<void> {
const startTime = Date.now();
const maxWaitMs = MAX_WAIT_SECONDS * 1000;
while (Date.now() - startTime < maxWaitMs) {
try {
await probeGateway();
console.log('Gateway is ready');
return;
} catch {}
await new Promise((resolve) => setTimeout(resolve, POLL_INTERVAL_MS));
}
throw new Error(`Gateway did not become ready within ${MAX_WAIT_SECONDS} seconds`);
}
function startContainers(): void {
console.log('Starting integration test infrastructure...');
try {
execSync(`docker compose -p ${PROJECT_NAME} -f "${composeFile}" up -d --build --wait`, {
stdio: 'inherit',
});
} catch (error) {
console.error('Failed to start Docker containers:', error);
throw error;
}
}
function showContainerLogs(): void {
try {
console.log('\n=== Container logs ===');
execSync(`docker compose -p ${PROJECT_NAME} -f "${composeFile}" logs --tail=100 fluxer_server`, {
stdio: 'inherit',
});
} catch {}
}
export default async function globalSetup(): Promise<void> {
console.log('\n=== Integration Test Global Setup ===\n');
startContainers();
try {
await waitForApi();
await waitForGateway();
} catch (error) {
showContainerLogs();
throw error;
}
console.log('\n=== Infrastructure ready ===\n');
}