From dd78399fc14c36b53cbeb811a6cbca5d0c6aab9c Mon Sep 17 00:00:00 2001 From: mike12345567 Date: Thu, 23 May 2024 17:56:48 +0100 Subject: [PATCH 1/3] Changing how content processing works for responses from REST API - try and reduce the chance of an error in the case of malformed data being returned. --- packages/server/src/integrations/rest.ts | 34 +++++++++++++------ .../server/src/integrations/utils/utils.ts | 5 ++- 2 files changed, 25 insertions(+), 14 deletions(-) diff --git a/packages/server/src/integrations/rest.ts b/packages/server/src/integrations/rest.ts index 7104c9daca..987f58423a 100644 --- a/packages/server/src/integrations/rest.ts +++ b/packages/server/src/integrations/rest.ts @@ -148,6 +148,10 @@ class RestIntegration implements IntegrationBase { response.headers, { downloadImages: this.config.downloadImages } ) + let contentLength = response.headers.get("content-length") + if (!contentLength && raw) { + contentLength = Buffer.byteLength(raw, "utf8").toString() + } if ( contentDisposition.includes("filename") || contentDisposition.includes("attachment") || @@ -156,36 +160,44 @@ class RestIntegration implements IntegrationBase { filename = path.basename(parse(contentDisposition).parameters?.filename) || "" } + + let triedParsing: boolean = false, + responseTxt: string | undefined + const hasContent = contentLength && parseInt(contentLength) > 0 try { if (filename) { return handleFileResponse(response, filename, this.startTimeMs) } else { + responseTxt = hasContent ? await response.text() : "" if (response.status === 204) { data = [] raw = "" - } else if (contentType.includes("application/json")) { - data = await response.json() - raw = JSON.stringify(data) + } else if (hasContent && contentType.includes("application/json")) { + triedParsing = true + data = JSON.parse(responseTxt) + raw = responseTxt } else if ( - contentType.includes("text/xml") || + (hasContent && contentType.includes("text/xml")) || contentType.includes("application/xml") ) { - let xmlResponse = await handleXml(response) + triedParsing = true + let xmlResponse = await handleXml(responseTxt) data = xmlResponse.data raw = xmlResponse.rawXml } else { - data = await response.text() + data = responseTxt raw = data as string } } } catch (err) { - throw `Failed to parse response body: ${err}` + if (triedParsing) { + data = responseTxt + raw = data as string + } else { + throw new Error(`Failed to parse response body: ${err}`) + } } - let contentLength = response.headers.get("content-length") - if (!contentLength && raw) { - contentLength = Buffer.byteLength(raw, "utf8").toString() - } const size = formatBytes(contentLength || "0") const time = `${Math.round(performance.now() - this.startTimeMs)}ms` headers = response.headers.raw() diff --git a/packages/server/src/integrations/utils/utils.ts b/packages/server/src/integrations/utils/utils.ts index 5bc90bc295..069bd750ae 100644 --- a/packages/server/src/integrations/utils/utils.ts +++ b/packages/server/src/integrations/utils/utils.ts @@ -485,9 +485,8 @@ export function isValidFilter(value: any) { return value != null && value !== "" } -export async function handleXml(response: any) { - let data, - rawXml = await response.text() +export async function handleXml(rawXml: string) { + let data data = (await xmlParser(rawXml, { explicitArray: false, From 62c407d846c5b0073b955143bc60c884a5032722 Mon Sep 17 00:00:00 2001 From: mike12345567 Date: Fri, 24 May 2024 12:15:28 +0100 Subject: [PATCH 2/3] Updating test cases. --- packages/server/src/integrations/rest.ts | 2 +- .../src/integrations/tests/rest.spec.ts | 25 +++++++++++++------ 2 files changed, 18 insertions(+), 9 deletions(-) diff --git a/packages/server/src/integrations/rest.ts b/packages/server/src/integrations/rest.ts index 987f58423a..57e4217722 100644 --- a/packages/server/src/integrations/rest.ts +++ b/packages/server/src/integrations/rest.ts @@ -168,7 +168,7 @@ class RestIntegration implements IntegrationBase { if (filename) { return handleFileResponse(response, filename, this.startTimeMs) } else { - responseTxt = hasContent ? await response.text() : "" + responseTxt = hasContent && response.text ? await response.text() : "" if (response.status === 204) { data = [] raw = "" diff --git a/packages/server/src/integrations/tests/rest.spec.ts b/packages/server/src/integrations/tests/rest.spec.ts index 144aefa576..be392f0a6f 100644 --- a/packages/server/src/integrations/tests/rest.spec.ts +++ b/packages/server/src/integrations/tests/rest.spec.ts @@ -1,19 +1,27 @@ jest.mock("node-fetch", () => { + const obj = { + my_next_cursor: 123, + } + const str = JSON.stringify(obj) return jest.fn(() => ({ headers: { raw: () => { - return { "content-type": ["application/json"] } + return { + "content-type": ["application/json"], + "content-length": str.length, + } }, get: (name: string) => { - if (name.toLowerCase() === "content-type") { + const lcName = name.toLowerCase() + if (lcName === "content-type") { return ["application/json"] + } else if (lcName === "content-length") { + return str.length } }, }, - json: jest.fn(() => ({ - my_next_cursor: 123, - })), - text: jest.fn(), + json: jest.fn(() => obj), + text: jest.fn(() => str), })) }) @@ -231,7 +239,8 @@ describe("REST Integration", () => { } it("should be able to parse JSON response", async () => { - const input = buildInput({ a: 1 }, null, "application/json") + const obj = { a: 1 } + const input = buildInput(obj, JSON.stringify(obj), "application/json") const output = await config.integration.parseResponse(input) expect(output.data).toEqual({ a: 1 }) expect(output.info.code).toEqual(200) @@ -261,7 +270,7 @@ describe("REST Integration", () => { test.each([...contentTypes, undefined])( "should not throw an error on 204 no content", async contentType => { - const input = buildInput(undefined, null, contentType, 204) + const input = buildInput(undefined, "", contentType, 204) const output = await config.integration.parseResponse(input) expect(output.data).toEqual([]) expect(output.extra.raw).toEqual("") From ae26f66cf91e8579b27a3aa3e5c354f19ac7356f Mon Sep 17 00:00:00 2001 From: mike12345567 Date: Fri, 24 May 2024 12:40:58 +0100 Subject: [PATCH 3/3] Fixing a test case. --- .../server/src/api/routes/tests/queries/rest.spec.ts | 10 ++++------ packages/server/src/integrations/rest.ts | 6 ++++-- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/packages/server/src/api/routes/tests/queries/rest.spec.ts b/packages/server/src/api/routes/tests/queries/rest.spec.ts index 5c41583244..1d5483017b 100644 --- a/packages/server/src/api/routes/tests/queries/rest.spec.ts +++ b/packages/server/src/api/routes/tests/queries/rest.spec.ts @@ -64,13 +64,11 @@ describe("rest", () => { cached = await getCachedVariable(basedOnQuery._id!, "foo") expect(cached).toBeNull() - nock("http://one.example.com") - .get("/") - .reply(200, [{ name: "one" }]) + const body1 = [{ name: "one" }] + const body2 = [{ name: "two" }] + nock("http://one.example.com").get("/").reply(200, body1) nock("http://two.example.com").get("/?test=one").reply(500) - nock("http://two.example.com") - .get("/?test=one") - .reply(200, [{ name: "two" }]) + nock("http://two.example.com").get("/?test=one").reply(200, body2) const res = await config.api.query.preview({ datasourceId: datasource._id!, diff --git a/packages/server/src/integrations/rest.ts b/packages/server/src/integrations/rest.ts index 57e4217722..451c319aa9 100644 --- a/packages/server/src/integrations/rest.ts +++ b/packages/server/src/integrations/rest.ts @@ -163,12 +163,14 @@ class RestIntegration implements IntegrationBase { let triedParsing: boolean = false, responseTxt: string | undefined - const hasContent = contentLength && parseInt(contentLength) > 0 try { if (filename) { return handleFileResponse(response, filename, this.startTimeMs) } else { - responseTxt = hasContent && response.text ? await response.text() : "" + responseTxt = response.text ? await response.text() : "" + const hasContent = + (contentLength && parseInt(contentLength) > 0) || + responseTxt.length > 0 if (response.status === 204) { data = [] raw = ""