From 75e0125c8d25c4b1002f39d9d0fac7792acc3d43 Mon Sep 17 00:00:00 2001 From: George Fu Date: Thu, 3 Oct 2024 16:11:14 -0400 Subject: [PATCH] fix(protocols): fix header serde, handle unset union payloads (#1417) --- .changeset/proud-bugs-deny.md | 5 +++ .changeset/witty-rings-do.md | 5 +++ .../src/compressionMiddleware.spec.ts | 2 +- .../src/compressionMiddleware.ts | 2 +- packages/smithy-client/src/index.ts | 2 + .../smithy-client/src/quote-header.spec.ts | 19 ++++++++ packages/smithy-client/src/quote-header.ts | 11 +++++ .../smithy-client/src/split-header.spec.ts | 22 +++++++++ packages/smithy-client/src/split-header.ts | 45 +++++++++++++++++++ .../test/functional/rpcv2cbor.spec.ts | 2 +- .../codegen/HttpProtocolTestGenerator.java | 6 ++- .../HttpBindingProtocolGenerator.java | 27 +++++++++-- 12 files changed, 139 insertions(+), 9 deletions(-) create mode 100644 .changeset/proud-bugs-deny.md create mode 100644 .changeset/witty-rings-do.md create mode 100644 packages/smithy-client/src/quote-header.spec.ts create mode 100644 packages/smithy-client/src/quote-header.ts create mode 100644 packages/smithy-client/src/split-header.spec.ts create mode 100644 packages/smithy-client/src/split-header.ts diff --git a/.changeset/proud-bugs-deny.md b/.changeset/proud-bugs-deny.md new file mode 100644 index 00000000000..5fd3207ef96 --- /dev/null +++ b/.changeset/proud-bugs-deny.md @@ -0,0 +1,5 @@ +--- +"@smithy/middleware-compression": patch +--- + +comma spacing diff --git a/.changeset/witty-rings-do.md b/.changeset/witty-rings-do.md new file mode 100644 index 00000000000..e56f9652bce --- /dev/null +++ b/.changeset/witty-rings-do.md @@ -0,0 +1,5 @@ +--- +"@smithy/smithy-client": minor +--- + +add quoteHeader function diff --git a/packages/middleware-compression/src/compressionMiddleware.spec.ts b/packages/middleware-compression/src/compressionMiddleware.spec.ts index a79ada81d48..95d9099748e 100644 --- a/packages/middleware-compression/src/compressionMiddleware.spec.ts +++ b/packages/middleware-compression/src/compressionMiddleware.spec.ts @@ -160,7 +160,7 @@ describe(compressionMiddleware.name, () => { body: mockCompressedBody, headers: { ...mockArgs.request.headers, - "content-encoding": [mockExistingContentEncoding, "gzip"].join(","), + "content-encoding": [mockExistingContentEncoding, "gzip"].join(", "), }, }, }); diff --git a/packages/middleware-compression/src/compressionMiddleware.ts b/packages/middleware-compression/src/compressionMiddleware.ts index df0026770f5..13f810537c7 100644 --- a/packages/middleware-compression/src/compressionMiddleware.ts +++ b/packages/middleware-compression/src/compressionMiddleware.ts @@ -82,7 +82,7 @@ export const compressionMiddleware = if (headers["content-encoding"]) { updatedHeaders = { ...headers, - "content-encoding": `${headers["content-encoding"]},${algorithm}`, + "content-encoding": `${headers["content-encoding"]}, ${algorithm}`, }; } else { updatedHeaders = { ...headers, "content-encoding": algorithm }; diff --git a/packages/smithy-client/src/index.ts b/packages/smithy-client/src/index.ts index ee08d54c3c5..4a4ac197236 100644 --- a/packages/smithy-client/src/index.ts +++ b/packages/smithy-client/src/index.ts @@ -18,7 +18,9 @@ export * from "./lazy-json"; export * from "./NoOpLogger"; export * from "./object-mapping"; export * from "./parse-utils"; +export * from "./quote-header"; export * from "./resolve-path"; export * from "./ser-utils"; export * from "./serde-json"; export * from "./split-every"; +export * from "./split-header"; diff --git a/packages/smithy-client/src/quote-header.spec.ts b/packages/smithy-client/src/quote-header.spec.ts new file mode 100644 index 00000000000..3609f96c22f --- /dev/null +++ b/packages/smithy-client/src/quote-header.spec.ts @@ -0,0 +1,19 @@ +import { quoteHeader } from "./quote-header"; + +describe(quoteHeader.name, () => { + it("should not wrap header elements that don't include the delimiter or double quotes", () => { + expect(quoteHeader("bc")).toBe("bc"); + }); + + it("should wrap header elements that include the delimiter", () => { + expect(quoteHeader("b,c")).toBe('"b,c"'); + }); + + it("should wrap header elements that include double quotes", () => { + expect(quoteHeader(`"bc"`)).toBe('"\\"bc\\""'); + }); + + it("should wrap header elements that include the delimiter and double quotes", () => { + expect(quoteHeader(`"b,c"`)).toBe('"\\"b,c\\""'); + }); +}); diff --git a/packages/smithy-client/src/quote-header.ts b/packages/smithy-client/src/quote-header.ts new file mode 100644 index 00000000000..d7eb7d60349 --- /dev/null +++ b/packages/smithy-client/src/quote-header.ts @@ -0,0 +1,11 @@ +/** + * @public + * @param part - header list element + * @returns quoted string if part contains delimiter. + */ +export function quoteHeader(part: string) { + if (part.includes(",") || part.includes('"')) { + part = `"${part.replace(/"/g, '\\"')}"`; + } + return part; +} diff --git a/packages/smithy-client/src/split-header.spec.ts b/packages/smithy-client/src/split-header.spec.ts new file mode 100644 index 00000000000..608abf76a06 --- /dev/null +++ b/packages/smithy-client/src/split-header.spec.ts @@ -0,0 +1,22 @@ +import { splitHeader } from "./split-header"; + +describe(splitHeader.name, () => { + it("should split a string by commas and trim only the comma delimited outer values", () => { + expect(splitHeader("abc")).toEqual(["abc"]); + expect(splitHeader("a,b,c")).toEqual(["a", "b", "c"]); + expect(splitHeader("a, b, c")).toEqual(["a", "b", "c"]); + expect(splitHeader("a , b , c")).toEqual(["a", "b", "c"]); + expect(splitHeader(`a , b , " c "`)).toEqual(["a", "b", " c "]); + expect(splitHeader(` a , , b`)).toEqual(["a", "", "b"]); + expect(splitHeader(`,,`)).toEqual(["", "", ""]); + expect(splitHeader(` , , `)).toEqual(["", "", ""]); + }); + it("should split a string by commas that are not in quotes, and remove outer quotes", () => { + expect(splitHeader('"b,c", "\\"def\\"", a')).toEqual(["b,c", '"def"', "a"]); + expect(splitHeader('"a,b,c", ""def"", "a,b ,c"')).toEqual(["a,b,c", '"def"', "a,b ,c"]); + expect(splitHeader(`""`)).toEqual([``]); + expect(splitHeader(``)).toEqual([``]); + expect(splitHeader(`\\"`)).toEqual([`"`]); + expect(splitHeader(`"`)).toEqual([`"`]); + }); +}); diff --git a/packages/smithy-client/src/split-header.ts b/packages/smithy-client/src/split-header.ts new file mode 100644 index 00000000000..e76aa577de6 --- /dev/null +++ b/packages/smithy-client/src/split-header.ts @@ -0,0 +1,45 @@ +/** + * @param value - header string value. + * @returns value split by commas that aren't in quotes. + */ +export const splitHeader = (value: string): string[] => { + const z = value.length; + const values = []; + + let withinQuotes = false; + let prevChar = undefined; + let anchor = 0; + + for (let i = 0; i < z; ++i) { + const char = value[i]; + switch (char) { + case `"`: + if (prevChar !== "\\") { + withinQuotes = !withinQuotes; + } + break; + case ",": + if (!withinQuotes) { + values.push(value.slice(anchor, i)); + anchor = i + 1; + } + break; + default: + } + prevChar = char; + } + + values.push(value.slice(anchor)); + + return values.map((v) => { + v = v.trim(); + const z = v.length; + if (z < 2) { + return v; + } + if (v[0] === `"` && v[z - 1] === `"`) { + v = v.slice(1, z - 1); + } + return v.replace(/\\"/g, '"'); + }); +}; diff --git a/private/smithy-rpcv2-cbor/test/functional/rpcv2cbor.spec.ts b/private/smithy-rpcv2-cbor/test/functional/rpcv2cbor.spec.ts index 940d62070fc..bf9b1a52b17 100644 --- a/private/smithy-rpcv2-cbor/test/functional/rpcv2cbor.spec.ts +++ b/private/smithy-rpcv2-cbor/test/functional/rpcv2cbor.spec.ts @@ -697,7 +697,7 @@ it("no_input:Request", async () => { expect(r.headers["smithy-protocol"]).toBeDefined(); expect(r.headers["smithy-protocol"]).toBe("rpc-v2-cbor"); - expect(r.body).toBeFalsy(); + expect(!r.body || r.body === `{}`).toBeTruthy(); } }); diff --git a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/HttpProtocolTestGenerator.java b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/HttpProtocolTestGenerator.java index 084e9a697d1..68743aae596 100644 --- a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/HttpProtocolTestGenerator.java +++ b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/HttpProtocolTestGenerator.java @@ -531,9 +531,11 @@ private void writeHttpHostAssertion(HttpRequestTestCase testCase) { } private void writeHttpBodyAssertions(String body, String mediaType, boolean isClientTest) { - // If we expect an empty body, expect it to be falsy. if (body.isEmpty()) { - writer.write("expect(r.body).toBeFalsy();"); + // If we expect an empty body, expect it to be falsy. + // Or, for JSON an empty object represents an empty body. + // mediaType is often UNKNOWN here. + writer.write("expect(!r.body || r.body === `{}`).toBeTruthy();"); return; } diff --git a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpBindingProtocolGenerator.java b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpBindingProtocolGenerator.java index 2df46ef0f47..a1a2e9e7cb1 100644 --- a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpBindingProtocolGenerator.java +++ b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpBindingProtocolGenerator.java @@ -1385,6 +1385,11 @@ private String getCollectionInputParam( switch (bindingType) { case HEADER: + if (collectionTarget.isStringShape()) { + context.getWriter().addImport( + "quoteHeader", "__quoteHeader", TypeScriptDependency.AWS_SMITHY_CLIENT); + return iteratedParam + ".map(__quoteHeader).join(', ')"; + } return iteratedParam + ".join(', ')"; case QUERY: case QUERY_PARAMS: @@ -2464,9 +2469,8 @@ private HttpBinding readPayload( + "= __expectObject(await parseBody(output.body, context));"); } else if (target instanceof UnionShape) { // If payload is a Union, then we need to parse the string into JavaScript object. - importUnionDeserializer(writer); writer.write("const data: Record | undefined " - + "= __expectUnion(await parseBody(output.body, context));"); + + "= await parseBody(output.body, context);"); } else if (target instanceof StringShape || target instanceof DocumentShape) { // If payload is String or Document, we need to collect body and convert binary to string. writer.write("const data: any = await collectBodyString(output.body, context);"); @@ -2474,8 +2478,22 @@ private HttpBinding readPayload( throw new CodegenException(String.format("Unexpected shape type bound to payload: `%s`", target.getType())); } - writer.write("contents.$L = $L;", binding.getMemberName(), getOutputValue(context, + + if (target instanceof UnionShape) { + writer.openBlock( + "if (Object.keys(data ?? {}).length) {", + "}", + () -> { + importUnionDeserializer(writer); + writer.write("contents.$L = __expectUnion($L);", binding.getMemberName(), getOutputValue(context, + Location.PAYLOAD, "data", binding.getMember(), target)); + } + ); + } else { + writer.write("contents.$L = $L;", binding.getMemberName(), getOutputValue(context, Location.PAYLOAD, "data", binding.getMember(), target)); + } + return binding; } @@ -2716,7 +2734,8 @@ private String getCollectionOutputParam( case HEADER: dataSource = "(" + dataSource + " || \"\")"; // Split these values on commas. - outputParam = dataSource + ".split(',')"; + context.getWriter().addImport("splitHeader", "__splitHeader", TypeScriptDependency.AWS_SMITHY_CLIENT); + outputParam = "__splitHeader(" + dataSource + ")"; // Headers that have HTTP_DATE formatted timestamps already contain a "," // in their formatted entry, so split on every other "," instead.