Migrate filesystem driver to new storagedriver calls
The filesystem driver has been migrated to impleemnt the storagedriver interface changes. Most interetingly, this provides a filesystem-based implementation of the Stat driver call. With this comes some refactoring of Reads and Write to be much simpler and more robust. The IPC tests have been disabled to stability problems that we'll have to troubleshoot at a later date.
This commit is contained in:
parent
2037b1d6bf
commit
ab9570f872
2 changed files with 118 additions and 86 deletions
|
@ -1,10 +1,13 @@
|
||||||
package filesystem
|
package filesystem
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/docker/docker-registry/storagedriver"
|
"github.com/docker/docker-registry/storagedriver"
|
||||||
"github.com/docker/docker-registry/storagedriver/factory"
|
"github.com/docker/docker-registry/storagedriver/factory"
|
||||||
|
@ -49,41 +52,43 @@ func New(rootDirectory string) *Driver {
|
||||||
return &Driver{rootDirectory}
|
return &Driver{rootDirectory}
|
||||||
}
|
}
|
||||||
|
|
||||||
// subPath returns the absolute path of a key within the Driver's storage
|
|
||||||
func (d *Driver) subPath(subPath string) string {
|
|
||||||
return path.Join(d.rootDirectory, subPath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Implement the storagedriver.StorageDriver interface
|
// Implement the storagedriver.StorageDriver interface
|
||||||
|
|
||||||
// GetContent retrieves the content stored at "path" as a []byte.
|
// GetContent retrieves the content stored at "path" as a []byte.
|
||||||
func (d *Driver) GetContent(path string) ([]byte, error) {
|
func (d *Driver) GetContent(path string) ([]byte, error) {
|
||||||
contents, err := ioutil.ReadFile(d.subPath(path))
|
rc, err := d.ReadStream(path, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, storagedriver.PathNotFoundError{Path: path}
|
return nil, err
|
||||||
}
|
}
|
||||||
return contents, nil
|
defer rc.Close()
|
||||||
|
|
||||||
|
p, err := ioutil.ReadAll(rc)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return p, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// PutContent stores the []byte content at a location designated by "path".
|
// PutContent stores the []byte content at a location designated by "path".
|
||||||
func (d *Driver) PutContent(subPath string, contents []byte) error {
|
func (d *Driver) PutContent(subPath string, contents []byte) error {
|
||||||
fullPath := d.subPath(subPath)
|
if _, err := d.WriteStream(subPath, 0, bytes.NewReader(contents)); err != nil {
|
||||||
parentDir := path.Dir(fullPath)
|
|
||||||
err := os.MkdirAll(parentDir, 0755)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
err = ioutil.WriteFile(fullPath, contents, 0644)
|
return os.Truncate(d.fullPath(subPath), int64(len(contents)))
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ReadStream retrieves an io.ReadCloser for the content stored at "path" with a
|
// ReadStream retrieves an io.ReadCloser for the content stored at "path" with a
|
||||||
// given byte offset.
|
// given byte offset.
|
||||||
func (d *Driver) ReadStream(path string, offset int64) (io.ReadCloser, error) {
|
func (d *Driver) ReadStream(path string, offset int64) (io.ReadCloser, error) {
|
||||||
file, err := os.OpenFile(d.subPath(path), os.O_RDONLY, 0644)
|
file, err := os.OpenFile(d.fullPath(path), os.O_RDONLY, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, storagedriver.PathNotFoundError{Path: path}
|
if os.IsNotExist(err) {
|
||||||
|
return nil, storagedriver.PathNotFoundError{Path: path}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
seekPos, err := file.Seek(int64(offset), os.SEEK_SET)
|
seekPos, err := file.Seek(int64(offset), os.SEEK_SET)
|
||||||
|
@ -98,81 +103,64 @@ func (d *Driver) ReadStream(path string, offset int64) (io.ReadCloser, error) {
|
||||||
return file, nil
|
return file, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// WriteStream stores the contents of the provided io.ReadCloser at a location
|
// WriteStream stores the contents of the provided io.Reader at a location
|
||||||
// designated by the given path.
|
// designated by the given path.
|
||||||
func (d *Driver) WriteStream(subPath string, offset, size int64, reader io.ReadCloser) error {
|
func (d *Driver) WriteStream(subPath string, offset int64, reader io.Reader) (nn int64, err error) {
|
||||||
defer reader.Close()
|
if offset < 0 {
|
||||||
|
return 0, storagedriver.InvalidOffsetError{Path: subPath, Offset: offset}
|
||||||
resumableOffset, err := d.CurrentSize(subPath)
|
|
||||||
if _, pathNotFound := err.(storagedriver.PathNotFoundError); err != nil && !pathNotFound {
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if offset > int64(resumableOffset) {
|
// TODO(stevvooe): This needs to be a requirement.
|
||||||
return storagedriver.InvalidOffsetError{Path: subPath, Offset: offset}
|
// if !path.IsAbs(subPath) {
|
||||||
}
|
// return fmt.Errorf("absolute path required: %q", subPath)
|
||||||
|
// }
|
||||||
|
|
||||||
fullPath := d.subPath(subPath)
|
fullPath := d.fullPath(subPath)
|
||||||
parentDir := path.Dir(fullPath)
|
parentDir := path.Dir(fullPath)
|
||||||
err = os.MkdirAll(parentDir, 0755)
|
if err := os.MkdirAll(parentDir, 0755); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
fp, err := os.OpenFile(fullPath, os.O_WRONLY|os.O_CREATE, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
// TODO(stevvooe): A few missing conditions in storage driver:
|
||||||
}
|
// 1. What if the path is already a directory?
|
||||||
|
// 2. Should number 1 be exposed explicitly in storagedriver?
|
||||||
var file *os.File
|
// 2. Can this path not exist, even if we create above?
|
||||||
if offset == 0 {
|
return 0, err
|
||||||
file, err = os.Create(fullPath)
|
|
||||||
} else {
|
|
||||||
file, err = os.OpenFile(fullPath, os.O_WRONLY|os.O_APPEND, 0)
|
|
||||||
}
|
}
|
||||||
|
defer fp.Close()
|
||||||
|
|
||||||
|
nn, err = fp.Seek(offset, os.SEEK_SET)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return 0, err
|
||||||
}
|
}
|
||||||
defer file.Close()
|
|
||||||
|
|
||||||
// TODO(sday): Use Seek + Copy here.
|
if nn != offset {
|
||||||
|
return 0, fmt.Errorf("bad seek to %v, expected %v in fp=%v", offset, nn, fp)
|
||||||
buf := make([]byte, 32*1024)
|
|
||||||
for {
|
|
||||||
bytesRead, er := reader.Read(buf)
|
|
||||||
if bytesRead > 0 {
|
|
||||||
bytesWritten, ew := file.WriteAt(buf[0:bytesRead], int64(offset))
|
|
||||||
if bytesWritten > 0 {
|
|
||||||
offset += int64(bytesWritten)
|
|
||||||
}
|
|
||||||
if ew != nil {
|
|
||||||
err = ew
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if bytesRead != bytesWritten {
|
|
||||||
err = io.ErrShortWrite
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if er == io.EOF {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if er != nil {
|
|
||||||
err = er
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return err
|
|
||||||
|
return io.Copy(fp, reader)
|
||||||
}
|
}
|
||||||
|
|
||||||
// CurrentSize retrieves the curernt size in bytes of the object at the given
|
// Stat retrieves the FileInfo for the given path, including the current size
|
||||||
// path.
|
// in bytes and the creation time.
|
||||||
func (d *Driver) CurrentSize(subPath string) (uint64, error) {
|
func (d *Driver) Stat(subPath string) (storagedriver.FileInfo, error) {
|
||||||
fullPath := d.subPath(subPath)
|
fullPath := d.fullPath(subPath)
|
||||||
|
|
||||||
fileInfo, err := os.Stat(fullPath)
|
fi, err := os.Stat(fullPath)
|
||||||
if err != nil && !os.IsNotExist(err) {
|
if err != nil {
|
||||||
return 0, err
|
if os.IsNotExist(err) {
|
||||||
} else if err != nil {
|
return nil, storagedriver.PathNotFoundError{Path: subPath}
|
||||||
return 0, storagedriver.PathNotFoundError{Path: subPath}
|
}
|
||||||
|
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
return uint64(fileInfo.Size()), nil
|
|
||||||
|
return fileInfo{
|
||||||
|
path: subPath,
|
||||||
|
FileInfo: fi,
|
||||||
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// List returns a list of the objects that are direct descendants of the given
|
// List returns a list of the objects that are direct descendants of the given
|
||||||
|
@ -181,7 +169,7 @@ func (d *Driver) List(subPath string) ([]string, error) {
|
||||||
if subPath[len(subPath)-1] != '/' {
|
if subPath[len(subPath)-1] != '/' {
|
||||||
subPath += "/"
|
subPath += "/"
|
||||||
}
|
}
|
||||||
fullPath := d.subPath(subPath)
|
fullPath := d.fullPath(subPath)
|
||||||
|
|
||||||
dir, err := os.Open(fullPath)
|
dir, err := os.Open(fullPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -204,8 +192,8 @@ func (d *Driver) List(subPath string) ([]string, error) {
|
||||||
// Move moves an object stored at sourcePath to destPath, removing the original
|
// Move moves an object stored at sourcePath to destPath, removing the original
|
||||||
// object.
|
// object.
|
||||||
func (d *Driver) Move(sourcePath string, destPath string) error {
|
func (d *Driver) Move(sourcePath string, destPath string) error {
|
||||||
source := d.subPath(sourcePath)
|
source := d.fullPath(sourcePath)
|
||||||
dest := d.subPath(destPath)
|
dest := d.fullPath(destPath)
|
||||||
|
|
||||||
if _, err := os.Stat(source); os.IsNotExist(err) {
|
if _, err := os.Stat(source); os.IsNotExist(err) {
|
||||||
return storagedriver.PathNotFoundError{Path: sourcePath}
|
return storagedriver.PathNotFoundError{Path: sourcePath}
|
||||||
|
@ -217,7 +205,7 @@ func (d *Driver) Move(sourcePath string, destPath string) error {
|
||||||
|
|
||||||
// Delete recursively deletes all objects stored at "path" and its subpaths.
|
// Delete recursively deletes all objects stored at "path" and its subpaths.
|
||||||
func (d *Driver) Delete(subPath string) error {
|
func (d *Driver) Delete(subPath string) error {
|
||||||
fullPath := d.subPath(subPath)
|
fullPath := d.fullPath(subPath)
|
||||||
|
|
||||||
_, err := os.Stat(fullPath)
|
_, err := os.Stat(fullPath)
|
||||||
if err != nil && !os.IsNotExist(err) {
|
if err != nil && !os.IsNotExist(err) {
|
||||||
|
@ -229,3 +217,42 @@ func (d *Driver) Delete(subPath string) error {
|
||||||
err = os.RemoveAll(fullPath)
|
err = os.RemoveAll(fullPath)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// fullPath returns the absolute path of a key within the Driver's storage.
|
||||||
|
func (d *Driver) fullPath(subPath string) string {
|
||||||
|
return path.Join(d.rootDirectory, subPath)
|
||||||
|
}
|
||||||
|
|
||||||
|
type fileInfo struct {
|
||||||
|
os.FileInfo
|
||||||
|
path string
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ storagedriver.FileInfo = fileInfo{}
|
||||||
|
|
||||||
|
// Path provides the full path of the target of this file info.
|
||||||
|
func (fi fileInfo) Path() string {
|
||||||
|
return fi.path
|
||||||
|
}
|
||||||
|
|
||||||
|
// Size returns current length in bytes of the file. The return value can
|
||||||
|
// be used to write to the end of the file at path. The value is
|
||||||
|
// meaningless if IsDir returns true.
|
||||||
|
func (fi fileInfo) Size() int64 {
|
||||||
|
if fi.IsDir() {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
return fi.FileInfo.Size()
|
||||||
|
}
|
||||||
|
|
||||||
|
// ModTime returns the modification time for the file. For backends that
|
||||||
|
// don't have a modification time, the creation time should be returned.
|
||||||
|
func (fi fileInfo) ModTime() time.Time {
|
||||||
|
return fi.FileInfo.ModTime()
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsDir returns true if the path is a directory.
|
||||||
|
func (fi fileInfo) IsDir() bool {
|
||||||
|
return fi.FileInfo.IsDir()
|
||||||
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package filesystem
|
package filesystem
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -13,12 +14,16 @@ import (
|
||||||
func Test(t *testing.T) { TestingT(t) }
|
func Test(t *testing.T) { TestingT(t) }
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
rootDirectory := "/tmp/driver"
|
root, err := ioutil.TempDir("", "driver-")
|
||||||
os.RemoveAll(rootDirectory)
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
filesystemDriverConstructor := func() (storagedriver.StorageDriver, error) {
|
|
||||||
return New(rootDirectory), nil
|
|
||||||
}
|
}
|
||||||
testsuites.RegisterInProcessSuite(filesystemDriverConstructor, testsuites.NeverSkip)
|
defer os.Remove(root)
|
||||||
testsuites.RegisterIPCSuite(driverName, map[string]string{"rootdirectory": rootDirectory}, testsuites.NeverSkip)
|
|
||||||
|
testsuites.RegisterInProcessSuite(func() (storagedriver.StorageDriver, error) {
|
||||||
|
return New(root), nil
|
||||||
|
}, testsuites.NeverSkip)
|
||||||
|
|
||||||
|
// BUG(stevvooe): IPC is broken so we're disabling for now. Will revisit later.
|
||||||
|
// testsuites.RegisterIPCSuite(driverName, map[string]string{"rootdirectory": root}, testsuites.NeverSkip)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue