From 1927db57e80731b41f4773a8fe450700d172007f Mon Sep 17 00:00:00 2001 From: "marian.jureczko" Date: Mon, 14 Oct 2024 08:56:30 +0200 Subject: [PATCH 1/3] bumping easy-random --- pom.xml | 4 +- .../test/ArrangerSealedInterfacesTest.java | 166 ++++++++++++++++++ 2 files changed, 168 insertions(+), 2 deletions(-) create mode 100644 src/test/java/com/ocadotechnology/gembus/test/ArrangerSealedInterfacesTest.java diff --git a/pom.xml b/pom.xml index d56f2e2..970f532 100644 --- a/pom.xml +++ b/pom.xml @@ -20,7 +20,7 @@ 4.0.0 com.ocadotechnology.gembus test-arranger - 1.6.2 + 1.6.3-SNAPSHOT jar test-arranger A tool for arranging test data with pseudo-random values. @@ -60,7 +60,7 @@ UTF-8 17 1.9.22 - 7.0.0 + 7.0.1-SNAPSHOT diff --git a/src/test/java/com/ocadotechnology/gembus/test/ArrangerSealedInterfacesTest.java b/src/test/java/com/ocadotechnology/gembus/test/ArrangerSealedInterfacesTest.java new file mode 100644 index 0000000..a8ade11 --- /dev/null +++ b/src/test/java/com/ocadotechnology/gembus/test/ArrangerSealedInterfacesTest.java @@ -0,0 +1,166 @@ +/* + * Copyright © 2020 Ocado (marian.jureczko@ocado.com) + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package com.ocadotechnology.gembus.test; + +import org.jeasy.random.ObjectCreationException; +import org.junit.jupiter.api.Test; + +import java.util.List; +import java.util.Map; + +import static com.ocadotechnology.gembus.test.Arranger.some; +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatThrownBy; + +public class ArrangerSealedInterfacesTest { + + @Test + void should_arrangeSealedInterface() { + //when + SealedInterfaceData actual = some(SealedInterfaceData.class); + + //then + assertThat(actual).isInstanceOfAny(ConcreteData1.class, ConcreteData2.class); + } + + @Test + void should_arrangeSealedInterfaceWithCustomArranger() { + //when + SealedInterfaceWithCustomArranger actual = some(SealedInterfaceWithCustomArranger.class); + + //then + assertThat(actual).isEqualTo(new ConcreteDataWithCustomArranger("expected-test-string")); + } + + @Test + void should_arrangeDataWithSealedInterfaceField() { + //when + DataWithAbstract actual = some(DataWithAbstract.class, "excludedSealedInterface", "nonSealedInterface"); + + //then + assertThat(actual.sealedInterfaceData()).isInstanceOfAny(ConcreteData1.class, ConcreteData2.class); + assertThat(actual.excludedSealedInterface()).isNull(); + } + + @Test + void should_arrangeDataWithSealedInterfaceFieldOverride() { + //when + ConcreteData1 expected = new ConcreteData1("expected-data"); + DataWithAbstract actual = some(DataWithAbstract.class, Map.of( + "sealedInterfaceData", () -> expected, + "excludedSealedInterface", () -> null, + "nonSealedInterface", () -> null)); + + //then + assertThat(actual.sealedInterfaceData()).isEqualTo(expected); + assertThat(actual.excludedSealedInterface()).isNull(); + } + + @Test + void should_arrangeDataWithNestedSealedInterfaceField() { + //when + NestedSealedInterfaceData actual = some(NestedSealedInterfaceData.class); + + //then + assertThat(actual).isInstanceOf(ConcreteNested1.class); + assertThat((ConcreteNested1) actual) + .extracting(ConcreteNested1::sealedInterfaceData) + .isInstanceOfAny(ConcreteData1.class, ConcreteData2.class); + + } + + @Test + void should_arrangeRecordWithListWithSealedInterfaceField() { + //when + RootRecordWithNestedList actual = some(RootRecordWithNestedList.class); + + //then + assertThat(actual.nestedRecordWithSealedInterfaces().get(0)) + .extracting(NestedRecordWithSealedInterface::sealedInterfaceData) + .isInstanceOfAny(ConcreteData1.class, ConcreteData2.class); + } + + @Test + void should_failForNonSealedInterface() { + //when + assertThatThrownBy(() -> some(DataWithAbstract.class, "excludedSealedInterface")) + + //then + .isInstanceOf(ObjectCreationException.class); + + } + +} + +record DataWithAbstract(Integer value, + SealedInterfaceData sealedInterfaceData, + ExcludedSealedInterface excludedSealedInterface, + NonSealedInterface nonSealedInterface) { + DataWithAbstract { + } + + DataWithAbstract(String name) { + this(0, new ConcreteData1(""), new ExcludedData1(), new NonSealedInterface() { + }); + } +} + +sealed interface SealedInterfaceData permits ConcreteData1, ConcreteData2 { +} + +record ConcreteData1(String value) implements SealedInterfaceData { +} + +final class ConcreteData2 implements SealedInterfaceData { + + Integer value; + +} + +sealed interface ExcludedSealedInterface permits ExcludedData1 { +} + +record ExcludedData1() implements ExcludedSealedInterface { +} + +interface NonSealedInterface { +} + +sealed interface NestedSealedInterfaceData permits ConcreteNested1 { +} + +record ConcreteNested1(SealedInterfaceData sealedInterfaceData) implements NestedSealedInterfaceData { +} + +sealed interface SealedInterfaceWithCustomArranger permits ConcreteDataWithCustomArranger { +} + +record ConcreteDataWithCustomArranger(String test) implements SealedInterfaceWithCustomArranger { +} + +class CustomSealedInterfaceArranger extends CustomArranger { + + @Override + protected SealedInterfaceWithCustomArranger instance() { + return new ConcreteDataWithCustomArranger("expected-test-string"); + } +} + +record RootRecordWithNestedList(List nestedRecordWithSealedInterfaces) { +} + +record NestedRecordWithSealedInterface(SealedInterfaceData sealedInterfaceData) { +} From 2a0e18fe5e4f11ea34b1e97814ba418ab5ddd34c Mon Sep 17 00:00:00 2001 From: "marian.jureczko" Date: Mon, 28 Oct 2024 08:38:42 +0100 Subject: [PATCH 2/3] release 1.6.3 --- README.md | 4 ++-- pom.xml | 28 ++++++++++++++-------------- 2 files changed, 16 insertions(+), 16 deletions(-) diff --git a/README.md b/README.md index a982868..91a51d7 100644 --- a/README.md +++ b/README.md @@ -35,14 +35,14 @@ product.setBrand("Ocado"); com.ocadotechnology.gembus test-arranger - 1.6.2 + 1.6.3 ``` ### Gradle ```groovy -testImplementation 'com.ocadotechnology.gembus:test-arranger:1.6.2' +testImplementation 'com.ocadotechnology.gembus:test-arranger:1.6.3' ``` ## Features diff --git a/pom.xml b/pom.xml index 970f532..15614f4 100644 --- a/pom.xml +++ b/pom.xml @@ -20,7 +20,7 @@ 4.0.0 com.ocadotechnology.gembus test-arranger - 1.6.3-SNAPSHOT + 1.6.3 jar test-arranger A tool for arranging test data with pseudo-random values. @@ -59,8 +59,8 @@ UTF-8 UTF-8 17 - 1.9.22 - 7.0.1-SNAPSHOT + 1.9.25 + 7.1.0 @@ -95,17 +95,17 @@ org.apache.commons commons-lang3 - 3.12.0 + 3.17.0 org.yaml snakeyaml - 2.2 + 2.3 io.github.classgraph classgraph - 4.8.165 + 4.8.177 org.jetbrains.kotlin @@ -130,7 +130,7 @@ org.assertj assertj-core - 3.25.1 + 3.26.3 test @@ -140,7 +140,7 @@ org.junit junit-bom - 5.10.1 + 5.11.3 pom import @@ -195,7 +195,7 @@ org.apache.maven.plugins maven-compiler-plugin - 3.12.1 + 3.13.0 @@ -229,13 +229,13 @@ maven-surefire-plugin - 3.2.5 + 3.5.1 com.mycila license-maven-plugin - 4.3 + 4.6 @@ -284,7 +284,7 @@ org.sonatype.plugins nexus-staging-maven-plugin - 1.6.13 + 1.7.0 true ossrh @@ -296,7 +296,7 @@ org.apache.maven.plugins maven-source-plugin - 3.3.0 + 3.3.1 attach-sources @@ -310,7 +310,7 @@ org.apache.maven.plugins maven-javadoc-plugin - 3.6.3 + 3.10.1 attach-javadocs From e00f72d9f80ab5348e480b08ff71920f00a4dd78 Mon Sep 17 00:00:00 2001 From: "marian.jureczko" Date: Mon, 28 Oct 2024 08:49:31 +0100 Subject: [PATCH 3/3] resolving merge conflicts --- README.md | 10 +- .../gembus/test/ArrangersConfigurer.java | 5 +- .../gembus/test/CustomArranger.java | 12 +-- .../gembus/test/EnhancedRandom.java | 94 ++----------------- .../gembus/test/ReflectionHelper.java | 2 - .../experimental/SealedInterfaceArranger.java | 34 ------- 6 files changed, 17 insertions(+), 140 deletions(-) delete mode 100644 src/main/java/com/ocadotechnology/gembus/test/experimental/SealedInterfaceArranger.java diff --git a/README.md b/README.md index 91a51d7..21fbbf8 100644 --- a/README.md +++ b/README.md @@ -318,13 +318,11 @@ Eventually, we may end up with something like this: class ShopFixture { Repository repo; public void shopWithNineProductsAndFourCustomers() { - Stream.generate(() -> Arranger.some(Product.class)) - .limit(9) - .forEach(p -> repo.save(p)); + Arranger.someObjects(Product.class, 9) + .forEach(p -> repo.save(p)); - Stream.generate(() -> Arranger.some(Customer.class)) - .limit(4) - .forEach(p -> repo.save(p)); + Arranger.someObjects(Customer.class, 4) + .forEach(p -> repo.save(p)); } } ``` diff --git a/src/main/java/com/ocadotechnology/gembus/test/ArrangersConfigurer.java b/src/main/java/com/ocadotechnology/gembus/test/ArrangersConfigurer.java index ba10274..c8f19e6 100644 --- a/src/main/java/com/ocadotechnology/gembus/test/ArrangersConfigurer.java +++ b/src/main/java/com/ocadotechnology/gembus/test/ArrangersConfigurer.java @@ -74,10 +74,11 @@ EnhancedRandom simplifiedRandom() { return randomWithArrangers(simplifiedArrangers, new EnhancedRandom.Builder(ArrangersConfigurer::getEasyRandomSimplifiedParameters)); } - EnhancedRandom randomForGivenConfiguration(Class type, boolean withoutGivenType, Map, CustomArranger> arrangers, Supplier parametersSupplier) { + EnhancedRandom randomForGivenConfiguration(Class type, Map, CustomArranger> arrangers, Supplier parametersSupplier) { EnhancedRandom.Builder randomBuilder = new EnhancedRandom.Builder(parametersSupplier); long seed = SeedHelper.calculateSeed(); - if (withoutGivenType && arrangers.get(type) != null) { + CustomArranger arrangerToUpdate = arrangers.get(type); + if (arrangerToUpdate != null) { seed = SeedHelper.customArrangerTypeSpecificSeedRespectingRandomSeedSetting(type); arrangers = withoutGivenType(arrangers, type); } diff --git a/src/main/java/com/ocadotechnology/gembus/test/CustomArranger.java b/src/main/java/com/ocadotechnology/gembus/test/CustomArranger.java index 32d1f86..4b90e3b 100644 --- a/src/main/java/com/ocadotechnology/gembus/test/CustomArranger.java +++ b/src/main/java/com/ocadotechnology/gembus/test/CustomArranger.java @@ -33,19 +33,9 @@ public abstract class CustomArranger { protected EnhancedRandom enhancedRandom = null; - protected final Class type; + protected final Class type = (Class) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0]; protected CustomArranger() { - this.type = (Class) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0]; - initEnhancedRandom(); - } - - protected CustomArranger(Class type) { - this.type = type; - initEnhancedRandom(); - } - - private void initEnhancedRandom() { if (ArrangersConfigurer.defaultInitialized.get()) { enhancedRandom = ArrangersConfigurer.instance().defaultRandom(); } else { diff --git a/src/main/java/com/ocadotechnology/gembus/test/EnhancedRandom.java b/src/main/java/com/ocadotechnology/gembus/test/EnhancedRandom.java index fc05eed..bca376b 100644 --- a/src/main/java/com/ocadotechnology/gembus/test/EnhancedRandom.java +++ b/src/main/java/com/ocadotechnology/gembus/test/EnhancedRandom.java @@ -15,27 +15,15 @@ */ package com.ocadotechnology.gembus.test; -import com.ocadotechnology.gembus.test.experimental.SealedInterfaceArranger; import org.jeasy.random.EasyRandom; import org.jeasy.random.EasyRandomParameters; import org.jeasy.random.api.Randomizer; -import java.lang.reflect.Field; -import java.lang.reflect.ParameterizedType; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Optional; -import java.util.Random; -import java.util.Set; +import java.util.*; import java.util.concurrent.ConcurrentHashMap; import java.util.function.Supplier; -import java.util.stream.Collectors; import java.util.stream.Stream; -import static java.util.function.Function.identity; - /** * Class for random object generator. */ @@ -96,90 +84,26 @@ public Stream objects(final Class type, final int amount, final String } private EasyRandom selectEasyRandomWithRespectToExclusion(Class type, String[] excludedFields) { - if (newEasyRandomWithCustomRandomizersOrFieldExclusionConfigIsRequired(type, excludedFields)) { - return createEasyRandomWithCustomRandomizersAndExclusions(type, excludedFields); + if (newEasyRandomWithFieldExclusionConfigIsRequired(type, excludedFields)) { + return createEasyRandomWithExclusions(type, excludedFields); } return easyRandom; } - private boolean newEasyRandomWithCustomRandomizersOrFieldExclusionConfigIsRequired(Class type, String[] excludedFields) { + private boolean newEasyRandomWithFieldExclusionConfigIsRequired(Class type, String[] excludedFields) { /* There is a logical inconsistency in using a custom arranger and field exclusion for the same type - the * exclusion can be configured in the custom arranger. Technically, creating an arranger with exclusion disables * the custom arranger for the type that is being instantiated. */ - return !arrangers.containsKey(type) && (excludedFields.length != 0 || isSealedInterface(type) || !nestedSealedInterfaceFields(type).isEmpty()); + return !arrangers.containsKey(type) && excludedFields.length != 0; } - private EasyRandom createEasyRandomWithCustomRandomizersAndExclusions(Class type, String[] excludedFields) { + private EasyRandom createEasyRandomWithExclusions(Class type, String[] excludedFields) { Set fields = new HashSet<>(Arrays.asList(excludedFields)); - var forSealedInterfaces = createCustomArrangersForSealedInterfaces(type, fields); - Set cacheKey = getCacheKey(fields, forSealedInterfaces.keySet()); - cache.computeIfAbsent(cacheKey, key -> { - HashMap, CustomArranger> enhancedArrangers = new HashMap<>(arrangers); - enhancedArrangers.putAll(forSealedInterfaces); - EnhancedRandom er = ArrangersConfigurer.instance() - .randomForGivenConfiguration(type, !forSealedInterfaces.containsKey(type), enhancedArrangers, () -> addExclusionToParameters(fields)); + cache.computeIfAbsent(fields, key -> { + EnhancedRandom er = ArrangersConfigurer.instance().randomForGivenConfiguration(type, arrangers, () -> addExclusionToParameters(fields)); return er.easyRandom; }); - return cache.get(cacheKey); - } - - private Set getCacheKey(Set fields, Set> sealedInterfaces) { - Set cacheKey = new HashSet<>(fields); - cacheKey.addAll(sealedInterfaces.stream().map(Class::getName).toList()); - return cacheKey; - } - - private Map, CustomArranger> createCustomArrangersForSealedInterfaces(Class type, Set excludedFields) { - Map, CustomArranger> sealedInterfaceArrangers = new HashMap<>(); - if (isSealedInterface(type)) { - sealedInterfaceArrangers.put(type, new SealedInterfaceArranger(type)); - } - sealedInterfaceArrangers.putAll(nestedSealedInterfaceFields(type) - .entrySet() - .stream() - .filter(entry -> !excludedFields.contains(entry.getKey())) - .map(Map.Entry::getValue) - .collect(Collectors.toMap(identity(), SealedInterfaceArranger::new))); - return sealedInterfaceArrangers; - } - - private Map> nestedSealedInterfaceFields(Class type) { - return allNestedFields(new HashMap<>(), type) - .entrySet() - .stream() - .filter(entry -> isSealedInterface(entry.getValue())) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - } - - private Map> allNestedFields(Map> acc, Class clazz) { - if (clazz == null || clazz.isPrimitive()) { - return Map.of(); - } - if (isSealedInterface(clazz)) { - for (Class permittedSubclasses : clazz.getPermittedSubclasses()) { - acc.putAll(allNestedFields(acc, permittedSubclasses)); - } - } - for (Field field : clazz.getDeclaredFields()) { - if (!acc.containsKey(field.getName())) { - if (field.getGenericType() instanceof ParameterizedType parameterizedType) { - for (var genericType : parameterizedType.getActualTypeArguments()) { - if(genericType instanceof Class genericTypeClass) { - acc.put(field.getName(), genericTypeClass); - acc.putAll(allNestedFields(acc, genericTypeClass)); - } - } - } else { - acc.put(field.getName(), field.getType()); - acc.putAll(allNestedFields(acc, field.getType())); - } - } - } - return acc; - } - - private static boolean isSealedInterface(Class type) { - return type.isSealed() && type.isInterface(); + return cache.get(fields); } private EasyRandomParameters addExclusionToParameters(Set fields) { diff --git a/src/main/java/com/ocadotechnology/gembus/test/ReflectionHelper.java b/src/main/java/com/ocadotechnology/gembus/test/ReflectionHelper.java index 4b182f6..efea474 100644 --- a/src/main/java/com/ocadotechnology/gembus/test/ReflectionHelper.java +++ b/src/main/java/com/ocadotechnology/gembus/test/ReflectionHelper.java @@ -15,7 +15,6 @@ */ package com.ocadotechnology.gembus.test; -import com.ocadotechnology.gembus.test.experimental.SealedInterfaceArranger; import io.github.classgraph.ClassGraph; import java.lang.reflect.Constructor; @@ -40,7 +39,6 @@ class ReflectionHelper { .loadClasses(CustomArranger.class, true) .stream() .filter(clazz -> isNotAbstract(clazz)) - .filter(clazz -> !SealedInterfaceArranger.class.equals(clazz)) .map(clazz -> extractConstructor(clazz)) .filter(constructor -> constructor.isPresent()) .map(constructor -> constructor.get()) diff --git a/src/main/java/com/ocadotechnology/gembus/test/experimental/SealedInterfaceArranger.java b/src/main/java/com/ocadotechnology/gembus/test/experimental/SealedInterfaceArranger.java deleted file mode 100644 index 90bdf8c..0000000 --- a/src/main/java/com/ocadotechnology/gembus/test/experimental/SealedInterfaceArranger.java +++ /dev/null @@ -1,34 +0,0 @@ -/* - * Copyright © 2020 Ocado (marian.jureczko@ocado.com) - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package com.ocadotechnology.gembus.test.experimental; - -import com.ocadotechnology.gembus.test.CustomArranger; - -public class SealedInterfaceArranger extends CustomArranger { - - public SealedInterfaceArranger() { - } - - public SealedInterfaceArranger(Class clazz) { - super(clazz); - } - - @Override - protected T instance() { - Class[] subclasses = type.getPermittedSubclasses(); - return (T) enhancedRandom.nextObject(subclasses[enhancedRandom.nextInt(subclasses.length)]); - } -}