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.
io.micronaut.data.runtime.query.internal.DefaultStoredQuery Maven / Gradle / Ivy
/*
* Copyright 2017-2022 original authors
*
* 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
*
* https://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 io.micronaut.data.runtime.query.internal;
import io.micronaut.core.annotation.AnnotationMetadata;
import io.micronaut.core.annotation.AnnotationValue;
import io.micronaut.core.annotation.Internal;
import io.micronaut.core.annotation.NonNull;
import io.micronaut.core.reflect.ReflectionUtils;
import io.micronaut.core.util.ArrayUtils;
import io.micronaut.core.util.StringUtils;
import io.micronaut.data.annotation.*;
import io.micronaut.data.intercept.annotation.DataMethod;
import io.micronaut.data.intercept.annotation.DataMethodQueryParameter;
import io.micronaut.data.model.AssociationUtils;
import io.micronaut.data.model.DataType;
import io.micronaut.data.model.JsonDataType;
import io.micronaut.data.model.query.JoinPath;
import io.micronaut.data.model.query.builder.sql.SqlQueryBuilder;
import io.micronaut.data.model.runtime.DefaultStoredDataOperation;
import io.micronaut.data.model.runtime.QueryParameterBinding;
import io.micronaut.data.model.runtime.StoredQuery;
import io.micronaut.data.operations.HintsCapableRepository;
import io.micronaut.inject.ExecutableMethod;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import static io.micronaut.data.intercept.annotation.DataMethod.META_MEMBER_PAGE_SIZE;
/**
* Represents a prepared query.
*
* @param The entity type
* @param The result type
*/
@Internal
public final class DefaultStoredQuery extends DefaultStoredDataOperation implements StoredQuery {
private static final String DATA_METHOD_ANN_NAME = DataMethod.class.getName();
@NonNull
private final Class resultType;
@NonNull
private final Class rootEntity;
@NonNull
private final String query;
private final String[] queryParts;
private final ExecutableMethod, ?> method;
private final boolean isDto;
private final boolean isOptimisticLock;
private final boolean isNative;
private final boolean isProcedure;
private final boolean isNumericPlaceHolder;
private final boolean hasPageable;
private final AnnotationMetadata annotationMetadata;
private final boolean isCount;
private final boolean hasResultConsumer;
private Map queryHints;
private Set joinFetchPaths = null;
private final List queryParameters;
private final boolean rawQuery;
private final boolean jsonEntity;
private final OperationType operationType;
private final Map> parameterExpressions;
/**
* The default constructor.
*
* @param method The target method
* @param resultType The result type of the query
* @param rootEntity The root entity of the query
* @param query The query itself
* @param isCount Is the query a count query
* @param repositoryOperations The repositoryOperations
*/
public DefaultStoredQuery(
@NonNull ExecutableMethod, ?> method,
@NonNull Class resultType,
@NonNull Class rootEntity,
@NonNull String query,
boolean isCount,
HintsCapableRepository repositoryOperations) {
super(method);
//noinspection unchecked
this.resultType = (Class) ReflectionUtils.getWrapperType(resultType);
this.rootEntity = rootEntity;
this.annotationMetadata = method.getAnnotationMetadata();
this.isNative = method.isTrue(Query.class, "nativeQuery");
this.isProcedure = method.isTrue(DataMethod.class, DataMethod.META_MEMBER_PROCEDURE);
this.hasResultConsumer = method.stringValue(DATA_METHOD_ANN_NAME, "sqlMappingFunction").isPresent();
this.isNumericPlaceHolder = method
.classValue(RepositoryConfiguration.class, "queryBuilder")
.map(c -> c == SqlQueryBuilder.class).orElse(false);
this.hasPageable = method.stringValue(DATA_METHOD_ANN_NAME, TypeRole.PAGEABLE).isPresent() ||
method.stringValue(DATA_METHOD_ANN_NAME, TypeRole.SORT).isPresent() ||
method.intValue(DATA_METHOD_ANN_NAME, META_MEMBER_PAGE_SIZE).orElse(-1) > -1;
if (isCount) {
Optional rawCountQueryString = method.stringValue(Query.class, DataMethod.META_MEMBER_RAW_COUNT_QUERY);
this.rawQuery = rawCountQueryString.isPresent();
this.query = rawCountQueryString.orElse(query);
String[] countQueryParts = method.stringValues(DataMethod.class, DataMethod.META_MEMBER_EXPANDABLE_COUNT_QUERY);
// for countBy queries this is empty, and we should use DataMethod.META_MEMBER_EXPANDABLE_QUERY value
if (ArrayUtils.isNotEmpty(countQueryParts)) {
this.queryParts = countQueryParts;
} else {
this.queryParts = method.stringValues(DataMethod.class, DataMethod.META_MEMBER_EXPANDABLE_QUERY);
}
} else {
Optional rawQueryString = method.stringValue(Query.class, DataMethod.META_MEMBER_RAW_QUERY);
this.rawQuery = rawQueryString.isPresent();
this.query = rawQueryString.orElse(query);
this.queryParts = method.stringValues(DataMethod.class, DataMethod.META_MEMBER_EXPANDABLE_QUERY);
}
this.method = method;
this.isDto = method.isTrue(DATA_METHOD_ANN_NAME, DataMethod.META_MEMBER_DTO);
this.isOptimisticLock = method.isTrue(DATA_METHOD_ANN_NAME, DataMethod.META_MEMBER_OPTIMISTIC_LOCK);
this.isCount = isCount;
AnnotationValue annotation = annotationMetadata.getAnnotation(DataMethod.class);
if (method.hasAnnotation(QueryHint.class)) {
List> values = method.getAnnotationValuesByType(QueryHint.class);
this.queryHints = new HashMap<>(values.size());
for (AnnotationValue value : values) {
String n = value.stringValue("name").orElse(null);
String v = value.stringValue("value").orElse(null);
if (StringUtils.isNotEmpty(n) && StringUtils.isNotEmpty(v)) {
queryHints.put(n, v);
}
}
}
Map queryHints = repositoryOperations.getQueryHints(this);
if (queryHints != Collections.EMPTY_MAP) {
if (this.queryHints != null) {
this.queryHints.putAll(queryHints);
} else {
this.queryHints = queryHints;
}
}
if (annotation == null) {
queryParameters = Collections.emptyList();
} else {
List> params = annotation.getAnnotations(DataMethod.META_MEMBER_PARAMETERS, DataMethodQueryParameter.class);
List queryParameters = new ArrayList<>(params.size());
for (AnnotationValue av : params) {
String[] propertyPath = av.stringValues(DataMethodQueryParameter.META_MEMBER_PROPERTY_PATH);
if (propertyPath.length == 0) {
propertyPath = av.stringValue(DataMethodQueryParameter.META_MEMBER_PROPERTY)
.map(property -> new String[]{property})
.orElse(null);
}
String[] parameterBindingPath = av.stringValues(DataMethodQueryParameter.META_MEMBER_PARAMETER_BINDING_PATH);
if (parameterBindingPath.length == 0) {
parameterBindingPath = null;
}
DataType dataType = isNumericPlaceHolder ? av.enumValue(DataMethodQueryParameter.META_MEMBER_DATA_TYPE, DataType.class).orElse(DataType.OBJECT) : null;
JsonDataType jsonDataType = dataType != null ? av.enumValue(DataMethodQueryParameter.META_MEMBER_JSON_DATA_TYPE, JsonDataType.class).orElse(JsonDataType.DEFAULT) : null;
queryParameters.add(
new StoredQueryParameter(
av.stringValue(DataMethodQueryParameter.META_MEMBER_NAME).orElse(null),
dataType,
jsonDataType,
av.intValue(DataMethodQueryParameter.META_MEMBER_PARAMETER_INDEX).orElse(-1),
parameterBindingPath,
propertyPath,
av.booleanValue(DataMethodQueryParameter.META_MEMBER_AUTO_POPULATED).orElse(false),
av.booleanValue(DataMethodQueryParameter.META_MEMBER_REQUIRES_PREVIOUS_POPULATED_VALUES).orElse(false),
av.classValue(DataMethodQueryParameter.META_MEMBER_CONVERTER).orElse(null),
av.booleanValue(DataMethodQueryParameter.META_MEMBER_EXPANDABLE).orElse(false),
av.booleanValue(DataMethodQueryParameter.META_MEMBER_EXPRESSION).orElse(false),
queryParameters
));
}
this.queryParameters = queryParameters;
}
this.jsonEntity = DataAnnotationUtils.hasJsonEntityRepresentationAnnotation(annotationMetadata);
this.operationType = method.enumValue(DataMethod.NAME, DataMethod.META_MEMBER_OPERATION_TYPE, DataMethod.OperationType.class)
.map(op -> OperationType.valueOf(op.name()))
.orElse(OperationType.QUERY);
this.parameterExpressions = annotationMetadata.getAnnotationValuesByType(ParameterExpression.class).stream()
.collect(Collectors.toMap(av -> av.stringValue("name").orElseThrow(), av -> av));
}
@Override
public List getQueryBindings() {
return queryParameters;
}
@NonNull
@Override
public Set getJoinFetchPaths() {
if (joinFetchPaths == null) {
Set set = AssociationUtils.getJoinFetchPaths(method);
this.joinFetchPaths = set.isEmpty() ? Collections.emptySet() : Collections.unmodifiableSet(set);
}
return joinFetchPaths;
}
/**
* @return The method
*/
public ExecutableMethod, ?> getMethod() {
return method;
}
@Override
public boolean isSingleResult() {
return !isCount() && getJoinFetchPaths().isEmpty();
}
@Override
public boolean hasResultConsumer() {
return this.hasResultConsumer;
}
@Override
public boolean isCount() {
return isCount;
}
@NonNull
@Override
public Map getQueryHints() {
if (queryHints != null) {
return queryHints;
}
return Collections.emptyMap();
}
@Override
public boolean isNative() {
return isNative;
}
@Override
public boolean isProcedure() {
return isProcedure;
}
@Override
public OperationType getOperationType() {
return operationType;
}
/**
* Is this a raw SQL query.
*
* @return The raw sql query.
*/
@Override
public boolean useNumericPlaceholders() {
return isNumericPlaceHolder;
}
/**
* @return Whether the query is a DTO query
*/
@Override
public boolean isDtoProjection() {
return isDto;
}
/**
* @return The result type
*/
@Override
@NonNull
public Class getResultType() {
return resultType;
}
@NonNull
@Override
public DataType getResultDataType() {
if (isCount) {
return DataType.LONG;
}
return annotationMetadata.enumValue(DATA_METHOD_ANN_NAME, DataMethod.META_MEMBER_RESULT_DATA_TYPE, DataType.class)
.orElse(DataType.OBJECT);
}
/**
* @return The ID type
*/
@SuppressWarnings("unchecked")
@Override
public Optional> getEntityIdentifierType() {
Optional o = annotationMetadata.classValue(DATA_METHOD_ANN_NAME, DataMethod.META_MEMBER_ID_TYPE);
return o;
}
/**
* @return The root entity type
*/
@Override
@NonNull
public Class getRootEntity() {
return rootEntity;
}
@Override
public boolean hasPageable() {
return hasPageable;
}
@Override
@NonNull
public String getQuery() {
return query;
}
@Override
public String[] getExpandableQueryParts() {
return queryParts;
}
@NonNull
@Override
public String getName() {
return method.getMethodName();
}
@Override
@NonNull
public Class>[] getArgumentTypes() {
return method.getArgumentTypes();
}
@Override
public boolean isOptimisticLock() {
return isOptimisticLock;
}
@Override
public boolean isRawQuery() {
return this.rawQuery;
}
@Override
public boolean isJsonEntity() {
return jsonEntity;
}
@Override
public Map> getParameterExpressions() {
return parameterExpressions;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
DefaultStoredQuery, ?> that = (DefaultStoredQuery, ?>) o;
return resultType.equals(that.resultType) &&
method.equals(that.method);
}
@Override
public int hashCode() {
return Objects.hash(resultType, method);
}
}