-
Notifications
You must be signed in to change notification settings - Fork 11
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[FR] Nullable checking to avoid null checking code generation #4
Comments
Given that jstachio is now on Java 17, I think we should try to standardize and use JSpecify annotations. JSpecify seems to be the future, at least I hope so. I agree that current support for null is unfortunate. I think this is caused by the fact that I've started static-mustache at the times before Java 8 and there were no convincing null-checking story then. |
I did some work on this and found some issues with TYPE_USE annotations across compile time boundaries. See jspecify/jspecify#365 The short of it is types that are not in the same compilation unit that JStachio is compiling do not have their |
Looking into this I think we may long-term look into fixing https://bugs.openjdk.org/browse/JDK-8225377 ourselves. And at the same time provide some crude work-around. As a work-around that is good enough for simplicity/functionality, we may want to always ignore
and document in the |
Note mostly to myself but a possible solution to checking for Nullable TYPE_USE given the JDK bug is using Jandex: https://smallrye.io/jandex/jandex/3.1.2/core/browsing.html |
And we are all excited about that :) The fix so far is just in the head version of javac, not any release. It will be part of Java 22, though that's a non-LTS release, when it comes out (scheduled for March 19). As you probably saw on the bug, Liam is looking to backport it to 11, 17, and 21. If all goes well, it would be in an update release of each of those, probably released in the same ballpark as Java 22. |
Thanks @cpovirk This is pretty exciting and I'm looking forward to continue making JStachio one of the safest templating languages. I think even those in the |
To determine if DeclaredType aka class instances are falsey we generate code to check if they are null.
While that seems like a good idea in the context of Javascript in most Java codebases today things are not null and if they are it is often a serious bug. JStachio exacerbates this because it will hide it instead of failing fast like you would if you were to write rendering code manually or using more programmatic templating like JTE or Rocker.
In my opinion this violates JStachio core philosophy of being as typesafe as possible and failing as fast as possible preferable during compile time.
Thus I propose an option where we support
TYPE_USE
@Nullable
annotations like JSpecify (not released yet), Eclipses and kind of Intellijs. Yes there are other nullable annotations that are notTYPE_USE
but those are far more difficult to support (frankly that ones that do not... they are fucking wrong and is disappointing because TYPE_USE has been around since JDK 8 but I will leave that rant for another day... findbugs aka spotbugs)What we will do is check if there is annotation is present on the DeclaredType that has a simple name of
Nullable
(potentially configurable) and if it is then we will do null checking otherwise it is assumed it is nonnull like a native type (and currentlyOptional
asOptional
already has this behavior).As a side note this will increase rendering performance slightly but that is not the reason we want to do it.
This is a lot of work but worth it. I'm not sure if I can get it in before end of year which I plan on 1.0.0.
cc @sviperll
EDIT
I should go over the various level of support of nullable we want to achieve
EXHAUST_NULLABLE
Not implemented yet
Force sectioning on a nullable variable
Example:
Template code like below will fail during compilation:
As message could be null.
To correct it would be to do the following:
NULL_CHECK_DISABLE
✅ implemented see:
NO_NULL_CHECKING
#134 and #164
Remove conditional checking if a variable is null if it is not
@Nullable
Currently anytime a section is accessed we check if its not null.
Example:
The code generated is roughly like:
Notice that
action
got called twice. It maybe a separate feature that we store the action return results as variable but it is relevant to nullable support as there is no guarantee calling the method twice will yield the same results. Furthermore the above hides potential bugs if we are expecting action to never be null.Anyway
NULL_CHECK_DISABLE
turned on would yield code like:N.B. That
NULL_CHECK_DISABLE
does not necessarily need to read nullable annotations to work unlikeEXHAUST_NULLABLE
.NULL_CHECK_DOTTED_PATH
Not implemented
There also could be various levels of
NULL_CHECK_DISABLE
in the case of dotted paths. That is maybe only the first name in path (model
in this case) is checked or not checked. Or maybe non dotted paths are treated differentlyFor example:
Should generate:
But maybe not. If nullable annotations are not available we need to assume they are testing for null to prevent NPE and thus we would always generate the above code for an undotted name (
model
).The question is what to do for:
Should model only be checked? Should model and action be checked? Should only action (last in dotted name) be checked?
Need to get feedback.
EXHAUST_NULLABLE
is not going to work at the moment or anything that needs to absolutely confirm something is nullable as TYPE_USE annotations are not available across compile time boundaries unless we do some--add-modules
hacks to load upSymbol
.Furthermore a separate feature that overlaps is storing property accessors on the stack as variables to avoid repeatedly calling the accessor methods.
The text was updated successfully, but these errors were encountered: