2014-12-13 17:27:06 -05:00
|
|
|
/*
|
|
|
|
* Mini Object Storage, (C) 2014 Minio, Inc.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2014-12-09 02:31:19 -05:00
|
|
|
package appendstorage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/gob"
|
2014-12-10 20:43:16 -05:00
|
|
|
"errors"
|
2014-12-10 23:40:53 -05:00
|
|
|
"io"
|
2014-12-09 02:31:19 -05:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"strconv"
|
2014-12-14 21:55:35 -05:00
|
|
|
"strings"
|
2014-12-09 02:31:19 -05:00
|
|
|
|
2014-12-29 19:35:56 -05:00
|
|
|
"github.com/minio-io/minio/pkg/checksum/crc32c"
|
|
|
|
"github.com/minio-io/minio/pkg/storage"
|
2014-12-09 02:31:19 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2014-12-10 23:40:53 -05:00
|
|
|
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 {
|
2014-12-15 01:13:37 -05:00
|
|
|
return nil, errors.New("Object not found")
|
2014-12-09 02:31:19 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
2014-12-10 23:40:53 -05:00
|
|
|
return bytes.NewBuffer(object), nil
|
2014-12-09 02:31:19 -05:00
|
|
|
}
|
|
|
|
|
2014-12-10 23:40:53 -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
|
|
|
}
|
2014-12-10 21:54:04 -05:00
|
|
|
offset, err := aStorage.file.Seek(0, os.SEEK_END)
|
2014-12-09 02:31:19 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2014-12-10 23:40:53 -05:00
|
|
|
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
|
2014-12-10 23:40:53 -05:00
|
|
|
header.Length = len(objectBytes)
|
2014-12-11 05:11:48 -05:00
|
|
|
header.Crc, err = crc32c.Crc32c(objectBytes)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2014-12-10 21:54:04 -05:00
|
|
|
aStorage.objects[objectPath] = header
|
2014-12-09 02:31:19 -05:00
|
|
|
var mapBuffer bytes.Buffer
|
|
|
|
encoder := gob.NewEncoder(&mapBuffer)
|
2014-12-10 21:54:04 -05:00
|
|
|
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
|
|
|
|
2014-12-14 21:55:35 -05:00
|
|
|
func (aStorage *appendStorage) List(objectPath string) ([]storage.ObjectDescription, error) {
|
2014-12-12 05:50:47 -05:00
|
|
|
var objectDescList []storage.ObjectDescription
|
|
|
|
for objectName, _ := range aStorage.objects {
|
2014-12-14 21:55:35 -05:00
|
|
|
if strings.HasPrefix(objectName, objectPath) {
|
|
|
|
var objectDescription storage.ObjectDescription
|
|
|
|
objectDescription.Name = objectName
|
|
|
|
objectDescription.Md5sum = ""
|
|
|
|
objectDescription.Murmur3 = ""
|
|
|
|
objectDescList = append(objectDescList, objectDescription)
|
|
|
|
}
|
2014-12-12 05:50:47 -05:00
|
|
|
}
|
|
|
|
if len(objectDescList) == 0 {
|
|
|
|
return nil, errors.New("No objects found")
|
|
|
|
}
|
|
|
|
return objectDescList, nil
|
2014-12-10 20:43:16 -05:00
|
|
|
}
|