All Downloads are FREE. Search and download functionalities are using the official Maven repository.
Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.kie.kogito.codegen.rules.QueryEndpointGenerator Maven / Gradle / Ivy
/*
* Copyright 2019 Red Hat, Inc. and/or its affiliates.
*
* 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
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* 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.kie.kogito.codegen.rules;
import java.util.NoSuchElementException;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import com.github.javaparser.ast.CompilationUnit;
import com.github.javaparser.ast.Modifier;
import com.github.javaparser.ast.NodeList;
import com.github.javaparser.ast.body.ClassOrInterfaceDeclaration;
import com.github.javaparser.ast.body.ConstructorDeclaration;
import com.github.javaparser.ast.body.FieldDeclaration;
import com.github.javaparser.ast.body.MethodDeclaration;
import com.github.javaparser.ast.body.VariableDeclarator;
import com.github.javaparser.ast.expr.AssignExpr;
import com.github.javaparser.ast.expr.CastExpr;
import com.github.javaparser.ast.expr.MethodCallExpr;
import com.github.javaparser.ast.expr.NameExpr;
import com.github.javaparser.ast.expr.ObjectCreationExpr;
import com.github.javaparser.ast.expr.StringLiteralExpr;
import com.github.javaparser.ast.stmt.BlockStmt;
import com.github.javaparser.ast.stmt.ReturnStmt;
import com.github.javaparser.ast.stmt.Statement;
import com.github.javaparser.ast.type.Type;
import org.drools.modelcompiler.builder.QueryModel;
import org.drools.modelcompiler.builder.BodyDeclarationComparator;
import org.kie.kogito.codegen.FileGenerator;
import org.kie.kogito.codegen.di.DependencyInjectionAnnotator;
import static com.github.javaparser.StaticJavaParser.parse;
import static com.github.javaparser.StaticJavaParser.parseClassOrInterfaceType;
import static org.drools.core.util.StringUtils.ucFirst;
import static org.drools.modelcompiler.builder.generator.DrlxParseUtil.classToReferenceType;
import static org.drools.modelcompiler.util.ClassUtil.toNonPrimitiveType;
public class QueryEndpointGenerator implements FileGenerator {
private final Class ruleUnit;
private final QueryModel query;
private final DependencyInjectionAnnotator annotator;
private final String name;
private final String endpointName;
private final String targetCanonicalName;
private final String generatedFilePath;
public QueryEndpointGenerator(Class ruleUnit, QueryModel query, DependencyInjectionAnnotator annotator ) {
this.ruleUnit = ruleUnit;
this.query = query;
this.name = toCamelCase(query.getName());
this.endpointName = toKebabCase(name);
this.annotator = annotator;
this.targetCanonicalName = ruleUnit.getSimpleName() + "Query" + name + "Endpoint";
this.generatedFilePath = (query.getNamespace() + "." + targetCanonicalName).replace('.', '/') + ".java";
}
@Override
public String generatedFilePath() {
return generatedFilePath;
}
@Override
public String generate() {
CompilationUnit cu = parse(
this.getClass().getResourceAsStream("/class-templates/rules/RestQueryTemplate.java"));
cu.setPackageDeclaration(query.getNamespace());
ClassOrInterfaceDeclaration clazz = cu
.findFirst(ClassOrInterfaceDeclaration.class)
.orElseThrow(() -> new NoSuchElementException("Compilation unit doesn't contain a class or interface declaration!"));
clazz.setName( targetCanonicalName );
cu.findAll(StringLiteralExpr.class).forEach(this::interpolateStrings);
FieldDeclaration ruleUnitDeclaration = clazz
.getFieldByName( "ruleUnit" )
.orElseThrow(() -> new NoSuchElementException("ClassOrInterfaceDeclaration doesn't contain a field named ruleUnit!"));
setUnitGeneric( ruleUnitDeclaration.getElementType() );
if (annotator != null) {
annotator.withInjection( ruleUnitDeclaration );
}
String returnType = getReturnType(clazz);
generateConstructors( clazz );
generateQueryMethod( clazz, returnType );
clazz.getMembers().sort(new BodyDeclarationComparator());
return cu.toString();
}
private void generateConstructors( ClassOrInterfaceDeclaration clazz ) {
for (ConstructorDeclaration c : clazz.getConstructors()) {
c.setName( targetCanonicalName );
if (!c.getParameters().isEmpty()) {
setUnitGeneric( c.getParameter( 0 ).getType() );
}
}
}
private void generateQueryMethod( ClassOrInterfaceDeclaration clazz, String returnType ) {
MethodDeclaration queryMethod = clazz.getMethodsByName( "executeQuery" ).get(0);
queryMethod.getParameter( 0 ).setType(ruleUnit.getCanonicalName() + "DTO");
setGeneric(queryMethod.getType(), returnType);
Statement statement = queryMethod
.getBody()
.orElseThrow(() -> new NoSuchElementException("A method declaration doesn't contain a body!"))
.getStatement( 0 );
statement.findAll( VariableDeclarator.class ).forEach( decl -> setUnitGeneric( decl.getType() ) );
}
private String getReturnType( ClassOrInterfaceDeclaration clazz ) {
MethodDeclaration toResultMethod = clazz.getMethodsByName( "toResult" ).get(0);
String returnType;
if (query.getBindings().size() == 1) {
returnType = query.getBindings().values().iterator().next().getCanonicalName();
Statement statement = toResultMethod
.getBody()
.orElseThrow(() -> new NoSuchElementException("A method declaration doesn't contain a body!"))
.getStatement( 0 );
statement.findAll( CastExpr.class ).get(0).setType( returnType );
} else {
returnType = "Result";
generateResultClass(clazz, toResultMethod);
}
toResultMethod.setType( returnType );
return returnType;
}
private void generateResultClass( ClassOrInterfaceDeclaration clazz, MethodDeclaration toResultMethod ) {
ClassOrInterfaceDeclaration resultClass = new ClassOrInterfaceDeclaration( new NodeList(Modifier.publicModifier(), Modifier.staticModifier()), false, "Result" );
clazz.addMember( resultClass );
ConstructorDeclaration constructor = resultClass.addConstructor( Modifier.Keyword.PUBLIC );
BlockStmt constructorBody = constructor.createBody();
ObjectCreationExpr resultCreation = new ObjectCreationExpr();
resultCreation.setType( "Result" );
BlockStmt resultMethodBody = toResultMethod.createBody();
resultMethodBody.addStatement( new ReturnStmt( resultCreation ) );
query.getBindings().forEach( ( name, type) -> {
resultClass.addField( type, name, Modifier.Keyword.PRIVATE, Modifier.Keyword.FINAL );
MethodDeclaration getterMethod = resultClass.addMethod( "get" + ucFirst(name), Modifier.Keyword.PUBLIC );
getterMethod.setType( type );
BlockStmt body = getterMethod.createBody();
body.addStatement( new ReturnStmt( new NameExpr( name ) ) );
constructor.addAndGetParameter( type, name );
constructorBody.addStatement( new AssignExpr( new NameExpr( "this." + name ), new NameExpr( name ), AssignExpr.Operator.ASSIGN ) );
MethodCallExpr callExpr = new MethodCallExpr( new NameExpr( "tuple" ), "get" );
callExpr.addArgument( new StringLiteralExpr( name ) );
resultCreation.addArgument( new CastExpr( classToReferenceType( type ), callExpr ) );
} );
}
private void setUnitGeneric(Type type) {
setGeneric(type, ruleUnit);
}
private void setGeneric(Type type, Class typeArgument) {
type.asClassOrInterfaceType().setTypeArguments( classToReferenceType( typeArgument ) );
}
private void setGeneric(Type type, String typeArgument) {
type.asClassOrInterfaceType().setTypeArguments( parseClassOrInterfaceType( toNonPrimitiveType( typeArgument ) ) );
}
private void interpolateStrings(StringLiteralExpr vv) {
String interpolated = vv.getValue()
.replace("$name$", name)
.replace("$endpointName$", endpointName)
.replace("$queryName$", query.getName());
vv.setString(interpolated);
}
private static String toCamelCase(String inputString) {
return Stream.of(inputString.split(" "))
.map( s -> s.length() > 1 ? s.substring( 0, 1 ).toUpperCase() + s.substring( 1 ) : s.substring( 0, 1 ).toUpperCase() )
.collect( Collectors.joining() );
}
private static String toKebabCase(String inputString) {
return inputString.replaceAll("(.)(\\p{Upper})", "$1-$2").toLowerCase();
}
}