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
)
2014-06-02 21:26:41 -04:00
type FsMagic uint64
const (
FsMagicBtrfs = FsMagic ( 0x9123683E )
FsMagicAufs = FsMagic ( 0x61756673 )
)
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.
2013-12-05 16:18:02 -05:00
Put ( id string )
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-04 18:22:34 -05:00
var (
2013-11-20 14:39:15 -05:00
DefaultDriver string
2013-11-04 18:22:34 -05:00
// All registred drivers
drivers map [ string ] InitFunc
// Slice of drivers that should be used in an order
priority = [ ] string {
"aufs" ,
2014-03-24 10:15:04 -04:00
"btrfs" ,
2013-11-04 18:22:34 -05:00
"devicemapper" ,
2013-11-25 13:28:17 -05:00
"vfs" ,
2014-08-19 05:23:55 -04:00
// experimental, has to be enabled manually for now
2014-12-03 07:57:23 -05:00
"overlay" ,
2013-11-04 18:22:34 -05:00
}
2014-03-27 12:41:06 -04:00
2014-05-29 15:55:59 -04: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 18:22:34 -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-05 13:34:28 -05:00
if prior != name {
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 , "," ) )
}
}