cli.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. package client
  2. import (
  3. "crypto/tls"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "net"
  8. "net/http"
  9. "os"
  10. "reflect"
  11. "strings"
  12. "text/template"
  13. "time"
  14. flag "github.com/docker/docker/pkg/mflag"
  15. "github.com/docker/docker/pkg/term"
  16. "github.com/docker/docker/registry"
  17. "github.com/docker/libtrust"
  18. )
  19. type DockerCli struct {
  20. proto string
  21. addr string
  22. configFile *registry.ConfigFile
  23. in io.ReadCloser
  24. out io.Writer
  25. err io.Writer
  26. key libtrust.PrivateKey
  27. tlsConfig *tls.Config
  28. scheme string
  29. // inFd holds file descriptor of the client's STDIN, if it's a valid file
  30. inFd uintptr
  31. // outFd holds file descriptor of the client's STDOUT, if it's a valid file
  32. outFd uintptr
  33. // isTerminalIn describes if client's STDIN is a TTY
  34. isTerminalIn bool
  35. // isTerminalOut describes if client's STDOUT is a TTY
  36. isTerminalOut bool
  37. transport *http.Transport
  38. }
  39. var funcMap = template.FuncMap{
  40. "json": func(v interface{}) string {
  41. a, _ := json.Marshal(v)
  42. return string(a)
  43. },
  44. }
  45. func (cli *DockerCli) getMethod(args ...string) (func(...string) error, bool) {
  46. camelArgs := make([]string, len(args))
  47. for i, s := range args {
  48. if len(s) == 0 {
  49. return nil, false
  50. }
  51. camelArgs[i] = strings.ToUpper(s[:1]) + strings.ToLower(s[1:])
  52. }
  53. methodName := "Cmd" + strings.Join(camelArgs, "")
  54. method := reflect.ValueOf(cli).MethodByName(methodName)
  55. if !method.IsValid() {
  56. return nil, false
  57. }
  58. return method.Interface().(func(...string) error), true
  59. }
  60. // Cmd executes the specified command
  61. func (cli *DockerCli) Cmd(args ...string) error {
  62. if len(args) > 1 {
  63. method, exists := cli.getMethod(args[:2]...)
  64. if exists {
  65. return method(args[2:]...)
  66. }
  67. }
  68. if len(args) > 0 {
  69. method, exists := cli.getMethod(args[0])
  70. if !exists {
  71. fmt.Println("Error: Command not found:", args[0])
  72. return cli.CmdHelp()
  73. }
  74. return method(args[1:]...)
  75. }
  76. return cli.CmdHelp()
  77. }
  78. func (cli *DockerCli) Subcmd(name, signature, description string) *flag.FlagSet {
  79. flags := flag.NewFlagSet(name, flag.ContinueOnError)
  80. flags.Usage = func() {
  81. options := ""
  82. if flags.FlagCountUndeprecated() > 0 {
  83. options = "[OPTIONS] "
  84. }
  85. fmt.Fprintf(cli.err, "\nUsage: docker %s %s%s\n\n%s\n\n", name, options, signature, description)
  86. flags.PrintDefaults()
  87. os.Exit(2)
  88. }
  89. return flags
  90. }
  91. func (cli *DockerCli) LoadConfigFile() (err error) {
  92. cli.configFile, err = registry.LoadConfig(os.Getenv("HOME"))
  93. if err != nil {
  94. fmt.Fprintf(cli.err, "WARNING: %s\n", err)
  95. }
  96. return err
  97. }
  98. func NewDockerCli(in io.ReadCloser, out, err io.Writer, key libtrust.PrivateKey, proto, addr string, tlsConfig *tls.Config) *DockerCli {
  99. var (
  100. inFd uintptr
  101. outFd uintptr
  102. isTerminalIn = false
  103. isTerminalOut = false
  104. scheme = "http"
  105. )
  106. if tlsConfig != nil {
  107. scheme = "https"
  108. }
  109. if in != nil {
  110. if file, ok := in.(*os.File); ok {
  111. inFd = file.Fd()
  112. isTerminalIn = term.IsTerminal(inFd)
  113. }
  114. }
  115. if out != nil {
  116. if file, ok := out.(*os.File); ok {
  117. outFd = file.Fd()
  118. isTerminalOut = term.IsTerminal(outFd)
  119. }
  120. }
  121. if err == nil {
  122. err = out
  123. }
  124. // The transport is created here for reuse during the client session
  125. tr := &http.Transport{
  126. TLSClientConfig: tlsConfig,
  127. }
  128. // Why 32? See issue 8035
  129. if proto == "unix" {
  130. // no need in compressing for local communications
  131. tr.DisableCompression = true
  132. tr.Dial = func(dial_network, dial_addr string) (net.Conn, error) {
  133. return net.DialTimeout(proto, addr, 32*time.Second)
  134. }
  135. } else {
  136. tr.Dial = (&net.Dialer{Timeout: 32 * time.Second}).Dial
  137. }
  138. return &DockerCli{
  139. proto: proto,
  140. addr: addr,
  141. in: in,
  142. out: out,
  143. err: err,
  144. key: key,
  145. inFd: inFd,
  146. outFd: outFd,
  147. isTerminalIn: isTerminalIn,
  148. isTerminalOut: isTerminalOut,
  149. tlsConfig: tlsConfig,
  150. scheme: scheme,
  151. transport: tr,
  152. }
  153. }