Skip to content

Commit

Permalink
Swap mocha+chai testing to vitest
Browse files Browse the repository at this point in the history
  • Loading branch information
dannon committed Jan 8, 2025
1 parent b54ee99 commit afe1454
Show file tree
Hide file tree
Showing 2 changed files with 111 additions and 90 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@
"url": "git://github.com/usegalaxy-eu/gie-nodejs-proxy"
},
"scripts": {
"test": "mocha --reporter spec",
"test": "vitest",
"lint": "eslint lib test",
"format": "prettier --no-config --write lib/*js test/*js",
"format-check": "prettier --no-config --check lib/*js test/*js"
Expand Down
199 changes: 110 additions & 89 deletions test/test.js → test/main.test.js
Original file line number Diff line number Diff line change
@@ -1,14 +1,15 @@
const { main } = require("../lib/main");
const { DynamicProxy } = require("../lib/proxy");
const http = require("http");
const url = require("url");
const path = require("path");
const fs = require("fs");
require("chai").should();
const axios = require("axios");
const axiosRetry = require("axios-retry");

// Needed because I can't figure out how to wait on the proxy server to get
import { describe, it, beforeAll, afterAll, expect } from "vitest";
import http from "http";
import url from "url";
import path from "path";
import fs from "fs";
import axios from "axios";
import axiosRetry from "axios-retry";

import { main } from "../lib/main.js";
import { DynamicProxy } from "../lib/proxy.js";

// Needed because we can’t figure out how to wait on the proxy server to get
// setup (server.listening stays false forever). This greatly reduces transient
// connection refused errors as a result.
axiosRetry(axios, { retries: 3 });
Expand Down Expand Up @@ -49,8 +50,10 @@ const testServer = http.createServer(function (req, res) {
return;
}

// if is a directory search for index file matching the extention
if (fs.statSync(pathname).isDirectory()) pathname += "/index" + ext;
// if is a directory, search for index file matching the extension
if (fs.statSync(pathname).isDirectory()) {
pathname += "/index" + ext;
}

// read file from file system
fs.readFile(pathname, function (err, data) {
Expand All @@ -68,70 +71,78 @@ const testServer = http.createServer(function (req, res) {

const waitForServer = async function (server, listening) {
for (;;) {
if (server.listening == listening) {
if (server.listening === listening) {
return;
}
await null; // prevents app from hanging
// prevents app from hanging
await null;
}
};

const useTestServer = function () {
before(async function () {
beforeAll(async () => {
testServer.listen(TEST_PORT);
await waitForServer(testServer, true);
});
after(async function () {

afterAll(async () => {
testServer.close();
await waitForServer(testServer, false);
});
};

const verifyProxyOnPort = async function (port, headers, path = "/README.md") {
let res = await axios.get(`http://localhost:${port}${path}`, {
headers: headers,
});
let { data } = res;
data.should.include("# A dynamic configurable reverse proxy");
const res = await axios.get(`http://localhost:${port}${path}`, { headers });
const { data } = res;
expect(data).toContain("# A dynamic configurable reverse proxy");
};

describe("test server", function () {
// --------------------------------------------------------------
// TESTS
// --------------------------------------------------------------

describe("test server", () => {
useTestServer();
it("should serve direct requests as files", async function () {

it("should serve direct requests as files", async () => {
await verifyProxyOnPort(TEST_PORT, {});
});
});

describe("DynamicProxy", function () {
describe("DynamicProxy", () => {
useTestServer();
describe("x-interactive-tool-* headers", function () {
it("should respect host and port", async function () {

describe("x-interactive-tool-* headers", () => {
it("should respect host and port", async () => {
const proxy = new DynamicProxy({ port: 5098, verbose: true });
proxy.listen();
// This never becomes True for the proxy server... why?
// await waitForServer(proxy.proxy, true);

const headers = {
"x-interactive-tool-host": "localhost",
"x-interactive-tool-port": TEST_PORT,
};
await verifyProxyOnPort(5098, headers);

proxy.close();
});
});

describe("x-interactive-tool-target headers", function () {
it("should respect host and port in one header", async function () {
describe("x-interactive-tool-target headers", () => {
it("should respect host and port in one header", async () => {
const proxy = new DynamicProxy({ port: 5097, verbose: true });
proxy.listen();

const headers = {
"x-interactive-tool-host": "localhost:" + TEST_PORT,
};
await verifyProxyOnPort(5097, headers);

proxy.close();
});
});

describe("map based forwarding using subdomain", function () {
it("should respect session map", async function () {
describe("map based forwarding using subdomain", () => {
it("should respect session map", async () => {
const sessionMap = {
coolkey: {
token: "cooltoken",
Expand All @@ -144,19 +155,21 @@ describe("DynamicProxy", function () {
const proxy = new DynamicProxy({
port: 5099,
verbose: true,
sessionMap: sessionMap,
sessionMap,
});
proxy.listen();

const headers = {
host: "coolkey-cooltoken.usegalaxy.org",
};
await verifyProxyOnPort(5099, headers);

proxy.close();
});
});

describe("map based path forwarding to full path", function () {
it("should respect session map with requires_path_in_url=true and leave path unmodified", async function () {
describe("map based path forwarding to full path", () => {
it("should respect session map with requires_path_in_url=true and leave path unmodified", async () => {
const sessionMap = {
coolkey: {
token: "cooltoken",
Expand All @@ -170,22 +183,24 @@ describe("DynamicProxy", function () {
const proxy = new DynamicProxy({
port: 5100,
verbose: true,
sessionMap: sessionMap,
sessionMap,
proxyPathPrefix: "/test_data/interactivetool/ep",
});
proxy.listen();

const headers = {
host: "usegalaxy.org",
};
const path =
"/test_data/interactivetool/ep/coolkey/cooltoken/extradir/README.md";
await verifyProxyOnPort(5100, headers, path);

proxy.close();
});
});

describe("map based path forwarding to top-level path (default)", function () {
it("should respect session map without requires_path_in_url and strip entry point path from url", async function () {
describe("map based path forwarding to top-level path (default)", () => {
it("should respect session map without requires_path_in_url and strip entry point path", async () => {
const sessionMap = {
coolkey: {
token: "cooltoken",
Expand All @@ -198,24 +213,26 @@ describe("DynamicProxy", function () {
const proxy = new DynamicProxy({
port: 5101,
verbose: true,
sessionMap: sessionMap,
sessionMap,
proxyPathPrefix: "/interactivetool/ep",
});
proxy.listen();

const headers = {
host: "usegalaxy.org",
};
// "/interactivetool/ep/coolkey/cooltoken" will be stripped from the path and
// "/test_data/extradir/README.md" will be read and validated
// "/interactivetool/ep/coolkey/cooltoken" will be stripped from the path
// so we actually load "/test_data/extradir/README.md".
const path =
"/interactivetool/ep/coolkey/cooltoken/test_data/extradir/README.md";
await verifyProxyOnPort(5101, headers, path);

proxy.close();
});
});

describe("map based path forwarding to top-level path (requires_path_in_url=false)", function () {
it("should respect session map with requires_path_in_url=false and strip entry point path from url", async function () {
describe("map based path forwarding to top-level path (requires_path_in_url=false)", () => {
it("should respect session map with requires_path_in_url=false", async () => {
const sessionMap = {
coolkey: {
token: "cooltoken",
Expand All @@ -229,64 +246,63 @@ describe("DynamicProxy", function () {
const proxy = new DynamicProxy({
port: 5102,
verbose: true,
sessionMap: sessionMap,
sessionMap,
proxyPathPrefix: "/interactivetool/ep",
});
proxy.listen();

const headers = {
host: "usegalaxy.org",
};
// "/interactivetool/ep/coolkey/cooltoken" will be stripped from the path and
// "/test_data/extradir/README.md" will be read and validated
// "/interactivetool/ep/coolkey/cooltoken" is stripped from the path
// so we actually load "/test_data/extradir/README.md".
const path =
"/interactivetool/ep/coolkey/cooltoken/test_data/extradir/README.md";
await verifyProxyOnPort(5102, headers, path);

proxy.close();
});
});

describe("map based path forwarding to top-level path with entry point path in header", function () {
it(
'should respect session map with requires_path_in_header_named="X-My-Header", strip entry point path from ' +
'url and instead provide it in header "X-My-Header"',
async function () {
const sessionMap = {
coolkey: {
token: "cooltoken",
target: {
host: "localhost",
port: TEST_PORT,
},
requires_path_in_header_named: "X-My-Header",
describe("map based path forwarding with entry point path in header", () => {
it('should respect session map with requires_path_in_header_named="X-My-Header"', async () => {
const sessionMap = {
coolkey: {
token: "cooltoken",
target: {
host: "localhost",
port: TEST_PORT,
},
};
const proxy = new DynamicProxy({
port: 5103,
verbose: true,
sessionMap: sessionMap,
proxyPathPrefix: "/interactivetool/ep",
});
proxy.listen();
const headers = {
host: "usegalaxy.org",
};

proxy.proxy.on("proxyReq", function (proxyReq) {
proxyReq
.getHeader("X-My-Header")
.should.equal("/interactivetool/ep/coolkey/cooltoken");
});

const path =
"/interactivetool/ep/coolkey/cooltoken/test_data/extradir/README.md";
await verifyProxyOnPort(5103, headers, path);
proxy.close();
},
);
requires_path_in_header_named: "X-My-Header",
},
};
const proxy = new DynamicProxy({
port: 5103,
verbose: true,
sessionMap,
proxyPathPrefix: "/interactivetool/ep",
});
proxy.listen();

const headers = { host: "usegalaxy.org" };

proxy.proxy.on("proxyReq", (proxyReq) => {
// Instead of proxyReq.getHeader(...).should.equal(...):
expect(proxyReq.getHeader("X-My-Header")).toBe(
"/interactivetool/ep/coolkey/cooltoken",
);
});

const path =
"/interactivetool/ep/coolkey/cooltoken/test_data/extradir/README.md";
await verifyProxyOnPort(5103, headers, path);

proxy.close();
});
});

describe("double proxying", function () {
it("should proxy across two servers", async function () {
describe("double proxying", () => {
it("should proxy across two servers", async () => {
const sessionMap = {
coolkey: {
token: "cooltoken",
Expand All @@ -299,32 +315,37 @@ describe("DynamicProxy", function () {
const outerProxy = new DynamicProxy({
port: 5200,
verbose: true,
sessionMap: sessionMap,
sessionMap,
forwardIP: "localhost",
forwardPort: 5201,
});
const innerProxy = new DynamicProxy({ port: 5201, verbose: true });
outerProxy.listen();
innerProxy.listen();

const headers = {
host: "coolkey-cooltoken.usegalaxy.org",
};
await verifyProxyOnPort(5200, headers);

innerProxy.close();
outerProxy.close();
});
});
});

describe("Main function", function () {
describe("Main function", () => {
useTestServer();
it("should parse simple arguments and start proxy", async function () {

it("should parse simple arguments and start proxy", async () => {
const proxy = main(["nodejs", "coolproxy", "--port", "5300", "--verbose"]);

const headers = {
"x-interactive-tool-host": "localhost",
"x-interactive-tool-port": TEST_PORT,
};
await verifyProxyOnPort(5300, headers);

proxy.close();
});
});

0 comments on commit afe1454

Please sign in to comment.