All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jwebmp.entityassist.querybuilder.builders.WhereExpression Maven / Gradle / Ivy

package com.jwebmp.entityassist.querybuilder.builders;

import com.jwebmp.entityassist.enumerations.Operand;
import com.jwebmp.logger.LogFactory;

import javax.persistence.criteria.CriteriaBuilder;
import javax.persistence.criteria.Expression;
import javax.persistence.criteria.From;
import javax.persistence.criteria.Predicate;
import javax.persistence.metamodel.Attribute;
import javax.persistence.metamodel.PluralAttribute;
import javax.persistence.metamodel.SingularAttribute;
import javax.validation.constraints.NotNull;
import java.util.Optional;
import java.util.logging.Logger;

final class WhereExpression
		implements IFilterExpression
{
	private static final Logger log = LogFactory.getLog("WhereExpression");

	private Attribute expressionAttribute;

	private Operand operand;
	private Object expressionValue;

	private From root;

	private CriteriaBuilder criteriaBuilder;

	WhereExpression()
	{
	}

	WhereExpression(Attribute expressionAttribute, Operand operand, Object expressionValue)
	{
		this.expressionAttribute = expressionAttribute;
		this.operand = operand;
		this.expressionValue = expressionValue;
	}

	@Override
	public int hashCode()
	{
		return super.hashCode();
	}

	@Override
	public boolean equals(Object obj)
	{
		return super.equals(obj);
	}

	@Override
	public Optional toPredicate(From entityRoot, CriteriaBuilder builder)
	{
		root = entityRoot;
		criteriaBuilder = builder;
		return processWhereExpression(this);
	}

	private Optional processWhereExpression(WhereExpression whereExpression)
	{
		Optional result;
		result = processWhereNulls(whereExpression);
		if (!result.isPresent())
		{
			result = processWhereEquals(whereExpression);
		}
		if (!result.isPresent())
		{
			result = processWhereLike(whereExpression);
		}
		if (!result.isPresent())
		{
			result = processWhereLists(whereExpression);
		}
		if (!result.isPresent())
		{
			result = processWhereCompare(whereExpression);
		}
		if (!result.isPresent())
		{
			log.severe("Unable to generate a where clause for the given expression");
		}
		return result;
	}

	@SuppressWarnings("unchecked")
	private Optional processWhereNulls(WhereExpression whereExpression)
	{
		Attribute attribute = whereExpression.getExpressionAttribute();
		switch (whereExpression.getOperand())
		{
			case Null:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().isNull(getRoot().get(SingularAttribute.class.cast(attribute))));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().isNull(getRoot().get(PluralAttribute.class.cast(attribute))));
				}
			}
			case NotNull:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().isNotNull(getRoot().get(SingularAttribute.class.cast(attribute))));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().isNotNull(getRoot().get(PluralAttribute.class.cast(attribute))));
				}
			}
			default:
			{
				return Optional.empty();
			}
		}
	}

	@SuppressWarnings("unchecked")
	private Optional processWhereEquals(WhereExpression whereExpression)
	{
		Attribute attribute = whereExpression.getExpressionAttribute();
		Object value = whereExpression.getExpressionValue();
		switch (whereExpression.getOperand())
		{
			case Equals:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().equal(getRoot().get(SingularAttribute.class.cast(attribute)), value));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().equal(getRoot().get(PluralAttribute.class.cast(attribute)), value));
				}
			}

			case NotEquals:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().notEqual(getRoot().get(SingularAttribute.class.cast(attribute)), value));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().notEqual(getRoot().get(PluralAttribute.class.cast(attribute)), value));
				}
			}
			default:
			{
				return Optional.empty();
			}
		}
	}

	@SuppressWarnings("unchecked")
	private Optional processWhereLike(WhereExpression whereExpression)
	{
		Attribute attribute = whereExpression.getExpressionAttribute();
		Object value = whereExpression.getExpressionValue();
		switch (whereExpression.getOperand())
		{
			case Like:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().like(getRoot().get(SingularAttribute.class.cast(attribute)), value.toString()));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().like(getRoot().get(PluralAttribute.class.cast(attribute)), value.toString()));
				}
			}
			case NotLike:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().notLike(getRoot().get(SingularAttribute.class.cast(attribute)), value.toString()));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().notLike(getRoot().get(PluralAttribute.class.cast(attribute)), value.toString()));
				}
			}
			default:
			{
				return Optional.empty();
			}
		}
	}

	@SuppressWarnings("unchecked")
	private Optional processWhereLists(WhereExpression whereExpression)
	{
		Attribute attribute = whereExpression.getExpressionAttribute();
		Object value = whereExpression.getExpressionValue();
		switch (whereExpression.getOperand())
		{
			case InList:
			{
				Expression path = null;
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					path = getRoot().get(SingularAttribute.class.cast(attribute));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					path = getRoot().get(PluralAttribute.class.cast(attribute));
				}
				CriteriaBuilder.In in = getCriteriaBuilder().in(path);
				IFilterExpression.buildInObject(in, value);
				return Optional.of(in);
			}
			case NotInList:
			{
				Expression path = null;
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					path = getRoot().get(SingularAttribute.class.cast(attribute));
				}
				else if (IFilterExpression.isPluralOrMapAttribute(attribute))
				{
					path = getRoot().get(PluralAttribute.class.cast(attribute));
				}
				CriteriaBuilder.In in = getCriteriaBuilder().in(path);
				IFilterExpression.buildInObject(in, value);
				return Optional.of(getCriteriaBuilder().not(in));
			}
			default:
			{
				return Optional.empty();
			}
		}
	}

	@SuppressWarnings("unchecked")
	@NotNull
	private > Optional processWhereCompare(WhereExpression whereExpression)
	{
		Attribute attribute = whereExpression.getExpressionAttribute();
		T value = (T) whereExpression.getExpressionValue();
		switch (whereExpression.getOperand())
		{
			case LessThan:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().lessThan(getRoot().get((SingularAttribute) attribute), value));
				}
			}
			case LessThanEqualTo:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().lessThanOrEqualTo(getRoot().get((SingularAttribute) attribute), value));
				}
			}
			case GreaterThan:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().greaterThan(getRoot().get((SingularAttribute) attribute), value));
				}
			}
			case GreaterThanEqualTo:
			{
				if (IFilterExpression.isSingularAttribute(attribute))
				{
					return Optional.of(getCriteriaBuilder().greaterThanOrEqualTo(getRoot().get((SingularAttribute) attribute), value));
				}
			}
			default:
			{
				return Optional.empty();
			}
		}
	}

	/**
	 * Returns the attribute associated
	 *
	 * @return The attribute
	 */
	public Attribute getExpressionAttribute()
	{
		return expressionAttribute;
	}

	/**
	 * Sets the attribute
	 *
	 * @param expressionAttribute
	 * 		The attribute
	 *
	 * @return This
	 */
	public WhereExpression setExpressionAttribute(Attribute expressionAttribute)
	{
		this.expressionAttribute = expressionAttribute;
		return this;
	}

	/**
	 * The applicable operand to apply
	 *
	 * @return The operand
	 */
	public Operand getOperand()
	{
		return operand;
	}

	/**
	 * Sets the operand to apply
	 *
	 * @param operand
	 * 		The operand
	 *
	 * @return This
	 */
	public WhereExpression setOperand(Operand operand)
	{
		this.operand = operand;
		return this;
	}

	/**
	 * Returns the sent in criteria builder set on toPredicate
	 *
	 * @return The builder
	 */
	private CriteriaBuilder getCriteriaBuilder()
	{
		return criteriaBuilder;
	}

	/**
	 * Gets the assigned root - only set on toPredicate
	 *
	 * @return The root or null
	 */
	private From getRoot()
	{
		return root;
	}

	/**
	 * Whatever the value object is
	 *
	 * @return The given value, nullable for isNull
	 */
	public Object getExpressionValue()
	{
		return expressionValue;
	}

	/**
	 * Whatever the value object is
	 *
	 * @param expressionValue
	 * 		The value, null for not and is null
	 *
	 * @return This
	 */
	public WhereExpression setExpressionValue(Object expressionValue)
	{
		this.expressionValue = expressionValue;
		return this;
	}
}