|  | // Copyright (C) 2009 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.gerrit.server.project.ProjectCache.illegalState; | 
|  | import static java.util.Objects.requireNonNull; | 
|  | import static java.util.stream.Collectors.toList; | 
|  | import static java.util.stream.Collectors.toMap; | 
|  |  | 
|  | import com.google.auto.value.AutoValue; | 
|  | import com.google.common.annotations.VisibleForTesting; | 
|  | import com.google.common.base.MoreObjects; | 
|  | import com.google.common.collect.HashBasedTable; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import com.google.common.collect.ImmutableListMultimap; | 
|  | import com.google.common.collect.ImmutableMap; | 
|  | import com.google.common.collect.ImmutableSet; | 
|  | import com.google.common.collect.ImmutableSetMultimap; | 
|  | import com.google.common.collect.ImmutableSortedSet; | 
|  | import com.google.common.collect.Iterables; | 
|  | import com.google.common.collect.ListMultimap; | 
|  | import com.google.common.collect.Lists; | 
|  | import com.google.common.collect.Maps; | 
|  | import com.google.common.collect.SetMultimap; | 
|  | import com.google.common.collect.Table; | 
|  | import com.google.common.flogger.FluentLogger; | 
|  | import com.google.common.primitives.Ints; | 
|  | import com.google.gerrit.common.Nullable; | 
|  | import com.google.gerrit.entities.Account; | 
|  | import com.google.gerrit.entities.AttentionSetUpdate; | 
|  | import com.google.gerrit.entities.Change; | 
|  | import com.google.gerrit.entities.ChangeMessage; | 
|  | import com.google.gerrit.entities.Comment; | 
|  | import com.google.gerrit.entities.HumanComment; | 
|  | import com.google.gerrit.entities.LabelTypes; | 
|  | import com.google.gerrit.entities.PatchSet; | 
|  | import com.google.gerrit.entities.PatchSetApproval; | 
|  | import com.google.gerrit.entities.Project; | 
|  | import com.google.gerrit.entities.Project.NameKey; | 
|  | import com.google.gerrit.entities.RefNames; | 
|  | import com.google.gerrit.entities.RobotComment; | 
|  | import com.google.gerrit.entities.SubmitRecord; | 
|  | import com.google.gerrit.entities.SubmitRequirement; | 
|  | import com.google.gerrit.entities.SubmitRequirementResult; | 
|  | import com.google.gerrit.entities.SubmitTypeRecord; | 
|  | import com.google.gerrit.exceptions.StorageException; | 
|  | import com.google.gerrit.extensions.restapi.BadRequestException; | 
|  | import com.google.gerrit.extensions.restapi.ResourceConflictException; | 
|  | import com.google.gerrit.index.RefState; | 
|  | import com.google.gerrit.server.ChangeMessagesUtil; | 
|  | import com.google.gerrit.server.CommentsUtil; | 
|  | import com.google.gerrit.server.CurrentUser; | 
|  | import com.google.gerrit.server.PatchSetUtil; | 
|  | import com.google.gerrit.server.ReviewerByEmailSet; | 
|  | import com.google.gerrit.server.ReviewerSet; | 
|  | import com.google.gerrit.server.ReviewerStatusUpdate; | 
|  | import com.google.gerrit.server.StarredChangesUtil; | 
|  | import com.google.gerrit.server.StarredChangesUtil.StarRef; | 
|  | import com.google.gerrit.server.approval.ApprovalsUtil; | 
|  | import com.google.gerrit.server.change.CommentThread; | 
|  | import com.google.gerrit.server.change.CommentThreads; | 
|  | import com.google.gerrit.server.change.MergeabilityCache; | 
|  | import com.google.gerrit.server.change.PureRevert; | 
|  | import com.google.gerrit.server.config.AllUsersName; | 
|  | import com.google.gerrit.server.config.TrackingFooters; | 
|  | import com.google.gerrit.server.git.GitRepositoryManager; | 
|  | import com.google.gerrit.server.git.MergeUtil; | 
|  | import com.google.gerrit.server.notedb.ChangeNotes; | 
|  | import com.google.gerrit.server.notedb.RobotCommentNotes; | 
|  | import com.google.gerrit.server.patch.DiffSummary; | 
|  | import com.google.gerrit.server.patch.DiffSummaryKey; | 
|  | import com.google.gerrit.server.patch.PatchListCache; | 
|  | import com.google.gerrit.server.patch.PatchListKey; | 
|  | import com.google.gerrit.server.patch.PatchListNotAvailableException; | 
|  | import com.google.gerrit.server.project.NoSuchChangeException; | 
|  | import com.google.gerrit.server.project.ProjectCache; | 
|  | import com.google.gerrit.server.project.ProjectState; | 
|  | import com.google.gerrit.server.project.SubmitRequirementsAdapter; | 
|  | import com.google.gerrit.server.project.SubmitRequirementsEvaluator; | 
|  | import com.google.gerrit.server.project.SubmitRequirementsUtil; | 
|  | import com.google.gerrit.server.project.SubmitRuleEvaluator; | 
|  | import com.google.gerrit.server.project.SubmitRuleOptions; | 
|  | import com.google.gerrit.server.util.time.TimeUtil; | 
|  | import com.google.inject.Inject; | 
|  | import com.google.inject.assistedinject.Assisted; | 
|  | import java.io.IOException; | 
|  | import java.time.Instant; | 
|  | import java.util.ArrayList; | 
|  | import java.util.Collection; | 
|  | import java.util.Collections; | 
|  | import java.util.HashMap; | 
|  | import java.util.LinkedHashSet; | 
|  | import java.util.List; | 
|  | import java.util.Map; | 
|  | import java.util.Optional; | 
|  | import java.util.Set; | 
|  | import java.util.function.Function; | 
|  | import java.util.stream.Collectors; | 
|  | import java.util.stream.Stream; | 
|  | import org.eclipse.jgit.lib.ObjectId; | 
|  | import org.eclipse.jgit.lib.PersonIdent; | 
|  | import org.eclipse.jgit.lib.Ref; | 
|  | import org.eclipse.jgit.lib.Repository; | 
|  | import org.eclipse.jgit.revwalk.FooterLine; | 
|  | import org.eclipse.jgit.revwalk.RevCommit; | 
|  | import org.eclipse.jgit.revwalk.RevWalk; | 
|  |  | 
|  | /** | 
|  | * ChangeData provides lazily loaded interface to change metadata loaded from NoteDb. It can be | 
|  | * constructed by loading from NoteDb, or calling setters. The latter happens when ChangeData is | 
|  | * retrieved through the change index. This happens for Applications that are performance sensitive | 
|  | * (eg. dashboard loads, git protocol negotiation) but can tolerate staleness. In that case, setting | 
|  | * lazyLoad=false disables loading from NoteDb, so we don't accidentally enable a slow path. | 
|  | */ | 
|  | public class ChangeData { | 
|  | private static final FluentLogger logger = FluentLogger.forEnclosingClass(); | 
|  |  | 
|  | public enum StorageConstraint { | 
|  | /** | 
|  | * This instance was loaded from the change index. Backfilling missing data from NoteDb is not | 
|  | * allowed. | 
|  | */ | 
|  | INDEX_ONLY, | 
|  | /** | 
|  | * This instance was loaded from the change index. Backfilling missing data from NoteDb is | 
|  | * allowed. | 
|  | */ | 
|  | INDEX_PRIMARY_NOTEDB_SECONDARY, | 
|  | /** This instance was loaded from NoteDb. */ | 
|  | NOTEDB_ONLY | 
|  | } | 
|  |  | 
|  | public static List<Change> asChanges(List<ChangeData> changeDatas) { | 
|  | List<Change> result = new ArrayList<>(changeDatas.size()); | 
|  | for (ChangeData cd : changeDatas) { | 
|  | result.add(cd.change()); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | public static Map<Change.Id, ChangeData> asMap(List<ChangeData> changes) { | 
|  | return changes.stream().collect(toMap(ChangeData::getId, Function.identity())); | 
|  | } | 
|  |  | 
|  | public static void ensureChangeLoaded(Iterable<ChangeData> changes) { | 
|  | ChangeData first = Iterables.getFirst(changes, null); | 
|  | if (first == null) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (ChangeData cd : changes) { | 
|  | cd.change(); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void ensureAllPatchSetsLoaded(Iterable<ChangeData> changes) { | 
|  | ChangeData first = Iterables.getFirst(changes, null); | 
|  | if (first == null) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (ChangeData cd : changes) { | 
|  | cd.patchSets(); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void ensureCurrentPatchSetLoaded(Iterable<ChangeData> changes) { | 
|  | ChangeData first = Iterables.getFirst(changes, null); | 
|  | if (first == null) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (ChangeData cd : changes) { | 
|  | cd.currentPatchSet(); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void ensureCurrentApprovalsLoaded(Iterable<ChangeData> changes) { | 
|  | ChangeData first = Iterables.getFirst(changes, null); | 
|  | if (first == null) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (ChangeData cd : changes) { | 
|  | cd.currentApprovals(); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void ensureMessagesLoaded(Iterable<ChangeData> changes) { | 
|  | ChangeData first = Iterables.getFirst(changes, null); | 
|  | if (first == null) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (ChangeData cd : changes) { | 
|  | cd.messages(); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static void ensureReviewedByLoadedForOpenChanges(Iterable<ChangeData> changes) { | 
|  | List<ChangeData> pending = new ArrayList<>(); | 
|  | for (ChangeData cd : changes) { | 
|  | if (cd.reviewedBy == null && cd.change().isNew()) { | 
|  | pending.add(cd); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!pending.isEmpty()) { | 
|  | ensureAllPatchSetsLoaded(pending); | 
|  | ensureMessagesLoaded(pending); | 
|  | for (ChangeData cd : pending) { | 
|  | cd.reviewedBy(); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | public static class Factory { | 
|  | private final AssistedFactory assistedFactory; | 
|  |  | 
|  | @Inject | 
|  | Factory(AssistedFactory assistedFactory) { | 
|  | this.assistedFactory = assistedFactory; | 
|  | } | 
|  |  | 
|  | public ChangeData create(Project.NameKey project, Change.Id id) { | 
|  | return assistedFactory.create(project, id, null, null); | 
|  | } | 
|  |  | 
|  | public ChangeData create(Change change) { | 
|  | return assistedFactory.create(change.getProject(), change.getId(), change, null); | 
|  | } | 
|  |  | 
|  | public ChangeData create(ChangeNotes notes) { | 
|  | return assistedFactory.create( | 
|  | notes.getChange().getProject(), notes.getChangeId(), notes.getChange(), notes); | 
|  | } | 
|  | } | 
|  |  | 
|  | public interface AssistedFactory { | 
|  | ChangeData create( | 
|  | Project.NameKey project, | 
|  | Change.Id id, | 
|  | @Nullable Change change, | 
|  | @Nullable ChangeNotes notes); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Create an instance for testing only. | 
|  | * | 
|  | * <p>Attempting to lazy load data will fail with NPEs. Callers may consider manually setting | 
|  | * fields that can be set. | 
|  | * | 
|  | * @param id change ID | 
|  | * @return instance for testing. | 
|  | */ | 
|  | public static ChangeData createForTest( | 
|  | Project.NameKey project, Change.Id id, int currentPatchSetId, ObjectId commitId) { | 
|  | ChangeData cd = | 
|  | new ChangeData( | 
|  | null, null, null, null, null, null, null, null, null, null, null, null, null, null, | 
|  | null, null, null, project, id, null, null); | 
|  | cd.currentPatchSet = | 
|  | PatchSet.builder() | 
|  | .id(PatchSet.id(id, currentPatchSetId)) | 
|  | .commitId(commitId) | 
|  | .uploader(Account.id(1000)) | 
|  | .createdOn(TimeUtil.now()) | 
|  | .build(); | 
|  | return cd; | 
|  | } | 
|  |  | 
|  | // Injected fields. | 
|  | private @Nullable final StarredChangesUtil starredChangesUtil; | 
|  | private final AllUsersName allUsersName; | 
|  | private final ApprovalsUtil approvalsUtil; | 
|  | private final ChangeMessagesUtil cmUtil; | 
|  | private final ChangeNotes.Factory notesFactory; | 
|  | private final CommentsUtil commentsUtil; | 
|  | private final GitRepositoryManager repoManager; | 
|  | private final MergeUtil.Factory mergeUtilFactory; | 
|  | private final MergeabilityCache mergeabilityCache; | 
|  | private final PatchListCache patchListCache; | 
|  | private final PatchSetUtil psUtil; | 
|  | private final ProjectCache projectCache; | 
|  | private final TrackingFooters trackingFooters; | 
|  | private final PureRevert pureRevert; | 
|  | private final SubmitRequirementsEvaluator submitRequirementsEvaluator; | 
|  | private final SubmitRequirementsUtil submitRequirementsUtil; | 
|  | private final SubmitRuleEvaluator.Factory submitRuleEvaluatorFactory; | 
|  |  | 
|  | // Required assisted injected fields. | 
|  | private final Project.NameKey project; | 
|  | private final Change.Id legacyId; | 
|  |  | 
|  | // Lazily populated fields, including optional assisted injected fields. | 
|  |  | 
|  | private final Map<SubmitRuleOptions, List<SubmitRecord>> submitRecords = | 
|  | Maps.newLinkedHashMapWithExpectedSize(1); | 
|  |  | 
|  | private Map<SubmitRequirement, SubmitRequirementResult> submitRequirements; | 
|  |  | 
|  | private StorageConstraint storageConstraint = StorageConstraint.NOTEDB_ONLY; | 
|  | private Change change; | 
|  | private ChangeNotes notes; | 
|  | private String commitMessage; | 
|  | private List<FooterLine> commitFooters; | 
|  | private PatchSet currentPatchSet; | 
|  | private Collection<PatchSet> patchSets; | 
|  | private ListMultimap<PatchSet.Id, PatchSetApproval> allApprovals; | 
|  | private List<PatchSetApproval> currentApprovals; | 
|  | private List<String> currentFiles; | 
|  | private Optional<DiffSummary> diffSummary; | 
|  | private Collection<HumanComment> publishedComments; | 
|  | private Collection<RobotComment> robotComments; | 
|  | private CurrentUser visibleTo; | 
|  | private List<ChangeMessage> messages; | 
|  | private Optional<ChangedLines> changedLines; | 
|  | private SubmitTypeRecord submitTypeRecord; | 
|  | private Boolean mergeable; | 
|  | private Set<String> hashtags; | 
|  | /** | 
|  | * Map from {@link com.google.gerrit.entities.Account.Id} to the tip of the edit ref for this | 
|  | * change and a given user. | 
|  | */ | 
|  | private Table<Account.Id, PatchSet.Id, ObjectId> editsByUser; | 
|  |  | 
|  | private Set<Account.Id> reviewedBy; | 
|  | /** | 
|  | * Map from {@link com.google.gerrit.entities.Account.Id} to the tip of the draft comments ref for | 
|  | * this change and the user. | 
|  | */ | 
|  | private Map<Account.Id, ObjectId> draftsByUser; | 
|  |  | 
|  | private ImmutableListMultimap<Account.Id, String> stars; | 
|  | private StarsOf starsOf; | 
|  | private ImmutableMap<Account.Id, StarRef> starRefs; | 
|  | private ReviewerSet reviewers; | 
|  | private ReviewerByEmailSet reviewersByEmail; | 
|  | private ReviewerSet pendingReviewers; | 
|  | private ReviewerByEmailSet pendingReviewersByEmail; | 
|  | private List<ReviewerStatusUpdate> reviewerUpdates; | 
|  | private PersonIdent author; | 
|  | private PersonIdent committer; | 
|  | private ImmutableSet<AttentionSetUpdate> attentionSet; | 
|  | private Integer parentCount; | 
|  | private Integer unresolvedCommentCount; | 
|  | private Integer totalCommentCount; | 
|  | private LabelTypes labelTypes; | 
|  | private Optional<Instant> mergedOn; | 
|  | private ImmutableSetMultimap<NameKey, RefState> refStates; | 
|  | private ImmutableList<byte[]> refStatePatterns; | 
|  |  | 
|  | @Inject | 
|  | private ChangeData( | 
|  | @Nullable StarredChangesUtil starredChangesUtil, | 
|  | ApprovalsUtil approvalsUtil, | 
|  | AllUsersName allUsersName, | 
|  | ChangeMessagesUtil cmUtil, | 
|  | ChangeNotes.Factory notesFactory, | 
|  | CommentsUtil commentsUtil, | 
|  | GitRepositoryManager repoManager, | 
|  | MergeUtil.Factory mergeUtilFactory, | 
|  | MergeabilityCache mergeabilityCache, | 
|  | PatchListCache patchListCache, | 
|  | PatchSetUtil psUtil, | 
|  | ProjectCache projectCache, | 
|  | TrackingFooters trackingFooters, | 
|  | PureRevert pureRevert, | 
|  | SubmitRequirementsEvaluator submitRequirementsEvaluator, | 
|  | SubmitRequirementsUtil submitRequirementsUtil, | 
|  | SubmitRuleEvaluator.Factory submitRuleEvaluatorFactory, | 
|  | @Assisted Project.NameKey project, | 
|  | @Assisted Change.Id id, | 
|  | @Assisted @Nullable Change change, | 
|  | @Assisted @Nullable ChangeNotes notes) { | 
|  | this.approvalsUtil = approvalsUtil; | 
|  | this.allUsersName = allUsersName; | 
|  | this.cmUtil = cmUtil; | 
|  | this.notesFactory = notesFactory; | 
|  | this.commentsUtil = commentsUtil; | 
|  | this.repoManager = repoManager; | 
|  | this.mergeUtilFactory = mergeUtilFactory; | 
|  | this.mergeabilityCache = mergeabilityCache; | 
|  | this.patchListCache = patchListCache; | 
|  | this.psUtil = psUtil; | 
|  | this.projectCache = projectCache; | 
|  | this.starredChangesUtil = starredChangesUtil; | 
|  | this.trackingFooters = trackingFooters; | 
|  | this.pureRevert = pureRevert; | 
|  | this.submitRequirementsEvaluator = submitRequirementsEvaluator; | 
|  | this.submitRequirementsUtil = submitRequirementsUtil; | 
|  | this.submitRuleEvaluatorFactory = submitRuleEvaluatorFactory; | 
|  |  | 
|  | this.project = project; | 
|  | this.legacyId = id; | 
|  |  | 
|  | this.change = change; | 
|  | this.notes = notes; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * If false, omit fields that require database/repo IO. | 
|  | * | 
|  | * <p>This is used to enforce that the dashboard is rendered from the index only. If {@code | 
|  | * lazyLoad} is on, the {@code ChangeData} object will load from the database ("lazily") when a | 
|  | * field accessor is called. | 
|  | */ | 
|  | public ChangeData setStorageConstraint(StorageConstraint storageConstraint) { | 
|  | this.storageConstraint = storageConstraint; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | public StorageConstraint getStorageConstraint() { | 
|  | return storageConstraint; | 
|  | } | 
|  |  | 
|  | /** Returns {@code true} if we allow reading data from NoteDb. */ | 
|  | public boolean lazyload() { | 
|  | return storageConstraint.ordinal() | 
|  | >= StorageConstraint.INDEX_PRIMARY_NOTEDB_SECONDARY.ordinal(); | 
|  | } | 
|  |  | 
|  | public AllUsersName getAllUsersNameForIndexing() { | 
|  | return allUsersName; | 
|  | } | 
|  |  | 
|  | @VisibleForTesting | 
|  | public void setCurrentFilePaths(List<String> filePaths) { | 
|  | PatchSet ps = currentPatchSet(); | 
|  | if (ps != null) { | 
|  | currentFiles = ImmutableList.copyOf(filePaths); | 
|  | } | 
|  | } | 
|  |  | 
|  | public List<String> currentFilePaths() { | 
|  | if (currentFiles == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | Optional<DiffSummary> p = getDiffSummary(); | 
|  | currentFiles = p.map(DiffSummary::getPaths).orElse(Collections.emptyList()); | 
|  | } | 
|  | return currentFiles; | 
|  | } | 
|  |  | 
|  | private Optional<DiffSummary> getDiffSummary() { | 
|  | if (diffSummary == null) { | 
|  | if (!lazyload()) { | 
|  | return Optional.empty(); | 
|  | } | 
|  |  | 
|  | Change c = change(); | 
|  | PatchSet ps = currentPatchSet(); | 
|  | if (c == null || ps == null || !loadCommitData()) { | 
|  | return Optional.empty(); | 
|  | } | 
|  |  | 
|  | PatchListKey pk = PatchListKey.againstBase(ps.commitId(), parentCount); | 
|  | DiffSummaryKey key = DiffSummaryKey.fromPatchListKey(pk); | 
|  | try { | 
|  | diffSummary = Optional.of(patchListCache.getDiffSummary(key, c.getProject())); | 
|  | } catch (PatchListNotAvailableException e) { | 
|  | diffSummary = Optional.empty(); | 
|  | } | 
|  | } | 
|  | return diffSummary; | 
|  | } | 
|  |  | 
|  | private Optional<ChangedLines> computeChangedLines() { | 
|  | Optional<DiffSummary> ds = getDiffSummary(); | 
|  | if (ds.isPresent()) { | 
|  | return Optional.of(ds.get().getChangedLines()); | 
|  | } | 
|  | return Optional.empty(); | 
|  | } | 
|  |  | 
|  | public Optional<ChangedLines> changedLines() { | 
|  | if (changedLines == null) { | 
|  | if (!lazyload()) { | 
|  | return Optional.empty(); | 
|  | } | 
|  | changedLines = computeChangedLines(); | 
|  | } | 
|  | return changedLines; | 
|  | } | 
|  |  | 
|  | public void setChangedLines(int insertions, int deletions) { | 
|  | changedLines = Optional.of(new ChangedLines(insertions, deletions)); | 
|  | } | 
|  |  | 
|  | public void setLinesInserted(int insertions) { | 
|  | changedLines = | 
|  | Optional.of( | 
|  | new ChangedLines( | 
|  | insertions, | 
|  | changedLines != null && changedLines.isPresent() | 
|  | ? changedLines.get().deletions | 
|  | : -1)); | 
|  | } | 
|  |  | 
|  | public void setLinesDeleted(int deletions) { | 
|  | changedLines = | 
|  | Optional.of( | 
|  | new ChangedLines( | 
|  | changedLines != null && changedLines.isPresent() | 
|  | ? changedLines.get().insertions | 
|  | : -1, | 
|  | deletions)); | 
|  | } | 
|  |  | 
|  | public void setNoChangedLines() { | 
|  | changedLines = Optional.empty(); | 
|  | } | 
|  |  | 
|  | public Change.Id getId() { | 
|  | return legacyId; | 
|  | } | 
|  |  | 
|  | public Project.NameKey project() { | 
|  | return project; | 
|  | } | 
|  |  | 
|  | boolean fastIsVisibleTo(CurrentUser user) { | 
|  | return visibleTo == user; | 
|  | } | 
|  |  | 
|  | void cacheVisibleTo(CurrentUser user) { | 
|  | visibleTo = user; | 
|  | } | 
|  |  | 
|  | public Change change() { | 
|  | if (change == null && lazyload()) { | 
|  | reloadChange(); | 
|  | } | 
|  | return change; | 
|  | } | 
|  |  | 
|  | public void setChange(Change c) { | 
|  | change = c; | 
|  | } | 
|  |  | 
|  | public Change reloadChange() { | 
|  | try { | 
|  | notes = notesFactory.createChecked(project, legacyId); | 
|  | } catch (NoSuchChangeException e) { | 
|  | throw new StorageException("Unable to load change " + legacyId, e); | 
|  | } | 
|  | change = notes.getChange(); | 
|  | setPatchSets(null); | 
|  | return change; | 
|  | } | 
|  |  | 
|  | public LabelTypes getLabelTypes() { | 
|  | if (labelTypes == null) { | 
|  | ProjectState state = projectCache.get(project()).orElseThrow(illegalState(project())); | 
|  | labelTypes = state.getLabelTypes(change().getDest()); | 
|  | } | 
|  | return labelTypes; | 
|  | } | 
|  |  | 
|  | public ChangeNotes notes() { | 
|  | if (notes == null) { | 
|  | if (!lazyload()) { | 
|  | throw new StorageException("ChangeNotes not available, lazyLoad = false"); | 
|  | } | 
|  | notes = notesFactory.create(project(), legacyId); | 
|  | } | 
|  | return notes; | 
|  | } | 
|  |  | 
|  | public PatchSet currentPatchSet() { | 
|  | if (currentPatchSet == null) { | 
|  | Change c = change(); | 
|  | if (c == null) { | 
|  | return null; | 
|  | } | 
|  | for (PatchSet p : patchSets()) { | 
|  | if (p.id().equals(c.currentPatchSetId())) { | 
|  | currentPatchSet = p; | 
|  | return p; | 
|  | } | 
|  | } | 
|  | } | 
|  | return currentPatchSet; | 
|  | } | 
|  |  | 
|  | public List<PatchSetApproval> currentApprovals() { | 
|  | if (currentApprovals == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | Change c = change(); | 
|  | if (c == null) { | 
|  | currentApprovals = Collections.emptyList(); | 
|  | } else { | 
|  | try { | 
|  | currentApprovals = | 
|  | ImmutableList.copyOf(approvalsUtil.byPatchSet(notes(), c.currentPatchSetId())); | 
|  | } catch (StorageException e) { | 
|  | if (e.getCause() instanceof NoSuchChangeException) { | 
|  | currentApprovals = Collections.emptyList(); | 
|  | } else { | 
|  | throw e; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return currentApprovals; | 
|  | } | 
|  |  | 
|  | public void setCurrentApprovals(List<PatchSetApproval> approvals) { | 
|  | currentApprovals = approvals; | 
|  | } | 
|  |  | 
|  | public String commitMessage() { | 
|  | if (commitMessage == null) { | 
|  | if (!loadCommitData()) { | 
|  | return null; | 
|  | } | 
|  | } | 
|  | return commitMessage; | 
|  | } | 
|  |  | 
|  | /** Returns the list of commit footers (which may be empty). */ | 
|  | public List<FooterLine> commitFooters() { | 
|  | if (commitFooters == null) { | 
|  | if (!loadCommitData()) { | 
|  | return ImmutableList.of(); | 
|  | } | 
|  | } | 
|  | return commitFooters; | 
|  | } | 
|  |  | 
|  | public ListMultimap<String, String> trackingFooters() { | 
|  | return trackingFooters.extract(commitFooters()); | 
|  | } | 
|  |  | 
|  | public PersonIdent getAuthor() { | 
|  | if (author == null) { | 
|  | if (!loadCommitData()) { | 
|  | return null; | 
|  | } | 
|  | } | 
|  | return author; | 
|  | } | 
|  |  | 
|  | public PersonIdent getCommitter() { | 
|  | if (committer == null) { | 
|  | if (!loadCommitData()) { | 
|  | return null; | 
|  | } | 
|  | } | 
|  | return committer; | 
|  | } | 
|  |  | 
|  | private boolean loadCommitData() { | 
|  | PatchSet ps = currentPatchSet(); | 
|  | if (ps == null) { | 
|  | return false; | 
|  | } | 
|  | try (Repository repo = repoManager.openRepository(project()); | 
|  | RevWalk walk = new RevWalk(repo)) { | 
|  | RevCommit c = walk.parseCommit(ps.commitId()); | 
|  | commitMessage = c.getFullMessage(); | 
|  | commitFooters = c.getFooterLines(); | 
|  | author = c.getAuthorIdent(); | 
|  | committer = c.getCommitterIdent(); | 
|  | parentCount = c.getParentCount(); | 
|  | } catch (IOException e) { | 
|  | throw new StorageException( | 
|  | String.format( | 
|  | "Loading commit %s for ps %d of change %d failed.", | 
|  | ps.commitId(), ps.id().get(), ps.id().changeId().get()), | 
|  | e); | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /** Returns the most recent update (i.e. status) per user. */ | 
|  | public ImmutableSet<AttentionSetUpdate> attentionSet() { | 
|  | if (attentionSet == null) { | 
|  | if (!lazyload()) { | 
|  | return ImmutableSet.of(); | 
|  | } | 
|  | attentionSet = notes().getAttentionSet(); | 
|  | } | 
|  | return attentionSet; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the {@link Optional} value of time when the change was merged. | 
|  | * | 
|  | * <p>The value can be set from index field, see {@link ChangeData#setMergedOn} or loaded from the | 
|  | * database (available in {@link ChangeNotes}) | 
|  | * | 
|  | * @return {@link Optional} value of time when the change was merged. | 
|  | * @throws StorageException if {@code lazyLoad} is off, {@link ChangeNotes} can not be loaded | 
|  | *     because we do not expect to call the database. | 
|  | */ | 
|  | public Optional<Instant> getMergedOn() throws StorageException { | 
|  | if (mergedOn == null) { | 
|  | // The value was not loaded yet, try to get from the database. | 
|  | mergedOn = notes().getMergedOn(); | 
|  | } | 
|  | return mergedOn; | 
|  | } | 
|  |  | 
|  | /** Sets the value e.g. when loading from index. */ | 
|  | public void setMergedOn(@Nullable Instant mergedOn) { | 
|  | this.mergedOn = Optional.ofNullable(mergedOn); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Sets the specified attention set. If two or more entries refer to the same user, throws an | 
|  | * {@link IllegalStateException}. | 
|  | */ | 
|  | public void setAttentionSet(ImmutableSet<AttentionSetUpdate> attentionSet) { | 
|  | if (attentionSet.stream().map(AttentionSetUpdate::account).distinct().count() | 
|  | != attentionSet.size()) { | 
|  | throw new IllegalStateException( | 
|  | String.format( | 
|  | "Stored attention set for change %d contains duplicate update", | 
|  | change.getId().get())); | 
|  | } | 
|  | this.attentionSet = attentionSet; | 
|  | } | 
|  |  | 
|  | /** Returns patches for the change, in patch set ID order. */ | 
|  | public Collection<PatchSet> patchSets() { | 
|  | if (patchSets == null) { | 
|  | patchSets = psUtil.byChange(notes()); | 
|  | } | 
|  | return patchSets; | 
|  | } | 
|  |  | 
|  | public void setPatchSets(Collection<PatchSet> patchSets) { | 
|  | this.currentPatchSet = null; | 
|  | this.patchSets = patchSets; | 
|  | } | 
|  |  | 
|  | /** Returns patch with the given ID, or null if it does not exist. */ | 
|  | public PatchSet patchSet(PatchSet.Id psId) { | 
|  | if (currentPatchSet != null && currentPatchSet.id().equals(psId)) { | 
|  | return currentPatchSet; | 
|  | } | 
|  | for (PatchSet ps : patchSets()) { | 
|  | if (ps.id().equals(psId)) { | 
|  | return ps; | 
|  | } | 
|  | } | 
|  | return null; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns all patch set approvals for the change, keyed by ID, ordered by timestamp within each | 
|  | * patch set. | 
|  | */ | 
|  | public ListMultimap<PatchSet.Id, PatchSetApproval> approvals() { | 
|  | if (allApprovals == null) { | 
|  | if (!lazyload()) { | 
|  | return ImmutableListMultimap.of(); | 
|  | } | 
|  | allApprovals = approvalsUtil.byChangeExcludingCopiedApprovals(notes()); | 
|  | } | 
|  | return allApprovals; | 
|  | } | 
|  |  | 
|  | /* @return legacy submit ('SUBM') approval label */ | 
|  | // TODO(mariasavtchouk): Deprecate legacy submit label, | 
|  | // see com.google.gerrit.entities.LabelId.LEGACY_SUBMIT_NAME | 
|  | public Optional<PatchSetApproval> getSubmitApproval() { | 
|  | return currentApprovals().stream().filter(PatchSetApproval::isLegacySubmit).findFirst(); | 
|  | } | 
|  |  | 
|  | public ReviewerSet reviewers() { | 
|  | if (reviewers == null) { | 
|  | if (!lazyload()) { | 
|  | // We are not allowed to load values from NoteDb. Reviewers were not populated with values | 
|  | // from the index. However, we need these values for permission checks. | 
|  | throw new IllegalStateException("reviewers not populated"); | 
|  | } | 
|  | reviewers = approvalsUtil.getReviewers(notes()); | 
|  | } | 
|  | return reviewers; | 
|  | } | 
|  |  | 
|  | public void setReviewers(ReviewerSet reviewers) { | 
|  | this.reviewers = reviewers; | 
|  | } | 
|  |  | 
|  | public ReviewerByEmailSet reviewersByEmail() { | 
|  | if (reviewersByEmail == null) { | 
|  | if (!lazyload()) { | 
|  | return ReviewerByEmailSet.empty(); | 
|  | } | 
|  | reviewersByEmail = notes().getReviewersByEmail(); | 
|  | } | 
|  | return reviewersByEmail; | 
|  | } | 
|  |  | 
|  | public void setReviewersByEmail(ReviewerByEmailSet reviewersByEmail) { | 
|  | this.reviewersByEmail = reviewersByEmail; | 
|  | } | 
|  |  | 
|  | public ReviewerByEmailSet getReviewersByEmail() { | 
|  | return reviewersByEmail; | 
|  | } | 
|  |  | 
|  | public void setPendingReviewers(ReviewerSet pendingReviewers) { | 
|  | this.pendingReviewers = pendingReviewers; | 
|  | } | 
|  |  | 
|  | public ReviewerSet getPendingReviewers() { | 
|  | return this.pendingReviewers; | 
|  | } | 
|  |  | 
|  | public ReviewerSet pendingReviewers() { | 
|  | if (pendingReviewers == null) { | 
|  | if (!lazyload()) { | 
|  | return ReviewerSet.empty(); | 
|  | } | 
|  | pendingReviewers = notes().getPendingReviewers(); | 
|  | } | 
|  | return pendingReviewers; | 
|  | } | 
|  |  | 
|  | public void setPendingReviewersByEmail(ReviewerByEmailSet pendingReviewersByEmail) { | 
|  | this.pendingReviewersByEmail = pendingReviewersByEmail; | 
|  | } | 
|  |  | 
|  | public ReviewerByEmailSet getPendingReviewersByEmail() { | 
|  | return pendingReviewersByEmail; | 
|  | } | 
|  |  | 
|  | public ReviewerByEmailSet pendingReviewersByEmail() { | 
|  | if (pendingReviewersByEmail == null) { | 
|  | if (!lazyload()) { | 
|  | return ReviewerByEmailSet.empty(); | 
|  | } | 
|  | pendingReviewersByEmail = notes().getPendingReviewersByEmail(); | 
|  | } | 
|  | return pendingReviewersByEmail; | 
|  | } | 
|  |  | 
|  | public List<ReviewerStatusUpdate> reviewerUpdates() { | 
|  | if (reviewerUpdates == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | reviewerUpdates = approvalsUtil.getReviewerUpdates(notes()); | 
|  | } | 
|  | return reviewerUpdates; | 
|  | } | 
|  |  | 
|  | public void setReviewerUpdates(List<ReviewerStatusUpdate> reviewerUpdates) { | 
|  | this.reviewerUpdates = reviewerUpdates; | 
|  | } | 
|  |  | 
|  | public List<ReviewerStatusUpdate> getReviewerUpdates() { | 
|  | return reviewerUpdates; | 
|  | } | 
|  |  | 
|  | public Collection<HumanComment> publishedComments() { | 
|  | if (publishedComments == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | publishedComments = commentsUtil.publishedHumanCommentsByChange(notes()); | 
|  | } | 
|  | return publishedComments; | 
|  | } | 
|  |  | 
|  | public Collection<RobotComment> robotComments() { | 
|  | if (robotComments == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | robotComments = commentsUtil.robotCommentsByChange(notes()); | 
|  | } | 
|  | return robotComments; | 
|  | } | 
|  |  | 
|  | public Integer unresolvedCommentCount() { | 
|  | if (unresolvedCommentCount == null) { | 
|  | if (!lazyload()) { | 
|  | return null; | 
|  | } | 
|  |  | 
|  | List<Comment> comments = | 
|  | Stream.concat(publishedComments().stream(), robotComments().stream()).collect(toList()); | 
|  |  | 
|  | ImmutableSet<CommentThread<Comment>> commentThreads = | 
|  | CommentThreads.forComments(comments).getThreads(); | 
|  | unresolvedCommentCount = | 
|  | (int) commentThreads.stream().filter(CommentThread::unresolved).count(); | 
|  | } | 
|  |  | 
|  | return unresolvedCommentCount; | 
|  | } | 
|  |  | 
|  | public void setUnresolvedCommentCount(Integer count) { | 
|  | this.unresolvedCommentCount = count; | 
|  | } | 
|  |  | 
|  | public Integer totalCommentCount() { | 
|  | if (totalCommentCount == null) { | 
|  | if (!lazyload()) { | 
|  | return null; | 
|  | } | 
|  |  | 
|  | // Fail on overflow. | 
|  | totalCommentCount = | 
|  | Ints.checkedCast((long) publishedComments().size() + robotComments().size()); | 
|  | } | 
|  | return totalCommentCount; | 
|  | } | 
|  |  | 
|  | public void setTotalCommentCount(Integer count) { | 
|  | this.totalCommentCount = count; | 
|  | } | 
|  |  | 
|  | public List<ChangeMessage> messages() { | 
|  | if (messages == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | messages = cmUtil.byChange(notes()); | 
|  | } | 
|  | return messages; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Similar to {@link #submitRequirements}, except that it also converts submit records resulting | 
|  | * from the evaluation of legacy submit rules to submit requirements. | 
|  | */ | 
|  | public Map<SubmitRequirement, SubmitRequirementResult> submitRequirementsIncludingLegacy() { | 
|  | Map<SubmitRequirement, SubmitRequirementResult> projectConfigReqs = submitRequirements(); | 
|  | Map<SubmitRequirement, SubmitRequirementResult> legacyReqs = | 
|  | SubmitRequirementsAdapter.getLegacyRequirements(this); | 
|  | return submitRequirementsUtil.mergeLegacyAndNonLegacyRequirements( | 
|  | projectConfigReqs, legacyReqs, this); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Get all evaluated submit requirements for this change, including those from parent projects. | 
|  | * For closed changes, submit requirements are read from the change notes. For active changes, | 
|  | * submit requirements are evaluated online. | 
|  | * | 
|  | * <p>For changes loaded from the index, the value will be set from index field {@link | 
|  | * com.google.gerrit.server.index.change.ChangeField#STORED_SUBMIT_REQUIREMENTS}. | 
|  | */ | 
|  | public Map<SubmitRequirement, SubmitRequirementResult> submitRequirements() { | 
|  | if (submitRequirements == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyMap(); | 
|  | } | 
|  | Change c = change(); | 
|  | if (c == null || !c.isClosed()) { | 
|  | // Open changes: Evaluate submit requirements online. | 
|  | submitRequirements = | 
|  | submitRequirementsEvaluator.evaluateAllRequirements(this, /* includeLegacy= */ false); | 
|  | return submitRequirements; | 
|  | } | 
|  | // Closed changes: Load submit requirement results from NoteDb. | 
|  | submitRequirements = | 
|  | notes().getSubmitRequirementsResult().stream() | 
|  | .filter(r -> !r.isLegacy()) | 
|  | .collect(Collectors.toMap(r -> r.submitRequirement(), Function.identity())); | 
|  | } | 
|  | return submitRequirements; | 
|  | } | 
|  |  | 
|  | public void setSubmitRequirements( | 
|  | Map<SubmitRequirement, SubmitRequirementResult> submitRequirements) { | 
|  | this.submitRequirements = submitRequirements; | 
|  | } | 
|  |  | 
|  | public List<SubmitRecord> submitRecords(SubmitRuleOptions options) { | 
|  | // If the change is not submitted yet, 'strict' and 'lenient' both have the same result. If the | 
|  | // change is submitted, SubmitRecord requested with 'strict' will contain just a single entry | 
|  | // that with status=CLOSED. The latter is cheap to evaluate as we don't have to run any actual | 
|  | // evaluation. | 
|  | List<SubmitRecord> records = submitRecords.get(options); | 
|  | if (records == null) { | 
|  | if (storageConstraint != StorageConstraint.NOTEDB_ONLY) { | 
|  | // Submit requirements are expensive. We allow loading them only if this change did not | 
|  | // originate from the change index and we can invest the extra time. | 
|  | logger.atWarning().log( | 
|  | "Tried to load SubmitRecords for change fetched from index %s: %d", | 
|  | project(), getId().get()); | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | records = submitRuleEvaluatorFactory.create(options).evaluate(this); | 
|  | submitRecords.put(options, records); | 
|  | if (!change().isClosed() && submitRecords.size() == 1) { | 
|  | // Cache the SubmitRecord with allowClosed = !allowClosed as the SubmitRecord are the same. | 
|  | submitRecords.put( | 
|  | options | 
|  | .toBuilder() | 
|  | .recomputeOnClosedChanges(!options.recomputeOnClosedChanges()) | 
|  | .build(), | 
|  | records); | 
|  | } | 
|  | } | 
|  | return records; | 
|  | } | 
|  |  | 
|  | public void setSubmitRecords(SubmitRuleOptions options, List<SubmitRecord> records) { | 
|  | submitRecords.put(options, records); | 
|  | } | 
|  |  | 
|  | public SubmitTypeRecord submitTypeRecord() { | 
|  | if (submitTypeRecord == null) { | 
|  | submitTypeRecord = | 
|  | submitRuleEvaluatorFactory.create(SubmitRuleOptions.defaults()).getSubmitType(this); | 
|  | } | 
|  | return submitTypeRecord; | 
|  | } | 
|  |  | 
|  | public void setMergeable(Boolean mergeable) { | 
|  | this.mergeable = mergeable; | 
|  | } | 
|  |  | 
|  | @Nullable | 
|  | public Boolean isMergeable() { | 
|  | if (mergeable == null) { | 
|  | Change c = change(); | 
|  | if (c == null) { | 
|  | return null; | 
|  | } | 
|  | if (c.isMerged()) { | 
|  | mergeable = true; | 
|  | } else if (c.isAbandoned()) { | 
|  | return null; | 
|  | } else if (c.isWorkInProgress()) { | 
|  | return null; | 
|  | } else { | 
|  | if (!lazyload()) { | 
|  | return null; | 
|  | } | 
|  | PatchSet ps = currentPatchSet(); | 
|  | if (ps == null) { | 
|  | return null; | 
|  | } | 
|  |  | 
|  | try (Repository repo = repoManager.openRepository(project())) { | 
|  | Ref ref = repo.getRefDatabase().exactRef(c.getDest().branch()); | 
|  | SubmitTypeRecord str = submitTypeRecord(); | 
|  | if (!str.isOk()) { | 
|  | // If submit type rules are broken, it's definitely not mergeable. | 
|  | // No need to log, as SubmitRuleEvaluator already did it for us. | 
|  | return false; | 
|  | } | 
|  | String mergeStrategy = | 
|  | mergeUtilFactory | 
|  | .create(projectCache.get(project()).orElseThrow(illegalState(project()))) | 
|  | .mergeStrategyName(); | 
|  | mergeable = | 
|  | mergeabilityCache.get(ps.commitId(), ref, str.type, mergeStrategy, c.getDest(), repo); | 
|  | } catch (IOException e) { | 
|  | throw new StorageException(e); | 
|  | } | 
|  | } | 
|  | } | 
|  | return mergeable; | 
|  | } | 
|  |  | 
|  | @Nullable | 
|  | public Boolean isMerge() { | 
|  | if (parentCount == null) { | 
|  | if (!loadCommitData()) { | 
|  | return null; | 
|  | } | 
|  | } | 
|  | return parentCount > 1; | 
|  | } | 
|  |  | 
|  | public Set<Account.Id> editsByUser() { | 
|  | return editRefs().rowKeySet(); | 
|  | } | 
|  |  | 
|  | public Table<Account.Id, PatchSet.Id, ObjectId> editRefs() { | 
|  | if (editsByUser == null) { | 
|  | if (!lazyload()) { | 
|  | return HashBasedTable.create(); | 
|  | } | 
|  | Change c = change(); | 
|  | if (c == null) { | 
|  | return HashBasedTable.create(); | 
|  | } | 
|  | editsByUser = HashBasedTable.create(); | 
|  | Change.Id id = requireNonNull(change.getId()); | 
|  | try (Repository repo = repoManager.openRepository(project())) { | 
|  | for (Ref ref : repo.getRefDatabase().getRefsByPrefix(RefNames.REFS_USERS)) { | 
|  | if (!RefNames.isRefsEdit(ref.getName())) { | 
|  | continue; | 
|  | } | 
|  | PatchSet.Id ps = PatchSet.Id.fromEditRef(ref.getName()); | 
|  | if (id.equals(ps.changeId())) { | 
|  | Account.Id accountId = Account.Id.fromRef(ref.getName()); | 
|  | if (accountId != null) { | 
|  | editsByUser.put(accountId, ps, ref.getObjectId()); | 
|  | } | 
|  | } | 
|  | } | 
|  | } catch (IOException e) { | 
|  | throw new StorageException(e); | 
|  | } | 
|  | } | 
|  | return editsByUser; | 
|  | } | 
|  |  | 
|  | public Set<Account.Id> draftsByUser() { | 
|  | return draftRefs().keySet(); | 
|  | } | 
|  |  | 
|  | public boolean isReviewedBy(Account.Id accountId) { | 
|  | return reviewedBy().contains(accountId); | 
|  | } | 
|  |  | 
|  | public Set<Account.Id> reviewedBy() { | 
|  | if (reviewedBy == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptySet(); | 
|  | } | 
|  | Change c = change(); | 
|  | if (c == null) { | 
|  | return Collections.emptySet(); | 
|  | } | 
|  | List<ReviewedByEvent> events = new ArrayList<>(); | 
|  | for (ChangeMessage msg : messages()) { | 
|  | if (msg.getAuthor() != null) { | 
|  | events.add(ReviewedByEvent.create(msg)); | 
|  | } | 
|  | } | 
|  | events = Lists.reverse(events); | 
|  | reviewedBy = new LinkedHashSet<>(); | 
|  | Account.Id owner = c.getOwner(); | 
|  | for (ReviewedByEvent event : events) { | 
|  | if (owner.equals(event.author())) { | 
|  | break; | 
|  | } | 
|  | reviewedBy.add(event.author()); | 
|  | } | 
|  | } | 
|  | return reviewedBy; | 
|  | } | 
|  |  | 
|  | public void setReviewedBy(Set<Account.Id> reviewedBy) { | 
|  | this.reviewedBy = reviewedBy; | 
|  | } | 
|  |  | 
|  | public Set<String> hashtags() { | 
|  | if (hashtags == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptySet(); | 
|  | } | 
|  | hashtags = notes().getHashtags(); | 
|  | } | 
|  | return hashtags; | 
|  | } | 
|  |  | 
|  | public void setHashtags(Set<String> hashtags) { | 
|  | this.hashtags = hashtags; | 
|  | } | 
|  |  | 
|  | public ImmutableListMultimap<Account.Id, String> stars() { | 
|  | if (stars == null) { | 
|  | if (!lazyload()) { | 
|  | return ImmutableListMultimap.of(); | 
|  | } | 
|  | ImmutableListMultimap.Builder<Account.Id, String> b = ImmutableListMultimap.builder(); | 
|  | for (Map.Entry<Account.Id, StarRef> e : starRefs().entrySet()) { | 
|  | b.putAll(e.getKey(), e.getValue().labels()); | 
|  | } | 
|  | return b.build(); | 
|  | } | 
|  | return stars; | 
|  | } | 
|  |  | 
|  | public void setStars(ListMultimap<Account.Id, String> stars) { | 
|  | this.stars = ImmutableListMultimap.copyOf(stars); | 
|  | } | 
|  |  | 
|  | public ImmutableMap<Account.Id, StarRef> starRefs() { | 
|  | if (starRefs == null) { | 
|  | if (!lazyload()) { | 
|  | return ImmutableMap.of(); | 
|  | } | 
|  | starRefs = requireNonNull(starredChangesUtil).byChange(legacyId); | 
|  | } | 
|  | return starRefs; | 
|  | } | 
|  |  | 
|  | public Set<String> stars(Account.Id accountId) { | 
|  | if (starsOf != null) { | 
|  | if (!starsOf.accountId().equals(accountId)) { | 
|  | starsOf = null; | 
|  | } | 
|  | } | 
|  | if (starsOf == null) { | 
|  | if (stars != null) { | 
|  | starsOf = StarsOf.create(accountId, stars.get(accountId)); | 
|  | } else { | 
|  | if (!lazyload()) { | 
|  | return ImmutableSet.of(); | 
|  | } | 
|  | starsOf = StarsOf.create(accountId, starredChangesUtil.getLabels(accountId, legacyId)); | 
|  | } | 
|  | } | 
|  | return starsOf.stars(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns {@code null} if {@code revertOf} is {@code null}; true if the change is a pure revert; | 
|  | * false otherwise. | 
|  | */ | 
|  | @Nullable | 
|  | public Boolean isPureRevert() { | 
|  | if (change().getRevertOf() == null) { | 
|  | return null; | 
|  | } | 
|  | try { | 
|  | return pureRevert.get(notes(), Optional.empty()); | 
|  | } catch (IOException | BadRequestException | ResourceConflictException e) { | 
|  | throw new StorageException("could not compute pure revert", e); | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | MoreObjects.ToStringHelper h = MoreObjects.toStringHelper(this); | 
|  | if (change != null) { | 
|  | h.addValue(change); | 
|  | } else { | 
|  | h.addValue(legacyId); | 
|  | } | 
|  | return h.toString(); | 
|  | } | 
|  |  | 
|  | public static class ChangedLines { | 
|  | public final int insertions; | 
|  | public final int deletions; | 
|  |  | 
|  | public ChangedLines(int insertions, int deletions) { | 
|  | this.insertions = insertions; | 
|  | this.deletions = deletions; | 
|  | } | 
|  | } | 
|  |  | 
|  | public SetMultimap<NameKey, RefState> getRefStates() { | 
|  | if (refStates == null) { | 
|  | if (!lazyload()) { | 
|  | return ImmutableSetMultimap.of(); | 
|  | } | 
|  |  | 
|  | ImmutableSetMultimap.Builder<NameKey, RefState> result = ImmutableSetMultimap.builder(); | 
|  | for (Table.Cell<Account.Id, PatchSet.Id, ObjectId> edit : editRefs().cellSet()) { | 
|  | result.put( | 
|  | project, | 
|  | RefState.create( | 
|  | RefNames.refsEdit( | 
|  | edit.getRowKey(), edit.getColumnKey().changeId(), edit.getColumnKey()), | 
|  | edit.getValue())); | 
|  | } | 
|  | starRefs().values().forEach(r -> result.put(allUsersName, RefState.of(r.ref()))); | 
|  |  | 
|  | // TODO: instantiating the notes is too much. We don't want to parse NoteDb, we just want the | 
|  | // refs. | 
|  | result.put(project, RefState.create(notes().getRefName(), notes().getMetaId())); | 
|  | notes().getRobotComments(); // Force loading robot comments. | 
|  | RobotCommentNotes robotNotes = notes().getRobotCommentNotes(); | 
|  | result.put(project, RefState.create(robotNotes.getRefName(), robotNotes.getMetaId())); | 
|  | draftRefs() | 
|  | .entrySet() | 
|  | .forEach( | 
|  | r -> | 
|  | result.put( | 
|  | allUsersName, | 
|  | RefState.create( | 
|  | RefNames.refsDraftComments(getId(), r.getKey()), r.getValue()))); | 
|  |  | 
|  | refStates = result.build(); | 
|  | } | 
|  |  | 
|  | return refStates; | 
|  | } | 
|  |  | 
|  | public void setRefStates(ImmutableSetMultimap<Project.NameKey, RefState> refStates) { | 
|  | this.refStates = refStates; | 
|  | if (draftsByUser == null) { | 
|  | // Recover draft refs as well. Draft comments are represented as refs in the repository. | 
|  | // ChangeData exposes #draftsByUser which just provides a Set of Account.Ids of users who | 
|  | // have drafts comments on this change. Recovering this list from RefStates makes it | 
|  | // available even on ChangeData instances retrieved from the index. | 
|  | draftsByUser = new HashMap<>(); | 
|  | if (refStates.containsKey(allUsersName)) { | 
|  | refStates.get(allUsersName).stream() | 
|  | .filter(r -> RefNames.isRefsDraftsComments(r.ref())) | 
|  | .forEach(r -> draftsByUser.put(Account.Id.fromRef(r.ref()), r.id())); | 
|  | } | 
|  | } | 
|  | if (editsByUser == null) { | 
|  | // Recover edit refs as well. Edits are represented as refs in the repository. | 
|  | // ChangeData exposes #editsByUser which just provides a Set of Account.Ids of users who | 
|  | // have edits on this change. Recovering this list from RefStates makes it available even | 
|  | // on ChangeData instances retrieved from the index. | 
|  | editsByUser = HashBasedTable.create(); | 
|  | if (refStates.containsKey(project())) { | 
|  | refStates.get(project()).stream() | 
|  | .filter(r -> RefNames.isRefsEdit(r.ref())) | 
|  | .forEach( | 
|  | r -> | 
|  | editsByUser.put( | 
|  | Account.Id.fromRef(r.ref()), PatchSet.Id.fromEditRef(r.ref()), r.id())); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | public ImmutableList<byte[]> getRefStatePatterns() { | 
|  | return refStatePatterns; | 
|  | } | 
|  |  | 
|  | public void setRefStatePatterns(Iterable<byte[]> refStatePatterns) { | 
|  | this.refStatePatterns = ImmutableList.copyOf(refStatePatterns); | 
|  | } | 
|  |  | 
|  | @AutoValue | 
|  | abstract static class ReviewedByEvent { | 
|  | private static ReviewedByEvent create(ChangeMessage msg) { | 
|  | return new AutoValue_ChangeData_ReviewedByEvent(msg.getAuthor(), msg.getWrittenOn()); | 
|  | } | 
|  |  | 
|  | public abstract Account.Id author(); | 
|  |  | 
|  | public abstract Instant ts(); | 
|  | } | 
|  |  | 
|  | @AutoValue | 
|  | abstract static class StarsOf { | 
|  | private static StarsOf create(Account.Id accountId, Iterable<String> stars) { | 
|  | return new AutoValue_ChangeData_StarsOf(accountId, ImmutableSortedSet.copyOf(stars)); | 
|  | } | 
|  |  | 
|  | public abstract Account.Id accountId(); | 
|  |  | 
|  | public abstract ImmutableSortedSet<String> stars(); | 
|  | } | 
|  |  | 
|  | private Map<Account.Id, ObjectId> draftRefs() { | 
|  | if (draftsByUser == null) { | 
|  | if (!lazyload()) { | 
|  | return Collections.emptyMap(); | 
|  | } | 
|  | Change c = change(); | 
|  | if (c == null) { | 
|  | return Collections.emptyMap(); | 
|  | } | 
|  |  | 
|  | draftsByUser = new HashMap<>(); | 
|  | for (Ref ref : commentsUtil.getDraftRefs(notes().getChangeId())) { | 
|  | Account.Id account = Account.Id.fromRefSuffix(ref.getName()); | 
|  | if (account != null | 
|  | // Double-check that any drafts exist for this user after | 
|  | // filtering out zombies. If some but not all drafts in the ref | 
|  | // were zombies, the returned Ref still includes those zombies; | 
|  | // this is suboptimal, but is ok for the purposes of | 
|  | // draftsByUser(), and easier than trying to rebuild the change at | 
|  | // this point. | 
|  | && !notes().getDraftComments(account, ref).isEmpty()) { | 
|  | draftsByUser.put(account, ref.getObjectId()); | 
|  | } | 
|  | } | 
|  | } | 
|  | return draftsByUser; | 
|  | } | 
|  | } |