 7d3a1e7737
			
		
	
	7d3a1e7737
	
	
	
		
			
			Update snapshot storage to use boltdb keys instead of protobuf. Add update function to snapshotter to allow updating labels. Add option type for creating snapshots to allow setting labels. Signed-off-by: Derek McGowan <derek@mcgstyle.net>
		
			
				
	
	
		
			323 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			323 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package naive
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"io/ioutil"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 
 | |
| 	"github.com/containerd/containerd/fs"
 | |
| 	"github.com/containerd/containerd/log"
 | |
| 	"github.com/containerd/containerd/mount"
 | |
| 	"github.com/containerd/containerd/plugin"
 | |
| 	"github.com/containerd/containerd/snapshot"
 | |
| 	"github.com/containerd/containerd/snapshot/storage"
 | |
| 	"github.com/pkg/errors"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	plugin.Register(&plugin.Registration{
 | |
| 		Type: plugin.SnapshotPlugin,
 | |
| 		ID:   "naive",
 | |
| 		Init: func(ic *plugin.InitContext) (interface{}, error) {
 | |
| 			return NewSnapshotter(ic.Root)
 | |
| 		},
 | |
| 	})
 | |
| }
 | |
| 
 | |
| 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()
 | |
| 	_, info, _, err := storage.GetInfo(ctx, key)
 | |
| 	if err != nil {
 | |
| 		return snapshot.Info{}, err
 | |
| 	}
 | |
| 
 | |
| 	return info, nil
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) Update(ctx context.Context, info snapshot.Info, fieldpaths ...string) (snapshot.Info, error) {
 | |
| 	ctx, t, err := o.ms.TransactionContext(ctx, true)
 | |
| 	if err != nil {
 | |
| 		return snapshot.Info{}, err
 | |
| 	}
 | |
| 
 | |
| 	info, err = storage.UpdateInfo(ctx, info, fieldpaths...)
 | |
| 	if err != nil {
 | |
| 		t.Rollback()
 | |
| 		return snapshot.Info{}, err
 | |
| 	}
 | |
| 
 | |
| 	if err := t.Commit(); err != nil {
 | |
| 		return snapshot.Info{}, err
 | |
| 	}
 | |
| 
 | |
| 	return info, nil
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) Usage(ctx context.Context, key string) (snapshot.Usage, error) {
 | |
| 	ctx, t, err := o.ms.TransactionContext(ctx, false)
 | |
| 	if err != nil {
 | |
| 		return snapshot.Usage{}, err
 | |
| 	}
 | |
| 	defer t.Rollback()
 | |
| 
 | |
| 	id, info, usage, err := storage.GetInfo(ctx, key)
 | |
| 	if err != nil {
 | |
| 		return snapshot.Usage{}, err
 | |
| 	}
 | |
| 
 | |
| 	if info.Kind == snapshot.KindActive {
 | |
| 		du, err := fs.DiskUsage(o.getSnapshotDir(id))
 | |
| 		if err != nil {
 | |
| 			return snapshot.Usage{}, err
 | |
| 		}
 | |
| 		usage = snapshot.Usage(du)
 | |
| 	}
 | |
| 
 | |
| 	return usage, nil
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) Prepare(ctx context.Context, key, parent string, opts ...snapshot.Opt) ([]mount.Mount, error) {
 | |
| 	return o.createSnapshot(ctx, snapshot.KindActive, key, parent, opts)
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) View(ctx context.Context, key, parent string, opts ...snapshot.Opt) ([]mount.Mount, error) {
 | |
| 	return o.createSnapshot(ctx, snapshot.KindView, key, parent, opts)
 | |
| }
 | |
| 
 | |
| // 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) ([]mount.Mount, error) {
 | |
| 	ctx, t, err := o.ms.TransactionContext(ctx, false)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	s, err := storage.GetSnapshot(ctx, key)
 | |
| 	t.Rollback()
 | |
| 	if err != nil {
 | |
| 		return nil, errors.Wrap(err, "failed to get snapshot mount")
 | |
| 	}
 | |
| 	return o.mounts(s), nil
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) Commit(ctx context.Context, name, key string, opts ...snapshot.Opt) error {
 | |
| 	ctx, t, err := o.ms.TransactionContext(ctx, true)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	id, _, _, err := storage.GetInfo(ctx, key)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	usage, err := fs.DiskUsage(o.getSnapshotDir(id))
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if _, err := storage.CommitActive(ctx, key, name, snapshot.Usage(usage), opts...); 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) createSnapshot(ctx context.Context, kind snapshot.Kind, key, parent string, opts []snapshot.Opt) ([]mount.Mount, error) {
 | |
| 	var (
 | |
| 		err      error
 | |
| 		path, td string
 | |
| 	)
 | |
| 
 | |
| 	if kind == snapshot.KindActive || 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
 | |
| 	}
 | |
| 
 | |
| 	s, err := storage.CreateSnapshot(ctx, kind, key, parent, opts...)
 | |
| 	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 snapshot")
 | |
| 	}
 | |
| 
 | |
| 	if td != "" {
 | |
| 		if len(s.ParentIDs) > 0 {
 | |
| 			parent := o.getSnapshotDir(s.ParentIDs[0])
 | |
| 			if err := fs.CopyDir(td, parent); err != nil {
 | |
| 				return nil, errors.Wrap(err, "copying of parent failed")
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		path = o.getSnapshotDir(s.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(s), nil
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) getSnapshotDir(id string) string {
 | |
| 	return filepath.Join(o.root, "snapshots", id)
 | |
| }
 | |
| 
 | |
| func (o *snapshotter) mounts(s storage.Snapshot) []mount.Mount {
 | |
| 	var (
 | |
| 		roFlag string
 | |
| 		source string
 | |
| 	)
 | |
| 
 | |
| 	if s.Kind == snapshot.KindView {
 | |
| 		roFlag = "ro"
 | |
| 	} else {
 | |
| 		roFlag = "rw"
 | |
| 	}
 | |
| 
 | |
| 	if len(s.ParentIDs) == 0 || s.Kind == snapshot.KindActive {
 | |
| 		source = o.getSnapshotDir(s.ID)
 | |
| 	} else {
 | |
| 		source = o.getSnapshotDir(s.ParentIDs[0])
 | |
| 	}
 | |
| 
 | |
| 	return []mount.Mount{
 | |
| 		{
 | |
| 			Source: source,
 | |
| 			Type:   "bind",
 | |
| 			Options: []string{
 | |
| 				roFlag,
 | |
| 				"rbind",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| }
 |