summaryrefslogtreecommitdiff
path: root/tst/server_request_response.test.ts
blob: 99e74d8d47924c338764d8be768959db9226e390 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
import {
    Either,
    JsonResponse,
    PenguenoRequest,
    PenguenoResponse,
    getResponseMetrics,
    MetricValueTag,
    type BaseRequest,
    type ServerTrace,
} from '../lib/index';
import { CollectingTrace, TestTraceable } from './test_utils';

const makeBaseRequest = (overrides: Partial<BaseRequest> = {}): BaseRequest => ({
    url: 'https://example.com/hello?x=1',
    method: 'GET',
    header: () => ({}),
    formData: async () => new FormData(),
    json: async () => ({}),
    text: async () => '',
    param: () => undefined,
    query: () => ({}),
    queries: () => ({}),
    ...overrides,
});

describe('server/request + server/response', () => {
    beforeEach(() => {
        jest.useFakeTimers();
        jest.setSystemTime(new Date('2020-01-01T00:00:00.000Z'));
    });

    afterEach(() => {
        jest.useRealTimers();
        jest.restoreAllMocks();
    });

    test('PenguenoRequest.from creates response headers', () => {
        jest.spyOn(globalThis.crypto, 'randomUUID').mockReturnValue('00000000-0000-0000-0000-000000000000');
        jest.spyOn(Math, 'random').mockReturnValue(0);

        const baseReq = makeBaseRequest();
        const trace = new CollectingTrace<ServerTrace>();
        const req = PenguenoRequest.from(TestTraceable.of(baseReq, trace));

        jest.setSystemTime(new Date('2020-01-01T00:00:01.000Z'));
        const headers = req.get().getResponseHeaders();

        expect(headers).toMatchObject({
            RequestId: '00000000-0000-0000-0000-000000000000',
            Hai: 'hewwo :D',
        });
        expect(headers.DeltaUnix).toBe('1000');
        expect(headers.RequestReceivedUnix).toBe('1577836800000');
        expect(headers.RequestHandleUnix).toBe('1577836801000');
    });

    test('PenguenoResponse sets headers and emits metrics', () => {
        jest.spyOn(globalThis.crypto, 'randomUUID').mockReturnValue('00000000-0000-0000-0000-000000000000');
        jest.spyOn(Math, 'random').mockReturnValue(0);

        const trace = new CollectingTrace<ServerTrace>();
        const req = PenguenoRequest.from(TestTraceable.of(makeBaseRequest(), trace));

        jest.setSystemTime(new Date('2020-01-01T00:00:01.000Z'));
        const res = new PenguenoResponse(req, 'hi', { status: 200, headers: { X: 'Y' } });

        expect(res.status).toBe(200);
        expect(res.statusText).toBe('OK');
        expect(res.headers['Content-Type']).toBe('text/plain; charset=utf-8');
        expect(res.headers.X).toBe('Y');

        const lastTrace = trace.events[trace.events.length - 1];
        expect(lastTrace).toBeDefined();

        const metricValues = lastTrace[lastTrace.length - 1] as any[];
        expect(Array.isArray(metricValues)).toBe(true);
        expect(
            metricValues.some((m) => m._tag === MetricValueTag && m.name === 'response.2xx.count' && m.value === 1),
        ).toBe(true);
    });

    test('JsonResponse formats Either bodies', () => {
        jest.spyOn(globalThis.crypto, 'randomUUID').mockReturnValue('00000000-0000-0000-0000-000000000000');

        const trace = new CollectingTrace<ServerTrace>();
        const req = PenguenoRequest.from(TestTraceable.of(makeBaseRequest(), trace));

        const ok = new JsonResponse(req, Either.right('yay'), { status: 200, headers: {} });
        expect(ok.headers['Content-Type']).toBe('application/json; charset=utf-8');
        expect(ok.body()).toBe('{"ok":"yay"}');

        const err = new JsonResponse(req, Either.left('nope'), { status: 400, headers: {} });
        expect(err.body()).toBe('{"error":"nope"}');
    });

    test('getResponseMetrics returns one active bucket', () => {
        jest.setSystemTime(new Date('2020-01-01T00:00:00.000Z'));
        const metrics = getResponseMetrics(201, 12);

        expect(metrics).toEqual(
            expect.arrayContaining([
                expect.objectContaining({ name: 'response.2xx.count', value: 1 }),
                expect.objectContaining({ name: 'response.2xx.time', value: 12 }),
            ]),
        );

        const inactive = metrics.filter((m) => m.value === 0);
        expect(inactive).toHaveLength(5);
    });
});