Skip to content

Commit 02335b1

Browse files
committed
config: adds branches to config for tracking branches against remotes, updates clone to track when cloning a branch. Fixes src-d#313
Signed-off-by: Jeremy Chambers <[email protected]>
1 parent c4ace4d commit 02335b1

6 files changed

+507
-7
lines changed

config/branch.go

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
package config
2+
3+
import (
4+
"errors"
5+
6+
"gopkg.in/src-d/go-git.v4/plumbing"
7+
format "gopkg.in/src-d/go-git.v4/plumbing/format/config"
8+
)
9+
10+
var (
11+
errBranchEmptyName = errors.New("branch config: empty name")
12+
errBranchInvalidMerge = errors.New("branch config: invalid merge")
13+
)
14+
15+
// Branch contains information on the
16+
// local branches and which remote to track
17+
type Branch struct {
18+
// Name of branch
19+
Name string
20+
// Remote name of remote to track
21+
Remote string
22+
// Merge is the local refspec for the branch
23+
Merge plumbing.ReferenceName
24+
25+
raw *format.Subsection
26+
}
27+
28+
// Validate validates fields of branch
29+
func (b *Branch) Validate() error {
30+
if b.Name == "" {
31+
return errBranchEmptyName
32+
}
33+
34+
if b.Merge != "" && !b.Merge.IsBranch() {
35+
return errBranchInvalidMerge
36+
}
37+
38+
return nil
39+
}
40+
41+
func (b *Branch) marshal() *format.Subsection {
42+
if b.raw == nil {
43+
b.raw = &format.Subsection{}
44+
}
45+
46+
b.raw.Name = b.Name
47+
48+
if b.Remote == "" {
49+
b.raw.RemoveOption(remoteSection)
50+
} else {
51+
b.raw.SetOption(remoteSection, b.Remote)
52+
}
53+
54+
if b.Merge == "" {
55+
b.raw.RemoveOption(mergeKey)
56+
} else {
57+
b.raw.SetOption(mergeKey, string(b.Merge))
58+
}
59+
60+
return b.raw
61+
}
62+
63+
func (b *Branch) unmarshal(s *format.Subsection) error {
64+
b.raw = s
65+
66+
b.Name = b.raw.Name
67+
b.Remote = b.raw.Options.Get(remoteSection)
68+
b.Merge = plumbing.ReferenceName(b.raw.Options.Get(mergeKey))
69+
70+
return b.Validate()
71+
}

config/branch_test.go

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
package config
2+
3+
import (
4+
. "gopkg.in/check.v1"
5+
"gopkg.in/src-d/go-git.v4/plumbing"
6+
)
7+
8+
type BranchSuite struct{}
9+
10+
var _ = Suite(&BranchSuite{})
11+
12+
func (b *BranchSuite) TestValidateName(c *C) {
13+
goodBranch := Branch{
14+
Name: "master",
15+
Remote: "some_remote",
16+
Merge: "refs/heads/master",
17+
}
18+
badBranch := Branch{
19+
Remote: "some_remote",
20+
Merge: "refs/heads/master",
21+
}
22+
c.Assert(goodBranch.Validate(), IsNil)
23+
c.Assert(badBranch.Validate(), NotNil)
24+
}
25+
26+
func (b *BranchSuite) TestValidateMerge(c *C) {
27+
goodBranch := Branch{
28+
Name: "master",
29+
Remote: "some_remote",
30+
Merge: "refs/heads/master",
31+
}
32+
badBranch := Branch{
33+
Name: "master",
34+
Remote: "some_remote",
35+
Merge: "blah",
36+
}
37+
c.Assert(goodBranch.Validate(), IsNil)
38+
c.Assert(badBranch.Validate(), NotNil)
39+
}
40+
41+
func (b *BranchSuite) TestMarshall(c *C) {
42+
expected := []byte(`[core]
43+
bare = false
44+
[branch "branch-tracking-on-clone"]
45+
remote = fork
46+
merge = refs/heads/branch-tracking-on-clone
47+
`)
48+
49+
cfg := NewConfig()
50+
cfg.Branches["branch-tracking-on-clone"] = &Branch{
51+
Name: "branch-tracking-on-clone",
52+
Remote: "fork",
53+
Merge: plumbing.ReferenceName("refs/heads/branch-tracking-on-clone"),
54+
}
55+
56+
actual, err := cfg.Marshal()
57+
c.Assert(err, IsNil)
58+
c.Assert(string(actual), Equals, string(expected))
59+
}
60+
61+
func (b *BranchSuite) TestUnmarshall(c *C) {
62+
input := []byte(`[core]
63+
bare = false
64+
[branch "branch-tracking-on-clone"]
65+
remote = fork
66+
merge = refs/heads/branch-tracking-on-clone
67+
`)
68+
69+
cfg := NewConfig()
70+
err := cfg.Unmarshal(input)
71+
c.Assert(err, IsNil)
72+
branch := cfg.Branches["branch-tracking-on-clone"]
73+
c.Assert(branch.Name, Equals, "branch-tracking-on-clone")
74+
c.Assert(branch.Remote, Equals, "fork")
75+
c.Assert(branch.Merge, Equals, plumbing.ReferenceName("refs/heads/branch-tracking-on-clone"))
76+
}

