| // Copyright (C) 2014 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package com.google.gerrit.server.restapi.change; |
| |
| import static com.google.common.collect.ImmutableList.toImmutableList; |
| import static com.google.gerrit.server.CommentsUtil.COMMENT_INFO_ORDER; |
| import static java.util.stream.Collectors.toList; |
| |
| import com.google.common.base.Strings; |
| import com.google.common.collect.ImmutableList; |
| import com.google.common.collect.ImmutableMap; |
| import com.google.common.collect.Streams; |
| import com.google.gerrit.common.Nullable; |
| import com.google.gerrit.entities.Change; |
| import com.google.gerrit.entities.Comment; |
| import com.google.gerrit.entities.CommentContext; |
| import com.google.gerrit.entities.FixReplacement; |
| import com.google.gerrit.entities.FixSuggestion; |
| import com.google.gerrit.entities.HumanComment; |
| import com.google.gerrit.entities.Project; |
| import com.google.gerrit.entities.RobotComment; |
| import com.google.gerrit.extensions.client.Comment.Range; |
| import com.google.gerrit.extensions.client.Side; |
| import com.google.gerrit.extensions.common.CommentInfo; |
| import com.google.gerrit.extensions.common.ContextLineInfo; |
| import com.google.gerrit.extensions.common.FixReplacementInfo; |
| import com.google.gerrit.extensions.common.FixSuggestionInfo; |
| import com.google.gerrit.extensions.common.RobotCommentInfo; |
| import com.google.gerrit.extensions.restapi.Url; |
| import com.google.gerrit.server.account.AccountLoader; |
| import com.google.gerrit.server.comment.CommentContextCache; |
| import com.google.gerrit.server.comment.CommentContextKey; |
| import com.google.gerrit.server.permissions.PermissionBackendException; |
| import com.google.inject.Inject; |
| import java.util.ArrayList; |
| import java.util.Collection; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.TreeMap; |
| |
| public class CommentJson { |
| |
| private final AccountLoader.Factory accountLoaderFactory; |
| private final CommentContextCache commentContextCache; |
| |
| private Project.NameKey project; |
| private Change.Id changeId; |
| |
| private boolean fillAccounts = true; |
| private boolean fillPatchSet; |
| private boolean fillCommentContext; |
| private int contextPadding; |
| |
| @Inject |
| CommentJson(AccountLoader.Factory accountLoaderFactory, CommentContextCache commentContextCache) { |
| this.accountLoaderFactory = accountLoaderFactory; |
| this.commentContextCache = commentContextCache; |
| } |
| |
| CommentJson setFillAccounts(boolean fillAccounts) { |
| this.fillAccounts = fillAccounts; |
| return this; |
| } |
| |
| CommentJson setFillPatchSet(boolean fillPatchSet) { |
| this.fillPatchSet = fillPatchSet; |
| return this; |
| } |
| |
| CommentJson setFillCommentContext(boolean fillCommentContext) { |
| this.fillCommentContext = fillCommentContext; |
| return this; |
| } |
| |
| CommentJson setContextPadding(int contextPadding) { |
| this.contextPadding = contextPadding; |
| return this; |
| } |
| |
| CommentJson setProjectKey(Project.NameKey project) { |
| this.project = project; |
| return this; |
| } |
| |
| CommentJson setChangeId(Change.Id changeId) { |
| this.changeId = changeId; |
| return this; |
| } |
| |
| public HumanCommentFormatter newHumanCommentFormatter() { |
| return new HumanCommentFormatter(); |
| } |
| |
| public RobotCommentFormatter newRobotCommentFormatter() { |
| return new RobotCommentFormatter(); |
| } |
| |
| private abstract class BaseCommentFormatter<F extends Comment, T extends CommentInfo> { |
| public T format(F comment) throws PermissionBackendException { |
| AccountLoader loader = fillAccounts ? accountLoaderFactory.create(true) : null; |
| T info = toInfo(comment, loader); |
| if (loader != null) { |
| loader.fill(); |
| } |
| return info; |
| } |
| |
| public Map<String, List<T>> format(Iterable<F> comments) throws PermissionBackendException { |
| AccountLoader loader = fillAccounts ? accountLoaderFactory.create(true) : null; |
| |
| Map<String, List<T>> out = new TreeMap<>(); |
| |
| for (F c : comments) { |
| T o = toInfo(c, loader); |
| List<T> list = out.get(o.path); |
| if (list == null) { |
| list = new ArrayList<>(); |
| out.put(o.path, list); |
| } |
| list.add(o); |
| } |
| |
| out.values().forEach(l -> l.sort(COMMENT_INFO_ORDER)); |
| |
| if (loader != null) { |
| loader.fill(); |
| } |
| |
| List<T> allComments = out.values().stream().flatMap(Collection::stream).collect(toList()); |
| if (fillCommentContext) { |
| addCommentContext(allComments); |
| } |
| allComments.forEach(c -> c.path = null); // we don't need path since it exists in the map keys |
| return out; |
| } |
| |
| public ImmutableList<T> formatAsList(Iterable<F> comments) throws PermissionBackendException { |
| AccountLoader loader = fillAccounts ? accountLoaderFactory.create(true) : null; |
| |
| ImmutableList<T> out = |
| Streams.stream(comments) |
| .map(c -> toInfo(c, loader)) |
| .sorted(COMMENT_INFO_ORDER) |
| .collect(toImmutableList()); |
| |
| if (loader != null) { |
| loader.fill(); |
| } |
| |
| if (fillCommentContext) { |
| addCommentContext(out); |
| } |
| |
| return out; |
| } |
| |
| protected void addCommentContext(List<T> allComments) { |
| List<CommentContextKey> keys = |
| allComments.stream().map(this::createCommentContextKey).collect(toList()); |
| ImmutableMap<CommentContextKey, CommentContext> allContext = commentContextCache.getAll(keys); |
| for (T c : allComments) { |
| CommentContextKey contextKey = createCommentContextKey(c); |
| CommentContext commentContext = allContext.get(contextKey); |
| c.contextLines = toContextLineInfoList(commentContext); |
| c.sourceContentType = commentContext.contentType(); |
| } |
| } |
| |
| protected List<ContextLineInfo> toContextLineInfoList(CommentContext commentContext) { |
| List<ContextLineInfo> result = new ArrayList<>(); |
| for (Map.Entry<Integer, String> e : commentContext.lines().entrySet()) { |
| result.add(new ContextLineInfo(e.getKey(), e.getValue())); |
| } |
| return result; |
| } |
| |
| protected CommentContextKey createCommentContextKey(T r) { |
| return CommentContextKey.builder() |
| .project(project) |
| .changeId(changeId) |
| .id(Url.decode(r.id)) // We reverse the encoding done while filling comment info |
| .path(r.path) |
| .patchset(r.patchSet) |
| .contextPadding(contextPadding) |
| .build(); |
| } |
| |
| protected abstract T toInfo(F comment, AccountLoader loader); |
| |
| protected void fillCommentInfo(Comment c, CommentInfo r, AccountLoader loader) { |
| if (fillPatchSet) { |
| r.patchSet = c.key.patchSetId; |
| } |
| r.id = Url.encode(c.key.uuid); |
| r.path = c.key.filename; |
| if (c.side <= 0) { |
| r.side = Side.PARENT; |
| if (c.side < 0) { |
| r.parent = -c.side; |
| } |
| } |
| if (c.lineNbr > 0) { |
| r.line = c.lineNbr; |
| } |
| r.inReplyTo = Url.encode(c.parentUuid); |
| r.message = Strings.emptyToNull(c.message); |
| r.updated = c.writtenOn; |
| r.range = toRange(c.range); |
| r.tag = c.tag; |
| if (loader != null) { |
| r.author = loader.get(c.author.getId()); |
| } |
| r.commitId = c.getCommitId().getName(); |
| } |
| |
| protected Range toRange(Comment.Range commentRange) { |
| Range range = null; |
| if (commentRange != null) { |
| range = new Range(); |
| range.startLine = commentRange.startLine; |
| range.startCharacter = commentRange.startChar; |
| range.endLine = commentRange.endLine; |
| range.endCharacter = commentRange.endChar; |
| } |
| return range; |
| } |
| } |
| |
| public class HumanCommentFormatter extends BaseCommentFormatter<HumanComment, CommentInfo> { |
| @Override |
| protected CommentInfo toInfo(HumanComment c, AccountLoader loader) { |
| CommentInfo ci = new CommentInfo(); |
| fillCommentInfo(c, ci, loader); |
| ci.unresolved = c.unresolved; |
| return ci; |
| } |
| |
| private HumanCommentFormatter() {} |
| } |
| |
| class RobotCommentFormatter extends BaseCommentFormatter<RobotComment, RobotCommentInfo> { |
| @Override |
| protected RobotCommentInfo toInfo(RobotComment c, AccountLoader loader) { |
| RobotCommentInfo rci = new RobotCommentInfo(); |
| rci.robotId = c.robotId; |
| rci.robotRunId = c.robotRunId; |
| rci.url = c.url; |
| rci.properties = c.properties; |
| rci.fixSuggestions = toFixSuggestionInfos(c.fixSuggestions); |
| fillCommentInfo(c, rci, loader); |
| return rci; |
| } |
| |
| private List<FixSuggestionInfo> toFixSuggestionInfos( |
| @Nullable List<FixSuggestion> fixSuggestions) { |
| if (fixSuggestions == null || fixSuggestions.isEmpty()) { |
| return null; |
| } |
| |
| return fixSuggestions.stream().map(this::toFixSuggestionInfo).collect(toList()); |
| } |
| |
| private FixSuggestionInfo toFixSuggestionInfo(FixSuggestion fixSuggestion) { |
| FixSuggestionInfo fixSuggestionInfo = new FixSuggestionInfo(); |
| fixSuggestionInfo.fixId = fixSuggestion.fixId; |
| fixSuggestionInfo.description = fixSuggestion.description; |
| fixSuggestionInfo.replacements = |
| fixSuggestion.replacements.stream().map(this::toFixReplacementInfo).collect(toList()); |
| return fixSuggestionInfo; |
| } |
| |
| private FixReplacementInfo toFixReplacementInfo(FixReplacement fixReplacement) { |
| FixReplacementInfo fixReplacementInfo = new FixReplacementInfo(); |
| fixReplacementInfo.path = fixReplacement.path; |
| fixReplacementInfo.range = toRange(fixReplacement.range); |
| fixReplacementInfo.replacement = fixReplacement.replacement; |
| return fixReplacementInfo; |
| } |
| |
| private RobotCommentFormatter() {} |
| } |
| } |