Skip to content
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

Open
agentgt opened this issue Nov 9, 2022 · 7 comments
Open

[FR] Nullable checking to avoid null checking code generation #4

agentgt opened this issue Nov 9, 2022 · 7 comments
Labels
enhancement New feature or request
Milestone

Comments

@agentgt
Copy link
Contributor

agentgt commented Nov 9, 2022

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 not TYPE_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 currently Optional as Optional 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:

@JStache
public record Model(@Nullable String message){}

Template code like below will fail during compilation:

{{message}}

As message could be null.

To correct it would be to do the following:

{{#message}}
{{.}} non null case
{{/message}}
{{^message}}
null case
{{/message}}

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:

{{model.action.message}}

The code generated is roughly like:

if (model != null) { 
   if (model.action() != null) {
      format(model.action().message());
   }
}

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:

format(model.action().message())

N.B. That NULL_CHECK_DISABLE does not necessarily need to read nullable annotations to work unlike EXHAUST_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 differently

For example:

{{#model}}
{{/model}}

Should generate:

if(model != null) {
}

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:

{{#model.action}}
{{/model.action}}

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 up Symbol.

Furthermore a separate feature that overlaps is storing property accessors on the stack as variables to avoid repeatedly calling the accessor methods.

@agentgt agentgt added the enhancement New feature or request label Nov 9, 2022
@sviperll
Copy link
Contributor

sviperll commented Nov 9, 2022

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.

@agentgt agentgt modified the milestones: v1.0.0, v0.9.0 Dec 5, 2022
@agentgt agentgt modified the milestones: v0.9.0, v1.1.0 Dec 19, 2022
@agentgt
Copy link
Contributor Author

agentgt commented Jan 9, 2023

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 Nullable annotations visible thus making JStachio believe they are nonnull.

@sviperll
Copy link
Contributor

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 Nullable annotations altogether, so that users doesn't have false hopes about nullness support. Instead we can provide something like

@JStachNulls({"a.b", "a.b.d"}) // a.b and "a.b.d" are allowed to be null
@JStach(template="{{#a}}{{#b}}B is not null with {{c}}{{#d}} and {{.}}{{/d}}{{/b}}{{/a}}")
record Tmpl(A a) {
}

and document in the @JStachNulls javadoc that JDK-8225377 fix is required for better long term nullness story support.

agentgt added a commit to agentgt/jstachio that referenced this issue May 4, 2023
agentgt added a commit that referenced this issue May 8, 2023
@agentgt agentgt modified the milestones: v1.1.0, v2.0.0 Jun 6, 2023
@agentgt agentgt pinned this issue Jun 6, 2023
@agentgt
Copy link
Contributor Author

agentgt commented Jun 8, 2023

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

@zimmi
Copy link

zimmi commented Nov 9, 2023

https://bugs.openjdk.org/browse/JDK-8225377 is fixed now.

@cpovirk
Copy link

cpovirk commented Nov 9, 2023

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.

@agentgt
Copy link
Contributor Author

agentgt commented Nov 9, 2023

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 Optional as parameters camp do not like putting them as fields (accessor or not) so perhaps JStachio supporting nullable annotations will help JSpecfiy uptake (albeit extremely minuscule every bit counts!).

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

No branches or pull requests

4 participants