Skip to content

Commit

Permalink
Merge pull request moby#2843 from fenollp/ssh-buildctl
Browse files Browse the repository at this point in the history
Port SSH connhelper from github.com/docker/cli/cli/connhelper/ssh
  • Loading branch information
tonistiigi committed May 4, 2022
2 parents 8e37441 + 5d33eed commit a7fa387
Show file tree
Hide file tree
Showing 3 changed files with 118 additions and 0 deletions.
78 changes: 78 additions & 0 deletions client/connhelper/ssh/ssh.go
@@ -0,0 +1,78 @@
// Package ssh provides connhelper for ssh://<SSH URL>
package ssh

import (
"context"
"net"
"net/url"

"github.com/docker/cli/cli/connhelper/commandconn"
"github.com/moby/buildkit/client/connhelper"
"github.com/pkg/errors"
)

func init() {
connhelper.Register("ssh", Helper)
}

// Helper returns helper for connecting through an SSH URL.
func Helper(u *url.URL) (*connhelper.ConnectionHelper, error) {
sp, err := SpecFromURL(u)
if err != nil {
return nil, err
}
return &connhelper.ConnectionHelper{
ContextDialer: func(ctx context.Context, addr string) (net.Conn, error) {
args := []string{}
if sp.User != "" {
args = append(args, "-l", sp.User)
}
if sp.Port != "" {
args = append(args, "-p", sp.Port)
}
args = append(args, "--", sp.Host)
args = append(args, "buildctl")
if socket := sp.Socket; socket != "" {
args = append(args, "--addr", "unix://"+socket)
}
args = append(args, "dial-stdio")
// using background context because context remains active for the duration of the process, after dial has completed
return commandconn.New(context.Background(), "ssh", args...)
},
}, nil
}

// Spec
type Spec struct {
User string
Host string
Port string
Socket string
}

// SpecFromURL creates Spec from URL.
// URL is like ssh://<user>@host:<port>
// Only <host> part is mandatory.
func SpecFromURL(u *url.URL) (*Spec, error) {
sp := Spec{
Host: u.Hostname(),
Port: u.Port(),
Socket: u.Path,
}
if user := u.User; user != nil {
sp.User = user.Username()
if _, ok := user.Password(); ok {
return nil, errors.New("plain-text password is not supported")
}
}
if sp.Host == "" {
return nil, errors.Errorf("no host specified")
}
if u.RawQuery != "" {
return nil, errors.Errorf("extra query after the host: %q", u.RawQuery)
}
if u.Fragment != "" {
return nil, errors.Errorf("extra fragment after the host: %q", u.Fragment)
}
return &sp, nil
}
39 changes: 39 additions & 0 deletions client/connhelper/ssh/ssh_test.go
@@ -0,0 +1,39 @@
package ssh

import (
"net/url"
"testing"

"github.com/stretchr/testify/require"
)

func TestSpecFromURL(t *testing.T) {
cases := map[string]*Spec{
"ssh://foo": {
Host: "foo",
},
"ssh://me@foo:10022/s/o/c/k/e/t.sock": {
User: "me", Host: "foo", Port: "10022", Socket: "/s/o/c/k/e/t.sock",
},
"ssh://me:passw0rd@foo": nil,
"ssh://foo/bar": {
Host: "foo", Socket: "/bar",
},
"ssh://foo?bar": nil,
"ssh://foo#bar": nil,
"ssh://": nil,
}
for s, expected := range cases {
u, err := url.Parse(s)
if err != nil {
t.Fatal(err)
}
got, err := SpecFromURL(u)
if expected != nil {
require.NoError(t, err)
require.EqualValues(t, expected, got, s)
} else {
require.Error(t, err, s)
}
}
}
1 change: 1 addition & 0 deletions cmd/buildctl/main.go
Expand Up @@ -7,6 +7,7 @@ import (
_ "github.com/moby/buildkit/client/connhelper/dockercontainer"
_ "github.com/moby/buildkit/client/connhelper/kubepod"
_ "github.com/moby/buildkit/client/connhelper/podmancontainer"
_ "github.com/moby/buildkit/client/connhelper/ssh"
bccommon "github.com/moby/buildkit/cmd/buildctl/common"
"github.com/moby/buildkit/solver/errdefs"
"github.com/moby/buildkit/util/apicaps"
Expand Down

0 comments on commit a7fa387

Please sign in to comment.