config/config.go

+64-2
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ type ConfigStorer interface {
2525
}
2626

2727
var (
28-
ErrInvalid = errors.New("config invalid remote")
28+
ErrInvalid = errors.New("config invalid key in remote or branch")
2929
ErrRemoteConfigNotFound = errors.New("remote config not found")
3030
ErrRemoteConfigEmptyURL = errors.New("remote config: empty URL")
3131
ErrRemoteConfigEmptyName = errors.New("remote config: empty name")
@@ -55,7 +55,9 @@ type Config struct {
5555
// Submodules list of repository submodules, the key of the map is the name
5656
// of the submodule, should equal to Submodule.Name.
5757
Submodules map[string]*Submodule
58-
58+
// Branches list of branches, the key is the branch name and should
59+
// equal Branch.Name
60+
Branches map[string]*Branch
5961
// Raw contains the raw information of a config file. The main goal is
6062
// preserve the parsed information from the original format, to avoid
6163
// dropping unsupported fields.
@@ -67,6 +69,7 @@ func NewConfig() *Config {
6769
config := &Config{
6870
Remotes: make(map[string]*RemoteConfig),
6971
Submodules: make(map[string]*Submodule),
72+
Branches: make(map[string]*Branch),
7073
Raw: format.New(),
7174
}
7275

@@ -87,19 +90,31 @@ func (c *Config) Validate() error {
8790
}
8891
}
8992

93+
for name, b := range c.Branches {
94+
if b.Name != name {
95+
return ErrInvalid
96+
}
97+
98+
if err := b.Validate(); err != nil {
99+
return err
100+
}
101+
}
102+
90103
return nil
91104
}
92105

93106
const (
94107
remoteSection = "remote"
95108
submoduleSection = "submodule"
109+
branchSection = "branch"
96110
coreSection = "core"
97111
packSection = "pack"
98112
fetchKey = "fetch"
99113
urlKey = "url"
100114
bareKey = "bare"
101115
worktreeKey = "worktree"
102116
windowKey = "window"
117+
mergeKey = "merge"
103118

104119
// DefaultPackWindow holds the number of previous objects used to
105120
// generate deltas. The value 10 is the same used by git command.
@@ -121,6 +136,11 @@ func (c *Config) Unmarshal(b []byte) error {
121136
return err
122137
}
123138
c.unmarshalSubmodules()
139+
140+
if err := c.unmarshalBranches(); err != nil {
141+
return err
142+
}
143+
124144
return c.unmarshalRemotes()
125145
}
126146

@@ -172,12 +192,27 @@ func (c *Config) unmarshalSubmodules() {
172192
}
173193
}
174194

195+
func (c *Config) unmarshalBranches() error {
196+
bs := c.Raw.Section(branchSection)
197+
for _, sub := range bs.Subsections {
198+
b := &Branch{}
199+
200+
if err := b.unmarshal(sub); err != nil {
201+
return err
202+
}
203+
204+
c.Branches[b.Name] = b
205+
}
206+
return nil
207+
}
208+
175209
// Marshal returns Config encoded as a git-config file.
176210
func (c *Config) Marshal() ([]byte, error) {
177211
c.marshalCore()
178212
c.marshalPack()
179213
c.marshalRemotes()
180214
c.marshalSubmodules()
215+
c.marshalBranches()
181216

182217
buf := bytes.NewBuffer(nil)
183218
if err := format.NewEncoder(buf).Encode(c.Raw); err != nil {
@@ -245,6 +280,33 @@ func (c *Config) marshalSubmodules() {
245280
}
246281
}
247282

283+
func (c *Config) marshalBranches() {
284+
s := c.Raw.Section(branchSection)
285+
newSubsections := make(format.Subsections, 0, len(c.Branches))
286+
added := make(map[string]bool)
287+
for _, subsection := range s.Subsections {
288+
if branch, ok := c.Branches[subsection.Name]; ok {
289+
newSubsections = append(newSubsections, branch.marshal())
290+
added[subsection.Name] = true
291+
}
292+
}
293+
294+
branchNames := make([]string, 0, len(c.Branches))
295+
for name := range c.Branches {
296+
branchNames = append(branchNames, name)
297+
}
298+
299+
sort.Strings(branchNames)
300+
301+
for _, name := range branchNames {
302+
if !added[name] {
303+
newSubsections = append(newSubsections, c.Branches[name].marshal())
304+
}
305+
}
306+
307+
s.Subsections = newSubsections
308+
}
309+
248310
// RemoteConfig contains the configuration for a given remote repository.
249311
type RemoteConfig struct {
250312
// Name of the remote

0 commit comments

Comments
 (0)