2022-06-05 01:15:39 +00:00
|
|
|
package needle
|
|
|
|
|
|
|
|
import (
|
2022-10-23 21:00:28 +00:00
|
|
|
"fmt"
|
2022-07-29 07:17:28 +00:00
|
|
|
"github.com/seaweedfs/seaweedfs/weed/glog"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/storage/backend"
|
|
|
|
. "github.com/seaweedfs/seaweedfs/weed/storage/types"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/util"
|
2022-06-05 01:15:39 +00:00
|
|
|
"io"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ReadNeedleData uses a needle without n.Data to read the content
|
|
|
|
// volumeOffset: the offset within the volume
|
|
|
|
// needleOffset: the offset within the needle Data
|
|
|
|
func (n *Needle) ReadNeedleData(r backend.BackendStorageFile, volumeOffset int64, data []byte, needleOffset int64) (count int, err error) {
|
|
|
|
|
|
|
|
sizeToRead := min(int64(len(data)), int64(n.DataSize)-needleOffset)
|
|
|
|
if sizeToRead <= 0 {
|
|
|
|
return 0, io.EOF
|
|
|
|
}
|
|
|
|
startOffset := volumeOffset + NeedleHeaderSize + DataSizeSize + needleOffset
|
|
|
|
|
|
|
|
count, err = r.ReadAt(data[:sizeToRead], startOffset)
|
|
|
|
if err != nil {
|
|
|
|
fileSize, _, _ := r.GetStat()
|
|
|
|
glog.Errorf("%s read %d %d size %d at offset %d fileSize %d: %v", r.Name(), n.Id, needleOffset, sizeToRead, volumeOffset, fileSize, err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadNeedleMeta fills all metadata except the n.Data
|
2022-06-05 22:24:02 +00:00
|
|
|
func (n *Needle) ReadNeedleMeta(r backend.BackendStorageFile, offset int64, size Size, version Version) (err error) {
|
2022-10-23 21:00:28 +00:00
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
err = fmt.Errorf("panic occurred: %+v", r)
|
|
|
|
}
|
|
|
|
}()
|
2022-06-05 01:15:39 +00:00
|
|
|
|
|
|
|
bytes := make([]byte, NeedleHeaderSize+DataSizeSize)
|
|
|
|
|
|
|
|
count, err := r.ReadAt(bytes, offset)
|
|
|
|
if count != NeedleHeaderSize+DataSizeSize || err != nil {
|
2022-06-05 22:24:02 +00:00
|
|
|
return err
|
2022-06-05 01:15:39 +00:00
|
|
|
}
|
|
|
|
n.ParseNeedleHeader(bytes)
|
2022-06-05 18:54:04 +00:00
|
|
|
if n.Size != size {
|
|
|
|
if OffsetSize == 4 && offset < int64(MaxPossibleVolumeSize) {
|
2022-06-05 22:24:02 +00:00
|
|
|
return ErrorSizeMismatch
|
2022-06-05 18:54:04 +00:00
|
|
|
}
|
|
|
|
}
|
2022-10-23 21:00:15 +00:00
|
|
|
if !n.Size.IsValid() {
|
|
|
|
return ErrorSizeInvalid
|
|
|
|
}
|
2022-06-05 01:15:39 +00:00
|
|
|
n.DataSize = util.BytesToUint32(bytes[NeedleHeaderSize : NeedleHeaderSize+DataSizeSize])
|
|
|
|
|
|
|
|
startOffset := offset + NeedleHeaderSize + DataSizeSize + int64(n.DataSize)
|
|
|
|
dataSize := GetActualSize(size, version)
|
|
|
|
stopOffset := offset + dataSize
|
|
|
|
metaSize := stopOffset - startOffset
|
|
|
|
metaSlice := make([]byte, int(metaSize))
|
|
|
|
|
|
|
|
count, err = r.ReadAt(metaSlice, startOffset)
|
|
|
|
if err != nil && int64(count) == metaSize {
|
|
|
|
err = nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
2022-06-05 22:24:02 +00:00
|
|
|
return err
|
2022-06-05 01:15:39 +00:00
|
|
|
}
|
|
|
|
var index int
|
|
|
|
index, err = n.readNeedleDataVersion2NonData(metaSlice)
|
|
|
|
|
2022-06-05 22:24:02 +00:00
|
|
|
n.Checksum = CRC(util.BytesToUint32(metaSlice[index : index+NeedleChecksumSize]))
|
2022-06-05 01:15:39 +00:00
|
|
|
if version == Version3 {
|
|
|
|
n.AppendAtNs = util.BytesToUint64(metaSlice[index+NeedleChecksumSize : index+NeedleChecksumSize+TimestampSize])
|
|
|
|
}
|
|
|
|
|
2022-06-05 22:24:02 +00:00
|
|
|
return err
|
2022-06-05 01:15:39 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func min(x, y int64) int64 {
|
|
|
|
if x < y {
|
|
|
|
return x
|
|
|
|
}
|
|
|
|
return y
|
|
|
|
}
|