test: cover middleware + core utils (logger, hooks, proxy)

This commit is contained in:
shamoon 2026-02-04 09:23:36 -08:00
parent bcdd4166a3
commit dfb25a2d61
6 changed files with 371 additions and 0 deletions

72
src/middleware.test.js Normal file
View File

@ -0,0 +1,72 @@
import { beforeEach, describe, expect, it, vi } from "vitest";
const { NextResponse } = vi.hoisted(() => ({
NextResponse: {
json: vi.fn((body, init) => ({ type: "json", body, init })),
next: vi.fn(() => ({ type: "next" })),
},
}));
vi.mock("next/server", () => ({ NextResponse }));
import { middleware } from "./middleware";
function createReq(host) {
return {
headers: {
get: (key) => (key === "host" ? host : null),
},
};
}
describe("middleware", () => {
const originalEnv = process.env;
const originalConsoleError = console.error;
beforeEach(() => {
vi.clearAllMocks();
process.env = { ...originalEnv };
console.error = originalConsoleError;
});
it("allows requests for default localhost hosts", () => {
process.env.PORT = "3000";
const res = middleware(createReq("localhost:3000"));
expect(NextResponse.next).toHaveBeenCalled();
expect(res).toEqual({ type: "next" });
});
it("blocks requests when host is not allowed", () => {
process.env.PORT = "3000";
const errSpy = vi.spyOn(console, "error").mockImplementation(() => {});
const res = middleware(createReq("evil.com"));
expect(errSpy).toHaveBeenCalled();
expect(NextResponse.json).toHaveBeenCalledWith(
{ error: "Host validation failed. See logs for more details." },
{ status: 400 },
);
expect(res.type).toBe("json");
expect(res.init.status).toBe(400);
});
it("allows requests when HOMEPAGE_ALLOWED_HOSTS is '*'", () => {
process.env.HOMEPAGE_ALLOWED_HOSTS = "*";
const res = middleware(createReq("anything.example"));
expect(NextResponse.next).toHaveBeenCalled();
expect(res).toEqual({ type: "next" });
});
it("allows requests when host is included in HOMEPAGE_ALLOWED_HOSTS", () => {
process.env.PORT = "3000";
process.env.HOMEPAGE_ALLOWED_HOSTS = "example.com:3000,other:3000";
const res = middleware(createReq("example.com:3000"));
expect(NextResponse.next).toHaveBeenCalled();
expect(res).toEqual({ type: "next" });
});
});

View File

@ -0,0 +1,27 @@
// @vitest-environment jsdom
import { render, screen, waitFor } from "@testing-library/react";
import { describe, expect, it, vi } from "vitest";
import useWindowFocus from "./window-focus";
function Fixture() {
const focused = useWindowFocus();
return <div data-testid="focused">{String(focused)}</div>;
}
describe("utils/hooks/window-focus", () => {
it("tracks focus/blur events", async () => {
vi.spyOn(document, "hasFocus").mockReturnValue(true);
render(<Fixture />);
expect(screen.getByTestId("focused")).toHaveTextContent("true");
window.dispatchEvent(new Event("blur"));
await waitFor(() => expect(screen.getByTestId("focused")).toHaveTextContent("false"));
window.dispatchEvent(new Event("focus"));
await waitFor(() => expect(screen.getByTestId("focused")).toHaveTextContent("true"));
});
});

View File

@ -0,0 +1,12 @@
import { describe, expect, it } from "vitest";
import { columnMap } from "./columns";
describe("utils/layout/columns", () => {
it("maps column counts to responsive grid classes", () => {
expect(columnMap).toHaveLength(9);
expect(columnMap[1]).toContain("grid-cols-1");
expect(columnMap[2]).toContain("md:grid-cols-2");
expect(columnMap[8]).toContain("lg:grid-cols-8");
});
});

99
src/utils/logger.test.js Normal file
View File

@ -0,0 +1,99 @@
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
const { state, winston, checkAndCopyConfig, getSettings } = vi.hoisted(() => {
const state = {
created: [],
lastCreateLoggerArgs: null,
};
function ConsoleTransport(opts) {
this.opts = opts;
}
function FileTransport(opts) {
this.opts = opts;
}
const createLogger = vi.fn((args) => {
state.lastCreateLoggerArgs = args;
const base = {
child: vi.fn(() => base),
debug: vi.fn(),
info: vi.fn(),
warn: vi.fn(),
error: vi.fn(),
};
state.created.push(base);
return base;
});
const winston = {
transports: { Console: ConsoleTransport, File: FileTransport },
format: {
combine: (...parts) => ({ parts }),
errors: () => ({}),
timestamp: () => ({}),
colorize: () => ({}),
printf: (fn) => fn,
},
createLogger,
};
return {
state,
winston,
checkAndCopyConfig: vi.fn(),
getSettings: vi.fn(() => ({ logpath: "/tmp" })),
};
});
vi.mock("winston", () => ({ default: winston, ...winston }));
vi.mock("utils/config/config", () => ({
default: checkAndCopyConfig,
CONF_DIR: "/conf",
getSettings,
}));
describe("utils/logger", () => {
const originalEnv = process.env;
const originalConsole = { ...console };
beforeEach(() => {
vi.clearAllMocks();
process.env = { ...originalEnv };
});
afterEach(() => {
// Restore patched console methods if init() ran.
Object.assign(console, originalConsole);
});
it("initializes winston on first createLogger() and caches per label", async () => {
vi.resetModules();
process.env.LOG_TARGETS = "stdout";
const createLogger = (await import("./logger")).default;
const a1 = createLogger("a");
const a2 = createLogger("a");
const b = createLogger("b");
expect(checkAndCopyConfig).toHaveBeenCalledWith("settings.yaml");
expect(winston.createLogger).toHaveBeenCalled();
expect(a1).toBe(a2);
expect(b).toBeDefined();
});
it("selects stdout/file/both transports based on LOG_TARGETS", async () => {
vi.resetModules();
process.env.LOG_TARGETS = "file";
const createLogger = (await import("./logger")).default;
createLogger("x");
const transports = state.lastCreateLoggerArgs.transports;
expect(transports).toHaveLength(1);
expect(transports[0].opts.filename).toBe("/tmp/logs/homepage.log");
});
});

