mirror of
https://github.com/sourcegit-scm/sourcegit
synced 2025-06-21 18:35:00 +00:00
120 lines
4.2 KiB
C#
120 lines
4.2 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
|
|
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)\"";
|
|
}
|
|
|
|
public List<Models.Branch> Result(out int localBranchesCount)
|
|
{
|
|
localBranchesCount = 0;
|
|
|
|
var branches = new List<Models.Branch>();
|
|
var rs = ReadToEnd();
|
|
if (!rs.IsSuccess)
|
|
return branches;
|
|
|
|
var lines = rs.StdOut.Split(['\r', '\n'], StringSplitOptions.RemoveEmptyEntries);
|
|
var remoteHeads = new Dictionary<string, string>();
|
|
foreach (var line in lines)
|
|
{
|
|
var b = ParseLine(line);
|
|
if (b != null)
|
|
{
|
|
branches.Add(b);
|
|
if (!b.IsLocal)
|
|
remoteHeads.Add(b.FullName, b.Head);
|
|
else
|
|
localBranchesCount++;
|
|
}
|
|
}
|
|
|
|
foreach (var b in branches)
|
|
{
|
|
if (b.IsLocal && !string.IsNullOrEmpty(b.Upstream))
|
|
{
|
|
if (remoteHeads.TryGetValue(b.Upstream, out var upstreamHead))
|
|
{
|
|
b.IsUpstreamGone = false;
|
|
|
|
if (b.TrackStatus == null)
|
|
b.TrackStatus = new QueryTrackStatus(WorkingDirectory, b.Head, upstreamHead).Result();
|
|
}
|
|
else
|
|
{
|
|
b.IsUpstreamGone = true;
|
|
|
|
if (b.TrackStatus == null)
|
|
b.TrackStatus = new Models.BranchTrackStatus();
|
|
}
|
|
}
|
|
}
|
|
|
|
return branches;
|
|
}
|
|
|
|
private Models.Branch ParseLine(string line)
|
|
{
|
|
var parts = line.Split('\0');
|
|
if (parts.Length != 6)
|
|
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 shortNameIdx = name.IndexOf('/', StringComparison.Ordinal);
|
|
if (shortNameIdx < 0)
|
|
return null;
|
|
|
|
branch.Remote = name.Substring(0, shortNameIdx);
|
|
branch.Name = name.Substring(branch.Remote.Length + 1);
|
|
branch.IsLocal = false;
|
|
}
|
|
else
|
|
{
|
|
branch.Name = refName;
|
|
branch.IsLocal = true;
|
|
}
|
|
|
|
branch.FullName = refName;
|
|
branch.CommitterDate = ulong.Parse(parts[1]);
|
|
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))
|
|
branch.TrackStatus = new Models.BranchTrackStatus();
|
|
|
|
return branch;
|
|
}
|
|
}
|
|
}
|