mirror of
https://github.com/outline/outline.git
synced 2026-06-13 03:14:59 +03:00
091346dfe8
* wip * Remove obsolete snapshots * simplify * chore(test): Convert mocks to TypeScript and tighten fetch mock types Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> * Remove unneccessary patches * Migrate to msw instead of custom fetch mock * Address PR review comments - Split chained vi.useFakeTimers().setSystemTime() into separate calls. - Switch test setup to dynamic imports so EventEmitter.defaultMaxListeners assignment runs before module init (static imports were hoisted above it). - Drop redundant NODE_ENV guard in monkeyPatchSequelizeErrorsForJest; its sole caller already gates on env.isTest. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
126 lines
3.2 KiB
TypeScript
126 lines
3.2 KiB
TypeScript
import type Koa from "koa";
|
|
import type { Mock } from "vitest";
|
|
import { requestErrorHandler } from "@server/logging/sentry";
|
|
import { InternalError, ValidationError, NotFoundError } from "./errors";
|
|
import onerror from "./onerror";
|
|
|
|
// Mock the requestErrorHandler from Sentry
|
|
vi.mock("@server/logging/sentry", () => ({
|
|
requestErrorHandler: vi.fn(),
|
|
}));
|
|
|
|
type MockCtx = {
|
|
headers: Record<string, string>;
|
|
headerSent: boolean;
|
|
writable: boolean;
|
|
accepts: Mock;
|
|
set: Mock;
|
|
res: { end: Mock };
|
|
status: number | undefined;
|
|
type: string | undefined;
|
|
body: unknown;
|
|
};
|
|
|
|
type ReportableError = Error & {
|
|
status?: number;
|
|
isReportable?: boolean;
|
|
};
|
|
|
|
describe("onerror", () => {
|
|
let app: Koa;
|
|
let ctx: MockCtx;
|
|
|
|
beforeEach(() => {
|
|
// Create a mock Koa app
|
|
app = {
|
|
context: {},
|
|
} as unknown as Koa;
|
|
|
|
// Apply the onerror middleware
|
|
onerror(app);
|
|
|
|
// Create a mock context
|
|
ctx = {
|
|
headers: {},
|
|
headerSent: false,
|
|
writable: true,
|
|
accepts: vi.fn(() => "json"),
|
|
set: vi.fn(),
|
|
res: {
|
|
end: vi.fn(),
|
|
},
|
|
status: undefined,
|
|
type: undefined,
|
|
body: undefined,
|
|
};
|
|
|
|
// Clear mock calls
|
|
(requestErrorHandler as Mock).mockClear();
|
|
});
|
|
|
|
it("should report InternalError to Sentry", () => {
|
|
const error = InternalError("Test internal error");
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).toHaveBeenCalledWith(error, ctx);
|
|
expect(ctx.status).toBe(500);
|
|
});
|
|
|
|
it("should not report ValidationError to Sentry", () => {
|
|
const error = ValidationError("Test validation error");
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).not.toHaveBeenCalled();
|
|
expect(ctx.status).toBe(400);
|
|
});
|
|
|
|
it("should not report NotFoundError to Sentry", () => {
|
|
const error = NotFoundError("Test not found error");
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).not.toHaveBeenCalled();
|
|
expect(ctx.status).toBe(404);
|
|
});
|
|
|
|
it("should report unknown errors without isReportable property to Sentry", () => {
|
|
const error = new Error("Unknown error") as ReportableError;
|
|
error.status = 500;
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).toHaveBeenCalledWith(error, ctx);
|
|
});
|
|
|
|
it("should report errors with invalid status codes to Sentry", () => {
|
|
const error = new Error("Invalid status error") as ReportableError;
|
|
error.status = 999;
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).toHaveBeenCalledWith(error, ctx);
|
|
});
|
|
|
|
it("should not report errors explicitly marked with isReportable: false", () => {
|
|
const error = new Error("Custom error") as ReportableError;
|
|
error.status = 500;
|
|
error.isReportable = false;
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("should report errors explicitly marked with isReportable: true", () => {
|
|
const error = new Error("Custom error") as ReportableError;
|
|
error.status = 400;
|
|
error.isReportable = true;
|
|
|
|
app.context.onerror.call(ctx, error);
|
|
|
|
expect(requestErrorHandler).toHaveBeenCalledWith(error, ctx);
|
|
});
|
|
});
|