commit
4fe6593fbe
1 changed files with 41 additions and 0 deletions
|
@ -8,6 +8,7 @@ import (
|
||||||
"restic"
|
"restic"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"restic/debug"
|
||||||
"restic/errors"
|
"restic/errors"
|
||||||
|
|
||||||
"restic/crypto"
|
"restic/crypto"
|
||||||
|
@ -189,14 +190,45 @@ func readHeaderLength(rd io.ReaderAt, size int64) (uint32, error) {
|
||||||
|
|
||||||
const maxHeaderSize = 16 * 1024 * 1024
|
const maxHeaderSize = 16 * 1024 * 1024
|
||||||
|
|
||||||
|
// we require at least one entry in the header, and one blob for a pack file
|
||||||
|
var minFileSize = entrySize + crypto.Extension
|
||||||
|
|
||||||
// readHeader reads the header at the end of rd. size is the length of the
|
// readHeader reads the header at the end of rd. size is the length of the
|
||||||
// whole data accessible in rd.
|
// whole data accessible in rd.
|
||||||
func readHeader(rd io.ReaderAt, size int64) ([]byte, error) {
|
func readHeader(rd io.ReaderAt, size int64) ([]byte, error) {
|
||||||
|
debug.Log("size: %v", size)
|
||||||
|
if size == 0 {
|
||||||
|
err := InvalidFileError{Message: "file is empty"}
|
||||||
|
return nil, errors.Wrap(err, "readHeader")
|
||||||
|
}
|
||||||
|
|
||||||
|
if size < int64(minFileSize) {
|
||||||
|
err := InvalidFileError{Message: "file is too small"}
|
||||||
|
return nil, errors.Wrap(err, "readHeader")
|
||||||
|
}
|
||||||
|
|
||||||
hl, err := readHeaderLength(rd, size)
|
hl, err := readHeaderLength(rd, size)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
debug.Log("header length: %v", size)
|
||||||
|
|
||||||
|
if hl == 0 {
|
||||||
|
err := InvalidFileError{Message: "header length is zero"}
|
||||||
|
return nil, errors.Wrap(err, "readHeader")
|
||||||
|
}
|
||||||
|
|
||||||
|
if hl < crypto.Extension {
|
||||||
|
err := InvalidFileError{Message: "header length is too small"}
|
||||||
|
return nil, errors.Wrap(err, "readHeader")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hl-crypto.Extension)%uint32(entrySize) != 0 {
|
||||||
|
err := InvalidFileError{Message: "header length is invalid"}
|
||||||
|
return nil, errors.Wrap(err, "readHeader")
|
||||||
|
}
|
||||||
|
|
||||||
if int64(hl) > size-int64(binary.Size(hl)) {
|
if int64(hl) > size-int64(binary.Size(hl)) {
|
||||||
return nil, errors.New("header is larger than file")
|
return nil, errors.New("header is larger than file")
|
||||||
}
|
}
|
||||||
|
@ -218,6 +250,15 @@ func readHeader(rd io.ReaderAt, size int64) ([]byte, error) {
|
||||||
return buf, nil
|
return buf, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InvalidFileError is return when a file is found that is not a pack file.
|
||||||
|
type InvalidFileError struct {
|
||||||
|
Message string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e InvalidFileError) Error() string {
|
||||||
|
return e.Message
|
||||||
|
}
|
||||||
|
|
||||||
// List returns the list of entries found in a pack file.
|
// List returns the list of entries found in a pack file.
|
||||||
func List(k *crypto.Key, rd io.ReaderAt, size int64) (entries []restic.Blob, err error) {
|
func List(k *crypto.Key, rd io.ReaderAt, size int64) (entries []restic.Blob, err error) {
|
||||||
buf, err := readHeader(rd, size)
|
buf, err := readHeader(rd, size)
|
||||||
|
|
Loading…
Reference in a new issue