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

Java 21 / JEP 441: Pattern Matching for switch result in com.palantir.javaformat.java.FormatterException #952

Open
matsev opened this issue Nov 16, 2023 · 15 comments

Comments

@matsev
Copy link

matsev commented Nov 16, 2023

Java 21 / JEP 441 : Pattern Matching for switch code result in com.palantir.javaformat.java.FormatterException

What happened?

This method compiles using JDK 21:

class Example {
    static String test(CharSequence charSequence) {
        return switch (charSequence) {
            case String s when s.isEmpty() -> "empty string";
            case String s -> "non-empty string";
            default -> "not a String";
        };
    }
}

However, when using the spotless-maven-plugin together with <palantirJavaFormat> results in an exception:

mvn spotless:apply
[...]
Unable to format file [path to Example.java]: com.palantir.javaformat.java.FormatterException: error: expected token: 'when'; generated - instead -> [Help 1]

Maven config:

                <plugin>
                    <groupId>com.diffplug.spotless</groupId>
                    <artifactId>spotless-maven-plugin</artifactId>
                    <version>2.40.0</version>
                    <configuration>
                        <java>
                            <palantirJavaFormat>
                                <version>2.38.0</version>
                            </palantirJavaFormat>
                        </java>
                    </configuration>
                </plugin>

Maven:

mvn -v
Apache Maven 3.9.5 (57804ffe001d7215b5e7bcb531cf83df38f93546)
Java version: 21, vendor: Amazon.com Inc.
Default locale: en_GB, platform encoding: UTF-8
OS name: "mac os x", version: "13.6.1", arch: "x86_64", family: "mac"

What did you want to happen?

The maven-spotless-plugin and planter-java-format should accept or reformat code that is supported by JDK 21

@koppor
Copy link
Contributor

koppor commented Nov 20, 2023

Can you manage to compile and try #935?

@matsev
Copy link
Author

matsev commented Nov 21, 2023

I have checked out PR #935 locally and I manage to build it if I use JDK 17 when running Cradle.

./gradlew -v

------------------------------------------------------------
Gradle 7.6.2
------------------------------------------------------------

Build time:   2023-06-30 15:42:51 UTC
Revision:     dab132169006b16e7ada4ab2456e0c9d6415b52a

Kotlin:       1.7.10
Groovy:       3.0.13
Ant:          Apache Ant(TM) version 1.10.11 compiled on July 10 2021
JVM:          17.0.8 (Amazon.com Inc. 17.0.8+7-LTS)
OS:           Mac OS X 13.6.1 x86_64
./gradlew build
[...]
BUILD SUCCESSFUL in 29s
93 actionable tasks: 56 executed, 25 from cache, 12 up-to-date

However, that PR fails when tried to build using JDK 21 as it is not supported for running Gradle according to its compatibility matrix:

./gradlew -v

------------------------------------------------------------
Gradle 7.6.2
------------------------------------------------------------

Build time:   2023-06-30 15:42:51 UTC
Revision:     dab132169006b16e7ada4ab2456e0c9d6415b52a

Kotlin:       1.7.10
Groovy:       3.0.13
Ant:          Apache Ant(TM) version 1.10.11 compiled on July 10 2021
JVM:          21 (Amazon.com Inc. 21+35-LTS)
OS:           Mac OS X 13.6.1 x86_64
./gradlew build
[...]
FAILURE: Build failed with an exception.

* What went wrong:
Execution failed for task ':gradle-palantir-java-format:compileGroovy'.
> Unrecoverable compilation error: startup failed:
  General error during conversion: Unsupported class file major version 65

@matsev matsev changed the title JEP 441: Pattern Matching for switch result in com.palantir.javaformat.java.FormatterException Java 21 / JEP 441: Pattern Matching for switch result in com.palantir.javaformat.java.FormatterException Dec 4, 2023
@mandrean
Copy link

mandrean commented Jan 4, 2024

Bump:

Java 21 is now supported from Gradle ~8.4 @matsev

image

@matsev
Copy link
Author

matsev commented Jan 8, 2024

@mandrean Thanks for the notification. I made an attempt to upgrade the Gradle wrapper to use Gradle ver 8.5, but there is some more work required to make this work, please see #977 for details

