2013-11-04 01:54:51 +00:00
|
|
|
package graphdriver
|
2013-10-31 18:07:54 -07:00
|
|
|
|
2013-11-04 15:22:34 -08:00
|
|
|
import (
|
2014-03-27 17:41:06 +01:00
|
|
|
"errors"
|
2013-11-04 15:22:34 -08:00
|
|
|
"fmt"
|
2013-11-19 03:13:22 -08:00
|
|
|
"os"
|
2013-11-08 11:36:58 -08:00
|
|
|
"path"
|
2014-06-05 12:50:53 -07:00
|
|
|
|
|
|
|
"github.com/dotcloud/docker/archive"
|
|
|
|
"github.com/dotcloud/docker/pkg/mount"
|
2013-11-04 15:22:34 -08:00
|
|
|
)
|
|
|
|
|
2014-06-02 19:26:41 -06:00
|
|
|
type FsMagic uint64
|
|
|
|
|
|
|
|
const (
|
|
|
|
FsMagicBtrfs = FsMagic(0x9123683E)
|
|
|
|
FsMagicAufs = FsMagic(0x61756673)
|
|
|
|
)
|
|
|
|
|
2014-06-05 10:34:20 +02:00
|
|
|
type InitFunc func(root string, options []string) (Driver, error)
|
2013-11-04 20:51:12 -08:00
|
|
|
|
|
|
|
type Driver interface {
|
2013-11-15 10:24:48 +01:00
|
|
|
String() string
|
|
|
|
|
2014-04-17 23:47:27 +00:00
|
|
|
Create(id, parent string) error
|
2013-11-07 20:34:01 +00:00
|
|
|
Remove(id string) error
|
2013-11-04 20:51:12 -08:00
|
|
|
|
2014-04-17 23:47:27 +00:00
|
|
|
Get(id, mountLabel string) (dir string, err error)
|
2013-12-05 22:18:02 +01:00
|
|
|
Put(id string)
|
2013-11-19 02:32:08 -08:00
|
|
|
Exists(id string) bool
|
2013-11-04 20:51:12 -08:00
|
|
|
|
2013-11-15 11:04:02 +01:00
|
|
|
Status() [][2]string
|
|
|
|
|
2013-11-11 17:17:38 -08:00
|
|
|
Cleanup() error
|
|
|
|
}
|
|
|
|
|
|
|
|
type Differ interface {
|
2013-11-11 12:09:26 -08:00
|
|
|
Diff(id string) (archive.Archive, error)
|
2013-11-08 14:54:20 -08:00
|
|
|
Changes(id string) ([]archive.Change, error)
|
2014-02-14 12:41:46 +01:00
|
|
|
ApplyDiff(id string, diff archive.ArchiveReader) error
|
2013-11-19 10:36:54 +01:00
|
|
|
DiffSize(id string) (bytes int64, err error)
|
2013-11-11 16:47:36 +01:00
|
|
|
}
|
|
|
|
|
2013-11-04 15:22:34 -08:00
|
|
|
var (
|
2013-11-20 11:39:15 -08:00
|
|
|
DefaultDriver string
|
2013-11-04 15:22:34 -08:00
|
|
|
// All registred drivers
|
|
|
|
drivers map[string]InitFunc
|
|
|
|
// Slice of drivers that should be used in an order
|
|
|
|
priority = []string{
|
|
|
|
"aufs",
|
2014-03-24 14:15:04 +00:00
|
|
|
"btrfs",
|
2013-11-04 15:22:34 -08:00
|
|
|
"devicemapper",
|
2013-11-25 10:28:17 -08:00
|
|
|
"vfs",
|
2013-11-04 15:22:34 -08:00
|
|
|
}
|
2014-03-27 17:41:06 +01:00
|
|
|
|
2014-05-29 22:55:59 +03:00
|
|
|
ErrNotSupported = errors.New("driver not supported")
|
|
|
|
ErrPrerequisites = errors.New("prerequisites for driver not satisfied (wrong filesystem?)")
|
|
|
|
ErrIncompatibleFS = fmt.Errorf("backing file system is unsupported for this graph driver")
|
2013-11-04 15:22:34 -08:00
|
|
|
)
|
|
|
|
|
2013-11-07 20:31:50 +00:00
|
|
|
func init() {
|
|
|
|
drivers = make(map[string]InitFunc)
|
|
|
|
}
|
|
|
|
|
2013-11-04 15:22:34 -08:00
|
|
|
func Register(name string, initFunc InitFunc) error {
|
|
|
|
if _, exists := drivers[name]; exists {
|
|
|
|
return fmt.Errorf("Name already registered %s", name)
|
|
|
|
}
|
|
|
|
drivers[name] = initFunc
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-06-05 10:34:20 +02:00
|
|
|
func GetDriver(name, home string, options []string) (Driver, error) {
|
2013-11-08 02:49:32 +00:00
|
|
|
if initFunc, exists := drivers[name]; exists {
|
2014-06-05 10:34:20 +02:00
|
|
|
return initFunc(path.Join(home, name), options)
|
2013-11-08 02:49:32 +00:00
|
|
|
}
|
2014-03-27 17:41:06 +01:00
|
|
|
return nil, ErrNotSupported
|
2013-11-08 02:49:32 +00:00
|
|
|
}
|
|
|
|
|
2014-06-05 10:34:20 +02:00
|
|
|
func New(root string, options []string) (driver Driver, err error) {
|
2013-11-20 11:39:15 -08:00
|
|
|
for _, name := range []string{os.Getenv("DOCKER_DRIVER"), DefaultDriver} {
|
2013-11-19 03:13:22 -08:00
|
|
|
if name != "" {
|
2014-06-05 10:34:20 +02:00
|
|
|
return GetDriver(name, root, options)
|
2013-11-19 03:13:22 -08:00
|
|
|
}
|
2013-11-08 02:49:32 +00:00
|
|
|
}
|
2013-11-19 03:13:22 -08:00
|
|
|
|
2013-11-04 15:22:34 -08:00
|
|
|
// Check for priority drivers first
|
|
|
|
for _, name := range priority {
|
2014-06-05 10:34:20 +02:00
|
|
|
driver, err = GetDriver(name, root, options)
|
2014-03-27 17:41:06 +01:00
|
|
|
if err != nil {
|
2014-05-29 22:55:59 +03:00
|
|
|
if err == ErrNotSupported || err == ErrPrerequisites || err == ErrIncompatibleFS {
|
2014-03-27 17:41:06 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
return nil, err
|
2013-11-04 15:22:34 -08:00
|
|
|
}
|
2013-11-08 02:49:32 +00:00
|
|
|
return driver, nil
|
2013-11-04 15:22:34 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check all registered drivers if no priority driver is found
|
|
|
|
for _, initFunc := range drivers {
|
2014-06-05 10:34:20 +02:00
|
|
|
if driver, err = initFunc(root, options); err != nil {
|
2014-05-29 22:55:59 +03:00
|
|
|
if err == ErrNotSupported || err == ErrPrerequisites || err == ErrIncompatibleFS {
|
2014-03-27 17:41:06 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
return nil, err
|
2013-11-04 15:22:34 -08:00
|
|
|
}
|
|
|
|
return driver, nil
|
|
|
|
}
|
2014-03-27 17:41:06 +01:00
|
|
|
return nil, fmt.Errorf("No supported storage backend found")
|
2013-11-04 15:22:34 -08:00
|
|
|
}
|
2014-06-05 12:50:53 -07:00
|
|
|
|
|
|
|
func MakePrivate(mountPoint string) error {
|
|
|
|
mounted, err := mount.Mounted(mountPoint)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !mounted {
|
|
|
|
if err := mount.Mount(mountPoint, mountPoint, "none", "bind,rw"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mount.ForceMount("", mountPoint, "none", "private")
|
|
|
|
}
|