com.abubusoft.kripton.processor.sqlite.SelectPaginatedResultHelper Maven / Gradle / Ivy
/*******************************************************************************
* Copyright 2015, 2017 Francesco Benincasa ([email protected]).
*
* 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 com.abubusoft.kripton.processor.sqlite;
import static com.abubusoft.kripton.processor.core.reflect.TypeUtility.typeName;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import javax.lang.model.element.Modifier;
import com.abubusoft.kripton.android.Logger;
import com.abubusoft.kripton.android.PageRequest;
import com.abubusoft.kripton.android.annotation.BindSqlSelect;
import com.abubusoft.kripton.android.sqlite.PagedResultImpl;
import com.abubusoft.kripton.common.Pair;
import com.abubusoft.kripton.common.SQLTypeAdapterUtils;
import com.abubusoft.kripton.processor.core.AnnotationAttributeType;
import com.abubusoft.kripton.processor.core.ImmutableUtility;
import com.abubusoft.kripton.processor.core.ModelAnnotation;
import com.abubusoft.kripton.processor.core.reflect.TypeUtility;
import com.abubusoft.kripton.processor.sqlite.grammars.jql.JQLChecker;
import com.abubusoft.kripton.processor.sqlite.grammars.jql.JQLProjection;
import com.abubusoft.kripton.processor.sqlite.model.SQLProperty;
import com.abubusoft.kripton.processor.sqlite.model.SQLiteDaoDefinition;
import com.abubusoft.kripton.processor.sqlite.model.SQLiteEntity;
import com.abubusoft.kripton.processor.sqlite.model.SQLiteModelMethod;
import com.abubusoft.kripton.processor.sqlite.transform.SQLTransformer;
import com.squareup.javapoet.ClassName;
import com.squareup.javapoet.MethodSpec;
import com.squareup.javapoet.MethodSpec.Builder;
import com.squareup.javapoet.ParameterSpec;
import com.squareup.javapoet.ParameterizedTypeName;
import com.squareup.javapoet.TypeName;
import com.squareup.javapoet.TypeSpec;
/**
* The Class SelectPaginatedResultHelper.
*
* @author Francesco Benincasa ([email protected])
* @param
* the generic type
* @since 29/01/2017
*/
public class SelectPaginatedResultHelper extends AbstractSelectCodeGenerator {
public static final String WITH_PAGE_REQUEST_PREFIX = "WithPageRequest";
/*
* (non-Javadoc)
*
* @see com.abubusoft.kripton.processor.sqlite.AbstractSelectCodeGenerator#
* generate(com.squareup.javapoet.TypeSpec.Builder, boolean,
* com.abubusoft.kripton.processor.sqlite.model.SQLiteModelMethod)
*/
@Override
public void generate(TypeSpec.Builder classBuilder, boolean mapFields, SQLiteModelMethod method) {
// SQLiteDaoDefinition daoDefinition = method.getParent();
String pagedResultName = buildSpecializedPagedResultClass(classBuilder, method);
Set fieldList = JQLChecker.getInstance().extractProjections(method, method.jql.value, method.getEntity());
{
MethodSpec.Builder methodBuilder = generateMethodBuilder(method);
// create PaginatedResult
createPagedResult(method, pagedResultName, methodBuilder);
generateCommonPart(method, classBuilder, methodBuilder, fieldList, GenerationType.NO_CONTENT, null, true, false, "paginatedResult");
methodBuilder.addStatement("return paginatedResult");
if (!method.isPagedLiveData()) {
classBuilder.addMethod(methodBuilder.build());
}
}
String selectTotalCountMethodName = method.getName() + "TotalCount";
// generate paged result method
{
//@formatter:off
MethodSpec.Builder methodBuilder = MethodSpec.methodBuilder(method.getName()).addModifiers(Modifier.PRIVATE);
generateMethodSignature(method,
methodBuilder,
TypeUtility.parameterizedTypeName(TypeUtility.className(List.class),
TypeUtility.typeName(method.getEntity().getElement())),
ParameterSpec.builder(TypeUtility.typeName(pagedResultName), "paginatedResult")
.build());
generateTotalCountUsage(method, methodBuilder, "paginatedResult", selectTotalCountMethodName);
generateCommonPart(method, classBuilder, methodBuilder, fieldList, GenerationType.NO_METHOD_SIGN, null, false, false, "paginatedResult",
JavadocPart.build(JavadocPartType.ADD_PARAMETER, "paginatedResult", "handler of paginated result"), JavadocPart.build(JavadocPartType.RETURN, "", "result list"));
methodBuilder.addComment("Specialized part II - $L - BEGIN", this.getClass().getSimpleName());
generateSpecializedPart(method, classBuilder, methodBuilder, fieldList, selectType.isMapFields());
methodBuilder.addComment("Specialized part II - $L - END", this.getClass().getSimpleName());
classBuilder.addMethod(methodBuilder.build());
//@formatter:on
}
// generate select count method
{
Set countfields = new HashSet<>();
countfields.add(JQLProjection.ProjectionBuilder.create().expression("count(*)").build());
MethodSpec.Builder methodBuilder = MethodSpec.methodBuilder(selectTotalCountMethodName).addModifiers(Modifier.PRIVATE);
generateMethodSignature(method, methodBuilder, TypeName.INT, ParameterSpec.builder(TypeUtility.typeName(pagedResultName), "paginatedResult").build());
generateCommonPart(method, classBuilder, methodBuilder, countfields, GenerationType.NO_METHOD_SIGN, TypeName.INT, false, true, "paginatedResult",
JavadocPart.build(JavadocPartType.ADD_PARAMETER, "paginatedResult", "handler of paginated result"), JavadocPart.build(JavadocPartType.RETURN, "", "total row count"));
methodBuilder.addComment("Specialized part II - $L - BEGIN", this.getClass().getSimpleName());
generateTotalSpecializedPart(method, methodBuilder);
methodBuilder.addComment("Specialized part II - $L - END", this.getClass().getSimpleName());
classBuilder.addMethod(methodBuilder.build());
}
// generate paged result method with pageRequest
{
String selectWithPageRequestMethodName = method.getName() + SelectPaginatedResultHelper.WITH_PAGE_REQUEST_PREFIX;
//@formatter:off
MethodSpec.Builder methodBuilder = MethodSpec.methodBuilder(selectWithPageRequestMethodName).addModifiers(Modifier.PRIVATE);
generateMethodSignature(method,
methodBuilder,
TypeUtility.parameterizedTypeName(TypeUtility.className(List.class),
TypeUtility.typeName(method.getEntity().getElement())),
ParameterSpec.builder(PageRequest.class, "pageRequest")
.build());
// generateTotalCountUsage(method, methodBuilder, "paginatedResult", selectTotalCountMethodName);
generateCommonPart(method, classBuilder, methodBuilder, fieldList, GenerationType.NO_METHOD_SIGN, null, false, false, "pageRequest",
JavadocPart.build(JavadocPartType.ADD_PARAMETER, "pageRequest", "page request"), JavadocPart.build(JavadocPartType.RETURN, "", "result list"));
methodBuilder.addComment("Specialized part II - $L - BEGIN", this.getClass().getSimpleName());
generateSpecializedPart(method, classBuilder, methodBuilder, fieldList, selectType.isMapFields());
methodBuilder.addComment("Specialized part II - $L - END", this.getClass().getSimpleName());
classBuilder.addMethod(methodBuilder.build());
//@formatter:on
}
}
private void generateTotalCountUsage(SQLiteModelMethod method, Builder methodBuilder, String paginatedResultName, String selectTotalCountMethodName) {
methodBuilder.addComment("total count - BEGIN");
methodBuilder.addCode(paginatedResultName + ".setTotalElements(this." + selectTotalCountMethodName + "(");
method.getParameters().forEach(p -> {
methodBuilder.addCode(p.value0 + ", ");
});
methodBuilder.addCode(paginatedResultName + "));\n");
methodBuilder.addComment("total count - END");
}
public static void createPagedResult(SQLiteModelMethod method, String pagedResultName, MethodSpec.Builder methodBuilder) {
String separator = "";
methodBuilder.addCode("final $L paginatedResult=new $L(", pagedResultName, pagedResultName);
for (Pair item : method.getParameters()) {
// field
methodBuilder.addCode(separator + "$L", item.value0);
separator = ", ";
}
methodBuilder.addCode(");\n");
}
/** Used to generate specialized Paged Result classes;. */
static int pagedResultCounter;
/*
* (non-Javadoc)
*
* @see com.abubusoft.kripton.processor.sqlite.AbstractSelectCodeGenerator#
* generateSpecializedPart(com.abubusoft.kripton.processor.sqlite.model.
* SQLiteModelMethod, com.squareup.javapoet.TypeSpec.Builder,
* com.squareup.javapoet.MethodSpec.Builder, java.util.Set, boolean)
*/
@Override
public void generateSpecializedPart(SQLiteModelMethod method, TypeSpec.Builder classBuilder, MethodSpec.Builder methodBuilder, Set fieldList, boolean mapFields) {
SQLiteEntity entity = method.getEntity();
// get return type (in this case is always a list)
ClassName returnRawListClazzName = ClassName.get(List.class);
TypeName entityClass = typeName(entity.getElement());
methodBuilder.addCode("\n");
methodBuilder.addStatement("$T<$T> resultList=new $T<$T>(_cursor.getCount())", List.class, entityClass, ArrayList.class, entityClass);
methodBuilder.addStatement("$T resultBean=null", entityClass);
// immutable management
if (entity.isImmutablePojo()) {
methodBuilder.addCode("\n");
methodBuilder.addComment("initialize temporary variable for immutable POJO");
ImmutableUtility.generateImmutableVariableInit(entity, methodBuilder);
}
methodBuilder.addCode("\n");
methodBuilder.beginControlFlow("if (_cursor.moveToFirst())");
// generate index from columns
methodBuilder.addCode("\n");
{
int i = 0;
for (JQLProjection a : fieldList) {
SQLProperty item = a.property;
methodBuilder.addStatement("int index$L=_cursor.getColumnIndex($S)", (i++), item.columnName);
if (item.hasTypeAdapter()) {
methodBuilder.addStatement("$T $LAdapter=$T.getAdapter($T.class)", item.typeAdapter.getAdapterTypeName(), item.getName(), SQLTypeAdapterUtils.class,
item.typeAdapter.getAdapterTypeName());
}
}
}
methodBuilder.addCode("\n");
methodBuilder.beginControlFlow("do\n");
// immutable management
if (entity.isImmutablePojo()) {
methodBuilder.addComment("reset temporary variable for immutable POJO");
ImmutableUtility.generateImmutableVariableReset(entity, methodBuilder);
} else {
methodBuilder.addCode("resultBean=new $T();\n\n", entityClass);
}
// generate mapping
int i = 0;
for (JQLProjection a : fieldList) {
SQLProperty item = a.property;
if (item.isNullable()) {
methodBuilder.addCode("if (!_cursor.isNull(index$L)) { ", i);
}
SQLTransformer.cursor2Java(method.getParent().getEntity(), methodBuilder, typeName(entity.getElement()), item, "resultBean", "_cursor", "index" + i + "");
methodBuilder.addCode(";");
if (item.isNullable()) {
methodBuilder.addCode(" }");
}
methodBuilder.addCode("\n");
i++;
}
methodBuilder.addCode("\n");
// immutable management
if (entity.isImmutablePojo()) {
methodBuilder.addComment("define immutable POJO");
ImmutableUtility.generateImmutableEntityCreation(entity, methodBuilder, "resultBean", false);
}
methodBuilder.addCode("resultList.add(resultBean);\n");
methodBuilder.endControlFlow("while (_cursor.moveToNext())");
methodBuilder.endControlFlow();
methodBuilder.addCode("\n");
// return list or immutable list
if (entity.isImmutablePojo()) {
methodBuilder.addCode("return ");
ImmutableUtility.generateImmutableCollectionIfPossible(entity, methodBuilder, "resultList", ParameterizedTypeName.get(returnRawListClazzName, entityClass));
methodBuilder.addCode(";\n");
} else {
methodBuilder.addCode("return resultList;\n");
}
methodBuilder.endControlFlow();
}
public void generateTotalSpecializedPart(SQLiteModelMethod method, MethodSpec.Builder methodBuilder) {
SQLiteDaoDefinition daoDefinition = method.getParent();
methodBuilder.addComment("manage query for total count eements");
methodBuilder.addStatement("int _result=-1");
methodBuilder.addCode("\n");
methodBuilder.beginControlFlow("if (_cursor.moveToFirst())");
methodBuilder.addStatement("_result=_cursor.getInt(0)");
methodBuilder.endControlFlow();
if (daoDefinition.isLogEnabled()) {
// generate log section - BEGIN
methodBuilder.addComment("log section for select BEGIN");
methodBuilder.beginControlFlow("if (_context.isLogEnabled())");
// manage log
methodBuilder.addComment("manage log");
// log for where parames
SqlBuilderHelper.generateLogForWhereParameters(method, methodBuilder);
methodBuilder.addStatement("$T.info($S, _result)", Logger.class, "Total elements found: %s");
// generate log section - END
methodBuilder.addComment("log section for select END");
}
methodBuilder.endControlFlow();
methodBuilder.addStatement("return _result");
methodBuilder.endControlFlow();
}
/**
* Build paginated result class handler.
*
* @param classBuilder
* the class builder
* @param method
* the method
* @return name of generated class
*/
public static String buildSpecializedPagedResultClass(TypeSpec.Builder classBuilder, SQLiteModelMethod method) {
// TypeName entityTypeName =
// TypeUtility.typeName(method.getParent().getEntityClassName());
TypeName entityTypeName = TypeUtility.typeName(method.getEntity().getName());
String pagedResultName = "PaginatedResult" + (pagedResultCounter++);
TypeSpec.Builder typeBuilder = TypeSpec.classBuilder(pagedResultName).addModifiers(Modifier.PUBLIC)
.superclass(TypeUtility.parameterizedTypeName(TypeUtility.className(PagedResultImpl.class), entityTypeName));
// add fields and define constructor
MethodSpec.Builder setupBuilder = MethodSpec.constructorBuilder();
MethodSpec.Builder executeBuilder = MethodSpec.methodBuilder("execute").addModifiers(Modifier.PUBLIC)
.returns(TypeUtility.parameterizedTypeName(TypeUtility.className(List.class), entityTypeName));
executeBuilder.addComment("Executor builder - BEGIN");
// method for execution with pageRequest
//@formatter:off
ClassName dataSourceClassName = BindDataSourceBuilder.generateDataSourceName(method.getParent().getParent());
MethodSpec.Builder executeWithPageRequestBuilder = MethodSpec.methodBuilder("execute")
.addModifiers(Modifier.PUBLIC)
.addParameter(PageRequest.class, "pageRequest")
.returns(TypeUtility.parameterizedTypeName(TypeUtility.className(List.class), entityTypeName));
executeWithPageRequestBuilder.addComment("Executor with pageRequet - BEGIN");
executeWithPageRequestBuilder.addCode("return $T.getInstance().executeBatch(daoFactory -> daoFactory.get$L().$L"+SelectPaginatedResultHelper.WITH_PAGE_REQUEST_PREFIX+"(",
dataSourceClassName,
method.getParentName(),
method.getName());
ClassName daoFactoryClassName = BindDaoFactoryBuilder.generateDaoFactoryClassName(method.getParent().getParent());
MethodSpec.Builder executeWithDaoFactortBuilder = MethodSpec.methodBuilder("execute")
.addParameter(daoFactoryClassName, "daoFactory")
.addModifiers(Modifier.PUBLIC)
.returns(TypeUtility.parameterizedTypeName(TypeUtility.className(List.class), entityTypeName));
executeWithDaoFactortBuilder.addCode("return daoFactory.get$L().$L(", method.getParentName(), method.getName());
//@formatter:on
if (!method.isPagedLiveData()) {
executeBuilder.addCode("list=$T.this.$L(", TypeUtility.typeName(method.getParent().getElement(), BindDaoBuilder.SUFFIX), method.getName());
}
// we have always a first parameter
String separator = "";
ParameterSpec parameterSpec;
for (Pair item : method.getParameters()) {
if (method.hasDynamicPageSizeConditions() && method.dynamicPageSizeName.equals(item.value0)) {
setupBuilder.addStatement("this.pageSize=$L", item.value0);
} else {
// field
typeBuilder.addField(item.value1, item.value0);
setupBuilder.addStatement("this.$L=$L", item.value0, item.value0);
}
// construtor
parameterSpec = ParameterSpec.builder(item.value1, item.value0).build();
setupBuilder.addParameter(parameterSpec);
// execute
if (method.dynamicPageSizeName != null && method.dynamicPageSizeName.equals(item.value0)) {
if (!method.isPagedLiveData()) {
executeBuilder.addCode(separator + "this.pageSize");
}
executeWithDaoFactortBuilder.addCode(separator + "this.pageSize");
executeWithPageRequestBuilder.addCode(separator + "this.pageSize");
} else {
if (!method.isPagedLiveData()) {
executeBuilder.addCode(separator + item.value0);
}
executeWithDaoFactortBuilder.addCode(separator + item.value0);
executeWithPageRequestBuilder.addCode(separator + item.value0);
}
separator = ", ";
}
if (method.isPagedLiveData()) {
// if method is not paged live data, add complete execute method
executeBuilder.addComment("paged result is used in live data, so this method must be empty");
executeBuilder.addStatement("return null");
} else {
executeBuilder.addCode(separator + "this);\n");
executeBuilder.addStatement("return list");
}
executeBuilder.addComment("Executor builder - END");
typeBuilder.addMethod(executeBuilder.build());
if (!method.hasDynamicPageSizeConditions()) {
ModelAnnotation annotation = method.getAnnotation(BindSqlSelect.class);
int pageSize = annotation.getAttributeAsInt(AnnotationAttributeType.PAGE_SIZE);
// in case pageSize is not specified (only for liveData)
if (pageSize == 0) {
pageSize = 20;
}
setupBuilder.addStatement("this.pageSize=$L", pageSize);
}
typeBuilder.addMethod(setupBuilder.build());
executeWithDaoFactortBuilder.addCode(separator + "this);\n");
executeWithPageRequestBuilder.addCode(separator + "pageRequest));\n");
executeWithPageRequestBuilder.addComment("Executor with pageRequet - END");
// add methods to type builder
typeBuilder.addMethod(executeWithDaoFactortBuilder.build());
typeBuilder.addMethod(executeWithPageRequestBuilder.build());
classBuilder.addType(typeBuilder.build());
return pagedResultName;
}
/**
*
* Return the simple current paged result inner class name.
*
* @return
*/
public static String getCurrentPagedResultClass() {
return "PaginatedResult" + (pagedResultCounter - 1);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy