com.google.gerrit.server.query.change.InternalChangeQuery Maven / Gradle / Ivy
// Copyright (C) 2014 The Android Open Source Project
//
// 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
//
// http://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.gerrit.server.query.change;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.gerrit.index.query.Predicate.and;
import static com.google.gerrit.index.query.Predicate.not;
import static com.google.gerrit.index.query.Predicate.or;
import static com.google.gerrit.server.query.change.ChangePredicates.EditByPredicateProvider;
import static com.google.gerrit.server.query.change.ChangeStatusPredicate.open;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.gerrit.common.UsedAt;
import com.google.gerrit.entities.Account;
import com.google.gerrit.entities.BranchNameKey;
import com.google.gerrit.entities.Change;
import com.google.gerrit.entities.Project;
import com.google.gerrit.entities.RefNames;
import com.google.gerrit.index.IndexConfig;
import com.google.gerrit.index.query.InternalQuery;
import com.google.gerrit.index.query.Predicate;
import com.google.gerrit.index.query.QueryParseException;
import com.google.gerrit.server.index.change.ChangeField;
import com.google.gerrit.server.index.change.ChangeIndexCollection;
import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.inject.Inject;
import com.google.inject.Provider;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.function.Supplier;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.Repository;
/**
 * Query wrapper for the change index.
 *
 * Instances are one-time-use. Other singleton classes should inject a Provider rather than
 * holding on to a single instance.
 */