View File

@ -0,0 +1,112 @@
import { beforeEach, describe, expect, it, vi } from "vitest";
const { state, cache, logger } = vi.hoisted(() => ({
state: {
response: {
statusCode: 200,
headers: { "content-type": "application/json" },
body: Buffer.from(""),
},
error: null,
},
cache: {
get: vi.fn(),
put: vi.fn(),
},
logger: {
debug: vi.fn(),
error: vi.fn(),
},
}));
vi.mock("follow-redirects", async () => {
const { EventEmitter } = await import("node:events");
const { Readable } = await import("node:stream");
function Agent(opts) {
this.opts = opts;
}
function makeRequest() {
return (url, params, cb) => {
const req = new EventEmitter();
req.write = vi.fn();
req.end = vi.fn(() => {
if (state.error) {
req.emit("error", state.error);
return;
}
const res = new Readable({
read() {
this.push(state.response.body);
this.push(null);
},
});
res.statusCode = state.response.statusCode;
res.headers = state.response.headers;
cb(res);
});
return req;
};
}
return {
http: { request: makeRequest(), Agent },
https: { request: makeRequest(), Agent },
};
});
vi.mock("memory-cache", () => ({
default: cache,
}));
vi.mock("utils/logger", () => ({
default: () => logger,
}));
describe("utils/proxy/http cachedRequest", () => {
beforeEach(() => {
vi.clearAllMocks();
state.error = null;
state.response = {
statusCode: 200,
headers: { "content-type": "application/json" },
body: Buffer.from(""),
};
vi.resetModules();
});
it("returns cached values without calling httpProxy", async () => {
cache.get.mockReturnValueOnce({ ok: true });
const httpMod = await import("./http");
const spy = vi.spyOn(httpMod, "httpProxy");
const data = await httpMod.cachedRequest("http://example.com");
expect(data).toEqual({ ok: true });
expect(spy).not.toHaveBeenCalled();
});
it("parses json buffer responses and caches the result", async () => {
cache.get.mockReturnValueOnce(null);
state.response.body = Buffer.from('{"a":1}');
const httpMod = await import("./http");
const data = await httpMod.cachedRequest("http://example.com/data", 1, "ua");
expect(data).toEqual({ a: 1 });
expect(cache.put).toHaveBeenCalledWith("http://example.com/data", { a: 1 }, 1 * 1000 * 60);
});
it("falls back to string when cachedRequest cannot parse json", async () => {
cache.get.mockReturnValueOnce(null);
state.response.body = Buffer.from("not-json");
const httpMod = await import("./http");
const data = await httpMod.cachedRequest("http://example.com/data", 1, "ua");
expect(data).toBe("not-json");
expect(logger.debug).toHaveBeenCalled();
});
});

View File

@ -0,0 +1,49 @@
import { beforeEach, describe, expect, it, vi } from "vitest";
const { useSWR } = vi.hoisted(() => ({ useSWR: vi.fn() }));
vi.mock("swr", () => ({
default: useSWR,
}));
import useWidgetAPI from "./use-widget-api";
describe("utils/proxy/use-widget-api", () => {
beforeEach(() => {
vi.clearAllMocks();
});
it("formats the proxy url and passes refreshInterval when provided in options", () => {
useSWR.mockReturnValue({ data: { ok: true }, error: undefined, mutate: "m" });
const widget = { service_group: "g", service_name: "s", index: 0 };
const result = useWidgetAPI(widget, "status", { refreshInterval: 123, foo: "bar" });
expect(useSWR).toHaveBeenCalledWith(
expect.stringContaining("/api/services/proxy?"),
expect.objectContaining({ refreshInterval: 123 }),
);
expect(result.data).toEqual({ ok: true });
expect(result.error).toBeUndefined();
expect(result.mutate).toBe("m");
});
it("returns data.error as the top-level error", () => {
const dataError = { message: "nope" };
useSWR.mockReturnValue({ data: { error: dataError }, error: undefined, mutate: vi.fn() });
const widget = { service_group: "g", service_name: "s", index: 0 };
const result = useWidgetAPI(widget, "status", {});
expect(result.error).toBe(dataError);
});
it("disables the request when endpoint is an empty string", () => {
useSWR.mockReturnValue({ data: undefined, error: undefined, mutate: vi.fn() });
const widget = { service_group: "g", service_name: "s", index: 0 };
useWidgetAPI(widget, "");
expect(useSWR).toHaveBeenCalledWith(null, {});
});
});