mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-06 13:32:54 +00:00
214 lines
7.3 KiB
TypeScript
214 lines
7.3 KiB
TypeScript
import { describe, expect, it, vi } from "vitest";
|
|
import {
|
|
classifyMSTeamsSendError,
|
|
formatMSTeamsSendErrorHint,
|
|
formatUnknownError,
|
|
isRevokedProxyError,
|
|
} from "./errors.js";
|
|
import { withRevokedProxyFallback } from "./revoked-context.js";
|
|
|
|
describe("msteams errors", () => {
|
|
it("formats unknown errors", () => {
|
|
expect(formatUnknownError("oops")).toBe("oops");
|
|
expect(formatUnknownError(null)).toBe("null");
|
|
});
|
|
|
|
it("classifies auth errors", () => {
|
|
expect(classifyMSTeamsSendError({ statusCode: 401 }).kind).toBe("auth");
|
|
expect(classifyMSTeamsSendError({ statusCode: 403 }).kind).toBe("auth");
|
|
});
|
|
|
|
it("classifies ContentStreamNotAllowed as permanent instead of auth", () => {
|
|
const result = classifyMSTeamsSendError({
|
|
statusCode: 403,
|
|
response: {
|
|
body: {
|
|
error: {
|
|
code: "ContentStreamNotAllowed",
|
|
},
|
|
},
|
|
},
|
|
});
|
|
expect(result.kind).toBe("permanent");
|
|
expect(result.statusCode).toBe(403);
|
|
expect(result.errorCode).toBe("ContentStreamNotAllowed");
|
|
});
|
|
|
|
it("classifies throttling errors and parses retry-after", () => {
|
|
const result = classifyMSTeamsSendError({ statusCode: 429, retryAfter: "1.5" });
|
|
expect(result.kind).toBe("throttled");
|
|
expect(result.statusCode).toBe(429);
|
|
expect(result.retryAfterMs).toBe(1500);
|
|
});
|
|
|
|
it("does not parse partial retry-after values", () => {
|
|
expect(
|
|
classifyMSTeamsSendError({ statusCode: 429, retryAfter: "1.5s" }).retryAfterMs,
|
|
).toBeUndefined();
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
response: { headers: { "retry-after": "2 seconds" } },
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
response: { headers: new Headers({ "retry-after": "3 seconds" }) },
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
});
|
|
|
|
it("ignores unsafe retry-after magnitudes", () => {
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
retryAfterMs: Number.MAX_SAFE_INTEGER + 1,
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
retryAfter: Number.MAX_SAFE_INTEGER,
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
retryAfter: "9007199254741",
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
response: { headers: { "retry-after": "9007199254741" } },
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
expect(
|
|
classifyMSTeamsSendError({
|
|
statusCode: 429,
|
|
response: { headers: new Headers({ "retry-after": "9007199254741" }) },
|
|
}).retryAfterMs,
|
|
).toBeUndefined();
|
|
});
|
|
|
|
it("does not parse partial or fractional status codes", () => {
|
|
expect(classifyMSTeamsSendError({ statusCode: "429oops" }).kind).toBe("unknown");
|
|
expect(classifyMSTeamsSendError({ statusCode: 429.5 }).kind).toBe("unknown");
|
|
expect(
|
|
classifyMSTeamsSendError({ response: { status: "503 temporarily unavailable" } }).kind,
|
|
).toBe("unknown");
|
|
});
|
|
|
|
it("classifies transient errors", () => {
|
|
const result = classifyMSTeamsSendError({ statusCode: 503 });
|
|
expect(result.kind).toBe("transient");
|
|
expect(result.statusCode).toBe(503);
|
|
});
|
|
|
|
it("classifies permanent 4xx errors", () => {
|
|
const result = classifyMSTeamsSendError({ statusCode: 400 });
|
|
expect(result.kind).toBe("permanent");
|
|
expect(result.statusCode).toBe(400);
|
|
});
|
|
|
|
it("provides actionable hints for common cases", () => {
|
|
expect(formatMSTeamsSendErrorHint({ kind: "auth" })).toContain("msteams");
|
|
expect(formatMSTeamsSendErrorHint({ kind: "throttled" })).toContain("throttled");
|
|
expect(
|
|
formatMSTeamsSendErrorHint({
|
|
kind: "permanent",
|
|
errorCode: "ContentStreamNotAllowed",
|
|
}),
|
|
).toContain("expired the content stream");
|
|
});
|
|
|
|
it("classifies transport-level network errors and provides smba egress hint (#77674)", () => {
|
|
const econnrefused = Object.assign(new Error("connect ECONNREFUSED"), { code: "ECONNREFUSED" });
|
|
const enotfound = Object.assign(new Error("getaddrinfo ENOTFOUND smba.trafficmanager.net"), {
|
|
code: "ENOTFOUND",
|
|
});
|
|
const etimedout = Object.assign(new Error("ETIMEDOUT"), { code: "ETIMEDOUT" });
|
|
|
|
const econnrefusedResult = classifyMSTeamsSendError(econnrefused);
|
|
expect(econnrefusedResult.kind).toBe("network");
|
|
expect(econnrefusedResult.errorCode).toBe("ECONNREFUSED");
|
|
const enotfoundResult = classifyMSTeamsSendError(enotfound);
|
|
expect(enotfoundResult.kind).toBe("network");
|
|
expect(enotfoundResult.errorCode).toBe("ENOTFOUND");
|
|
const etimedoutResult = classifyMSTeamsSendError(etimedout);
|
|
expect(etimedoutResult.kind).toBe("network");
|
|
expect(etimedoutResult.errorCode).toBe("ETIMEDOUT");
|
|
|
|
// Hints for network errors must mention smba (Connector endpoint) and egress
|
|
expect(formatMSTeamsSendErrorHint({ kind: "network" })).toContain("smba");
|
|
expect(formatMSTeamsSendErrorHint({ kind: "network" })).toContain("egress");
|
|
});
|
|
|
|
it("still classifies HTTP errors as unknown when no status code and no network code", () => {
|
|
expect(classifyMSTeamsSendError(new Error("unexpected error")).kind).toBe("unknown");
|
|
expect(classifyMSTeamsSendError(null).kind).toBe("unknown");
|
|
});
|
|
|
|
describe("isRevokedProxyError", () => {
|
|
it("returns true for revoked proxy TypeError", () => {
|
|
expect(
|
|
isRevokedProxyError(new TypeError("Cannot perform 'set' on a proxy that has been revoked")),
|
|
).toBe(true);
|
|
expect(
|
|
isRevokedProxyError(new TypeError("Cannot perform 'get' on a proxy that has been revoked")),
|
|
).toBe(true);
|
|
});
|
|
|
|
it("returns false for non-TypeError errors", () => {
|
|
expect(isRevokedProxyError(new Error("proxy that has been revoked"))).toBe(false);
|
|
});
|
|
|
|
it("returns false for unrelated TypeErrors", () => {
|
|
expect(isRevokedProxyError(new TypeError("undefined is not a function"))).toBe(false);
|
|
});
|
|
|
|
it("returns false for non-error values", () => {
|
|
expect(isRevokedProxyError(null)).toBe(false);
|
|
expect(isRevokedProxyError("proxy that has been revoked")).toBe(false);
|
|
});
|
|
});
|
|
|
|
describe("withRevokedProxyFallback", () => {
|
|
it("returns primary result when no error occurs", async () => {
|
|
await expect(
|
|
withRevokedProxyFallback({
|
|
run: async () => "ok",
|
|
onRevoked: async () => "fallback",
|
|
}),
|
|
).resolves.toBe("ok");
|
|
});
|
|
|
|
it("uses fallback when proxy-revoked TypeError is thrown", async () => {
|
|
const onRevokedLog = vi.fn();
|
|
await expect(
|
|
withRevokedProxyFallback({
|
|
run: async () => {
|
|
throw new TypeError("Cannot perform 'get' on a proxy that has been revoked");
|
|
},
|
|
onRevoked: async () => "fallback",
|
|
onRevokedLog,
|
|
}),
|
|
).resolves.toBe("fallback");
|
|
expect(onRevokedLog).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it("rethrows non-revoked errors", async () => {
|
|
const err = Object.assign(new Error("boom"), { statusCode: 500 });
|
|
await expect(
|
|
withRevokedProxyFallback({
|
|
run: async () => {
|
|
throw err;
|
|
},
|
|
onRevoked: async () => "fallback",
|
|
}),
|
|
).rejects.toBe(err);
|
|
});
|
|
});
|
|
});
|