2018-05-13 15:11:26 +08:00
|
|
|
package filer2
|
|
|
|
|
2018-05-21 08:06:09 +08:00
|
|
|
import (
|
2018-09-10 07:26:11 +08:00
|
|
|
"fmt"
|
|
|
|
"hash/fnv"
|
2020-03-27 19:35:31 +08:00
|
|
|
"math"
|
2018-05-28 02:52:26 +08:00
|
|
|
"sort"
|
2018-11-19 03:51:38 +08:00
|
|
|
"sync"
|
2018-05-21 08:08:54 +08:00
|
|
|
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
|
2018-05-21 08:06:09 +08:00
|
|
|
)
|
2018-05-16 15:08:44 +08:00
|
|
|
|
2018-05-21 08:06:09 +08:00
|
|
|
func TotalSize(chunks []*filer_pb.FileChunk) (size uint64) {
|
2018-05-13 15:11:26 +08:00
|
|
|
for _, c := range chunks {
|
|
|
|
t := uint64(c.Offset + int64(c.Size))
|
|
|
|
if size < t {
|
|
|
|
size = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-04-08 23:12:00 +08:00
|
|
|
func ETag(entry *filer_pb.Entry) (etag string) {
|
|
|
|
if entry.Attributes == nil || entry.Attributes.Md5 == nil {
|
2020-04-09 00:13:26 +08:00
|
|
|
return ETagChunks(entry.Chunks)
|
2020-04-08 23:12:00 +08:00
|
|
|
}
|
|
|
|
return fmt.Sprintf("%x", entry.Attributes.Md5)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ETagEntry(entry *Entry) (etag string) {
|
|
|
|
if entry.Attr.Md5 == nil {
|
2020-04-09 00:13:26 +08:00
|
|
|
return ETagChunks(entry.Chunks)
|
2020-04-08 23:12:00 +08:00
|
|
|
}
|
|
|
|
return fmt.Sprintf("%x", entry.Attr.Md5)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ETagChunks(chunks []*filer_pb.FileChunk) (etag string) {
|
2018-09-10 07:25:43 +08:00
|
|
|
if len(chunks) == 1 {
|
|
|
|
return chunks[0].ETag
|
|
|
|
}
|
|
|
|
|
|
|
|
h := fnv.New32a()
|
|
|
|
for _, c := range chunks {
|
|
|
|
h.Write([]byte(c.ETag))
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%x", h.Sum32())
|
|
|
|
}
|
|
|
|
|
2020-07-20 08:59:43 +08:00
|
|
|
func CompactFileChunks(lookupFileIdFn LookupFileIdFunctionType, chunks []*filer_pb.FileChunk) (compacted, garbage []*filer_pb.FileChunk) {
|
2018-05-21 15:00:28 +08:00
|
|
|
|
2020-07-20 08:59:43 +08:00
|
|
|
visibles, _ := NonOverlappingVisibleIntervals(lookupFileIdFn, chunks)
|
2018-05-21 15:00:28 +08:00
|
|
|
|
|
|
|
fileIds := make(map[string]bool)
|
|
|
|
for _, interval := range visibles {
|
|
|
|
fileIds[interval.fileId] = true
|
|
|
|
}
|
|
|
|
for _, chunk := range chunks {
|
2019-06-23 11:04:56 +08:00
|
|
|
if _, found := fileIds[chunk.GetFileIdString()]; found {
|
2018-05-21 15:00:28 +08:00
|
|
|
compacted = append(compacted, chunk)
|
|
|
|
} else {
|
|
|
|
garbage = append(garbage, chunk)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-21 08:06:09 +08:00
|
|
|
return
|
2018-05-13 15:11:26 +08:00
|
|
|
}
|
2018-05-21 08:06:09 +08:00
|
|
|
|
2020-07-20 08:59:43 +08:00
|
|
|
func MinusChunks(lookupFileIdFn LookupFileIdFunctionType, as, bs []*filer_pb.FileChunk) (delta []*filer_pb.FileChunk, err error) {
|
|
|
|
|
|
|
|
aData, aMeta, aErr := ResolveChunkManifest(lookupFileIdFn, as)
|
|
|
|
if aErr != nil {
|
|
|
|
return nil, aErr
|
|
|
|
}
|
|
|
|
bData, bMeta, bErr := ResolveChunkManifest(lookupFileIdFn, bs)
|
|
|
|
if bErr != nil {
|
|
|
|
return nil, bErr
|
|
|
|
}
|
|
|
|
|
|
|
|
delta = append(delta, DoMinusChunks(aData, bData)...)
|
|
|
|
delta = append(delta, DoMinusChunks(aMeta, bMeta)...)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func DoMinusChunks(as, bs []*filer_pb.FileChunk) (delta []*filer_pb.FileChunk) {
|
2018-05-22 18:26:38 +08:00
|
|
|
|
|
|
|
fileIds := make(map[string]bool)
|
2019-06-23 11:04:56 +08:00
|
|
|
for _, interval := range bs {
|
|
|
|
fileIds[interval.GetFileIdString()] = true
|
2018-05-22 18:26:38 +08:00
|
|
|
}
|
2019-06-23 11:04:56 +08:00
|
|
|
for _, chunk := range as {
|
|
|
|
if _, found := fileIds[chunk.GetFileIdString()]; !found {
|
|
|
|
delta = append(delta, chunk)
|
2018-05-22 18:26:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-05-24 16:22:37 +08:00
|
|
|
type ChunkView struct {
|
|
|
|
FileId string
|
|
|
|
Offset int64
|
|
|
|
Size uint64
|
|
|
|
LogicOffset int64
|
2020-04-14 12:58:10 +08:00
|
|
|
ChunkSize uint64
|
2020-03-06 16:49:47 +08:00
|
|
|
CipherKey []byte
|
2020-03-29 04:42:35 +08:00
|
|
|
IsGzipped bool
|
2018-05-24 16:22:37 +08:00
|
|
|
}
|
|
|
|
|
2020-04-14 12:58:10 +08:00
|
|
|
func (cv *ChunkView) IsFullChunk() bool {
|
|
|
|
return cv.Size == cv.ChunkSize
|
|
|
|
}
|
|
|
|
|
2020-07-20 08:59:43 +08:00
|
|
|
func ViewFromChunks(lookupFileIdFn LookupFileIdFunctionType, chunks []*filer_pb.FileChunk, offset int64, size int64) (views []*ChunkView) {
|
2018-05-24 13:28:54 +08:00
|
|
|
|
2020-07-20 08:59:43 +08:00
|
|
|
visibles, _ := NonOverlappingVisibleIntervals(lookupFileIdFn, chunks)
|
2018-12-30 16:51:44 +08:00
|
|
|
|
|
|
|
return ViewFromVisibleIntervals(visibles, offset, size)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-03-22 16:37:46 +08:00
|
|
|
func ViewFromVisibleIntervals(visibles []VisibleInterval, offset int64, size int64) (views []*ChunkView) {
|
2018-05-24 13:28:54 +08:00
|
|
|
|
2020-03-22 16:37:46 +08:00
|
|
|
stop := offset + size
|
2020-03-27 19:35:31 +08:00
|
|
|
if size == math.MaxInt64 {
|
|
|
|
stop = math.MaxInt64
|
|
|
|
}
|
|
|
|
if stop < offset {
|
|
|
|
stop = math.MaxInt64
|
|
|
|
}
|
2018-05-24 13:28:54 +08:00
|
|
|
|
|
|
|
for _, chunk := range visibles {
|
2020-03-09 12:39:33 +08:00
|
|
|
|
2018-05-25 14:19:56 +08:00
|
|
|
if chunk.start <= offset && offset < chunk.stop && offset < stop {
|
2018-05-24 16:22:37 +08:00
|
|
|
views = append(views, &ChunkView{
|
|
|
|
FileId: chunk.fileId,
|
|
|
|
Offset: offset - chunk.start, // offset is the data starting location in this file id
|
|
|
|
Size: uint64(min(chunk.stop, stop) - offset),
|
|
|
|
LogicOffset: offset,
|
2020-04-14 12:58:10 +08:00
|
|
|
ChunkSize: chunk.chunkSize,
|
2020-03-06 16:49:47 +08:00
|
|
|
CipherKey: chunk.cipherKey,
|
2020-03-29 04:42:35 +08:00
|
|
|
IsGzipped: chunk.isGzipped,
|
2018-05-24 13:28:54 +08:00
|
|
|
})
|
|
|
|
offset = min(chunk.stop, stop)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return views
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-01-01 07:10:14 +08:00
|
|
|
func logPrintf(name string, visibles []VisibleInterval) {
|
2018-05-28 02:56:49 +08:00
|
|
|
/*
|
2018-11-23 16:26:15 +08:00
|
|
|
log.Printf("%s len %d", name, len(visibles))
|
|
|
|
for _, v := range visibles {
|
|
|
|
log.Printf("%s: => %+v", name, v)
|
|
|
|
}
|
2018-05-28 02:56:49 +08:00
|
|
|
*/
|
2018-05-13 15:11:26 +08:00
|
|
|
}
|
2018-05-21 08:06:09 +08:00
|
|
|
|
2018-11-19 03:51:38 +08:00
|
|
|
var bufPool = sync.Pool{
|
|
|
|
New: func() interface{} {
|
2018-12-30 16:51:44 +08:00
|
|
|
return new(VisibleInterval)
|
2018-11-19 03:51:38 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2019-01-06 14:23:44 +08:00
|
|
|
func MergeIntoVisibles(visibles, newVisibles []VisibleInterval, chunk *filer_pb.FileChunk) []VisibleInterval {
|
2018-11-19 12:31:39 +08:00
|
|
|
|
2020-06-20 23:15:49 +08:00
|
|
|
newV := newVisibleInterval(chunk.Offset, chunk.Offset+int64(chunk.Size), chunk.GetFileIdString(), chunk.Mtime, chunk.Size, chunk.CipherKey, chunk.IsCompressed)
|
2018-11-19 12:31:39 +08:00
|
|
|
|
|
|
|
length := len(visibles)
|
|
|
|
if length == 0 {
|
|
|
|
return append(visibles, newV)
|
|
|
|
}
|
|
|
|
last := visibles[length-1]
|
|
|
|
if last.stop <= chunk.Offset {
|
|
|
|
return append(visibles, newV)
|
|
|
|
}
|
|
|
|
|
2018-11-22 08:25:13 +08:00
|
|
|
logPrintf(" before", visibles)
|
2018-11-19 02:07:30 +08:00
|
|
|
for _, v := range visibles {
|
|
|
|
if v.start < chunk.Offset && chunk.Offset < v.stop {
|
2020-04-14 12:58:10 +08:00
|
|
|
newVisibles = append(newVisibles, newVisibleInterval(v.start, chunk.Offset, v.fileId, v.modifiedTime, chunk.Size, v.cipherKey, v.isGzipped))
|
2018-05-21 08:06:09 +08:00
|
|
|
}
|
2018-11-19 02:07:30 +08:00
|
|
|
chunkStop := chunk.Offset + int64(chunk.Size)
|
|
|
|
if v.start < chunkStop && chunkStop < v.stop {
|
2020-04-14 12:58:10 +08:00
|
|
|
newVisibles = append(newVisibles, newVisibleInterval(chunkStop, v.stop, v.fileId, v.modifiedTime, chunk.Size, v.cipherKey, v.isGzipped))
|
2018-05-21 08:06:09 +08:00
|
|
|
}
|
2018-11-22 08:25:13 +08:00
|
|
|
if chunkStop <= v.start || v.stop <= chunk.Offset {
|
2018-11-19 02:07:30 +08:00
|
|
|
newVisibles = append(newVisibles, v)
|
|
|
|
}
|
2018-05-21 08:06:09 +08:00
|
|
|
}
|
2018-11-19 12:31:39 +08:00
|
|
|
newVisibles = append(newVisibles, newV)
|
|
|
|
|
2018-11-19 13:59:53 +08:00
|
|
|
logPrintf(" append", newVisibles)
|
|
|
|
|
|
|
|
for i := len(newVisibles) - 1; i >= 0; i-- {
|
|
|
|
if i > 0 && newV.start < newVisibles[i-1].start {
|
2018-11-19 12:31:39 +08:00
|
|
|
newVisibles[i] = newVisibles[i-1]
|
|
|
|
} else {
|
|
|
|
newVisibles[i] = newV
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2018-11-19 13:59:53 +08:00
|
|
|
logPrintf(" sorted", newVisibles)
|
2018-11-19 12:31:39 +08:00
|
|
|
|
2018-11-19 13:24:58 +08:00
|
|
|
return newVisibles
|
2018-11-19 02:07:30 +08:00
|
|
|
}
|
2018-05-21 08:06:09 +08:00
|
|
|
|
2020-07-20 08:59:43 +08:00
|
|
|
// NonOverlappingVisibleIntervals translates the file chunk into VisibleInterval in memory
|
|
|
|
// If the file chunk content is a chunk manifest
|
|
|
|
func NonOverlappingVisibleIntervals(lookupFileIdFn LookupFileIdFunctionType, chunks []*filer_pb.FileChunk) (visibles []VisibleInterval, err error) {
|
|
|
|
|
|
|
|
chunks, _, err = ResolveChunkManifest(lookupFileIdFn, chunks)
|
2018-05-21 08:06:09 +08:00
|
|
|
|
2018-11-19 02:07:30 +08:00
|
|
|
sort.Slice(chunks, func(i, j int) bool {
|
|
|
|
return chunks[i].Mtime < chunks[j].Mtime
|
|
|
|
})
|
2018-05-21 08:06:09 +08:00
|
|
|
|
2019-01-06 14:23:44 +08:00
|
|
|
var newVisibles []VisibleInterval
|
2018-11-19 02:07:30 +08:00
|
|
|
for _, chunk := range chunks {
|
2020-03-09 12:39:33 +08:00
|
|
|
|
2019-01-06 14:23:44 +08:00
|
|
|
newVisibles = MergeIntoVisibles(visibles, newVisibles, chunk)
|
2018-11-19 13:24:58 +08:00
|
|
|
t := visibles[:0]
|
2019-01-06 14:23:44 +08:00
|
|
|
visibles = newVisibles
|
|
|
|
newVisibles = t
|
2018-05-21 08:06:09 +08:00
|
|
|
|
2018-11-19 13:59:53 +08:00
|
|
|
logPrintf("add", visibles)
|
|
|
|
|
|
|
|
}
|
2018-05-21 08:06:09 +08:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// find non-overlapping visible intervals
|
|
|
|
// visible interval map to one file chunk
|
|
|
|
|
2018-12-30 16:51:44 +08:00
|
|
|
type VisibleInterval struct {
|
2018-05-21 08:06:09 +08:00
|
|
|
start int64
|
|
|
|
stop int64
|
|
|
|
modifiedTime int64
|
|
|
|
fileId string
|
2020-04-14 12:58:10 +08:00
|
|
|
chunkSize uint64
|
2020-03-06 16:49:47 +08:00
|
|
|
cipherKey []byte
|
2020-03-09 12:39:33 +08:00
|
|
|
isGzipped bool
|
2018-05-21 08:06:09 +08:00
|
|
|
}
|
|
|
|
|
2020-04-14 12:58:10 +08:00
|
|
|
func newVisibleInterval(start, stop int64, fileId string, modifiedTime int64, chunkSize uint64, cipherKey []byte, isGzipped bool) VisibleInterval {
|
2019-01-01 07:10:14 +08:00
|
|
|
return VisibleInterval{
|
2018-11-19 12:31:39 +08:00
|
|
|
start: start,
|
|
|
|
stop: stop,
|
|
|
|
fileId: fileId,
|
|
|
|
modifiedTime: modifiedTime,
|
2020-04-14 12:58:10 +08:00
|
|
|
chunkSize: chunkSize,
|
2020-03-06 16:49:47 +08:00
|
|
|
cipherKey: cipherKey,
|
2020-03-09 12:39:33 +08:00
|
|
|
isGzipped: isGzipped,
|
2018-11-19 12:31:39 +08:00
|
|
|
}
|
2018-05-21 08:06:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func min(x, y int64) int64 {
|
|
|
|
if x <= y {
|
|
|
|
return x
|
|
|
|
}
|
|
|
|
return y
|
|
|
|
}
|