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

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

The newest version!
// Copyright (C) 2021 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.collect.ImmutableSet.toImmutableSet;

import com.google.common.base.CharMatcher;
import com.google.common.collect.ImmutableSet;
import com.google.gerrit.common.UsedAt;
import com.google.gerrit.entities.Account;
import com.google.gerrit.entities.Change;
import com.google.gerrit.entities.PatchSet;
import com.google.gerrit.entities.Project;
import com.google.gerrit.git.ObjectIds;
import com.google.gerrit.index.query.Predicate;
import com.google.gerrit.index.query.QueryParseException;
import com.google.gerrit.server.DraftCommentsReader;
import com.google.gerrit.server.StarredChangesReader;
import com.google.gerrit.server.change.HashtagsUtil;
import com.google.gerrit.server.index.change.ChangeField;
import com.google.inject.ImplementedBy;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Locale;

/** Predicates that match against {@link ChangeData}. */
public class ChangePredicates {
  private ChangePredicates() {}

  /**
   * Returns a predicate that matches changes where the provided {@link
   * com.google.gerrit.entities.Account.Id} is in the attention set.
   */
  public static Predicate attentionSet(Account.Id id) {
    return new ChangeIndexPredicate(ChangeField.ATTENTION_SET_USERS, id.toString());
  }

  /**
   * Returns a predicate that matches changes that are a revert of the provided {@link
   * com.google.gerrit.entities.Change.Id}.
   */
  public static Predicate revertOf(Change.Id revertOf) {
    return new ChangeIndexCardinalPredicate(ChangeField.REVERT_OF, revertOf.toString(), 1);
  }

  /**
   * Returns a predicate that matches changes that have a comment authored by the provided {@link
   * com.google.gerrit.entities.Account.Id}.
   */
  public static Predicate commentBy(Account.Id id) {
    return new ChangeIndexPredicate(ChangeField.COMMENTBY_SPEC, id.toString());
  }

  @ImplementedBy(IndexEditByPredicateProvider.class)
  public interface EditByPredicateProvider {

    /**
     * Returns a predicate that matches changes where the provided {@link
     * com.google.gerrit.entities.Account.Id} has a pending change edit.
     */
    Predicate editBy(Account.Id id) throws QueryParseException;
  }

  /**
   * A default implementation of {@link EditByPredicateProvider}, based on th {@link
   * ChangeField#EDITBY_SPEC} index field.
   */
  public static class IndexEditByPredicateProvider implements EditByPredicateProvider {
    @Override
    public Predicate editBy(Account.Id id) {
      return new ChangeIndexPredicate(ChangeField.EDITBY_SPEC, id.toString());
    }
  }

