install.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. // +build experimental
  2. package plugin
  3. import (
  4. "bufio"
  5. "fmt"
  6. "strings"
  7. "github.com/docker/docker/api/client"
  8. "github.com/docker/docker/cli"
  9. "github.com/docker/docker/reference"
  10. "github.com/docker/docker/registry"
  11. "github.com/docker/engine-api/types"
  12. "github.com/spf13/cobra"
  13. "golang.org/x/net/context"
  14. )
  15. type pluginOptions struct {
  16. name string
  17. grantPerms bool
  18. disable bool
  19. }
  20. func newInstallCommand(dockerCli *client.DockerCli) *cobra.Command {
  21. var options pluginOptions
  22. cmd := &cobra.Command{
  23. Use: "install [OPTIONS] PLUGIN",
  24. Short: "Install a plugin",
  25. Args: cli.ExactArgs(1), // TODO: allow for set args
  26. RunE: func(cmd *cobra.Command, args []string) error {
  27. options.name = args[0]
  28. return runInstall(dockerCli, options)
  29. },
  30. }
  31. flags := cmd.Flags()
  32. flags.BoolVar(&options.grantPerms, "grant-all-permissions", false, "grant all permissions necessary to run the plugin")
  33. flags.BoolVar(&options.disable, "disable", false, "do not enable the plugin on install")
  34. return cmd
  35. }
  36. func runInstall(dockerCli *client.DockerCli, opts pluginOptions) error {
  37. named, err := reference.ParseNamed(opts.name) // FIXME: validate
  38. if err != nil {
  39. return err
  40. }
  41. if reference.IsNameOnly(named) {
  42. named = reference.WithDefaultTag(named)
  43. }
  44. ref, ok := named.(reference.NamedTagged)
  45. if !ok {
  46. return fmt.Errorf("invalid name: %s", named.String())
  47. }
  48. ctx := context.Background()
  49. repoInfo, err := registry.ParseRepositoryInfo(named)
  50. if err != nil {
  51. return err
  52. }
  53. authConfig := dockerCli.ResolveAuthConfig(ctx, repoInfo.Index)
  54. encodedAuth, err := client.EncodeAuthToBase64(authConfig)
  55. if err != nil {
  56. return err
  57. }
  58. registryAuthFunc := dockerCli.RegistryAuthenticationPrivilegedFunc(repoInfo.Index, "plugin install")
  59. options := types.PluginInstallOptions{
  60. RegistryAuth: encodedAuth,
  61. Disabled: opts.disable,
  62. AcceptAllPermissions: opts.grantPerms,
  63. AcceptPermissionsFunc: acceptPrivileges(dockerCli, opts.name),
  64. // TODO: Rename PrivilegeFunc, it has nothing to do with privileges
  65. PrivilegeFunc: registryAuthFunc,
  66. }
  67. if err := dockerCli.Client().PluginInstall(ctx, ref.String(), options); err != nil {
  68. return err
  69. }
  70. fmt.Fprintln(dockerCli.Out(), opts.name)
  71. return nil
  72. }
  73. func acceptPrivileges(dockerCli *client.DockerCli, name string) func(privileges types.PluginPrivileges) (bool, error) {
  74. return func(privileges types.PluginPrivileges) (bool, error) {
  75. fmt.Fprintf(dockerCli.Out(), "Plugin %q is requesting the following privileges:\n", name)
  76. for _, privilege := range privileges {
  77. fmt.Fprintf(dockerCli.Out(), " - %s: %v\n", privilege.Name, privilege.Value)
  78. }
  79. fmt.Fprint(dockerCli.Out(), "Do you grant the above permissions? [y/N] ")
  80. reader := bufio.NewReader(dockerCli.In())
  81. line, _, err := reader.ReadLine()
  82. if err != nil {
  83. return false, err
  84. }
  85. return strings.ToLower(string(line)) == "y", nil
  86. }
  87. }