org.sonarsource.sonarlint.ls.util.GitUtils Maven / Gradle / Ivy
/*
* SonarLint Language Server
* Copyright (C) 2009-2024 SonarSource SA
* mailto:info AT sonarsource DOT com
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
package org.sonarsource.sonarlint.ls.util;
import java.io.IOException;
import java.net.URI;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.eclipse.jgit.lib.Constants;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.lib.RepositoryBuilder;
import org.eclipse.jgit.revwalk.RevWalk;
import org.eclipse.jgit.revwalk.RevWalkUtils;
import org.eclipse.jgit.revwalk.filter.RevFilter;
import org.sonarsource.sonarlint.ls.log.LanguageClientLogger;
import static java.util.Comparator.naturalOrder;
public class GitUtils {
private GitUtils() {
// util class
}
@CheckForNull
public static Repository getRepositoryForDir(Path projectDir, LanguageClientLogger logOutput) {
try {
var builder = new RepositoryBuilder()
.findGitDir(projectDir.toFile())
.setMustExist(true);
if (builder.getGitDir() == null) {
logOutput.error("Not inside a Git work tree: " + projectDir);
return null;
}
return builder.build();
} catch (IOException e) {
logOutput.errorWithStackTrace("Couldn't access repository for path " + projectDir, e);
}
return null;
}
@CheckForNull
public static String electBestMatchingServerBranchForCurrentHead(Repository repo, Set serverCandidateNames,
@Nullable String serverMainBranch, LanguageClientLogger logOutput) {
try {
String currentBranch = repo.getBranch();
if (currentBranch != null && serverCandidateNames.contains(currentBranch)) {
return currentBranch;
}
var head = repo.exactRef(Constants.HEAD);
if (head == null) {
// Not sure if this is possible to not have a HEAD, but just in case
return null;
}
Map> branchesPerDistance = new HashMap<>();
for (String serverBranchName : serverCandidateNames) {
var shortBranchName = Repository.shortenRefName(serverBranchName);
var localFullBranchName = Constants.R_HEADS + shortBranchName;
var branchRef = repo.exactRef(localFullBranchName);
if (branchRef == null) {
continue;
}
int distance = distance(repo, head, branchRef);
branchesPerDistance.computeIfAbsent(distance, d -> new HashSet<>()).add(serverBranchName);
}
if (branchesPerDistance.isEmpty()) {
return null;
}
return branchesPerDistance.keySet().stream().min(naturalOrder())
.map(minDistance -> {
var bestCandidates = branchesPerDistance.get(minDistance);
if (serverMainBranch != null && bestCandidates.contains(serverMainBranch)) {
// Favor the main branch when there are multiple candidates with the same distance
return serverMainBranch;
}
return bestCandidates.iterator().next();
}).orElse(null);
} catch (IOException e) {
logOutput.errorWithStackTrace("Couldn't find best matching branch.", e);
return null;
}
}
public static boolean isCurrentBranch(String folderUri, String expectedBranch, LanguageClientLogger logOutput) {
var repo = GitUtils.getRepositoryForDir(Paths.get(URI.create(folderUri)), logOutput);
if (repo == null) {
return false;
}
try {
var branch = repo.getBranch();
return branch != null && branch.equals(expectedBranch);
} catch (IOException e) {
return false;
}
}
private static int distance(Repository repository, Ref from, Ref to) throws IOException {
try (var walk = new RevWalk(repository)) {
var fromCommit = walk.parseCommit(from.getObjectId());
var toCommit = walk.parseCommit(to.getObjectId());
walk.setRevFilter(RevFilter.MERGE_BASE);
walk.markStart(fromCommit);
walk.markStart(toCommit);
var mergeBase = walk.next();
walk.reset();
walk.setRevFilter(RevFilter.ALL);
int aheadCount = RevWalkUtils.count(walk, fromCommit, mergeBase);
int behindCount = RevWalkUtils.count(walk, toCommit,
mergeBase);
return aheadCount + behindCount;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy