Skip to content
This repository was archived by the owner on Sep 11, 2020. It is now read-only.

Commit 174f373

Browse files
authored
Merge pull request #942 from jfontan/improvement/maintain-packfiles-open
storage/dotgit: add KeepDescriptors option
2 parents 2f15838 + 8176f08 commit 174f373

File tree

5 files changed

+131
-2
lines changed

5 files changed

+131
-2
lines changed

storage/filesystem/dotgit/dotgit.go

+42-1
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,9 @@ type Options struct {
6262
// ExclusiveAccess means that the filesystem is not modified externally
6363
// while the repo is open.
6464
ExclusiveAccess bool
65+
// KeepDescriptors makes the file descriptors to be reused but they will
66+
// need to be manually closed calling Close().
67+
KeepDescriptors bool
6568
}
6669

6770
// The DotGit type represents a local git repository on disk. This
@@ -78,6 +81,8 @@ type DotGit struct {
7881
objectMap map[plumbing.Hash]struct{}
7982
packList []plumbing.Hash
8083
packMap map[plumbing.Hash]struct{}
84+
85+
files map[string]billy.File
8186
}
8287

8388
// New returns a DotGit value ready to be used. The path argument must
@@ -123,6 +128,28 @@ func (d *DotGit) Initialize() error {
123128
return nil
124129
}
125130

