mirror of
https://github.com/docker/compose.git
synced 2025-07-25 22:54:54 +02:00
Merge pull request #82 from chris-crone/context-use
Add context use command
This commit is contained in:
commit
cc23137945
@ -36,11 +36,13 @@ import (
|
|||||||
"github.com/hashicorp/go-multierror"
|
"github.com/hashicorp/go-multierror"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
|
|
||||||
|
cliconfig "github.com/docker/api/cli/config"
|
||||||
|
cliopts "github.com/docker/api/cli/options"
|
||||||
"github.com/docker/api/context/store"
|
"github.com/docker/api/context/store"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ContextCommand manages contexts
|
// ContextCommand manages contexts
|
||||||
func ContextCommand() *cobra.Command {
|
func ContextCommand(opts *cliopts.GlobalOpts) *cobra.Command {
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "context",
|
Use: "context",
|
||||||
Short: "Manage contexts",
|
Short: "Manage contexts",
|
||||||
@ -50,6 +52,7 @@ func ContextCommand() *cobra.Command {
|
|||||||
createCommand(),
|
createCommand(),
|
||||||
listCommand(),
|
listCommand(),
|
||||||
removeCommand(),
|
removeCommand(),
|
||||||
|
useCommand(opts),
|
||||||
)
|
)
|
||||||
|
|
||||||
return cmd
|
return cmd
|
||||||
@ -84,6 +87,7 @@ func createCommand() *cobra.Command {
|
|||||||
func listCommand() *cobra.Command {
|
func listCommand() *cobra.Command {
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "list",
|
Use: "list",
|
||||||
|
Short: "List available contexts",
|
||||||
Aliases: []string{"ls"},
|
Aliases: []string{"ls"},
|
||||||
Args: cobra.NoArgs,
|
Args: cobra.NoArgs,
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
@ -95,7 +99,8 @@ func listCommand() *cobra.Command {
|
|||||||
|
|
||||||
func removeCommand() *cobra.Command {
|
func removeCommand() *cobra.Command {
|
||||||
return &cobra.Command{
|
return &cobra.Command{
|
||||||
Use: "rm",
|
Use: "rm CONTEXT [CONTEXT...]",
|
||||||
|
Short: "Remove one or more contexts",
|
||||||
Aliases: []string{"remove"},
|
Aliases: []string{"remove"},
|
||||||
Args: cobra.MinimumNArgs(1),
|
Args: cobra.MinimumNArgs(1),
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
@ -104,6 +109,17 @@ func removeCommand() *cobra.Command {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func useCommand(opts *cliopts.GlobalOpts) *cobra.Command {
|
||||||
|
return &cobra.Command{
|
||||||
|
Use: "use CONTEXT",
|
||||||
|
Short: "Set the default context",
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
|
return runUse(cmd.Context(), opts.Config, args[0])
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func runCreate(ctx context.Context, opts createOpts, name string, contextType string) error {
|
func runCreate(ctx context.Context, opts createOpts, name string, contextType string) error {
|
||||||
switch contextType {
|
switch contextType {
|
||||||
case "aci":
|
case "aci":
|
||||||
@ -160,3 +176,18 @@ func runRemove(ctx context.Context, args []string) error {
|
|||||||
}
|
}
|
||||||
return errs.ErrorOrNil()
|
return errs.ErrorOrNil()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func runUse(ctx context.Context, configDir string, name string) error {
|
||||||
|
s := store.ContextStore(ctx)
|
||||||
|
// Match behavior of existing CLI
|
||||||
|
if name != store.DefaultContextName {
|
||||||
|
if _, err := s.Get(name, nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err := cliconfig.WriteCurrentContext(configDir, name); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Println(name)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
96
cli/config/config.go
Normal file
96
cli/config/config.go
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
/*
|
||||||
|
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 config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LoadFile loads the docker configuration
|
||||||
|
func LoadFile(dir string) (*File, error) {
|
||||||
|
f := &File{}
|
||||||
|
err := loadFile(configFilePath(dir), &f)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return f, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// WriteCurrentContext writes the selected current context to the Docker
|
||||||
|
// configuration file. Note, the validity of the context is not checked.
|
||||||
|
func WriteCurrentContext(dir string, name string) error {
|
||||||
|
m := map[string]interface{}{}
|
||||||
|
path := configFilePath(dir)
|
||||||
|
err := loadFile(path, &m)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// Match existing CLI behavior
|
||||||
|
if name == "default" {
|
||||||
|
delete(m, currentContextKey)
|
||||||
|
} else {
|
||||||
|
m[currentContextKey] = name
|
||||||
|
}
|
||||||
|
return writeFile(path, m)
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeFile(path string, content map[string]interface{}) error {
|
||||||
|
d, err := json.MarshalIndent(content, "", "\t")
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "unable to marshal config")
|
||||||
|
}
|
||||||
|
err = ioutil.WriteFile(path, d, 0644)
|
||||||
|
return errors.Wrap(err, "unable to write config file")
|
||||||
|
}
|
||||||
|
|
||||||
|
func loadFile(path string, dest interface{}) error {
|
||||||
|
data, err := ioutil.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
// Not an error if there is no config, we're just using defaults
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return errors.Wrap(err, "unable to read config file")
|
||||||
|
}
|
||||||
|
err = json.Unmarshal(data, dest)
|
||||||
|
return errors.Wrap(err, "unable to unmarshal config")
|
||||||
|
}
|
||||||
|
|
||||||
|
func configFilePath(dir string) string {
|
||||||
|
return filepath.Join(dir, ConfigFileName)
|
||||||
|
}
|
||||||
|
|
||||||
|
// File contains the current context from the docker configuration file
|
||||||
|
type File struct {
|
||||||
|
CurrentContext string `json:"currentContext,omitempty"`
|
||||||
|
}
|
@ -25,44 +25,36 @@
|
|||||||
THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package context
|
package config
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/spf13/pflag"
|
||||||
)
|
)
|
||||||
|
|
||||||
// LoadConfigFile loads the docker configuration
|
const (
|
||||||
func LoadConfigFile(configDir string, configFileName string) (*ConfigFile, error) {
|
// ConfigFileName is the name of config file
|
||||||
filename := filepath.Join(configDir, configFileName)
|
ConfigFileName = "config.json"
|
||||||
configFile := &ConfigFile{
|
// ConfigFileDir is the default folder where the config file is stored
|
||||||
Filename: filename,
|
ConfigFileDir = ".docker"
|
||||||
}
|
// ConfigFlagName is the name of the config flag
|
||||||
|
ConfigFlagName = "config"
|
||||||
|
)
|
||||||
|
|
||||||
if _, err := os.Stat(filename); err == nil {
|
// ConfigFlags are the global CLI flags
|
||||||
file, err := os.Open(filename)
|
// nolint stutter
|
||||||
if err != nil {
|
type ConfigFlags struct {
|
||||||
return nil, fmt.Errorf("can't read %s: %w", filename, err)
|
Config string
|
||||||
}
|
|
||||||
// nolint errcheck
|
|
||||||
defer file.Close()
|
|
||||||
err = json.NewDecoder(file).Decode(&configFile)
|
|
||||||
if err != nil {
|
|
||||||
err = fmt.Errorf("can't read %s: %w", filename, err)
|
|
||||||
}
|
|
||||||
return configFile, err
|
|
||||||
} else if !os.IsNotExist(err) {
|
|
||||||
// if file is there but we can't stat it for any reason other
|
|
||||||
// than it doesn't exist then stop
|
|
||||||
return nil, fmt.Errorf("can't read %s: %w", filename, err)
|
|
||||||
}
|
|
||||||
return configFile, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ConfigFile contains the current context from the docker configuration file
|
// AddConfigFlags adds persistent (global) flags
|
||||||
type ConfigFile struct {
|
func (c *ConfigFlags) AddConfigFlags(flags *pflag.FlagSet) {
|
||||||
Filename string `json:"-"` // Note: for internal use only
|
flags.StringVar(&c.Config, ConfigFlagName, filepath.Join(home(), ConfigFileDir), "Location of the client config files `DIRECTORY`")
|
||||||
CurrentContext string `json:"currentContext,omitempty"`
|
}
|
||||||
|
|
||||||
|
func home() string {
|
||||||
|
home, _ := os.UserHomeDir()
|
||||||
|
return home
|
||||||
}
|
}
|
34
cli/config/keys.go
Normal file
34
cli/config/keys.go
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
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 config
|
||||||
|
|
||||||
|
const (
|
||||||
|
// currentContextKey is the key used in the Docker config file to set the
|
||||||
|
// default context
|
||||||
|
currentContextKey = "currentContext"
|
||||||
|
)
|
42
cli/main.go
42
cli/main.go
@ -30,13 +30,13 @@ package main
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
|
|
||||||
@ -48,6 +48,8 @@ import (
|
|||||||
"github.com/docker/api/cli/cmd"
|
"github.com/docker/api/cli/cmd"
|
||||||
"github.com/docker/api/cli/cmd/compose"
|
"github.com/docker/api/cli/cmd/compose"
|
||||||
"github.com/docker/api/cli/cmd/run"
|
"github.com/docker/api/cli/cmd/run"
|
||||||
|
cliconfig "github.com/docker/api/cli/config"
|
||||||
|
cliopts "github.com/docker/api/cli/options"
|
||||||
apicontext "github.com/docker/api/context"
|
apicontext "github.com/docker/api/context"
|
||||||
"github.com/docker/api/context/store"
|
"github.com/docker/api/context/store"
|
||||||
)
|
)
|
||||||
@ -56,17 +58,12 @@ var (
|
|||||||
runningOwnCommand bool
|
runningOwnCommand bool
|
||||||
)
|
)
|
||||||
|
|
||||||
type mainOpts struct {
|
|
||||||
apicontext.Flags
|
|
||||||
debug bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
// initial hack to get the path of the project's bin dir
|
// initial hack to get the path of the project's bin dir
|
||||||
// into the env of this cli for development
|
// into the env of this cli for development
|
||||||
path, err := filepath.Abs(filepath.Dir(os.Args[0]))
|
path, err := filepath.Abs(filepath.Dir(os.Args[0]))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
fatal(errors.Wrap(err, "unable to get absolute bin path"))
|
||||||
}
|
}
|
||||||
if err := os.Setenv("PATH", fmt.Sprintf("%s:%s", os.Getenv("PATH"), path)); err != nil {
|
if err := os.Setenv("PATH", fmt.Sprintf("%s:%s", os.Getenv("PATH"), path)); err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
@ -84,7 +81,7 @@ func isOwnCommand(cmd *cobra.Command) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
var opts mainOpts
|
var opts cliopts.GlobalOpts
|
||||||
root := &cobra.Command{
|
root := &cobra.Command{
|
||||||
Use: "docker",
|
Use: "docker",
|
||||||
Long: "docker for the 2020s",
|
Long: "docker for the 2020s",
|
||||||
@ -103,7 +100,7 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
root.AddCommand(
|
root.AddCommand(
|
||||||
cmd.ContextCommand(),
|
cmd.ContextCommand(&opts),
|
||||||
cmd.PsCommand(),
|
cmd.PsCommand(),
|
||||||
cmd.ServeCommand(),
|
cmd.ServeCommand(),
|
||||||
run.Command(),
|
run.Command(),
|
||||||
@ -122,21 +119,25 @@ func main() {
|
|||||||
helpFunc(cmd, args)
|
helpFunc(cmd, args)
|
||||||
})
|
})
|
||||||
|
|
||||||
root.PersistentFlags().BoolVarP(&opts.debug, "debug", "d", false, "enable debug output in the logs")
|
root.PersistentFlags().BoolVarP(&opts.Debug, "debug", "d", false, "enable debug output in the logs")
|
||||||
opts.AddFlags(root.PersistentFlags())
|
opts.AddConfigFlags(root.PersistentFlags())
|
||||||
|
opts.AddContextFlags(root.PersistentFlags())
|
||||||
|
|
||||||
// populate the opts with the global flags
|
// populate the opts with the global flags
|
||||||
_ = root.PersistentFlags().Parse(os.Args[1:])
|
_ = root.PersistentFlags().Parse(os.Args[1:])
|
||||||
if opts.debug {
|
if opts.Debug {
|
||||||
logrus.SetLevel(logrus.DebugLevel)
|
logrus.SetLevel(logrus.DebugLevel)
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := newSigContext()
|
ctx, cancel := newSigContext()
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
config, err := apicontext.LoadConfigFile(opts.Config, "config.json")
|
if opts.Config == "" {
|
||||||
|
fatal(errors.New("config path cannot be empty"))
|
||||||
|
}
|
||||||
|
config, err := cliconfig.LoadFile(opts.Config)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logrus.Fatal("unable ot find configuration")
|
fatal(errors.Wrap(err, "unable to find configuration file"))
|
||||||
}
|
}
|
||||||
currentContext := opts.Context
|
currentContext := opts.Context
|
||||||
if currentContext == "" {
|
if currentContext == "" {
|
||||||
@ -146,15 +147,11 @@ func main() {
|
|||||||
currentContext = "default"
|
currentContext = "default"
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx = apicontext.WithCurrentContext(ctx, currentContext)
|
|
||||||
if err != nil {
|
|
||||||
logrus.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
s, err := store.New(store.WithRoot(opts.Config))
|
s, err := store.New(store.WithRoot(opts.Config))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logrus.Fatal(err)
|
fatal(errors.Wrap(err, "unable to create context store"))
|
||||||
}
|
}
|
||||||
|
ctx = apicontext.WithCurrentContext(ctx, currentContext)
|
||||||
ctx = store.WithContextStore(ctx, s)
|
ctx = store.WithContextStore(ctx, s)
|
||||||
|
|
||||||
if err = root.ExecuteContext(ctx); err != nil {
|
if err = root.ExecuteContext(ctx); err != nil {
|
||||||
@ -201,3 +198,8 @@ func execMoby(ctx context.Context) {
|
|||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func fatal(err error) {
|
||||||
|
fmt.Fprint(os.Stderr, err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
40
cli/options/options.go
Normal file
40
cli/options/options.go
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
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 options
|
||||||
|
|
||||||
|
import (
|
||||||
|
apicontext "github.com/docker/api/context"
|
||||||
|
cliconfig "github.com/docker/api/cli/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GlobalOpts contains the global CLI options
|
||||||
|
type GlobalOpts struct {
|
||||||
|
apicontext.ContextFlags
|
||||||
|
cliconfig.ConfigFlags
|
||||||
|
Debug bool
|
||||||
|
}
|
@ -29,31 +29,17 @@ package context
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
|
||||||
|
|
||||||
"github.com/mitchellh/go-homedir"
|
|
||||||
"github.com/spf13/pflag"
|
"github.com/spf13/pflag"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
// ContextFlags are the global CLI flags
|
||||||
// ConfigFileName is the name of config file
|
// nolint stutter
|
||||||
ConfigFileName = "config.json"
|
type ContextFlags struct {
|
||||||
configFileDir = ".docker"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Flags are the global cli flags
|
|
||||||
type Flags struct {
|
|
||||||
Config string
|
|
||||||
Context string
|
Context string
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddFlags adds persistent (global) flags
|
// AddContextFlags adds persistent (global) flags
|
||||||
func (c *Flags) AddFlags(flags *pflag.FlagSet) {
|
func (c *ContextFlags) AddContextFlags(flags *pflag.FlagSet) {
|
||||||
flags.StringVar(&c.Config, "config", filepath.Join(home(), configFileDir), "Location of the client config files `DIRECTORY`")
|
|
||||||
flags.StringVarP(&c.Context, "context", "c", os.Getenv("DOCKER_CONTEXT"), "context")
|
flags.StringVarP(&c.Context, "context", "c", os.Getenv("DOCKER_CONTEXT"), "context")
|
||||||
}
|
}
|
||||||
|
|
||||||
func home() string {
|
|
||||||
home, _ := homedir.Dir()
|
|
||||||
return home
|
|
||||||
}
|
|
||||||
|
@ -36,9 +36,15 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
|
||||||
"github.com/docker/api/errdefs"
|
|
||||||
"github.com/opencontainers/go-digest"
|
"github.com/opencontainers/go-digest"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
|
"github.com/docker/api/errdefs"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// DefaultContextName is an automatically generated local context
|
||||||
|
DefaultContextName = "default"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -187,6 +193,9 @@ func (s *store) GetType(meta *Metadata) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *store) Create(name string, data TypedContext) error {
|
func (s *store) Create(name string, data TypedContext) error {
|
||||||
|
if name == DefaultContextName {
|
||||||
|
return errors.Wrap(errdefs.ErrAlreadyExists, objectName(name))
|
||||||
|
}
|
||||||
dir := contextdirOf(name)
|
dir := contextdirOf(name)
|
||||||
metaDir := filepath.Join(s.root, contextsDir, metadataDir, dir)
|
metaDir := filepath.Join(s.root, contextsDir, metadataDir, dir)
|
||||||
if _, err := os.Stat(metaDir); !os.IsNotExist(err) {
|
if _, err := os.Stat(metaDir); !os.IsNotExist(err) {
|
||||||
@ -242,6 +251,9 @@ func (s *store) List() ([]*Metadata, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *store) Remove(name string) error {
|
func (s *store) Remove(name string) error {
|
||||||
|
if name == DefaultContextName {
|
||||||
|
return errors.Wrap(errdefs.ErrForbidden, objectName(name))
|
||||||
|
}
|
||||||
dir := filepath.Join(s.root, contextsDir, metadataDir, contextdirOf(name))
|
dir := filepath.Join(s.root, contextsDir, metadataDir, contextdirOf(name))
|
||||||
// Check if directory exists because os.RemoveAll returns nil if it doesn't
|
// Check if directory exists because os.RemoveAll returns nil if it doesn't
|
||||||
if _, err := os.Stat(dir); os.IsNotExist(err) {
|
if _, err := os.Stat(dir); os.IsNotExist(err) {
|
||||||
|
@ -36,6 +36,8 @@ var (
|
|||||||
ErrNotFound = errors.New("not found")
|
ErrNotFound = errors.New("not found")
|
||||||
// ErrAlreadyExists is returned when an object already exists
|
// ErrAlreadyExists is returned when an object already exists
|
||||||
ErrAlreadyExists = errors.New("already exists")
|
ErrAlreadyExists = errors.New("already exists")
|
||||||
|
// ErrForbidden is returned when an operation is not permitted
|
||||||
|
ErrForbidden = errors.New("forbidden")
|
||||||
// ErrUnknown is returned when the error type is unmapped
|
// ErrUnknown is returned when the error type is unmapped
|
||||||
ErrUnknown = errors.New("unknown")
|
ErrUnknown = errors.New("unknown")
|
||||||
)
|
)
|
||||||
@ -50,6 +52,11 @@ func IsAlreadyExistsError(err error) bool {
|
|||||||
return errors.Is(err, ErrAlreadyExists)
|
return errors.Is(err, ErrAlreadyExists)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsForbiddenError returns true if the unwrapped error is ErrForbidden
|
||||||
|
func IsForbiddenError(err error) bool {
|
||||||
|
return errors.Is(err, ErrForbidden)
|
||||||
|
}
|
||||||
|
|
||||||
// IsUnknownError returns true if the unwrapped error is ErrUnknown
|
// IsUnknownError returns true if the unwrapped error is ErrUnknown
|
||||||
func IsUnknownError(err error) bool {
|
func IsUnknownError(err error) bool {
|
||||||
return errors.Is(err, ErrUnknown)
|
return errors.Is(err, ErrUnknown)
|
||||||
|
@ -31,19 +31,33 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestIsNotFound(t *testing.T) {
|
func TestIsNotFound(t *testing.T) {
|
||||||
err := errors.Wrap(ErrNotFound, `object "name"`)
|
err := errors.Wrap(ErrNotFound, `object "name"`)
|
||||||
require.True(t, IsNotFoundError(err))
|
assert.True(t, IsNotFoundError(err))
|
||||||
|
|
||||||
require.False(t, IsNotFoundError(errors.New("another error")))
|
assert.False(t, IsNotFoundError(errors.New("another error")))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestIsAlreadyExists(t *testing.T) {
|
func TestIsAlreadyExists(t *testing.T) {
|
||||||
err := errors.Wrap(ErrAlreadyExists, `object "name"`)
|
err := errors.Wrap(ErrAlreadyExists, `object "name"`)
|
||||||
require.True(t, IsAlreadyExistsError(err))
|
assert.True(t, IsAlreadyExistsError(err))
|
||||||
|
|
||||||
require.False(t, IsAlreadyExistsError(errors.New("another error")))
|
assert.False(t, IsAlreadyExistsError(errors.New("another error")))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIsForbidden(t *testing.T) {
|
||||||
|
err := errors.Wrap(ErrForbidden, `object "name"`)
|
||||||
|
assert.True(t, IsForbiddenError(err))
|
||||||
|
|
||||||
|
assert.False(t, IsForbiddenError(errors.New("another error")))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIsUnknown(t *testing.T) {
|
||||||
|
err := errors.Wrap(ErrUnknown, `object "name"`)
|
||||||
|
assert.True(t, IsUnknownError(err))
|
||||||
|
|
||||||
|
assert.False(t, IsUnknownError(errors.New("another error")))
|
||||||
}
|
}
|
||||||
|
1
go.mod
1
go.mod
@ -23,7 +23,6 @@ require (
|
|||||||
github.com/gorilla/mux v1.7.4 // indirect
|
github.com/gorilla/mux v1.7.4 // indirect
|
||||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0
|
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0
|
||||||
github.com/hashicorp/go-multierror v1.1.0
|
github.com/hashicorp/go-multierror v1.1.0
|
||||||
github.com/mitchellh/go-homedir v1.1.0
|
|
||||||
github.com/morikuni/aec v1.0.0 // indirect
|
github.com/morikuni/aec v1.0.0 // indirect
|
||||||
github.com/onsi/gomega v1.9.0
|
github.com/onsi/gomega v1.9.0
|
||||||
github.com/opencontainers/go-digest v1.0.0-rc1
|
github.com/opencontainers/go-digest v1.0.0-rc1
|
||||||
|
@ -20,7 +20,7 @@ func main() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
It("should be initialized with default context", func() {
|
It("should be initialized with default context", func() {
|
||||||
NewCommand("docker", "context", "use", "default").ExecOrDie()
|
NewDockerCommand("context", "use", "default").ExecOrDie()
|
||||||
output := NewCommand("docker", "context", "ls").ExecOrDie()
|
output := NewCommand("docker", "context", "ls").ExecOrDie()
|
||||||
Expect(output).To(Not(ContainSubstring("test-example")))
|
Expect(output).To(Not(ContainSubstring("test-example")))
|
||||||
Expect(output).To(ContainSubstring("default *"))
|
Expect(output).To(ContainSubstring("default *"))
|
||||||
@ -52,9 +52,10 @@ func main() {
|
|||||||
// Expect(output).To(ContainSubstring("test-example context acitest created"))
|
// Expect(output).To(ContainSubstring("test-example context acitest created"))
|
||||||
})
|
})
|
||||||
defer NewDockerCommand("context", "rm", "test-example").ExecOrDie()
|
defer NewDockerCommand("context", "rm", "test-example").ExecOrDie()
|
||||||
|
defer NewDockerCommand("context", "use", "default").ExecOrDie()
|
||||||
|
|
||||||
It("uses the test context", func() {
|
It("uses the test context", func() {
|
||||||
currentContext := NewCommand("docker", "context", "use", "test-example").ExecOrDie()
|
currentContext := NewDockerCommand("context", "use", "test-example").ExecOrDie()
|
||||||
Expect(currentContext).To(ContainSubstring("test-example"))
|
Expect(currentContext).To(ContainSubstring("test-example"))
|
||||||
output := NewCommand("docker", "context", "ls").ExecOrDie()
|
output := NewCommand("docker", "context", "ls").ExecOrDie()
|
||||||
Expect(output).To(ContainSubstring("test-example *"))
|
Expect(output).To(ContainSubstring("test-example *"))
|
||||||
|
Loading…
x
Reference in New Issue
Block a user