package org.springframework.data.jpa.repository.query;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.IntStream;
import javax.persistence.EntityManager;
import javax.persistence.NamedStoredProcedureQuery;
import javax.persistence.ParameterMode;
import javax.persistence.StoredProcedureQuery;
import javax.persistence.TypedQuery;
import org.springframework.data.jpa.repository.query.JpaParameters.JpaParameter;
import org.springframework.data.repository.query.Parameter;
import org.springframework.data.repository.query.ParametersParameterAccessor;
import org.springframework.data.repository.query.QueryMethod;
import org.springframework.data.repository.query.ResultProcessor;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;
import org.springframework.util.StringUtils;
class StoredProcedureJpaQuery extends AbstractJpaQuery {
private final StoredProcedureAttributes procedureAttributes;
private final boolean useNamedParameters;
private final QueryParameterSetter.QueryMetadataCache metadataCache = new QueryParameterSetter.QueryMetadataCache();
StoredProcedureJpaQuery(JpaQueryMethod method, EntityManager em) {
super(method, em);
this.procedureAttributes = method.getProcedureAttributes();
this.useNamedParameters = useNamedParameters(method);
}
private static boolean useNamedParameters(QueryMethod method) {
for (Parameter parameter : method.getParameters()) {
if (parameter.isNamedParameter()) {
return true;
}
}
return false;
}
@Override
protected StoredProcedureQuery createQuery(JpaParametersParameterAccessor accessor) {
return applyHints(doCreateQuery(accessor), getQueryMethod());
}
@Override
protected StoredProcedureQuery doCreateQuery(JpaParametersParameterAccessor accessor) {
StoredProcedureQuery storedProcedure = createStoredProcedure();
QueryParameterSetter.QueryMetadata metadata = metadataCache.getMetadata("singleton", storedProcedure);
return parameterBinder.get().bind(storedProcedure, metadata, accessor);
}
@Override
protected TypedQuery<Long> doCreateCountQuery(JpaParametersParameterAccessor accessor) {
throw new UnsupportedOperationException("StoredProcedureQuery does not support count queries!");
}
@Nullable
Object (StoredProcedureQuery storedProcedureQuery) {
Assert.notNull(storedProcedureQuery, "StoredProcedureQuery must not be null!");
if (!procedureAttributes.hasReturnValue()) {
return null;
}
Map<String, Object> outputValues = new HashMap<>();
List<String> parameterNames = procedureAttributes.getOutputParameterNames();
for (int i = 0; i < parameterNames.size(); i++) {
String name = parameterNames.get(i);
outputValues.put(name, extractOutputParameter(storedProcedureQuery, i));
}
return outputValues.size() == 1 ? outputValues.values().iterator().next() : outputValues;
}
private Object (StoredProcedureQuery storedProcedureQuery, Integer index) {
String outputParameterName = procedureAttributes.getOutputParameterNames().get(index);
JpaParameters parameters = getQueryMethod().getParameters();
return extractOutputParameterValue(storedProcedureQuery, outputParameterName, index,
parameters.getNumberOfParameters());
}
private Object (StoredProcedureQuery storedProcedureQuery, String name, Integer index,
int offset) {
return useNamedParameters && StringUtils.hasText(name) ?
storedProcedureQuery.getOutputParameterValue(name)
: storedProcedureQuery.getOutputParameterValue(offset + index + 1);
}
private StoredProcedureQuery createStoredProcedure() {
return procedureAttributes.isNamedStoredProcedure() ? newNamedStoredProcedureQuery()
: newAdhocStoredProcedureQuery();
}
private StoredProcedureQuery newNamedStoredProcedureQuery() {
return getEntityManager().createNamedStoredProcedureQuery(procedureAttributes.getProcedureName());
}
private StoredProcedureQuery newAdhocStoredProcedureQuery() {
JpaParameters params = getQueryMethod().getParameters();
String procedureName = procedureAttributes.getProcedureName();
StoredProcedureQuery procedureQuery = getEntityManager().createStoredProcedureQuery(procedureName);
for (JpaParameter param : params) {
if (!param.isBindable()) {
continue;
}
if (useNamedParameters) {
procedureQuery.registerStoredProcedureParameter(
param.getName()
.orElseThrow(() -> new IllegalArgumentException(ParameterBinder.PARAMETER_NEEDS_TO_BE_NAMED)),
param.getType(), ParameterMode.IN);
} else {
procedureQuery.registerStoredProcedureParameter(param.getIndex() + 1, param.getType(), ParameterMode.IN);
}
}
if (procedureAttributes.hasReturnValue()) {
ParameterMode mode = ParameterMode.OUT;
IntStream.range(0, procedureAttributes.getOutputParameterTypes().size()).forEach(i -> {
Class<?> outputParameterType = procedureAttributes.getOutputParameterTypes().get(i);
if (useNamedParameters) {
String outputParameterName = procedureAttributes.getOutputParameterNames().get(i);
procedureQuery.registerStoredProcedureParameter(outputParameterName, outputParameterType, mode);
} else {
procedureQuery.registerStoredProcedureParameter(params.getNumberOfParameters() + i + 1, outputParameterType,
mode);
}
});
}
return procedureQuery;
}
}