Skip to content

Commit

Permalink
Migrate UserType implementations to hibernate 6 (#40)
Browse files Browse the repository at this point in the history
* WIP

* WIP

* WIP

* WIP

* WIP

* WIP

* Apply suggestions from code review

Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>

* New line at end of file

* imports

* No need to add hibernate-core-5 to src/main

* Apply suggestions from code review

Co-authored-by: Tim te Beek <tim@moderne.io>

* Get rid of static mutable field
Order methodmatchers alphabetically

* Reduced cycles to 1

* Remove debug code

* Organize imports

* Apply FindImplementations

* Extract returnedClass with visitor instead of streams API

* Reduce warnings

* Reduce unnecessary nesting and matching

* Demonstrate sensitivity to variable names

* Use parameter index instead of name

* Simplify conditionals as parameterized type is never null

* Pass along parameterizedType

* Drop JetBrains annotation for JSpecify

---------

Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
Co-authored-by: Tim te Beek <tim@moderne.io>
  • Loading branch information
3 people authored Oct 8, 2024
1 parent d7c9403 commit 1b924b0
Show file tree
Hide file tree
Showing 6 changed files with 453 additions and 4 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,7 @@ public J.Annotation visitAnnotation(J.Annotation annotation, ExecutionContext ct
String converterFQN = String.format("org.hibernate.type.%s", converterName);

ann = JavaTemplate.builder(String.format("@Convert(converter = %s.class)", converterName))
.javaParser(JavaParser.fromJavaVersion().classpathFromResources(ctx, "hibernate-core", "jakarta.persistence-api"))
.javaParser(JavaParser.fromJavaVersion().classpathFromResources(ctx, "hibernate-core-6+", "jakarta.persistence-api"))
.imports(converterFQN, "jakarta.persistence.Convert")
.contextSensitive()
.build().apply(getCursor(), ann.getCoordinates().replace());
Expand Down
251 changes: 251 additions & 0 deletions src/main/java/org/openrewrite/hibernate/MigrateUserType.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,251 @@
/*
* 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.hibernate;

import org.jspecify.annotations.Nullable;
import org.openrewrite.ExecutionContext;
import org.openrewrite.Preconditions;
import org.openrewrite.Recipe;
import org.openrewrite.TreeVisitor;
import org.openrewrite.internal.ListUtils;
import org.openrewrite.java.*;
import org.openrewrite.java.search.FindImplementations;
import org.openrewrite.java.search.FindMethodDeclaration;
import org.openrewrite.java.tree.*;
import org.openrewrite.marker.Markers;

import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicReference;

import static org.openrewrite.Tree.randomId;

public class MigrateUserType extends Recipe {

private static final String USER_TYPE = "org.hibernate.usertype.UserType";
private static final MethodMatcher ASSEMBLE = new MethodMatcher("* assemble(java.io.Serializable, java.lang.Object)");
private static final MethodMatcher DEEP_COPY = new MethodMatcher("* deepCopy(java.lang.Object)");
private static final MethodMatcher DISASSEMBLE = new MethodMatcher("* disassemble(java.lang.Object)");
private static final MethodMatcher EQUALS = new MethodMatcher("* equals(java.lang.Object, java.lang.Object)");
private static final MethodMatcher HASHCODE = new MethodMatcher("* hashCode(java.lang.Object)");
private static final MethodMatcher NULL_SAFE_GET_STRING_ARRAY = new MethodMatcher("* nullSafeGet(java.sql.ResultSet, java.lang.String[], org.hibernate.engine.spi.SharedSessionContractImplementor, java.lang.Object)");
private static final MethodMatcher NULL_SAFE_SET = new MethodMatcher("* nullSafeSet(java.sql.PreparedStatement, java.lang.Object, int, org.hibernate.engine.spi.SharedSessionContractImplementor)");
private static final MethodMatcher NULL_SAFE_GET_INT = new MethodMatcher("* nullSafeGet(java.sql.ResultSet, int, org.hibernate.engine.spi.SharedSessionContractImplementor, java.lang.Object)");
private static final MethodMatcher REPLACE = new MethodMatcher("* replace(java.lang.Object, java.lang.Object, java.lang.Object)");
private static final MethodMatcher RESULT_SET_STRING_PARAM = new MethodMatcher("java.sql.ResultSet *(java.lang.String)");
private static final MethodMatcher RETURNED_CLASS = new MethodMatcher("* returnedClass()");
private static final MethodMatcher SQL_TYPES = new MethodMatcher("* sqlTypes()");

@Override
public String getDisplayName() {
return "Migrate `UserType` to Hibernate 6";
}

@Override
public String getDescription() {
return "With Hibernate 6 the `UserType` interface received a type parameter making it more strictly typed. " +
"This recipe applies the changes required to adhere to this change.";
}

@Override
public TreeVisitor<?, ExecutionContext> getVisitor() {
return Preconditions.check(Preconditions.and(
new FindImplementations(USER_TYPE).getVisitor(),
// This method only exists on the Hibernate 6 variant of UserType, so as a precondition this shouldn't exist
Preconditions.not(new FindMethodDeclaration("* getSqlType()", true).getVisitor())
), new JavaVisitor<ExecutionContext>() {
@Override
public J visitClassDeclaration(J.ClassDeclaration classDecl, ExecutionContext ctx) {
J.ClassDeclaration cd = classDecl;
J.FieldAccess parameterizedType = getReturnedClass(cd);
cd = cd.withImplements(ListUtils.map(cd.getImplements(), impl -> {
if (TypeUtils.isAssignableTo(USER_TYPE, impl.getType()) && parameterizedType != null) {
return TypeTree.build("UserType<" + parameterizedType.getTarget() + ">").withType(JavaType.buildType(USER_TYPE)).withPrefix(Space.SINGLE_SPACE);
}
return impl;
}));
if (parameterizedType != null) {
getCursor().putMessage("parameterizedType", parameterizedType);
}
return super.visitClassDeclaration(cd, ctx);
}

@SuppressWarnings("ConstantConditions")
private J.@Nullable FieldAccess getReturnedClass(J.ClassDeclaration cd) {
AtomicReference<J.FieldAccess> reference = new AtomicReference<>();
new JavaIsoVisitor<AtomicReference<J.FieldAccess>>() {
@Override
public J.MethodDeclaration visitMethodDeclaration(J.MethodDeclaration method, AtomicReference<J.FieldAccess> ref) {
// Only visit top level method returnedClass
return RETURNED_CLASS.matches(method, cd) ? super.visitMethodDeclaration(method, ref) : method;
}

@Override
public J.Return visitReturn(J.Return _return, AtomicReference<J.FieldAccess> ref) {
ref.set((J.FieldAccess) _return.getExpression());
return _return;
}
}.visitNonNull(cd, reference);
return reference.get();
}

@Override
public J visitMethodDeclaration(J.MethodDeclaration method, ExecutionContext ctx) {
J.MethodDeclaration md = method;
J.ClassDeclaration cd = getCursor().firstEnclosing(J.ClassDeclaration.class);
J.FieldAccess parameterizedType = getCursor().getNearestMessage("parameterizedType");
if (cd == null || parameterizedType == null) {
return md;
}
if (SQL_TYPES.matches(md, cd)) {
if (md.getBody() != null) {
Optional<J.Return> ret = md.getBody().getStatements().stream().filter(J.Return.class::isInstance).map(J.Return.class::cast).findFirst();
if (ret.isPresent()) {
if (ret.get().getExpression() instanceof J.NewArray) {
J.NewArray newArray = (J.NewArray) ret.get().getExpression();
if (newArray.getInitializer() != null) {
String template = "@Override\n" +
"public int getSqlType() {\n" +
" return #{any()};\n" +
"}";
md = JavaTemplate.builder(template)
.javaParser(JavaParser.fromJavaVersion())
.build()
.apply(getCursor(), md.getCoordinates().replace(), newArray.getInitializer().get(0)).withId(md.getId());
}
}

}
}
} else if (RETURNED_CLASS.matches(md, cd)) {
md = md.withReturnTypeExpression(TypeTree.build("Class<" + parameterizedType.getTarget() + ">"));
if (md.getReturnTypeExpression() != null) {
md = md.withPrefix(md.getReturnTypeExpression().getPrefix());
}
} else if (EQUALS.matches(md, cd)) {
md = changeParameterTypes(md, Arrays.asList(0, 1), parameterizedType);
} else if (HASHCODE.matches(md, cd)) {
md = changeParameterTypes(md, Collections.singletonList(0), parameterizedType);
} else if (NULL_SAFE_GET_STRING_ARRAY.matches(md, cd)) {
String template = "@Override\n" +
"public BigDecimal nullSafeGet(ResultSet rs, int position, SharedSessionContractImplementor session, Object owner) throws SQLException {\n" +
"}";
J.MethodDeclaration updatedParam = JavaTemplate.builder(template)
.javaParser(JavaParser.fromJavaVersion().classpathFromResources(ctx, "hibernate-core"))
.imports("java.math.BigDecimal", "java.sql.ResultSet", "java.sql.SQLException", "org.hibernate.engine.spi.SharedSessionContractImplementor")
.build()
.apply(getCursor(), md.getCoordinates().replace());
md = updatedParam.withId(md.getId()).withBody(md.getBody());
} else if (NULL_SAFE_SET.matches(md, cd)) {
md = changeParameterTypes(md, Collections.singletonList(1), parameterizedType);
} else if (DEEP_COPY.matches(md, cd)) {
md = md.withReturnTypeExpression(parameterizedType.getTarget().withPrefix(Space.SINGLE_SPACE));
if (md.getReturnTypeExpression() != null) {
md = md.withPrefix(md.getReturnTypeExpression().getPrefix());
}
md = changeParameterTypes(md, Collections.singletonList(0), parameterizedType);
} else if (DISASSEMBLE.matches(md, cd)) {
md = changeParameterTypes(md, Collections.singletonList(0), parameterizedType);
if (md.getBody() != null) {
md = md.withBody(md.getBody().withStatements(ListUtils.map(md.getBody().getStatements(), stmt -> {
if (stmt instanceof J.Return) {
J.Return r = (J.Return) stmt;
if (r.getExpression() instanceof J.TypeCast) {
J.TypeCast tc = (J.TypeCast) r.getExpression();
if (TypeUtils.isOfType(parameterizedType.getTarget().getType(), tc.getClazz().getType())) {
return r.withExpression(tc.getExpression());
}
}
}
return stmt;
})));
}
} else if (ASSEMBLE.matches(md, cd)) {
md = md.withReturnTypeExpression(parameterizedType.getTarget().withPrefix(Space.SINGLE_SPACE));
if (md.getReturnTypeExpression() != null) {
md = md.withPrefix(md.getReturnTypeExpression().getPrefix());
}
if (md.getBody() != null) {
md = md.withBody(md.getBody().withStatements(ListUtils.map(md.getBody().getStatements(), stmt -> {
if (stmt instanceof J.Return) {
J.Return r = (J.Return) stmt;
if (r.getExpression() != null && !TypeUtils.isOfType(parameterizedType.getTarget().getType(), r.getExpression().getType())) {
return r.withExpression(new J.TypeCast(randomId(), Space.EMPTY, Markers.EMPTY, new J.ControlParentheses<>(randomId(), Space.EMPTY, Markers.EMPTY,
new JRightPadded<>(TypeTree.build("BigDecimal").withType(parameterizedType.getTarget().getType()), Space.EMPTY, Markers.EMPTY)), r.getExpression()));
}
}
return stmt;
})));
}
} else if (REPLACE.matches(md, cd)) {
md = md.withReturnTypeExpression(parameterizedType.getTarget().withPrefix(Space.SINGLE_SPACE));
if (md.getReturnTypeExpression() != null) {
md = md.withPrefix(md.getReturnTypeExpression().getPrefix());
}
md = changeParameterTypes(md, Arrays.asList(0, 1), parameterizedType);
}
updateCursor(md);
md = (J.MethodDeclaration) super.visitMethodDeclaration(md, ctx);
return maybeAutoFormat(method, md, ctx);
}

private J.MethodDeclaration changeParameterTypes(J.MethodDeclaration md, List<Integer> paramIndexes, J.FieldAccess parameterizedType) {
if (md.getMethodType() != null) {
JavaType.Method met = md.getMethodType().withParameterTypes(ListUtils.map(md.getMethodType().getParameterTypes(),
(index, type) -> {
if (paramIndexes.contains(index)) {
type = TypeUtils.isOfType(JavaType.buildType("java.lang.Object"), type) ? parameterizedType.getTarget().getType() : type;
}
return type;
}));
return md.withParameters(ListUtils.map(md.getParameters(), (index, param) -> {
if (param instanceof J.VariableDeclarations && paramIndexes.contains(index)) {
param = ((J.VariableDeclarations) param)
.withType(parameterizedType.getTarget().getType()).withTypeExpression((TypeTree) parameterizedType.getTarget())
.withVariables(ListUtils.map(((J.VariableDeclarations) param).getVariables(), var -> {
var = var.withType(parameterizedType.getTarget().getType());
if (var.getVariableType() != null && parameterizedType.getTarget().getType() != null) {
var = var.withVariableType(var.getVariableType().withType(parameterizedType.getTarget().getType()).withOwner(met));
}
return var;
}));
}
return param;
}));
}
return md;
}

@Override
public J visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) {
J.MethodInvocation mi = (J.MethodInvocation) super.visitMethodInvocation(method, ctx);
if (RESULT_SET_STRING_PARAM.matches(mi)) {
J.MethodDeclaration md = getCursor().firstEnclosing(J.MethodDeclaration.class);
J.ClassDeclaration cd = getCursor().firstEnclosing(J.ClassDeclaration.class);
if (md != null && cd != null && NULL_SAFE_GET_INT.matches(md, cd)) {
mi = mi.withArguments(Collections.singletonList(((J.VariableDeclarations) md.getParameters().get(1)).getVariables().get(0).getName()));
if (mi.getMethodType() != null) {
mi = mi.withMethodType(mi.getMethodType().withParameterTypes(Collections.singletonList(JavaType.buildType("int"))));
}
}
}
return mi;
}
});
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ class MigrateBooleanMappingsTest implements RewriteTest {
public void defaults(RecipeSpec spec) {
spec.recipe(new MigrateBooleanMappings())
.parser(JavaParser.fromJavaVersion()
.classpathFromResources(new InMemoryExecutionContext(), "hibernate-core", "jakarta.persistence-api")
.classpathFromResources(new InMemoryExecutionContext(), "hibernate-core-6+", "jakarta.persistence-api")
);
}

Expand Down
Loading

0 comments on commit 1b924b0

Please sign in to comment.