openclaw/src/infra/system-presence.test.ts
2026-03-13 19:03:01 +00:00

121 lines
3.8 KiB
TypeScript

import { randomUUID } from "node:crypto";
import { afterEach, describe, expect, it, vi } from "vitest";
import { listSystemPresence, updateSystemPresence, upsertPresence } from "./system-presence.js";
describe("system-presence", () => {
afterEach(() => {
vi.useRealTimers();
});
it("dedupes entries across sources by case-insensitive instanceId key", () => {
const instanceIdUpper = `AaBb-${randomUUID()}`.toUpperCase();
const instanceIdLower = instanceIdUpper.toLowerCase();
upsertPresence(instanceIdUpper, {
host: "openclaw",
mode: "ui",
instanceId: instanceIdUpper,
reason: "connect",
});
updateSystemPresence({
text: "Node: Peter-Mac-Studio (10.0.0.1) · ui 2.0.0 · last input 5s ago · mode ui · reason beacon",
instanceId: instanceIdLower,
host: "Peter-Mac-Studio",
ip: "10.0.0.1",
mode: "ui",
version: "2.0.0",
lastInputSeconds: 5,
reason: "beacon",
});
const matches = listSystemPresence().filter(
(e) => (e.instanceId ?? "").toLowerCase() === instanceIdLower,
);
expect(matches).toHaveLength(1);
expect(matches[0]?.host).toBe("Peter-Mac-Studio");
expect(matches[0]?.ip).toBe("10.0.0.1");
expect(matches[0]?.lastInputSeconds).toBe(5);
});
it("merges roles and scopes for the same device", () => {
const deviceId = randomUUID();
upsertPresence(deviceId, {
deviceId,
host: "openclaw",
roles: ["operator"],
scopes: ["operator.admin"],
reason: "connect",
});
upsertPresence(deviceId, {
deviceId,
roles: ["node"],
scopes: ["system.run"],
reason: "connect",
});
const entry = listSystemPresence().find((e) => e.deviceId === deviceId);
expect(entry?.roles).toEqual(expect.arrayContaining(["operator", "node"]));
expect(entry?.scopes).toEqual(expect.arrayContaining(["operator.admin", "system.run"]));
});
it("parses node presence text and normalizes the update key", () => {
const update = updateSystemPresence({
text: "Node: Relay-Host (10.0.0.9) · app 2.1.0 · last input 7s ago · mode ui · reason beacon",
instanceId: " Mixed-Case-Node ",
});
expect(update.key).toBe("mixed-case-node");
expect(update.changedKeys).toEqual(["host", "ip", "version", "mode", "reason"]);
expect(update.next).toMatchObject({
host: "Relay-Host",
ip: "10.0.0.9",
version: "2.1.0",
lastInputSeconds: 7,
mode: "ui",
reason: "beacon",
text: "Node: Relay-Host (10.0.0.9) · app 2.1.0 · last input 7s ago · mode ui · reason beacon",
});
});
it("drops blank role and scope entries while keeping fallback text", () => {
const deviceId = randomUUID();
upsertPresence(deviceId, {
deviceId,
host: "relay-host",
mode: "operator",
roles: [" operator ", "", " "],
scopes: ["operator.admin", "", " "],
});
const entry = listSystemPresence().find((candidate) => candidate.deviceId === deviceId);
expect(entry?.roles).toEqual(["operator"]);
expect(entry?.scopes).toEqual(["operator.admin"]);
expect(entry?.text).toBe("Node: relay-host · mode operator");
});
it("prunes stale non-self entries after TTL", () => {
vi.useFakeTimers();
vi.setSystemTime(Date.now());
const deviceId = randomUUID();
upsertPresence(deviceId, {
deviceId,
host: "stale-host",
mode: "ui",
reason: "connect",
});
expect(listSystemPresence().some((entry) => entry.deviceId === deviceId)).toBe(true);
vi.advanceTimersByTime(5 * 60 * 1000 + 1);
const entries = listSystemPresence();
expect(entries.some((entry) => entry.deviceId === deviceId)).toBe(false);
expect(entries.some((entry) => entry.reason === "self")).toBe(true);
});
});