using System;
using System.Collections.Generic;
using System.Text.RegularExpressions;
namespace SourceGit.Git {
///
/// Git branch
///
public class Branch {
private static readonly string PRETTY_FORMAT = @"$%(refname)$%(objectname)$%(HEAD)$%(upstream)$%(upstream:track)$%(contents:subject)";
private static readonly Regex PARSE = new Regex(@"\$(.*)\$(.*)\$([\* ])\$(.*)\$(.*?)\$(.*)");
private static readonly Regex AHEAD = new Regex(@"ahead (\d+)");
private static readonly Regex BEHIND = new Regex(@"behind (\d+)");
///
/// Branch type.
///
public enum Type {
Normal,
Feature,
Release,
Hotfix,
}
///
/// Branch name
///
public string Name { get; set; } = "";
///
/// Full name.
///
public string FullName { get; set; } = "";
///
/// Head ref
///
public string Head { get; set; } = "";
///
/// Subject for head ref.
///
public string HeadSubject { get; set; } = "";
///
/// Is local branch
///
public bool IsLocal { get; set; } = false;
///
/// Branch type.
///
public Type Kind { get; set; } = Type.Normal;
///
/// Remote name. Only used for remote branch
///
public string Remote { get; set; } = "";
///
/// Upstream. Only used for local branches.
///
public string Upstream { get; set; }
///
/// Track information for upstream. Only used for local branches.
///
public string UpstreamTrack { get; set; }
///
/// Is current branch. Only used for local branches.
///
public bool IsCurrent { get; set; }
///
/// Is this branch's HEAD same with upstream?
///
public bool IsSameWithUpstream => string.IsNullOrEmpty(UpstreamTrack);
///
/// Enable filter in log histories.
///
public bool IsFiltered { get; set; }
///
/// Load branches.
///
///
public static List Load(Repository repo) {
var localPrefix = "refs/heads/";
var remotePrefix = "refs/remotes/";
var branches = new List();
var remoteBranches = new List();
repo.RunCommand("branch -l --all -v --format=\"" + PRETTY_FORMAT + "\"", line => {
var match = PARSE.Match(line);
if (!match.Success) return;
var branch = new Branch();
var refname = match.Groups[1].Value;
if (refname.EndsWith("/HEAD")) return;
if (refname.StartsWith(localPrefix, StringComparison.Ordinal)) {
branch.Name = refname.Substring(localPrefix.Length);
branch.IsLocal = true;
} else if (refname.StartsWith(remotePrefix, StringComparison.Ordinal)) {
var name = refname.Substring(remotePrefix.Length);
branch.Remote = name.Substring(0, name.IndexOf('/'));
branch.Name = name;
branch.IsLocal = false;
remoteBranches.Add(refname);
}
branch.FullName = refname;
branch.Head = match.Groups[2].Value;
branch.IsCurrent = match.Groups[3].Value == "*";
branch.Upstream = match.Groups[4].Value;
branch.UpstreamTrack = ParseTrack(match.Groups[5].Value);
branch.HeadSubject = match.Groups[6].Value;
branches.Add(branch);
});
// Fixed deleted remote branch
foreach (var b in branches) {
if (!string.IsNullOrEmpty(b.Upstream) && !remoteBranches.Contains(b.Upstream)) {
b.Upstream = null;
}
}
return branches;
}
///
/// Create new branch.
///
///
///
///
public static void Create(Repository repo, string name, string startPoint) {
var errs = repo.RunCommand($"branch {name} {startPoint}", null);
if (errs != null) App.RaiseError(errs);
}
///
/// Rename branch
///
///
///
public void Rename(Repository repo, string name) {
var errs = repo.RunCommand($"branch -M {Name} {name}", null);
if (errs != null) App.RaiseError(errs);
}
///
/// Change upstream
///
///
///
public void SetUpstream(Repository repo, string upstream) {
var errs = repo.RunCommand($"branch {Name} -u {upstream}", null);
if (errs != null) App.RaiseError(errs);
repo.Branches(true);
repo.OnBranchChanged?.Invoke();
}
///
/// Delete branch.
///
///
public void Delete(Repository repo) {
string errs = null;
if (!IsLocal) {
errs = repo.RunCommand($"-c credential.helper=manager push {Remote} --delete {Name.Substring(Name.IndexOf('/')+1)}", null);
} else {
errs = repo.RunCommand($"branch -D {Name}", null);
}
if (errs != null) App.RaiseError(errs);
}
private static string ParseTrack(string data) {
if (string.IsNullOrEmpty(data)) return "";
string track = "";
var ahead = AHEAD.Match(data);
if (ahead.Success) {
track += ahead.Groups[1].Value + "↑ ";
}
var behind = BEHIND.Match(data);
if (behind.Success) {
track += behind.Groups[1].Value + "↓";
}
return track.Trim();
}
}
}