@@ -25,7 +25,7 @@ type ConfigStorer interface {
25
25
}
26
26
27
27
var (
28
- ErrInvalid = errors .New ("config invalid remote" )
28
+ ErrInvalid = errors .New ("config invalid key in remote or branch " )
29
29
ErrRemoteConfigNotFound = errors .New ("remote config not found" )
30
30
ErrRemoteConfigEmptyURL = errors .New ("remote config: empty URL" )
31
31
ErrRemoteConfigEmptyName = errors .New ("remote config: empty name" )
@@ -55,7 +55,9 @@ type Config struct {
55
55
// Submodules list of repository submodules, the key of the map is the name
56
56
// of the submodule, should equal to Submodule.Name.
57
57
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
59
61
// Raw contains the raw information of a config file. The main goal is
60
62
// preserve the parsed information from the original format, to avoid
61
63
// dropping unsupported fields.
@@ -67,6 +69,7 @@ func NewConfig() *Config {
67
69
config := & Config {
68
70
Remotes : make (map [string ]* RemoteConfig ),
69
71
Submodules : make (map [string ]* Submodule ),
72
+ Branches : make (map [string ]* Branch ),
70
73
Raw : format .New (),
71
74
}
72
75
@@ -87,19 +90,31 @@ func (c *Config) Validate() error {
87
90
}
88
91
}
89
92
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
+
90
103
return nil
91
104
}
92
105
93
106
const (
94
107
remoteSection = "remote"
95
108
submoduleSection = "submodule"
109
+ branchSection = "branch"
96
110
coreSection = "core"
97
111
packSection = "pack"
98
112
fetchKey = "fetch"
99
113
urlKey = "url"
100
114
bareKey = "bare"
101
115
worktreeKey = "worktree"
102
116
windowKey = "window"
117
+ mergeKey = "merge"
103
118
104
119
// DefaultPackWindow holds the number of previous objects used to
105
120
// generate deltas. The value 10 is the same used by git command.
@@ -121,6 +136,11 @@ func (c *Config) Unmarshal(b []byte) error {
121
136
return err
122
137
}
123
138
c .unmarshalSubmodules ()
139
+
140
+ if err := c .unmarshalBranches (); err != nil {
141
+ return err
142
+ }
143
+
124
144
return c .unmarshalRemotes ()
125
145
}
126
146
@@ -172,12 +192,27 @@ func (c *Config) unmarshalSubmodules() {
172
192
}
173
193
}
174
194
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
+
175
209
// Marshal returns Config encoded as a git-config file.
176
210
func (c * Config ) Marshal () ([]byte , error ) {
177
211
c .marshalCore ()
178
212
c .marshalPack ()
179
213
c .marshalRemotes ()
180
214
c .marshalSubmodules ()
215
+ c .marshalBranches ()
181
216
182
217
buf := bytes .NewBuffer (nil )
183
218
if err := format .NewEncoder (buf ).Encode (c .Raw ); err != nil {
@@ -245,6 +280,33 @@ func (c *Config) marshalSubmodules() {
245
280
}
246
281
}
247
282
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
+
248
310
// RemoteConfig contains the configuration for a given remote repository.
249
311
type RemoteConfig struct {
250
312
// Name of the remote
0 commit comments