-
Notifications
You must be signed in to change notification settings - Fork 892
/
Copy pathBranchUpdater.cs
199 lines (178 loc) · 6.81 KB
/
BranchUpdater.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
using System;
using System.Globalization;
using LibGit2Sharp.Core;
namespace LibGit2Sharp
{
/// <summary>
/// Exposes properties of a branch that can be updated.
/// </summary>
public class BranchUpdater
{
private readonly Repository repo;
private readonly Branch branch;
/// <summary>
/// Needed for mocking purposes.
/// </summary>
protected BranchUpdater()
{ }
internal BranchUpdater(Repository repo, Branch branch)
{
Ensure.ArgumentNotNull(repo, "repo");
Ensure.ArgumentNotNull(branch, "branch");
this.repo = repo;
this.branch = branch;
}
/// <summary>
/// Sets the upstream information for the branch.
/// <para>
/// Passing null or string.Empty will unset the upstream.
/// </para>
/// <para>
/// The upstream branch name is with respect to the current repository.
/// So, passing "refs/remotes/origin/master" will set the current branch
/// to track "refs/heads/master" on the origin. Passing in
/// "refs/heads/master" will result in the branch tracking the local
/// master branch.
/// </para>
/// </summary>
public virtual string TrackedBranch
{
set
{
if (string.IsNullOrEmpty(value))
{
UnsetUpstream();
return;
}
SetUpstream(value);
}
}
/// <summary>
/// Set the upstream branch for this branch.
/// <para>
/// To track the "master" branch on the "origin" remote, set the
/// <see cref="Remote"/> property to "origin" and the <see cref="UpstreamBranch"/>
/// property to "refs/heads/master".
/// </para>
/// </summary>
public virtual string UpstreamBranch
{
set
{
SetUpstreamBranch(value);
}
}
/// <summary>
/// Set the upstream remote for this branch.
/// <para>
/// To track the "master" branch on the "origin" remote, set the
/// <see cref="Remote"/> property to "origin" and the <see cref="UpstreamBranch"/>
/// property to "refs/heads/master".
/// </para>
/// </summary>
public virtual string Remote
{
set
{
SetUpstreamRemote(value);
}
}
private void UnsetUpstream()
{
SetUpstreamRemote(string.Empty);
SetUpstreamBranch(string.Empty);
}
/// <summary>
/// Set the upstream information for the current branch.
/// <para>
/// The upstream branch name is with respect to the current repository.
/// So, passing "refs/remotes/origin/master" will set the current branch
/// to track "refs/heads/master" on the origin. Passing in
/// "refs/heads/master" will result in the branch tracking the local
/// master branch.
/// </para>
/// </summary>
/// <param name="upstreamBranchName">The remote branch to track (e.g. refs/remotes/origin/master).</param>
private void SetUpstream(string upstreamBranchName)
{
if (branch.IsRemote)
{
throw new LibGit2SharpException("Cannot set upstream branch on a remote branch.");
}
string remoteName;
string branchName;
GetUpstreamInformation(upstreamBranchName, out remoteName, out branchName);
SetUpstreamRemote(remoteName);
SetUpstreamBranch(branchName);
}
/// <summary>
/// Set the upstream merge branch for the local branch.
/// </summary>
/// <param name="mergeBranchName">The merge branch in the upstream remote's namespace.</param>
private void SetUpstreamBranch(string mergeBranchName)
{
string configKey = string.Format(CultureInfo.InvariantCulture, "branch.{0}.merge", branch.FriendlyName);
if (string.IsNullOrEmpty(mergeBranchName))
{
repo.Config.Unset(configKey);
}
else
{
repo.Config.Set(configKey, mergeBranchName);
}
}
/// <summary>
/// Set the upstream remote for the local branch.
/// </summary>
/// <param name="remoteName">The name of the remote to set as the upstream branch.</param>
private void SetUpstreamRemote(string remoteName)
{
string configKey = string.Format(CultureInfo.InvariantCulture, "branch.{0}.remote", branch.FriendlyName);
if (string.IsNullOrEmpty(remoteName))
{
repo.Config.Unset(configKey);
}
else
{
if (!remoteName.Equals(".", StringComparison.Ordinal))
{
// Verify that remote exists.
using (repo.Network.Remotes.RemoteForName(remoteName)) { }
}
repo.Config.Set(configKey, remoteName);
}
}
/// <summary>
/// Get the upstream remote and merge branch name from a Canonical branch name.
/// This will return the remote name (or ".") if a local branch for the remote name.
/// </summary>
/// <param name="canonicalName">The canonical branch name to parse.</param>
/// <param name="remoteName">The name of the corresponding remote the branch belongs to
/// or "." if it is a local branch.</param>
/// <param name="mergeBranchName">The name of the upstream branch to merge into.</param>
private void GetUpstreamInformation(string canonicalName, out string remoteName, out string mergeBranchName)
{
remoteName = null;
mergeBranchName = null;
if (canonicalName.LooksLikeLocalBranch())
{
remoteName = ".";
mergeBranchName = canonicalName;
}
else if (canonicalName.LooksLikeRemoteTrackingBranch())
{
remoteName = Proxy.git_branch_remote_name(repo.Handle, canonicalName, true);
using (var remote = repo.Network.Remotes.RemoteForName(remoteName))
{
mergeBranchName = remote.FetchSpecTransformToSource(canonicalName);
}
}
else
{
throw new ArgumentException(string.Format(CultureInfo.InvariantCulture,
"'{0}' does not look like a valid canonical branch name.",
canonicalName));
}
}
}
}