Skip to content

Commit 45b810a

Browse files
christophstroblmp911de
authored andcommitted
Use enclosing class constructor parameter only for non-static inner classes.
Closes #3038 Original pull request: #3039
1 parent eb4b0b4 commit 45b810a

File tree

2 files changed

+88
-1
lines changed

2 files changed

+88
-1
lines changed

src/main/java/org/springframework/data/mapping/Parameter.java

+3-1
Original file line numberDiff line numberDiff line change
@@ -24,13 +24,15 @@
2424
import org.springframework.data.util.TypeInformation;
2525
import org.springframework.lang.Nullable;
2626
import org.springframework.util.Assert;
27+
import org.springframework.util.ClassUtils;
2728
import org.springframework.util.StringUtils;
2829

2930
/**
3031
* Value object to represent constructor parameters.
3132
*
3233
* @param <T> the type of the parameter
3334
* @author Oliver Gierke
35+
* @author Christoph Strobl
3436
*/
3537
public class Parameter<T, P extends PersistentProperty<P>> {
3638

@@ -72,7 +74,7 @@ public Parameter(@Nullable String name, TypeInformation<T> type, Annotation[] an
7274
}
7375

7476
Class<T> owningType = entity.getType();
75-
return owningType.isMemberClass() && type.getType().equals(owningType.getEnclosingClass());
77+
return ClassUtils.isInnerClass(owningType) && type.getType().equals(owningType.getEnclosingClass());
7678
});
7779

7880
this.hasSpelExpression = Lazy.of(() -> StringUtils.hasText(getSpelExpression()));

src/test/java/org/springframework/data/mapping/ParameterUnitTests.java

+85
Original file line numberDiff line numberDiff line change
@@ -16,19 +16,27 @@
1616
package org.springframework.data.mapping;
1717

1818
import static org.assertj.core.api.Assertions.*;
19+
import static org.mockito.Mockito.*;
1920

2021
import java.lang.annotation.Annotation;
2122

2223
import org.junit.jupiter.api.Test;
2324
import org.junit.jupiter.api.extension.ExtendWith;
2425
import org.mockito.Mock;
26+
import org.mockito.Mockito;
2527
import org.mockito.junit.jupiter.MockitoExtension;
28+
import org.springframework.data.mapping.ParameterUnitTests.IFace.ClassMember;
29+
import org.springframework.data.mapping.ParameterUnitTests.IFace.RecordMember;
30+
import org.springframework.data.mapping.ParameterUnitTests.StaticType.NonStaticInner;
31+
import org.springframework.data.mapping.ParameterUnitTests.StaticType.RecordInner;
32+
import org.springframework.data.mapping.ParameterUnitTests.StaticType.StaticInner;
2633
import org.springframework.data.util.TypeInformation;
2734

2835
/**
2936
* Unit tests for {@link Parameter}.
3037
*
3138
* @author Oliver Gierke
39+
* @author Christoph Strobl
3240
*/
3341
@ExtendWith(MockitoExtension.class)
3442
class ParameterUnitTests<P extends PersistentProperty<P>> {
@@ -87,4 +95,81 @@ void twoParametersWithDifferenTypeAreNotEqual() {
8795

8896
assertThat(left).isNotEqualTo(right);
8997
}
98+
99+
@Test // GH-3038
100+
void shouldNotConsiderRecordTypeOfInterfaceEnclosingClassParameter() {
101+
102+
PersistentEntity pe = Mockito.mock(PersistentEntity.class);
103+
when(pe.getType()).thenReturn(RecordMember.class);
104+
105+
Parameter<IFace, P> iFace = new Parameter<IFace, P>("iFace", TypeInformation.of(IFace.class), annotations, pe);
106+
assertThat(iFace.isEnclosingClassParameter()).isFalse();
107+
}
108+
109+
@Test // GH-3038
110+
void shouldNotConsiderMemberTypeOfInterfaceEnclosingClassParameter() {
111+
112+
PersistentEntity pe = Mockito.mock(PersistentEntity.class);
113+
when(pe.getType()).thenReturn(ClassMember.class);
114+
115+
Parameter<IFace, P> iFace = new Parameter<IFace, P>("iFace", TypeInformation.of(IFace.class), annotations, pe);
116+
assertThat(iFace.isEnclosingClassParameter()).isFalse();
117+
}
118+
119+
@Test // GH-3038
120+
void shouldConsiderMemberTypeOfClassEnclosingClassParameter() {
121+
122+
PersistentEntity pe = Mockito.mock(PersistentEntity.class);
123+
when(pe.getType()).thenReturn(NonStaticInner.class);
124+
125+
Parameter<StaticType, P> iFace = new Parameter<StaticType, P>("outer", TypeInformation.of(StaticType.class),
126+
annotations, pe);
127+
assertThat(iFace.isEnclosingClassParameter()).isTrue();
128+
}
129+
130+
@Test // GH-3038
131+
void shouldNotConsiderStaticMemberTypeOfClassEnclosingClassParameter() {
132+
133+
PersistentEntity pe = Mockito.mock(PersistentEntity.class);
134+
when(pe.getType()).thenReturn(StaticInner.class);
135+
136+
Parameter<StaticType, P> iFace = new Parameter<StaticType, P>("outer", TypeInformation.of(StaticType.class),
137+
annotations, pe);
138+
assertThat(iFace.isEnclosingClassParameter()).isFalse();
139+
}
140+
141+
@Test // GH-3038
142+
void shouldNotConsiderRecordMemberTypeOfClassEnclosingClassParameter() {
143+
144+
PersistentEntity pe = Mockito.mock(PersistentEntity.class);
145+
when(pe.getType()).thenReturn(RecordInner.class);
146+
147+
Parameter<StaticType, P> iFace = new Parameter<StaticType, P>("outer", TypeInformation.of(StaticType.class),
148+
annotations, pe);
149+
assertThat(iFace.isEnclosingClassParameter()).isFalse();
150+
}
151+
152+
interface IFace {
153+
154+
record RecordMember(IFace iFace) {
155+
}
156+
157+
class ClassMember {
158+
ClassMember(IFace iface) {}
159+
}
160+
}
161+
162+
static class StaticType {
163+
164+
class NonStaticInner {
165+
NonStaticInner(StaticType outer) {}
166+
}
167+
168+
static class StaticInner {
169+
StaticInner(StaticType outer) {}
170+
}
171+
172+
record RecordInner(StaticType outer) {
173+
}
174+
}
90175
}

0 commit comments

Comments
 (0)