| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206 |
- /*
- 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 (
- "context"
- "fmt"
- "os"
- "os/exec"
- "os/signal"
- "path/filepath"
- "syscall"
- "github.com/pkg/errors"
- "github.com/sirupsen/logrus"
- "github.com/spf13/cobra"
- // Backend registrations
- _ "github.com/docker/api/azure"
- _ "github.com/docker/api/example"
- _ "github.com/docker/api/moby"
- "github.com/docker/api/cli/cmd"
- "github.com/docker/api/cli/cmd/compose"
- contextcmd "github.com/docker/api/cli/cmd/context"
- "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"
- "github.com/docker/api/context/store"
- )
- var (
- runningOwnCommand bool
- )
- func init() {
- // initial hack to get the path of the project's bin dir
- // into the env of this cli for development
- path, err := filepath.Abs(filepath.Dir(os.Args[0]))
- if err != nil {
- 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 {
- panic(err)
- }
- }
- func isOwnCommand(cmd *cobra.Command) bool {
- if cmd == nil {
- return false
- }
- if cmd.Name() == "context" || cmd.Name() == "serve" {
- return true
- }
- return isOwnCommand(cmd.Parent())
- }
- func main() {
- var opts cliopts.GlobalOpts
- root := &cobra.Command{
- Use: "docker",
- Long: "docker for the 2020s",
- SilenceErrors: true,
- SilenceUsage: true,
- PersistentPreRunE: func(cmd *cobra.Command, args []string) error {
- runningOwnCommand = isOwnCommand(cmd)
- if !runningOwnCommand {
- execMoby(cmd.Context())
- }
- return nil
- },
- RunE: func(cmd *cobra.Command, args []string) error {
- return cmd.Help()
- },
- }
- root.AddCommand(
- contextcmd.Command(&opts),
- cmd.PsCommand(),
- cmd.ServeCommand(),
- run.Command(),
- cmd.ExecCommand(),
- cmd.LogsCommand(),
- cmd.RmCommand(),
- compose.Command(),
- )
- helpFunc := root.HelpFunc()
- root.SetHelpFunc(func(cmd *cobra.Command, args []string) {
- runningOwnCommand = isOwnCommand(cmd)
- if !runningOwnCommand {
- execMoby(cmd.Context())
- }
- helpFunc(cmd, args)
- })
- root.PersistentFlags().BoolVarP(&opts.Debug, "debug", "d", false, "enable debug output in the logs")
- opts.AddConfigFlags(root.PersistentFlags())
- opts.AddContextFlags(root.PersistentFlags())
- // populate the opts with the global flags
- _ = root.PersistentFlags().Parse(os.Args[1:])
- if opts.Debug {
- logrus.SetLevel(logrus.DebugLevel)
- }
- ctx, cancel := newSigContext()
- defer cancel()
- if opts.Config == "" {
- fatal(errors.New("config path cannot be empty"))
- }
- config, err := cliconfig.LoadFile(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"
- }
- s, err := store.New(store.WithRoot(opts.Config))
- if err != nil {
- fatal(errors.Wrap(err, "unable to create context store"))
- }
- ctx = apicontext.WithCurrentContext(ctx, currentContext)
- ctx = store.WithContextStore(ctx, s)
- if err = root.ExecuteContext(ctx); err != nil {
- // Context should always be handled by new CLI
- if runningOwnCommand {
- fmt.Fprintln(os.Stderr, err)
- os.Exit(1)
- }
- execMoby(ctx)
- fmt.Println(err)
- os.Exit(1)
- }
- }
- func newSigContext() (context.Context, func()) {
- ctx, cancel := context.WithCancel(context.Background())
- s := make(chan os.Signal)
- signal.Notify(s, syscall.SIGTERM, syscall.SIGINT)
- go func() {
- <-s
- cancel()
- }()
- return ctx, cancel
- }
- func execMoby(ctx context.Context) {
- currentContext := apicontext.CurrentContext(ctx)
- s := store.ContextStore(ctx)
- _, err := s.Get(currentContext, nil)
- // Only run original docker command if the current context is not
- // ours.
- if err != nil {
- cmd := exec.Command("docker", os.Args[1:]...)
- cmd.Stdin = os.Stdin
- cmd.Stdout = os.Stdout
- cmd.Stderr = os.Stderr
- if err := cmd.Run(); err != nil {
- if exiterr, ok := err.(*exec.ExitError); ok {
- os.Exit(exiterr.ExitCode())
- }
- os.Exit(1)
- }
- os.Exit(0)
- }
- }
- func fatal(err error) {
- fmt.Fprint(os.Stderr, err)
- os.Exit(1)
- }
|