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

com.google.appengine.api.datastore.PreparedQueryImpl Maven / Gradle / Ivy

/*
 * Copyright 2021 Google LLC
 *
 * 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 com.google.appengine.api.datastore;

import static com.google.appengine.api.datastore.FetchOptions.Builder.withLimit;
import static com.google.common.base.Preconditions.checkArgument;

import java.util.Iterator;
import java.util.List;

/**
 * Implements PreparedQuery.
 *
 */
class PreparedQueryImpl extends BasePreparedQuery {
  private final Query query;
  private final Transaction txn;
  private final QueryRunner queryRunner;

  public PreparedQueryImpl(Query query, Transaction txn, QueryRunner queryRunner) {
    this.query = query;
    this.txn = txn;
    this.queryRunner = queryRunner;

    // TODO Move this check and the one that follows into the
    // LocalDatastoreService (it may already be there).
    checkArgument(
        txn == null || query.getAncestor() != null,
        "Only ancestor queries are allowed inside transactions.");
    TransactionImpl.ensureTxnActive(txn);
  }

  private QueryResultIteratorImpl runQuery(FetchOptions fetchOptions) {
    return new QueryResultIteratorImpl(
        this, queryRunner.runQuery(fetchOptions, query, txn), fetchOptions, txn);
  }

  @Override
  public List asList(FetchOptions fetchOptions) {
    return new LazyList(runQuery(fetchOptions));
  }

  @Override
  public QueryResultList asQueryResultList(FetchOptions fetchOptions) {
    FetchOptions override = new FetchOptions(fetchOptions);
    if (override.getCompile() == null) {
      // default to compile so that cursors are supported by default
      override.compile(true);
    }
    LazyList lazyList = new LazyList(runQuery(override));
    return lazyList;
  }

  @Override
  public Iterator asIterator(FetchOptions fetchOptions) {
    return runQuery(fetchOptions);
  }

  @Override
  public QueryResultIterator asQueryResultIterator(FetchOptions fetchOptions) {
    if (fetchOptions.getCompile() == null) {
      // default to compile so that cursors are supported by default
      fetchOptions = new FetchOptions(fetchOptions).compile(true);
    }
    return runQuery(fetchOptions);
  }

  @Override
  public Entity asSingleEntity() throws TooManyResultsException {
    List entities = asList(withLimit(2));
    if (entities.isEmpty()) {
      return null;
    } else if (entities.size() != 1) {
      throw new TooManyResultsException();
    }
    return entities.get(0);
  }

  /**
   * Counts the number of entities in the result set.
   *
   * 

This method will run a query that will offset past all results and return the number of * results that have been skipped in the process. * *

(offset, limit) is converted to (offset + limit, 0) so that no results are actually * returned. The resulting count is max(0, skipped - offset). This is the number of entities in * the range [offset, offset + limit) which is the count. */ @Override public int countEntities(FetchOptions fetchOptions) { FetchOptions overrideOptions = new FetchOptions(fetchOptions); // Converting form (offset, limit) to (offset + limit, 0) // NOTE: Currently the datastore will automatically covert this // the query into a keys_only query because the limit is set to 0. overrideOptions.limit(0); if (fetchOptions.getLimit() != null) { if (fetchOptions.getOffset() != null) { // Using max to handle overflow int offset = fetchOptions.getLimit() + fetchOptions.getOffset(); overrideOptions.offset(offset >= 0 ? offset : Integer.MAX_VALUE); } else { overrideOptions.offset(fetchOptions.getLimit()); } } else { overrideOptions.offset(Integer.MAX_VALUE); } // Returning max(0, skipped - offset) int count = runQuery(overrideOptions).getNumSkipped(); if (fetchOptions.getOffset() != null) { // Count cannot be negative if (count < fetchOptions.getOffset()) { count = 0; } else { count = count - fetchOptions.getOffset(); } } return count; } @Override public String toString() { return query + (txn != null ? " IN " + txn : ""); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy