2016-07-21 17:31:14 -07:00
|
|
|
/*
|
2019-04-09 11:39:42 -07:00
|
|
|
* MinIO Cloud Storage, (C) 2016, 2017, 2017 MinIO, Inc.
|
2016-07-21 17:31:14 -07:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2016-08-18 16:23:42 -07:00
|
|
|
package cmd
|
2016-05-20 20:48:47 -07:00
|
|
|
|
|
|
|
import (
|
2018-04-05 15:04:40 -07:00
|
|
|
"context"
|
2018-03-27 17:23:10 -07:00
|
|
|
"encoding/hex"
|
|
|
|
"encoding/json"
|
2017-01-16 17:05:00 -08:00
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
2019-02-28 11:01:25 -08:00
|
|
|
"net/http"
|
2017-01-16 17:05:00 -08:00
|
|
|
"os"
|
|
|
|
pathutil "path"
|
2019-02-28 11:01:25 -08:00
|
|
|
"time"
|
2017-01-16 17:05:00 -08:00
|
|
|
|
2018-04-05 15:04:40 -07:00
|
|
|
"github.com/minio/minio/cmd/logger"
|
2017-01-16 17:05:00 -08:00
|
|
|
"github.com/minio/minio/pkg/lock"
|
|
|
|
"github.com/minio/minio/pkg/mimedb"
|
2017-04-04 09:14:03 -07:00
|
|
|
"github.com/tidwall/gjson"
|
2016-05-20 20:48:47 -07:00
|
|
|
)
|
|
|
|
|
2017-05-14 12:05:51 -07:00
|
|
|
// FS format, and object metadata.
|
2016-05-24 21:24:20 -07:00
|
|
|
const (
|
2017-05-14 12:05:51 -07:00
|
|
|
// fs.json object metadata.
|
|
|
|
fsMetaJSONFile = "fs.json"
|
2016-05-24 21:24:20 -07:00
|
|
|
)
|
|
|
|
|
2017-05-14 12:05:51 -07:00
|
|
|
// FS metadata constants.
|
|
|
|
const (
|
|
|
|
// FS backend meta 1.0.0 version.
|
|
|
|
fsMetaVersion100 = "1.0.0"
|
|
|
|
|
|
|
|
// FS backend meta 1.0.1 version.
|
2018-03-27 17:23:10 -07:00
|
|
|
fsMetaVersion101 = "1.0.1"
|
2017-05-14 12:05:51 -07:00
|
|
|
|
2018-03-27 17:23:10 -07:00
|
|
|
// FS backend meta 1.0.2
|
2019-04-09 11:39:42 -07:00
|
|
|
// Removed the fields "Format" and "MinIO" from fsMetaV1 as they were unused. Added "Checksum" field - to be used in future for bit-rot protection.
|
2018-03-27 17:23:10 -07:00
|
|
|
fsMetaVersion = "1.0.2"
|
2017-05-14 12:05:51 -07:00
|
|
|
|
|
|
|
// Add more constants here.
|
|
|
|
)
|
|
|
|
|
2018-03-27 17:23:10 -07:00
|
|
|
// FSChecksumInfoV1 - carries checksums of individual blocks on disk.
|
|
|
|
type FSChecksumInfoV1 struct {
|
|
|
|
Algorithm string
|
|
|
|
Blocksize int64
|
|
|
|
Hashes [][]byte
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON marshals the FSChecksumInfoV1 struct
|
|
|
|
func (c FSChecksumInfoV1) MarshalJSON() ([]byte, error) {
|
|
|
|
type checksuminfo struct {
|
|
|
|
Algorithm string `json:"algorithm"`
|
|
|
|
Blocksize int64 `json:"blocksize"`
|
|
|
|
Hashes []string `json:"hashes"`
|
|
|
|
}
|
|
|
|
var hashes []string
|
|
|
|
for _, h := range c.Hashes {
|
|
|
|
hashes = append(hashes, hex.EncodeToString(h))
|
|
|
|
}
|
|
|
|
info := checksuminfo{
|
|
|
|
Algorithm: c.Algorithm,
|
|
|
|
Hashes: hashes,
|
|
|
|
Blocksize: c.Blocksize,
|
|
|
|
}
|
|
|
|
return json.Marshal(info)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON unmarshals the the given data into the FSChecksumInfoV1 struct
|
|
|
|
func (c *FSChecksumInfoV1) UnmarshalJSON(data []byte) error {
|
|
|
|
type checksuminfo struct {
|
|
|
|
Algorithm string `json:"algorithm"`
|
|
|
|
Blocksize int64 `json:"blocksize"`
|
|
|
|
Hashes []string `json:"hashes"`
|
|
|
|
}
|
|
|
|
|
|
|
|
var info checksuminfo
|
|
|
|
err := json.Unmarshal(data, &info)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.Algorithm = info.Algorithm
|
|
|
|
c.Blocksize = info.Blocksize
|
|
|
|
var hashes [][]byte
|
|
|
|
for _, hashStr := range info.Hashes {
|
|
|
|
h, err := hex.DecodeString(hashStr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
hashes = append(hashes, h)
|
|
|
|
}
|
|
|
|
c.Hashes = hashes
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-05-20 20:48:47 -07:00
|
|
|
// A fsMetaV1 represents a metadata header mapping keys to sets of values.
|
|
|
|
type fsMetaV1 struct {
|
|
|
|
Version string `json:"version"`
|
2018-03-27 17:23:10 -07:00
|
|
|
// checksums of blocks on disk.
|
|
|
|
Checksum FSChecksumInfoV1 `json:"checksum,omitempty"`
|
|
|
|
// Metadata map for current object.
|
|
|
|
Meta map[string]string `json:"meta,omitempty"`
|
|
|
|
// parts info for current object - used in encryption.
|
2019-01-05 14:16:43 -08:00
|
|
|
Parts []ObjectPartInfo `json:"parts,omitempty"`
|
2016-05-20 20:48:47 -07:00
|
|
|
}
|
|
|
|
|
2017-05-14 12:05:51 -07:00
|
|
|
// IsValid - tells if the format is sane by validating the version
|
|
|
|
// string and format style.
|
|
|
|
func (m fsMetaV1) IsValid() bool {
|
2018-03-27 17:23:10 -07:00
|
|
|
return isFSMetaValid(m.Version)
|
2017-05-14 12:05:51 -07:00
|
|
|
}
|
|
|
|
|
2018-03-27 17:23:10 -07:00
|
|
|
// Verifies if the backend format metadata is same by validating
|
|
|
|
// the version string.
|
|
|
|
func isFSMetaValid(version string) bool {
|
|
|
|
return (version == fsMetaVersion || version == fsMetaVersion100 || version == fsMetaVersion101)
|
2017-05-14 12:05:51 -07:00
|
|
|
}
|
|
|
|
|
2017-01-16 17:05:00 -08:00
|
|
|
// Converts metadata to object info.
|
|
|
|
func (m fsMetaV1) ToObjectInfo(bucket, object string, fi os.FileInfo) ObjectInfo {
|
|
|
|
if len(m.Meta) == 0 {
|
|
|
|
m.Meta = make(map[string]string)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Guess content-type from the extension if possible.
|
|
|
|
if m.Meta["content-type"] == "" {
|
2018-10-02 11:48:17 +05:30
|
|
|
m.Meta["content-type"] = mimedb.TypeByExtension(pathutil.Ext(object))
|
2017-01-16 17:05:00 -08:00
|
|
|
}
|
|
|
|
|
2017-10-16 17:20:54 -07:00
|
|
|
if hasSuffix(object, slashSeparator) {
|
|
|
|
m.Meta["etag"] = emptyETag // For directories etag is d41d8cd98f00b204e9800998ecf8427e
|
|
|
|
m.Meta["content-type"] = "application/octet-stream"
|
|
|
|
}
|
|
|
|
|
2017-01-16 17:05:00 -08:00
|
|
|
objInfo := ObjectInfo{
|
|
|
|
Bucket: bucket,
|
|
|
|
Name: object,
|
|
|
|
}
|
|
|
|
|
2017-04-04 09:14:03 -07:00
|
|
|
// We set file info only if its valid.
|
2017-01-16 17:05:00 -08:00
|
|
|
objInfo.ModTime = timeSentinel
|
|
|
|
if fi != nil {
|
|
|
|
objInfo.ModTime = fi.ModTime()
|
|
|
|
objInfo.Size = fi.Size()
|
2017-10-16 17:20:54 -07:00
|
|
|
if fi.IsDir() {
|
|
|
|
// Directory is always 0 bytes in S3 API, treat it as such.
|
|
|
|
objInfo.Size = 0
|
|
|
|
objInfo.IsDir = fi.IsDir()
|
|
|
|
}
|
2017-01-16 17:05:00 -08:00
|
|
|
}
|
|
|
|
|
2017-05-14 12:05:51 -07:00
|
|
|
objInfo.ETag = extractETag(m.Meta)
|
2017-01-16 17:05:00 -08:00
|
|
|
objInfo.ContentType = m.Meta["content-type"]
|
|
|
|
objInfo.ContentEncoding = m.Meta["content-encoding"]
|
2018-04-06 17:20:02 +02:00
|
|
|
if storageClass, ok := m.Meta[amzStorageClass]; ok {
|
|
|
|
objInfo.StorageClass = storageClass
|
|
|
|
} else {
|
|
|
|
objInfo.StorageClass = globalMinioDefaultStorageClass
|
|
|
|
}
|
2019-02-28 11:01:25 -08:00
|
|
|
var (
|
|
|
|
t time.Time
|
|
|
|
e error
|
|
|
|
)
|
|
|
|
if exp, ok := m.Meta["expires"]; ok {
|
|
|
|
if t, e = time.Parse(http.TimeFormat, exp); e == nil {
|
|
|
|
objInfo.Expires = t.UTC()
|
|
|
|
}
|
|
|
|
}
|
2017-05-14 12:05:51 -07:00
|
|
|
// etag/md5Sum has already been extracted. We need to
|
|
|
|
// remove to avoid it from appearing as part of
|
|
|
|
// response headers. e.g, X-Minio-* or X-Amz-*.
|
2018-01-04 11:44:45 +05:30
|
|
|
objInfo.UserDefined = cleanMetadata(m.Meta)
|
2017-01-16 17:05:00 -08:00
|
|
|
|
2018-03-01 20:37:57 +01:00
|
|
|
// All the parts per object.
|
|
|
|
objInfo.Parts = m.Parts
|
|
|
|
|
2017-01-16 17:05:00 -08:00
|
|
|
// Success..
|
|
|
|
return objInfo
|
|
|
|
}
|
|
|
|
|
2017-01-25 12:29:06 -08:00
|
|
|
func (m *fsMetaV1) WriteTo(lk *lock.LockedFile) (n int64, err error) {
|
2018-02-06 15:37:48 -08:00
|
|
|
if err = jsonSave(lk, m); err != nil {
|
|
|
|
return 0, err
|
2017-01-16 17:05:00 -08:00
|
|
|
}
|
2018-02-06 15:37:48 -08:00
|
|
|
fi, err := lk.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
2016-05-20 20:48:47 -07:00
|
|
|
}
|
2018-02-06 15:37:48 -08:00
|
|
|
return fi.Size(), nil
|
2016-05-20 20:48:47 -07:00
|
|
|
}
|
|
|
|
|
2017-04-04 09:14:03 -07:00
|
|
|
func parseFSVersion(fsMetaBuf []byte) string {
|
|
|
|
return gjson.GetBytes(fsMetaBuf, "version").String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseFSMetaMap(fsMetaBuf []byte) map[string]string {
|
|
|
|
// Get xlMetaV1.Meta map.
|
|
|
|
metaMapResult := gjson.GetBytes(fsMetaBuf, "meta").Map()
|
|
|
|
metaMap := make(map[string]string)
|
|
|
|
for key, valResult := range metaMapResult {
|
|
|
|
metaMap[key] = valResult.String()
|
|
|
|
}
|
|
|
|
return metaMap
|
|
|
|
}
|
|
|
|
|
2019-01-05 14:16:43 -08:00
|
|
|
func parseFSPartsArray(fsMetaBuf []byte) []ObjectPartInfo {
|
2018-03-01 20:37:57 +01:00
|
|
|
// Get xlMetaV1.Parts array
|
2019-01-05 14:16:43 -08:00
|
|
|
var partsArray []ObjectPartInfo
|
2018-03-01 20:37:57 +01:00
|
|
|
|
|
|
|
partsArrayResult := gjson.GetBytes(fsMetaBuf, "parts")
|
|
|
|
partsArrayResult.ForEach(func(key, part gjson.Result) bool {
|
|
|
|
partJSON := part.String()
|
|
|
|
number := gjson.Get(partJSON, "number").Int()
|
|
|
|
name := gjson.Get(partJSON, "name").String()
|
|
|
|
etag := gjson.Get(partJSON, "etag").String()
|
|
|
|
size := gjson.Get(partJSON, "size").Int()
|
2018-09-28 09:06:17 +05:30
|
|
|
actualSize := gjson.Get(partJSON, "actualSize").Int()
|
2019-01-05 14:16:43 -08:00
|
|
|
partsArray = append(partsArray, ObjectPartInfo{
|
2018-09-28 09:06:17 +05:30
|
|
|
Number: int(number),
|
|
|
|
Name: name,
|
|
|
|
ETag: etag,
|
|
|
|
Size: size,
|
|
|
|
ActualSize: int64(actualSize),
|
2018-03-01 20:37:57 +01:00
|
|
|
})
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
return partsArray
|
|
|
|
}
|
|
|
|
|
2018-04-05 15:04:40 -07:00
|
|
|
func (m *fsMetaV1) ReadFrom(ctx context.Context, lk *lock.LockedFile) (n int64, err error) {
|
2017-04-04 09:14:03 -07:00
|
|
|
var fsMetaBuf []byte
|
2017-01-25 12:29:06 -08:00
|
|
|
fi, err := lk.Stat()
|
|
|
|
if err != nil {
|
2018-04-05 15:04:40 -07:00
|
|
|
logger.LogIf(ctx, err)
|
|
|
|
return 0, err
|
2017-01-25 12:29:06 -08:00
|
|
|
}
|
|
|
|
|
2017-04-04 09:14:03 -07:00
|
|
|
fsMetaBuf, err = ioutil.ReadAll(io.NewSectionReader(lk, 0, fi.Size()))
|
2016-09-01 02:12:57 +05:30
|
|
|
if err != nil {
|
2018-04-05 15:04:40 -07:00
|
|
|
logger.LogIf(ctx, err)
|
|
|
|
return 0, err
|
2016-09-01 02:12:57 +05:30
|
|
|
}
|
2017-01-16 17:05:00 -08:00
|
|
|
|
2017-04-04 09:14:03 -07:00
|
|
|
if len(fsMetaBuf) == 0 {
|
2018-04-05 15:04:40 -07:00
|
|
|
logger.LogIf(ctx, io.EOF)
|
|
|
|
return 0, io.EOF
|
2016-08-25 22:09:01 +05:30
|
|
|
}
|
2017-01-16 17:05:00 -08:00
|
|
|
|
2017-04-04 09:14:03 -07:00
|
|
|
// obtain version.
|
|
|
|
m.Version = parseFSVersion(fsMetaBuf)
|
|
|
|
|
2017-05-14 12:05:51 -07:00
|
|
|
// Verify if the format is valid, return corrupted format
|
|
|
|
// for unrecognized formats.
|
2018-03-27 17:23:10 -07:00
|
|
|
if !isFSMetaValid(m.Version) {
|
2018-04-05 15:04:40 -07:00
|
|
|
logger.GetReqInfo(ctx).AppendTags("file", lk.Name())
|
|
|
|
logger.LogIf(ctx, errCorruptedFormat)
|
|
|
|
return 0, errCorruptedFormat
|
2017-05-14 12:05:51 -07:00
|
|
|
}
|
|
|
|
|
2018-03-01 20:37:57 +01:00
|
|
|
// obtain parts information
|
|
|
|
m.Parts = parseFSPartsArray(fsMetaBuf)
|
|
|
|
|
2017-04-04 09:14:03 -07:00
|
|
|
// obtain metadata.
|
|
|
|
m.Meta = parseFSMetaMap(fsMetaBuf)
|
|
|
|
|
2017-01-16 17:05:00 -08:00
|
|
|
// Success.
|
2017-04-04 09:14:03 -07:00
|
|
|
return int64(len(fsMetaBuf)), nil
|
2016-09-01 02:12:57 +05:30
|
|
|
}
|
|
|
|
|
2016-05-26 03:15:01 -07:00
|
|
|
// newFSMetaV1 - initializes new fsMetaV1.
|
|
|
|
func newFSMetaV1() (fsMeta fsMetaV1) {
|
|
|
|
fsMeta = fsMetaV1{}
|
2017-01-18 12:24:34 -08:00
|
|
|
fsMeta.Version = fsMetaVersion
|
2016-05-26 03:15:01 -07:00
|
|
|
return fsMeta
|
|
|
|
}
|