Skip to content

Commit 0499708

Browse files
committed
DATACMNS-1114 - Introduced usage of nullable annotations for API validation.
Marked all packages with Spring Frameworks @NonNullApi. Added Spring's @nullable to methods, parameters and fields that take or produce null values. Adapted using code to make sure the IDE can evaluate the null flow properly. Fixed Javadoc in places where an invalid null handling policy was advertised. Strengthened null requirements for types that expose null-instances. Removed null handling from converters for JodaTime and ThreeTenBP. Introduced factory methods Page.empty() and Page.empty(Pageable). Introduced default methods getRequiredGetter(), …Setter() and …Field() on PersistentProperty to allow non-nullable lookups of members. The same for TypeInformation.getrequiredActualType(), …SuperTypeInformation(). Tweaked PersistentPropertyCreator.addPropertiesForRemainingDescriptors() to filter unsuitable PropertyDescriptors before actually trying to create a Property instance from them as the new stronger nullability requirements would cause exceptions downstream. Lazy.get() now expects a non-null return value. Clients being able to cope with null need to call ….orElse(…). Original pull request: #232.
1 parent d9b16d8 commit 0499708

File tree

234 files changed

+2272
-1177
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

234 files changed

+2272
-1177
lines changed

pom.xml

-1
Original file line numberDiff line numberDiff line change
@@ -293,7 +293,6 @@
293293
<plugin>
294294
<groupId>org.apache.maven.plugins</groupId>
295295
<artifactId>maven-enforcer-plugin</artifactId>
296-
<version>1.3.1</version>
297296
<executions>
298297
<execution>
299298
<id>enforce-rules</id>
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
/**
22
* Core annotations being used by Spring Data.
33
*/
4-
package org.springframework.data.annotation;
4+
@org.springframework.lang.NonNullApi
5+
package org.springframework.data.annotation;

