package naive import ( "context" "io/ioutil" "os" "path/filepath" "github.com/containerd/containerd" "github.com/containerd/containerd/fs" "github.com/containerd/containerd/log" "github.com/containerd/containerd/plugin" "github.com/containerd/containerd/snapshot" "github.com/containerd/containerd/snapshot/storage" "github.com/pkg/errors" ) func init() { plugin.Register("snapshot-naive", &plugin.Registration{ Type: plugin.SnapshotPlugin, Init: func(ic *plugin.InitContext) (interface{}, error) { return NewSnapshotter(filepath.Join(ic.Root, "snapshot", "naive")) }, }) } type snapshotter struct { root string ms *storage.MetaStore } // NewSnapshotter returns a Snapshotter which copies layers on the underlying // file system. A metadata file is stored under the root. func NewSnapshotter(root string) (snapshot.Snapshotter, error) { if err := os.MkdirAll(root, 0700); err != nil { return nil, err } ms, err := storage.NewMetaStore(filepath.Join(root, "metadata.db")) if err != nil { return nil, err } if err := os.Mkdir(filepath.Join(root, "snapshots"), 0700); err != nil && !os.IsExist(err) { return nil, err } return &snapshotter{ root: root, ms: ms, }, nil } // Stat returns the info for an active or committed snapshot by name or // key. // // Should be used for parent resolution, existence checks and to discern // the kind of snapshot. func (o *snapshotter) Stat(ctx context.Context, key string) (snapshot.Info, error) { ctx, t, err := o.ms.TransactionContext(ctx, false) if err != nil { return snapshot.Info{}, err } defer t.Rollback() return storage.GetInfo(ctx, key) } func (o *snapshotter) Prepare(ctx context.Context, key, parent string) ([]containerd.Mount, error) { return o.createActive(ctx, key, parent, false) } func (o *snapshotter) View(ctx context.Context, key, parent string) ([]containerd.Mount, error) { return o.createActive(ctx, key, parent, true) } // Mounts returns the mounts for the transaction identified by key. Can be // called on an read-write or readonly transaction. // // This can be used to recover mounts after calling View or Prepare. func (o *snapshotter) Mounts(ctx context.Context, key string) ([]containerd.Mount, error) { ctx, t, err := o.ms.TransactionContext(ctx, false) if err != nil { return nil, err } active, err := storage.GetActive(ctx, key) t.Rollback() if err != nil { return nil, errors.Wrap(err, "failed to get active mount") } return o.mounts(active), nil } func (o *snapshotter) Commit(ctx context.Context, name, key string) error { ctx, t, err := o.ms.TransactionContext(ctx, true) if err != nil { return err } if _, err := storage.CommitActive(ctx, key, name); err != nil { if rerr := t.Rollback(); rerr != nil { log.G(ctx).WithError(rerr).Warn("Failure rolling back transaction") } return errors.Wrap(err, "failed to commit snapshot") } return t.Commit() } // Remove abandons the transaction identified by key. All resources // associated with the key will be removed. func (o *snapshotter) Remove(ctx context.Context, key string) (err error) { ctx, t, err := o.ms.TransactionContext(ctx, true) if err != nil { return err } defer func() { if err != nil && t != nil { if rerr := t.Rollback(); rerr != nil { log.G(ctx).WithError(rerr).Warn("Failure rolling back transaction") } } }() id, _, err := storage.Remove(ctx, key) if err != nil { return errors.Wrap(err, "failed to remove") } path := o.getSnapshotDir(id) renamed := filepath.Join(o.root, "snapshots", "rm-"+id) if err := os.Rename(path, renamed); err != nil { if !os.IsNotExist(err) { return errors.Wrap(err, "failed to rename") } renamed = "" } err = t.Commit() t = nil if err != nil { if renamed != "" { if err1 := os.Rename(renamed, path); err1 != nil { // May cause inconsistent data on disk log.G(ctx).WithError(err1).WithField("path", renamed).Errorf("Failed to rename after failed commit") } } return errors.Wrap(err, "failed to commit") } if renamed != "" { if err := os.RemoveAll(renamed); err != nil { // Must be cleaned up, any "rm-*" could be removed if no active transactions log.G(ctx).WithError(err).WithField("path", renamed).Warnf("Failed to remove root filesystem") } } return nil } // Walk the committed snapshots. func (o *snapshotter) Walk(ctx context.Context, fn func(context.Context, snapshot.Info) error) error { ctx, t, err := o.ms.TransactionContext(ctx, false) if err != nil { return err } defer t.Rollback() return storage.WalkInfo(ctx, fn) } func (o *snapshotter) createActive(ctx context.Context, key, parent string, readonly bool) ([]containerd.Mount, error) { var ( err error path, td string ) if !readonly || parent == "" { td, err = ioutil.TempDir(filepath.Join(o.root, "snapshots"), "new-") if err != nil { return nil, errors.Wrap(err, "failed to create temp dir") } defer func() { if err != nil { if td != "" { if err1 := os.RemoveAll(td); err1 != nil { err = errors.Wrapf(err, "remove failed: %v", err1) } } if path != "" { if err1 := os.RemoveAll(path); err1 != nil { err = errors.Wrapf(err, "failed to remove path: %v", err1) } } } }() } ctx, t, err := o.ms.TransactionContext(ctx, true) if err != nil { return nil, err } active, err := storage.CreateActive(ctx, key, parent, readonly) if err != nil { if rerr := t.Rollback(); rerr != nil { log.G(ctx).WithError(rerr).Warn("Failure rolling back transaction") } return nil, errors.Wrap(err, "failed to create active") } if td != "" { if len(active.ParentIDs) > 0 { parent := o.getSnapshotDir(active.ParentIDs[0]) if err := fs.CopyDir(td, parent); err != nil { return nil, errors.Wrap(err, "copying of parent failed") } } path = o.getSnapshotDir(active.ID) if err := os.Rename(td, path); err != nil { if rerr := t.Rollback(); rerr != nil { log.G(ctx).WithError(rerr).Warn("Failure rolling back transaction") } return nil, errors.Wrap(err, "failed to rename") } td = "" } if err := t.Commit(); err != nil { return nil, errors.Wrap(err, "commit failed") } return o.mounts(active), nil } func (o *snapshotter) getSnapshotDir(id string) string { return filepath.Join(o.root, "snapshots", id) } func (o *snapshotter) mounts(active storage.Active) []containerd.Mount { var ( roFlag string source string ) if active.Readonly { roFlag = "ro" } else { roFlag = "rw" } if len(active.ParentIDs) == 0 || !active.Readonly { source = o.getSnapshotDir(active.ID) } else { source = o.getSnapshotDir(active.ParentIDs[0]) } return []containerd.Mount{ { Source: source, Type: "bind", Options: []string{ roFlag, "rbind", }, }, } }