diff --git a/changelog/@unreleased/pr-2386.v2.yml b/changelog/@unreleased/pr-2386.v2.yml new file mode 100644 index 000000000..0361d6e35 --- /dev/null +++ b/changelog/@unreleased/pr-2386.v2.yml @@ -0,0 +1,5 @@ +type: improvement +improvement: + description: '[FR] Add Serialization Optimization for Primitive Collection Types' + links: + - https://github.com/palantir/conjure-java/pull/2386 diff --git a/conjure-java-core/src/integrationInput/java/com/palantir/product/PrimitiveExample.java b/conjure-java-core/src/integrationInput/java/com/palantir/product/PrimitiveExample.java new file mode 100644 index 000000000..8491c3beb --- /dev/null +++ b/conjure-java-core/src/integrationInput/java/com/palantir/product/PrimitiveExample.java @@ -0,0 +1,181 @@ +package com.palantir.product; + +import com.fasterxml.jackson.annotation.JsonIgnoreProperties; +import com.fasterxml.jackson.annotation.JsonInclude; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.annotation.JsonSetter; +import com.fasterxml.jackson.annotation.Nulls; +import com.fasterxml.jackson.databind.annotation.JsonDeserialize; +import com.google.errorprone.annotations.CheckReturnValue; +import com.palantir.conjure.java.lib.internal.ConjureCollections; +import com.palantir.logsafe.Preconditions; +import com.palantir.logsafe.SafeArg; +import com.palantir.logsafe.exceptions.SafeIllegalArgumentException; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Nonnull; +import javax.annotation.Nullable; +import javax.annotation.processing.Generated; + +@JsonDeserialize(builder = PrimitiveExample.Builder.class) +@Generated("com.palantir.conjure.java.types.BeanGenerator") +public final class PrimitiveExample { + private final List ints; + + private final List doubles; + + private int memoizedHashCode; + + private PrimitiveExample(List ints, List doubles) { + validateFields(ints, doubles); + this.ints = ConjureCollections.unmodifiableList(ints); + this.doubles = ConjureCollections.unmodifiableList(doubles); + } + + @JsonProperty("ints") + @JsonInclude(JsonInclude.Include.NON_EMPTY) + public List getInts() { + return this.ints; + } + + @JsonProperty("doubles") + @JsonInclude(JsonInclude.Include.NON_EMPTY) + public List getDoubles() { + return this.doubles; + } + + @Override + public boolean equals(@Nullable Object other) { + return this == other || (other instanceof PrimitiveExample && equalTo((PrimitiveExample) other)); + } + + private boolean equalTo(PrimitiveExample other) { + if (this.memoizedHashCode != 0 + && other.memoizedHashCode != 0 + && this.memoizedHashCode != other.memoizedHashCode) { + return false; + } + return this.ints.equals(other.ints) && this.doubles.equals(other.doubles); + } + + @Override + public int hashCode() { + int result = memoizedHashCode; + if (result == 0) { + int hash = 1; + hash = 31 * hash + this.ints.hashCode(); + hash = 31 * hash + this.doubles.hashCode(); + result = hash; + memoizedHashCode = result; + } + return result; + } + + @Override + public String toString() { + return "PrimitiveExample{ints: " + ints + ", doubles: " + doubles + '}'; + } + + public static PrimitiveExample of(List ints, List doubles) { + return builder().ints(ints).doubles(doubles).build(); + } + + private static void validateFields(List ints, List doubles) { + List missingFields = null; + missingFields = addFieldIfMissing(missingFields, ints, "ints"); + missingFields = addFieldIfMissing(missingFields, doubles, "doubles"); + if (missingFields != null) { + throw new SafeIllegalArgumentException( + "Some required fields have not been set", SafeArg.of("missingFields", missingFields)); + } + } + + private static List addFieldIfMissing(List prev, Object fieldValue, String fieldName) { + List missingFields = prev; + if (fieldValue == null) { + if (missingFields == null) { + missingFields = new ArrayList<>(2); + } + missingFields.add(fieldName); + } + return missingFields; + } + + public static Builder builder() { + return new Builder(); + } + + @Generated("com.palantir.conjure.java.types.BeanBuilderGenerator") + @JsonIgnoreProperties(ignoreUnknown = true) + public static final class Builder { + boolean _buildInvoked; + + private List ints = ConjureCollections.newNonNullIntegerList(); + + private List doubles = ConjureCollections.newNonNullDoubleList(); + + private Builder() {} + + public Builder from(PrimitiveExample other) { + checkNotBuilt(); + ints(other.getInts()); + doubles(other.getDoubles()); + return this; + } + + @JsonSetter(value = "ints", nulls = Nulls.SKIP, contentNulls = Nulls.FAIL) + public Builder ints(@Nonnull Iterable ints) { + checkNotBuilt(); + this.ints = + ConjureCollections.newNonNullIntegerList(Preconditions.checkNotNull(ints, "ints cannot be null")); + return this; + } + + public Builder addAllInts(@Nonnull Iterable ints) { + checkNotBuilt(); + ConjureCollections.addAllAndCheckNonNull( + this.ints, Preconditions.checkNotNull(ints, "ints cannot be null")); + return this; + } + + public Builder ints(int ints) { + checkNotBuilt(); + Preconditions.checkNotNull(ints, "ints cannot be null"); + this.ints.add(ints); + return this; + } + + @JsonSetter(value = "doubles", nulls = Nulls.SKIP, contentNulls = Nulls.FAIL) + public Builder doubles(@Nonnull Iterable doubles) { + checkNotBuilt(); + this.doubles = ConjureCollections.newNonNullDoubleList( + Preconditions.checkNotNull(doubles, "doubles cannot be null")); + return this; + } + + public Builder addAllDoubles(@Nonnull Iterable doubles) { + checkNotBuilt(); + ConjureCollections.addAllAndCheckNonNull( + this.doubles, Preconditions.checkNotNull(doubles, "doubles cannot be null")); + return this; + } + + public Builder doubles(double doubles) { + checkNotBuilt(); + Preconditions.checkNotNull(doubles, "doubles cannot be null"); + this.doubles.add(doubles); + return this; + } + + @CheckReturnValue + public PrimitiveExample build() { + checkNotBuilt(); + this._buildInvoked = true; + return new PrimitiveExample(ints, doubles); + } + + private void checkNotBuilt() { + Preconditions.checkState(!_buildInvoked, "Build has already been called"); + } + } +} diff --git a/conjure-java-core/src/test/java/com/palantir/conjure/java/NonNullCollectionsTest.java b/conjure-java-core/src/test/java/com/palantir/conjure/java/NonNullCollectionsTest.java index e6e8ed6b8..0c835857b 100644 --- a/conjure-java-core/src/test/java/com/palantir/conjure/java/NonNullCollectionsTest.java +++ b/conjure-java-core/src/test/java/com/palantir/conjure/java/NonNullCollectionsTest.java @@ -25,12 +25,15 @@ import com.palantir.conjure.java.serialization.ObjectMappers; import com.palantir.product.CovariantListExample; import com.palantir.product.ListExample; +import com.palantir.product.PrimitiveExample; import java.util.Collections; +import java.util.List; import java.util.Optional; import org.junit.jupiter.api.Test; public class NonNullCollectionsTest { - private static final ObjectMapper objectMapper = ObjectMappers.newClientObjectMapper(); + private static final ObjectMapper clientMapper = ObjectMappers.newClientObjectMapper(); + private static final ObjectMapper serverMapper = ObjectMappers.newServerJsonMapper(); @Test public void throwsNpe() { @@ -55,7 +58,7 @@ public void testOptionalSerialization() throws JsonProcessingException { .optionalItems(Collections.singleton(Optional.empty())) .build(); - assertThat(objectMapper.readValue(objectMapper.writeValueAsString(listExample), ListExample.class)) + assertThat(clientMapper.readValue(clientMapper.writeValueAsString(listExample), ListExample.class)) .isEqualTo(listExample); // non-null collections will add "contentNulls = Nulls.FAIL" to the JsonSetter annotation. This will cause deser @@ -64,12 +67,30 @@ public void testOptionalSerialization() throws JsonProcessingException { .addAllItems(Collections.singleton(Optional.empty())) .build(); assertThatExceptionOfType(InvalidNullException.class) - .isThrownBy(() -> objectMapper.readValue( - objectMapper.writeValueAsString(covariantListExample), CovariantListExample.class)); + .isThrownBy(() -> clientMapper.readValue( + clientMapper.writeValueAsString(covariantListExample), CovariantListExample.class)); // Similarly, setting a null in the builder also breaks assertThatExceptionOfType(NullPointerException.class).isThrownBy(() -> CovariantListExample.builder() .addAllItems(Collections.singleton(null)) .build()); } + + @Test + public void testSerDeOptimizationRespectsConjureEmptyCollections() throws JsonProcessingException { + PrimitiveExample expected = PrimitiveExample.builder().build(); + assertThat(clientMapper.writeValueAsString(expected)) + .describedAs("Does not serialize any empty collections, even when optimizing for primitives") + .isEqualTo("{}"); + } + + @Test + public void testSerializationRoundtrip() throws JsonProcessingException { + PrimitiveExample expected = PrimitiveExample.builder() + .ints(List.of(1, 2, 3)) + .doubles(List.of(1.1, 2.2, 3.3)) + .build(); + String serialized = serverMapper.writeValueAsString(expected); + assertThat(expected).isEqualTo(clientMapper.readValue(serialized, PrimitiveExample.class)); + } } diff --git a/conjure-java-core/src/test/java/com/palantir/conjure/java/types/ObjectGeneratorTests.java b/conjure-java-core/src/test/java/com/palantir/conjure/java/types/ObjectGeneratorTests.java index f7a67c38e..92d4b3a7e 100644 --- a/conjure-java-core/src/test/java/com/palantir/conjure/java/types/ObjectGeneratorTests.java +++ b/conjure-java-core/src/test/java/com/palantir/conjure/java/types/ObjectGeneratorTests.java @@ -168,6 +168,21 @@ public void testObjectGenerator_excludeEmptyCollections() throws IOException { assertThatFilesAreTheSame(files, REFERENCE_FILES_FOLDER); } + @Test + public void testObjectGenerator_primitiveCollections() throws IOException { + ConjureDefinition def = + Conjure.parse(ImmutableList.of(new File("src/test/resources/primitive-collections.yml"))); + List files = new GenerationCoordinator( + MoreExecutors.directExecutor(), + ImmutableSet.of(new ObjectGenerator(Options.builder() + .excludeEmptyCollections(true) + .nonNullCollections(true) + .build()))) + .emit(def, tempDir); + + assertThatFilesAreTheSame(files, REFERENCE_FILES_FOLDER); + } + @Test public void testConjureImports() throws IOException { ConjureDefinition conjure = Conjure.parse(ImmutableList.of( diff --git a/conjure-java-core/src/test/resources/primitive-collections.yml b/conjure-java-core/src/test/resources/primitive-collections.yml new file mode 100644 index 000000000..492bd0b90 --- /dev/null +++ b/conjure-java-core/src/test/resources/primitive-collections.yml @@ -0,0 +1,8 @@ +types: + definitions: + default-package: com.palantir.product + objects: + PrimitiveExample: + fields: + ints: list + doubles: list diff --git a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureCollections.java b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureCollections.java index c8a5a4df8..89a076992 100644 --- a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureCollections.java +++ b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureCollections.java @@ -38,8 +38,33 @@ private ConjureCollections() { // cannot instantiate } + /* + * This is bizarre. Allow me to explain... + * + * We do _not_ want to expose the Conjure*List types externally + * but we also want the optimizations they provide to make it thru + * to jackson for serialization. So the runtime type needs to be + * preserved while also not exposing the type :phew:. + * + * To achieve this we have to do some gymnastics surrounding the type + * system. We need this to return the type of the list given, but also + * return specific Conjure types when detected. This requires that we + * erase the type info, but we know this is safe because we are directly + * returning the same type which is by definition the identity function. + * Therefore the input List is the same types as the output List. + */ public static List unmodifiableList(List list) { - return Collections.unmodifiableList(list); + // Return the unmodifiable version of the Eclipse types + if (list instanceof ConjureIntegerList) { + return (List) ((ConjureIntegerList) list).asUnmodifiable(); + } else if (list instanceof ConjureDoubleList) { + return (List) ((ConjureDoubleList) list).asUnmodifiable(); + } else if (list instanceof ConjureSafeLongList) { + return (List) ((ConjureSafeLongList) list).asUnmodifiable(); + } else { + // Otherwise use the JDK types + return Collections.unmodifiableList(list); + } } @SuppressWarnings("unchecked") diff --git a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureDoubleList.java b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureDoubleList.java index 58d63bbaa..ebf359c19 100644 --- a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureDoubleList.java +++ b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureDoubleList.java @@ -16,10 +16,11 @@ package com.palantir.conjure.java.lib.internal; +import com.fasterxml.jackson.annotation.JsonValue; import java.util.AbstractList; import java.util.Collection; import java.util.RandomAccess; -import org.eclipse.collections.impl.list.mutable.primitive.DoubleArrayList; +import org.eclipse.collections.api.list.primitive.MutableDoubleList; import org.eclipse.collections.impl.utility.Iterate; /** @@ -27,9 +28,9 @@ * a BoxedMutableDoubleList will be released. Once available, ConjureDoubleList should be replaced with that. */ final class ConjureDoubleList extends AbstractList implements RandomAccess { - private final DoubleArrayList delegate; + private final MutableDoubleList delegate; - ConjureDoubleList(DoubleArrayList delegate) { + ConjureDoubleList(MutableDoubleList delegate) { this.delegate = delegate; } @@ -69,4 +70,15 @@ public void clear() { public Double set(int index, Double element) { return delegate.set(index, element); } + + ConjureDoubleList asUnmodifiable() { + return new ConjureDoubleList(delegate.asUnmodifiable()); + } + + // Cannot be named 'toArray' as that conflicts with the #toArray in AbstractList + // This is a serialization optimization that avoids boxing, but does copy + @JsonValue + double[] jacksonSerialize() { + return delegate.toArray(); + } } diff --git a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureIntegerList.java b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureIntegerList.java index 79ecd32b6..83aace125 100644 --- a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureIntegerList.java +++ b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureIntegerList.java @@ -16,10 +16,11 @@ package com.palantir.conjure.java.lib.internal; +import com.fasterxml.jackson.annotation.JsonValue; import java.util.AbstractList; import java.util.Collection; import java.util.RandomAccess; -import org.eclipse.collections.impl.list.mutable.primitive.IntArrayList; +import org.eclipse.collections.api.list.primitive.MutableIntList; import org.eclipse.collections.impl.utility.Iterate; /** @@ -27,9 +28,9 @@ * a BoxedMutableIntList will be released. Once available, ConjureIntegerList should be replaced with that. */ final class ConjureIntegerList extends AbstractList implements RandomAccess { - private final IntArrayList delegate; + private final MutableIntList delegate; - ConjureIntegerList(IntArrayList delegate) { + ConjureIntegerList(MutableIntList delegate) { this.delegate = delegate; } @@ -69,4 +70,15 @@ public void clear() { public Integer set(int index, Integer element) { return delegate.set(index, element); } + + ConjureIntegerList asUnmodifiable() { + return new ConjureIntegerList(delegate.asUnmodifiable()); + } + + // Cannot be named 'toArray' as that conflicts with the #toArray in AbstractList + // This is a serialization optimization that avoids boxing, but does copy + @JsonValue + int[] jacksonSerialize() { + return delegate.toArray(); + } } diff --git a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureSafeLongList.java b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureSafeLongList.java index 7995d8285..f8cb5cdfe 100644 --- a/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureSafeLongList.java +++ b/conjure-lib/src/main/java/com/palantir/conjure/java/lib/internal/ConjureSafeLongList.java @@ -16,11 +16,12 @@ package com.palantir.conjure.java.lib.internal; +import com.fasterxml.jackson.annotation.JsonValue; import com.palantir.conjure.java.lib.SafeLong; import java.util.AbstractList; import java.util.Collection; import java.util.RandomAccess; -import org.eclipse.collections.impl.list.mutable.primitive.LongArrayList; +import org.eclipse.collections.api.list.primitive.MutableLongList; import org.eclipse.collections.impl.utility.Iterate; /** @@ -28,9 +29,9 @@ * with SafeLongs. */ final class ConjureSafeLongList extends AbstractList implements RandomAccess { - private final LongArrayList delegate; + private final MutableLongList delegate; - ConjureSafeLongList(LongArrayList delegate) { + ConjureSafeLongList(MutableLongList delegate) { this.delegate = delegate; } @@ -70,4 +71,15 @@ public void clear() { public SafeLong set(int index, SafeLong element) { return SafeLong.of(delegate.set(index, element.longValue())); } + + ConjureSafeLongList asUnmodifiable() { + return new ConjureSafeLongList(delegate.asUnmodifiable()); + } + + // Cannot be named 'toArray' as that conflicts with the #toArray in AbstractList + // This is a serialization optimization that avoids boxing, but does copy + @JsonValue + long[] jacksonSerialize() { + return delegate.toArray(); + } }