From 3ff86af581794c2f18e9e3b8eecc4140c1b49be3 Mon Sep 17 00:00:00 2001 From: Andrey G Date: Wed, 29 May 2024 15:59:09 +0300 Subject: [PATCH] FMWK-451 Find by "not equals" doesn't return entities with nonexistent path (#749) --- .../AbstractAerospikeDataConfiguration.java | 5 +++++ .../data/aerospike/query/FilterOperation.java | 22 +++++++------------ .../noindex/findBy/NotEqualTests.java | 11 +++------- .../blocking/noindex/findBy/NotInTests.java | 3 ++- 4 files changed, 18 insertions(+), 23 deletions(-) diff --git a/src/main/java/org/springframework/data/aerospike/config/AbstractAerospikeDataConfiguration.java b/src/main/java/org/springframework/data/aerospike/config/AbstractAerospikeDataConfiguration.java index f6e4828ff..921700037 100644 --- a/src/main/java/org/springframework/data/aerospike/config/AbstractAerospikeDataConfiguration.java +++ b/src/main/java/org/springframework/data/aerospike/config/AbstractAerospikeDataConfiguration.java @@ -64,6 +64,11 @@ public QueryEngine queryEngine(IAerospikeClient aerospikeClient, long queryMaxRecords = settings.getDataSettings().getQueryMaxRecords(); log.debug("AerospikeDataSettings.queryMaxRecords: {}", queryMaxRecords); queryEngine.setQueryMaxRecords(queryMaxRecords); + if (!settings.getDataSettings().isWriteSortedMaps()) { + log.info("AerospikeDataSettings.writeSortedMaps is set to false, " + + "Maps and POJOs will be written as unsorted Maps (degrades performance of Map-related operations ," + + " does not allow comparing Maps)"); + } return queryEngine; } diff --git a/src/main/java/org/springframework/data/aerospike/query/FilterOperation.java b/src/main/java/org/springframework/data/aerospike/query/FilterOperation.java index 12ac9a620..486603145 100644 --- a/src/main/java/org/springframework/data/aerospike/query/FilterOperation.java +++ b/src/main/java/org/springframework/data/aerospike/query/FilterOperation.java @@ -223,27 +223,19 @@ public Filter sIndexFilter(Map qualifierMap) { public Exp filterExp(Map qualifierMap) { return getMetadataExp(qualifierMap).orElseGet(() -> { Value value = getValue(qualifierMap); - return switch (value.getType()) { + Exp ne = switch (value.getType()) { // FMWK-175: Exp.ne() does not return null bins, so Exp.not(Exp.binExists()) is added - case INTEGER -> { - Exp ne = Exp.ne(Exp.intBin(getBinName(qualifierMap)), Exp.val(value.toLong())); - yield Exp.or(Exp.not(Exp.binExists(getBinName(qualifierMap))), ne); - } + case INTEGER -> Exp.ne(Exp.intBin(getBinName(qualifierMap)), Exp.val(value.toLong())); case STRING -> { if (ignoreCase(qualifierMap)) { String equalsRegexp = getStringEquals(value.toString()); - Exp regexCompare = Exp.not(Exp.regexCompare(equalsRegexp, RegexFlag.ICASE, + yield Exp.not(Exp.regexCompare(equalsRegexp, RegexFlag.ICASE, Exp.stringBin(getBinName(qualifierMap)))); - yield Exp.or(Exp.not(Exp.binExists(getBinName(qualifierMap))), regexCompare); } else { - Exp ne = Exp.ne(Exp.stringBin(getBinName(qualifierMap)), Exp.val(value.toString())); - yield Exp.or(Exp.not(Exp.binExists(getBinName(qualifierMap))), ne); + yield Exp.ne(Exp.stringBin(getBinName(qualifierMap)), Exp.val(value.toString())); } } - case BOOL -> { - Exp ne = Exp.ne(Exp.boolBin(getBinName(qualifierMap)), Exp.val((Boolean) value.getObject())); - yield Exp.or(Exp.not(Exp.binExists(getBinName(qualifierMap))), ne); - } + case BOOL -> Exp.ne(Exp.boolBin(getBinName(qualifierMap)), Exp.val((Boolean) value.getObject())); case MAP -> getFilterExp(Exp.val((Map) value.getObject()), getBinName(qualifierMap), Exp::ne, Exp::mapBin); case LIST -> getFilterExp(Exp.val((List) value.getObject()), getBinName(qualifierMap), Exp::ne, @@ -251,6 +243,7 @@ public Exp filterExp(Map qualifierMap) { default -> throw new IllegalArgumentException("NOTEQ FilterExpression unsupported particle type: " + value.getClass().getSimpleName()); }; + return Exp.or(Exp.not(Exp.binExists(getBinName(qualifierMap))), ne); }); } @@ -550,7 +543,8 @@ public Filter sIndexFilter(Map qualifierMap) { MAP_VAL_NOTEQ_BY_KEY { @Override public Exp filterExp(Map qualifierMap) { - return getFilterExpMapValNotEqOrFail(qualifierMap, Exp::ne); + Exp binIsNull = Exp.not(Exp.binExists(getBinName(qualifierMap))); + return Exp.or(binIsNull, getFilterExpMapValNotEqOrFail(qualifierMap, Exp::ne)); } @Override diff --git a/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotEqualTests.java b/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotEqualTests.java index 80d3f2fcd..4e88bc188 100644 --- a/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotEqualTests.java +++ b/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotEqualTests.java @@ -37,22 +37,17 @@ void findBySimplePropertyNotEqual_String() { @Test void findByNestedSimplePropertyNotEqual() { - String zipCode = "C0123456789"; - assertThat(carter.getAddress().getZipCode()).isNotEqualTo(zipCode); - assertThat(repository.findByAddressZipCodeIsNot(zipCode)).contains(carter); - oliver.setFriend(alicia); repository.save(oliver); dave.setFriend(oliver); repository.save(dave); carter.setFriend(dave); repository.save(carter); + assertThat(carter.getFriend().getAge()).isEqualTo(42); + // find all records where friend's age is not 42 and all without friend.age List result = repository.findByFriendAgeIsNot(42); - - assertThat(result) - .hasSize(2) - .containsExactlyInAnyOrder(dave, oliver); + assertThat(result).doesNotContain(carter); TestUtils.setFriendsToNull(repository, oliver, dave, carter); } diff --git a/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotInTests.java b/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotInTests.java index 8ec86780e..62a0d3d5a 100644 --- a/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotInTests.java +++ b/src/test/java/org/springframework/data/aerospike/repository/query/blocking/noindex/findBy/NotInTests.java @@ -32,7 +32,8 @@ void findBySimplePropertyNotIn_String() { void findByNestedSimplePropertyNotIn_String() { assertThat(carter.getAddress().getZipCode()).isEqualTo("C0124"); assertThat(dave.getAddress().getZipCode()).isEqualTo("C0123"); - assertThat(repository.findByAddressZipCodeNotIn(List.of("C0123", "C0125"))).containsOnly(carter); + // find all records where address' zipCode is not C0123 or C0125, and all without address.zipCode + assertThat(repository.findByAddressZipCodeNotIn(List.of("C0123", "C0125"))).doesNotContain(dave); } @Test