mirror of https://github.com/docker/compose.git
Merge pull request #162 from docker/login_command
Move login to root command
This commit is contained in:
commit
ec5e61b645
|
@ -29,8 +29,6 @@ package context
|
|||
|
||||
import (
|
||||
"github.com/spf13/cobra"
|
||||
|
||||
"github.com/docker/api/cli/cmd/context/login"
|
||||
)
|
||||
|
||||
// Command manages contexts
|
||||
|
@ -46,7 +44,6 @@ func Command() *cobra.Command {
|
|||
removeCommand(),
|
||||
showCommand(),
|
||||
useCommand(),
|
||||
login.Command(),
|
||||
inspectCommand(),
|
||||
)
|
||||
|
||||
|
|
|
@ -39,8 +39,7 @@ func inspectCommand() *cobra.Command {
|
|||
Short: "Display detailed information on one or more contexts",
|
||||
Args: cobra.ExactArgs(1),
|
||||
RunE: func(cmd *cobra.Command, args []string) error {
|
||||
dockerclassic.Exec(cmd.Context())
|
||||
return nil
|
||||
return dockerclassic.ExecCmd(cmd)
|
||||
},
|
||||
}
|
||||
return cmd
|
||||
|
|
|
@ -1,36 +0,0 @@
|
|||
package login
|
||||
|
||||
import (
|
||||
"github.com/pkg/errors"
|
||||
"github.com/spf13/cobra"
|
||||
|
||||
"github.com/docker/api/client"
|
||||
)
|
||||
|
||||
// Command returns the compose command with its child commands
|
||||
func Command() *cobra.Command {
|
||||
command := &cobra.Command{
|
||||
Short: "Cloud login for docker contexts",
|
||||
Use: "login",
|
||||
}
|
||||
command.AddCommand(
|
||||
azureLoginCommand(),
|
||||
)
|
||||
return command
|
||||
}
|
||||
|
||||
func azureLoginCommand() *cobra.Command {
|
||||
azureLoginCmd := &cobra.Command{
|
||||
Use: "azure",
|
||||
RunE: func(cmd *cobra.Command, args []string) error {
|
||||
ctx := cmd.Context()
|
||||
cs, err := client.GetCloudService(ctx, "aci")
|
||||
if err != nil {
|
||||
return errors.Wrap(err, "cannot connect to backend")
|
||||
}
|
||||
return cs.Login(ctx, nil)
|
||||
},
|
||||
}
|
||||
|
||||
return azureLoginCmd
|
||||
}
|
|
@ -0,0 +1,52 @@
|
|||
package login
|
||||
|
||||
import (
|
||||
"strings"
|
||||
|
||||
"github.com/pkg/errors"
|
||||
"github.com/spf13/cobra"
|
||||
|
||||
"github.com/docker/api/cli/dockerclassic"
|
||||
"github.com/docker/api/client"
|
||||
)
|
||||
|
||||
// Command returns the login command
|
||||
func Command() *cobra.Command {
|
||||
cmd := &cobra.Command{
|
||||
Use: "login [OPTIONS] [SERVER] | login azure",
|
||||
Short: "Log in to a Docker registry",
|
||||
Long: "Log in to a Docker registry or cloud backend.\nIf no registry server is specified, the default is defined by the daemon.",
|
||||
Args: cobra.MaximumNArgs(1),
|
||||
RunE: runLogin,
|
||||
}
|
||||
// define flags for backward compatibility with docker-classic
|
||||
flags := cmd.Flags()
|
||||
flags.StringP("username", "u", "", "Username")
|
||||
flags.StringP("password", "p", "", "Password")
|
||||
flags.BoolP("password-stdin", "", false, "Take the password from stdin")
|
||||
|
||||
return cmd
|
||||
}
|
||||
|
||||
func runLogin(cmd *cobra.Command, args []string) error {
|
||||
if len(args) == 1 && !strings.Contains(args[0], ".") {
|
||||
backend := args[0]
|
||||
switch backend {
|
||||
case "azure":
|
||||
return cloudLogin(cmd, "aci")
|
||||
default:
|
||||
return errors.New("Unknown backend type for cloud login : " + backend)
|
||||
}
|
||||
}
|
||||
return dockerclassic.ExecCmd(cmd)
|
||||
}
|
||||
|
||||
func cloudLogin(cmd *cobra.Command, backendType string) error {
|
||||
ctx := cmd.Context()
|
||||
cs, err := client.GetCloudService(ctx, backendType)
|
||||
if err != nil {
|
||||
return errors.Wrap(err, "cannot connect to backend")
|
||||
}
|
||||
return cs.Login(ctx, nil)
|
||||
|
||||
}
|
|
@ -6,11 +6,13 @@ import (
|
|||
"os"
|
||||
"os/exec"
|
||||
|
||||
"github.com/spf13/cobra"
|
||||
|
||||
apicontext "github.com/docker/api/context"
|
||||
"github.com/docker/api/context/store"
|
||||
)
|
||||
|
||||
// Exec will delegate the cli command to docker-classic
|
||||
// Exec delegates to docker-classic
|
||||
func Exec(ctx context.Context) {
|
||||
currentContext := apicontext.CurrentContext(ctx)
|
||||
s := store.ContextStore(ctx)
|
||||
|
@ -34,3 +36,9 @@ func Exec(ctx context.Context) {
|
|||
os.Exit(0)
|
||||
}
|
||||
}
|
||||
|
||||
// ExecCmd delegates the cli command to docker-classic. The error is never returned (process will exit with docker classic exit code), the return type is to make it easier to use with cobra commands
|
||||
func ExecCmd(command *cobra.Command) error {
|
||||
Exec(command.Context())
|
||||
return nil
|
||||
}
|
||||
|
|
10
cli/main.go
10
cli/main.go
|
@ -37,6 +37,8 @@ import (
|
|||
"syscall"
|
||||
"time"
|
||||
|
||||
"github.com/docker/api/cli/cmd/login"
|
||||
|
||||
"github.com/docker/api/cli/dockerclassic"
|
||||
|
||||
"github.com/pkg/errors"
|
||||
|
@ -60,6 +62,11 @@ import (
|
|||
|
||||
var (
|
||||
runningOwnCommand bool
|
||||
ownCommands = map[string]struct{}{
|
||||
"context": {},
|
||||
"login": {},
|
||||
"serve": {},
|
||||
}
|
||||
)
|
||||
|
||||
func init() {
|
||||
|
@ -80,7 +87,7 @@ func isOwnCommand(cmd *cobra.Command) bool {
|
|||
if cmd == nil {
|
||||
return false
|
||||
}
|
||||
if cmd.Name() == "context" || cmd.Name() == "serve" {
|
||||
if _, ok := ownCommands[cmd.Name()]; ok {
|
||||
return true
|
||||
}
|
||||
return isOwnCommand(cmd.Parent())
|
||||
|
@ -114,6 +121,7 @@ func main() {
|
|||
cmd.LogsCommand(),
|
||||
cmd.RmCommand(),
|
||||
compose.Command(),
|
||||
login.Command(),
|
||||
)
|
||||
|
||||
helpFunc := root.HelpFunc()
|
||||
|
|
|
@ -33,6 +33,11 @@ import (
|
|||
"path/filepath"
|
||||
"testing"
|
||||
|
||||
"github.com/docker/api/cli/cmd"
|
||||
"github.com/docker/api/cli/cmd/context"
|
||||
"github.com/docker/api/cli/cmd/login"
|
||||
"github.com/docker/api/cli/cmd/run"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
|
||||
"github.com/docker/api/cli/config"
|
||||
|
@ -70,3 +75,13 @@ func TestDetermineCurrentContext(t *testing.T) {
|
|||
require.NoError(t, err)
|
||||
require.Equal(t, "other-context", c)
|
||||
}
|
||||
|
||||
func TestCheckOwnCommand(t *testing.T) {
|
||||
require.True(t, isOwnCommand(login.Command()))
|
||||
require.True(t, isOwnCommand(context.Command()))
|
||||
require.True(t, isOwnCommand(cmd.ServeCommand()))
|
||||
require.False(t, isOwnCommand(run.Command()))
|
||||
require.False(t, isOwnCommand(cmd.ExecCommand()))
|
||||
require.False(t, isOwnCommand(cmd.LogsCommand()))
|
||||
require.False(t, isOwnCommand(cmd.PsCommand()))
|
||||
}
|
||||
|
|
|
@ -76,6 +76,24 @@ func (s *E2eSuite) TestContextLegacy() {
|
|||
})
|
||||
}
|
||||
|
||||
func (s *E2eSuite) TestClassicLoginWithparameters() {
|
||||
output, err := s.NewDockerCommand("login", "-u", "nouser", "-p", "wrongpasword").Exec()
|
||||
Expect(output).To(ContainSubstring("Get https://registry-1.docker.io/v2/: unauthorized: incorrect username or password"))
|
||||
Expect(err).NotTo(BeNil())
|
||||
}
|
||||
|
||||
func (s *E2eSuite) TestClassicLogin() {
|
||||
output, err := s.NewDockerCommand("login", "someregistry.docker.io").Exec()
|
||||
Expect(output).To(ContainSubstring("Cannot perform an interactive login from a non TTY device"))
|
||||
Expect(err).NotTo(BeNil())
|
||||
}
|
||||
|
||||
func (s *E2eSuite) TestCloudLogin() {
|
||||
output, err := s.NewDockerCommand("login", "mycloudbackend").Exec()
|
||||
Expect(output).To(ContainSubstring("Unknown backend type for cloud login : mycloudbackend"))
|
||||
Expect(err).NotTo(BeNil())
|
||||
}
|
||||
|
||||
func (s *E2eSuite) TestSetupError() {
|
||||
It("should display an error if cannot shell out to docker-classic", func() {
|
||||
err := os.Setenv("PATH", s.BinDir)
|
||||
|
|
Loading…
Reference in New Issue