  /**
   * Returns a predicate that matches changes where the provided {@link
   * com.google.gerrit.entities.Account.Id} has a pending draft comment.
   *
   * 

The predicates filter by "legacy_id_str" field. */ @UsedAt(UsedAt.Project.GOOGLE) public static Predicate draftBy( DraftCommentsReader draftCommentsReader, Account.Id id) { ImmutableSet> changeIdPredicates = draftCommentsReader.getChangesWithDrafts(id).stream() .map(ChangePredicates::idStr) .collect(toImmutableSet()); return changeIdPredicates.isEmpty() ? ChangeIndexPredicate.none() : Predicate.or(changeIdPredicates); } /** * Returns a predicate that matches changes where the provided {@link * com.google.gerrit.entities.Account.Id} has starred changes with {@code label}. * *

The predicates filter by "legacy_id_str" field. */ @UsedAt(UsedAt.Project.GOOGLE) public static Predicate starBy( StarredChangesReader starredChangesReader, Account.Id id) { ImmutableSet> starredChanges = starredChangesReader.byAccountId(id).stream() .map(ChangePredicates::idStr) .collect(toImmutableSet()); return starredChanges.isEmpty() ? ChangeIndexPredicate.none() : Predicate.or(starredChanges); } /** * Returns a predicate that matches changes that were reviewed by any of the provided {@link * com.google.gerrit.entities.Account.Id}. */ public static Predicate reviewedBy(Collection ids) { List> predicates = new ArrayList<>(ids.size()); for (Account.Id id : ids) { predicates.add(new ChangeIndexPredicate(ChangeField.REVIEWEDBY_SPEC, id.toString())); } return Predicate.or(predicates); } /** Returns a predicate that matches changes that were not yet reviewed. */ public static Predicate unreviewed() { return Predicate.not( new ChangeIndexPredicate(ChangeField.REVIEWEDBY_SPEC, ChangeField.NOT_REVIEWED.toString())); } /** * Returns a predicate that matches the change with the provided {@link * com.google.gerrit.entities.Change.Id}. */ public static Predicate idStr(Change.Id id) { return idStr(id.toString()); } public static Predicate idStr(String id) { return new ChangeIndexCardinalPredicate( ChangeField.NUMERIC_ID_STR_SPEC, ChangeQueryBuilder.FIELD_CHANGE, id, 1); } /** * Returns a predicate that matches the change number with the provided {@link * com.google.gerrit.entities.Change.Id}. */ public static Predicate changeNumber( Change.Id id, ChangeQueryBuilder.Arguments args) { if (args.getSchema().hasField(ChangeField.CHANGENUM_SPEC)) { return new ChangeIndexCardinalPredicate( ChangeField.CHANGENUM_SPEC, ChangeQueryBuilder.FIELD_CHANGE, id.toString(), 1); } return idStr(id); } /** * Returns a predicate that matches changes owned by the provided {@link * com.google.gerrit.entities.Account.Id}. */ public static Predicate owner(Account.Id id) { return new ChangeIndexCardinalPredicate(ChangeField.OWNER_SPEC, id.toString(), 5000); } /** * Returns a predicate that matches changes where the latest patch set was uploaded by the * provided {@link com.google.gerrit.entities.Account.Id}. */ public static Predicate uploader(Account.Id id) { return new ChangeIndexPredicate(ChangeField.UPLOADER_SPEC, id.toString()); } /** * Returns a predicate that matches changes that are a cherry pick of the provided {@link * com.google.gerrit.entities.Change.Id}. */ public static Predicate cherryPickOf(Change.Id id) { return new ChangeIndexPredicate(ChangeField.CHERRY_PICK_OF_CHANGE, id.toString()); } /** * Returns a predicate that matches changes that are a cherry pick of the provided {@link * com.google.gerrit.entities.PatchSet.Id}. */ public static Predicate cherryPickOf(PatchSet.Id psId) { return Predicate.and( cherryPickOf(psId.changeId()), new ChangeIndexPredicate(ChangeField.CHERRY_PICK_OF_PATCHSET, String.valueOf(psId.get()))); } /** * Returns a predicate that matches changes in the provided {@link * com.google.gerrit.entities.Project.NameKey}. */ public static Predicate project(Project.NameKey id) { return new ChangeIndexCardinalPredicate(ChangeField.PROJECT_SPEC, id.get(), 1_000_000); } /** Returns a predicate that matches changes targeted at the provided {@code refName}. */ public static Predicate ref(String refName) { return new ChangeIndexCardinalPredicate(ChangeField.REF_SPEC, refName, 10_000); } /** Returns a predicate that matches changes in the provided {@code topic}. */ public static Predicate exactTopic(String topic) { return new ChangeIndexPredicate(ChangeField.EXACT_TOPIC, topic); } /** Returns a predicate that matches changes in the provided {@code topic}. */ public static Predicate fuzzyTopic(String topic) { return new ChangeIndexPredicate(ChangeField.FUZZY_TOPIC, topic); } /** Returns a predicate that matches changes in the provided {@code topic}. Used with prefixes */ public static Predicate prefixTopic(String topic) { return new ChangeIndexPredicate(ChangeField.PREFIX_TOPIC, topic); } /** Returns a predicate that matches changes submitted in the provided {@code changeSet}. */ public static Predicate submissionId(String changeSet) { return new ChangeIndexPredicate(ChangeField.SUBMISSIONID_SPEC, changeSet); } /** Returns a predicate that matches changes that modified the provided {@code path}. */ public static Predicate path(String path) { return new ChangeIndexPredicate(ChangeField.PATH_SPEC, path); } /** Returns a predicate that matches changes tagged with the provided {@code hashtag}. */ public static Predicate hashtag(String hashtag) { // Use toLowerCase without locale to match behavior in ChangeField. return new ChangeIndexPredicate( ChangeField.HASHTAG_SPEC, HashtagsUtil.cleanupHashtag(hashtag).toLowerCase(Locale.US)); } /** Returns a predicate that matches changes tagged with the provided {@code hashtag}. */ public static Predicate fuzzyHashtag(String hashtag) { // Use toLowerCase without locale to match behavior in ChangeField. return new ChangeIndexPredicate( ChangeField.FUZZY_HASHTAG, HashtagsUtil.cleanupHashtag(hashtag).toLowerCase(Locale.US)); } /** * Returns a predicate that matches changes in the provided {@code hashtag}. Used with prefixes */ public static Predicate prefixHashtag(String hashtag) { // Use toLowerCase without locale to match behavior in ChangeField. return new ChangeIndexPredicate( ChangeField.PREFIX_HASHTAG, HashtagsUtil.cleanupHashtag(hashtag).toLowerCase(Locale.US)); } /** Returns a predicate that matches changes that modified the provided {@code file}. */ public static Predicate file(ChangeQueryBuilder.Arguments args, String file) { Predicate eqPath = path(file); if (!args.getSchema().hasField(ChangeField.FILE_PART_SPEC)) { return eqPath; } return Predicate.or(eqPath, new ChangeIndexPredicate(ChangeField.FILE_PART_SPEC, file)); } /** * Returns a predicate that matches changes with the provided {@code footer} in their commit * message. */ public static Predicate footer(String footer) { int indexEquals = footer.indexOf('='); int indexColon = footer.indexOf(':'); // footer key cannot contain '=' if (indexEquals > 0 && (indexEquals < indexColon || indexColon < 0)) { footer = footer.substring(0, indexEquals) + ": " + footer.substring(indexEquals + 1); } return new ChangeIndexPredicate(ChangeField.FOOTER_SPEC, footer.toLowerCase(Locale.US)); } /** * Returns a predicate that matches changes with the provided {@code footer} name in their commit * message. */ public static Predicate hasFooter(String footerName) { return new ChangeIndexPredicate(ChangeField.FOOTER_NAME, footerName); } /** * Returns a predicate that matches changes that modified files in the provided {@code directory}. */ public static Predicate directory(String directory) { return new ChangeIndexPredicate( ChangeField.DIRECTORY_SPEC, CharMatcher.is('/').trimFrom(directory).toLowerCase(Locale.US)); } /** Returns a predicate that matches changes with the provided {@code trackingId}. */ public static Predicate trackingId(String trackingId) { return new ChangeIndexCardinalPredicate(ChangeField.TR_SPEC, trackingId, 5); } /** Returns a predicate that matches changes authored by the provided {@code exactAuthor}. */ public static Predicate exactAuthor(String exactAuthor) { return new ChangeIndexPredicate( ChangeField.EXACT_AUTHOR_SPEC, exactAuthor.toLowerCase(Locale.US)); } /** Returns a predicate that matches changes authored by the provided {@code author}. */ public static Predicate author(String author) { return new ChangeIndexPredicate(ChangeField.AUTHOR_PARTS_SPEC, author); } /** * Returns a predicate that matches changes where the patch set was committed by {@code * exactCommitter}. */ public static Predicate exactCommitter(String exactCommitter) { return new ChangeIndexPredicate( ChangeField.EXACT_COMMITTER_SPEC, exactCommitter.toLowerCase(Locale.US)); } /** * Returns a predicate that matches changes where the patch set was committed by {@code * committer}. */ public static Predicate committer(String comitter) { return new ChangeIndexPredicate( ChangeField.COMMITTER_PARTS_SPEC, comitter.toLowerCase(Locale.US)); } /** Returns a predicate that matches changes whose ID starts with the provided {@code id}. */ public static Predicate idPrefix(String id) { return new ChangeIndexCardinalPredicate(ChangeField.CHANGE_ID_SPEC, id, 5); } /** * Returns a predicate that matches changes in a project that has the provided {@code prefix} in * its name. */ public static Predicate projectPrefix(String prefix) { return new ChangeIndexPredicate(ChangeField.PROJECTS_SPEC, prefix); } /** * Returns a predicate that matches changes where a patch set has the provided {@code commitId} * either as prefix or as full {@link org.eclipse.jgit.lib.ObjectId}. */ public static Predicate commitPrefix(String commitId) { if (commitId.length() == ObjectIds.STR_LEN) { return new ChangeIndexCardinalPredicate(ChangeField.EXACT_COMMIT_SPEC, commitId, 5); } return new ChangeIndexCardinalPredicate(ChangeField.COMMIT_SPEC, commitId, 5); } /** * Returns a predicate that matches changes where the provided {@code message} appears in the * commit message. Uses full-text search semantics. */ public static Predicate message(String message) { return new ChangeIndexPredicate(ChangeField.COMMIT_MESSAGE, message); } public static Predicate subject(String subject) { return new ChangeIndexPredicate(ChangeField.SUBJECT_SPEC, subject); } public static Predicate prefixSubject(String subject) { return new ChangeIndexPredicate(ChangeField.PREFIX_SUBJECT_SPEC, subject); } /** * Returns a predicate that matches changes where the provided {@code comment} appears in any * comment on any patch set of the change. Uses full-text search semantics. */ public static Predicate comment(String comment) { return new ChangeIndexPredicate(ChangeField.COMMENT_SPEC, comment); } /** * Returns a predicate that matches with changes having a specific submit rule evaluating to a * certain result. Value should be in the form of "$ruleName=$status" with $ruleName equals to * '$plugin_name~$rule_name' and $rule_name equals to the name of the class that implements the * {@link com.google.gerrit.server.rules.SubmitRule}. For gerrit core rules, $ruleName should be * in the form of 'gerrit~$rule_name'. */ public static Predicate submitRuleStatus(String value) { return new ChangeIndexPredicate(ChangeField.SUBMIT_RULE_RESULT_SPEC, value); } /** * Returns a predicate that matches with changes that are pure reverts if {@code value} is equal * to "1", or non-pure reverts if {@code value} is "0". */ public static Predicate pureRevert(String value) { return new ChangeIndexPredicate(ChangeField.IS_PURE_REVERT_SPEC, value); } /** * Returns a predicate that matches with changes that are submittable if {@code value} is equal to * "1", or non-submittable if {@code value} is "0". * *

The computation of this field is based on the evaluation of {@link * com.google.gerrit.entities.SubmitRequirement}s. */ public static Predicate isSubmittable(String value) { return new ChangeIndexPredicate(ChangeField.IS_SUBMITTABLE_SPEC, value); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy