2016-05-16 11:50:55 -04:00
|
|
|
// +build experimental
|
|
|
|
|
|
|
|
package plugin
|
|
|
|
|
|
|
|
import (
|
2016-08-10 19:48:17 -04:00
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
2016-05-16 11:50:55 -04:00
|
|
|
"fmt"
|
2016-08-10 19:48:17 -04:00
|
|
|
"io/ioutil"
|
2016-05-16 11:50:55 -04:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
|
|
"github.com/docker/docker/pkg/archive"
|
|
|
|
"github.com/docker/docker/pkg/stringid"
|
|
|
|
"github.com/docker/docker/plugin/distribution"
|
|
|
|
"github.com/docker/docker/reference"
|
|
|
|
"github.com/docker/engine-api/types"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Disable deactivates a plugin, which implies that they cannot be used by containers.
|
|
|
|
func (pm *Manager) Disable(name string) error {
|
|
|
|
p, err := pm.get(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-07-18 11:02:12 -04:00
|
|
|
if err := pm.disable(p); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
pm.pluginEventLogger(p.PluginObj.ID, name, "disable")
|
|
|
|
return nil
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Enable activates a plugin, which implies that they are ready to be used by containers.
|
|
|
|
func (pm *Manager) Enable(name string) error {
|
|
|
|
p, err := pm.get(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-08-09 14:49:28 -04:00
|
|
|
if err := pm.enable(p, false); err != nil {
|
2016-07-18 11:02:12 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
pm.pluginEventLogger(p.PluginObj.ID, name, "enable")
|
|
|
|
return nil
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Inspect examines a plugin manifest
|
|
|
|
func (pm *Manager) Inspect(name string) (tp types.Plugin, err error) {
|
|
|
|
p, err := pm.get(name)
|
|
|
|
if err != nil {
|
|
|
|
return tp, err
|
|
|
|
}
|
2016-07-18 11:02:12 -04:00
|
|
|
return p.PluginObj, nil
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
|
2016-07-18 11:02:12 -04:00
|
|
|
// Pull pulls a plugin and computes the privileges required to install it.
|
2016-05-16 11:50:55 -04:00
|
|
|
func (pm *Manager) Pull(name string, metaHeader http.Header, authConfig *types.AuthConfig) (types.PluginPrivileges, error) {
|
|
|
|
ref, err := reference.ParseNamed(name)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Debugf("error in reference.ParseNamed: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
name = ref.String()
|
|
|
|
|
|
|
|
if p, _ := pm.get(name); p != nil {
|
|
|
|
logrus.Debugf("plugin already exists")
|
|
|
|
return nil, fmt.Errorf("%s exists", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
pluginID := stringid.GenerateNonCryptoID()
|
|
|
|
|
|
|
|
if err := os.MkdirAll(filepath.Join(pm.libRoot, pluginID), 0755); err != nil {
|
|
|
|
logrus.Debugf("error in MkdirAll: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
pd, err := distribution.Pull(name, pm.registryService, metaHeader, authConfig)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Debugf("error in distribution.Pull(): %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := distribution.WritePullData(pd, filepath.Join(pm.libRoot, pluginID), true); err != nil {
|
|
|
|
logrus.Debugf("error in distribution.WritePullData(): %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
p := pm.newPlugin(ref, pluginID)
|
|
|
|
if err := pm.initPlugin(p); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
pm.Lock()
|
|
|
|
pm.plugins[pluginID] = p
|
|
|
|
pm.nameToID[name] = pluginID
|
|
|
|
pm.save()
|
|
|
|
pm.Unlock()
|
|
|
|
|
2016-07-18 11:02:12 -04:00
|
|
|
pm.pluginEventLogger(pluginID, name, "pull")
|
|
|
|
return computePrivileges(&p.PluginObj.Manifest), nil
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// List displays the list of plugins and associated metadata.
|
|
|
|
func (pm *Manager) List() ([]types.Plugin, error) {
|
|
|
|
out := make([]types.Plugin, 0, len(pm.plugins))
|
|
|
|
for _, p := range pm.plugins {
|
2016-07-18 11:02:12 -04:00
|
|
|
out = append(out, p.PluginObj)
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
return out, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Push pushes a plugin to the store.
|
|
|
|
func (pm *Manager) Push(name string, metaHeader http.Header, authConfig *types.AuthConfig) error {
|
|
|
|
p, err := pm.get(name)
|
2016-06-27 11:41:53 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-07-18 11:02:12 -04:00
|
|
|
dest := filepath.Join(pm.libRoot, p.PluginObj.ID)
|
2016-08-10 19:48:17 -04:00
|
|
|
config, err := ioutil.ReadFile(filepath.Join(dest, "manifest.json"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var dummy types.Plugin
|
|
|
|
err = json.Unmarshal(config, &dummy)
|
2016-05-16 11:50:55 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-06-27 11:41:53 -04:00
|
|
|
|
2016-05-16 11:50:55 -04:00
|
|
|
rootfs, err := archive.Tar(filepath.Join(dest, "rootfs"), archive.Gzip)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-06-24 23:57:21 -04:00
|
|
|
defer rootfs.Close()
|
|
|
|
|
2016-08-10 19:48:17 -04:00
|
|
|
_, err = distribution.Push(name, pm.registryService, metaHeader, authConfig, ioutil.NopCloser(bytes.NewReader(config)), rootfs)
|
2016-05-16 11:50:55 -04:00
|
|
|
// XXX: Ignore returning digest for now.
|
|
|
|
// Since digest needs to be written to the ProgressWriter.
|
2016-06-22 13:40:32 -04:00
|
|
|
return err
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Remove deletes plugin's root directory.
|
2016-07-22 11:24:54 -04:00
|
|
|
func (pm *Manager) Remove(name string, config *types.PluginRmConfig) error {
|
2016-05-16 11:50:55 -04:00
|
|
|
p, err := pm.get(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-07-22 11:24:54 -04:00
|
|
|
if err := pm.remove(p, config.ForceRemove); err != nil {
|
2016-07-18 11:02:12 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
pm.pluginEventLogger(p.PluginObj.ID, name, "remove")
|
|
|
|
return nil
|
2016-05-16 11:50:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set sets plugin args
|
|
|
|
func (pm *Manager) Set(name string, args []string) error {
|
|
|
|
p, err := pm.get(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return pm.set(p, args)
|
|
|
|
}
|