blob: 17a0c9d5b39c7c474c8aaef085a5898c6adc6677 [file] [log] [blame]
// Copyright (C) 2021 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.gerrit.server.change;
import static java.util.stream.Collectors.toSet;
import com.google.common.collect.ImmutableMap;
import com.google.gerrit.entities.Project;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.permissions.PermissionBackend;
import com.google.gerrit.server.permissions.PermissionBackend.RefFilterOptions;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.inject.Inject;
import java.io.IOException;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.eclipse.jgit.errors.IncorrectObjectTypeException;
import org.eclipse.jgit.errors.MissingObjectException;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.RefDatabase;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;
public class IncludedInRefs {
protected final GitRepositoryManager repoManager;
protected final PermissionBackend permissionBackend;
@Inject
IncludedInRefs(GitRepositoryManager repoManager, PermissionBackend permissionBackend) {
this.repoManager = repoManager;
this.permissionBackend = permissionBackend;
}
public Map<String, Set<String>> apply(
Project.NameKey project, Set<String> commits, Set<String> refNames)
throws IOException, PermissionBackendException {
try (Repository repo = repoManager.openRepository(project)) {
Set<Ref> visibleRefs = getVisibleRefs(repo, refNames, project);
if (!visibleRefs.isEmpty()) {
try (RevWalk revWalk = new RevWalk(repo)) {
revWalk.setRetainBody(false);
Set<RevCommit> revCommits = getRevCommits(commits, revWalk);
if (!revCommits.isEmpty()) {
return commitsIncludedIn(
revCommits, IncludedInUtil.getSortedRefs(visibleRefs, revWalk), revWalk);
}
}
}
}
return ImmutableMap.of();
}
private Set<Ref> getVisibleRefs(Repository repo, Set<String> refNames, Project.NameKey project)
throws PermissionBackendException {
RefDatabase refDb = repo.getRefDatabase();
Set<Ref> refs = new HashSet<>();
for (String refName : refNames) {
try {
Ref ref = refDb.exactRef(refName);
if (ref != null) {
refs.add(ref);
}
} catch (IOException e) {
// Ignore and continue to process rest of the refs so as to keep
// the behavior similar to the ref not being visible to the user.
// This will ensure that there is no information leak about the
// ref when the ref is corrupted and is not visible to the user.
}
}
return filterReadableRefs(project, refs, repo);
}
private Set<RevCommit> getRevCommits(Set<String> commits, RevWalk revWalk) throws IOException {
Set<RevCommit> revCommits = new HashSet<>();
for (String commit : commits) {
try {
revCommits.add(revWalk.parseCommit(ObjectId.fromString(commit)));
} catch (MissingObjectException | IncorrectObjectTypeException | IllegalArgumentException e) {
// Ignore and continue to process the rest of the commits so as to keep
// the behavior similar to the commit not being included in any of the
// visible specified refs. This will ensure that there is no information
// leak about the commit when the commit is not visible to the user.
}
}
return revCommits;
}
private Map<String, Set<String>> commitsIncludedIn(
Collection<RevCommit> commits, Collection<Ref> refs, RevWalk revWalk) throws IOException {
Map<String, Set<String>> refsByCommit = new HashMap<>();
for (RevCommit commit : commits) {
List<Ref> matchingRefs = revWalk.getMergedInto(commit, refs);
if (matchingRefs.size() > 0) {
refsByCommit.put(
commit.getName(), matchingRefs.stream().map(Ref::getName).collect(toSet()));
}
}
return refsByCommit;
}
/**
* Filter readable refs according to the caller's refs visibility.
*
* @param project specific Gerrit project.
* @param inputRefs a list of refs
* @param repo repository opened for the Gerrit project.
* @return set of visible refs to the caller
*/
private Set<Ref> filterReadableRefs(Project.NameKey project, Set<Ref> inputRefs, Repository repo)
throws PermissionBackendException {
PermissionBackend.ForProject perm = permissionBackend.currentUser().project(project);
return perm.filter(inputRefs, repo, RefFilterOptions.defaults()).stream().collect(toSet());
}
}