mirror of
https://github.com/gethomepage/homepage.git
synced 2026-04-01 15:52:16 -07:00
165 lines
5.1 KiB
JavaScript
165 lines
5.1 KiB
JavaScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
import createMockRes from "test-utils/create-mock-res";
|
|
|
|
const { httpProxy, getServiceWidget, cache, logger } = vi.hoisted(() => {
|
|
const store = new Map();
|
|
return {
|
|
httpProxy: vi.fn(),
|
|
getServiceWidget: vi.fn(),
|
|
cache: {
|
|
get: vi.fn((k) => store.get(k)),
|
|
put: vi.fn((k, v) => store.set(k, v)),
|
|
del: vi.fn((k) => store.delete(k)),
|
|
_reset: () => store.clear(),
|
|
},
|
|
logger: {
|
|
debug: vi.fn(),
|
|
error: vi.fn(),
|
|
},
|
|
};
|
|
});
|
|
|
|
vi.mock("memory-cache", () => ({
|
|
default: cache,
|
|
...cache,
|
|
}));
|
|
vi.mock("utils/logger", () => ({
|
|
default: () => logger,
|
|
}));
|
|
vi.mock("utils/config/service-helpers", () => ({
|
|
default: getServiceWidget,
|
|
}));
|
|
vi.mock("utils/proxy/http", () => ({
|
|
httpProxy,
|
|
}));
|
|
vi.mock("widgets/widgets", () => ({
|
|
default: {
|
|
crowdsec: {
|
|
api: "{url}/{endpoint}",
|
|
loginURL: "{url}/login",
|
|
},
|
|
},
|
|
}));
|
|
|
|
import crowdsecProxyHandler from "./proxy";
|
|
|
|
describe("widgets/crowdsec/proxy", () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
cache._reset();
|
|
});
|
|
|
|
it("logs in, caches a token, and uses it for requests", async () => {
|
|
getServiceWidget.mockResolvedValue({
|
|
type: "crowdsec",
|
|
url: "http://cs",
|
|
username: "machine",
|
|
password: "pw",
|
|
});
|
|
|
|
httpProxy
|
|
.mockResolvedValueOnce([
|
|
200,
|
|
"application/json",
|
|
JSON.stringify({ token: "tok", expire: new Date(Date.now() + 60_000).toISOString() }),
|
|
])
|
|
.mockResolvedValueOnce([200, "application/json", Buffer.from("data")]);
|
|
|
|
const req = { query: { group: "g", service: "svc", endpoint: "alerts", index: "0" } };
|
|
const res = createMockRes();
|
|
|
|
await crowdsecProxyHandler(req, res);
|
|
|
|
expect(httpProxy).toHaveBeenCalledTimes(2);
|
|
expect(httpProxy.mock.calls[1][1].headers.Authorization).toBe("Bearer tok");
|
|
expect(res.statusCode).toBe(200);
|
|
expect(res.body).toEqual(Buffer.from("data"));
|
|
});
|
|
|
|
it("returns 500 if token cannot be obtained", async () => {
|
|
getServiceWidget.mockResolvedValue({ type: "crowdsec", url: "http://cs", username: "machine", password: "pw" });
|
|
httpProxy.mockResolvedValueOnce([200, "application/json", JSON.stringify({ expire: "2099-01-01T00:00:00Z" })]);
|
|
|
|
const req = { query: { group: "g", service: "svc", endpoint: "alerts", index: "0" } };
|
|
const res = createMockRes();
|
|
|
|
await crowdsecProxyHandler(req, res);
|
|
|
|
expect(res.statusCode).toBe(500);
|
|
expect(res.body).toEqual({ error: "Failed to authenticate with Crowdsec" });
|
|
});
|
|
|
|
it("re-authenticates and retries once when API returns 401", async () => {
|
|
getServiceWidget.mockResolvedValue({
|
|
type: "crowdsec",
|
|
url: "http://cs",
|
|
username: "machine",
|
|
password: "pw",
|
|
});
|
|
|
|
httpProxy
|
|
.mockResolvedValueOnce([
|
|
200,
|
|
"application/json",
|
|
JSON.stringify({ token: "tok-old", expire: new Date(Date.now() + 60_000).toISOString() }),
|
|
])
|
|
.mockResolvedValueOnce([401, "application/json", Buffer.from("bad token")])
|
|
.mockResolvedValueOnce([
|
|
200,
|
|
"application/json",
|
|
JSON.stringify({ token: "tok-new", expire: new Date(Date.now() + 60_000).toISOString() }),
|
|
])
|
|
.mockResolvedValueOnce([200, "application/json", Buffer.from("data")]);
|
|
|
|
const req = { query: { group: "g", service: "svc", endpoint: "alerts", index: "0" } };
|
|
const res = createMockRes();
|
|
|
|
await crowdsecProxyHandler(req, res);
|
|
|
|
expect(httpProxy).toHaveBeenCalledTimes(4);
|
|
expect(httpProxy.mock.calls[3][1].headers.Authorization).toBe("Bearer tok-new");
|
|
expect(res.statusCode).toBe(200);
|
|
expect(res.body).toEqual(Buffer.from("data"));
|
|
});
|
|
|
|
it("returns 500 when 401 refresh fails to get a new token", async () => {
|
|
getServiceWidget.mockResolvedValue({
|
|
type: "crowdsec",
|
|
url: "http://cs",
|
|
username: "machine",
|
|
password: "pw",
|
|
});
|
|
|
|
httpProxy
|
|
.mockResolvedValueOnce([
|
|
200,
|
|
"application/json",
|
|
JSON.stringify({ token: "tok-old", expire: new Date(Date.now() + 60_000).toISOString() }),
|
|
])
|
|
.mockResolvedValueOnce([401, "application/json", Buffer.from("bad token")])
|
|
.mockResolvedValueOnce([500, "application/json", JSON.stringify({ error: "no token" })]);
|
|
|
|
const req = { query: { group: "g", service: "svc", endpoint: "alerts", index: "0" } };
|
|
const res = createMockRes();
|
|
|
|
await crowdsecProxyHandler(req, res);
|
|
|
|
expect(res.statusCode).toBe(500);
|
|
expect(res.body).toEqual({ error: "Failed to authenticate with Crowdsec" });
|
|
});
|
|
|
|
it("returns 500 when login response is not JSON", async () => {
|
|
getServiceWidget.mockResolvedValue({ type: "crowdsec", url: "http://cs", username: "machine", password: "pw" });
|
|
httpProxy.mockResolvedValueOnce([200, "text/plain", "not-json"]);
|
|
|
|
const req = { query: { group: "g", service: "svc", endpoint: "alerts", index: "0" } };
|
|
const res = createMockRes();
|
|
|
|
await crowdsecProxyHandler(req, res);
|
|
|
|
expect(res.statusCode).toBe(500);
|
|
expect(res.body).toEqual({ error: "Failed to authenticate with Crowdsec" });
|
|
});
|
|
});
|