login.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. package client
  2. import (
  3. "bufio"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "os"
  8. "strings"
  9. "github.com/docker/docker/api/types"
  10. "github.com/docker/docker/cliconfig"
  11. flag "github.com/docker/docker/pkg/mflag"
  12. "github.com/docker/docker/pkg/term"
  13. "github.com/docker/docker/registry"
  14. )
  15. // CmdLogin logs in or registers a user to a Docker registry service.
  16. //
  17. // If no server is specified, the user will be logged into or registered to the registry's index server.
  18. //
  19. // Usage: docker login SERVER
  20. func (cli *DockerCli) CmdLogin(args ...string) error {
  21. cmd := cli.Subcmd("login", "[SERVER]", "Register or log in to a Docker registry server, if no server is\nspecified \""+registry.IndexServerAddress()+"\" is the default.", true)
  22. cmd.Require(flag.Max, 1)
  23. var username, password, email string
  24. cmd.StringVar(&username, []string{"u", "-username"}, "", "Username")
  25. cmd.StringVar(&password, []string{"p", "-password"}, "", "Password")
  26. cmd.StringVar(&email, []string{"e", "-email"}, "", "Email")
  27. cmd.ParseFlags(args, true)
  28. serverAddress := registry.IndexServerAddress()
  29. if len(cmd.Args()) > 0 {
  30. serverAddress = cmd.Arg(0)
  31. }
  32. promptDefault := func(prompt string, configDefault string) {
  33. if configDefault == "" {
  34. fmt.Fprintf(cli.out, "%s: ", prompt)
  35. } else {
  36. fmt.Fprintf(cli.out, "%s (%s): ", prompt, configDefault)
  37. }
  38. }
  39. readInput := func(in io.Reader, out io.Writer) string {
  40. reader := bufio.NewReader(in)
  41. line, _, err := reader.ReadLine()
  42. if err != nil {
  43. fmt.Fprintln(out, err.Error())
  44. os.Exit(1)
  45. }
  46. return string(line)
  47. }
  48. authconfig, ok := cli.configFile.AuthConfigs[serverAddress]
  49. if !ok {
  50. authconfig = cliconfig.AuthConfig{}
  51. }
  52. if username == "" {
  53. promptDefault("Username", authconfig.Username)
  54. username = readInput(cli.in, cli.out)
  55. username = strings.Trim(username, " ")
  56. if username == "" {
  57. username = authconfig.Username
  58. }
  59. }
  60. // Assume that a different username means they may not want to use
  61. // the password or email from the config file, so prompt them
  62. if username != authconfig.Username {
  63. if password == "" {
  64. oldState, err := term.SaveState(cli.inFd)
  65. if err != nil {
  66. return err
  67. }
  68. fmt.Fprintf(cli.out, "Password: ")
  69. term.DisableEcho(cli.inFd, oldState)
  70. password = readInput(cli.in, cli.out)
  71. fmt.Fprint(cli.out, "\n")
  72. term.RestoreTerminal(cli.inFd, oldState)
  73. if password == "" {
  74. return fmt.Errorf("Error : Password Required")
  75. }
  76. }
  77. if email == "" {
  78. promptDefault("Email", authconfig.Email)
  79. email = readInput(cli.in, cli.out)
  80. if email == "" {
  81. email = authconfig.Email
  82. }
  83. }
  84. } else {
  85. // However, if they don't override the username use the
  86. // password or email from the cmd line if specified. IOW, allow
  87. // then to change/override them. And if not specified, just
  88. // use what's in the config file
  89. if password == "" {
  90. password = authconfig.Password
  91. }
  92. if email == "" {
  93. email = authconfig.Email
  94. }
  95. }
  96. authconfig.Username = username
  97. authconfig.Password = password
  98. authconfig.Email = email
  99. authconfig.ServerAddress = serverAddress
  100. cli.configFile.AuthConfigs[serverAddress] = authconfig
  101. stream, statusCode, err := cli.call("POST", "/auth", cli.configFile.AuthConfigs[serverAddress], nil)
  102. if statusCode == 401 {
  103. delete(cli.configFile.AuthConfigs, serverAddress)
  104. if err2 := cli.configFile.Save(); err2 != nil {
  105. fmt.Fprintf(cli.out, "WARNING: could not save config file: %v\n", err2)
  106. }
  107. return err
  108. }
  109. if err != nil {
  110. return err
  111. }
  112. var response types.AuthResponse
  113. if err := json.NewDecoder(stream).Decode(&response); err != nil {
  114. // Upon error, remove entry
  115. delete(cli.configFile.AuthConfigs, serverAddress)
  116. return err
  117. }
  118. if err := cli.configFile.Save(); err != nil {
  119. return fmt.Errorf("Error saving config file: %v", err)
  120. }
  121. fmt.Fprintf(cli.out, "WARNING: login credentials saved in %s\n", cli.configFile.Filename())
  122. if response.Status != "" {
  123. fmt.Fprintf(cli.out, "%s\n", response.Status)
  124. }
  125. return nil
  126. }