Skip to content

Commit

Permalink
Qute: fix generated ValueResolvers
Browse files Browse the repository at this point in the history
- fixes quarkusio#43489
- consider default methods in class hierarchy; previously, interface default methods were only considered if a
ValueResolver was generated for an interface
- also consider inherited fields in a class hierarchy
  • Loading branch information
mkouba committed Sep 30, 2024
1 parent 96d8967 commit 483009a
Show file tree
Hide file tree
Showing 13 changed files with 212 additions and 33 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ public class DefaultMethodValidationSuccessTest {
@RegisterExtension
static final QuarkusUnitTest config = new QuarkusUnitTest()
.withApplicationRoot((jar) -> jar
.addClasses(Movie.class, MovieExtensions.class)
.addClasses(Name.class, Something.class)
.addAsResource(new StringAsset(
"{@io.quarkus.qute.deployment.typesafe.DefaultMethodValidationSuccessTest$Name name}Hello {name.fullName()}::{name.fullName}!"),
"templates/name.html"));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -266,41 +266,59 @@ private boolean implementResolve(ClassCreator valueResolver, String clazzName, C
ResultHandle paramsCount = resolve.invokeInterfaceMethod(Descriptors.COLLECTION_SIZE, params);
Function<FieldInfo, String> fieldToGetterFun = forceGettersFunction != null ? forceGettersFunction.apply(clazz) : null;

// First collect and sort methods (getters must come before is/has properties, etc.)
List<MethodKey> methods = new ArrayList<>();
for (MethodInfo method : clazz.methods()) {
if (filter.test(method)) {
methods.add(new MethodKey(method));
// First collect methods and fields from the class hierarchy
Set<MethodKey> methods = new HashSet<>();
List<FieldInfo> fields = new ArrayList<>();
ClassInfo target = clazz;
while (target != null) {
for (MethodInfo method : target.methods()) {
if (filter.test(method)) {
methods.add(new MethodKey(method));
}
}
}
methods.sort(null);

if (!ignoreSuperclasses && !clazz.isEnum()) {
DotName superName = clazz.superName();
while (superName != null && !superName.equals(DotNames.OBJECT)) {
ClassInfo superClass = index.getClassByName(superName);
if (superClass != null) {
for (MethodInfo method : superClass.methods()) {
if (filter.test(method)) {
methods.add(new MethodKey(method));
}
}
superName = superClass.superName();
} else {
superName = null;
LOGGER.warnf("Skipping super class %s - not found in the index", clazz.superClassType());
for (FieldInfo field : target.fields()) {
if (filter.test(field)) {
fields.add(field);
}
}
DotName superName = target.superName();
if (ignoreSuperclasses || target.isEnum() || superName == null || superName.equals(DotNames.OBJECT)) {
target = null;
} else {
target = index.getClassByName(superName);
if (target == null) {
LOGGER.warnf("Skipping super class %s - not found in the index", superName);
}
}
}

List<FieldInfo> fields = new ArrayList<>();
for (FieldInfo field : clazz.fields()) {
if (filter.test(field)) {
fields.add(field);
// Find non-implemented default interface methods
target = clazz;
while (target != null) {
for (DotName interfaceName : target.interfaceNames()) {
ClassInfo interfaceClass = index.getClassByName(interfaceName);
if (interfaceClass == null) {
LOGGER.warnf("Skipping implemented interface %s - not found in the index", interfaceName);
continue;
}
for (MethodInfo method : interfaceClass.methods()) {
if (method.isDefault() && filter.test(method)) {
methods.add(new MethodKey(method));
}
}
}
DotName superName = target.superName();
if (ignoreSuperclasses || target.isEnum() || superName == null || superName.equals(DotNames.OBJECT)) {
target = null;
} else {
target = index.getClassByName(superName);
}
}

if (methods.isEmpty() && fields.isEmpty()) {
// Sort methods, getters must come before is/has properties, etc.
List<MethodKey> sortedMethods = methods.stream().sorted().toList();

if (sortedMethods.isEmpty() && fields.isEmpty()) {
// No members
return false;
}
Expand All @@ -311,7 +329,7 @@ private boolean implementResolve(ClassCreator valueResolver, String clazzName, C
Map<Match, List<MethodInfo>> matches = new HashMap<>();
Map<Match, List<MethodInfo>> varargsMatches = new HashMap<>();

for (MethodKey methodKey : methods) {
for (MethodKey methodKey : sortedMethods) {
MethodInfo method = methodKey.method;
if (method.parametersCount() == 0) {
noParamMethods.add(methodKey);
Expand Down Expand Up @@ -365,7 +383,7 @@ private boolean implementResolve(ClassCreator valueResolver, String clazzName, C
public void accept(BytecodeCreator bc) {
Type returnType = method.returnType();
ResultHandle invokeRet;
if (Modifier.isInterface(clazz.flags())) {
if (Modifier.isInterface(method.declaringClass().flags())) {
invokeRet = bc.invokeInterfaceMethod(MethodDescriptor.of(method), base);
} else {
invokeRet = bc.invokeVirtualMethod(MethodDescriptor.of(method), base);
Expand All @@ -378,7 +396,7 @@ public void accept(BytecodeCreator bc) {

for (FieldInfo field : fields) {
String getterName = fieldToGetterFun != null ? fieldToGetterFun.apply(field) : null;
if (getterName != null && noneMethodMatches(methods, getterName) && matchedNames.add(getterName)) {
if (getterName != null && noneMethodMatches(sortedMethods, getterName) && matchedNames.add(getterName)) {
LOGGER.debugf("Forced getter added: %s", field);
List<String> matching;
if (matchedNames.add(field.name())) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -170,7 +170,7 @@ public void testArrays() {
assertEquals("true,false,false,false,false,", engine.parse("{#for i in 5}{i_isFirst},{/for}").render());
}

private Resolver newResolver(String className)
public static Resolver newResolver(String className)
throws ClassNotFoundException, InstantiationException, IllegalAccessException, IllegalArgumentException,
InvocationTargetException, NoSuchMethodException, SecurityException {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
Expand All @@ -181,7 +181,7 @@ private Resolver newResolver(String className)
return (Resolver) clazz.getDeclaredConstructor().newInstance();
}

static Index index(Class<?>... classes) throws IOException {
public static Index index(Class<?>... classes) throws IOException {
Indexer indexer = new Indexer();
for (Class<?> clazz : classes) {
try (InputStream stream = SimpleGeneratorTest.class.getClassLoader()
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package io.quarkus.qute.generator.hierarchy;

public interface FirstLevel {

default int firstLevel() {
return 1;
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
package io.quarkus.qute.generator.hierarchy;

import static org.junit.jupiter.api.Assertions.assertEquals;

import java.io.IOException;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;

import org.jboss.jandex.DotName;
import org.jboss.jandex.Index;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;

import io.quarkus.qute.Engine;
import io.quarkus.qute.EngineBuilder;
import io.quarkus.qute.ValueResolver;
import io.quarkus.qute.generator.SimpleGeneratorTest;
import io.quarkus.qute.generator.TestClassOutput;
import io.quarkus.qute.generator.ValueResolverGenerator;

public class HierarchyTest {

static Set<String> generatedTypes = new HashSet<>();

@BeforeAll
public static void init() throws IOException {
TestClassOutput classOutput = new TestClassOutput();
Index index = SimpleGeneratorTest.index(Level1.class, Level2.class, Level3.class, Level4.class, FirstLevel.class,
SecondLevel.class);
ValueResolverGenerator generator = ValueResolverGenerator.builder().setIndex(index).setClassOutput(classOutput)
.addClass(index.getClassByName(DotName.createSimple(Level4.class.getName())))
.build();

generator.generate();
generatedTypes.addAll(generator.getGeneratedTypes());
}

@Test
public void testHierarchy() throws Exception {
EngineBuilder builder = Engine.builder().addDefaults();
for (String generatedType : generatedTypes) {
builder.addValueResolver((ValueResolver) SimpleGeneratorTest.newResolver(generatedType));
}
Engine engine = builder.build();

Level4 level4 = new Level4();
assertEquals(1, level4.getLevel1());
assertEquals(1, level4.firstLevel());
assertEquals(2, level4.secondLevel());
assertEquals(4, level4.getLevel4());
assertEquals(4, level4.overridenLevel);

assertEquals("1::1::2::4::4",
engine.parse(
"{level.level1}::{level.firstLevel}::{level.secondLevel}::{level.getLevel4}::{level.overridenLevel}")
.render(Map.of("level", level4)));
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
package io.quarkus.qute.generator.hierarchy;

public class Level1 implements FirstLevel {

public int firstLevel = 1;

public int getLevel1() {
return 1;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
package io.quarkus.qute.generator.hierarchy;

public class Level2 extends Level1 implements SecondLevel {

public int getLevel2() {
return 2;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package io.quarkus.qute.generator.hierarchy;

public class Level3 extends Level2 {

public int overridenLevel = 3;

public int getLevel3() {
return 3;
}

// This method should be overriden
public int getLevel4() {
return 34;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
package io.quarkus.qute.generator.hierarchy;

public class Level4 extends Level3 {

public int overridenLevel = 4;

public int getLevel4() {
return 4;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package io.quarkus.qute.generator.hierarchy;

public interface SecondLevel {

default int secondLevel() {
return 2;
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
package io.quarkus.it.qute;

import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;

import io.quarkus.qute.TemplateInstance;

@Path("/defaultmethod")
public class DefaultMethodResource {

@GET
@Produces(MediaType.TEXT_PLAIN)
public TemplateInstance get() {
return new name(new Name());
}

record name(Name name) implements TemplateInstance {
};

public static class Name implements Something {

public String name() {
return "M";
}
}

public interface Something {

String name();

default String fullName() {
return name() + "K";
}
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Hello {name.fullName}!
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ public void testTemplates() throws InterruptedException {
.contentType(is(ContentType.HTML.toString()))
.body(containsString("Hello Ciri!"));
RestAssured.when().get("/beer").then().body(containsString("Beer Pilsner, completed: true, done: true"));
RestAssured.when().get("/defaultmethod").then().body(containsString("Hello MK!"));

Check failure on line 30 in integration-tests/qute/src/test/java/io/quarkus/it/qute/QuteTestCase.java

View workflow job for this annotation

GitHub Actions / Build summary for 483009a8a91ca53e90deb16e4e266981d18e3bc8

JVM Tests - JDK 17 Windows

java.lang.AssertionError: 1 expectation failed. Response body doesn't match expectation.
Raw output
java.lang.AssertionError: 
1 expectation failed.
Response body doesn't match expectation.
Expected: a string containing "Hello MK!"
  Actual: {"details":"Error id 819c6141-edf5-4b9d-85e9-c46fe6c50f2c-1, java.lang.NoClassDefFoundError: io/quarkus/it/qute/DefaultMethodResource$Name (wrong name: io/quarkus/it/qute/DefaultMethodResource$name)","stack":"java.lang.NoClassDefFoundError: io/quarkus/it/qute/DefaultMethodResource$Name (wrong name: io/quarkus/it/qute/DefaultMethodResource$name)\r\n\tat java.base/java.lang.ClassLoader.defineClass1(Native Method)\r\n\tat java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1017)\r\n\tat io.quarkus.bootstrap.classloading.QuarkusClassLoader.loadClass(QuarkusClassLoader.java:531)\r\n\tat io.quarkus.bootstrap.classloading.QuarkusClassLoader.loadClass(QuarkusClassLoader.java:490)\r\n\tat io.quarkus.it.qute.DefaultMethodResource.get(DefaultMethodResource.java:16)\r\n\tat io.quarkus.it.qute.DefaultMethodResource$quarkusrestinvoker$get_e7c49ae2ccf8df6f97f84d062655557ffcb3616d.invoke(Unknown Source)\r\n\tat org.jboss.resteasy.reactive.server.handlers.InvocationHandler.handle(InvocationHandler.java:29)\r\n\tat io.quarkus.resteasy.reactive.server.runtime.QuarkusResteasyReactiveRequestContext.invokeHandler(QuarkusResteasyReactiveRequestContext.java:141)\r\n\tat org.jboss.resteasy.reactive.common.core.AbstractResteasyReactiveContext.run(AbstractResteasyReactiveContext.java:147)\r\n\tat io.quarkus.vertx.core.runtime.VertxCoreRecorder$14.runWith(VertxCoreRecorder.java:627)\r\n\tat org.jboss.threads.EnhancedQueueExecutor$Task.doRunWith(EnhancedQueueExecutor.java:2516)\r\n\tat org.jboss.threads.EnhancedQueueExecutor$Task.run(EnhancedQueueExecutor.java:2495)\r\n\tat org.jboss.threads.EnhancedQueueExecutor$ThreadBody.run(EnhancedQueueExecutor.java:1521)\r\n\tat org.jboss.threads.DelegatingRunnable.run(DelegatingRunnable.java:11)\r\n\tat org.jboss.threads.ThreadLocalResettingRunnable.run(ThreadLocalResettingRunnable.java:11)\r\n\tat io.netty.util.concurrent.FastThreadLocalRunnable.run(FastThreadLocalRunnable.java:30)\r\n\tat java.base/java.lang.Thread.run(Thread.java:840)"}

	at java.base/jdk.internal.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
	at java.base/jdk.internal.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:77)
	at java.base/jdk.internal.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
	at java.base/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:500)
	at java.base/java.lang.reflect.Constructor.newInstance(Constructor.java:481)
	at org.codehaus.groovy.reflection.CachedConstructor.invoke(CachedConstructor.java:73)
	at org.codehaus.groovy.runtime.callsite.ConstructorSite$ConstructorSiteNoUnwrapNoCoerce.callConstructor(ConstructorSite.java:108)
	at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCallConstructor(CallSiteArray.java:57)
	at org.codehaus.groovy.runtime.callsite.AbstractCallSite.callConstructor(AbstractCallSite.java:263)
	at org.codehaus.groovy.runtime.callsite.AbstractCallSite.callConstructor(AbstractCallSite.java:277)
	at io.restassured.internal.ResponseSpecificationImpl$HamcrestAssertionClosure.validate(ResponseSpecificationImpl.groovy:512)
	at io.restassured.internal.ResponseSpecificationImpl$HamcrestAssertionClosure$validate$1.call(Unknown Source)
	at io.restassured.internal.ResponseSpecificationImpl.validateResponseIfRequired(ResponseSpecificationImpl.groovy:696)
	at io.restassured.internal.ResponseSpecificationImpl.this$2$validateResponseIfRequired(ResponseSpecificationImpl.groovy)
	at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
	at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:77)
	at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
	at java.base/java.lang.reflect.Method.invoke(Method.java:569)
	at org.codehaus.groovy.runtime.callsite.PlainObjectMetaMethodSite.doInvoke(PlainObjectMetaMethodSite.java:43)
	at org.codehaus.groovy.runtime.callsite.PogoMetaMethodSite$PogoCachedMethodSiteNoUnwrapNoCoerce.invoke(PogoMetaMethodSite.java:198)
	at org.codehaus.groovy.runtime.callsite.PogoMetaMethodSite.callCurrent(PogoMetaMethodSite.java:62)
	at org.codehaus.groovy.runtime.callsite.AbstractCallSite.callCurrent(AbstractCallSite.java:185)
	at io.restassured.internal.ResponseSpecificationImpl.body(ResponseSpecificationImpl.groovy:107)
	at io.restassured.internal.ValidatableResponseOptionsImpl.body(ValidatableResponseOptionsImpl.java:238)
	at io.quarkus.it.qute.QuteTestCase.testTemplates(QuteTestCase.java:30)
	at java.base/java.lang.reflect.Method.invoke(Method.java:569)
	at io.quarkus.test.junit.QuarkusTestExtension.runExtensionMethod(QuarkusTestExtension.java:971)
	at io.quarkus.test.junit.QuarkusTestExtension.interceptTestMethod(QuarkusTestExtension.java:821)
	at java.base/java.util.ArrayList.forEach(ArrayList.java:1511)
	at java.base/java.util.ArrayList.forEach(ArrayList.java:1511)
	Suppressed: java.lang.NoClassDefFoundError: io/quarkus/it/qute/DefaultMethodResource$Name (wrong name: io/quarkus/it/qute/DefaultMethodResource$name)
		at java.base/java.lang.ClassLoader.defineClass1(Native Method)
		at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1017)
		at io.quarkus.bootstrap.classloading.QuarkusClassLoader.loadClass(QuarkusClassLoader.java:531)
		at io.quarkus.bootstrap.classloading.QuarkusClassLoader.loadClass(QuarkusClassLoader.java:490)
		at io.quarkus.it.qute.DefaultMethodResource.get(DefaultMethodResource.java:16)
		at io.quarkus.it.qute.DefaultMethodResource$quarkusrestinvoker$get_e7c49ae2ccf8df6f97f84d062655557ffcb3616d.invoke(Unknown Source)
		at org.jboss.resteasy.reactive.server.handlers.InvocationHandler.handle(InvocationHandler.java:29)
		at io.quarkus.resteasy.reactive.server.runtime.QuarkusResteasyReactiveRequestContext.invokeHandler(QuarkusResteasyReactiveRequestContext.java:141)
		at org.jboss.resteasy.reactive.common.core.AbstractResteasyReactiveContext.run(AbstractResteasyReactiveContext.java:147)
		at io.quarkus.vertx.core.runtime.VertxCoreRecorder$14.runWith(VertxCoreRecorder.java:627)
		at org.jboss.threads.EnhancedQueueExecutor$Task.doRunWith(EnhancedQueueExecutor.java:2516)
		at org.jboss.threads.EnhancedQueueExecutor$Task.run(EnhancedQueueExecutor.java:2495)
		at org.jboss.threads.EnhancedQueueExecutor$ThreadBody.run(EnhancedQueueExecutor.java:1521)
		at org.jboss.threads.DelegatingRunnable.run(DelegatingRunnable.java:11)
		at org.jboss.threads.ThreadLocalResettingRunnable.run(ThreadLocalResettingRunnable.java:11)
		at io.netty.util.concurrent.FastThreadLocalRunnable.run(FastThreadLocalRunnable.java:30)
		at java.base/java.lang.Thread.run(Thread.java:840)
}

@Test
Expand Down

0 comments on commit 483009a

Please sign in to comment.