|
@@ -1,6 +1,7 @@
|
|
package docker
|
|
package docker
|
|
|
|
|
|
import (
|
|
import (
|
|
|
|
+ "archive/tar"
|
|
"bytes"
|
|
"bytes"
|
|
"encoding/json"
|
|
"encoding/json"
|
|
"flag"
|
|
"flag"
|
|
@@ -10,14 +11,12 @@ import (
|
|
"github.com/dotcloud/docker/utils"
|
|
"github.com/dotcloud/docker/utils"
|
|
"io"
|
|
"io"
|
|
"io/ioutil"
|
|
"io/ioutil"
|
|
- "mime/multipart"
|
|
|
|
"net"
|
|
"net"
|
|
"net/http"
|
|
"net/http"
|
|
"net/http/httputil"
|
|
"net/http/httputil"
|
|
"net/url"
|
|
"net/url"
|
|
"os"
|
|
"os"
|
|
"os/signal"
|
|
"os/signal"
|
|
- "path"
|
|
|
|
"path/filepath"
|
|
"path/filepath"
|
|
"reflect"
|
|
"reflect"
|
|
"regexp"
|
|
"regexp"
|
|
@@ -29,7 +28,7 @@ import (
|
|
"unicode"
|
|
"unicode"
|
|
)
|
|
)
|
|
|
|
|
|
-const VERSION = "0.4.2"
|
|
|
|
|
|
+const VERSION = "0.4.4"
|
|
|
|
|
|
var (
|
|
var (
|
|
GITCOMMIT string
|
|
GITCOMMIT string
|
|
@@ -40,8 +39,8 @@ func (cli *DockerCli) getMethod(name string) (reflect.Method, bool) {
|
|
return reflect.TypeOf(cli).MethodByName(methodName)
|
|
return reflect.TypeOf(cli).MethodByName(methodName)
|
|
}
|
|
}
|
|
|
|
|
|
-func ParseCommands(addr string, port int, args ...string) error {
|
|
|
|
- cli := NewDockerCli(addr, port)
|
|
|
|
|
|
+func ParseCommands(proto, addr string, args ...string) error {
|
|
|
|
+ cli := NewDockerCli(proto, addr)
|
|
|
|
|
|
if len(args) > 0 {
|
|
if len(args) > 0 {
|
|
method, exists := cli.getMethod(args[0])
|
|
method, exists := cli.getMethod(args[0])
|
|
@@ -74,7 +73,7 @@ func (cli *DockerCli) CmdHelp(args ...string) error {
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- help := fmt.Sprintf("Usage: docker [OPTIONS] COMMAND [arg...]\n -H=\"%s:%d\": Host:port to bind/connect to\n\nA self-sufficient runtime for linux containers.\n\nCommands:\n", cli.host, cli.port)
|
|
|
|
|
|
+ help := fmt.Sprintf("Usage: docker [OPTIONS] COMMAND [arg...]\n -H=[tcp://%s:%d]: tcp://host:port to bind/connect to or unix://path/to/socker to use\n\nA self-sufficient runtime for linux containers.\n\nCommands:\n", DEFAULTHTTPHOST, DEFAULTHTTPPORT)
|
|
for _, command := range [][2]string{
|
|
for _, command := range [][2]string{
|
|
{"attach", "Attach to a running container"},
|
|
{"attach", "Attach to a running container"},
|
|
{"build", "Build a container from a Dockerfile"},
|
|
{"build", "Build a container from a Dockerfile"},
|
|
@@ -131,8 +130,33 @@ func (cli *DockerCli) CmdInsert(args ...string) error {
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+// mkBuildContext returns an archive of an empty context with the contents
|
|
|
|
+// of `dockerfile` at the path ./Dockerfile
|
|
|
|
+func mkBuildContext(dockerfile string, files [][2]string) (Archive, error) {
|
|
|
|
+ buf := new(bytes.Buffer)
|
|
|
|
+ tw := tar.NewWriter(buf)
|
|
|
|
+ files = append(files, [2]string{"Dockerfile", dockerfile})
|
|
|
|
+ for _, file := range files {
|
|
|
|
+ name, content := file[0], file[1]
|
|
|
|
+ hdr := &tar.Header{
|
|
|
|
+ Name: name,
|
|
|
|
+ Size: int64(len(content)),
|
|
|
|
+ }
|
|
|
|
+ if err := tw.WriteHeader(hdr); err != nil {
|
|
|
|
+ return nil, err
|
|
|
|
+ }
|
|
|
|
+ if _, err := tw.Write([]byte(content)); err != nil {
|
|
|
|
+ return nil, err
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ if err := tw.Close(); err != nil {
|
|
|
|
+ return nil, err
|
|
|
|
+ }
|
|
|
|
+ return buf, nil
|
|
|
|
+}
|
|
|
|
+
|
|
func (cli *DockerCli) CmdBuild(args ...string) error {
|
|
func (cli *DockerCli) CmdBuild(args ...string) error {
|
|
- cmd := Subcmd("build", "[OPTIONS] PATH | -", "Build a new container image from the source code at PATH")
|
|
|
|
|
|
+ cmd := Subcmd("build", "[OPTIONS] PATH | URL | -", "Build a new container image from the source code at PATH")
|
|
tag := cmd.String("t", "", "Tag to be applied to the resulting image in case of success")
|
|
tag := cmd.String("t", "", "Tag to be applied to the resulting image in case of success")
|
|
if err := cmd.Parse(args); err != nil {
|
|
if err := cmd.Parse(args); err != nil {
|
|
return nil
|
|
return nil
|
|
@@ -143,76 +167,55 @@ func (cli *DockerCli) CmdBuild(args ...string) error {
|
|
}
|
|
}
|
|
|
|
|
|
var (
|
|
var (
|
|
- multipartBody io.Reader
|
|
|
|
- file io.ReadCloser
|
|
|
|
- contextPath string
|
|
|
|
|
|
+ context Archive
|
|
|
|
+ isRemote bool
|
|
|
|
+ err error
|
|
)
|
|
)
|
|
|
|
|
|
- // Init the needed component for the Multipart
|
|
|
|
- buff := bytes.NewBuffer([]byte{})
|
|
|
|
- multipartBody = buff
|
|
|
|
- w := multipart.NewWriter(buff)
|
|
|
|
- boundary := strings.NewReader("\r\n--" + w.Boundary() + "--\r\n")
|
|
|
|
-
|
|
|
|
- compression := Bzip2
|
|
|
|
-
|
|
|
|
if cmd.Arg(0) == "-" {
|
|
if cmd.Arg(0) == "-" {
|
|
- file = os.Stdin
|
|
|
|
- } else {
|
|
|
|
- // Send Dockerfile from arg/Dockerfile (deprecate later)
|
|
|
|
- f, err := os.Open(path.Join(cmd.Arg(0), "Dockerfile"))
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
- }
|
|
|
|
- file = f
|
|
|
|
- // Send context from arg
|
|
|
|
- // Create a FormFile multipart for the context if needed
|
|
|
|
- // FIXME: Use NewTempArchive in order to have the size and avoid too much memory usage?
|
|
|
|
- context, err := Tar(cmd.Arg(0), compression)
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
- }
|
|
|
|
- // NOTE: Do this in case '.' or '..' is input
|
|
|
|
- absPath, err := filepath.Abs(cmd.Arg(0))
|
|
|
|
|
|
+ // As a special case, 'docker build -' will build from an empty context with the
|
|
|
|
+ // contents of stdin as a Dockerfile
|
|
|
|
+ dockerfile, err := ioutil.ReadAll(os.Stdin)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- wField, err := w.CreateFormFile("Context", filepath.Base(absPath)+"."+compression.Extension())
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
- }
|
|
|
|
- // FIXME: Find a way to have a progressbar for the upload too
|
|
|
|
- sf := utils.NewStreamFormatter(false)
|
|
|
|
- io.Copy(wField, utils.ProgressReader(ioutil.NopCloser(context), -1, os.Stdout, sf.FormatProgress("Caching Context", "%v/%v (%v)"), sf))
|
|
|
|
- multipartBody = io.MultiReader(multipartBody, boundary)
|
|
|
|
|
|
+ context, err = mkBuildContext(string(dockerfile), nil)
|
|
|
|
+ } else if utils.IsURL(cmd.Arg(0)) || utils.IsGIT(cmd.Arg(0)) {
|
|
|
|
+ isRemote = true
|
|
|
|
+ } else {
|
|
|
|
+ context, err = Tar(cmd.Arg(0), Uncompressed)
|
|
}
|
|
}
|
|
- // Create a FormFile multipart for the Dockerfile
|
|
|
|
- wField, err := w.CreateFormFile("Dockerfile", "Dockerfile")
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
|
|
+ var body io.Reader
|
|
|
|
+ // Setup an upload progress bar
|
|
|
|
+ // FIXME: ProgressReader shouldn't be this annoyning to use
|
|
|
|
+ if context != nil {
|
|
|
|
+ sf := utils.NewStreamFormatter(false)
|
|
|
|
+ body = utils.ProgressReader(ioutil.NopCloser(context), 0, os.Stderr, sf.FormatProgress("Uploading context", "%v bytes%0.0s%0.0s"), sf)
|
|
}
|
|
}
|
|
- io.Copy(wField, file)
|
|
|
|
- multipartBody = io.MultiReader(multipartBody, boundary)
|
|
|
|
-
|
|
|
|
|
|
+ // Upload the build context
|
|
v := &url.Values{}
|
|
v := &url.Values{}
|
|
v.Set("t", *tag)
|
|
v.Set("t", *tag)
|
|
- // Send the multipart request with correct content-type
|
|
|
|
- req, err := http.NewRequest("POST", fmt.Sprintf("http://%s:%d%s?%s", cli.host, cli.port, "/build", v.Encode()), multipartBody)
|
|
|
|
|
|
+ if isRemote {
|
|
|
|
+ v.Set("remote", cmd.Arg(0))
|
|
|
|
+ }
|
|
|
|
+ req, err := http.NewRequest("POST", fmt.Sprintf("/v%g/build?%s", APIVERSION, v.Encode()), body)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- req.Header.Set("Content-Type", w.FormDataContentType())
|
|
|
|
- if contextPath != "" {
|
|
|
|
- req.Header.Set("X-Docker-Context-Compression", compression.Flag())
|
|
|
|
- fmt.Println("Uploading Context...")
|
|
|
|
|
|
+ if context != nil {
|
|
|
|
+ req.Header.Set("Content-Type", "application/tar")
|
|
}
|
|
}
|
|
-
|
|
|
|
- resp, err := http.DefaultClient.Do(req)
|
|
|
|
|
|
+ dial, err := net.Dial(cli.proto, cli.addr)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
|
|
+ clientconn := httputil.NewClientConn(dial, nil)
|
|
|
|
+ resp, err := clientconn.Do(req)
|
|
|
|
+ defer clientconn.Close()
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
defer resp.Body.Close()
|
|
defer resp.Body.Close()
|
|
-
|
|
|
|
// Check for errors
|
|
// Check for errors
|
|
if resp.StatusCode < 200 || resp.StatusCode >= 400 {
|
|
if resp.StatusCode < 200 || resp.StatusCode >= 400 {
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
@@ -311,6 +314,7 @@ func (cli *DockerCli) CmdLogin(args ...string) error {
|
|
email = cli.authConfig.Email
|
|
email = cli.authConfig.Email
|
|
}
|
|
}
|
|
} else {
|
|
} else {
|
|
|
|
+ password = cli.authConfig.Password
|
|
email = cli.authConfig.Email
|
|
email = cli.authConfig.Email
|
|
}
|
|
}
|
|
term.RestoreTerminal(oldState)
|
|
term.RestoreTerminal(oldState)
|
|
@@ -319,7 +323,14 @@ func (cli *DockerCli) CmdLogin(args ...string) error {
|
|
cli.authConfig.Password = password
|
|
cli.authConfig.Password = password
|
|
cli.authConfig.Email = email
|
|
cli.authConfig.Email = email
|
|
|
|
|
|
- body, _, err := cli.call("POST", "/auth", cli.authConfig)
|
|
|
|
|
|
+ body, statusCode, err := cli.call("POST", "/auth", cli.authConfig)
|
|
|
|
+ if statusCode == 401 {
|
|
|
|
+ cli.authConfig.Username = ""
|
|
|
|
+ cli.authConfig.Password = ""
|
|
|
|
+ cli.authConfig.Email = ""
|
|
|
|
+ auth.SaveConfig(cli.authConfig)
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
@@ -332,7 +343,7 @@ func (cli *DockerCli) CmdLogin(args ...string) error {
|
|
}
|
|
}
|
|
auth.SaveConfig(cli.authConfig)
|
|
auth.SaveConfig(cli.authConfig)
|
|
if out2.Status != "" {
|
|
if out2.Status != "" {
|
|
- fmt.Print(out2.Status)
|
|
|
|
|
|
+ fmt.Println(out2.Status)
|
|
}
|
|
}
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
@@ -1044,10 +1055,10 @@ func (cli *DockerCli) CmdLogs(args ...string) error {
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
|
|
|
|
- if err := cli.stream("POST", "/containers/"+cmd.Arg(0)+"/attach?logs=1&stdout=1", nil, os.Stdout); err != nil {
|
|
|
|
|
|
+ if err := cli.hijack("POST", "/containers/"+cmd.Arg(0)+"/attach?logs=1&stdout=1", false, nil, os.Stdout); err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- if err := cli.stream("POST", "/containers/"+cmd.Arg(0)+"/attach?logs=1&stderr=1", nil, os.Stderr); err != nil {
|
|
|
|
|
|
+ if err := cli.hijack("POST", "/containers/"+cmd.Arg(0)+"/attach?logs=1&stderr=1", false, nil, os.Stderr); err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
return nil
|
|
@@ -1078,37 +1089,18 @@ func (cli *DockerCli) CmdAttach(args ...string) error {
|
|
return fmt.Errorf("Impossible to attach to a stopped container, start it first")
|
|
return fmt.Errorf("Impossible to attach to a stopped container, start it first")
|
|
}
|
|
}
|
|
|
|
|
|
- splitStderr := container.Config.Tty
|
|
|
|
-
|
|
|
|
- connections := 1
|
|
|
|
- if splitStderr {
|
|
|
|
- connections += 1
|
|
|
|
- }
|
|
|
|
- chErrors := make(chan error, connections)
|
|
|
|
if container.Config.Tty {
|
|
if container.Config.Tty {
|
|
cli.monitorTtySize(cmd.Arg(0))
|
|
cli.monitorTtySize(cmd.Arg(0))
|
|
}
|
|
}
|
|
- if splitStderr {
|
|
|
|
- go func() {
|
|
|
|
- chErrors <- cli.hijack("POST", "/containers/"+cmd.Arg(0)+"/attach?stream=1&stderr=1", false, nil, os.Stderr)
|
|
|
|
- }()
|
|
|
|
- }
|
|
|
|
|
|
+
|
|
v := url.Values{}
|
|
v := url.Values{}
|
|
v.Set("stream", "1")
|
|
v.Set("stream", "1")
|
|
v.Set("stdin", "1")
|
|
v.Set("stdin", "1")
|
|
v.Set("stdout", "1")
|
|
v.Set("stdout", "1")
|
|
- if !splitStderr {
|
|
|
|
- v.Set("stderr", "1")
|
|
|
|
- }
|
|
|
|
- go func() {
|
|
|
|
- chErrors <- cli.hijack("POST", "/containers/"+cmd.Arg(0)+"/attach?"+v.Encode(), container.Config.Tty, os.Stdin, os.Stdout)
|
|
|
|
- }()
|
|
|
|
- for connections > 0 {
|
|
|
|
- err := <-chErrors
|
|
|
|
- if err != nil {
|
|
|
|
- return err
|
|
|
|
- }
|
|
|
|
- connections -= 1
|
|
|
|
|
|
+ v.Set("stderr", "1")
|
|
|
|
+
|
|
|
|
+ if err := cli.hijack("POST", "/containers/"+cmd.Arg(0)+"/attach?"+v.Encode(), container.Config.Tty, os.Stdin, os.Stdout); err != nil {
|
|
|
|
+ return err
|
|
}
|
|
}
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|
|
@@ -1334,7 +1326,7 @@ func (cli *DockerCli) call(method, path string, data interface{}) ([]byte, int,
|
|
params = bytes.NewBuffer(buf)
|
|
params = bytes.NewBuffer(buf)
|
|
}
|
|
}
|
|
|
|
|
|
- req, err := http.NewRequest(method, fmt.Sprintf("http://%s:%d/v%g%s", cli.host, cli.port, APIVERSION, path), params)
|
|
|
|
|
|
+ req, err := http.NewRequest(method, fmt.Sprintf("/v%g%s", APIVERSION, path), params)
|
|
if err != nil {
|
|
if err != nil {
|
|
return nil, -1, err
|
|
return nil, -1, err
|
|
}
|
|
}
|
|
@@ -1344,7 +1336,13 @@ func (cli *DockerCli) call(method, path string, data interface{}) ([]byte, int,
|
|
} else if method == "POST" {
|
|
} else if method == "POST" {
|
|
req.Header.Set("Content-Type", "plain/text")
|
|
req.Header.Set("Content-Type", "plain/text")
|
|
}
|
|
}
|
|
- resp, err := http.DefaultClient.Do(req)
|
|
|
|
|
|
+ dial, err := net.Dial(cli.proto, cli.addr)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return nil, -1, err
|
|
|
|
+ }
|
|
|
|
+ clientconn := httputil.NewClientConn(dial, nil)
|
|
|
|
+ resp, err := clientconn.Do(req)
|
|
|
|
+ defer clientconn.Close()
|
|
if err != nil {
|
|
if err != nil {
|
|
if strings.Contains(err.Error(), "connection refused") {
|
|
if strings.Contains(err.Error(), "connection refused") {
|
|
return nil, -1, fmt.Errorf("Can't connect to docker daemon. Is 'docker -d' running on this host?")
|
|
return nil, -1, fmt.Errorf("Can't connect to docker daemon. Is 'docker -d' running on this host?")
|
|
@@ -1369,7 +1367,7 @@ func (cli *DockerCli) stream(method, path string, in io.Reader, out io.Writer) e
|
|
if (method == "POST" || method == "PUT") && in == nil {
|
|
if (method == "POST" || method == "PUT") && in == nil {
|
|
in = bytes.NewReader([]byte{})
|
|
in = bytes.NewReader([]byte{})
|
|
}
|
|
}
|
|
- req, err := http.NewRequest(method, fmt.Sprintf("http://%s:%d/v%g%s", cli.host, cli.port, APIVERSION, path), in)
|
|
|
|
|
|
+ req, err := http.NewRequest(method, fmt.Sprintf("/v%g%s", APIVERSION, path), in)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
@@ -1377,7 +1375,13 @@ func (cli *DockerCli) stream(method, path string, in io.Reader, out io.Writer) e
|
|
if method == "POST" {
|
|
if method == "POST" {
|
|
req.Header.Set("Content-Type", "plain/text")
|
|
req.Header.Set("Content-Type", "plain/text")
|
|
}
|
|
}
|
|
- resp, err := http.DefaultClient.Do(req)
|
|
|
|
|
|
+ dial, err := net.Dial(cli.proto, cli.addr)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
|
|
+ clientconn := httputil.NewClientConn(dial, nil)
|
|
|
|
+ resp, err := clientconn.Do(req)
|
|
|
|
+ defer clientconn.Close()
|
|
if err != nil {
|
|
if err != nil {
|
|
if strings.Contains(err.Error(), "connection refused") {
|
|
if strings.Contains(err.Error(), "connection refused") {
|
|
return fmt.Errorf("Can't connect to docker daemon. Is 'docker -d' running on this host?")
|
|
return fmt.Errorf("Can't connect to docker daemon. Is 'docker -d' running on this host?")
|
|
@@ -1385,6 +1389,7 @@ func (cli *DockerCli) stream(method, path string, in io.Reader, out io.Writer) e
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
defer resp.Body.Close()
|
|
defer resp.Body.Close()
|
|
|
|
+
|
|
if resp.StatusCode < 200 || resp.StatusCode >= 400 {
|
|
if resp.StatusCode < 200 || resp.StatusCode >= 400 {
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
if err != nil {
|
|
@@ -1422,19 +1427,24 @@ func (cli *DockerCli) stream(method, path string, in io.Reader, out io.Writer) e
|
|
}
|
|
}
|
|
|
|
|
|
func (cli *DockerCli) hijack(method, path string, setRawTerminal bool, in *os.File, out io.Writer) error {
|
|
func (cli *DockerCli) hijack(method, path string, setRawTerminal bool, in *os.File, out io.Writer) error {
|
|
|
|
+
|
|
req, err := http.NewRequest(method, fmt.Sprintf("/v%g%s", APIVERSION, path), nil)
|
|
req, err := http.NewRequest(method, fmt.Sprintf("/v%g%s", APIVERSION, path), nil)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
|
|
+ req.Header.Set("User-Agent", "Docker-Client/"+VERSION)
|
|
req.Header.Set("Content-Type", "plain/text")
|
|
req.Header.Set("Content-Type", "plain/text")
|
|
- dial, err := net.Dial("tcp", fmt.Sprintf("%s:%d", cli.host, cli.port))
|
|
|
|
|
|
+
|
|
|
|
+ dial, err := net.Dial(cli.proto, cli.addr)
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
clientconn := httputil.NewClientConn(dial, nil)
|
|
clientconn := httputil.NewClientConn(dial, nil)
|
|
- clientconn.Do(req)
|
|
|
|
defer clientconn.Close()
|
|
defer clientconn.Close()
|
|
|
|
|
|
|
|
+ // Server hijacks the connection, error 'connection closed' expected
|
|
|
|
+ clientconn.Do(req)
|
|
|
|
+
|
|
rwc, br := clientconn.Hijack()
|
|
rwc, br := clientconn.Hijack()
|
|
defer rwc.Close()
|
|
defer rwc.Close()
|
|
|
|
|
|
@@ -1510,13 +1520,13 @@ func Subcmd(name, signature, description string) *flag.FlagSet {
|
|
return flags
|
|
return flags
|
|
}
|
|
}
|
|
|
|
|
|
-func NewDockerCli(addr string, port int) *DockerCli {
|
|
|
|
|
|
+func NewDockerCli(proto, addr string) *DockerCli {
|
|
authConfig, _ := auth.LoadConfig(os.Getenv("HOME"))
|
|
authConfig, _ := auth.LoadConfig(os.Getenv("HOME"))
|
|
- return &DockerCli{addr, port, authConfig}
|
|
|
|
|
|
+ return &DockerCli{proto, addr, authConfig}
|
|
}
|
|
}
|
|
|
|
|
|
type DockerCli struct {
|
|
type DockerCli struct {
|
|
- host string
|
|
|
|
- port int
|
|
|
|
|
|
+ proto string
|
|
|
|
+ addr string
|
|
authConfig *auth.AuthConfig
|
|
authConfig *auth.AuthConfig
|
|
}
|
|
}
|