src/main/java/org/springframework/data/auditing/AuditingHandler.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -226,7 +226,7 @@ private Optional<TemporalAccessor> touchDate(AuditableBeanWrapper wrapper, boole
226226
*/
227227
public void afterPropertiesSet() {
228228

229-
if (auditorAware == null) {
229+
if (!auditorAware.isPresent()) {
230230
LOGGER.debug("No AuditorAware set! Auditing will not be applied!");
231231
}
232232
}

src/main/java/org/springframework/data/auditing/DefaultAuditableBeanWrapperFactory.java

+5-2
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@
3434
import org.springframework.data.domain.Auditable;
3535
import org.springframework.data.util.ReflectionUtils;
3636
import org.springframework.format.support.DefaultFormattingConversionService;
37+
import org.springframework.lang.Nullable;
3738
import org.springframework.util.Assert;
3839

3940
/**
@@ -88,7 +89,7 @@ public AuditableInterfaceBeanWrapper(Auditable<Object, ?, TemporalAccessor> audi
8889

8990
this.auditable = auditable;
9091
this.type = (Class<? extends TemporalAccessor>) ResolvableType.forClass(Auditable.class, auditable.getClass())
91-
.getGeneric(2).getRawClass();
92+
.getGeneric(2).resolve(TemporalAccessor.class);
9293
}
9394

9495
/*
@@ -183,6 +184,7 @@ public DateConvertingAuditableBeanWrapper() {
183184
* @param source must not be {@literal null}.
184185
* @return
185186
*/
187+
@Nullable
186188
protected Object getDateValueToSet(TemporalAccessor value, Class<?> targetType, Object source) {
187189

188190
if (TemporalAccessor.class.equals(targetType)) {
@@ -213,6 +215,7 @@ protected Object getDateValueToSet(TemporalAccessor value, Class<?> targetType,
213215
* Returns the given object as {@link Calendar}.
214216
*
215217
* @param source can be {@literal null}.
218+
* @param target must not be {@literal null}.
216219
* @return
217220
*/
218221
@SuppressWarnings("unchecked")
@@ -299,7 +302,7 @@ public Optional<TemporalAccessor> getLastModifiedDate() {
299302
return getAsTemporalAccessor(metadata.getLastModifiedDateField().map(field -> {
300303

301304
Object value = org.springframework.util.ReflectionUtils.getField(field, target);
302-
return Optional.class.isInstance(value) ? ((Optional<?>) value).orElse(null) : value;
305+
return value instanceof Optional ? ((Optional<?>) value).orElse(null) : value;
303306

304307
}), TemporalAccessor.class);
305308
}

src/main/java/org/springframework/data/auditing/config/AnnotationAuditingConfiguration.java

+10-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
package org.springframework.data.auditing.config;
1717

1818
import java.lang.annotation.Annotation;
19+
import java.util.Map;
1920

2021
import org.springframework.core.annotation.AnnotationAttributes;
2122
import org.springframework.core.type.AnnotationMetadata;
@@ -30,6 +31,8 @@
3031
*/
3132
public class AnnotationAuditingConfiguration implements AuditingConfiguration {
3233

34+
private static final String MISSING_ANNOTATION_ATTRIBUTES = "Couldn't find annotation attributes for %s in %s!";
35+
3336
private final AnnotationAttributes attributes;
3437

3538
/**
@@ -44,7 +47,13 @@ public AnnotationAuditingConfiguration(AnnotationMetadata metadata, Class<? exte
4447
Assert.notNull(metadata, "AnnotationMetadata must not be null!");
4548
Assert.notNull(annotation, "Annotation must not be null!");
4649

47-
this.attributes = new AnnotationAttributes(metadata.getAnnotationAttributes(annotation.getName()));
50+
Map<String, Object> attributesSource = metadata.getAnnotationAttributes(annotation.getName());
51+
52+
if (attributesSource == null) {
53+
throw new IllegalArgumentException(String.format(MISSING_ANNOTATION_ATTRIBUTES, annotation, metadata));
54+
}
55+
56+
this.attributes = new AnnotationAttributes(attributesSource);
4857
}
4958

5059
/*

src/main/java/org/springframework/data/auditing/config/AuditingHandlerBeanDefinitionParser.java

+4
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717

1818
import static org.springframework.beans.factory.support.BeanDefinitionBuilder.*;
1919

20+
import javax.annotation.Nonnull;
21+
2022
import org.springframework.aop.framework.ProxyFactoryBean;
2123
import org.springframework.aop.target.LazyInitTargetSource;
2224
import org.springframework.beans.factory.config.BeanDefinition;
@@ -51,6 +53,7 @@ public class AuditingHandlerBeanDefinitionParser extends AbstractSingleBeanDefin
5153
*
5254
* @param mappingContextBeanName must not be {@literal null} or empty.
5355
*/
56+
@SuppressWarnings("null")
5457
public AuditingHandlerBeanDefinitionParser(String mappingContextBeanName) {
5558

5659
Assert.hasText(mappingContextBeanName, "MappingContext bean name must not be null!");
@@ -70,6 +73,7 @@ public String getResolvedBeanName() {
7073
* (non-Javadoc)
7174
* @see org.springframework.beans.factory.xml.AbstractSingleBeanDefinitionParser#getBeanClass(org.w3c.dom.Element)
7275
*/
76+
@Nonnull
7377
@Override
7478
protected Class<?> getBeanClass(Element element) {
7579
return AuditingHandler.class;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
/**
2+
* Types to abstract authentication concepts.
3+
*/
4+
@org.springframework.lang.NonNullApi
5+
package org.springframework.data.auditing.config;
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
/**
22
* General support for entity auditing.
33
*/
4-
package org.springframework.data.auditing;
4+
@org.springframework.lang.NonNullApi
5+
package org.springframework.data.auditing;

src/main/java/org/springframework/data/authentication/UserCredentials.java

+10-5
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
*/
1616
package org.springframework.data.authentication;
1717

18+
import org.springframework.lang.Nullable;
1819
import org.springframework.util.ObjectUtils;
1920
import org.springframework.util.StringUtils;
2021

@@ -28,8 +29,7 @@ public class UserCredentials {
2829

2930
public static final UserCredentials NO_CREDENTIALS = new UserCredentials(null, null);
3031

31-
private final String username;
32-
private final String password;
32+
private final @Nullable String username, password;
3333

3434
/**
3535
* Creates a new {@link UserCredentials} instance from the given username and password. Empty {@link String}s provided
@@ -38,7 +38,7 @@ public class UserCredentials {
3838
* @param username
3939
* @param password
4040
*/
41-
public UserCredentials(String username, String password) {
41+
public UserCredentials(@Nullable String username, @Nullable String password) {
4242
this.username = StringUtils.hasText(username) ? username : null;
4343
this.password = StringUtils.hasText(password) ? password : null;
4444
}
@@ -48,6 +48,7 @@ public UserCredentials(String username, String password) {
4848
*
4949
* @return the username
5050
*/
51+
@Nullable
5152
public String getUsername() {
5253
return username;
5354
}
@@ -57,6 +58,7 @@ public String getUsername() {
5758
*
5859
* @return the password
5960
*/
61+
@Nullable
6062
public String getPassword() {
6163
return password;
6264
}
@@ -85,9 +87,12 @@ public boolean hasPassword() {
8587
*
8688
* @return the obfuscated password
8789
*/
90+
@Nullable
8891
public String getObfuscatedPassword() {
8992

90-
if (!hasPassword()) {
93+
String password = this.password;
94+
95+
if (password == null) {
9196
return null;
9297
}
9398

@@ -125,7 +130,7 @@ public String toString() {
125130
* @see java.lang.Object#equals(java.lang.Object)
126131
*/
127132
@Override
128-
public boolean equals(Object obj) {
133+
public boolean equals(@Nullable Object obj) {
129134

130135
if (obj == this) {
131136
return true;
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
/**
22
* Types to abstract authentication concepts.
33
*/
4-
package org.springframework.data.authentication;
4+
@org.springframework.lang.NonNullApi
5+
package org.springframework.data.authentication;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
/*
2+
* Copyright 2017 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+
* http://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.config;
17+
18+
import org.springframework.beans.factory.config.BeanDefinition;
19+
import org.springframework.beans.factory.xml.XmlReaderContext;
20+
import org.springframework.core.io.ResourceLoader;
21+
import org.springframework.util.Assert;
22+
23+
/**
24+
* Helper class to centralize common functionality that needs to be used in various places of the configuration
25+
* implementation.
26+
*
27+
* @author Oliver Gierke
28+
* @since 2.0
29+
* @soundtrack Richard Spaven - The Self (feat. Jordan Rakei)
30+
*/
31+
public interface ConfigurationUtils {
32+
33+
/**
34+
* Returns the {@link ResourceLoader} from the given {@link XmlReaderContext}.
35+
*
36+
* @param context must not be {@literal null}.
37+
* @return
38+
* @throws IllegalArgumentException if no {@link ResourceLoader} can be obtained from the {@link XmlReaderContext}.
39+
*/
40+
public static ResourceLoader getRequiredResourceLoader(XmlReaderContext context) {
41+
42+
Assert.notNull(context, "XmlReaderContext must not be null!");
43+
44+
ResourceLoader resourceLoader = context.getResourceLoader();
45+
46+
if (resourceLoader == null) {
47+
throw new IllegalArgumentException("Could not obtain ResourceLoader from XmlReaderContext!");
48+
}
49+
50+
return resourceLoader;
51+
}
52+
53+
/**
54+
* Returns the {@link ClassLoader} used by the given {@link XmlReaderContext}.
55+
*
56+
* @param context must not be {@literal null}.
57+
* @return
58+
* @throws IllegalArgumentException if no {@link ClassLoader} can be obtained from the given {@link XmlReaderContext}.
59+
*/
60+
public static ClassLoader getRequiredClassLoader(XmlReaderContext context) {
61+
return getRequiredClassLoader(getRequiredResourceLoader(context));
62+
}
63+
64+
/**
65+
* Returns the {@link ClassLoader} used by the given {@link ResourceLoader}.
66+
*
67+
* @param resourceLoader must not be {@literal null}.
68+
* @return
69+
* @throws IllegalArgumentException if the given {@link ResourceLoader} does not expose a {@link ClassLoader}.
70+
*/
71+
public static ClassLoader getRequiredClassLoader(ResourceLoader resourceLoader) {
72+
73+
Assert.notNull(resourceLoader, "ResourceLoader must not be null!");
74+
75+
ClassLoader classLoader = resourceLoader.getClassLoader();
76+
77+
if (classLoader == null) {
78+
throw new IllegalArgumentException("Could not obtain ClassLoader from ResourceLoader!");
79+
}
80+
81+
return classLoader;
82+
}
83+
84+
/**
85+
* Returns the bean class name of the given {@link BeanDefinition}.
86+
*
87+
* @param beanDefinition must not be {@literal null}.
88+
* @return
89+
* @throws IllegalArgumentException if the given {@link BeanDefinition} does not contain a bean class name.
90+
*/
91+
public static String getRequiredBeanClassName(BeanDefinition beanDefinition) {
92+
93+
Assert.notNull(beanDefinition, "BeanDefinition must not be null!");
94+
95+
String result = beanDefinition.getBeanClassName();
96+
97+
if (result == null) {
98+
throw new IllegalArgumentException(
99+
String.format("Could not obtain required bean class name from BeanDefinition!", beanDefinition));
100+
}
101+
102+
return result;
103+
}
104+
}

src/main/java/org/springframework/data/config/ParsingUtils.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
2222
import org.springframework.beans.factory.xml.BeanDefinitionParser;
2323
import org.springframework.beans.factory.xml.ParserContext;
24+
import org.springframework.lang.Nullable;
2425
import org.springframework.util.Assert;
2526
import org.springframework.util.StringUtils;
2627
import org.w3c.dom.Element;
@@ -120,7 +121,7 @@ public static AbstractBeanDefinition getSourceBeanDefinition(BeanDefinitionBuild
120121
* @param source
121122
* @return
122123
*/
123-
public static AbstractBeanDefinition getSourceBeanDefinition(BeanDefinitionBuilder builder, Object source) {
124+
public static AbstractBeanDefinition getSourceBeanDefinition(BeanDefinitionBuilder builder, @Nullable Object source) {
124125

125126
Assert.notNull(builder, "Builder must not be null!");
126127

src/main/java/org/springframework/data/config/TypeFilterParser.java

+10-7
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@
3030
import org.springframework.core.type.filter.AssignableTypeFilter;
3131
import org.springframework.core.type.filter.RegexPatternTypeFilter;
3232
import org.springframework.core.type.filter.TypeFilter;
33+
import org.springframework.lang.Nullable;
3334
import org.springframework.util.Assert;
3435
import org.w3c.dom.Element;
3536
import org.w3c.dom.Node;
@@ -55,7 +56,7 @@ public class TypeFilterParser {
5556
* @param readerContext must not be {@literal null}.
5657
*/
5758
public TypeFilterParser(XmlReaderContext readerContext) {
58-
this(readerContext, readerContext.getResourceLoader().getClassLoader());
59+
this(readerContext, ConfigurationUtils.getRequiredClassLoader(readerContext));
5960
}
6061

6162
/**
@@ -92,13 +93,14 @@ public Collection<TypeFilter> parseTypeFilters(Element element, Type type) {
9293
Node node = nodeList.item(i);
9394
Element childElement = type.getElement(node);
9495

95-
if (childElement != null) {
96+
if (childElement == null) {
97+
continue;
98+
}
9699

97-
try {
98-
filters.add(createTypeFilter(childElement, classLoader));
99-
} catch (RuntimeException e) {
100-
readerContext.error(e.getMessage(), readerContext.extractSource(element), e.getCause());
101-
}
100+
try {
101+
filters.add(createTypeFilter(childElement, classLoader));
102+
} catch (RuntimeException e) {
103+
readerContext.error(e.getMessage(), readerContext.extractSource(element), e.getCause());
102104
}
103105
}
104106

@@ -224,6 +226,7 @@ private Type(String elementName) {
224226
* @param node
225227
* @return
226228
*/
229+
@Nullable
227230
Element getElement(Node node) {
228231

229232
if (node.getNodeType() == Node.ELEMENT_NODE) {
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
/**
22
* Basic support for creating custom Spring namespaces and JavaConfig.
33
*/
4-
package org.springframework.data.config;
4+
@org.springframework.lang.NonNullApi
5+
package org.springframework.data.config;

0 commit comments

Comments
 (0)