|
@ -28,7 +28,7 @@ func HasChunkManifest(chunks []*filer_pb.FileChunk) bool { |
|
|
|
|
|
|
|
|
func SeparateManifestChunks(chunks []*filer_pb.FileChunk) (manifestChunks, nonManifestChunks []*filer_pb.FileChunk) { |
|
|
func SeparateManifestChunks(chunks []*filer_pb.FileChunk) (manifestChunks, nonManifestChunks []*filer_pb.FileChunk) { |
|
|
for _, c := range chunks { |
|
|
for _, c := range chunks { |
|
|
if !c.IsChunkManifest { |
|
|
|
|
|
|
|
|
if c.IsChunkManifest { |
|
|
manifestChunks = append(manifestChunks, c) |
|
|
manifestChunks = append(manifestChunks, c) |
|
|
} else { |
|
|
} else { |
|
|
nonManifestChunks = append(nonManifestChunks, c) |
|
|
nonManifestChunks = append(nonManifestChunks, c) |
|
@ -37,7 +37,7 @@ func SeparateManifestChunks(chunks []*filer_pb.FileChunk) (manifestChunks, nonMa |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
func ResolveChunkManifest(lookupFileIdFn LookupFileIdFunctionType, chunks []*filer_pb.FileChunk) (dataChunks, manifestChunks []*filer_pb.FileChunk, manefestResolveErr error) { |
|
|
|
|
|
|
|
|
func ResolveChunkManifest(lookupFileIdFn LookupFileIdFunctionType, chunks []*filer_pb.FileChunk) (dataChunks, manifestChunks []*filer_pb.FileChunk, manifestResolveErr error) { |
|
|
// TODO maybe parallel this
|
|
|
// TODO maybe parallel this
|
|
|
for _, chunk := range chunks { |
|
|
for _, chunk := range chunks { |
|
|
if !chunk.IsChunkManifest { |
|
|
if !chunk.IsChunkManifest { |
|
@ -45,19 +45,14 @@ func ResolveChunkManifest(lookupFileIdFn LookupFileIdFunctionType, chunks []*fil |
|
|
continue |
|
|
continue |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
// IsChunkManifest
|
|
|
|
|
|
data, err := fetchChunk(lookupFileIdFn, chunk.FileId, chunk.CipherKey, chunk.IsCompressed) |
|
|
|
|
|
|
|
|
resolvedChunks, err := ResolveOneChunkManifest(lookupFileIdFn, chunk) |
|
|
if err != nil { |
|
|
if err != nil { |
|
|
return chunks, nil, fmt.Errorf("fail to read manifest %s: %v", chunk.FileId, err) |
|
|
|
|
|
} |
|
|
|
|
|
m := &filer_pb.FileChunkManifest{} |
|
|
|
|
|
if err := proto.Unmarshal(data, m); err != nil { |
|
|
|
|
|
return chunks, nil, fmt.Errorf("fail to unmarshal manifest %s: %v", chunk.FileId, err) |
|
|
|
|
|
|
|
|
return chunks, nil, err |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
manifestChunks = append(manifestChunks, chunk) |
|
|
manifestChunks = append(manifestChunks, chunk) |
|
|
// recursive
|
|
|
// recursive
|
|
|
filer_pb.AfterEntryDeserialization(m.Chunks) |
|
|
|
|
|
dchunks, mchunks, subErr := ResolveChunkManifest(lookupFileIdFn, m.Chunks) |
|
|
|
|
|
|
|
|
dchunks, mchunks, subErr := ResolveChunkManifest(lookupFileIdFn, resolvedChunks) |
|
|
if subErr != nil { |
|
|
if subErr != nil { |
|
|
return chunks, nil, subErr |
|
|
return chunks, nil, subErr |
|
|
} |
|
|
} |
|
@ -67,6 +62,26 @@ func ResolveChunkManifest(lookupFileIdFn LookupFileIdFunctionType, chunks []*fil |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
func ResolveOneChunkManifest(lookupFileIdFn LookupFileIdFunctionType, chunk *filer_pb.FileChunk) (dataChunks []*filer_pb.FileChunk, manifestResolveErr error) { |
|
|
|
|
|
if !chunk.IsChunkManifest { |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// IsChunkManifest
|
|
|
|
|
|
data, err := fetchChunk(lookupFileIdFn, chunk.GetFileIdString(), chunk.CipherKey, chunk.IsCompressed) |
|
|
|
|
|
if err != nil { |
|
|
|
|
|
return nil, fmt.Errorf("fail to read manifest %s: %v", chunk.GetFileIdString(), err) |
|
|
|
|
|
} |
|
|
|
|
|
m := &filer_pb.FileChunkManifest{} |
|
|
|
|
|
if err := proto.Unmarshal(data, m); err != nil { |
|
|
|
|
|
return nil, fmt.Errorf("fail to unmarshal manifest %s: %v", chunk.GetFileIdString(), err) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// recursive
|
|
|
|
|
|
filer_pb.AfterEntryDeserialization(m.Chunks) |
|
|
|
|
|
return m.Chunks, nil |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
// TODO fetch from cache for weed mount?
|
|
|
// TODO fetch from cache for weed mount?
|
|
|
func fetchChunk(lookupFileIdFn LookupFileIdFunctionType, fileId string, cipherKey []byte, isGzipped bool) ([]byte, error) { |
|
|
func fetchChunk(lookupFileIdFn LookupFileIdFunctionType, fileId string, cipherKey []byte, isGzipped bool) ([]byte, error) { |
|
|
urlString, err := lookupFileIdFn(fileId) |
|
|
urlString, err := lookupFileIdFn(fileId) |
|
|