Files
lemonspace_app/tests/convex/ai-errors.test.ts
Matthias Meister 463830f178 feat(canvas): implement local node size pinning and reconciliation logic
- Added functions to handle local node size pins, ensuring that node sizes are preserved during reconciliation.
- Updated `reconcileCanvasFlowNodes` to incorporate size pinning logic.
- Enhanced tests to verify the correct behavior of size pinning in various scenarios.
- Updated related components to support new size pinning functionality.
2026-04-10 08:48:34 +02:00

106 lines
3.1 KiB
TypeScript

import { describe, expect, it } from "vitest";
import { ConvexError } from "convex/values";
import { FreepikApiError } from "@/convex/freepik";
import {
categorizeError,
formatTerminalStatusMessage,
getVideoPollDelayMs,
isVideoPollTimedOut,
} from "@/convex/ai_errors";
describe("ai error helpers", () => {
it("marks provider 503 failures as retryable", () => {
const result = categorizeError(new Error("OpenRouter API error 503"));
expect(result).toEqual({ category: "provider", retryable: true });
});
it("maps Freepik timeout to timeout category", () => {
const error = new FreepikApiError({
code: "timeout",
message: "Task polling timeout",
retryable: true,
status: 504,
});
const result = categorizeError(error);
expect(result).toEqual({ category: "timeout", retryable: true });
});
it("formats terminal status with translated transient prefix", () => {
expect(formatTerminalStatusMessage(new Error("network disconnected"))).toBe(
"Netzwerk: network disconnected",
);
});
it("formats structured-output invalid json with human-readable provider message", () => {
expect(
formatTerminalStatusMessage(
new ConvexError({ code: "OPENROUTER_STRUCTURED_OUTPUT_INVALID_JSON" }),
),
).toBe("Provider: Strukturierte Antwort konnte nicht gelesen werden");
});
it("formats structured-output missing content with human-readable provider message", () => {
expect(
formatTerminalStatusMessage(
new ConvexError({ code: "OPENROUTER_STRUCTURED_OUTPUT_MISSING_CONTENT" }),
),
).toBe("Provider: Strukturierte Antwort fehlt");
});
it("formats structured-output http error with provider prefix and server message", () => {
expect(
formatTerminalStatusMessage(
new ConvexError({
code: "OPENROUTER_STRUCTURED_OUTPUT_HTTP_ERROR",
status: 503,
message: "OpenRouter API error 503: Upstream timeout",
}),
),
).toBe("Provider: OpenRouter API error 503: Upstream timeout");
});
it("formats structured-output http error without falling back to raw code", () => {
expect(
formatTerminalStatusMessage(
new ConvexError({ code: "OPENROUTER_STRUCTURED_OUTPUT_HTTP_ERROR" }),
),
).toBe("Provider: Anfrage fehlgeschlagen");
});
it("uses staged poll delays", () => {
expect(getVideoPollDelayMs(1)).toBe(5000);
expect(getVideoPollDelayMs(9)).toBe(10000);
expect(getVideoPollDelayMs(20)).toBe(20000);
});
it("detects poll timeout by attempts and elapsed time", () => {
expect(
isVideoPollTimedOut({
attempt: 31,
maxAttempts: 30,
elapsedMs: 1000,
maxTotalMs: 600000,
}),
).toBe(true);
expect(
isVideoPollTimedOut({
attempt: 10,
maxAttempts: 30,
elapsedMs: 700000,
maxTotalMs: 600000,
}),
).toBe(true);
expect(
isVideoPollTimedOut({
attempt: 10,
maxAttempts: 30,
elapsedMs: 200000,
maxTotalMs: 600000,
}),
).toBe(false);
});
});