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

Commit 76efca1

Browse files
committed
Add sideband support for push
1 parent 3ca3702 commit 76efca1

File tree

8 files changed

+118
-6
lines changed

8 files changed

+118
-6
lines changed

options.go

+3
Original file line numberDiff line numberDiff line change
@@ -160,6 +160,9 @@ type PushOptions struct {
160160
RefSpecs []config.RefSpec
161161
// Auth credentials, if required, to use with the remote repository.
162162
Auth transport.AuthMethod
163+
// Progress is where the human readable information sent by the server is
164+
// stored, if nil nothing is stored.
165+
Progress sideband.Progress
163166
}
164167

165168
// Validate validates the fields and sets the default values.

plumbing/protocol/packp/updreq.go

+4
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import (
66

77
"gopkg.in/src-d/go-git.v4/plumbing"
88
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp/capability"
9+
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp/sideband"
910
)
1011

1112
var (
@@ -21,6 +22,9 @@ type ReferenceUpdateRequest struct {
2122
Shallow *plumbing.Hash
2223
// Packfile contains an optional packfile reader.
2324
Packfile io.ReadCloser
25+
26+
// Progress receives sideband progress messages from the server
27+
Progress sideband.Progress
2428
}
2529

2630
// New returns a pointer to a new ReferenceUpdateRequest value.

plumbing/transport/http/receive_pack.go

+13
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@ import (
99

1010
"gopkg.in/src-d/go-git.v4/plumbing"
1111
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp"
12+
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp/capability"
13+
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp/sideband"
1214
"gopkg.in/src-d/go-git.v4/plumbing/transport"
1315
"gopkg.in/src-d/go-git.v4/utils/ioutil"
1416
)
@@ -52,6 +54,17 @@ func (s *rpSession) ReceivePack(ctx context.Context, req *packp.ReferenceUpdateR
5254
return nil, err
5355
}
5456

57+
var d *sideband.Demuxer
58+
if req.Capabilities.Supports(capability.Sideband64k) {
59+
d = sideband.NewDemuxer(sideband.Sideband64k, r)
60+
} else if req.Capabilities.Supports(capability.Sideband) {
61+
d = sideband.NewDemuxer(sideband.Sideband, r)
62+
}
63+
if d != nil {
64+
d.Progress = req.Progress
65+
r = d
66+
}
67+
5568
rc := ioutil.NewReadCloser(r, res.Body)
5669

5770
report := packp.NewReportStatus()

plumbing/transport/internal/common/common.go

+16-2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ import (
1818
"gopkg.in/src-d/go-git.v4/plumbing/format/pktline"
1919
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp"
2020
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp/capability"
21+
"gopkg.in/src-d/go-git.v4/plumbing/protocol/packp/sideband"
2122
"gopkg.in/src-d/go-git.v4/plumbing/transport"
2223
"gopkg.in/src-d/go-git.v4/utils/ioutil"
2324
)
@@ -298,13 +299,26 @@ func (s *session) ReceivePack(ctx context.Context, req *packp.ReferenceUpdateReq
298299
}
299300

300301
if !req.Capabilities.Supports(capability.ReportStatus) {
301-
// If we have neither report-status or sideband, we can only
302+
// If we don't have report-status, we can only
302303
// check return value error.
303304
return nil, s.Command.Close()
304305
}
305306

307+
r := s.StdoutContext(ctx)
308+
309+
var d *sideband.Demuxer
310+
if req.Capabilities.Supports(capability.Sideband64k) {
311+
d = sideband.NewDemuxer(sideband.Sideband64k, r)
312+
} else if req.Capabilities.Supports(capability.Sideband) {
313+
d = sideband.NewDemuxer(sideband.Sideband, r)
314+
}
315+
if d != nil {
316+
d.Progress = req.Progress
317+
r = d
318+
}
319+
306320
report := packp.NewReportStatus()
307-
if err := report.Decode(s.StdoutContext(ctx)); err != nil {
321+
if err := report.Decode(r); err != nil {
308322
return nil, err
309323
}
310324

remote.go

+21-4
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,6 @@ func (r *Remote) Push(o *PushOptions) error {
6565
// operation is complete, an error is returned. The context only affects to the
6666
// transport operations.
6767
func (r *Remote) PushContext(ctx context.Context, o *PushOptions) error {
68-
// TODO: Sideband support
6968
if err := o.Validate(); err != nil {
7069
return err
7170
}
@@ -108,9 +107,8 @@ func (r *Remote) PushContext(ctx context.Context, o *PushOptions) error {
108107
return ErrDeleteRefNotSupported
109108
}
110109

111-
req := packp.NewReferenceUpdateRequestFromCapabilities(ar.Capabilities)
112-
if err := r.addReferencesToUpdate(o.RefSpecs, remoteRefs, req); err != nil {
113-
110+
req, err := r.newReferenceUpdateRequest(o, remoteRefs, ar)
111+
if err != nil {
114112
return err
115113
}
116114

@@ -158,6 +156,25 @@ func (r *Remote) PushContext(ctx context.Context, o *PushOptions) error {
158156
return r.updateRemoteReferenceStorage(req, rs)
159157
}
160158

159+
func (r *Remote) newReferenceUpdateRequest(o *PushOptions, remoteRefs storer.ReferenceStorer, ar *packp.AdvRefs) (*packp.ReferenceUpdateRequest, error) {
160+
req := packp.NewReferenceUpdateRequestFromCapabilities(ar.Capabilities)
161+
162+
if o.Progress != nil {
163+
req.Progress = o.Progress
164+
if ar.Capabilities.Supports(capability.Sideband64k) {
165+
req.Capabilities.Set(capability.Sideband64k)
166+
} else if ar.Capabilities.Supports(capability.Sideband) {
167+
req.Capabilities.Set(capability.Sideband)
168+
}
169+
}
170+
171+
if err := r.addReferencesToUpdate(o.RefSpecs, remoteRefs, req); err != nil {
172+
return nil, err
173+
}
174+
175+
return req, nil
176+
}
177+
161178
func (r *Remote) updateRemoteReferenceStorage(
162179
req *packp.ReferenceUpdateRequest,
163180
result *packp.ReportStatus,

repository_test.go

+41
Original file line numberDiff line numberDiff line change
@@ -719,6 +719,47 @@ func (s *RepositorySuite) TestPushContext(c *C) {
719719
c.Assert(err, NotNil)
720720
}
721721

722+
// installPreReceiveHook installs a pre-receive hook in the .git
723+
// directory at path which prints message m before exiting
724+
// successfully.
725+
func installPreReceiveHook(c *C, path, m string) {
726+
hooks := filepath.Join(path, "hooks")
727+
err := os.MkdirAll(hooks, 0777)
728+
c.Assert(err, IsNil)
729+
730+
err = ioutil.WriteFile(filepath.Join(hooks, "pre-receive"), preReceiveHook(m), 0777)
731+
c.Assert(err, IsNil)
732+
}
733+
734+
func (s *RepositorySuite) TestPushWithProgress(c *C) {
735+
url := c.MkDir()
736+
server, err := PlainInit(url, true)
737+
c.Assert(err, IsNil)
738+
739+
m := "Receiving..."
740+
installPreReceiveHook(c, url, m)
741+
742+
_, err = s.Repository.CreateRemote(&config.RemoteConfig{
743+
Name: "bar",
744+
URLs: []string{url},
745+
})
746+
c.Assert(err, IsNil)
747+
748+
var p bytes.Buffer
749+
err = s.Repository.Push(&PushOptions{
750+
RemoteName: "bar",
751+
Progress: &p,
752+
})
753+
c.Assert(err, IsNil)
754+
755+
AssertReferences(c, server, map[string]string{
756+
"refs/heads/master": "6ecf0ef2c2dffb796033e5a02219af86ec6584e5",
757+
"refs/heads/branch": "e8d3ffab552895c19b9fcf7aa264d277cde33881",
758+
})
759+
760+
c.Assert((&p).Bytes(), DeepEquals, []byte(m))
761+
}
762+
722763
func (s *RepositorySuite) TestPushDepth(c *C) {
723764
url := c.MkDir()
724765
server, err := PlainClone(url, true, &CloneOptions{

repository_unix_test.go

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
// +build !plan9,!windows
2+
3+
package git
4+
5+
import "fmt"
6+
7+
// preReceiveHook returns the bytes of a pre-receive hook script
8+
// that prints m before exiting successfully
9+
func preReceiveHook(m string) []byte {
10+
return []byte(fmt.Sprintf("#!/bin/sh\nprintf '%s'\n", m))
11+
}

repository_windows_test.go

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
package git
2+
3+
import "fmt"
4+
5+
// preReceiveHook returns the bytes of a pre-receive hook script
6+
// that prints m before exiting successfully
7+
func preReceiveHook(m string) []byte {
8+
return []byte(fmt.Sprintf("#!C:/Program\\ Files/Git/usr/bin/sh.exe\nprintf '%s'\n", m))
9+
}

0 commit comments

Comments
 (0)