2016-08-31 17:18:51 +00:00
|
|
|
package restic
|
|
|
|
|
|
|
|
import (
|
2016-09-01 19:37:59 +00:00
|
|
|
"crypto/rand"
|
2016-08-31 17:18:51 +00:00
|
|
|
"encoding/hex"
|
|
|
|
"encoding/json"
|
2019-03-22 19:30:29 +00:00
|
|
|
"fmt"
|
2016-09-01 19:37:59 +00:00
|
|
|
"io"
|
2016-08-31 17:18:51 +00:00
|
|
|
|
2017-07-23 12:21:03 +00:00
|
|
|
"github.com/restic/restic/internal/errors"
|
2020-03-19 10:27:19 +00:00
|
|
|
|
|
|
|
"github.com/minio/sha256-simd"
|
2016-08-31 17:18:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Hash returns the ID for data.
|
|
|
|
func Hash(data []byte) ID {
|
|
|
|
return sha256.Sum256(data)
|
|
|
|
}
|
|
|
|
|
2016-09-01 19:37:59 +00:00
|
|
|
// idSize contains the size of an ID, in bytes.
|
|
|
|
const idSize = sha256.Size
|
2016-08-31 17:18:51 +00:00
|
|
|
|
|
|
|
// ID references content within a repository.
|
2016-09-01 19:37:59 +00:00
|
|
|
type ID [idSize]byte
|
2016-08-31 17:18:51 +00:00
|
|
|
|
|
|
|
// ParseID converts the given string to an ID.
|
|
|
|
func ParseID(s string) (ID, error) {
|
|
|
|
b, err := hex.DecodeString(s)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return ID{}, errors.Wrap(err, "hex.DecodeString")
|
|
|
|
}
|
|
|
|
|
2016-09-01 19:37:59 +00:00
|
|
|
if len(b) != idSize {
|
2016-08-31 17:18:51 +00:00
|
|
|
return ID{}, errors.New("invalid length for hash")
|
|
|
|
}
|
|
|
|
|
|
|
|
id := ID{}
|
|
|
|
copy(id[:], b)
|
|
|
|
|
|
|
|
return id, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (id ID) String() string {
|
|
|
|
return hex.EncodeToString(id[:])
|
|
|
|
}
|
|
|
|
|
2017-02-08 23:43:10 +00:00
|
|
|
// NewRandomID returns a randomly generated ID. When reading from rand fails,
|
2016-09-01 19:37:59 +00:00
|
|
|
// the function panics.
|
|
|
|
func NewRandomID() ID {
|
|
|
|
id := ID{}
|
|
|
|
_, err := io.ReadFull(rand.Reader, id[:])
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
2016-08-31 17:18:51 +00:00
|
|
|
const shortStr = 4
|
|
|
|
|
|
|
|
// Str returns the shortened string version of id.
|
|
|
|
func (id *ID) Str() string {
|
|
|
|
if id == nil {
|
|
|
|
return "[nil]"
|
|
|
|
}
|
|
|
|
|
|
|
|
if id.IsNull() {
|
|
|
|
return "[null]"
|
|
|
|
}
|
|
|
|
|
|
|
|
return hex.EncodeToString(id[:shortStr])
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsNull returns true iff id only consists of null bytes.
|
|
|
|
func (id ID) IsNull() bool {
|
|
|
|
var nullID ID
|
|
|
|
|
|
|
|
return id == nullID
|
|
|
|
}
|
|
|
|
|
|
|
|
// Equal compares an ID to another other.
|
|
|
|
func (id ID) Equal(other ID) bool {
|
|
|
|
return id == other
|
|
|
|
}
|
|
|
|
|
|
|
|
// EqualString compares this ID to another one, given as a string.
|
|
|
|
func (id ID) EqualString(other string) (bool, error) {
|
|
|
|
s, err := hex.DecodeString(other)
|
|
|
|
if err != nil {
|
|
|
|
return false, errors.Wrap(err, "hex.DecodeString")
|
|
|
|
}
|
|
|
|
|
|
|
|
id2 := ID{}
|
|
|
|
copy(id2[:], s)
|
|
|
|
|
|
|
|
return id == id2, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON returns the JSON encoding of id.
|
|
|
|
func (id ID) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(id.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON parses the JSON-encoded data and stores the result in id.
|
|
|
|
func (id *ID) UnmarshalJSON(b []byte) error {
|
2019-03-22 19:30:29 +00:00
|
|
|
// check string length
|
|
|
|
if len(b) < 2 {
|
|
|
|
return fmt.Errorf("invalid ID: %q", b)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(b)%2 != 0 {
|
|
|
|
return fmt.Errorf("invalid ID length: %q", b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check string delimiters
|
|
|
|
if b[0] != '"' && b[0] != '\'' {
|
|
|
|
return fmt.Errorf("invalid start of string: %q", b[0])
|
|
|
|
}
|
|
|
|
|
|
|
|
last := len(b) - 1
|
|
|
|
if b[0] != b[last] {
|
|
|
|
return fmt.Errorf("starting string delimiter (%q) does not match end (%q)", b[0], b[last])
|
|
|
|
}
|
|
|
|
|
|
|
|
// strip JSON string delimiters
|
|
|
|
b = b[1:last]
|
|
|
|
|
|
|
|
if len(b) != 2*len(id) {
|
|
|
|
return fmt.Errorf("invalid length for ID")
|
2016-08-31 17:18:51 +00:00
|
|
|
}
|
|
|
|
|
2019-03-22 19:30:29 +00:00
|
|
|
_, err := hex.Decode(id[:], b)
|
2016-08-31 17:18:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "hex.Decode")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2017-01-22 16:53:00 +00:00
|
|
|
|
|
|
|
// IDFromHash returns the ID for the hash.
|
|
|
|
func IDFromHash(hash []byte) (id ID) {
|
|
|
|
if len(hash) != idSize {
|
|
|
|
panic("invalid hash type, not enough/too many bytes")
|
|
|
|
}
|
|
|
|
|
|
|
|
copy(id[:], hash)
|
|
|
|
return id
|
|
|
|
}
|