2014-11-18 00:29:42 +00:00
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"time"
|
2014-11-19 22:39:32 +00:00
|
|
|
|
|
|
|
"github.com/docker/docker-registry/digest"
|
2014-11-18 00:29:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// LayerService provides operations on layer files in a backend storage.
|
|
|
|
type LayerService interface {
|
|
|
|
// Exists returns true if the layer exists.
|
2014-11-19 22:39:32 +00:00
|
|
|
Exists(name string, digest digest.Digest) (bool, error)
|
2014-11-18 00:29:42 +00:00
|
|
|
|
|
|
|
// Fetch the layer identifed by TarSum.
|
2014-11-19 22:39:32 +00:00
|
|
|
Fetch(name string, digest digest.Digest) (Layer, error)
|
2014-11-18 00:29:42 +00:00
|
|
|
|
2014-11-19 22:39:32 +00:00
|
|
|
// Upload begins a layer upload to repository identified by name,
|
|
|
|
// returning a handle.
|
|
|
|
Upload(name string) (LayerUpload, error)
|
2014-11-18 00:29:42 +00:00
|
|
|
|
|
|
|
// Resume continues an in progress layer upload, returning the current
|
|
|
|
// state of the upload.
|
2014-11-19 22:39:32 +00:00
|
|
|
Resume(uuid string) (LayerUpload, error)
|
2014-11-18 00:29:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Layer provides a readable and seekable layer object. Typically,
|
|
|
|
// implementations are *not* goroutine safe.
|
|
|
|
type Layer interface {
|
|
|
|
// http.ServeContent requires an efficient implementation of
|
|
|
|
// ReadSeeker.Seek(0, os.SEEK_END).
|
|
|
|
io.ReadSeeker
|
|
|
|
io.Closer
|
|
|
|
|
|
|
|
// Name returns the repository under which this layer is linked.
|
|
|
|
Name() string // TODO(stevvooe): struggling with nomenclature: should this be "repo" or "name"?
|
|
|
|
|
2014-11-19 22:39:32 +00:00
|
|
|
// Digest returns the unique digest of the blob, which is the tarsum for
|
|
|
|
// layers.
|
|
|
|
Digest() digest.Digest
|
2014-11-18 00:29:42 +00:00
|
|
|
|
|
|
|
// CreatedAt returns the time this layer was created. Until we implement
|
|
|
|
// Stat call on storagedriver, this just returns the zero time.
|
|
|
|
CreatedAt() time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
// LayerUpload provides a handle for working with in-progress uploads.
|
|
|
|
// Instances can be obtained from the LayerService.Upload and
|
|
|
|
// LayerService.Resume.
|
|
|
|
type LayerUpload interface {
|
|
|
|
io.WriteCloser
|
|
|
|
|
|
|
|
// UUID returns the identifier for this upload.
|
|
|
|
UUID() string
|
|
|
|
|
|
|
|
// Name of the repository under which the layer will be linked.
|
|
|
|
Name() string
|
|
|
|
|
|
|
|
// Offset returns the position of the last byte written to this layer.
|
|
|
|
Offset() int64
|
|
|
|
|
|
|
|
// Finish marks the upload as completed, returning a valid handle to the
|
2014-11-19 22:39:32 +00:00
|
|
|
// uploaded layer. The final size and digest are validated against the
|
|
|
|
// contents of the uploaded layer.
|
|
|
|
Finish(size int64, digest digest.Digest) (Layer, error)
|
2014-11-18 00:29:42 +00:00
|
|
|
|
|
|
|
// Cancel the layer upload process.
|
|
|
|
Cancel() error
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
// ErrLayerUnknown returned when layer cannot be found.
|
|
|
|
ErrLayerUnknown = fmt.Errorf("unknown layer")
|
|
|
|
|
|
|
|
// ErrLayerExists returned when layer already exists
|
|
|
|
ErrLayerExists = fmt.Errorf("layer exists")
|
|
|
|
|
|
|
|
// ErrLayerTarSumVersionUnsupported when tarsum is unsupported version.
|
|
|
|
ErrLayerTarSumVersionUnsupported = fmt.Errorf("unsupported tarsum version")
|
|
|
|
|
|
|
|
// ErrLayerUploadUnknown returned when upload is not found.
|
|
|
|
ErrLayerUploadUnknown = fmt.Errorf("layer upload unknown")
|
|
|
|
|
2014-11-19 22:39:32 +00:00
|
|
|
// ErrLayerInvalidDigest returned when tarsum check fails.
|
|
|
|
ErrLayerInvalidDigest = fmt.Errorf("invalid layer digest")
|
2014-11-18 00:29:42 +00:00
|
|
|
|
|
|
|
// ErrLayerInvalidLength returned when length check fails.
|
|
|
|
ErrLayerInvalidLength = fmt.Errorf("invalid layer length")
|
|
|
|
)
|