/*
 * Copyright 2008-2020 the original author or 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 org.springframework.data.repository.query.parser;

import java.util.Collections;
import java.util.Iterator;
import java.util.Optional;

import org.springframework.data.domain.Sort;
import org.springframework.data.repository.query.ParameterAccessor;
import org.springframework.data.repository.query.ParametersParameterAccessor;
import org.springframework.data.repository.query.parser.PartTree.OrPart;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;

Base class for query creators that create criteria based queries from a PartTree.
Author:Oliver Gierke, Mark Paluch, Christoph Strobl
Type parameters:
  • <T> – the actual query type to be created
  • <S> – the intermediate criteria type
/** * Base class for query creators that create criteria based queries from a {@link PartTree}. * * @param <T> the actual query type to be created * @param <S> the intermediate criteria type * @author Oliver Gierke * @author Mark Paluch * @author Christoph Strobl */
public abstract class AbstractQueryCreator<T, S> { private final Optional<ParameterAccessor> parameters; private final PartTree tree;
Creates a new AbstractQueryCreator for the given PartTree. This will cause null be handed for the Iterator in the callback methods.
Params:
  • tree – must not be null.
Since:2.0
/** * Creates a new {@link AbstractQueryCreator} for the given {@link PartTree}. This will cause {@literal null} be * handed for the {@link Iterator} in the callback methods. * * @param tree must not be {@literal null}. * @since 2.0 */
public AbstractQueryCreator(PartTree tree) { this(tree, Optional.empty()); }
Creates a new AbstractQueryCreator for the given PartTree and ParametersParameterAccessor. The latter is used to hand actual parameter values into the callback methods as well as to apply dynamic sorting via a Sort parameter.
Params:
  • tree – must not be null.
  • parameters – must not be null.
/** * Creates a new {@link AbstractQueryCreator} for the given {@link PartTree} and {@link ParametersParameterAccessor}. * The latter is used to hand actual parameter values into the callback methods as well as to apply dynamic sorting * via a {@link Sort} parameter. * * @param tree must not be {@literal null}. * @param parameters must not be {@literal null}. */
public AbstractQueryCreator(PartTree tree, ParameterAccessor parameters) { this(tree, Optional.of(parameters)); } private AbstractQueryCreator(PartTree tree, Optional<ParameterAccessor> parameters) { Assert.notNull(tree, "PartTree must not be null"); Assert.notNull(parameters, "ParameterAccessor must not be null"); this.tree = tree; this.parameters = parameters; }
Creates the actual query object.
Returns:
/** * Creates the actual query object. * * @return */
public T createQuery() { return createQuery(parameters.map(ParameterAccessor::getSort) // .orElse(Sort.unsorted())); }
Creates the actual query object applying the given Sort parameter. Use this method in case you haven't provided a ParameterAccessor in the first place but want to apply dynamic sorting nevertheless.
Params:
  • dynamicSort – must not be null.
Returns:
/** * Creates the actual query object applying the given {@link Sort} parameter. Use this method in case you haven't * provided a {@link ParameterAccessor} in the first place but want to apply dynamic sorting nevertheless. * * @param dynamicSort must not be {@literal null}. * @return */
public T createQuery(Sort dynamicSort) { Assert.notNull(dynamicSort, "DynamicSort must not be null!"); return complete(createCriteria(tree), tree.getSort().and(dynamicSort)); }
Actual query building logic. Traverses the PartTree and invokes callback methods to delegate actual criteria creation and concatenation.
Params:
  • tree – must not be null.
Returns:
/** * Actual query building logic. Traverses the {@link PartTree} and invokes callback methods to delegate actual * criteria creation and concatenation. * * @param tree must not be {@literal null}. * @return */
@Nullable private S createCriteria(PartTree tree) { S base = null; Iterator<Object> iterator = parameters.map(ParameterAccessor::iterator).orElse(Collections.emptyIterator()); for (OrPart node : tree) { Iterator<Part> parts = node.iterator(); if (!parts.hasNext()) { throw new IllegalStateException(String.format("No part found in PartTree %s!", tree)); } S criteria = create(parts.next(), iterator); while (parts.hasNext()) { criteria = and(parts.next(), criteria, iterator); } base = base == null ? criteria : or(base, criteria); } return base; }
Creates a new atomic instance of the criteria object.
Params:
  • part – must not be null.
  • iterator – must not be null.
Returns:
/** * Creates a new atomic instance of the criteria object. * * @param part must not be {@literal null}. * @param iterator must not be {@literal null}. * @return */
protected abstract S create(Part part, Iterator<Object> iterator);
Creates a new criteria object from the given part and and-concatenates it to the given base criteria.
Params:
  • part – must not be null.
  • base – will never be null.
  • iterator – must not be null.
Returns:
/** * Creates a new criteria object from the given part and and-concatenates it to the given base criteria. * * @param part must not be {@literal null}. * @param base will never be {@literal null}. * @param iterator must not be {@literal null}. * @return */
protected abstract S and(Part part, S base, Iterator<Object> iterator);
Or-concatenates the given base criteria to the given new criteria.
Params:
  • base – must not be null.
  • criteria – must not be null.
Returns:
/** * Or-concatenates the given base criteria to the given new criteria. * * @param base must not be {@literal null}. * @param criteria must not be {@literal null}. * @return */
protected abstract S or(S base, S criteria);
Actually creates the query object applying the given criteria object and Sort definition.
Params:
  • criteria – can be null.
  • sort – must not be null.
Returns:
/** * Actually creates the query object applying the given criteria object and {@link Sort} definition. * * @param criteria can be {@literal null}. * @param sort must not be {@literal null}. * @return */
protected abstract T complete(@Nullable S criteria, Sort sort); }