160 lines
5.9 KiB
TypeScript
160 lines
5.9 KiB
TypeScript
|
/*
|
||
|
Copyright 2023 The Matrix.org Foundation C.I.C.
|
||
|
|
||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
you may not use this file except in compliance with the License.
|
||
|
You may obtain a copy of the License at
|
||
|
|
||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
||
|
Unless required by applicable law or agreed to in writing, software
|
||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
See the License for the specific language governing permissions and
|
||
|
limitations under the License.
|
||
|
*/
|
||
|
|
||
|
import path, { basename } from "node:path";
|
||
|
import os from "node:os";
|
||
|
import * as fse from "fs-extra";
|
||
|
import { BrowserContext, TestInfo } from "@playwright/test";
|
||
|
|
||
|
import { getFreePort } from "../utils/port";
|
||
|
import { Docker } from "../docker";
|
||
|
import { PG_PASSWORD, PostgresDocker } from "../postgres";
|
||
|
import { HomeserverInstance } from "../homeserver";
|
||
|
import { Instance as MailhogInstance } from "../mailhog";
|
||
|
|
||
|
// Docker tag to use for `ghcr.io/matrix-org/matrix-authentication-service` image.
|
||
|
// We use a debug tag so that we have a shell and can run all 3 necessary commands in one run.
|
||
|
const TAG = "0.8.0-debug";
|
||
|
|
||
|
export interface ProxyInstance {
|
||
|
containerId: string;
|
||
|
postgresId: string;
|
||
|
configDir: string;
|
||
|
port: number;
|
||
|
}
|
||
|
|
||
|
async function cfgDirFromTemplate(opts: {
|
||
|
postgresHost: string;
|
||
|
synapseUrl: string;
|
||
|
masPort: string;
|
||
|
smtpPort: string;
|
||
|
}): Promise<{
|
||
|
configDir: string;
|
||
|
}> {
|
||
|
const configPath = path.join(__dirname, "config.yaml");
|
||
|
const tempDir = await fse.mkdtemp(path.join(os.tmpdir(), "react-sdk-mas-"));
|
||
|
|
||
|
const outputHomeserver = path.join(tempDir, "config.yaml");
|
||
|
console.log(`Gen ${configPath} -> ${outputHomeserver}`);
|
||
|
let config = await fse.readFile(configPath, "utf8");
|
||
|
config = config.replace(/{{MAS_PORT}}/g, opts.masPort);
|
||
|
config = config.replace(/{{POSTGRES_HOST}}/g, opts.postgresHost);
|
||
|
config = config.replace(/{{POSTGRES_PASSWORD}}/g, PG_PASSWORD);
|
||
|
config = config.replace(/%{{SMTP_PORT}}/g, opts.smtpPort);
|
||
|
config = config.replace(/{{SYNAPSE_URL}}/g, opts.synapseUrl);
|
||
|
|
||
|
await fse.writeFile(outputHomeserver, config);
|
||
|
|
||
|
// Allow anyone to read, write and execute in the temp directory
|
||
|
// so that the DIND setup that we use to update the playwright screenshots work without any issues.
|
||
|
await fse.chmod(tempDir, 0o757);
|
||
|
|
||
|
return {
|
||
|
configDir: tempDir,
|
||
|
};
|
||
|
}
|
||
|
|
||
|
export class MatrixAuthenticationService {
|
||
|
private readonly masDocker = new Docker();
|
||
|
private readonly postgresDocker = new PostgresDocker("mas");
|
||
|
private instance: ProxyInstance;
|
||
|
public port: number;
|
||
|
|
||
|
constructor(private context: BrowserContext) {}
|
||
|
|
||
|
async prepare(): Promise<{ port: number }> {
|
||
|
this.port = await getFreePort();
|
||
|
return { port: this.port };
|
||
|
}
|
||
|
|
||
|
async start(homeserver: HomeserverInstance, mailhog: MailhogInstance): Promise<ProxyInstance> {
|
||
|
console.log(new Date(), "Starting mas...");
|
||
|
|
||
|
if (!this.port) await this.prepare();
|
||
|
const port = this.port;
|
||
|
const { containerId: postgresId, ipAddress: postgresIp } = await this.postgresDocker.start();
|
||
|
const { configDir } = await cfgDirFromTemplate({
|
||
|
masPort: port.toString(),
|
||
|
postgresHost: postgresIp,
|
||
|
synapseUrl: homeserver.config.dockerUrl,
|
||
|
smtpPort: mailhog.smtpPort.toString(),
|
||
|
});
|
||
|
|
||
|
console.log(new Date(), "starting mas container...", TAG);
|
||
|
const containerId = await this.masDocker.run({
|
||
|
image: "ghcr.io/matrix-org/matrix-authentication-service:" + TAG,
|
||
|
containerName: "react-sdk-playwright-mas",
|
||
|
params: ["-p", `${port}:8080/tcp`, "-v", `${configDir}:/config`, "--entrypoint", "sh"],
|
||
|
cmd: [
|
||
|
"-c",
|
||
|
"mas-cli database migrate --config /config/config.yaml && " +
|
||
|
"mas-cli config sync --config /config/config.yaml && " +
|
||
|
"mas-cli server --config /config/config.yaml",
|
||
|
],
|
||
|
});
|
||
|
console.log(new Date(), "started!");
|
||
|
|
||
|
// Set up redirects
|
||
|
const baseUrl = `http://localhost:${port}`;
|
||
|
for (const path of [
|
||
|
"**/_matrix/client/*/login",
|
||
|
"**/_matrix/client/*/login/**",
|
||
|
"**/_matrix/client/*/logout",
|
||
|
"**/_matrix/client/*/refresh",
|
||
|
]) {
|
||
|
await this.context.route(path, async (route) => {
|
||
|
await route.continue({
|
||
|
url: new URL(route.request().url().split("/").slice(3).join("/"), baseUrl).href,
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
|
||
|
this.instance = { containerId, postgresId, port, configDir };
|
||
|
return this.instance;
|
||
|
}
|
||
|
|
||
|
async stop(testInfo: TestInfo): Promise<void> {
|
||
|
if (!this.instance) return; // nothing to stop
|
||
|
const id = this.instance.containerId;
|
||
|
const logPath = path.join("playwright", "logs", "matrix-authentication-service", id);
|
||
|
await fse.ensureDir(logPath);
|
||
|
await this.masDocker.persistLogsToFile({
|
||
|
stdoutFile: path.join(logPath, "stdout.log"),
|
||
|
stderrFile: path.join(logPath, "stderr.log"),
|
||
|
});
|
||
|
|
||
|
await this.masDocker.stop();
|
||
|
await this.postgresDocker.stop();
|
||
|
|
||
|
if (testInfo.status !== "passed") {
|
||
|
const logs = [path.join(logPath, "stdout.log"), path.join(logPath, "stderr.log")];
|
||
|
for (const path of logs) {
|
||
|
await testInfo.attach(`mas-${basename(path)}`, {
|
||
|
path,
|
||
|
contentType: "text/plain",
|
||
|
});
|
||
|
}
|
||
|
await testInfo.attach("mas-config.yaml", {
|
||
|
path: path.join(this.instance.configDir, "config.yaml"),
|
||
|
contentType: "text/plain",
|
||
|
});
|
||
|
}
|
||
|
|
||
|
await fse.remove(this.instance.configDir);
|
||
|
console.log(new Date(), "Stopped mas.");
|
||
|
}
|
||
|
}
|