2013-11-03 20:54:51 -05:00
package graphdriver
2013-10-31 21:07:54 -04:00
2013-11-04 18:22:34 -05:00
import (
2014-03-27 12:41:06 -04:00
"errors"
2013-11-04 18:22:34 -05:00
"fmt"
2013-11-19 06:13:22 -05:00
"os"
2013-11-08 14:36:58 -05:00
"path"
2014-11-15 12:54:21 -05:00
"strings"
2014-06-05 15:50:53 -04:00
2014-11-15 12:54:21 -05:00
log "github.com/Sirupsen/logrus"
2014-09-30 02:23:36 -04:00
"github.com/docker/docker/pkg/archive"
2013-11-04 18:22:34 -05:00
)
2015-01-08 19:22:38 -05:00
type FsMagic uint32
2014-06-02 21:26:41 -04:00
const (
2015-01-15 16:40:39 -05:00
FsMagicBtrfs = FsMagic ( 0x9123683E )
FsMagicAufs = FsMagic ( 0x61756673 )
FsMagicExtfs = FsMagic ( 0x0000EF53 )
FsMagicCramfs = FsMagic ( 0x28cd3d45 )
FsMagicRamFs = FsMagic ( 0x858458f6 )
FsMagicTmpFs = FsMagic ( 0x01021994 )
FsMagicSquashFs = FsMagic ( 0x73717368 )
FsMagicNfsFs = FsMagic ( 0x00006969 )
FsMagicReiserFs = FsMagic ( 0x52654973 )
FsMagicSmbFs = FsMagic ( 0x0000517B )
FsMagicJffs2Fs = FsMagic ( 0x000072b6 )
2015-01-23 17:00:15 -05:00
FsMagicZfs = FsMagic ( 0x2fc12fc1 )
2015-03-04 06:01:39 -05:00
FsMagicXfs = FsMagic ( 0x58465342 )
2015-01-15 16:40:39 -05:00
FsMagicUnsupported = FsMagic ( 0x00000000 )
)
var (
DefaultDriver string
// All registred drivers
drivers map [ string ] InitFunc
// Slice of drivers that should be used in an order
priority = [ ] string {
"aufs" ,
"btrfs" ,
"devicemapper" ,
"vfs" ,
// experimental, has to be enabled manually for now
"overlay" ,
}
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" )
FsNames = map [ FsMagic ] string {
FsMagicAufs : "aufs" ,
FsMagicBtrfs : "btrfs" ,
FsMagicExtfs : "extfs" ,
FsMagicCramfs : "cramfs" ,
FsMagicRamFs : "ramfs" ,
FsMagicTmpFs : "tmpfs" ,
FsMagicSquashFs : "squashfs" ,
FsMagicNfsFs : "nfs" ,
FsMagicReiserFs : "reiserfs" ,
FsMagicSmbFs : "smb" ,
FsMagicJffs2Fs : "jffs2" ,
2015-01-23 17:00:15 -05:00
FsMagicZfs : "zfs" ,
2015-03-04 06:01:39 -05:00
FsMagicXfs : "xfs" ,
2015-01-15 16:40:39 -05:00
FsMagicUnsupported : "unsupported" ,
}
2014-06-02 21:26:41 -04:00
)
2014-06-05 04:34:20 -04:00
type InitFunc func ( root string , options [ ] string ) ( Driver , error )
2013-11-04 23:51:12 -05:00
2014-09-16 15:13:50 -04:00
// ProtoDriver defines the basic capabilities of a driver.
// This interface exists solely to be a minimum set of methods
// for client code which choose not to implement the entire Driver
// interface and use the NaiveDiffDriver wrapper constructor.
//
// Use of ProtoDriver directly by client code is not recommended.
type ProtoDriver interface {
2014-09-10 23:30:52 -04:00
// String returns a string representation of this driver.
2013-11-15 04:24:48 -05:00
String ( ) string
2014-09-10 23:30:52 -04:00
// Create creates a new, empty, filesystem layer with the
// specified id and parent. Parent may be "".
2014-04-17 19:47:27 -04:00
Create ( id , parent string ) error
2014-09-16 15:13:50 -04:00
// Remove attempts to remove the filesystem layer with this id.
2013-11-07 15:34:01 -05:00
Remove ( id string ) error
2014-09-10 23:30:52 -04:00
// Get returns the mountpoint for the layered filesystem referred
// to by this id. You can optionally specify a mountLabel or "".
// Returns the absolute path to the mounted layered filesystem.
2014-04-17 19:47:27 -04:00
Get ( id , mountLabel string ) ( dir string , err error )
2014-09-10 23:30:52 -04:00
// Put releases the system resources for the specified id,
// e.g, unmounting layered filesystem.
2015-01-09 17:14:52 -05:00
Put ( id string ) error
2014-09-10 23:30:52 -04:00
// Exists returns whether a filesystem layer with the specified
// ID exists on this driver.
2013-11-19 05:32:08 -05:00
Exists ( id string ) bool
2014-09-10 23:30:52 -04:00
// Status returns a set of key-value pairs which give low
// level diagnostic status about this driver.
2013-11-15 05:04:02 -05:00
Status ( ) [ ] [ 2 ] string
2014-09-10 23:30:52 -04:00
// Cleanup performs necessary tasks to release resources
// held by the driver, e.g., unmounting all layered filesystems
// known to this driver.
2013-11-11 20:17:38 -05:00
Cleanup ( ) error
}
2014-09-16 15:13:50 -04:00
// Driver is the interface for layered/snapshot file system drivers.
2014-09-10 23:30:52 -04:00
type Driver interface {
2014-09-16 15:13:50 -04:00
ProtoDriver
2014-09-10 23:30:52 -04:00
// Diff produces an archive of the changes between the specified
// layer and its parent layer which may be "".
Diff ( id , parent string ) ( archive . Archive , error )
// Changes produces a list of changes between the specified layer
// and its parent layer. If parent is "", then all changes will be ADD changes.
Changes ( id , parent string ) ( [ ] archive . Change , error )
// ApplyDiff extracts the changeset from the given diff into the
// layer with the specified id and parent, returning the size of the
// new layer in bytes.
2014-12-17 21:26:03 -05:00
ApplyDiff ( id , parent string , diff archive . ArchiveReader ) ( size int64 , err error )
2014-09-10 23:30:52 -04:00
// DiffSize calculates the changes between the specified id
// and its parent and returns the size in bytes of the changes
// relative to its base filesystem directory.
2014-12-17 21:26:03 -05:00
DiffSize ( id , parent string ) ( size int64 , err error )
2013-11-11 10:47:36 -05:00
}
2013-11-07 15:31:50 -05:00
func init ( ) {
drivers = make ( map [ string ] InitFunc )
}
2013-11-04 18:22:34 -05: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 04:34:20 -04:00
func GetDriver ( name , home string , options [ ] string ) ( Driver , error ) {
2013-11-07 21:49:32 -05:00
if initFunc , exists := drivers [ name ] ; exists {
2014-06-05 04:34:20 -04:00
return initFunc ( path . Join ( home , name ) , options )
2013-11-07 21:49:32 -05:00
}
2014-03-27 12:41:06 -04:00
return nil , ErrNotSupported
2013-11-07 21:49:32 -05:00
}
2014-06-05 04:34:20 -04:00
func New ( root string , options [ ] string ) ( driver Driver , err error ) {
2013-11-20 14:39:15 -05:00
for _ , name := range [ ] string { os . Getenv ( "DOCKER_DRIVER" ) , DefaultDriver } {
2013-11-19 06:13:22 -05:00
if name != "" {
2014-06-05 04:34:20 -04:00
return GetDriver ( name , root , options )
2013-11-19 06:13:22 -05:00
}
2013-11-07 21:49:32 -05:00
}
2013-11-19 06:13:22 -05:00
2013-11-04 18:22:34 -05:00
// Check for priority drivers first
for _ , name := range priority {
2014-06-05 04:34:20 -04:00
driver , err = GetDriver ( name , root , options )
2014-03-27 12:41:06 -04:00
if err != nil {
2014-05-29 15:55:59 -04:00
if err == ErrNotSupported || err == ErrPrerequisites || err == ErrIncompatibleFS {
2014-03-27 12:41:06 -04:00
continue
}
return nil , err
2013-11-04 18:22:34 -05:00
}
2014-11-15 12:54:21 -05:00
checkPriorDriver ( name , root )
2013-11-07 21:49:32 -05:00
return driver , nil
2013-11-04 18:22:34 -05:00
}
// Check all registered drivers if no priority driver is found
2014-11-15 12:54:21 -05:00
for name , initFunc := range drivers {
2014-06-05 04:34:20 -04:00
if driver , err = initFunc ( root , options ) ; err != nil {
2014-05-29 15:55:59 -04:00
if err == ErrNotSupported || err == ErrPrerequisites || err == ErrIncompatibleFS {
2014-03-27 12:41:06 -04:00
continue
}
return nil , err
2013-11-04 18:22:34 -05:00
}
2014-11-15 12:54:21 -05:00
checkPriorDriver ( name , root )
2013-11-04 18:22:34 -05:00
return driver , nil
}
2014-03-27 12:41:06 -04:00
return nil , fmt . Errorf ( "No supported storage backend found" )
2013-11-04 18:22:34 -05:00
}
2014-11-15 12:54:21 -05:00
2015-01-05 13:34:28 -05:00
func checkPriorDriver ( name , root string ) {
priorDrivers := [ ] string { }
2014-11-15 12:54:21 -05:00
for prior := range drivers {
2015-01-14 20:28:50 -05:00
if prior != name && prior != "vfs" {
2015-01-05 13:34:28 -05:00
if _ , err := os . Stat ( path . Join ( root , prior ) ) ; err == nil {
priorDrivers = append ( priorDrivers , prior )
}
2014-11-15 12:54:21 -05:00
}
}
if len ( priorDrivers ) > 0 {
log . Warnf ( "graphdriver %s selected. Warning: your graphdriver directory %s already contains data managed by other graphdrivers: %s" , name , root , strings . Join ( priorDrivers , "," ) )
}
}