Skip to content

Commit 33dbc9a

Browse files
committed
Introduce PrimiviteProvider to determine nullability definitions.
1 parent 622c6f3 commit 33dbc9a

File tree

6 files changed

+125
-5
lines changed

6 files changed

+125
-5
lines changed

src/main/java/org/springframework/data/util/Nullability.java

+19
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,9 @@
6262
* }
6363
* }
6464
* </pre>
65+
*
66+
* Note that nullability is also expressed through using specific types. Primitives ({@code int}, {@code char} etc.) are
67+
* non-nullable by definition and cannot be {@code null}. {@code void}/{@code Void} types are {@code null}-only types.
6568
* <p>
6669
* {@code javax.annotation.Nonnull} is suitable for composition of meta-annotations and expresses via
6770
* {@code javax.annotation.Nonnull#when()} in which cases non-nullability is applicable. Nullability introspection
@@ -270,6 +273,15 @@ default Nullability forReturnType() {
270273
*/
271274
Nullability forParameter(Parameter parameter);
272275

276+
/**
277+
* Returns a {@link Nullability} instance for a method parameter by index.
278+
*
279+
* @param index the method parameter index.
280+
* @return a {@link Nullability} instance for a method parameter.
281+
* @throws IndexOutOfBoundsException if the method parameter index is out of bounds.
282+
*/
283+
Nullability forParameter(int index);
284+
273285
/**
274286
* Returns a {@link Nullability} instance for a method parameter.
275287
*
@@ -281,6 +293,13 @@ default Nullability forParameter(MethodParameter parameter) {
281293
return parameter.getParameterIndex() == -1 ? forReturnType() : forParameter(parameter.getParameter());
282294
}
283295

296+
/**
297+
* Returns the method parameter count.
298+
*
299+
* @return the method parameter count.
300+
*/
301+
int getParameterCount();
302+
284303
}
285304

286305
}

src/main/java/org/springframework/data/util/NullabilityIntrospector.java

+51-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,9 @@ class NullabilityIntrospector implements Nullability.Introspector {
5353
private static final List<NullabilityProvider> providers;
5454

5555
static {
56-
providers = new ArrayList<>(4);
56+
providers = new ArrayList<>(5);
57+
58+
providers.add(new PrimitiveProvider());
5759

5860
if (Jsr305Provider.isAvailable()) {
5961
providers.add(new Jsr305Provider());
@@ -202,6 +204,39 @@ static abstract class NullabilityProvider {
202204
abstract Spec evaluate(AnnotatedElement element, ElementType elementType);
203205
}
204206

207+
/**
208+
* Provider considering primitive types.
209+
*/
210+
static class PrimitiveProvider extends NullabilityProvider {
211+
212+
@Override
213+
Spec evaluate(AnnotatedElement element, ElementType elementType) {
214+
215+
Class<?> type = null;
216+
217+
if (element instanceof Method m) {
218+
type = m.getReturnType();
219+
}
220+
221+
if (element instanceof Parameter p) {
222+
type = p.getType();
223+
}
224+
225+
if (type != null) {
226+
227+
if (ReflectionUtils.isVoid(type)) {
228+
return Spec.NULLABLE;
229+
}
230+
231+
if (type.isPrimitive()) {
232+
return Spec.NON_NULL;
233+
}
234+
}
235+
236+
return Spec.UNSPECIFIED;
237+
}
238+
}
239+
205240
/**
206241
* Spring provider leveraging {@link NonNullApi @NonNullApi}, {@link NonNullFields @NonNullFields},
207242
* {@link NonNull @NonNull}, and {@link Nullable @Nullable} annotations.
@@ -515,6 +550,21 @@ public Nullability forParameter(Parameter parameter) {
515550

516551
return nullability;
517552
}
553+
554+
@Override
555+
public Nullability forParameter(int index) {
556+
557+
if (index >= method.getParameterCount() || index < 0) {
558+
throw new IndexOutOfBoundsException();
559+
}
560+
561+
return forParameter(method.getParameters()[index]);
562+
}
563+
564+
@Override
565+
public int getParameterCount() {
566+
return method.getParameterCount();
567+
}
518568
}
519569

520570
/**
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
/*
2+
* Copyright 2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.util;
17+
18+
import static org.assertj.core.api.Assertions.*;
19+
20+
import java.lang.reflect.Method;
21+
22+
import org.junit.jupiter.api.Test;
23+
24+
/**
25+
* Unit tests for {@link Nullability}.
26+
*
27+
* @author Mark Paluch
28+
*/
29+
class NullabilityUnitTests {
30+
31+
@Test
32+
void shouldConsiderPrimitiveNullability() throws NoSuchMethodException {
33+
34+
Method method = getClass().getDeclaredMethod("someMethod", Integer.TYPE);
35+
Nullability.MethodNullability methodNullability = Nullability.forMethod(method);
36+
37+
// method return type
38+
assertThat(methodNullability.isDeclared()).isTrue();
39+
assertThat(methodNullability.isNullable()).isTrue();
40+
41+
Nullability pn = methodNullability.forParameter(0);
42+
43+
// method return type
44+
assertThat(pn.isDeclared()).isTrue();
45+
assertThat(pn.isNullable()).isFalse();
46+
}
47+
48+
void someMethod(int i) {
49+
50+
}
51+
}

src/test/java/org/springframework/data/util/nonnull/type/Jsr305NonnullAnnotatedType.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,6 @@
2323
@Nonnull
2424
public interface Jsr305NonnullAnnotatedType {
2525

26-
void someMethod(String arg);
26+
String someMethod(String arg);
2727

2828
}

src/test/java/org/springframework/data/util/nonnull/type/NonAnnotatedType.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,8 @@
2020
*/
2121
public class NonAnnotatedType {
2222

23-
void someMethod(String arg){
24-
return;
23+
String someMethod(String arg) {
24+
return "";
2525
}
2626

2727
}

src/test/java/org/springframework/data/util/nonnull/type/NonNullableParameters.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,5 +23,5 @@
2323
@ParametersAreNonnullByDefault
2424
public interface NonNullableParameters {
2525

26-
void someMethod(String arg);
26+
String someMethod(String arg);
2727
}

0 commit comments

Comments
 (0)