From 70275bd823f35d3e3bc7f6aa64fcf7df6aa7307b Mon Sep 17 00:00:00 2001 From: George Fu Date: Mon, 9 Dec 2024 16:35:55 -0500 Subject: [PATCH] remove String extension in LazyJsonString (#1468) * remove String extension in LazyJsonString * uniform usage of LazyJsonString * preserve LazyJsonString interface --- .changeset/warm-dragons-wash.md | 5 + packages/smithy-client/src/lazy-json.spec.ts | 14 ++- packages/smithy-client/src/lazy-json.ts | 104 ++++++++++-------- .../typescript/codegen/SymbolVisitor.java | 8 +- .../HttpProtocolGeneratorUtils.java | 4 +- .../codegen/SymbolProviderTest.java | 2 +- .../DocumentMemberDeserVisitorTest.java | 2 +- .../DocumentMemberSerVisitorTest.java | 2 +- 8 files changed, 86 insertions(+), 55 deletions(-) create mode 100644 .changeset/warm-dragons-wash.md diff --git a/.changeset/warm-dragons-wash.md b/.changeset/warm-dragons-wash.md new file mode 100644 index 00000000000..2060a2de94f --- /dev/null +++ b/.changeset/warm-dragons-wash.md @@ -0,0 +1,5 @@ +--- +"@smithy/smithy-client": minor +--- + +remove String extension in LazyJsonString diff --git a/packages/smithy-client/src/lazy-json.spec.ts b/packages/smithy-client/src/lazy-json.spec.ts index 835fae1bcb0..1a7b747250c 100644 --- a/packages/smithy-client/src/lazy-json.spec.ts +++ b/packages/smithy-client/src/lazy-json.spec.ts @@ -1,8 +1,9 @@ import { describe, expect, test as it } from "vitest"; import { LazyJsonString } from "./lazy-json"; + describe("LazyJsonString", () => { - it("should has string methods", () => { + it("should have string methods", () => { const jsonValue = new LazyJsonString('"foo"'); expect(jsonValue.length).toBe(5); expect(jsonValue.toString()).toBe('"foo"'); @@ -22,17 +23,22 @@ describe("LazyJsonString", () => { it("can instantiate from LazyJsonString class", () => { const original = new LazyJsonString('"foo"'); - const newOne = LazyJsonString.fromObject(original); + const newOne = LazyJsonString.from(original); expect(newOne.toString()).toBe('"foo"'); }); it("can instantiate from String class", () => { - const jsonValue = LazyJsonString.fromObject(new String('"foo"')); + const jsonValue = LazyJsonString.from(new String('"foo"')); expect(jsonValue.toString()).toBe('"foo"'); }); it("can instantiate from object", () => { - const jsonValue = LazyJsonString.fromObject({ foo: "bar" }); + const jsonValue = LazyJsonString.from({ foo: "bar" }); expect(jsonValue.toString()).toBe('{"foo":"bar"}'); }); + + it("passes instanceof String check", () => { + const jsonValue = LazyJsonString.from({ foo: "bar" }); + expect(jsonValue).toBeInstanceOf(String); + }); }); diff --git a/packages/smithy-client/src/lazy-json.ts b/packages/smithy-client/src/lazy-json.ts index 6a9d4b416a9..02aaf8f23bd 100644 --- a/packages/smithy-client/src/lazy-json.ts +++ b/packages/smithy-client/src/lazy-json.ts @@ -1,57 +1,73 @@ /** - * Lazy String holder for JSON typed contents. + * @public + * + * A model field with this type means that you may provide a JavaScript + * object in lieu of a JSON string, and it will be serialized to JSON + * automatically before being sent in a request. + * + * For responses, you will receive a "LazyJsonString", which is a boxed String object + * with additional mixin methods. + * To get the string value, call `.toString()`, or to get the JSON object value, + * call `.deserializeJSON()` or parse it yourself. */ - -interface StringWrapper { - new (arg: any): String; -} +export type AutomaticJsonStringConversion = Parameters[0] | LazyJsonString; /** - * Because of https://github.com/microsoft/tslib/issues/95, - * TS 'extends' shim doesn't support extending native types like String. - * So here we create StringWrapper that duplicate everything from String - * class including its prototype chain. So we can extend from here. - * * @internal + * */ -// @ts-ignore StringWrapper implementation is not a simple constructor -export const StringWrapper: StringWrapper = function () { - //@ts-ignore 'this' cannot be assigned to any, but Object.getPrototypeOf accepts any - const Class = Object.getPrototypeOf(this).constructor; - const Constructor = Function.bind.apply(String, [null as any, ...arguments]); - //@ts-ignore Call wrapped String constructor directly, don't bother typing it. - const instance = new Constructor(); - Object.setPrototypeOf(instance, Class.prototype); - return instance as String; -}; -StringWrapper.prototype = Object.create(String.prototype, { - constructor: { - value: StringWrapper, - enumerable: false, - writable: true, - configurable: true, - }, -}); -Object.setPrototypeOf(StringWrapper, String); +export interface LazyJsonString extends String { + new (s: string): typeof LazyJsonString; + + /** + * @returns the JSON parsing of the string value. + */ + deserializeJSON(): any; + + /** + * @returns the original string value rather than a JSON.stringified value. + */ + toJSON(): string; +} /** * @internal + * + * Extension of the native String class in the previous implementation + * has negative global performance impact on method dispatch for strings, + * and is generally discouraged. + * + * This current implementation may look strange, but is necessary to preserve the interface and + * behavior of extending the String class. */ -export class LazyJsonString extends StringWrapper { - deserializeJSON(): any { - return JSON.parse(super.toString()); - } +export function LazyJsonString(val: string): void { + const str = Object.assign(new String(val), { + deserializeJSON() { + return JSON.parse(String(val)); + }, - toJSON(): string { - return super.toString(); - } + toString() { + return String(val); + }, - static fromObject(object: any): LazyJsonString { - if (object instanceof LazyJsonString) { - return object; - } else if (object instanceof String || typeof object === "string") { - return new LazyJsonString(object); - } - return new LazyJsonString(JSON.stringify(object)); - } + toJSON() { + return String(val); + }, + }); + + return str as never; } + +LazyJsonString.from = (object: any): LazyJsonString => { + if (object && typeof object === "object" && (object instanceof LazyJsonString || "deserializeJSON" in object)) { + return object as any; + } else if (typeof object === "string" || Object.getPrototypeOf(object) === String.prototype) { + return LazyJsonString(String(object) as string) as any; + } + return LazyJsonString(JSON.stringify(object)) as any; +}; + +/** + * @deprecated use from. + */ +LazyJsonString.fromObject = LazyJsonString.from; diff --git a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/SymbolVisitor.java b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/SymbolVisitor.java index 4e0aef6c0b2..b8ff3d8c09b 100644 --- a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/SymbolVisitor.java +++ b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/SymbolVisitor.java @@ -300,8 +300,12 @@ public Symbol stringShape(StringShape shape) { if (mediaTypeTrait.isPresent()) { String mediaType = mediaTypeTrait.get().getValue(); if (CodegenUtils.isJsonMediaType(mediaType)) { - Symbol.Builder builder = createSymbolBuilder(shape, "__LazyJsonString | string"); - return addSmithyUseImport(builder, "LazyJsonString", "__LazyJsonString").build(); + Symbol.Builder builder = createSymbolBuilder(shape, "__AutomaticJsonStringConversion | string"); + return addSmithyUseImport( + builder, + "AutomaticJsonStringConversion", + "__AutomaticJsonStringConversion" + ).build(); } else { LOGGER.warning(() -> "Found unsupported mediatype " + mediaType + " on String shape: " + shape); } diff --git a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpProtocolGeneratorUtils.java b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpProtocolGeneratorUtils.java index 849cc03d80f..fa74c2ece7f 100644 --- a/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpProtocolGeneratorUtils.java +++ b/smithy-typescript-codegen/src/main/java/software/amazon/smithy/typescript/codegen/integration/HttpProtocolGeneratorUtils.java @@ -176,7 +176,7 @@ public static String getStringInputParam(GenerationContext context, Shape shape, if (CodegenUtils.isJsonMediaType(mediaType)) { TypeScriptWriter writer = context.getWriter(); writer.addImport("LazyJsonString", "__LazyJsonString", TypeScriptDependency.AWS_SMITHY_CLIENT); - return "__LazyJsonString.fromObject(" + dataSource + ")"; + return "__LazyJsonString.from(" + dataSource + ")"; } else { LOGGER.warning(() -> "Found unsupported mediatype " + mediaType + " on String shape: " + shape); } @@ -210,7 +210,7 @@ public static String getStringOutputParam(GenerationContext context, if (CodegenUtils.isJsonMediaType(mediaType)) { TypeScriptWriter writer = context.getWriter(); writer.addImport("LazyJsonString", "__LazyJsonString", TypeScriptDependency.AWS_SMITHY_CLIENT); - return "new __LazyJsonString(" + dataSource + ")"; + return "__LazyJsonString.from(" + dataSource + ")"; } else { LOGGER.warning(() -> "Found unsupported mediatype " + mediaType + " on String shape: " + shape); } diff --git a/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/SymbolProviderTest.java b/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/SymbolProviderTest.java index 65b659c047e..4d2cc5f5395 100644 --- a/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/SymbolProviderTest.java +++ b/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/SymbolProviderTest.java @@ -191,7 +191,7 @@ public void usesLazyJsonStringForJsonMediaType() { SymbolProvider provider = new SymbolVisitor(model, settings); Symbol memberSymbol = provider.toSymbol(member); - assertThat(memberSymbol.getName(), equalTo("__LazyJsonString | string")); + assertThat(memberSymbol.getName(), equalTo("__AutomaticJsonStringConversion | string")); } @Test diff --git a/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberDeserVisitorTest.java b/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberDeserVisitorTest.java index 9d306538565..80b762db823 100644 --- a/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberDeserVisitorTest.java +++ b/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberDeserVisitorTest.java @@ -98,7 +98,7 @@ public static Collection validMemberTargetTypes() { {StringShape.builder().id(id).build(), "__expectString(" + DATA_SOURCE + ")", source}, { StringShape.builder().id(id).addTrait(new MediaTypeTrait("foo+json")).build(), - "new __LazyJsonString(" + DATA_SOURCE + ")", + "__LazyJsonString.from(" + DATA_SOURCE + ")", source }, {BlobShape.builder().id(id).build(), "context.base64Decoder(" + DATA_SOURCE + ")", source}, diff --git a/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberSerVisitorTest.java b/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberSerVisitorTest.java index fcee0c535f6..f82420f3f95 100644 --- a/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberSerVisitorTest.java +++ b/smithy-typescript-codegen/src/test/java/software/amazon/smithy/typescript/codegen/integration/DocumentMemberSerVisitorTest.java @@ -83,7 +83,7 @@ public static Collection validMemberTargetTypes() { {StringShape.builder().id(id).build(), DATA_SOURCE}, { StringShape.builder().id(id).addTrait(new MediaTypeTrait("foo+json")).build(), - "__LazyJsonString.fromObject(" + DATA_SOURCE + ")" + "__LazyJsonString.from(" + DATA_SOURCE + ")" }, {BlobShape.builder().id(id).build(), "context.base64Encoder(" + DATA_SOURCE + ")"}, {DocumentShape.builder().id(id).build(), delegate},