| // 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.common.base.Preconditions.checkNotNull; |
| import static com.google.gerrit.server.ApprovalsUtil.sortApprovals; |
| import static java.util.stream.Collectors.toList; |
| import static java.util.stream.Collectors.toMap; |
| import static java.util.stream.Collectors.toSet; |
| |
| import com.google.auto.value.AutoValue; |
| import com.google.common.base.MoreObjects; |
| 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.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.gerrit.common.Nullable; |
| import com.google.gerrit.common.data.LabelTypes; |
| import com.google.gerrit.common.data.SubmitRecord; |
| import com.google.gerrit.common.data.SubmitTypeRecord; |
| import com.google.gerrit.extensions.restapi.AuthException; |
| import com.google.gerrit.extensions.restapi.BadRequestException; |
| import com.google.gerrit.extensions.restapi.ResourceConflictException; |
| import com.google.gerrit.reviewdb.client.Account; |
| import com.google.gerrit.reviewdb.client.Change; |
| import com.google.gerrit.reviewdb.client.ChangeMessage; |
| import com.google.gerrit.reviewdb.client.Comment; |
| import com.google.gerrit.reviewdb.client.PatchSet; |
| import com.google.gerrit.reviewdb.client.PatchSetApproval; |
| import com.google.gerrit.reviewdb.client.Project; |
| import com.google.gerrit.reviewdb.client.RefNames; |
| import com.google.gerrit.reviewdb.client.RobotComment; |
| import com.google.gerrit.reviewdb.server.ReviewDb; |
| import com.google.gerrit.server.AnonymousUser; |
| import com.google.gerrit.server.ApprovalsUtil; |
| import com.google.gerrit.server.ChangeMessagesUtil; |
| import com.google.gerrit.server.CommentsUtil; |
| import com.google.gerrit.server.CurrentUser; |
| import com.google.gerrit.server.IdentifiedUser; |
| 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.account.AccountCache; |
| import com.google.gerrit.server.account.Accounts; |
| import com.google.gerrit.server.account.Emails; |
| import com.google.gerrit.server.change.ChangeResource; |
| import com.google.gerrit.server.change.GetPureRevert; |
| import com.google.gerrit.server.change.MergeabilityCache; |
| 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.NotesMigration; |
| import com.google.gerrit.server.patch.DiffSummary; |
| import com.google.gerrit.server.patch.PatchListCache; |
| import com.google.gerrit.server.patch.PatchListNotAvailableException; |
| import com.google.gerrit.server.project.ChangeControl; |
| 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.SubmitRuleEvaluator; |
| import com.google.gerrit.server.project.SubmitRuleOptions; |
| import com.google.gerrit.server.update.ChangeContext; |
| import com.google.gwtorm.server.OrmException; |
| import com.google.gwtorm.server.ResultSet; |
| import com.google.inject.Inject; |
| import com.google.inject.assistedinject.Assisted; |
| import java.io.IOException; |
| import java.sql.Timestamp; |
| 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.Predicate; |
| import java.util.stream.Stream; |
| import org.eclipse.jgit.errors.IncorrectObjectTypeException; |
| import org.eclipse.jgit.errors.MissingObjectException; |
| import org.eclipse.jgit.errors.RepositoryNotFoundException; |
| 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; |
| |
| public class ChangeData { |
| private static final int BATCH_SIZE = 50; |
| |
| public static List<Change> asChanges(List<ChangeData> changeDatas) throws OrmException { |
| 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, cd -> cd)); |
| } |
| |
| public static void ensureChangeLoaded(Iterable<ChangeData> changes) throws OrmException { |
| ChangeData first = Iterables.getFirst(changes, null); |
| if (first == null) { |
| return; |
| } else if (first.notesMigration.readChanges()) { |
| for (ChangeData cd : changes) { |
| cd.change(); |
| } |
| return; |
| } |
| |
| Map<Change.Id, ChangeData> missing = new HashMap<>(); |
| for (ChangeData cd : changes) { |
| if (cd.change == null) { |
| missing.put(cd.getId(), cd); |
| } |
| } |
| if (missing.isEmpty()) { |
| return; |
| } |
| for (ChangeNotes notes : first.notesFactory.create(first.db, missing.keySet())) { |
| missing.get(notes.getChangeId()).change = notes.getChange(); |
| } |
| } |
| |
| public static void ensureAllPatchSetsLoaded(Iterable<ChangeData> changes) throws OrmException { |
| ChangeData first = Iterables.getFirst(changes, null); |
| if (first == null) { |
| return; |
| } else if (first.notesMigration.readChanges()) { |
| for (ChangeData cd : changes) { |
| cd.patchSets(); |
| } |
| return; |
| } |
| |
| List<ResultSet<PatchSet>> results = new ArrayList<>(BATCH_SIZE); |
| for (List<ChangeData> batch : Iterables.partition(changes, BATCH_SIZE)) { |
| results.clear(); |
| for (ChangeData cd : batch) { |
| if (cd.patchSets == null) { |
| results.add(cd.db.patchSets().byChange(cd.getId())); |
| } else { |
| results.add(null); |
| } |
| } |
| for (int i = 0; i < batch.size(); i++) { |
| ResultSet<PatchSet> result = results.get(i); |
| if (result != null) { |
| batch.get(i).patchSets = result.toList(); |
| } |
| } |
| } |
| } |
| |
| public static void ensureCurrentPatchSetLoaded(Iterable<ChangeData> changes) throws OrmException { |
| ChangeData first = Iterables.getFirst(changes, null); |
| if (first == null) { |
| return; |
| } else if (first.notesMigration.readChanges()) { |
| for (ChangeData cd : changes) { |
| cd.currentPatchSet(); |
| } |
| return; |
| } |
| |
| Map<PatchSet.Id, ChangeData> missing = new HashMap<>(); |
| for (ChangeData cd : changes) { |
| if (cd.currentPatchSet == null && cd.patchSets == null) { |
| missing.put(cd.change().currentPatchSetId(), cd); |
| } |
| } |
| if (missing.isEmpty()) { |
| return; |
| } |
| for (PatchSet ps : first.db.patchSets().get(missing.keySet())) { |
| missing.get(ps.getId()).currentPatchSet = ps; |
| } |
| } |
| |
| public static void ensureCurrentApprovalsLoaded(Iterable<ChangeData> changes) |
| throws OrmException { |
| ChangeData first = Iterables.getFirst(changes, null); |
| if (first == null) { |
| return; |
| } else if (first.notesMigration.readChanges()) { |
| for (ChangeData cd : changes) { |
| cd.currentApprovals(); |
| } |
| return; |
| } |
| |
| List<ResultSet<PatchSetApproval>> results = new ArrayList<>(BATCH_SIZE); |
| for (List<ChangeData> batch : Iterables.partition(changes, BATCH_SIZE)) { |
| results.clear(); |
| for (ChangeData cd : batch) { |
| if (cd.currentApprovals == null) { |
| PatchSet.Id psId = cd.change().currentPatchSetId(); |
| results.add(cd.db.patchSetApprovals().byPatchSet(psId)); |
| } else { |
| results.add(null); |
| } |
| } |
| for (int i = 0; i < batch.size(); i++) { |
| ResultSet<PatchSetApproval> result = results.get(i); |
| if (result != null) { |
| batch.get(i).currentApprovals = sortApprovals(result); |
| } |
| } |
| } |
| } |
| |
| public static void ensureMessagesLoaded(Iterable<ChangeData> changes) throws OrmException { |
| ChangeData first = Iterables.getFirst(changes, null); |
| if (first == null) { |
| return; |
| } else if (first.notesMigration.readChanges()) { |
| for (ChangeData cd : changes) { |
| cd.messages(); |
| } |
| return; |
| } |
| |
| List<ResultSet<ChangeMessage>> results = new ArrayList<>(BATCH_SIZE); |
| for (List<ChangeData> batch : Iterables.partition(changes, BATCH_SIZE)) { |
| results.clear(); |
| for (ChangeData cd : batch) { |
| if (cd.messages == null) { |
| PatchSet.Id psId = cd.change().currentPatchSetId(); |
| results.add(cd.db.changeMessages().byPatchSet(psId)); |
| } else { |
| results.add(null); |
| } |
| } |
| for (int i = 0; i < batch.size(); i++) { |
| ResultSet<ChangeMessage> result = results.get(i); |
| if (result != null) { |
| batch.get(i).messages = result.toList(); |
| } |
| } |
| } |
| } |
| |
| public static void ensureReviewedByLoadedForOpenChanges(Iterable<ChangeData> changes) |
| throws OrmException { |
| List<ChangeData> pending = new ArrayList<>(); |
| for (ChangeData cd : changes) { |
| if (cd.reviewedBy == null && cd.change().getStatus().isOpen()) { |
| pending.add(cd); |
| } |
| } |
| |
| if (!pending.isEmpty()) { |
| ensureAllPatchSetsLoaded(pending); |
| ensureMessagesLoaded(pending); |
| for (ChangeData cd : pending) { |
| cd.reviewedBy(); |
| } |
| } |
| } |
| |
| public static class Factory { |
| private final AssistedFactory assistedFactory; |
| private final ChangeControl.GenericFactory changeControlFactory; |
| |
| @Inject |
| Factory(AssistedFactory assistedFactory, ChangeControl.GenericFactory changeControlFactory) { |
| this.assistedFactory = assistedFactory; |
| this.changeControlFactory = changeControlFactory; |
| } |
| |
| public ChangeData create(ReviewDb db, Project.NameKey project, Change.Id id) { |
| return assistedFactory.create(db, project, id, null, null, null); |
| } |
| |
| public ChangeData create(ReviewDb db, Change change) { |
| return assistedFactory.create(db, change.getProject(), change.getId(), change, null, null); |
| } |
| |
| public ChangeData create(ReviewDb db, ChangeNotes notes) { |
| return assistedFactory.create( |
| db, notes.getChange().getProject(), notes.getChangeId(), notes.getChange(), notes, null); |
| } |
| |
| public ChangeData create(ReviewDb db, ChangeControl control) { |
| return assistedFactory.create( |
| db, |
| control.getChange().getProject(), |
| control.getId(), |
| control.getChange(), |
| control.getNotes(), |
| control); |
| } |
| |
| // TODO(hiesel): Remove these after ChangeControl is removed from ChangeData |
| public ChangeData create(ReviewDb db, ChangeResource rsrc) throws NoSuchChangeException { |
| return create(db, changeControlFactory.controlFor(rsrc.getNotes(), rsrc.getUser())); |
| } |
| |
| public ChangeData create(ReviewDb db, ChangeContext ctx) throws NoSuchChangeException { |
| return create(db, changeControlFactory.controlFor(ctx.getNotes(), ctx.getUser())); |
| } |
| } |
| |
| public interface AssistedFactory { |
| ChangeData create( |
| ReviewDb db, |
| Project.NameKey project, |
| Change.Id id, |
| @Nullable Change change, |
| @Nullable ChangeNotes notes, |
| @Nullable ChangeControl control); |
| } |
| |
| /** |
| * 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) { |
| ChangeData cd = |
| new ChangeData( |
| null, null, null, null, null, null, null, null, null, null, null, null, null, null, |
| null, null, null, null, null, null, null, project, id, null, null, null); |
| cd.currentPatchSet = new PatchSet(new PatchSet.Id(id, currentPatchSetId)); |
| return cd; |
| } |
| |
| // Injected fields. |
| private @Nullable final StarredChangesUtil starredChangesUtil; |
| private final AccountCache accountCache; |
| private final Accounts accounts; |
| private final AllUsersName allUsersName; |
| private final ApprovalsUtil approvalsUtil; |
| private final ChangeControl.GenericFactory changeControlFactory; |
| private final ChangeMessagesUtil cmUtil; |
| private final ChangeNotes.Factory notesFactory; |
| private final CommentsUtil commentsUtil; |
| private final Emails emails; |
| private final GitRepositoryManager repoManager; |
| private final IdentifiedUser.GenericFactory userFactory; |
| private final MergeUtil.Factory mergeUtilFactory; |
| private final MergeabilityCache mergeabilityCache; |
| private final NotesMigration notesMigration; |
| private final PatchListCache patchListCache; |
| private final PatchSetUtil psUtil; |
| private final ProjectCache projectCache; |
| private final TrackingFooters trackingFooters; |
| private final GetPureRevert pureRevert; |
| |
| // Required assisted injected fields. |
| private final ReviewDb db; |
| 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 boolean lazyLoad = true; |
| 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 Map<Integer, List<String>> files; |
| private Map<Integer, Optional<DiffSummary>> diffSummaries; |
| private Collection<Comment> publishedComments; |
| private Collection<RobotComment> robotComments; |
| private CurrentUser visibleTo; |
| private ChangeControl changeControl; |
| private List<ChangeMessage> messages; |
| private Optional<ChangedLines> changedLines; |
| private SubmitTypeRecord submitTypeRecord; |
| private Boolean mergeable; |
| private Set<String> hashtags; |
| private Map<Account.Id, Ref> editsByUser; |
| private Set<Account.Id> reviewedBy; |
| private Map<Account.Id, Ref> 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 Integer unresolvedCommentCount; |
| private LabelTypes labelTypes; |
| |
| private ImmutableList<byte[]> refStates; |
| private ImmutableList<byte[]> refStatePatterns; |
| |
| @Inject |
| private ChangeData( |
| @Nullable StarredChangesUtil starredChangesUtil, |
| AccountCache accountCache, |
| Accounts accounts, |
| ApprovalsUtil approvalsUtil, |
| AllUsersName allUsersName, |
| ChangeControl.GenericFactory changeControlFactory, |
| ChangeMessagesUtil cmUtil, |
| ChangeNotes.Factory notesFactory, |
| CommentsUtil commentsUtil, |
| Emails emails, |
| GitRepositoryManager repoManager, |
| IdentifiedUser.GenericFactory userFactory, |
| MergeUtil.Factory mergeUtilFactory, |
| MergeabilityCache mergeabilityCache, |
| NotesMigration notesMigration, |
| PatchListCache patchListCache, |
| PatchSetUtil psUtil, |
| ProjectCache projectCache, |
| TrackingFooters trackingFooters, |
| GetPureRevert pureRevert, |
| @Assisted ReviewDb db, |
| @Assisted Project.NameKey project, |
| @Assisted Change.Id id, |
| @Assisted @Nullable Change change, |
| @Assisted @Nullable ChangeNotes notes, |
| @Assisted @Nullable ChangeControl control) { |
| this.accountCache = accountCache; |
| this.accounts = accounts; |
| this.approvalsUtil = approvalsUtil; |
| this.allUsersName = allUsersName; |
| this.changeControlFactory = changeControlFactory; |
| this.cmUtil = cmUtil; |
| this.notesFactory = notesFactory; |
| this.commentsUtil = commentsUtil; |
| this.emails = emails; |
| this.repoManager = repoManager; |
| this.userFactory = userFactory; |
| this.mergeUtilFactory = mergeUtilFactory; |
| this.mergeabilityCache = mergeabilityCache; |
| this.notesMigration = notesMigration; |
| this.patchListCache = patchListCache; |
| this.psUtil = psUtil; |
| this.projectCache = projectCache; |
| this.starredChangesUtil = starredChangesUtil; |
| this.trackingFooters = trackingFooters; |
| this.pureRevert = pureRevert; |
| |
| // May be null in tests when created via createForTest above, in which case lazy-loading will |
| // intentionally fail with NPE. Still not marked @Nullable in the constructor, to force callers |
| // using Guice to pass a non-null value. |
| this.db = db; |
| |
| this.project = project; |
| this.legacyId = id; |
| |
| this.change = change; |
| this.notes = notes; |
| this.changeControl = control; |
| } |
| |
| public ChangeData setLazyLoad(boolean load) { |
| lazyLoad = load; |
| return this; |
| } |
| |
| public ReviewDb db() { |
| return db; |
| } |
| |
| public AllUsersName getAllUsersNameForIndexing() { |
| return allUsersName; |
| } |
| |
| private Map<Integer, List<String>> initFiles() { |
| if (files == null) { |
| files = new HashMap<>(); |
| } |
| return files; |
| } |
| |
| public void setCurrentFilePaths(List<String> filePaths) throws OrmException { |
| PatchSet ps = currentPatchSet(); |
| if (ps != null) { |
| initFiles().put(ps.getPatchSetId(), ImmutableList.copyOf(filePaths)); |
| } |
| } |
| |
| public List<String> currentFilePaths() throws OrmException { |
| PatchSet ps = currentPatchSet(); |
| return ps != null ? filePaths(ps) : null; |
| } |
| |
| public List<String> filePaths(PatchSet ps) throws OrmException { |
| Integer psId = ps.getPatchSetId(); |
| List<String> r = initFiles().get(psId); |
| if (r == null) { |
| Change c = change(); |
| if (c == null) { |
| return null; |
| } |
| |
| Optional<DiffSummary> p = getDiffSummary(c, ps); |
| if (!p.isPresent()) { |
| List<String> emptyFileList = Collections.emptyList(); |
| if (lazyLoad) { |
| files.put(ps.getPatchSetId(), emptyFileList); |
| } |
| return emptyFileList; |
| } |
| |
| r = p.get().getPaths(); |
| files.put(psId, r); |
| } |
| return r; |
| } |
| |
| private Optional<DiffSummary> getDiffSummary(Change c, PatchSet ps) { |
| Integer psId = ps.getId().get(); |
| if (diffSummaries == null) { |
| diffSummaries = new HashMap<>(); |
| } |
| Optional<DiffSummary> r = diffSummaries.get(psId); |
| if (r == null) { |
| if (!lazyLoad) { |
| return Optional.empty(); |
| } |
| try { |
| r = Optional.of(patchListCache.getDiffSummary(c, ps)); |
| } catch (PatchListNotAvailableException e) { |
| r = Optional.empty(); |
| } |
| diffSummaries.put(psId, r); |
| } |
| return r; |
| } |
| |
| private Optional<ChangedLines> computeChangedLines() throws OrmException { |
| Change c = change(); |
| if (c == null) { |
| return Optional.empty(); |
| } |
| PatchSet ps = currentPatchSet(); |
| if (ps == null) { |
| return Optional.empty(); |
| } |
| Optional<DiffSummary> ds = getDiffSummary(c, ps); |
| if (ds.isPresent()) { |
| return Optional.of(ds.get().getChangedLines()); |
| } |
| return Optional.empty(); |
| } |
| |
| public Optional<ChangedLines> changedLines() throws OrmException { |
| 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 setNoChangedLines() { |
| changedLines = Optional.empty(); |
| } |
| |
| public Change.Id getId() { |
| return legacyId; |
| } |
| |
| public Project.NameKey project() { |
| return project; |
| } |
| |
| boolean fastIsVisibleTo(CurrentUser user) { |
| return visibleTo == user; |
| } |
| |
| public boolean hasChangeControl() { |
| return changeControl != null; |
| } |
| |
| public ChangeControl changeControl() throws OrmException { |
| if (changeControl == null) { |
| Change c = change(); |
| try { |
| changeControl = changeControlFactory.controlFor(db, c, userFactory.create(c.getOwner())); |
| } catch (NoSuchChangeException e) { |
| throw new OrmException(e); |
| } |
| } |
| return changeControl; |
| } |
| |
| public ChangeControl changeControl(CurrentUser user) throws OrmException { |
| if (changeControl != null) { |
| CurrentUser oldUser = user; |
| if (sameUser(user, oldUser)) { |
| return changeControl; |
| } |
| throw new IllegalStateException("user already specified: " + changeControl.getUser()); |
| } |
| |
| if (change != null) { |
| changeControl = changeControlFactory.controlFor(db, change, user); |
| } else { |
| changeControl = changeControlFactory.controlFor(db, project(), legacyId, user); |
| } |
| return changeControl; |
| } |
| |
| private static boolean sameUser(CurrentUser a, CurrentUser b) { |
| // TODO(dborowitz): This is a hack; general CurrentUser equality would be |
| // better. |
| if (a.isInternalUser() && b.isInternalUser()) { |
| return true; |
| } else if (a instanceof AnonymousUser && b instanceof AnonymousUser) { |
| return true; |
| } else if (a.isIdentifiedUser() && b.isIdentifiedUser()) { |
| return a.getAccountId().equals(b.getAccountId()); |
| } |
| return false; |
| } |
| |
| void cacheVisibleTo(ChangeControl ctl) { |
| visibleTo = ctl.getUser(); |
| changeControl = ctl; |
| } |
| |
| public Change change() throws OrmException { |
| if (change == null && lazyLoad) { |
| reloadChange(); |
| } |
| return change; |
| } |
| |
| public void setChange(Change c) { |
| change = c; |
| } |
| |
| public Change reloadChange() throws OrmException { |
| try { |
| notes = notesFactory.createChecked(db, project, legacyId); |
| } catch (NoSuchChangeException e) { |
| throw new OrmException("Unable to load change " + legacyId, e); |
| } |
| change = notes.getChange(); |
| setPatchSets(null); |
| return change; |
| } |
| |
| public LabelTypes getLabelTypes() throws OrmException { |
| if (labelTypes == null) { |
| ProjectState state; |
| try { |
| state = projectCache.checkedGet(project()); |
| } catch (IOException e) { |
| throw new OrmException("project state not available", e); |
| } |
| labelTypes = |
| state.getLabelTypes(changeControl().getChange().getDest(), changeControl().getUser()); |
| } |
| return labelTypes; |
| } |
| |
| public ChangeNotes notes() throws OrmException { |
| if (notes == null) { |
| if (!lazyLoad) { |
| throw new OrmException("ChangeNotes not available, lazyLoad = false"); |
| } |
| notes = notesFactory.create(db, project(), legacyId); |
| } |
| return notes; |
| } |
| |
| public PatchSet currentPatchSet() throws OrmException { |
| if (currentPatchSet == null) { |
| Change c = change(); |
| if (c == null) { |
| return null; |
| } |
| for (PatchSet p : patchSets()) { |
| if (p.getId().equals(c.currentPatchSetId())) { |
| currentPatchSet = p; |
| return p; |
| } |
| } |
| } |
| return currentPatchSet; |
| } |
| |
| public List<PatchSetApproval> currentApprovals() throws OrmException { |
| if (currentApprovals == null) { |
| if (!lazyLoad) { |
| return Collections.emptyList(); |
| } |
| Change c = change(); |
| if (c == null) { |
| currentApprovals = Collections.emptyList(); |
| } else { |
| try { |
| currentApprovals = |
| ImmutableList.copyOf( |
| approvalsUtil.byPatchSet( |
| db, notes(), changeControl().getUser(), c.currentPatchSetId(), null, null)); |
| } catch (OrmException 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() throws IOException, OrmException { |
| if (commitMessage == null) { |
| if (!loadCommitData()) { |
| return null; |
| } |
| } |
| return commitMessage; |
| } |
| |
| public List<FooterLine> commitFooters() throws IOException, OrmException { |
| if (commitFooters == null) { |
| if (!loadCommitData()) { |
| return null; |
| } |
| } |
| return commitFooters; |
| } |
| |
| public ListMultimap<String, String> trackingFooters() throws IOException, OrmException { |
| return trackingFooters.extract(commitFooters()); |
| } |
| |
| public PersonIdent getAuthor() throws IOException, OrmException { |
| if (author == null) { |
| if (!loadCommitData()) { |
| return null; |
| } |
| } |
| return author; |
| } |
| |
| public PersonIdent getCommitter() throws IOException, OrmException { |
| if (committer == null) { |
| if (!loadCommitData()) { |
| return null; |
| } |
| } |
| return committer; |
| } |
| |
| private boolean loadCommitData() |
| throws OrmException, RepositoryNotFoundException, IOException, MissingObjectException, |
| IncorrectObjectTypeException { |
| PatchSet ps = currentPatchSet(); |
| if (ps == null) { |
| return false; |
| } |
| String sha1 = ps.getRevision().get(); |
| try (Repository repo = repoManager.openRepository(project()); |
| RevWalk walk = new RevWalk(repo)) { |
| RevCommit c = walk.parseCommit(ObjectId.fromString(sha1)); |
| commitMessage = c.getFullMessage(); |
| commitFooters = c.getFooterLines(); |
| author = c.getAuthorIdent(); |
| committer = c.getCommitterIdent(); |
| } |
| return true; |
| } |
| |
| /** |
| * @return patches for the change, in patch set ID order. |
| * @throws OrmException an error occurred reading the database. |
| */ |
| public Collection<PatchSet> patchSets() throws OrmException { |
| if (patchSets == null) { |
| patchSets = psUtil.byChange(db, notes()); |
| } |
| return patchSets; |
| } |
| |
| /** |
| * @return patches for the change visible to the current user. |
| * @throws OrmException an error occurred reading the database. |
| */ |
| public Collection<PatchSet> visiblePatchSets() throws OrmException { |
| Predicate<? super PatchSet> predicate = |
| ps -> { |
| try { |
| return changeControl().isPatchVisible(ps, db); |
| } catch (OrmException e) { |
| return false; |
| } |
| }; |
| return patchSets().stream().filter(predicate).collect(toList()); |
| } |
| |
| public void setPatchSets(Collection<PatchSet> patchSets) { |
| this.currentPatchSet = null; |
| this.patchSets = patchSets; |
| } |
| |
| /** |
| * @return patch with the given ID, or null if it does not exist. |
| * @throws OrmException an error occurred reading the database. |
| */ |
| public PatchSet patchSet(PatchSet.Id psId) throws OrmException { |
| if (currentPatchSet != null && currentPatchSet.getId().equals(psId)) { |
| return currentPatchSet; |
| } |
| for (PatchSet ps : patchSets()) { |
| if (ps.getId().equals(psId)) { |
| return ps; |
| } |
| } |
| return null; |
| } |
| |
| /** |
| * @return all patch set approvals for the change, keyed by ID, ordered by timestamp within each |
| * patch set. |
| * @throws OrmException an error occurred reading the database. |
| */ |
| public ListMultimap<PatchSet.Id, PatchSetApproval> approvals() throws OrmException { |
| if (allApprovals == null) { |
| if (!lazyLoad) { |
| return ImmutableListMultimap.of(); |
| } |
| allApprovals = approvalsUtil.byChange(db, notes()); |
| } |
| return allApprovals; |
| } |
| |
| /** |
| * @return The submit ('SUBM') approval label |
| * @throws OrmException an error occurred reading the database. |
| */ |
| public Optional<PatchSetApproval> getSubmitApproval() throws OrmException { |
| return currentApprovals().stream().filter(PatchSetApproval::isLegacySubmit).findFirst(); |
| } |
| |
| public ReviewerSet reviewers() throws OrmException { |
| if (reviewers == null) { |
| if (!lazyLoad) { |
| return ReviewerSet.empty(); |
| } |
| reviewers = approvalsUtil.getReviewers(notes(), approvals().values()); |
| } |
| return reviewers; |
| } |
| |
| public void setReviewers(ReviewerSet reviewers) { |
| this.reviewers = reviewers; |
| } |
| |
| public ReviewerSet getReviewers() { |
| return reviewers; |
| } |
| |
| public ReviewerByEmailSet reviewersByEmail() throws OrmException { |
| 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() throws OrmException { |
| 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() throws OrmException { |
| if (pendingReviewersByEmail == null) { |
| if (!lazyLoad) { |
| return ReviewerByEmailSet.empty(); |
| } |
| pendingReviewersByEmail = notes().getPendingReviewersByEmail(); |
| } |
| return pendingReviewersByEmail; |
| } |
| |
| public List<ReviewerStatusUpdate> reviewerUpdates() throws OrmException { |
| 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<Comment> publishedComments() throws OrmException { |
| if (publishedComments == null) { |
| if (!lazyLoad) { |
| return Collections.emptyList(); |
| } |
| publishedComments = commentsUtil.publishedByChange(db, notes()); |
| } |
| return publishedComments; |
| } |
| |
| public Collection<RobotComment> robotComments() throws OrmException { |
| if (robotComments == null) { |
| if (!lazyLoad) { |
| return Collections.emptyList(); |
| } |
| robotComments = commentsUtil.robotCommentsByChange(notes()); |
| } |
| return robotComments; |
| } |
| |
| public Integer unresolvedCommentCount() throws OrmException { |
| if (unresolvedCommentCount == null) { |
| if (!lazyLoad) { |
| return null; |
| } |
| |
| List<Comment> comments = |
| Stream.concat(publishedComments().stream(), robotComments().stream()).collect(toList()); |
| Set<String> nonLeafSet = comments.stream().map(c -> c.parentUuid).collect(toSet()); |
| |
| Long count = |
| comments.stream().filter(c -> (c.unresolved && !nonLeafSet.contains(c.key.uuid))).count(); |
| unresolvedCommentCount = count.intValue(); |
| } |
| return unresolvedCommentCount; |
| } |
| |
| public void setUnresolvedCommentCount(Integer count) { |
| this.unresolvedCommentCount = count; |
| } |
| |
| public List<ChangeMessage> messages() throws OrmException { |
| if (messages == null) { |
| if (!lazyLoad) { |
| return Collections.emptyList(); |
| } |
| messages = cmUtil.byChange(db, notes()); |
| } |
| return messages; |
| } |
| |
| public List<SubmitRecord> submitRecords(SubmitRuleOptions options) throws OrmException { |
| List<SubmitRecord> records = submitRecords.get(options); |
| if (records == null) { |
| if (!lazyLoad) { |
| return Collections.emptyList(); |
| } |
| records = |
| new SubmitRuleEvaluator(accountCache, accounts, emails, this) |
| .setOptions(options) |
| .evaluate(); |
| submitRecords.put(options, records); |
| } |
| return records; |
| } |
| |
| @Nullable |
| public List<SubmitRecord> getSubmitRecords(SubmitRuleOptions options) { |
| return submitRecords.get(options); |
| } |
| |
| public void setSubmitRecords(SubmitRuleOptions options, List<SubmitRecord> records) { |
| submitRecords.put(options, records); |
| } |
| |
| public SubmitTypeRecord submitTypeRecord() throws OrmException { |
| if (submitTypeRecord == null) { |
| submitTypeRecord = |
| new SubmitRuleEvaluator(accountCache, accounts, emails, this).getSubmitType(); |
| } |
| return submitTypeRecord; |
| } |
| |
| public void setMergeable(Boolean mergeable) { |
| this.mergeable = mergeable; |
| } |
| |
| public Boolean isMergeable() throws OrmException { |
| if (mergeable == null) { |
| Change c = change(); |
| if (c == null) { |
| return null; |
| } |
| if (c.getStatus() == Change.Status.MERGED) { |
| mergeable = true; |
| } else if (c.getStatus() == Change.Status.ABANDONED) { |
| return null; |
| } else if (c.isWorkInProgress()) { |
| return null; |
| } else { |
| if (!lazyLoad) { |
| return null; |
| } |
| PatchSet ps = currentPatchSet(); |
| try { |
| if (ps == null || !changeControl().isPatchVisible(ps, db)) { |
| return null; |
| } |
| } catch (OrmException e) { |
| if (e.getCause() instanceof NoSuchChangeException) { |
| return null; |
| } |
| throw e; |
| } |
| |
| try (Repository repo = repoManager.openRepository(project())) { |
| Ref ref = repo.getRefDatabase().exactRef(c.getDest().get()); |
| 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())).mergeStrategyName(); |
| mergeable = |
| mergeabilityCache.get( |
| ObjectId.fromString(ps.getRevision().get()), |
| ref, |
| str.type, |
| mergeStrategy, |
| c.getDest(), |
| repo); |
| } catch (IOException e) { |
| throw new OrmException(e); |
| } |
| } |
| } |
| return mergeable; |
| } |
| |
| public Set<Account.Id> editsByUser() throws OrmException { |
| return editRefs().keySet(); |
| } |
| |
| public Map<Account.Id, Ref> editRefs() throws OrmException { |
| if (editsByUser == null) { |
| if (!lazyLoad) { |
| return Collections.emptyMap(); |
| } |
| Change c = change(); |
| if (c == null) { |
| return Collections.emptyMap(); |
| } |
| editsByUser = new HashMap<>(); |
| Change.Id id = checkNotNull(change.getId()); |
| try (Repository repo = repoManager.openRepository(project())) { |
| for (Map.Entry<String, Ref> e : |
| repo.getRefDatabase().getRefs(RefNames.REFS_USERS).entrySet()) { |
| if (id.equals(Change.Id.fromEditRefPart(e.getKey()))) { |
| editsByUser.put(Account.Id.fromRefPart(e.getKey()), e.getValue()); |
| } |
| } |
| } catch (IOException e) { |
| throw new OrmException(e); |
| } |
| } |
| return editsByUser; |
| } |
| |
| public Set<Account.Id> draftsByUser() throws OrmException { |
| return draftRefs().keySet(); |
| } |
| |
| public Map<Account.Id, Ref> draftRefs() throws OrmException { |
| if (draftsByUser == null) { |
| if (!lazyLoad) { |
| return Collections.emptyMap(); |
| } |
| Change c = change(); |
| if (c == null) { |
| return Collections.emptyMap(); |
| } |
| |
| draftsByUser = new HashMap<>(); |
| if (notesMigration.readChanges()) { |
| 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); |
| } |
| } |
| } else { |
| for (Comment sc : commentsUtil.draftByChange(db, notes())) { |
| draftsByUser.put(sc.author.getId(), null); |
| } |
| } |
| } |
| return draftsByUser; |
| } |
| |
| public Set<Account.Id> reviewedBy() throws OrmException { |
| 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() throws OrmException { |
| 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() throws OrmException { |
| 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() throws OrmException { |
| if (starRefs == null) { |
| if (!lazyLoad) { |
| return ImmutableMap.of(); |
| } |
| starRefs = checkNotNull(starredChangesUtil).byChange(legacyId); |
| } |
| return starRefs; |
| } |
| |
| public Set<String> stars(Account.Id accountId) throws OrmException { |
| 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(); |
| } |
| |
| /** |
| * @return {@code null} if {@code revertOf} is {@code null}; true if the change is a pure revert; |
| * false otherwise. |
| */ |
| @Nullable |
| public Boolean isPureRevert() throws OrmException { |
| if (change().getRevertOf() == null) { |
| return null; |
| } |
| try { |
| return pureRevert.getPureRevert(notes()).isPureRevert; |
| } catch (IOException | BadRequestException | AuthException | ResourceConflictException e) { |
| throw new OrmException("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 ImmutableList<byte[]> getRefStates() { |
| return refStates; |
| } |
| |
| public void setRefStates(Iterable<byte[]> refStates) { |
| this.refStates = ImmutableList.copyOf(refStates); |
| } |
| |
| 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 Timestamp 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(); |
| } |
| } |