-
Notifications
You must be signed in to change notification settings - Fork 687
Nullability annotation support
What we call nullability annotation are the set of annotations used to describe the nullability value of elements.
@Nonnull
private String s; // "s" should never be null
@CheckForNull // The return value of "f()" should always be checked for null
String f(@Nullable Object o) {
// You should consider the case where "o" is null
}
The Java analyzer contains rules helping users to consistently use such annotations and is also using them to improve the precision of the analysis. The current ecosystem is a bit complex, there are many different annotation providers, sometimes with a slightly different understanding of the problem. It is important for us to have a common understanding of the problem and to apply it in a consistent way.
Disclaimer: This document does not aim to show how things should work or the best way to use annotations, but rather a description of how we use nullability annotations in order to provide relevant analysis results.
In the context of the analyzer, we will use 3 different notions, representing the nullability value of an element (argument, field, return value of a method, local variable).
- Non-null
The element is never null. It can therefore be safely dereferenced and does not need to be checked for null.
- Strong Nullable
An element which nullability can not be statically determined, it should always be checked for null. Typical example: Null Dereference Check. A method strongly nullable should always be checked for null.
- Weak Nullable
An element that can be null or not depending on the context. A user should therefore themself determine if a null value is acceptable and if he should check it for null. The notion of weak nullable is not directly needed in our rules, only though nullable.
- Nullable
Combination of Weak and Strong Nullable. Everything that could be null at one point.
This represents the different places where an annotation can be found.
- Variable
When the parameter, field or local variable is directly annotated.
- Method
When the method return value is directly annotated.
- Class
Applies to the whole class.
- Package
Package level annotation. The nullability value of this annotation will apply to the whole package
We have seen previously that the nullability value targets multiple elements:
- argument
- field
- return value of a method
- local variable
A given annotation can target only some of these elements.
When an annotation is itself annotated with a nullability annotation, we call this a meta-annotation.
The different levels represents also the priority, from variable to package. For example, something annotated @Nonnull
at package level but is directly annotated with @Nullable
will be nullable.
Meta-annotations at a given level have lower priority than when directly annotated, but the priority of level still applies.
The exact list of supported annotations is not fixed, the best way to have reliable information is to check the code itself: JSymbolMetadataNullabilityHelper.java#L67. Every annotation is then assigned a list of Levels (where can you put this annotation?) and a list of Targets (what element will be null when this annotation is used).