plugin_install.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. package client // import "github.com/docker/docker/client"
  2. import (
  3. "context"
  4. "encoding/json"
  5. "io"
  6. "net/http"
  7. "net/url"
  8. "github.com/distribution/reference"
  9. "github.com/docker/docker/api/types"
  10. "github.com/docker/docker/api/types/registry"
  11. "github.com/docker/docker/errdefs"
  12. "github.com/pkg/errors"
  13. )
  14. // PluginInstall installs a plugin
  15. func (cli *Client) PluginInstall(ctx context.Context, name string, options types.PluginInstallOptions) (rc io.ReadCloser, err error) {
  16. query := url.Values{}
  17. if _, err := reference.ParseNormalizedNamed(options.RemoteRef); err != nil {
  18. return nil, errors.Wrap(err, "invalid remote reference")
  19. }
  20. query.Set("remote", options.RemoteRef)
  21. privileges, err := cli.checkPluginPermissions(ctx, query, options)
  22. if err != nil {
  23. return nil, err
  24. }
  25. // set name for plugin pull, if empty should default to remote reference
  26. query.Set("name", name)
  27. resp, err := cli.tryPluginPull(ctx, query, privileges, options.RegistryAuth)
  28. if err != nil {
  29. return nil, err
  30. }
  31. name = resp.header.Get("Docker-Plugin-Name")
  32. pr, pw := io.Pipe()
  33. go func() { // todo: the client should probably be designed more around the actual api
  34. _, err := io.Copy(pw, resp.body)
  35. if err != nil {
  36. pw.CloseWithError(err)
  37. return
  38. }
  39. defer func() {
  40. if err != nil {
  41. delResp, _ := cli.delete(ctx, "/plugins/"+name, nil, nil)
  42. ensureReaderClosed(delResp)
  43. }
  44. }()
  45. if len(options.Args) > 0 {
  46. if err := cli.PluginSet(ctx, name, options.Args); err != nil {
  47. pw.CloseWithError(err)
  48. return
  49. }
  50. }
  51. if options.Disabled {
  52. pw.Close()
  53. return
  54. }
  55. enableErr := cli.PluginEnable(ctx, name, types.PluginEnableOptions{Timeout: 0})
  56. pw.CloseWithError(enableErr)
  57. }()
  58. return pr, nil
  59. }
  60. func (cli *Client) tryPluginPrivileges(ctx context.Context, query url.Values, registryAuth string) (serverResponse, error) {
  61. return cli.get(ctx, "/plugins/privileges", query, http.Header{
  62. registry.AuthHeader: {registryAuth},
  63. })
  64. }
  65. func (cli *Client) tryPluginPull(ctx context.Context, query url.Values, privileges types.PluginPrivileges, registryAuth string) (serverResponse, error) {
  66. return cli.post(ctx, "/plugins/pull", query, privileges, http.Header{
  67. registry.AuthHeader: {registryAuth},
  68. })
  69. }
  70. func (cli *Client) checkPluginPermissions(ctx context.Context, query url.Values, options types.PluginInstallOptions) (types.PluginPrivileges, error) {
  71. resp, err := cli.tryPluginPrivileges(ctx, query, options.RegistryAuth)
  72. if errdefs.IsUnauthorized(err) && options.PrivilegeFunc != nil {
  73. // todo: do inspect before to check existing name before checking privileges
  74. newAuthHeader, privilegeErr := options.PrivilegeFunc()
  75. if privilegeErr != nil {
  76. ensureReaderClosed(resp)
  77. return nil, privilegeErr
  78. }
  79. options.RegistryAuth = newAuthHeader
  80. resp, err = cli.tryPluginPrivileges(ctx, query, options.RegistryAuth)
  81. }
  82. if err != nil {
  83. ensureReaderClosed(resp)
  84. return nil, err
  85. }
  86. var privileges types.PluginPrivileges
  87. if err := json.NewDecoder(resp.body).Decode(&privileges); err != nil {
  88. ensureReaderClosed(resp)
  89. return nil, err
  90. }
  91. ensureReaderClosed(resp)
  92. if !options.AcceptAllPermissions && options.AcceptPermissionsFunc != nil && len(privileges) > 0 {
  93. accept, err := options.AcceptPermissionsFunc(privileges)
  94. if err != nil {
  95. return nil, err
  96. }
  97. if !accept {
  98. return nil, errors.Errorf("permission denied while installing plugin %s", options.RemoteRef)
  99. }
  100. }
  101. return privileges, nil
  102. }