123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 |
- package main
- import (
- "errors"
- "fmt"
- "os"
- "github.com/Sirupsen/logrus"
- "github.com/docker/docker/api/types/versions"
- "github.com/docker/docker/cli"
- "github.com/docker/docker/cli/command"
- "github.com/docker/docker/cli/command/commands"
- cliconfig "github.com/docker/docker/cli/config"
- "github.com/docker/docker/cli/debug"
- cliflags "github.com/docker/docker/cli/flags"
- "github.com/docker/docker/dockerversion"
- "github.com/docker/docker/pkg/term"
- "github.com/spf13/cobra"
- "github.com/spf13/pflag"
- )
- func newDockerCommand(dockerCli *command.DockerCli) *cobra.Command {
- opts := cliflags.NewClientOptions()
- var flags *pflag.FlagSet
- cmd := &cobra.Command{
- Use: "docker [OPTIONS] COMMAND [ARG...]",
- Short: "A self-sufficient runtime for containers",
- SilenceUsage: true,
- SilenceErrors: true,
- TraverseChildren: true,
- Args: noArgs,
- RunE: func(cmd *cobra.Command, args []string) error {
- if opts.Version {
- showVersion()
- return nil
- }
- return dockerCli.ShowHelp(cmd, args)
- },
- PersistentPreRunE: func(cmd *cobra.Command, args []string) error {
- // daemon command is special, we redirect directly to another binary
- if cmd.Name() == "daemon" {
- return nil
- }
- // flags must be the top-level command flags, not cmd.Flags()
- opts.Common.SetDefaultOptions(flags)
- dockerPreRun(opts)
- if err := dockerCli.Initialize(opts); err != nil {
- return err
- }
- return isSupported(cmd, dockerCli.Client().ClientVersion(), dockerCli.HasExperimental())
- },
- }
- cli.SetupRootCommand(cmd)
- cmd.SetHelpFunc(func(ccmd *cobra.Command, args []string) {
- if dockerCli.Client() == nil { // when using --help, PersistenPreRun is not called, so initialization is needed.
- // flags must be the top-level command flags, not cmd.Flags()
- opts.Common.SetDefaultOptions(flags)
- dockerPreRun(opts)
- dockerCli.Initialize(opts)
- }
- if err := isSupported(ccmd, dockerCli.Client().ClientVersion(), dockerCli.HasExperimental()); err != nil {
- ccmd.Println(err)
- return
- }
- hideUnsupportedFeatures(ccmd, dockerCli.Client().ClientVersion(), dockerCli.HasExperimental())
- if err := ccmd.Help(); err != nil {
- ccmd.Println(err)
- }
- })
- flags = cmd.Flags()
- flags.BoolVarP(&opts.Version, "version", "v", false, "Print version information and quit")
- flags.StringVar(&opts.ConfigDir, "config", cliconfig.Dir(), "Location of client config files")
- opts.Common.InstallFlags(flags)
- cmd.SetOutput(dockerCli.Out())
- cmd.AddCommand(newDaemonCommand())
- commands.AddCommands(cmd, dockerCli)
- return cmd
- }
- func noArgs(cmd *cobra.Command, args []string) error {
- if len(args) == 0 {
- return nil
- }
- return fmt.Errorf(
- "docker: '%s' is not a docker command.\nSee 'docker --help'", args[0])
- }
- func main() {
- // Set terminal emulation based on platform as required.
- stdin, stdout, stderr := term.StdStreams()
- logrus.SetOutput(stderr)
- dockerCli := command.NewDockerCli(stdin, stdout, stderr)
- cmd := newDockerCommand(dockerCli)
- if err := cmd.Execute(); err != nil {
- if sterr, ok := err.(cli.StatusError); ok {
- if sterr.Status != "" {
- fmt.Fprintln(stderr, sterr.Status)
- }
- // StatusError should only be used for errors, and all errors should
- // have a non-zero exit status, so never exit with 0
- if sterr.StatusCode == 0 {
- os.Exit(1)
- }
- os.Exit(sterr.StatusCode)
- }
- fmt.Fprintln(stderr, err)
- os.Exit(1)
- }
- }
- func showVersion() {
- fmt.Printf("Docker version %s, build %s\n", dockerversion.Version, dockerversion.GitCommit)
- }
- func dockerPreRun(opts *cliflags.ClientOptions) {
- cliflags.SetLogLevel(opts.Common.LogLevel)
- if opts.ConfigDir != "" {
- cliconfig.SetDir(opts.ConfigDir)
- }
- if opts.Common.Debug {
- debug.Enable()
- }
- }
- func hideUnsupportedFeatures(cmd *cobra.Command, clientVersion string, hasExperimental bool) {
- cmd.Flags().VisitAll(func(f *pflag.Flag) {
- // hide experimental flags
- if !hasExperimental {
- if _, ok := f.Annotations["experimental"]; ok {
- f.Hidden = true
- }
- }
- // hide flags not supported by the server
- if flagVersion, ok := f.Annotations["version"]; ok && len(flagVersion) == 1 && versions.LessThan(clientVersion, flagVersion[0]) {
- f.Hidden = true
- }
- })
- for _, subcmd := range cmd.Commands() {
- // hide experimental subcommands
- if !hasExperimental {
- if _, ok := subcmd.Tags["experimental"]; ok {
- subcmd.Hidden = true
- }
- }
- // hide subcommands not supported by the server
- if subcmdVersion, ok := subcmd.Tags["version"]; ok && versions.LessThan(clientVersion, subcmdVersion) {
- subcmd.Hidden = true
- }
- }
- }
- func isSupported(cmd *cobra.Command, clientVersion string, hasExperimental bool) error {
- if !hasExperimental {
- if _, ok := cmd.Tags["experimental"]; ok {
- return errors.New("only supported with experimental daemon")
- }
- }
- if cmdVersion, ok := cmd.Tags["version"]; ok && versions.LessThan(clientVersion, cmdVersion) {
- return fmt.Errorf("only supported with daemon version >= %s", cmdVersion)
- }
- return nil
- }
|