131+
// Close closes all opened files.
132+
func (d *DotGit) Close() error {
133+
var firstError error
134+
if d.files != nil {
135+
for _, f := range d.files {
136+
err := f.Close()
137+
if err != nil && firstError == nil {
138+
firstError = err
139+
continue
140+
}
141+
}
142+
143+
d.files = nil
144+
}
145+
146+
if firstError != nil {
147+
return firstError
148+
}
149+
150+
return nil
151+
}
152+
126153
// ConfigWriter returns a file pointer for write to the config file
127154
func (d *DotGit) ConfigWriter() (billy.File, error) {
128155
return d.fs.Create(configPath)
@@ -217,12 +244,22 @@ func (d *DotGit) objectPackPath(hash plumbing.Hash, extension string) string {
217244
}
218245

219246
func (d *DotGit) objectPackOpen(hash plumbing.Hash, extension string) (billy.File, error) {
247+
if d.files == nil {
248+
d.files = make(map[string]billy.File)
249+
}
250+
220251
err := d.hasPack(hash)
221252
if err != nil {
222253
return nil, err
223254
}
224255

225-
pack, err := d.fs.Open(d.objectPackPath(hash, extension))
256+
path := d.objectPackPath(hash, extension)
257+
f, ok := d.files[path]
258+
if ok {
259+
return f, nil
260+
}
261+
262+
pack, err := d.fs.Open(path)
226263
if err != nil {
227264
if os.IsNotExist(err) {
228265
return nil, ErrPackfileNotFound
@@ -231,6 +268,10 @@ func (d *DotGit) objectPackOpen(hash plumbing.Hash, extension string) (billy.Fil
231268
return nil, err
232269
}
233270

271+
if d.options.KeepDescriptors && extension == "pack" {
272+
d.files[path] = pack
273+
}
274+
234275
return pack, nil
235276
}
236277

storage/filesystem/dotgit/dotgit_test.go

+39
Original file line numberDiff line numberDiff line change
@@ -465,6 +465,45 @@ func (s *SuiteDotGit) TestObjectPack(c *C) {
465465
c.Assert(filepath.Ext(pack.Name()), Equals, ".pack")
466466
}
467467

468+
func (s *SuiteDotGit) TestObjectPackWithKeepDescriptors(c *C) {
469+
f := fixtures.Basic().ByTag(".git").One()
470+
fs := f.DotGit()
471+
dir := NewWithOptions(fs, Options{KeepDescriptors: true})
472+
473+
pack, err := dir.ObjectPack(f.PackfileHash)
474+
c.Assert(err, IsNil)
475+
c.Assert(filepath.Ext(pack.Name()), Equals, ".pack")
476+
477+
// Move to an specific offset
478+
pack.Seek(42, os.SEEK_SET)
479+
480+
pack2, err := dir.ObjectPack(f.PackfileHash)
481+
c.Assert(err, IsNil)
482+
483+
// If the file is the same the offset should be the same
484+
offset, err := pack2.Seek(0, os.SEEK_CUR)
485+
c.Assert(err, IsNil)
486+
c.Assert(offset, Equals, int64(42))
487+
488+
err = dir.Close()
489+
c.Assert(err, IsNil)
490+
491+
pack2, err = dir.ObjectPack(f.PackfileHash)
492+
c.Assert(err, IsNil)
493+
494+
// If the file is opened again its offset should be 0
495+
offset, err = pack2.Seek(0, os.SEEK_CUR)
496+
c.Assert(err, IsNil)
497+
c.Assert(offset, Equals, int64(0))
498+
499+
err = pack2.Close()
500+
c.Assert(err, IsNil)
501+
502+
err = dir.Close()
503+
c.Assert(err, NotNil)
504+
505+
}
506+
468507
func (s *SuiteDotGit) TestObjectPackIdx(c *C) {
469508
f := fixtures.Basic().ByTag(".git").One()
470509
fs := f.DotGit()

storage/filesystem/object.go

+9-1
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,7 @@ func (s *ObjectStorage) loadIdxFile(h plumbing.Hash) (err error) {
7474
}
7575

7676
defer ioutil.CheckClose(f, &err)
77+
7778
idxf := idxfile.NewMemoryIndex()
7879
d := idxfile.NewDecoder(f)
7980
if err = d.Decode(idxf); err != nil {
@@ -280,7 +281,9 @@ func (s *ObjectStorage) getFromPackfile(h plumbing.Hash, canBeDelta bool) (
280281
return nil, err
281282
}
282283

283-
defer ioutil.CheckClose(f, &err)
284+
if !s.options.KeepDescriptors {
285+
defer ioutil.CheckClose(f, &err)
286+
}
284287

285288
idx := s.index[pack]
286289
if canBeDelta {
@@ -423,6 +426,11 @@ func (s *ObjectStorage) buildPackfileIters(t plumbing.ObjectType, seen map[plumb
423426
}, nil
424427
}
425428

429+
// Close closes all opened files.
430+
func (s *ObjectStorage) Close() error {
431+
return s.dir.Close()
432+
}
433+
426434
type lazyPackfilesIter struct {
427435
hashes []plumbing.Hash
428436
open func(h plumbing.Hash) (storer.EncodedObjectIter, error)

storage/filesystem/object_test.go

+37
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ package filesystem
22

33
import (
44
"io/ioutil"
5+
"os"
56
"testing"
67

78
"gopkg.in/src-d/go-git.v4/plumbing"
@@ -48,6 +49,42 @@ func (s *FsSuite) TestGetFromPackfile(c *C) {
4849
})
4950
}
5051

52+
func (s *FsSuite) TestGetFromPackfileKeepDescriptors(c *C) {
53+
fixtures.Basic().ByTag(".git").Test(c, func(f *fixtures.Fixture) {
54+
fs := f.DotGit()
55+
dg := dotgit.NewWithOptions(fs, dotgit.Options{KeepDescriptors: true})
56+
o, err := NewObjectStorageWithOptions(dg, Options{KeepDescriptors: true})
57+
c.Assert(err, IsNil)
58+
59+
expected := plumbing.NewHash("6ecf0ef2c2dffb796033e5a02219af86ec6584e5")
60+
obj, err := o.EncodedObject(plumbing.AnyObject, expected)
61+
c.Assert(err, IsNil)
62+
c.Assert(obj.Hash(), Equals, expected)
63+
64+
packfiles, err := dg.ObjectPacks()
65+
c.Assert(err, IsNil)
66+
67+
pack1, err := dg.ObjectPack(packfiles[0])
68+
c.Assert(err, IsNil)
69+
70+
pack1.Seek(42, os.SEEK_SET)
71+
72+
err = o.Close()
73+
c.Assert(err, IsNil)
74+
75+
pack2, err := dg.ObjectPack(packfiles[0])
76+
c.Assert(err, IsNil)
77+
78+
offset, err := pack2.Seek(0, os.SEEK_CUR)
79+
c.Assert(err, IsNil)
80+
c.Assert(offset, Equals, int64(0))
81+
82+
err = o.Close()
83+
c.Assert(err, IsNil)
84+
85+
})
86+
}
87+
5188
func (s *FsSuite) TestGetFromPackfileMultiplePackfiles(c *C) {
5289
fs := fixtures.ByTag(".git").ByTag("multi-packfile").One().DotGit()
5390
o, err := NewObjectStorage(dotgit.New(fs))

storage/filesystem/storage.go

+4
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,9 @@ type Options struct {
2727
// ExclusiveAccess means that the filesystem is not modified externally
2828
// while the repo is open.
2929
ExclusiveAccess bool
30+
// KeepDescriptors makes the file descriptors to be reused but they will
31+
// need to be manually closed calling Close().
32+
KeepDescriptors bool
3033
}
3134

3235
// NewStorage returns a new Storage backed by a given `fs.Filesystem`
@@ -41,6 +44,7 @@ func NewStorageWithOptions(
4144
) (*Storage, error) {
4245
dirOps := dotgit.Options{
4346
ExclusiveAccess: ops.ExclusiveAccess,
47+
KeepDescriptors: ops.KeepDescriptors,
4448
}
4549

4650
dir := dotgit.NewWithOptions(fs, dirOps)

0 commit comments

Comments
 (0)