mirror of https://github.com/docker/cli.git
191 lines
9.9 KiB
Go
191 lines
9.9 KiB
Go
package container
|
|
|
|
import (
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/docker/cli/cli/command"
|
|
"github.com/docker/cli/cli/command/completion"
|
|
"github.com/docker/docker/api/types/container"
|
|
"github.com/moby/sys/capability"
|
|
"github.com/moby/sys/signal"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
// allCaps is the magic value for "all capabilities".
|
|
const allCaps = "ALL"
|
|
|
|
// allLinuxCapabilities is a list of all known Linux capabilities.
|
|
//
|
|
// TODO(thaJeztah): add descriptions, and enable descriptions for our completion scripts (cobra.CompletionOptions.DisableDescriptions is currently set to "true")
|
|
// TODO(thaJeztah): consider what casing we want to use for completion (see below);
|
|
//
|
|
// We need to consider what format is most convenient; currently we use the
|
|
// canonical name (uppercase and "CAP_" prefix), however, tab-completion is
|
|
// case-sensitive by default, so requires the user to type uppercase letters
|
|
// to filter the list of options.
|
|
//
|
|
// Bash completion provides a `completion-ignore-case on` option to make completion
|
|
// case-insensitive (https://askubuntu.com/a/87066), but it looks to be a global
|
|
// option; the current cobra.CompletionOptions also don't provide this as an option
|
|
// to be used in the generated completion-script.
|
|
//
|
|
// Fish completion has `smartcase` (by default?) which matches any case if
|
|
// all of the input is lowercase.
|
|
//
|
|
// Zsh does not appear have a dedicated option, but allows setting matching-rules
|
|
// (see https://superuser.com/a/1092328).
|
|
var allLinuxCapabilities = sync.OnceValue(func() []string {
|
|
caps := capability.ListKnown()
|
|
out := make([]string, 0, len(caps)+1)
|
|
out = append(out, allCaps)
|
|
for _, c := range caps {
|
|
out = append(out, "CAP_"+strings.ToUpper(c.String()))
|
|
}
|
|
return out
|
|
})
|
|
|
|
// restartPolicies is a list of all valid restart-policies..
|
|
//
|
|
// TODO(thaJeztah): add descriptions, and enable descriptions for our completion scripts (cobra.CompletionOptions.DisableDescriptions is currently set to "true")
|
|
var restartPolicies = []string{
|
|
string(container.RestartPolicyDisabled),
|
|
string(container.RestartPolicyAlways),
|
|
string(container.RestartPolicyOnFailure),
|
|
string(container.RestartPolicyUnlessStopped),
|
|
}
|
|
|
|
// addCompletions adds the completions that `run` and `create` have in common.
|
|
func addCompletions(cmd *cobra.Command, dockerCli command.Cli) {
|
|
_ = cmd.RegisterFlagCompletionFunc("add-host", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("annotation", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("attach", completion.FromList("stderr", "stdin", "stdout"))
|
|
_ = cmd.RegisterFlagCompletionFunc("blkio-weight", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("blkio-weight-device", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cap-add", completeLinuxCapabilityNames)
|
|
_ = cmd.RegisterFlagCompletionFunc("cap-drop", completeLinuxCapabilityNames)
|
|
_ = cmd.RegisterFlagCompletionFunc("cgroup-parent", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cgroupns", completion.FromList("host", "private"))
|
|
_ = cmd.RegisterFlagCompletionFunc("cpu-period", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpu-quota", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpu-rt-period", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpu-rt-runtime", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpu-shares", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpus", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpuset-cpus", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("cpuset-mems", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("device-cgroup-rule", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("device-read-bps", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("device-read-iops", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("device-write-bps", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("device-write-iops", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("dns", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("dns-option", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("dns-search", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("domainname", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("entrypoint", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("expose", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("env", completion.EnvVarNames)
|
|
_ = cmd.RegisterFlagCompletionFunc("env-file", completion.FileNames)
|
|
_ = cmd.RegisterFlagCompletionFunc("gpus", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("group-add", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("health-cmd", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("health-interval", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("health-retries", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("health-start-interval", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("health-start-period", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("health-timeout", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("hostname", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("ip", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("ip6", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("ipc", completeIpc(dockerCli))
|
|
_ = cmd.RegisterFlagCompletionFunc("isolation", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("kernel-memory", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("label", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("link", completeLink(dockerCli))
|
|
_ = cmd.RegisterFlagCompletionFunc("link-local-ip", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("log-driver", completion.NoComplete) // TODO complete drivers
|
|
_ = cmd.RegisterFlagCompletionFunc("log-opt", completion.NoComplete) // TODO complete driver options
|
|
_ = cmd.RegisterFlagCompletionFunc("mac-address", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("memory", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("memory-reservation", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("memory-swap", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("memory-swappiness", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("mount", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("name", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("network", completion.NetworkNames(dockerCli))
|
|
_ = cmd.RegisterFlagCompletionFunc("network-alias", completion.NoComplete)
|
|
_ = cmd.RegisterFlagCompletionFunc("platform", completion.Platforms)
|
|
_ = cmd.RegisterFlagCompletionFunc("pull", completion.FromList(PullImageAlways, PullImageMissing, PullImageNever))
|
|
_ = cmd.RegisterFlagCompletionFunc("restart", completeRestartPolicies)
|
|
_ = cmd.RegisterFlagCompletionFunc("stop-signal", completeSignals)
|
|
_ = cmd.RegisterFlagCompletionFunc("volumes-from", completion.ContainerNames(dockerCli, true))
|
|
}
|
|
|
|
// completeIpc implements shell completion for the `--ipc` option of `run` and `create`.
|
|
// The completion is partly composite.
|
|
func completeIpc(cli command.Cli) func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
|
return func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
|
if len(toComplete) > 0 && strings.HasPrefix("container", toComplete) { //nolint:gocritic
|
|
return []string{"container:"}, cobra.ShellCompDirectiveNoSpace
|
|
}
|
|
if strings.HasPrefix(toComplete, "container:") {
|
|
names, _ := completion.ContainerNames(cli, true)(cmd, args, toComplete)
|
|
return prefixWith("container:", names), cobra.ShellCompDirectiveNoFileComp
|
|
}
|
|
return []string{"container:", "host", "none", "private", "shareable"}, cobra.ShellCompDirectiveNoFileComp
|
|
}
|
|
}
|
|
|
|
// completeLink implements shell completion for the `--link` option of `run` and `create`.
|
|
func completeLink(cli command.Cli) func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
|
return func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
|
return postfixWith(":", containerNames(cli, cmd, args, toComplete)), cobra.ShellCompDirectiveNoSpace
|
|
}
|
|
}
|
|
|
|
// containerNames contacts the API to get names and optionally IDs of containers.
|
|
// In case of an error, an empty list is returned.
|
|
func containerNames(dockerCLI completion.APIClientProvider, cmd *cobra.Command, args []string, toComplete string) []string {
|
|
names, _ := completion.ContainerNames(dockerCLI, true)(cmd, args, toComplete)
|
|
if names == nil {
|
|
return []string{}
|
|
}
|
|
return names
|
|
}
|
|
|
|
// prefixWith prefixes every element in the slice with the given prefix.
|
|
func prefixWith(prefix string, values []string) []string {
|
|
result := make([]string, len(values))
|
|
for i, v := range values {
|
|
result[i] = prefix + v
|
|
}
|
|
return result
|
|
}
|
|
|
|
// postfixWith appends postfix to every element in the slice.
|
|
func postfixWith(postfix string, values []string) []string {
|
|
result := make([]string, len(values))
|
|
for i, v := range values {
|
|
result[i] = v + postfix
|
|
}
|
|
return result
|
|
}
|
|
|
|
func completeLinuxCapabilityNames(cmd *cobra.Command, args []string, toComplete string) (names []string, _ cobra.ShellCompDirective) {
|
|
return completion.FromList(allLinuxCapabilities()...)(cmd, args, toComplete)
|
|
}
|
|
|
|
func completeRestartPolicies(cmd *cobra.Command, args []string, toComplete string) (names []string, _ cobra.ShellCompDirective) {
|
|
return completion.FromList(restartPolicies...)(cmd, args, toComplete)
|
|
}
|
|
|
|
func completeSignals(cmd *cobra.Command, args []string, toComplete string) (names []string, _ cobra.ShellCompDirective) {
|
|
// TODO(thaJeztah): do we want to provide the full list here, or a subset?
|
|
signalNames := make([]string, 0, len(signal.SignalMap))
|
|
for k := range signal.SignalMap {
|
|
signalNames = append(signalNames, k)
|
|
}
|
|
return completion.FromList(signalNames...)(cmd, args, toComplete)
|
|
}
|