blob: ea3dd0a33b4f8677ecdad2c5a1c530fc7327b5de [file] [log] [blame]
// Copyright (C) 2023 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.notedb;
import com.google.gerrit.entities.Account;
import com.google.gerrit.entities.Change;
import com.google.gerrit.entities.Comment;
import com.google.gerrit.entities.HumanComment;
import com.google.gerrit.entities.PatchSet;
import com.google.gerrit.entities.RefNames;
import com.google.gerrit.exceptions.StorageException;
import com.google.gerrit.server.CommentsUtil;
import com.google.gerrit.server.DraftCommentsReader;
import com.google.gerrit.server.IdentifiedUser;
import com.google.gerrit.server.config.AllUsersName;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.query.change.ChangeNumberVirtualIdAlgorithm;
import com.google.inject.Singleton;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import javax.inject.Inject;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.Repository;
@Singleton
public class DraftCommentsNotesReader implements DraftCommentsReader {
private final DraftCommentNotes.Factory draftCommentNotesFactory;
private final GitRepositoryManager repoManager;
private final AllUsersName allUsers;
private final ChangeNumberVirtualIdAlgorithm virtualIdAlgorithm;
@Inject
DraftCommentsNotesReader(
DraftCommentNotes.Factory draftCommentNotesFactory,
GitRepositoryManager repoManager,
AllUsersName allUsers,
ChangeNumberVirtualIdAlgorithm virtualIdAlgorithm) {
this.draftCommentNotesFactory = draftCommentNotesFactory;
this.repoManager = repoManager;
this.allUsers = allUsers;
this.virtualIdAlgorithm = virtualIdAlgorithm;
}
@Override
public Optional<HumanComment> getDraftComment(
ChangeNotes notes, IdentifiedUser author, Comment.Key key) {
return getDraftsByChangeAndDraftAuthor(notes, author.getAccountId()).stream()
.filter(c -> key.equals(c.key))
.findFirst();
}
@Override
public List<HumanComment> getDraftsByChangeAndDraftAuthor(ChangeNotes notes, Account.Id author) {
return sort(new ArrayList<>(notes.getDraftComments(author, getVirtualId(notes))));
}
@Override
public List<HumanComment> getDraftsByChangeAndDraftAuthor(Change.Id changeId, Account.Id author) {
return sort(
new ArrayList<>(draftCommentNotesFactory.create(changeId, author).load().getComments()));
}
@Override
public List<HumanComment> getDraftsByPatchSetAndDraftAuthor(
ChangeNotes notes, PatchSet.Id psId, Account.Id author) {
return sort(
notes.load().getDraftComments(author, getVirtualId(notes)).stream()
.filter(c -> c.key.patchSetId == psId.get())
.collect(Collectors.toList()));
}
@Override
public List<HumanComment> getDraftsByChangeForAllAuthors(ChangeNotes notes) {
List<HumanComment> comments = new ArrayList<>();
for (Ref ref : getDraftRefs(notes)) {
Account.Id account = Account.Id.fromRefSuffix(ref.getName());
if (account != null) {
comments.addAll(getDraftsByChangeAndDraftAuthor(notes, account));
}
}
return sort(comments);
}
@Override
public Set<Account.Id> getUsersWithDrafts(ChangeNotes changeNotes) {
Set<Account.Id> res = new HashSet<>();
for (Ref ref : getDraftRefs(changeNotes)) {
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.
&& !changeNotes.getDraftComments(account, ref).isEmpty()) {
res.add(account);
}
}
return res;
}
@Override
public Set<Change.Id> getChangesWithDrafts(Account.Id author) {
Set<Change.Id> changes = new HashSet<>();
try (Repository repo = repoManager.openRepository(allUsers)) {
for (Ref ref : repo.getRefDatabase().getRefsByPrefix(RefNames.REFS_DRAFT_COMMENTS)) {
Integer accountIdFromRef = RefNames.parseRefSuffix(ref.getName());
if (accountIdFromRef != null && accountIdFromRef == author.get()) {
Change.Id changeId = Change.Id.fromAllUsersRef(ref.getName());
if (changeId == null) {
continue;
}
changes.add(changeId);
}
}
} catch (IOException e) {
throw new StorageException(e);
}
return changes;
}
private List<Ref> getDraftRefs(ChangeNotes notes) {
try (Repository repo = repoManager.openRepository(allUsers)) {
return repo.getRefDatabase()
.getRefsByPrefix(RefNames.refsDraftCommentsPrefix(getVirtualId(notes)));
} catch (IOException e) {
throw new StorageException(e);
}
}
private List<HumanComment> sort(List<HumanComment> comments) {
return CommentsUtil.sort(comments);
}
private Change.Id getVirtualId(ChangeNotes notes) {
return virtualIdAlgorithm == null
? notes.getChangeId()
: virtualIdAlgorithm.apply(notes.getServerId(), notes.getChangeId());
}
}