From f6fcd27a097ddd6fef646dc85230d06ad04034d2 Mon Sep 17 00:00:00 2001 From: Christopher Crone Date: Mon, 18 May 2020 14:38:21 +0200 Subject: [PATCH 1/2] Factor determination of current context Signed-off-by: Christopher Crone --- cli/main.go | 27 ++++++++++++------ cli/main_test.go | 71 ++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 89 insertions(+), 9 deletions(-) create mode 100644 cli/main_test.go diff --git a/cli/main.go b/cli/main.go index bf9e9bdee..bb943b06b 100644 --- a/cli/main.go +++ b/cli/main.go @@ -136,16 +136,10 @@ func main() { if opts.Config == "" { fatal(errors.New("config path cannot be empty")) } - config, err := cliconfig.LoadFile(opts.Config) + + currentContext, err := determineCurrentContext(opts.Context, opts.Config) if err != nil { - fatal(errors.Wrap(err, "unable to find configuration file")) - } - currentContext := opts.Context - if currentContext == "" { - currentContext = config.CurrentContext - } - if currentContext == "" { - currentContext = "default" + fatal(errors.New("unable to determine current context")) } s, err := store.New(store.WithRoot(opts.Config)) @@ -200,6 +194,21 @@ func execMoby(ctx context.Context) { } } +func determineCurrentContext(flag string, configDir string) (string, error) { + res := flag + if res == "" { + config, err := cliconfig.LoadFile(configDir) + if err != nil { + return "", err + } + res = config.CurrentContext + } + if res == "" { + res = "default" + } + return res, nil +} + func fatal(err error) { fmt.Fprint(os.Stderr, err) os.Exit(1) diff --git a/cli/main_test.go b/cli/main_test.go new file mode 100644 index 000000000..a8a6c9f7d --- /dev/null +++ b/cli/main_test.go @@ -0,0 +1,71 @@ +/* + Copyright (c) 2020 Docker Inc. + + Permission is hereby granted, free of charge, to any person + obtaining a copy of this software and associated documentation + files (the "Software"), to deal in the Software without + restriction, including without limitation the rights to use, copy, + modify, merge, publish, distribute, sublicense, and/or sell copies + of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, + INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. + IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, + DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, + TORT OR OTHERWISE, + ARISING FROM, OUT OF OR IN CONNECTION WITH + THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +*/ + +package main + +import ( + "io/ioutil" + "os" + "path/filepath" + "testing" + + "github.com/stretchr/testify/require" + + "github.com/docker/api/cli/config" +) + +var contextSetConfig = []byte(`{ + "currentContext": "some-context" +}`) + +func TestDetermineCurrentContext(t *testing.T) { + d, err := ioutil.TempDir("", "") + defer os.RemoveAll(d) + require.NoError(t, err) + err = ioutil.WriteFile(filepath.Join(d, config.ConfigFileName), contextSetConfig, 0644) + require.NoError(t, err) + + // If nothing set, fallback to default + c, err := determineCurrentContext("", "") + require.NoError(t, err) + require.Equal(t, "default", c) + + // If context flag set, use that + c, err = determineCurrentContext("other-context", "") + require.NoError(t, err) + require.Equal(t, "other-context", c) + + // If no context flag, use config + c, err = determineCurrentContext("", d) + require.NoError(t, err) + require.Equal(t, "some-context", c) + + // Ensure context flag overrides config + c, err = determineCurrentContext("other-context", d) + require.NoError(t, err) + require.Equal(t, "other-context", c) +} From 10d826eb108d8ec8acf9c3e8a06739dc254c22dd Mon Sep 17 00:00:00 2001 From: Christopher Crone Date: Mon, 18 May 2020 13:57:38 +0200 Subject: [PATCH 2/2] Add context show command Signed-off-by: Christopher Crone --- cli/cmd/context/context.go | 1 + cli/cmd/context/show.go | 63 ++++++++++++++++++++++++++++++++++++++ tests/e2e/e2e.go | 6 +++- 3 files changed, 69 insertions(+), 1 deletion(-) create mode 100644 cli/cmd/context/show.go diff --git a/cli/cmd/context/context.go b/cli/cmd/context/context.go index 29ceb7d45..441e507ab 100644 --- a/cli/cmd/context/context.go +++ b/cli/cmd/context/context.go @@ -46,6 +46,7 @@ func Command(opts *cliopts.GlobalOpts) *cobra.Command { createCommand(), listCommand(), removeCommand(), + showCommand(opts), useCommand(opts), login.Command(), ) diff --git a/cli/cmd/context/show.go b/cli/cmd/context/show.go new file mode 100644 index 000000000..54f4bd504 --- /dev/null +++ b/cli/cmd/context/show.go @@ -0,0 +1,63 @@ +/* + Copyright (c) 2020 Docker Inc. + + Permission is hereby granted, free of charge, to any person + obtaining a copy of this software and associated documentation + files (the "Software"), to deal in the Software without + restriction, including without limitation the rights to use, copy, + modify, merge, publish, distribute, sublicense, and/or sell copies + of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, + INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. + IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, + DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, + TORT OR OTHERWISE, + ARISING FROM, OUT OF OR IN CONNECTION WITH + THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +*/ + +package context + +import ( + "context" + "fmt" + + "github.com/spf13/cobra" + + cliopts "github.com/docker/api/cli/options" + apicontext "github.com/docker/api/context" + "github.com/docker/api/context/store" +) + +func showCommand(opts *cliopts.GlobalOpts) *cobra.Command { + return &cobra.Command{ + Use: "show", + Short: "Print the current context", + Args: cobra.NoArgs, + RunE: func(cmd *cobra.Command, args []string) error { + return runShow(cmd.Context(), opts) + }, + } +} + +func runShow(ctx context.Context, opts *cliopts.GlobalOpts) error { + s := store.ContextStore(ctx) + name := apicontext.CurrentContext(ctx) + // Match behavior of existing CLI + if name != store.DefaultContextName { + if _, err := s.Get(name, nil); err != nil { + return err + } + } + fmt.Println(name) + return nil +} diff --git a/tests/e2e/e2e.go b/tests/e2e/e2e.go index a0fe70cf3..e54e6a1b5 100644 --- a/tests/e2e/e2e.go +++ b/tests/e2e/e2e.go @@ -21,7 +21,9 @@ func main() { It("should be initialized with default context", func() { NewDockerCommand("context", "use", "default").ExecOrDie() - output := NewCommand("docker", "context", "ls").ExecOrDie() + output := NewDockerCommand("context", "show").ExecOrDie() + Expect(output).To(ContainSubstring("default")) + output = NewCommand("docker", "context", "ls").ExecOrDie() Expect(output).To(Not(ContainSubstring("test-example"))) Expect(output).To(ContainSubstring("default *")) }) @@ -59,6 +61,8 @@ func main() { Expect(currentContext).To(ContainSubstring("test-example")) output := NewCommand("docker", "context", "ls").ExecOrDie() Expect(output).To(ContainSubstring("test-example *")) + output = NewDockerCommand("context", "show").ExecOrDie() + Expect(output).To(ContainSubstring("test-example")) }) It("can run ps command", func() {