Files
caddy-proxy-manager/tests/unit/api-routes/l4-proxy-hosts.test.ts
akanealw 99819b70ff
Some checks failed
Build and Push Docker Images (Trusted) / build-and-push (., docker/caddy/Dockerfile, caddy) (push) Has been cancelled
Build and Push Docker Images (Trusted) / build-and-push (., docker/l4-port-manager/Dockerfile, l4-port-manager) (push) Has been cancelled
Build and Push Docker Images (Trusted) / build-and-push (., docker/web/Dockerfile, web) (push) Has been cancelled
Tests / test (push) Has been cancelled
added caddy-proxy-manager for testing
2026-04-21 22:49:08 +00:00

292 lines
9.5 KiB
TypeScript
Executable File

import { describe, it, expect, vi, beforeEach } from 'vitest';
vi.mock('@/src/lib/models/l4-proxy-hosts', () => ({
listL4ProxyHosts: vi.fn(),
createL4ProxyHost: vi.fn(),
getL4ProxyHost: vi.fn(),
updateL4ProxyHost: vi.fn(),
deleteL4ProxyHost: vi.fn(),
}));
vi.mock('@/src/lib/api-auth', () => {
const ApiAuthError = class extends Error {
status: number;
constructor(msg: string, status: number) { super(msg); this.status = status; this.name = 'ApiAuthError'; }
};
return {
requireApiAdmin: vi.fn().mockResolvedValue({ userId: 1, role: 'admin', authMethod: 'bearer' }),
requireApiUser: vi.fn().mockResolvedValue({ userId: 1, role: 'admin', authMethod: 'bearer' }),
apiErrorResponse: vi.fn((error: unknown) => {
const { NextResponse: NR } = require('next/server');
if (error instanceof ApiAuthError) {
return NR.json({ error: error.message }, { status: error.status });
}
return NR.json({ error: error instanceof Error ? error.message : 'Internal server error' }, { status: 500 });
}),
ApiAuthError,
};
});
import { GET as listGET, POST } from '@/app/api/v1/l4-proxy-hosts/route';
import { GET as getGET, PUT, DELETE } from '@/app/api/v1/l4-proxy-hosts/[id]/route';
import { listL4ProxyHosts, createL4ProxyHost, getL4ProxyHost, updateL4ProxyHost, deleteL4ProxyHost } from '@/src/lib/models/l4-proxy-hosts';
import { requireApiAdmin } from '@/src/lib/api-auth';
const mockList = vi.mocked(listL4ProxyHosts);
const mockCreate = vi.mocked(createL4ProxyHost);
const mockGet = vi.mocked(getL4ProxyHost);
const mockUpdate = vi.mocked(updateL4ProxyHost);
const mockDelete = vi.mocked(deleteL4ProxyHost);
const mockRequireApiAdmin = vi.mocked(requireApiAdmin);
function createMockRequest(options: { method?: string; body?: unknown } = {}): any {
return {
headers: { get: () => null },
method: options.method ?? 'GET',
nextUrl: { pathname: '/api/v1/l4-proxy-hosts', searchParams: new URLSearchParams() },
json: async () => options.body ?? {},
};
}
const sampleHost = {
id: 1,
name: 'SSH Forward',
listen_port: 2222,
forward_host: '10.0.0.5',
forward_port: 22,
protocol: 'tcp',
enabled: true,
created_at: '2026-01-01',
};
beforeEach(() => {
vi.clearAllMocks();
mockRequireApiAdmin.mockResolvedValue({ userId: 1, role: 'admin', authMethod: 'bearer' });
});
describe('GET /api/v1/l4-proxy-hosts', () => {
it('returns list of L4 proxy hosts', async () => {
mockList.mockResolvedValue([sampleHost] as any);
const response = await listGET(createMockRequest());
const data = await response.json();
expect(response.status).toBe(200);
expect(data).toEqual([sampleHost]);
});
it('returns 401 on auth failure', async () => {
const { ApiAuthError } = await import('@/src/lib/api-auth');
mockRequireApiAdmin.mockRejectedValue(new ApiAuthError('Unauthorized', 401));
const response = await listGET(createMockRequest());
expect(response.status).toBe(401);
});
});
describe('POST /api/v1/l4-proxy-hosts', () => {
it('creates an L4 proxy host and returns 201', async () => {
const body = { name: 'New L4', listen_port: 3333, forward_host: '10.0.0.6', forward_port: 33 };
mockCreate.mockResolvedValue({ id: 2, ...body } as any);
const response = await POST(createMockRequest({ method: 'POST', body }));
const data = await response.json();
expect(response.status).toBe(201);
expect(data.id).toBe(2);
expect(mockCreate).toHaveBeenCalledWith(body, 1);
});
});
describe('GET /api/v1/l4-proxy-hosts/[id]', () => {
it('returns an L4 proxy host by id', async () => {
mockGet.mockResolvedValue(sampleHost as any);
const response = await getGET(createMockRequest(), { params: Promise.resolve({ id: '1' }) });
const data = await response.json();
expect(response.status).toBe(200);
expect(data).toEqual(sampleHost);
});
it('returns 404 for non-existent host', async () => {
mockGet.mockResolvedValue(null as any);
const response = await getGET(createMockRequest(), { params: Promise.resolve({ id: '999' }) });
const data = await response.json();
expect(response.status).toBe(404);
expect(data.error).toBe('Not found');
});
});
describe('PUT /api/v1/l4-proxy-hosts/[id]', () => {
it('updates an L4 proxy host', async () => {
const body = { listen_port: 4444 };
mockUpdate.mockResolvedValue({ ...sampleHost, listen_port: 4444 } as any);
const response = await PUT(createMockRequest({ method: 'PUT', body }), { params: Promise.resolve({ id: '1' }) });
const data = await response.json();
expect(response.status).toBe(200);
expect(data.listen_port).toBe(4444);
expect(mockUpdate).toHaveBeenCalledWith(1, body, 1);
});
it('returns 500 when host not found', async () => {
mockUpdate.mockRejectedValue(new Error('not found'));
const response = await PUT(createMockRequest({ method: 'PUT', body: { listen_port: 4444 } }), { params: Promise.resolve({ id: '999' }) });
const data = await response.json();
expect(response.status).toBe(500);
expect(data.error).toBe('not found');
});
});
describe('DELETE /api/v1/l4-proxy-hosts/[id]', () => {
it('deletes an L4 proxy host', async () => {
mockDelete.mockResolvedValue(undefined as any);
const response = await DELETE(createMockRequest({ method: 'DELETE' }), { params: Promise.resolve({ id: '1' }) });
const data = await response.json();
expect(response.status).toBe(200);
expect(data).toEqual({ ok: true });
expect(mockDelete).toHaveBeenCalledWith(1, 1);
});
it('returns 500 when host not found', async () => {
mockDelete.mockRejectedValue(new Error('not found'));
const response = await DELETE(createMockRequest({ method: 'DELETE' }), { params: Promise.resolve({ id: '999' }) });
const data = await response.json();
expect(response.status).toBe(500);
expect(data.error).toBe('not found');
});
});
describe('POST /api/v1/l4-proxy-hosts (all options)', () => {
it('creates L4 host with all options', async () => {
const fullBody = {
name: "PostgreSQL Proxy",
listen_addresses: [":5432"],
matchers: ["db.example.com"],
upstreams: ["db-primary:5432", "db-replica:5432"],
protocol: "tcp",
matcher_type: "tls_sni",
tls_termination: true,
proxy_protocol_version: "v2",
enabled: true,
load_balancer: {
enabled: true,
policy: "least_conn",
tryDuration: "10s",
tryInterval: "500ms",
retries: 2,
activeHealthCheck: {
enabled: true,
port: 5432,
interval: "15s",
timeout: "3s",
},
passiveHealthCheck: {
enabled: true,
failDuration: "30s",
maxFails: 3,
unhealthyLatency: "5s",
},
},
dns_resolver: {
enabled: true,
resolvers: ["1.1.1.1"],
fallbacks: [],
timeout: "3s",
},
upstream_dns_resolution: {
enabled: true,
family: "ipv4",
},
geoblock: {
enabled: true,
block_countries: ["CN"],
block_continents: [],
block_asns: [],
block_cidrs: [],
block_ips: [],
allow_countries: [],
allow_continents: [],
allow_asns: [],
allow_cidrs: [],
allow_ips: [],
trusted_proxies: [],
fail_closed: true,
response_status: 403,
response_body: "Blocked",
response_headers: {},
redirect_url: "",
},
geoblock_mode: "override",
};
const returnValue = { id: 10, ...fullBody, created_at: '2026-03-26T00:00:00Z', updated_at: '2026-03-26T00:00:00Z' };
mockCreate.mockResolvedValue(returnValue as any);
const response = await POST(createMockRequest({ method: 'POST', body: fullBody }));
const data = await response.json();
expect(response.status).toBe(201);
expect(data.id).toBe(10);
expect(mockCreate).toHaveBeenCalledWith(fullBody, 1);
});
it('creates UDP L4 host without TLS', async () => {
const body = {
name: "DNS Proxy",
listen_addresses: [":53"],
matchers: [],
upstreams: ["dns:53"],
protocol: "udp",
matcher_type: "none",
tls_termination: false,
enabled: true,
};
const returnValue = { id: 11, ...body, created_at: '2026-03-26T00:00:00Z', updated_at: '2026-03-26T00:00:00Z' };
mockCreate.mockResolvedValue(returnValue as any);
const response = await POST(createMockRequest({ method: 'POST', body }));
const data = await response.json();
expect(response.status).toBe(201);
expect(data.id).toBe(11);
expect(data.protocol).toBe("udp");
expect(data.tls_termination).toBe(false);
expect(data.matchers).toEqual([]);
expect(mockCreate).toHaveBeenCalledWith(body, 1);
});
});
describe('PUT /api/v1/l4-proxy-hosts/[id] (partial update)', () => {
it('updates L4 host matcher and protocol', async () => {
const partialBody = {
matcher_type: "http_host",
matchers: ["new.example.com"],
proxy_protocol_version: "v1",
};
const updated = { ...sampleHost, ...partialBody };
mockUpdate.mockResolvedValue(updated as any);
const response = await PUT(createMockRequest({ method: 'PUT', body: partialBody }), { params: Promise.resolve({ id: '1' }) });
const data = await response.json();
expect(response.status).toBe(200);
expect(mockUpdate).toHaveBeenCalledWith(1, partialBody, 1);
expect(data.matcher_type).toBe("http_host");
expect(data.matchers).toEqual(["new.example.com"]);
expect(data.proxy_protocol_version).toBe("v1");
});
});