Skip to content

Commit e9ebcc2

Browse files
davidbodenJoel Costigliola
authored and
Joel Costigliola
committed
DualValue hash code is not computed properly because it calls actual.hashCode() and expected.hashCode() when equals compares actual and expected references.
Fix #3340
1 parent d51cfd9 commit e9ebcc2

File tree

2 files changed

+83
-2
lines changed

2 files changed

+83
-2
lines changed

assertj-core/src/main/java/org/assertj/core/api/recursive/comparison/DualValue.java

+6-2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
package org.assertj.core.api.recursive.comparison;
1414

1515
import static java.lang.String.format;
16+
import static java.lang.System.identityHashCode;
1617
import static java.util.Collections.unmodifiableList;
1718
import static java.util.Objects.requireNonNull;
1819
import static org.assertj.core.api.recursive.comparison.FieldLocation.rootFieldLocation;
@@ -23,7 +24,6 @@
2324
import java.util.LinkedHashSet;
2425
import java.util.List;
2526
import java.util.Map;
26-
import java.util.Objects;
2727
import java.util.Optional;
2828
import java.util.OptionalDouble;
2929
import java.util.OptionalInt;
@@ -61,7 +61,11 @@ public DualValue(FieldLocation fieldLocation, Object actualFieldValue, Object ex
6161
this.fieldLocation = requireNonNull(fieldLocation, "fieldLocation must not be null");
6262
actual = actualFieldValue;
6363
expected = expectedFieldValue;
64-
hashCode = Objects.hash(actual, expected);
64+
hashCode = computeHashCode();
65+
}
66+
67+
private int computeHashCode() {
68+
return identityHashCode(actual) + identityHashCode(expected) + fieldLocation.hashCode();
6569
}
6670

6771
@Override
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
/*
2+
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
3+
* the License. You may obtain a copy of the License at
4+
*
5+
* http://www.apache.org/licenses/LICENSE-2.0
6+
*
7+
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
8+
* an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
9+
* specific language governing permissions and limitations under the License.
10+
*
11+
* Copyright 2012-2024 the original author or authors.
12+
*/
13+
package org.assertj.core.api.recursive.comparison;
14+
15+
import org.junit.jupiter.api.Test;
16+
17+
import nl.jqno.equalsverifier.EqualsVerifier;
18+
19+
import java.util.Objects;
20+
21+
import static java.util.Objects.hash;
22+
import static org.assertj.core.api.BDDAssertions.then;
23+
import static org.assertj.core.util.Lists.list;
24+
25+
class DualValue_Test {
26+
27+
@Test
28+
void should_honor_equals_contract() {
29+
EqualsVerifier.forClass(DualValue.class)
30+
.withNonnullFields("fieldLocation")
31+
.withCachedHashCode("hashCode", "computeHashCode", new DualValue(list(), "foo", "bar"))
32+
.verify();
33+
}
34+
35+
private static final class MutableType {
36+
String changeMe;
37+
38+
private MutableType(String startingValue) {
39+
changeMe = startingValue;
40+
}
41+
42+
private void setChangeMe(String nextValue) {
43+
changeMe = nextValue;
44+
}
45+
46+
@Override
47+
public boolean equals(Object o) {
48+
if (this == o) return true;
49+
if (o == null || getClass() != o.getClass()) return false;
50+
MutableType that = (MutableType) o;
51+
return Objects.equals(changeMe, that.changeMe);
52+
}
53+
54+
@Override
55+
public int hashCode() {
56+
return hash(changeMe);
57+
}
58+
}
59+
60+
@Test
61+
void should_return_same_hashcode_after_mutation() {
62+
// GIVEN
63+
MutableType actual = new MutableType("One");
64+
MutableType expected = new MutableType("One");
65+
DualValue dualValue1 = new DualValue(list(), actual, expected);
66+
DualValue dualValue2 = new DualValue(list(), actual, expected);
67+
int hashCodeBeforeMutation = dualValue1.hashCode();
68+
// WHEN
69+
actual.setChangeMe("Another value");
70+
// THEN
71+
then(dualValue1).isEqualTo(dualValue2)
72+
.hasSameHashCodeAs(dualValue2);
73+
int hashCodeAfterMutation = dualValue1.hashCode();
74+
then(hashCodeAfterMutation).isEqualTo(hashCodeBeforeMutation);
75+
}
76+
77+
}

0 commit comments

Comments
 (0)