public class InternalChangeQuery extends InternalQuery {
  private static Predicate ref(BranchNameKey branch) {
    return ChangePredicates.ref(branch.branch());
  }
  private static Predicate change(Change.Key key) {
    return ChangePredicates.idPrefix(key.get());
  }
  private static Predicate project(Project.NameKey project) {
    return ChangePredicates.project(project);
  }
  private static Predicate status(Change.Status status) {
    return ChangeStatusPredicate.forStatus(status);
  }
  private Predicate editBy(Account.Id accountId) throws QueryParseException {
    return editByPredicateProvider.editBy(accountId);
  }
  private static Predicate commit(String id) {
    return ChangePredicates.commitPrefix(id);
  }
  private final ChangeData.Factory changeDataFactory;
  private final ChangeNotes.Factory notesFactory;
  private final EditByPredicateProvider editByPredicateProvider;
  private final Provider queryBuilderArgsProvider;
  @Inject
  InternalChangeQuery(
      ChangeQueryProcessor queryProcessor,
      ChangeIndexCollection indexes,
      IndexConfig indexConfig,
      ChangeData.Factory changeDataFactory,
      ChangeNotes.Factory notesFactory,
      EditByPredicateProvider editByPredicateProvider,
      Provider queryBuilderArgsProvider) {
    super(queryProcessor, indexes, indexConfig);
    this.changeDataFactory = changeDataFactory;
    this.notesFactory = notesFactory;
    this.editByPredicateProvider = editByPredicateProvider;
    this.queryBuilderArgsProvider = queryBuilderArgsProvider;
  }
  public List byKey(Change.Key key) {
    return byKeyPrefix(key.get());
  }
  public List byKeyPrefix(String prefix) {
    return query(ChangePredicates.idPrefix(prefix));
  }
  public List byLegacyChangeId(Change.Id id) {
    return query(ChangePredicates.idStr(id));
  }
  public List byChangeNumber(Change.Id id) {
    return query(ChangePredicates.changeNumber(id, queryBuilderArgsProvider.get()));
  }
  @UsedAt(UsedAt.Project.GOOGLE)
  public List byLegacyChangeIds(Collection ids) {
    List> preds = new ArrayList<>(ids.size());
    for (Change.Id id : ids) {
      preds.add(ChangePredicates.idStr(id));
    }
    return query(or(preds));
  }
  @UsedAt(UsedAt.Project.GOOGLE)
  public List byCustomKeyedValue(String keyValue) {
    return query(new ChangeIndexPredicate(ChangeField.CUSTOM_KEYED_VALUES_SPEC, keyValue));
  }
  public List byBranchKey(BranchNameKey branch, Change.Key key) {
    return query(byBranchKeyPred(branch, key));
  }
  private static Predicate byBranchKeyPred(BranchNameKey branch, Change.Key key) {
    return and(ref(branch), project(branch.project()), change(key));
  }
  public List byProject(Project.NameKey project) {
    return query(project(project));
  }
  public List byBranchOpen(BranchNameKey branch) {
    return query(and(ref(branch), project(branch.project()), open()));
  }
  public List byBranchNew(BranchNameKey branch) {
    return query(and(ref(branch), project(branch.project()), status(Change.Status.NEW)));
  }
  public Iterable byCommitsOnBranchNotMerged(
      Repository repo, BranchNameKey branch, Collection hashes) throws IOException {
    return byCommitsOnBranchNotMerged(
        repo,
        branch,
        hashes,
        // Account for all commit predicates plus ref, project, status.
        indexConfig.maxTerms() - 3);
  }
  @VisibleForTesting
  Iterable byCommitsOnBranchNotMerged(
      Repository repo, BranchNameKey branch, Collection hashes, int indexLimit)
      throws IOException {
    if (hashes.size() > indexLimit || !indexes.getSearchIndex().isEnabled()) {
      return byCommitsOnBranchNotMergedFromDatabase(repo, branch, hashes);
    }
    return byCommitsOnBranchNotMergedFromIndex(branch, hashes);
  }
  private List byCommitsOnBranchNotMergedFromDatabase(
      Repository repo, BranchNameKey branch, Collection hashes) throws IOException {
    Set changeIds = Sets.newHashSetWithExpectedSize(hashes.size());
    String lastPrefix = null;
    for (Ref ref : repo.getRefDatabase().getRefsByPrefix(RefNames.REFS_CHANGES)) {
      String r = ref.getName();
      if ((lastPrefix != null && r.startsWith(lastPrefix))
          || !hashes.contains(ref.getObjectId().name())) {
        continue;
      }
      Change.Id id = Change.Id.fromRef(r);
      if (id == null) {
        continue;
      }
      if (changeIds.add(id)) {
        lastPrefix = r.substring(0, r.lastIndexOf('/'));
      }
    }
    List notes =
        notesFactory.create(
            repo,
            branch.project(),
            changeIds,
            cn -> {
              Change c = cn.getChange();
              return c.getDest().equals(branch) && !c.isMerged();
            });
    return Lists.transform(notes, n -> changeDataFactory.create(n));
  }
  private ImmutableList byCommitsOnBranchNotMergedFromIndex(
      BranchNameKey branch, Collection hashes) {
    return query(
        and(
            ref(branch),
            project(branch.project()),
            not(status(Change.Status.MERGED)),
            or(commits(hashes))));
  }
  private static List> commits(Collection hashes) {
    List> commits = new ArrayList<>(hashes.size());
    for (String s : hashes) {
      commits.add(commit(s));
    }
    return commits;
  }
  public List byProjectOpen(Project.NameKey project) {
    return query(and(project(project), open()));
  }
  public List byTopicOpen(String topic) {
    return query(and(ChangePredicates.exactTopic(topic), open()));
  }
  public List byOpenEditByUser(Account.Id accountId) throws QueryParseException {
    return query(editBy(accountId));
  }
  public List byCommit(ObjectId id) {
    return byCommit(id.name());
  }
  public List byCommit(String hash) {
    return query(commit(hash));
  }
  public List byProjectCommit(Project.NameKey project, ObjectId id) {
    return byProjectCommit(project, id.name());
  }
  public List byProjectCommit(Project.NameKey project, String hash) {
    return query(and(project(project), commit(hash)));
  }
  public List byProjectCommits(Project.NameKey project, List hashes) {
    int n = indexConfig.maxTerms() - 1;
    checkArgument(hashes.size() <= n, "cannot exceed %s commits", n);
    return query(and(project(project), or(commits(hashes))));
  }
  public List byBranchCommit(String project, String branch, String hash) {
    return query(byBranchCommitPred(project, branch, hash));
  }
  public List byBranchCommit(BranchNameKey branch, String hash) {
    return byBranchCommit(branch.project().get(), branch.branch(), hash);
  }
  public List byBranchCommitOpen(String project, String branch, String hash) {
    return query(and(byBranchCommitPred(project, branch, hash), open()));
  }
  public static Predicate byBranchCommitOpenPred(
      Project.NameKey project, String branch, String hash) {
    return and(byBranchCommitPred(project.get(), branch, hash), open());
  }
  private static Predicate byBranchCommitPred(
      String project, String branch, String hash) {
    return and(
        ChangePredicates.project(Project.nameKey(project)),
        ChangePredicates.ref(branch),
        commit(hash));
  }
  public List bySubmissionId(String cs) {
    if (Strings.isNullOrEmpty(cs)) {
      return Collections.emptyList();
    }
    return query(ChangePredicates.submissionId(cs));
  }
  private static Predicate byProjectGroupsPredicate(
      IndexConfig indexConfig, Project.NameKey project, Collection groups) {
    int n = indexConfig.maxTerms() - 1;
    checkArgument(groups.size() <= n, "cannot exceed %s groups", n);
    List groupPredicates = new ArrayList<>(groups.size());
    for (String g : groups) {
      groupPredicates.add(new GroupPredicate(g));
    }
    return and(project(project), or(groupPredicates));
  }
  public static ImmutableList byProjectGroups(
      Provider queryProvider,
      IndexConfig indexConfig,
      Project.NameKey project,
      Collection groups) {
    // These queries may be complex along multiple dimensions:
    //  * Many groups per change, if there are very many patch sets. This requires partitioning the
    //    list of predicates and combining results.
    //  * Many changes with the same set of groups, if the relation chain is very long. This
    //    requires querying exhaustively with pagination.
    // For both cases, we need to invoke the queryProvider multiple times, since each
    // InternalChangeQuery is single-use.
    Supplier querySupplier = () -> queryProvider.get().enforceVisibility(true);
    int batchSize = indexConfig.maxTerms() - 1;
    if (groups.size() <= batchSize) {
      return queryExhaustively(
          querySupplier, byProjectGroupsPredicate(indexConfig, project, groups));
    }
    Set seen = new HashSet<>();
    ImmutableList.Builder result = ImmutableList.builder();
    for (List part : Iterables.partition(groups, batchSize)) {
      for (ChangeData cd :
          queryExhaustively(querySupplier, byProjectGroupsPredicate(indexConfig, project, part))) {
        if (!seen.add(cd.virtualId())) {
          result.add(cd);
        }
      }
    }
    return result.build();
  }
}