mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
39bcaee47b
* additional * ambiguous * anonymous * anything * application * because * before * building * capabilities * circumstances * commit * committer * compresses * concatenated * config * container * container's * current * definition * delimiter * disassociates * discovery * distributed * doesnotexist * downloads * duplicates * either * enhancing * enumerate * escapable * exactly * expect * expectations * expected * explicitly * false * filesystem * following * forbidden * git with * healthcheck * ignore * independent * inheritance * investigating * irrelevant * it * logging * looking * membership * mimic * minimum * modify * mountpoint * multiline * notifier * outputting * outside * overridden * override * parsable * plugins * precedence * propagation * provided * provides * registries * repositories * returning * settings * should * signals * someone * something * specifically * successfully * synchronize * they've * thinking * uninitialized * unintentionally * unmarshaling * unnamed * unreferenced * verify Signed-off-by: Josh Soref <jsoref@gmail.com> Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
184 lines
4.1 KiB
Go
184 lines
4.1 KiB
Go
package volumedrivers
|
|
|
|
import (
|
|
"errors"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
"github.com/docker/docker/volume"
|
|
)
|
|
|
|
var (
|
|
errNoSuchVolume = errors.New("no such volume")
|
|
)
|
|
|
|
type volumeDriverAdapter struct {
|
|
name string
|
|
baseHostPath string
|
|
capabilities *volume.Capability
|
|
proxy *volumeDriverProxy
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) Name() string {
|
|
return a.name
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) Create(name string, opts map[string]string) (volume.Volume, error) {
|
|
if err := a.proxy.Create(name, opts); err != nil {
|
|
return nil, err
|
|
}
|
|
return &volumeAdapter{
|
|
proxy: a.proxy,
|
|
name: name,
|
|
driverName: a.name,
|
|
baseHostPath: a.baseHostPath,
|
|
}, nil
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) Remove(v volume.Volume) error {
|
|
return a.proxy.Remove(v.Name())
|
|
}
|
|
|
|
func hostPath(baseHostPath, path string) string {
|
|
if baseHostPath != "" {
|
|
path = filepath.Join(baseHostPath, path)
|
|
}
|
|
return path
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) List() ([]volume.Volume, error) {
|
|
ls, err := a.proxy.List()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var out []volume.Volume
|
|
for _, vp := range ls {
|
|
out = append(out, &volumeAdapter{
|
|
proxy: a.proxy,
|
|
name: vp.Name,
|
|
baseHostPath: a.baseHostPath,
|
|
driverName: a.name,
|
|
eMount: hostPath(a.baseHostPath, vp.Mountpoint),
|
|
})
|
|
}
|
|
return out, nil
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) Get(name string) (volume.Volume, error) {
|
|
v, err := a.proxy.Get(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// plugin may have returned no volume and no error
|
|
if v == nil {
|
|
return nil, errNoSuchVolume
|
|
}
|
|
|
|
return &volumeAdapter{
|
|
proxy: a.proxy,
|
|
name: v.Name,
|
|
driverName: a.Name(),
|
|
eMount: v.Mountpoint,
|
|
createdAt: v.CreatedAt,
|
|
status: v.Status,
|
|
baseHostPath: a.baseHostPath,
|
|
}, nil
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) Scope() string {
|
|
cap := a.getCapabilities()
|
|
return cap.Scope
|
|
}
|
|
|
|
func (a *volumeDriverAdapter) getCapabilities() volume.Capability {
|
|
if a.capabilities != nil {
|
|
return *a.capabilities
|
|
}
|
|
cap, err := a.proxy.Capabilities()
|
|
if err != nil {
|
|
// `GetCapabilities` is a not a required endpoint.
|
|
// On error assume it's a local-only driver
|
|
logrus.Warnf("Volume driver %s returned an error while trying to query its capabilities, using default capabilities: %v", a.name, err)
|
|
return volume.Capability{Scope: volume.LocalScope}
|
|
}
|
|
|
|
// don't spam the warn log below just because the plugin didn't provide a scope
|
|
if len(cap.Scope) == 0 {
|
|
cap.Scope = volume.LocalScope
|
|
}
|
|
|
|
cap.Scope = strings.ToLower(cap.Scope)
|
|
if cap.Scope != volume.LocalScope && cap.Scope != volume.GlobalScope {
|
|
logrus.Warnf("Volume driver %q returned an invalid scope: %q", a.Name(), cap.Scope)
|
|
cap.Scope = volume.LocalScope
|
|
}
|
|
|
|
a.capabilities = &cap
|
|
return cap
|
|
}
|
|
|
|
type volumeAdapter struct {
|
|
proxy *volumeDriverProxy
|
|
name string
|
|
baseHostPath string
|
|
driverName string
|
|
eMount string // ephemeral host volume path
|
|
createdAt time.Time // time the directory was created
|
|
status map[string]interface{}
|
|
}
|
|
|
|
type proxyVolume struct {
|
|
Name string
|
|
Mountpoint string
|
|
CreatedAt time.Time
|
|
Status map[string]interface{}
|
|
}
|
|
|
|
func (a *volumeAdapter) Name() string {
|
|
return a.name
|
|
}
|
|
|
|
func (a *volumeAdapter) DriverName() string {
|
|
return a.driverName
|
|
}
|
|
|
|
func (a *volumeAdapter) Path() string {
|
|
if len(a.eMount) == 0 {
|
|
mountpoint, _ := a.proxy.Path(a.name)
|
|
a.eMount = hostPath(a.baseHostPath, mountpoint)
|
|
}
|
|
return a.eMount
|
|
}
|
|
|
|
func (a *volumeAdapter) CachedPath() string {
|
|
return a.eMount
|
|
}
|
|
|
|
func (a *volumeAdapter) Mount(id string) (string, error) {
|
|
mountpoint, err := a.proxy.Mount(a.name, id)
|
|
a.eMount = hostPath(a.baseHostPath, mountpoint)
|
|
return a.eMount, err
|
|
}
|
|
|
|
func (a *volumeAdapter) Unmount(id string) error {
|
|
err := a.proxy.Unmount(a.name, id)
|
|
if err == nil {
|
|
a.eMount = ""
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (a *volumeAdapter) CreatedAt() (time.Time, error) {
|
|
return a.createdAt, nil
|
|
}
|
|
func (a *volumeAdapter) Status() map[string]interface{} {
|
|
out := make(map[string]interface{}, len(a.status))
|
|
for k, v := range a.status {
|
|
out[k] = v
|
|
}
|
|
return out
|
|
}
|