diff --git a/packages/shared.net/src/client/createOpenApiRequest.test.ts b/packages/shared.net/src/client/createOpenApiRequest.test.ts index 58735b554..ba0593bd0 100644 --- a/packages/shared.net/src/client/createOpenApiRequest.test.ts +++ b/packages/shared.net/src/client/createOpenApiRequest.test.ts @@ -16,6 +16,7 @@ import type { OpenApiRequest } from "@osdk/gateway/types"; import { describe, expect, expectTypeOf, it, vi } from "vitest"; +import { PalantirApiError } from "../PalantirApiError.js"; import { createOpenApiRequest } from "./createOpenApiRequest.js"; describe("createOpenApiRequest", () => { @@ -23,6 +24,7 @@ describe("createOpenApiRequest", () => { const mockFetch = vi.fn(); mockFetch.mockResolvedValue({ + ok: true, json: () => Promise.resolve({ test: 1 }), }); @@ -66,6 +68,7 @@ describe("createOpenApiRequest", () => { const mockFetch = vi.fn(); mockFetch.mockResolvedValue({ + ok: true, json: () => Promise.resolve({ test: 1 }), }); @@ -97,6 +100,7 @@ describe("createOpenApiRequest", () => { const mockFetch = vi.fn(); mockFetch.mockResolvedValue({ + ok: true, json: () => Promise.resolve({ test: 1 }), }); @@ -123,6 +127,7 @@ describe("createOpenApiRequest", () => { const mockFetch = vi.fn(); mockFetch.mockResolvedValue({ + ok: true, json: () => Promise.resolve({ test: 1 }), }); @@ -150,6 +155,7 @@ describe("createOpenApiRequest", () => { const stream = new ReadableStream(); mockFetch.mockResolvedValue({ + ok: true, body: stream, }); @@ -194,6 +200,7 @@ describe("createOpenApiRequest", () => { const blob = new Blob(); mockFetch.mockResolvedValue({ + ok: true, blob: () => Promise.resolve(blob), }); @@ -234,4 +241,36 @@ describe("createOpenApiRequest", () => { }, ); }); + + it("handles error status codes", async () => { + const mockFetch = vi.fn(); + + const mockResponse: ReturnType = Promise.resolve({ + ok: false, + status: 500, + json: () => + Promise.resolve({ + errorCode: "INTERNAL", + errorName: "Default:Internal", + errorInstanceId: "00000000-0000-0000-0000-000000000000", + parameters: {}, + }), + } as Response); + + mockFetch.mockImplementationOnce(() => mockResponse); + + const request = createOpenApiRequest<{}>( + "http://example.com", + mockFetch, + undefined, + false, + ); + + try { + await request("POST", "/"); + expect.fail(); + } catch (error) { + expect(error).toBeInstanceOf(PalantirApiError); + } + }); }); diff --git a/packages/shared.net/src/client/createOpenApiRequest.ts b/packages/shared.net/src/client/createOpenApiRequest.ts index 676b2e127..55bcc6229 100644 --- a/packages/shared.net/src/client/createOpenApiRequest.ts +++ b/packages/shared.net/src/client/createOpenApiRequest.ts @@ -15,6 +15,8 @@ */ import type { OpenApiRequest } from "@osdk/gateway/types"; +import { PalantirApiError } from "../PalantirApiError.js"; +import { UnknownError } from "../UnknownError.js"; import { replaceHttpIfNotLocalhost } from "../util/index.js"; export function createOpenApiRequest< @@ -73,6 +75,12 @@ export function createOpenApiRequest< headers: headersInit, }); + // error status codes are not thrown by fetch automatically, + // we have to look at the ok property and behave accordingly + if (!response.ok) { + throw await convertError(response); + } + if (responseMediaType && responseMediaType === "*/*") { if (asReadableStream) { return response.body; @@ -104,3 +112,25 @@ function withHttps(url: string): string { ? replaceHttpIfNotLocalhost(url) : `${httpsProtocol}${url}`; } + +async function convertError(response: Response) { + try { + const convertedError = await response.json(); + return new PalantirApiError( + convertedError.message, + convertedError.errorName, + convertedError.errorCode, + response.status, + convertedError.errorInstanceId, + convertedError.parameters, + ); + } catch (e) { + if (e instanceof Error) { + return new UnknownError(e.message, "UNKNOWN"); + } + return new UnknownError( + "Unable to parse error response", + "UNKNOWN", + ); + } +}