mirror of
https://fastgit.cc/github.com/sourcegit-scm/sourcegit
synced 2026-04-24 19:02:39 +08:00
119 lines
4.1 KiB
C#
119 lines
4.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace SourceGit.Commands
|
|
{
|
|
public class QueryBranches : Command
|
|
{
|
|
private const string PREFIX_LOCAL = "refs/heads/";
|
|
private const string PREFIX_REMOTE = "refs/remotes/";
|
|
private const string PREFIX_DETACHED_AT = "(HEAD detached at";
|
|
private const string PREFIX_DETACHED_FROM = "(HEAD detached from";
|
|
|
|
public QueryBranches(string repo)
|
|
{
|
|
WorkingDirectory = repo;
|
|
Context = repo;
|
|
Args = "branch -l --all -v --format=\"%(refname)%00%(committerdate:unix)%00%(objectname)%00%(HEAD)%00%(upstream)%00%(upstream:trackshort)%00%(worktreepath)\"";
|
|
}
|
|
|
|
public async Task<List<Models.Branch>> GetResultAsync()
|
|
{
|
|
var branches = new List<Models.Branch>();
|
|
var rs = await ReadToEndAsync().ConfigureAwait(false);
|
|
if (!rs.IsSuccess)
|
|
return branches;
|
|
|
|
var lines = rs.StdOut.Split(['\r', '\n'], StringSplitOptions.RemoveEmptyEntries);
|
|
var mismatched = new HashSet<string>();
|
|
var remotes = new Dictionary<string, Models.Branch>();
|
|
foreach (var line in lines)
|
|
{
|
|
var b = ParseLine(line, mismatched);
|
|
if (b != null)
|
|
{
|
|
branches.Add(b);
|
|
if (!b.IsLocal)
|
|
remotes.Add(b.FullName, b);
|
|
}
|
|
}
|
|
|
|
foreach (var b in branches)
|
|
{
|
|
if (b.IsLocal && !string.IsNullOrEmpty(b.Upstream))
|
|
{
|
|
if (remotes.TryGetValue(b.Upstream, out var upstream))
|
|
{
|
|
b.IsUpstreamGone = false;
|
|
|
|
if (mismatched.Contains(b.FullName))
|
|
await new QueryTrackStatus(WorkingDirectory).GetResultAsync(b, upstream).ConfigureAwait(false);
|
|
}
|
|
else
|
|
{
|
|
b.IsUpstreamGone = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return branches;
|
|
}
|
|
|
|
private Models.Branch ParseLine(string line, HashSet<string> mismatched)
|
|
{
|
|
var parts = line.Split('\0');
|
|
if (parts.Length != 7)
|
|
return null;
|
|
|
|
var branch = new Models.Branch();
|
|
var refName = parts[0];
|
|
if (refName.EndsWith("/HEAD", StringComparison.Ordinal))
|
|
return null;
|
|
|
|
branch.IsDetachedHead = refName.StartsWith(PREFIX_DETACHED_AT, StringComparison.Ordinal) ||
|
|
refName.StartsWith(PREFIX_DETACHED_FROM, StringComparison.Ordinal);
|
|
|
|
if (refName.StartsWith(PREFIX_LOCAL, StringComparison.Ordinal))
|
|
{
|
|
branch.Name = refName.Substring(PREFIX_LOCAL.Length);
|
|
branch.IsLocal = true;
|
|
}
|
|
else if (refName.StartsWith(PREFIX_REMOTE, StringComparison.Ordinal))
|
|
{
|
|
var name = refName.Substring(PREFIX_REMOTE.Length);
|
|
var nameParts = name.Split('/', 2);
|
|
if (nameParts.Length != 2)
|
|
return null;
|
|
|
|
branch.Remote = nameParts[0];
|
|
branch.Name = nameParts[1];
|
|
branch.IsLocal = false;
|
|
}
|
|
else
|
|
{
|
|
branch.Name = refName;
|
|
branch.IsLocal = true;
|
|
}
|
|
|
|
ulong.TryParse(parts[1], out var committerDate);
|
|
|
|
branch.FullName = refName;
|
|
branch.CommitterDate = committerDate;
|
|
branch.Head = parts[2];
|
|
branch.IsCurrent = parts[3] == "*";
|
|
branch.Upstream = parts[4];
|
|
branch.IsUpstreamGone = false;
|
|
|
|
if (branch.IsLocal &&
|
|
!string.IsNullOrEmpty(branch.Upstream) &&
|
|
!string.IsNullOrEmpty(parts[5]) &&
|
|
!parts[5].Equals("=", StringComparison.Ordinal))
|
|
mismatched.Add(branch.FullName);
|
|
|
|
branch.WorktreePath = parts[6];
|
|
return branch;
|
|
}
|
|
}
|
|
}
|