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.crnk.jpa.internal.query.backend.criteria.JpaCriteriaQueryBackend Maven / Gradle / Ivy
package io.crnk.jpa.internal.query.backend.criteria;
import io.crnk.core.engine.internal.utils.PreconditionUtil;
import io.crnk.core.queryspec.Direction;
import io.crnk.core.queryspec.FilterOperator;
import io.crnk.jpa.internal.query.JoinRegistry;
import io.crnk.jpa.internal.query.MetaComputedAttribute;
import io.crnk.jpa.internal.query.QueryUtil;
import io.crnk.jpa.internal.query.backend.JpaQueryBackend;
import io.crnk.jpa.meta.MetaEntity;
import io.crnk.jpa.query.criteria.JpaCriteriaExpressionFactory;
import io.crnk.meta.model.MetaAttribute;
import io.crnk.meta.model.MetaAttributePath;
import io.crnk.meta.model.MetaDataObject;
import io.crnk.meta.model.MetaKey;
import javax.persistence.EntityManager;
import javax.persistence.criteria.*;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
public class JpaCriteriaQueryBackend implements JpaQueryBackend, Order, Predicate, Expression>> {
protected CriteriaBuilder cb;
private CriteriaQuery criteriaQuery;
private JoinRegistry, Expression>> joinHelper;
private From root;
private Root> parentFrom;
private JpaCriteriaQueryImpl queryImpl;
@SuppressWarnings({"unchecked", "rawtypes"})
public JpaCriteriaQueryBackend(JpaCriteriaQueryImpl query, EntityManager em, Class clazz, MetaDataObject parentMeta,
MetaAttribute parentAttr, boolean parentIdSelection) {
this.queryImpl = query;
cb = em.getCriteriaBuilder();
criteriaQuery = (CriteriaQuery) cb.createQuery();
if (parentMeta != null) {
parentFrom = criteriaQuery.from(parentMeta.getImplementationClass());
root = parentFrom.join(parentAttr.getName());
joinHelper = new JoinRegistry<>(this, query);
joinHelper.putJoin(new MetaAttributePath(), root);
if (parentIdSelection) {
Expression> parentIdExpr = getParentIdExpression(parentAttr);
criteriaQuery.multiselect((List) Arrays.asList(parentIdExpr, root));
} else {
criteriaQuery.select(root);
}
} else {
root = criteriaQuery.from(clazz);
joinHelper = new JoinRegistry<>(this, query);
joinHelper.putJoin(new MetaAttributePath(), root);
criteriaQuery.select(root);
}
}
private Expression> getParentIdExpression(MetaAttribute parentAttr) {
MetaEntity parentEntity = (MetaEntity) parentAttr.getParent();
MetaKey primaryKey = parentEntity.getPrimaryKey();
List elements = primaryKey.getElements();
PreconditionUtil.assertFalse("composite primary keys not supported yet", elements.size() != 1);
MetaAttribute primaryKeyAttr = elements.get(0);
return parentFrom.get(primaryKeyAttr.getName());
}
@Override
public Expression> getAttribute(MetaAttributePath attrPath) {
return joinHelper.getEntityAttribute(attrPath);
}
@Override
public void addPredicate(Predicate predicate) {
Predicate restriction = criteriaQuery.getRestriction();
if (restriction != null) {
criteriaQuery.where(restriction, predicate);
} else {
criteriaQuery.where(predicate);
}
}
@Override
public From, ?> getRoot() {
return root;
}
@Override
public void setOrder(List list) {
criteriaQuery.orderBy(list);
}
@Override
public List getOrderList() {
return new ArrayList<>(criteriaQuery.getOrderList());
}
@Override
public Order newSort(Expression> expr, Direction dir) {
if (dir == Direction.ASC) {
return cb.asc(expr);
} else {
return cb.desc(expr);
}
}
@Override
public void distinct() {
criteriaQuery.distinct(true);
}
public CriteriaQuery getCriteriaQuery() {
return criteriaQuery;
}
@Override
public void addParentPredicate(MetaAttribute primaryKeyAttr) {
List> parentIds = queryImpl.getParentIds();
Path parentIdPath = parentFrom.get(primaryKeyAttr.getName());
addPredicate(parentIdPath.in(new ArrayList(parentIds)));
}
@Override
public boolean hasManyRootsFetchesOrJoins() {
return QueryUtil.hasManyRootsFetchesOrJoins(criteriaQuery);
}
@Override
public void addSelection(Expression> expression, String name) {
Selection> selection = criteriaQuery.getSelection();
List> newSelection = new ArrayList<>();
if (selection != null) {
if (selection.isCompoundSelection()) {
newSelection.addAll(selection.getCompoundSelectionItems());
} else {
newSelection.add(selection);
}
}
newSelection.add(expression);
criteriaQuery.multiselect(newSelection);
}
@Override
public Expression> getExpression(Order order) {
return order.getExpression();
}
@Override
public boolean containsRelation(Expression> expression) {
return QueryUtil.containsRelation(expression);
}
public Predicate ilike(Expression expr, String val) {
return cb.like(cb.lower(expr), val.toLowerCase());
}
private Predicate negateIfNeeded(Predicate p, FilterOperator fc) {
if (fc.equals(FilterOperator.NEQ))
return cb.not(p);
return p;
}
@Override
public Predicate buildPredicate(FilterOperator operator, MetaAttributePath attrPath, Object value) {
Expression> attr = getAttribute(attrPath);
return buildPredicate(operator, attr, value);
}
@SuppressWarnings({"rawtypes"})
public Predicate buildPredicate(FilterOperator operator, Expression> expressionObj, Object value) {
Expression expression = expressionObj;
expression = handleConversions(expression, operator);
return handle(expression, operator, value);
}
@SuppressWarnings({"rawtypes", "unchecked"})
private Predicate handle(Expression expression, FilterOperator operator, Object value) {
if (operator == FilterOperator.EQ || operator == FilterOperator.NEQ) {
return handleEquals(expression, operator, value);
} else if (operator == FilterOperator.LIKE) {
return ilike(expression, value.toString());
} else if (operator == FilterOperator.GT) {
return cb.greaterThan(expression, (Comparable) value);
} else if (operator == FilterOperator.LT) {
return cb.lessThan(expression, (Comparable) value);
} else if (operator == FilterOperator.GE) {
return cb.greaterThanOrEqualTo(expression, (Comparable) value);
} else {
PreconditionUtil.verify(operator == FilterOperator.LE, "unexpected operator %s", operator);
return cb.lessThanOrEqualTo(expression, (Comparable) value);
}
}
@SuppressWarnings({"rawtypes", "unchecked"})
private Predicate handleEquals(Expression> expression, FilterOperator operator, Object value) {
if (value instanceof List) {
Predicate p = expression.in(((List>) value).toArray());
return negateIfNeeded(p, operator);
} else if (Collection.class.isAssignableFrom(expression.getJavaType())) {
Predicate p = cb.literal(value).in(expression);
return negateIfNeeded(p, operator);
} else if (expression instanceof MapJoin) {
Predicate p = cb.literal(value).in(((MapJoin) expression).value());
return negateIfNeeded(p, operator);
} else if (value == null) {
return negateIfNeeded(cb.isNull(expression), operator);
}
return negateIfNeeded(cb.equal(expression, value), operator);
}
private Expression> handleConversions(Expression> expression, FilterOperator operator) {
// convert to String for LIKE operators
if (expression.getJavaType() != String.class && (operator == FilterOperator.LIKE)) {
return expression.as(String.class);
} else {
return expression;
}
}
@Override
public Predicate and(List predicates) {
return cb.and(predicates.toArray(new Predicate[predicates.size()]));
}
@Override
public Predicate not(Predicate predicate) {
return cb.not(predicate);
}
@Override
public Predicate or(List predicates) {
return cb.or(predicates.toArray(new Predicate[predicates.size()]));
}
@Override
public Expression> joinMapValue(Expression> currentCriteriaPath, MetaAttribute pathElement, Object key) {
MapJoin mapJoin = ((From, ?>) currentCriteriaPath).joinMap(pathElement.getName(),
JoinType.LEFT);
Predicate mapJoinCondition = cb.equal(mapJoin.key(), key);
Predicate nullCondition = cb.isNull(mapJoin.key());
addPredicate(cb.or(mapJoinCondition, nullCondition));
return mapJoin;
}
@Override
public Class> getJavaElementType(Expression> currentCriteriaPath) {
return currentCriteriaPath.getJavaType();
}
@Override
@SuppressWarnings({"rawtypes", "unchecked"})
public Expression> getAttribute(Expression> currentCriteriaPath, MetaAttribute pathElement) {
if (pathElement instanceof MetaComputedAttribute) {
MetaComputedAttribute projAttr = (MetaComputedAttribute) pathElement;
JpaCriteriaExpressionFactory expressionFactory = (JpaCriteriaExpressionFactory>) queryImpl.getComputedAttrs()
.get(projAttr);
From from = (From) currentCriteriaPath;
return expressionFactory.getExpression(from, getCriteriaQuery());
} else {
return ((Path>) currentCriteriaPath).get(pathElement.getName());
}
}
@SuppressWarnings({"unchecked", "rawtypes"})
@Override
public Expression> joinSubType(Expression> currentCriteriaPath, Class> entityType) {
return cb.treat((Path) currentCriteriaPath, entityType);
}
@SuppressWarnings("unchecked")
@Override
public From, ?> doJoin(MetaAttribute targetAttr, JoinType joinType, From, ?> parent) {
if (targetAttr instanceof MetaComputedAttribute) {
MetaComputedAttribute projAttr = (MetaComputedAttribute) targetAttr;
@SuppressWarnings("rawtypes")
JpaCriteriaExpressionFactory expressionFactory = (JpaCriteriaExpressionFactory>) queryImpl.getComputedAttrs()
.get(projAttr);
return (From, ?>) expressionFactory.getExpression(parent, getCriteriaQuery());
} else {
return parent.join(targetAttr.getName(), joinType);
}
}
}