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

JavaEE to Quarkus 2 Declarative Migration Recipe #73

Merged
merged 23 commits into from
Jul 12, 2024
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
d645b00
JavaEE to Quarkus 2 Declarative Migration Recipe
ammachado Feb 4, 2024
d1451f4
Add JavaEEtoQuarkus2MigrationTest and verify pom changes
timtebeek Feb 4, 2024
475857e
Merge branch 'main' into main
timtebeek Feb 4, 2024
f61e285
Merge branch 'main' into main
timtebeek Feb 5, 2024
2065265
Apply suggestions from code review
timtebeek Feb 5, 2024
6743def
Merge branch 'main' into main
timtebeek Feb 5, 2024
88efbda
Split Recipe to Maven Dependencies Upgrade YML and Code Tranformation…
ammachado Feb 5, 2024
c31622f
Updated Tests
ammachado Feb 5, 2024
60cc5e1
Remove recipe from quarkus.yml
ammachado Feb 5, 2024
8d5c734
Merge all three JavaEEtoQuarkus2 recipes into one file
timtebeek Feb 6, 2024
af2b54f
Update src/test/java/org/openrewrite/quarkus/quarkus2/JavaEEtoQuarkus…
timtebeek Feb 6, 2024
9509f27
Updated Dependency Migration Description to reflect that this migrate…
ShatanikB Feb 7, 2024
1867feb
Merge branch 'main' into main
timtebeek Feb 12, 2024
421ff81
Merge branch 'main' into main
timtebeek Mar 15, 2024
0ade672
Merge "main" into "main"
ShatanikB Apr 24, 2024
12e8a53
Updated dependency versions and tests
ShatanikB Apr 24, 2024
a509c5a
Merge branch 'main' into main
timtebeek May 2, 2024
57b3683
Merge branch 'main' into main
ShatanikB May 21, 2024
8f9d4a5
Merge branch 'main' into main
timtebeek Jun 14, 2024
23ef2d3
Move recipes and reformat tests
timtebeek Jul 12, 2024
ca1d498
Merge branch 'main' into main
timtebeek Jul 12, 2024
f1cb466
Break up dependency & maven recipes; expect to remove war plugin
timtebeek Jul 12, 2024
b25b2de
Bump failsafe and surefire plugin versions
timtebeek Jul 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ dependencies {
implementation("org.openrewrite:rewrite-properties:${rewriteVersion}")

implementation("org.openrewrite.recipe:rewrite-java-dependencies:${rewriteVersion}")
implementation("org.openrewrite.recipe:rewrite-migrate-java:${rewriteVersion}")

runtimeOnly("org.openrewrite:rewrite-java-17:${rewriteVersion}")

Expand All @@ -38,6 +39,8 @@ dependencies {
testImplementation("javax.xml.bind:jaxb-api:2.4.0-b180830.0359")
testImplementation("jakarta.xml.bind:jakarta.xml.bind-api:3.0.0")

testImplementation("javax:javaee-api:7.0")

testRuntimeOnly("org.openrewrite:rewrite-java-17:${rewriteVersion}")

testRuntimeOnly("io.quarkus:quarkus-grpc:1.13.+")
Expand Down
182 changes: 182 additions & 0 deletions src/main/resources/META-INF/rewrite/javaee-to-quarkus.yml
timtebeek marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,182 @@
#
# Copyright 2024 the original author or authors.
# <p>
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# <p>
# https://www.apache.org/licenses/LICENSE-2.0
# <p>
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
type: specs.openrewrite.org/v1beta/recipe
name: org.openrewrite.quarkus.quarkus2.JavaEEtoQuarkus2Migration
displayName: Migrate JavaEE to Quarkus 2
description: These recipes help with the migration of a JavaEE application using EJBs and Hibernate to Quarkus 2. Additional transformations like JSF, JMS, Quarkus Tests may be necessary.
recipeList:
# Migrate Maven Dependencies
- org.openrewrite.quarkus.quarkus2.JavaEEtoQuarkus2MavenDependencyMigration
# TODO Migrate Gradle Dependencies
# Migrate Code
- org.openrewrite.quarkus.quarkus2.JavaEEtoQuarkus2CodeMigration

---
type: specs.openrewrite.org/v1beta/recipe
name: org.openrewrite.quarkus.quarkus2.JavaEEtoQuarkus2MavenDependencyMigration
displayName: Migrate JavaEE Maven Dependencies to Quarkus 2
description: Upgrade Standard JavaEE dependencies to Quarkus 2 dependencies.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have a recipe locally that does exactly this (and most part of the dependency handling of this recipe). It adds the Camel Extensions for Quarkus, and the output is closer to what quarkus init generates. I can contribute it on a separate PR (but I need openrewrite/rewrite#4006 merged before that).

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

#4006 is now merged if you'd like to open that other PR @ammachado

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi @ammachado Did you have a PR for similar functionality ?

recipeList:
# Add Quarkus BOM
- org.openrewrite.maven.AddManagedDependency:
groupId: io.quarkus.platform
artifactId: quarkus-bom
version: '2.x'
type: pom
scope: import

# Add Basic Quarkus Extensions
- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-arc
- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-resteasy
- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-resteasy-jackson
- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-undertow
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure I would add this one except if the application is explicitly using servlet.

Especially since having undertow around changes the behavior of Quarkus.

Now this can probably be done as a follow-up.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added undertow to be able to use the 'SessionScoped' annotation as per this . Making the assumption that @javax.ejb.Stateful should be mapped to @javax.enterprise.context.SessionScoped in CDI

- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-hibernate-orm
- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-jdbc-h2
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same here, I don't think I would add these two except if the application makes use of persistence.

Now this can probably be done as a follow-up.

- org.openrewrite.java.dependencies.AddDependency:
groupId: io.quarkus
artifactId: quarkus-junit5
scope: test
- org.openrewrite.java.dependencies.AddDependency:
groupId: io.rest-assured
artifactId: rest-assured
scope: test

# Add Maven Plugins
- org.openrewrite.maven.AddPlugin:
groupId: io.quarkus.platform
artifactId: quarkus-maven-plugin
version: '2.16.12.Final'
timtebeek marked this conversation as resolved.
Show resolved Hide resolved
executions: <executions><execution><goals><goal>build</goal><goal>generate-code</goal><goal>generate-code-tests</goal></goals></execution></executions>
- org.openrewrite.maven.AddPlugin:
groupId: org.apache.maven.plugins
artifactId: maven-compiler-plugin
version: '3.10.1'
timtebeek marked this conversation as resolved.
Show resolved Hide resolved
configuration: <configuration><compilerArgs><arg>-parameters</arg></compilerArgs></configuration>
- org.openrewrite.maven.AddPlugin:
groupId: org.apache.maven.plugins
artifactId: maven-surefire-plugin
version: '3.0.0-M7'
timtebeek marked this conversation as resolved.
Show resolved Hide resolved
configuration: <configuration><systemPropertyVariables><java.util.logging.manager>org.jboss.logmanager.LogManager</java.util.logging.manager><maven.home>${maven.home}</maven.home></systemPropertyVariables></configuration>
- org.openrewrite.maven.AddPlugin:
groupId: org.apache.maven.plugins
artifactId: maven-failsafe-plugin
version: '3.0.0-M7'
timtebeek marked this conversation as resolved.
Show resolved Hide resolved
executions: <executions><execution><goals><goal>integration-test</goal><goal>verify</goal></goals></execution></executions>
configuration: <configuration><systemPropertyVariables><native.image.path>${project.build.directory}/${project.build.finalName}-runner</native.image.path><java.util.logging.manager>org.jboss.logmanager.LogManager</java.util.logging.manager><maven.home>${maven.home}</maven.home></systemPropertyVariables></configuration>
- org.openrewrite.maven.AddProfile:
id: native
activation: <activation><property><name>native</name></property></activation>
properties: <properties><skipITs>false</skipITs><quarkus.package.type>native</quarkus.package.type></properties>
- org.openrewrite.quarkus.ConfigureQuarkusMavenPluginWithReasonableDefaults
- org.openrewrite.maven.BestPractices

# Remove JavaEE dependencies
- org.openrewrite.java.dependencies.RemoveDependency:
groupId: javax*
artifactId: javaee-api
- org.openrewrite.java.dependencies.RemoveDependency:
groupId: javax*
artifactId: cdi-api
- org.openrewrite.java.dependencies.RemoveDependency:
groupId: javax*
artifactId: javax*

# Prep for Java 11 upgrade
- org.openrewrite.maven.AddProperty:
key: maven.compiler.source
value: 11
- org.openrewrite.maven.AddProperty:
key: maven.compiler.target
value: 11
timtebeek marked this conversation as resolved.
Show resolved Hide resolved
- org.openrewrite.maven.ChangePackaging:
groupId: '*'
artifactId: '*'
packaging: jar

---
type: specs.openrewrite.org/v1beta/recipe
name: org.openrewrite.quarkus.quarkus2.JavaEEtoQuarkus2CodeMigration
displayName: Migrate JavaEE Code to Quarkus 2
description: Migrate Standard JavaEE Code to Quarkus 2
recipeList:
# Convert some EJB annotations to CDI
- org.openrewrite.java.ChangeType:
oldFullyQualifiedTypeName: javax.ejb.Stateless
newFullyQualifiedTypeName: javax.enterprise.context.Dependent
- org.openrewrite.java.ChangeType:
oldFullyQualifiedTypeName: javax.ejb.Stateful
newFullyQualifiedTypeName: javax.enterprise.context.SessionScoped
- org.openrewrite.java.ChangeType:
oldFullyQualifiedTypeName: javax.ejb.Singleton
newFullyQualifiedTypeName: javax.enterprise.context.ApplicationScoped
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.ejb.EJB
attributeName: name
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.ejb.EJB
attributeName: description
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.ejb.EJB
attributeName: beanName
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.ejb.EJB
attributeName: beanInterface
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.ejb.EJB
attributeName: mappedName
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.ejb.EJB
attributeName: lookup
- org.openrewrite.java.ChangeType:
oldFullyQualifiedTypeName: javax.ejb.EJB
newFullyQualifiedTypeName: javax.inject.Inject
- org.openrewrite.java.RemoveAnnotation:
annotationPattern: '@javax.ejb.Local'

# Convert JPA annotations
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.persistence.PersistenceContext
attributeName: name
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.persistence.PersistenceContext
attributeName: unitName
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.persistence.PersistenceContext
attributeName: type
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.persistence.PersistenceContext
attributeName: synchronization
- org.openrewrite.java.RemoveAnnotationAttribute:
annotationType: javax.persistence.PersistenceContext
attributeName: properties
- org.openrewrite.java.ChangeType:
oldFullyQualifiedTypeName: javax.persistence.PersistenceContext
newFullyQualifiedTypeName: javax.inject.Inject

# Migrate to Java 11
- org.openrewrite.java.migrate.Java8toJava11
2 changes: 1 addition & 1 deletion src/main/resources/META-INF/rewrite/quarkus.yml
Original file line number Diff line number Diff line change
Expand Up @@ -140,4 +140,4 @@ recipeList:
- org.openrewrite.java.ChangePackage:
oldPackageName: io.vertx.core.http.HttpMethod
newPackageName: io.quarkus.vertx.web.Route.HttpMethod
recursive: false
recursive: false
Original file line number Diff line number Diff line change
@@ -0,0 +1,168 @@
/*
* Copyright 2024 the original author or authors.
* <p>
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* <p>
* https://www.apache.org/licenses/LICENSE-2.0
* <p>
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.openrewrite.quarkus.quarkus2;
timtebeek marked this conversation as resolved.
Show resolved Hide resolved

import org.junit.jupiter.api.Test;
import org.openrewrite.DocumentExample;
import org.openrewrite.java.JavaParser;
import org.openrewrite.test.RecipeSpec;
import org.openrewrite.test.RewriteTest;

import static org.openrewrite.java.Assertions.java;

class JavaEEtoQuarkus2CodeTranformationsTest implements RewriteTest {

@Override
public void defaults(RecipeSpec spec) {
spec.parser(JavaParser.fromJavaVersion()
.logCompilationWarningsAndErrors(true)
.classpath("javaee-api"))
.recipeFromResources("org.openrewrite.quarkus.quarkus2.JavaEEtoQuarkus2CodeMigration");
}

@Test
@DocumentExample
void javaEEtoQuarkus2CodeTransformationsTest() {
rewriteRun(
java(
//language=java
"""
package org.acme;

import javax.ejb.EJB;
import javax.ejb.Local;
import javax.ejb.SessionContext;
import javax.ejb.Singleton;
import javax.ejb.Stateful;
import javax.ejb.Stateless;

import javax.annotation.Resource;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;

@Stateless
public class PingEJBSLS {

@PersistenceContext
private EntityManager entityManager;

@Resource
private SessionContext context;

@EJB
private PingEJBLocal pingEJBLocal;

@EJB(lookup = "java:global/PingEJBSingleton")
private PingEJBSingleton pingEJBSingleton;

public String getMsg() {

return "PingEJBSLS: " + pingEJBLocal.getMsg() + " " + pingEJBSingleton.getMsg();
}

}

@Stateful
@Local
public class PingEJBLocal {

private static int hitCount;

public String getMsg() {

return "PingEJBLocal: " + hitCount++;
}

}

@Singleton
public class PingEJBSingleton {

private static int hitCount;

@PersistenceContext
private EntityManager entityManager;

public String getMsg() {

return "PingEJBSingleton: " + hitCount++;
}
}
""",
//language=java
"""
package org.acme;

import javax.ejb.SessionContext;
import javax.enterprise.context.ApplicationScoped;
import javax.enterprise.context.Dependent;
import javax.enterprise.context.SessionScoped;
import javax.inject.Inject;
import javax.annotation.Resource;
import javax.persistence.EntityManager;

@Dependent
public class PingEJBSLS {

@Inject
private EntityManager entityManager;

@Resource
private SessionContext context;

@Inject
private PingEJBLocal pingEJBLocal;

@Inject
private PingEJBSingleton pingEJBSingleton;

public String getMsg() {

return "PingEJBSLS: " + pingEJBLocal.getMsg() + " " + pingEJBSingleton.getMsg();
}

}

@SessionScoped
public class PingEJBLocal {

private static int hitCount;

public String getMsg() {

return "PingEJBLocal: " + hitCount++;
}

}

@ApplicationScoped
public class PingEJBSingleton {

private static int hitCount;

@Inject
private EntityManager entityManager;

public String getMsg() {

return "PingEJBSingleton: " + hitCount++;
}
}
"""
)
);
}
}
Loading