-
-
Notifications
You must be signed in to change notification settings - Fork 1.2k
Expand file tree
/
Copy pathorganizationAccessToken.test.ts
More file actions
89 lines (71 loc) · 2.71 KB
/
organizationAccessToken.test.ts
File metadata and controls
89 lines (71 loc) · 2.71 KB
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
import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
const { findFirstMock, updateManyMock } = vi.hoisted(() => ({
findFirstMock: vi.fn(),
updateManyMock: vi.fn(),
}));
vi.mock("~/db.server", () => ({
prisma: {
organizationAccessToken: {
findFirst: findFirstMock,
updateMany: updateManyMock,
},
},
$replica: {},
}));
vi.mock("~/utils/tokens.server", () => ({
hashToken: (t: string) => `hashed:${t}`,
}));
vi.mock("./logger.server", () => ({
logger: { warn: vi.fn(), error: vi.fn() },
}));
import {
authenticateOrganizationAccessToken,
OAT_LAST_ACCESSED_THROTTLE_MS,
} from "~/services/organizationAccessToken.server";
beforeEach(() => {
vi.useFakeTimers();
vi.setSystemTime(new Date("2026-01-01T00:00:00.000Z"));
findFirstMock.mockReset();
updateManyMock.mockReset();
updateManyMock.mockResolvedValue({ count: 1 });
});
afterEach(() => {
vi.useRealTimers();
});
describe("authenticateOrganizationAccessToken — lastAccessedAt throttle", () => {
test("issues a conditional updateMany that skips writes when lastAccessedAt is recent", async () => {
findFirstMock.mockResolvedValueOnce({
id: "oat_123",
organizationId: "org_1",
hashedToken: "hashed:tr_oat_validtoken",
});
const result = await authenticateOrganizationAccessToken("tr_oat_validtoken");
expect(result).toEqual({ organizationId: "org_1" });
expect(updateManyMock).toHaveBeenCalledTimes(1);
const call = updateManyMock.mock.calls[0][0];
expect(call.where.id).toBe("oat_123");
expect(call.where.revokedAt).toBeNull();
expect(call.data.lastAccessedAt).toBeInstanceOf(Date);
// The WHERE clause should require the existing lastAccessedAt to be null
// or strictly older than the throttle window — that's the entire point.
expect(call.where.OR).toEqual([
{ lastAccessedAt: null },
{ lastAccessedAt: { lt: expect.any(Date) } },
]);
// With fake timers, the cutoff lands exactly throttle-ms before "now".
const cutoff = call.where.OR[1].lastAccessedAt.lt as Date;
expect(cutoff.getTime()).toBe(Date.now() - OAT_LAST_ACCESSED_THROTTLE_MS);
});
test("skips updateMany when token is not found", async () => {
findFirstMock.mockResolvedValueOnce(null);
const result = await authenticateOrganizationAccessToken("tr_oat_validtoken");
expect(result).toBeUndefined();
expect(updateManyMock).not.toHaveBeenCalled();
});
test("skips updateMany when token doesn't start with prefix", async () => {
const result = await authenticateOrganizationAccessToken("not_an_oat");
expect(result).toBeUndefined();
expect(findFirstMock).not.toHaveBeenCalled();
expect(updateManyMock).not.toHaveBeenCalled();
});
});