Files
openclaw/src/gateway/channel-health-policy.test.ts
Peter Steinberger 9d7d961db8 fix: restore Telegram webhook-mode health after restarts
Landed from contributor PR #39313 by @fellanH.

Co-authored-by: Felix Hellström <30758862+fellanH@users.noreply.github.com>
2026-03-08 02:27:18 +00:00

272 lines
7.0 KiB
TypeScript

import { describe, expect, it } from "vitest";
import { evaluateChannelHealth, resolveChannelRestartReason } from "./channel-health-policy.js";
describe("evaluateChannelHealth", () => {
it("treats disabled accounts as healthy unmanaged", () => {
const evaluation = evaluateChannelHealth(
{
running: false,
enabled: false,
configured: true,
},
{
channelId: "discord",
now: 100_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "unmanaged" });
});
it("uses channel connect grace before flagging disconnected", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: false,
enabled: true,
configured: true,
lastStartAt: 95_000,
},
{
channelId: "discord",
now: 100_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "startup-connect-grace" });
});
it("treats active runs as busy even when disconnected", () => {
const now = 100_000;
const evaluation = evaluateChannelHealth(
{
running: true,
connected: false,
enabled: true,
configured: true,
activeRuns: 1,
lastRunActivityAt: now - 30_000,
},
{
channelId: "discord",
now,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "busy" });
});
it("flags stale busy channels as stuck when run activity is too old", () => {
const now = 100_000;
const evaluation = evaluateChannelHealth(
{
running: true,
connected: false,
enabled: true,
configured: true,
activeRuns: 1,
lastRunActivityAt: now - 26 * 60_000,
},
{
channelId: "discord",
now,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: false, reason: "stuck" });
});
it("ignores inherited busy flags until current lifecycle reports run activity", () => {
const now = 100_000;
const evaluation = evaluateChannelHealth(
{
running: true,
connected: false,
enabled: true,
configured: true,
lastStartAt: now - 30_000,
busy: true,
activeRuns: 1,
lastRunActivityAt: now - 31_000,
},
{
channelId: "discord",
now,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: false, reason: "disconnected" });
});
it("flags stale sockets when no events arrive beyond threshold", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: true,
enabled: true,
configured: true,
lastStartAt: 0,
lastEventAt: 0,
},
{
channelId: "discord",
now: 100_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: false, reason: "stale-socket" });
});
it("skips stale-socket detection for telegram long-polling channels", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: true,
enabled: true,
configured: true,
lastStartAt: 0,
lastEventAt: null,
},
{
channelId: "telegram",
now: 100_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "healthy" });
});
it("skips stale-socket detection for channels in webhook mode", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: true,
enabled: true,
configured: true,
lastStartAt: 0,
lastEventAt: 0,
mode: "webhook",
},
{
channelId: "discord",
now: 100_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "healthy" });
});
it("does not flag stale sockets for channels without event tracking", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: true,
enabled: true,
configured: true,
lastStartAt: 0,
lastEventAt: null,
},
{
channelId: "discord",
now: 100_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "healthy" });
});
it("does not flag stale sockets without an active connected socket", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
enabled: true,
configured: true,
lastStartAt: 0,
lastEventAt: 0,
},
{
channelId: "slack",
now: 75_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "healthy" });
});
it("ignores inherited event timestamps from a previous lifecycle", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: true,
enabled: true,
configured: true,
lastStartAt: 50_000,
lastEventAt: 10_000,
},
{
channelId: "slack",
now: 75_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: true, reason: "healthy" });
});
it("flags inherited event timestamps after the lifecycle exceeds the stale threshold", () => {
const evaluation = evaluateChannelHealth(
{
running: true,
connected: true,
enabled: true,
configured: true,
lastStartAt: 50_000,
lastEventAt: 10_000,
},
{
channelId: "slack",
now: 140_000,
channelConnectGraceMs: 10_000,
staleEventThresholdMs: 30_000,
},
);
expect(evaluation).toEqual({ healthy: false, reason: "stale-socket" });
});
});
describe("resolveChannelRestartReason", () => {
it("maps not-running + high reconnect attempts to gave-up", () => {
const reason = resolveChannelRestartReason(
{
running: false,
reconnectAttempts: 10,
},
{ healthy: false, reason: "not-running" },
);
expect(reason).toBe("gave-up");
});
it("maps disconnected to disconnected instead of stuck", () => {
const reason = resolveChannelRestartReason(
{
running: true,
connected: false,
enabled: true,
configured: true,
},
{ healthy: false, reason: "disconnected" },
);
expect(reason).toBe("disconnected");
});
});