Skip to content

Commit

Permalink
Run e2e tests in CI with latest prebuilt backend binary (#131)
Browse files Browse the repository at this point in the history
Getting the old path of a backend directory was too much trouble, we need convex-backend changes to make this work.
  • Loading branch information
thomasballinger authored Dec 4, 2024
1 parent aeafbf2 commit c6dc7dd
Show file tree
Hide file tree
Showing 14 changed files with 6,254 additions and 75 deletions.
41 changes: 41 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
name: Test

on:
push:
branches: [main]
pull_request:
branches: [main]

jobs:
test:
runs-on: ubuntu-latest

steps:
- uses: actions/checkout@v4

- uses: extractions/setup-just@v2

# TODO this doesn't pass on Node.js 20 but it should, something with Next.js
- name: Use Node.js 18.x
uses: actions/setup-node@v4
with:
node-version: 18.x
cache: "npm"

- name: Install dependencies
run: |
npm ci
cd test; npm i
cd ../test-nextjs; npm i
cd ../test-router; npm i
- name: test
run: npm run test:once

- name: Install Playwright Browsers
working-directory: test-nextjs
run: npx playwright install --with-deps

- name: e2e test
working-directory: test-nextjs
run: npm run test
6 changes: 4 additions & 2 deletions test-nextjs/.gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -35,5 +35,7 @@ yarn-error.log*
*.tsbuildinfo
next-env.d.ts

# Ignored for the template, you probably want to remove it:
package-lock.json
# for tests
convex-backend.zip
convex-local-backend
convex-local-backend.log
2 changes: 1 addition & 1 deletion test-nextjs/Justfile
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ reset-local-backend:

# (*) Run the open source convex backend on port 3210
run-local-backend *ARGS:
cd $CONVEX_LOCAL_BACKEND_PATH && just run-local-backend --port 3210
cd $CONVEX_LOCAL_BACKEND_PATH && just run-local-backend --port 3210 --site-proxy-port 3211

# This uses the default admin key for local backends, which is safe as long as the backend is
# running locally.
Expand Down
249 changes: 193 additions & 56 deletions test-nextjs/backendHarness.js
Original file line number Diff line number Diff line change
@@ -1,30 +1,58 @@
// Taken from https://github.com/get-convex/convex-chess/blob/main/backendHarness.js
// Run a command against a fresh local backend, handling setting up and tearing down the backend.
// If CONVEX_LOCAL_BACKEND_PATH is set, expect a convex local backend repo and use that.
// Otherwise download the latest prebuild binary.
//
// Based on from https://github.com/get-convex/convex-chess/blob/main/backendHarness.js
// and
// https://github.com/get-convex/convex-js/blob/main/src/cli/lib/localDeployment/run.ts

const http = require("http");
const fsPromises = require("fs/promises");
const { spawn, exec, execSync } = require("child_process");
const { existsSync, unlinkSync } = require("fs");
const { Readable } = require("stream");
const AdmZip = require("adm-zip");
const { promisify } = require("util");

// Run a command against a fresh local backend, handling setting up and tearing down the backend.
function logToStderr(...args) {
const strings = args.map((arg) =>
typeof arg === "string"
? arg
: util.inspect(arg, {
colors: true,
depth: null,
maxArrayLength: null,
breakLength: process.stdout.columns || 80,
}),
);

process.stderr.write(strings.join(" ") + "\n");
}

// Checks for a local backend running on port 3210.
const parsedUrl = new URL("http://127.0.0.1:3210");
const backendCloudPort = 3210;
const backendSitePort = 3211;
const parsedUrl = new URL(`http://127.0.0.1:${backendCloudPort}`);

async function isBackendRunning(backendUrl) {
return new Promise ((resolve) => {
http
.request(
{
hostname: backendUrl.hostname,
port: backendUrl.port,
path: "/version",
method: "GET",
},
(res) => {
resolve(res.statusCode === 200)
}
)
.on("error", () => { resolve(false) })
.end();
})
return new Promise((resolve) => {
http
.request(
{
hostname: backendUrl.hostname,
port: backendUrl.port,
path: "/version",
method: "GET",
},
(res) => {
resolve(res.statusCode === 200);
},
)
.on("error", () => {
resolve(false);
})
.end();
});
}

function sleep(ms) {
Expand All @@ -33,66 +61,175 @@ function sleep(ms) {

const waitForLocalBackendRunning = async (backendUrl) => {
let isRunning = await isBackendRunning(backendUrl);
let i = 0
let i = 0;
while (!isRunning) {
if (i % 10 === 0) {
// Progress messages every ~5 seconds
console.log("Waiting for backend to be running...")
logToStderr("Waiting for backend to be running...");
}
await sleep(500);
isRunning = await isBackendRunning(backendUrl);
i += 1
isDead = backendProcess.exitCode !== null;
if (isDead) {
throw new Error("Backend exited");
}
i += 1;
}
return
};

}

let backendProcess = null
let backendProcess = null;

function cleanup() {
if (backendProcess !== null) {
console.log("Cleaning up running backend")
backendProcess.kill("SIGTERM")
execSync("just reset-local-backend")
logToStderr("Cleaning up running backend");
backendProcess.kill("SIGTERM");
if (useDownloadedBinary) {
execSync(
"rm -rf convex_local_storage && rm -f convex_local_backend.sqlite3",
);
} else {
execSync("just reset-local-backend");
}
}
}

function getDownloadPath() {
switch (process.platform) {
case "darwin":
if (process.arch === "arm64") {
return "convex-local-backend-aarch64-apple-darwin.zip";
} else if (process.arch === "x64") {
return "convex-local-backend-x86_64-apple-darwin.zip";
}
break;
case "linux":
if (process.arch === "arm64") {
return "convex-local-backend-aarch64-unknown-linux-gnu.zip";
} else if (process.arch === "x64") {
return "convex-local-backend-x86_64-unknown-linux-gnu.zip";
}
break;
case "win32":
return "convex-local-backend-x86_64-pc-windows-msvc.zip";
}
return null;
}

async function downloadedBinaryProcess() {
const latest = await fetch(
"https://github.com/get-convex/convex-backend/releases/latest",
{ redirect: "manual" },
);
if (latest.status !== 302) {
throw new Error(`Error downloading ${latest}`);
}
const latestUrl = latest.headers.get("location");
const version = latestUrl.split("/").pop();
logToStderr(`Downloading latest backend binary, ${version}`);
const downloadPath = getDownloadPath();
const response = await fetch(
`https://github.com/get-convex/convex-backend/releases/download/${version}/${downloadPath}`,
);

const zipLocation = "convex-backend.zip";
if (existsSync(zipLocation)) {
unlinkSync(zipLocation);
}
const fileHandle = await fsPromises.open(zipLocation, "wx", 0o644);
try {
for await (const chunk of Readable.fromWeb(response.body)) {
await fileHandle.write(chunk);
}
} finally {
await fileHandle.close();
}
logToStderr("Downloaded zip file");

const zip = new AdmZip(zipLocation);
zip.extractAllTo(".", true);
const p = "./convex-local-backend";
await promisify(exec)(`chmod +x ${p}`);
return spawn(
p,
["--port", backendCloudPort, "--site-proxy-port", backendSitePort],
{ env: { CONVEX_TRACE_FILE: "1" } },
);
}

async function runWithLocalBackend(command, backendUrl) {
if (process.env.CONVEX_LOCAL_BACKEND_PATH === undefined) {
console.error("Please set environment variable CONVEX_LOCAL_BACKEND_PATH first")
process.exit(1)
if (useDownloadedBinary) {
console.error(
"environment variable CONVEX_LOCAL_BACKEND_PATH not set, using prebuild binary",
);
}
const isRunning = await isBackendRunning(backendUrl);
if (isRunning) {
console.error("Looks like local backend is already running. Cancel it and restart this command.")
process.exit(1)
console.error(
"Looks like local backend is already running. Cancel it and restart this command.",
);
process.exit(1);
}

if (useDownloadedBinary) {
execSync(
"rm -rf convex_local_storage && rm -f convex_local_backend.sqlite3",
);
} else {
execSync("just reset-local-backend");
}
execSync("just reset-local-backend")
backendProcess = exec("CONVEX_TRACE_FILE=1 just run-local-backend")
await waitForLocalBackendRunning(backendUrl)
console.log("Backend running! Logs can be found in $CONVEX_LOCAL_BACKEND_PATH/convex-local-backend.log")
let logLocation;
if (useDownloadedBinary) {
backendProcess = await downloadedBinaryProcess();
logLocation = `convex-local-backend.log`;
} else {
backendProcess = spawn("just", ["run-local-backend"], {
env: {
...process.env,
CONVEX_TRACE_FILE: "1",
},
cwd: process.env.CONVEX_LOCAL_BACKEND_PATH,
});
logLocation = `${process.env.CONVEX_LOCAL_BACKEND_PATH}/convex-local-backend.log`;
}
backendProcess.stdout.pipe(process.stderr);
backendProcess.stderr.pipe(process.stderr);

await waitForLocalBackendRunning(backendUrl);
logToStderr("Backend running! Logs can be found in", logLocation);
logToStderr("Running command", command);
const innerCommand = new Promise((resolve) => {
const c = spawn(command, { shell: true, stdio: "pipe", env: {...process.env, FORCE_COLOR: true } })
c.stdout.on('data', (data) => {
const c = spawn(command, {
shell: true,
stdio: "pipe",
env: { ...process.env, FORCE_COLOR: true },
});
c.stdout.on("data", (data) => {
process.stdout.write(data);
})
c.stderr.on('data', (data) => {
});

c.stderr.on("data", (data) => {
process.stderr.write(data);
})
c.on('exit', (code) => {
console.log('inner command exited with code ' + code.toString())
resolve(code)
})
});

c.on("exit", (code) => {
logToStderr(`inner command exited with code ${code.toString()}`);
resolve(code);
});
});
return innerCommand;
}

runWithLocalBackend(process.argv[2], parsedUrl).then((code) => {
cleanup()
process.exit(code)
}).catch(() => {
cleanup()
process.exit(1)
})
const useDownloadedBinary = process.env.CONVEX_LOCAL_BACKEND_PATH === undefined;

(async function main() {
let code = undefined;
try {
code = await runWithLocalBackend(process.argv[2], parsedUrl);
} finally {
cleanup();
}
// undefined means exception was thrown
if (code !== undefined) {
process.exit(code);
}
})();
1 change: 0 additions & 1 deletion test-nextjs/convex/_generated/api.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@
*
* THIS CODE IS AUTOMATICALLY GENERATED.
*
* Generated by [email protected].
* To regenerate, run `npx convex dev`.
* @module
*/
Expand Down
1 change: 0 additions & 1 deletion test-nextjs/convex/_generated/api.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@
*
* THIS CODE IS AUTOMATICALLY GENERATED.
*
* Generated by [email protected].
* To regenerate, run `npx convex dev`.
* @module
*/
Expand Down
1 change: 0 additions & 1 deletion test-nextjs/convex/_generated/dataModel.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@
*
* THIS CODE IS AUTOMATICALLY GENERATED.
*
* Generated by [email protected].
* To regenerate, run `npx convex dev`.
* @module
*/
Expand Down
1 change: 0 additions & 1 deletion test-nextjs/convex/_generated/server.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@
*
* THIS CODE IS AUTOMATICALLY GENERATED.
*
* Generated by [email protected].
* To regenerate, run `npx convex dev`.
* @module
*/
Expand Down
1 change: 0 additions & 1 deletion test-nextjs/convex/_generated/server.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@
*
* THIS CODE IS AUTOMATICALLY GENERATED.
*
* Generated by [email protected].
* To regenerate, run `npx convex dev`.
* @module
*/
Expand Down
Loading

0 comments on commit c6dc7dd

Please sign in to comment.