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

com.google.gerrit.server.query.change.InternalChangeQuery Maven / Gradle / Ivy

There is a newer version: 3.11.0-rc3
Show newest version
// 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(); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy