minio/pkgs/storage/appendstorage/append_storage.go

123 lines
2.6 KiB
Go
Raw Normal View History

2014-12-09 02:31:19 -05:00
package appendstorage
import (
"bytes"
"encoding/gob"
2014-12-10 20:43:16 -05:00
"errors"
"io"
2014-12-09 02:31:19 -05:00
"io/ioutil"
"os"
"path"
"strconv"
"github.com/minio-io/minio/pkgs/checksum/crc32c"
2014-12-09 02:31:19 -05:00
"github.com/minio-io/minio/pkgs/storage"
)
type appendStorage struct {
RootDir string
file *os.File
objects map[string]Header
objectsFile string
}
type Header struct {
Path string
Offset int64
Length int
2014-12-11 04:45:58 -05:00
Crc uint32
2014-12-09 02:31:19 -05:00
}
func NewStorage(rootDir string, slice int) (storage.ObjectStorage, error) {
rootPath := path.Join(rootDir, strconv.Itoa(slice))
// TODO verify and fix partial writes
file, err := os.OpenFile(rootPath, os.O_APPEND|os.O_CREATE|os.O_RDWR, 0600)
if err != nil {
return &appendStorage{}, err
}
objectsFile := path.Join(rootDir, strconv.Itoa(slice)+".map")
objects := make(map[string]Header)
if _, err := os.Stat(objectsFile); err == nil {
mapFile, err := os.Open(objectsFile)
defer mapFile.Close()
if err != nil {
return &appendStorage{}, nil
}
dec := gob.NewDecoder(mapFile)
err = dec.Decode(&objects)
2014-12-11 04:45:58 -05:00
if err != nil && err != io.EOF {
2014-12-09 02:31:19 -05:00
return &appendStorage{}, nil
}
}
if err != nil {
return &appendStorage{}, err
}
return &appendStorage{
RootDir: rootDir,
file: file,
objects: objects,
objectsFile: objectsFile,
}, nil
}
func (storage *appendStorage) Get(objectPath string) (io.Reader, error) {
2014-12-09 02:31:19 -05:00
header, ok := storage.objects[objectPath]
if ok == false {
return nil, nil
}
offset := header.Offset
length := header.Length
2014-12-11 18:06:29 -05:00
crc := header.Crc
2014-12-09 02:31:19 -05:00
object := make([]byte, length)
_, err := storage.file.ReadAt(object, offset)
if err != nil {
return nil, err
}
2014-12-11 18:06:29 -05:00
newcrc, err := crc32c.Crc32c(object)
if err != nil {
return nil, err
}
if newcrc != crc {
return nil, err
}
return bytes.NewBuffer(object), nil
2014-12-09 02:31:19 -05:00
}
func (aStorage *appendStorage) Put(objectPath string, object io.Reader) error {
2014-12-09 02:31:19 -05:00
header := Header{
Path: objectPath,
Offset: 0,
Length: 0,
2014-12-11 04:45:58 -05:00
Crc: 0,
2014-12-09 02:31:19 -05:00
}
offset, err := aStorage.file.Seek(0, os.SEEK_END)
2014-12-09 02:31:19 -05:00
if err != nil {
return err
}
objectBytes, err := ioutil.ReadAll(object)
if err != nil {
return err
}
if _, err := aStorage.file.Write(objectBytes); err != nil {
2014-12-09 02:31:19 -05:00
return err
}
header.Offset = offset
header.Length = len(objectBytes)
header.Crc, err = crc32c.Crc32c(objectBytes)
if err != nil {
return err
}
aStorage.objects[objectPath] = header
2014-12-09 02:31:19 -05:00
var mapBuffer bytes.Buffer
encoder := gob.NewEncoder(&mapBuffer)
encoder.Encode(aStorage.objects)
ioutil.WriteFile(aStorage.objectsFile, mapBuffer.Bytes(), 0600)
2014-12-09 02:31:19 -05:00
return nil
}
2014-12-10 20:43:16 -05:00
func (aStorage *appendStorage) List(listPath string) ([]storage.ObjectDescription, error) {
2014-12-10 20:43:16 -05:00
return nil, errors.New("Not Implemented")
}