1
0
Fork 0
mirror of https://github.com/moby/moby.git synced 2022-11-09 12:21:53 -05:00

Add container migration to aufs driver

This commit is contained in:
Michael Crosby 2013-11-18 17:20:03 -08:00
parent f88b760809
commit 29f07f8544
3 changed files with 85 additions and 28 deletions

View file

@ -107,6 +107,8 @@ func (AufsDriver) Status() [][2]string {
return nil return nil
} }
// Exists returns true if the given id is registered with
// this driver
func (a AufsDriver) Exists(id string) bool { func (a AufsDriver) Exists(id string) bool {
if _, err := os.Lstat(path.Join(a.rootPath(), "diff", id)); err != nil { if _, err := os.Lstat(path.Join(a.rootPath(), "diff", id)); err != nil {
return false return false

View file

@ -6,17 +6,14 @@ import (
"io/ioutil" "io/ioutil"
"os" "os"
"path" "path"
"time"
) )
type imageMetadata struct { type metadata struct {
ID string `json:"id"` ID string `json:"id"`
ParentID string `json:"parent,omitempty"` ParentID string `json:"parent,omitempty"`
Created time.Time `json:"created"` Image string `json:"Image,omitempty"`
DockerVersion string `json:"docker_version,omitempty"`
Architecture string `json:"architecture,omitempty"`
parent *imageMetadata parent *metadata
} }
func pathExists(pth string) bool { func pathExists(pth string) bool {
@ -27,48 +24,106 @@ func pathExists(pth string) bool {
} }
// Migrate existing images and containers from docker < 0.7.x // Migrate existing images and containers from docker < 0.7.x
func (a *AufsDriver) Migrate(pth string) error { //
// The format pre 0.7 is for docker to store the metadata and filesystem
// content in the same directory. For the migration to work we need to move Image layer
// data from /var/lib/docker/graph/<id>/layers to the diff of the registered id.
//
// Next we need to migrate the container's rw layer to diff of the driver. After the
// contents are migrated we need to register the image and container ids with the
// driver.
//
// For the migration we try to move the folder containing the layer files, if that
// fails because the data is currently mounted we will fallback to creating a
// symlink.
func (a *AufsDriver) Migrate(pth string, setupInit func(p string) error) error {
if pathExists(path.Join(pth, "graph")) {
if err := a.migrateImages(path.Join(pth, "graph")); err != nil {
return err
}
return a.migrateContainers(path.Join(pth, "containers"), setupInit)
}
return nil
}
func (a *AufsDriver) migrateContainers(pth string, setupInit func(p string) error) error {
fis, err := ioutil.ReadDir(pth)
if err != nil {
return err
}
for _, fi := range fis {
if id := fi.Name(); fi.IsDir() && pathExists(path.Join(pth, id, "rw")) && !a.Exists(id) {
if err := tryRelocate(path.Join(pth, id, "rw"), path.Join(a.rootPath(), "diff", id)); err != nil {
return err
}
metadata, err := loadMetadata(path.Join(pth, id, "config.json"))
if err != nil {
return err
}
initID := fmt.Sprintf("%s-init", id)
if err := a.Create(initID, metadata.Image); err != nil {
return err
}
initPath, err := a.Get(initID)
if err != nil {
return err
}
// setup init layer
if err := setupInit(initPath); err != nil {
return err
}
if err := a.Create(id, initID); err != nil {
return err
}
}
}
return nil
}
func (a *AufsDriver) migrateImages(pth string) error {
fis, err := ioutil.ReadDir(pth) fis, err := ioutil.ReadDir(pth)
if err != nil { if err != nil {
return err return err
} }
var ( var (
metadata = make(map[string]*imageMetadata) m = make(map[string]*metadata)
current *imageMetadata current *metadata
exists bool exists bool
) )
// Load metadata
for _, fi := range fis { for _, fi := range fis {
if id := fi.Name(); fi.IsDir() && pathExists(path.Join(pth, id, "layer")) && !a.Exists(id) { if id := fi.Name(); fi.IsDir() && pathExists(path.Join(pth, id, "layer")) && !a.Exists(id) {
if current, exists = metadata[id]; !exists { if current, exists = m[id]; !exists {
current, err = loadMetadata(pth, id) current, err = loadMetadata(path.Join(pth, id, "json"))
if err != nil { if err != nil {
return err return err
} }
metadata[id] = current m[id] = current
} }
} }
} }
// Recreate tree for _, v := range m {
for _, v := range metadata { v.parent = m[v.ParentID]
v.parent = metadata[v.ParentID]
} }
// Perform image migration for _, v := range m {
for _, v := range metadata { if err := a.migrateImage(v, pth); err != nil {
if err := migrateImage(v, a, pth); err != nil {
return err return err
} }
} }
return nil return nil
} }
func migrateImage(m *imageMetadata, a *AufsDriver, pth string) error { func (a *AufsDriver) migrateImage(m *metadata, pth string) error {
if !pathExists(path.Join(a.rootPath(), "diff", m.ID)) { if !pathExists(path.Join(a.rootPath(), "diff", m.ID)) {
if m.parent != nil { if m.parent != nil {
migrateImage(m.parent, a, pth) a.migrateImage(m.parent, pth)
} }
if err := tryRelocate(path.Join(pth, m.ID, "layer"), path.Join(a.rootPath(), "diff", m.ID)); err != nil { if err := tryRelocate(path.Join(pth, m.ID, "layer"), path.Join(a.rootPath(), "diff", m.ID)); err != nil {
return err return err
@ -92,15 +147,15 @@ func tryRelocate(oldPath, newPath string) error {
return nil return nil
} }
func loadMetadata(pth, id string) (*imageMetadata, error) { func loadMetadata(pth string) (*metadata, error) {
f, err := os.Open(path.Join(pth, id, "json")) f, err := os.Open(pth)
if err != nil { if err != nil {
return nil, err return nil, err
} }
defer f.Close() defer f.Close()
var ( var (
out = &imageMetadata{} out = &metadata{}
dec = json.NewDecoder(f) dec = json.NewDecoder(f)
) )

View file

@ -630,7 +630,7 @@ func NewRuntimeFromDirectory(config *DaemonConfig) (*Runtime, error) {
} }
if ad, ok := driver.(*aufs.AufsDriver); ok { if ad, ok := driver.(*aufs.AufsDriver); ok {
if err := ad.Migrate(path.Join(config.Root, "graph")); err != nil { if err := ad.Migrate(config.Root, setupInitLayer); err != nil {
return nil, err return nil, err
} }
} }