@thijmenketel
Copy link

Any progress on this? We're starting to use more JDK21 features and this is preventing us to use when.

@piotrooo
Copy link

👍 urgent and necessary

@rohan-changejar
Copy link

Any update on this, we have moved to java 21 and using the new switch case features seem impossible without this change

@talios
Copy link

talios commented Sep 3, 2024

Do you happen to have any updates on this issue? Is there anything specific holding it up that we can do to help resolve this?

Java 23 is just around the corner now, and I fear the lack of support for some of these newer constructs will become more and more of a problem for adoption (of both palantir-java-format and modern java itself).

@shollander
Copy link

google-java-format, on which this project is based, has supported Java 21 features for over 9 months: https://github.com/google/google-java-format/releases/tag/v1.19.0.

This project does not appear to be well maintained.

@carlmolemans
Copy link

@matsev Any progress on this? We were forced to rely on the google java format.

@CRogers
Copy link
Contributor

CRogers commented Jan 7, 2025

We're now actively working on 21 support.

For context, we discovered a number of JDK 21 bugs that has made rolling services onto JDK 21 runtime difficult. Particularly for arm64 on linux, which we are some of the heaviest users, including some miscompilations from bytecode to native code (!!) that resulted in serious issues. We're over that hump now with virtually all our services on 21 runtime, so are now in position to start moving code to use source level 21 internally, so are going to build out support in the formatter.

We're evaluating moving to using a graal native-image version of the formatter to make upgrading easier. This is actually likely the bulk of the work.

  • Currently, to use 21 source features with the formatter you'd need to ensure that both your Gradle daemon (for spotless) and IntelliJ use a 21 JDK. Providing a graal native-image should make distribution of the formatter much easier and not require these upgrades, especially as:
  • It also means that we should be able to ship preview/non-LTS changes in the formatter earlier.
  • Finally, there's a serious performance problem with the current setup where in IntelliJ we spin up new JDKs every time you format to ensure that the correct JVM --add-exports args, which is very slow, especially when formatting lots of files from inside IntelliJ (eg after an automated refactor). From our benchmarking, starting a native-image process is >10x faster, and means don't need to manage the lifecycle of a long running separate process or require manual action to change the JDK args of IntelliJ itself.

@matsev
Copy link
Author

matsev commented Jan 7, 2025

@carlmolemans I am sorry, I am not a maintainer of this repo. I filed this issue some time ago and I also made some initial minor efforts to get the ball in motion. I have subscribed to PR #935 as some progress has been shared in the comments, but not as elaborate as @CRogers' comment above.

@CRogers thanks for the update.

@talios
Copy link

talios commented Jan 7, 2025

so are now in position to start moving code to use source level 21 internally,

@CRogers does that mean even if NOT using 21 source features or higher, the build will still need to run from a 21+ JDK?

With the (potential) change to using native, does that mean any integration tools will just need to shell out to the formatter process, rather than call an API? ( spotless, or other build tool integration).

Great to see work on this finally. Cheers for a great formatter.

@CRogers
Copy link
Contributor

CRogers commented Jan 8, 2025

@CRogers does that mean even if NOT using 21 source features or higher, the build will still need to run from a 21+ JDK?

With the current formatter distribution setup, if you want to use source level X, you need to run the Gradle daemon (that spotless runs in) with a JDK >=X. The change to native means this is no longer the case, you can run the build with whatever JDK you want (although the Gradle plugin itself will have some minimum JDK requirement). Fwiw, Gradle 9 will require 17 daemon anyway.

With the (potential) change to using native, does that mean any integration tools will just need to shell out to the formatter process, rather than call an API? ( spotless, or other build tool integration).

Yes, although the spotless step is defined in this repo, so we'll update it. Similarly for the intellij plugin. If someone else has made a custom integration they'll need to update it too (although we'll likely still ship the jar for the formatter so maybe it'll continue to work).

@talios
Copy link

talios commented Jan 8, 2025

Right, that all sounds like a larger ecosystem reason - using spotless, and gradle. I'm currently using the spotless maven plugin, but also looking at writing support for https://github.com/rife2/bld .

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

No branches or pull requests

10 participants