123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 |
- package client
- import (
- "errors"
- "fmt"
- "io"
- "net/http"
- "os"
- "runtime"
- "github.com/docker/docker/api"
- "github.com/docker/docker/cli"
- "github.com/docker/docker/cliconfig"
- "github.com/docker/docker/cliconfig/credentials"
- "github.com/docker/docker/dockerversion"
- "github.com/docker/docker/opts"
- "github.com/docker/docker/pkg/term"
- "github.com/docker/engine-api/client"
- "github.com/docker/go-connections/sockets"
- "github.com/docker/go-connections/tlsconfig"
- )
- // DockerCli represents the docker command line client.
- // Instances of the client can be returned from NewDockerCli.
- type DockerCli struct {
- // initializing closure
- init func() error
- // configFile has the client configuration file
- configFile *cliconfig.ConfigFile
- // in holds the input stream and closer (io.ReadCloser) for the client.
- in io.ReadCloser
- // out holds the output stream (io.Writer) for the client.
- out io.Writer
- // err holds the error stream (io.Writer) for the client.
- err io.Writer
- // keyFile holds the key file as a string.
- keyFile string
- // inFd holds the file descriptor of the client's STDIN (if valid).
- inFd uintptr
- // outFd holds file descriptor of the client's STDOUT (if valid).
- outFd uintptr
- // isTerminalIn indicates whether the client's STDIN is a TTY
- isTerminalIn bool
- // isTerminalOut indicates whether the client's STDOUT is a TTY
- isTerminalOut bool
- // client is the http client that performs all API operations
- client client.APIClient
- // state holds the terminal state
- state *term.State
- }
- // Initialize calls the init function that will setup the configuration for the client
- // such as the TLS, tcp and other parameters used to run the client.
- func (cli *DockerCli) Initialize() error {
- if cli.init == nil {
- return nil
- }
- return cli.init()
- }
- // CheckTtyInput checks if we are trying to attach to a container tty
- // from a non-tty client input stream, and if so, returns an error.
- func (cli *DockerCli) CheckTtyInput(attachStdin, ttyMode bool) error {
- // In order to attach to a container tty, input stream for the client must
- // be a tty itself: redirecting or piping the client standard input is
- // incompatible with `docker run -t`, `docker exec -t` or `docker attach`.
- if ttyMode && attachStdin && !cli.isTerminalIn {
- return errors.New("cannot enable tty mode on non tty input")
- }
- return nil
- }
- // PsFormat returns the format string specified in the configuration.
- // String contains columns and format specification, for example {{ID}}\t{{Name}}.
- func (cli *DockerCli) PsFormat() string {
- return cli.configFile.PsFormat
- }
- // ImagesFormat returns the format string specified in the configuration.
- // String contains columns and format specification, for example {{ID}}\t{{Name}}.
- func (cli *DockerCli) ImagesFormat() string {
- return cli.configFile.ImagesFormat
- }
- func (cli *DockerCli) setRawTerminal() error {
- if cli.isTerminalIn && os.Getenv("NORAW") == "" {
- state, err := term.SetRawTerminal(cli.inFd)
- if err != nil {
- return err
- }
- cli.state = state
- }
- return nil
- }
- func (cli *DockerCli) restoreTerminal(in io.Closer) error {
- if cli.state != nil {
- term.RestoreTerminal(cli.inFd, cli.state)
- }
- // WARNING: DO NOT REMOVE THE OS CHECK !!!
- // For some reason this Close call blocks on darwin..
- // As the client exists right after, simply discard the close
- // until we find a better solution.
- if in != nil && runtime.GOOS != "darwin" {
- return in.Close()
- }
- return nil
- }
- // NewDockerCli returns a DockerCli instance with IO output and error streams set by in, out and err.
- // The key file, protocol (i.e. unix) and address are passed in as strings, along with the tls.Config. If the tls.Config
- // is set the client scheme will be set to https.
- // The client will be given a 32-second timeout (see https://github.com/docker/docker/pull/8035).
- func NewDockerCli(in io.ReadCloser, out, err io.Writer, clientFlags *cli.ClientFlags) *DockerCli {
- cli := &DockerCli{
- in: in,
- out: out,
- err: err,
- keyFile: clientFlags.Common.TrustKey,
- }
- cli.init = func() error {
- clientFlags.PostParse()
- configFile, e := cliconfig.Load(cliconfig.ConfigDir())
- if e != nil {
- fmt.Fprintf(cli.err, "WARNING: Error loading config file:%v\n", e)
- }
- if !configFile.ContainsAuth() {
- credentials.DetectDefaultStore(configFile)
- }
- cli.configFile = configFile
- host, err := getServerHost(clientFlags.Common.Hosts, clientFlags.Common.TLSOptions)
- if err != nil {
- return err
- }
- customHeaders := cli.configFile.HTTPHeaders
- if customHeaders == nil {
- customHeaders = map[string]string{}
- }
- customHeaders["User-Agent"] = clientUserAgent()
- verStr := api.DefaultVersion.String()
- if tmpStr := os.Getenv("DOCKER_API_VERSION"); tmpStr != "" {
- verStr = tmpStr
- }
- httpClient, err := newHTTPClient(host, clientFlags.Common.TLSOptions)
- if err != nil {
- return err
- }
- client, err := client.NewClient(host, verStr, httpClient, customHeaders)
- if err != nil {
- return err
- }
- cli.client = client
- if cli.in != nil {
- cli.inFd, cli.isTerminalIn = term.GetFdInfo(cli.in)
- }
- if cli.out != nil {
- cli.outFd, cli.isTerminalOut = term.GetFdInfo(cli.out)
- }
- return nil
- }
- return cli
- }
- func getServerHost(hosts []string, tlsOptions *tlsconfig.Options) (host string, err error) {
- switch len(hosts) {
- case 0:
- host = os.Getenv("DOCKER_HOST")
- case 1:
- host = hosts[0]
- default:
- return "", errors.New("Please specify only one -H")
- }
- host, err = opts.ParseHost(tlsOptions != nil, host)
- return
- }
- func newHTTPClient(host string, tlsOptions *tlsconfig.Options) (*http.Client, error) {
- if tlsOptions == nil {
- // let the api client configure the default transport.
- return nil, nil
- }
- config, err := tlsconfig.Client(*tlsOptions)
- if err != nil {
- return nil, err
- }
- tr := &http.Transport{
- TLSClientConfig: config,
- }
- proto, addr, _, err := client.ParseHost(host)
- if err != nil {
- return nil, err
- }
- sockets.ConfigureTransport(tr, proto, addr)
- return &http.Client{
- Transport: tr,
- }, nil
- }
- func clientUserAgent() string {
- return "Docker-Client/" + dockerversion.Version + " (" + runtime.GOOS + ")"
- }
|