2017-02-07 21:33:23 +00:00
|
|
|
package digest
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto"
|
|
|
|
"fmt"
|
|
|
|
"hash"
|
|
|
|
"io"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Algorithm identifies and implementation of a digester by an identifier.
|
|
|
|
// Note the that this defines both the hash algorithm used and the string
|
|
|
|
// encoding.
|
|
|
|
type Algorithm string
|
|
|
|
|
|
|
|
// supported digest types
|
|
|
|
const (
|
|
|
|
SHA256 Algorithm = "sha256" // sha256 with hex encoding
|
|
|
|
SHA384 Algorithm = "sha384" // sha384 with hex encoding
|
|
|
|
SHA512 Algorithm = "sha512" // sha512 with hex encoding
|
|
|
|
|
|
|
|
// Canonical is the primary digest algorithm used with the distribution
|
|
|
|
// project. Other digests may be used but this one is the primary storage
|
|
|
|
// digest.
|
|
|
|
Canonical = SHA256
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// TODO(stevvooe): Follow the pattern of the standard crypto package for
|
|
|
|
// registration of digests. Effectively, we are a registerable set and
|
|
|
|
// common symbol access.
|
|
|
|
|
|
|
|
// algorithms maps values to hash.Hash implementations. Other algorithms
|
|
|
|
// may be available but they cannot be calculated by the digest package.
|
|
|
|
algorithms = map[Algorithm]crypto.Hash{
|
|
|
|
SHA256: crypto.SHA256,
|
|
|
|
SHA384: crypto.SHA384,
|
|
|
|
SHA512: crypto.SHA512,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// Available returns true if the digest type is available for use. If this
|
2017-07-06 14:28:13 +00:00
|
|
|
// returns false, Digester and Hash will return nil.
|
2017-02-07 21:33:23 +00:00
|
|
|
func (a Algorithm) Available() bool {
|
|
|
|
h, ok := algorithms[a]
|
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// check availability of the hash, as well
|
|
|
|
return h.Available()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a Algorithm) String() string {
|
|
|
|
return string(a)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Size returns number of bytes returned by the hash.
|
|
|
|
func (a Algorithm) Size() int {
|
|
|
|
h, ok := algorithms[a]
|
|
|
|
if !ok {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return h.Size()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set implemented to allow use of Algorithm as a command line flag.
|
|
|
|
func (a *Algorithm) Set(value string) error {
|
|
|
|
if value == "" {
|
|
|
|
*a = Canonical
|
|
|
|
} else {
|
|
|
|
// just do a type conversion, support is queried with Available.
|
|
|
|
*a = Algorithm(value)
|
|
|
|
}
|
|
|
|
|
2017-07-06 14:28:13 +00:00
|
|
|
if !a.Available() {
|
|
|
|
return ErrDigestUnsupported
|
|
|
|
}
|
|
|
|
|
2017-02-07 21:33:23 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-07-06 14:28:13 +00:00
|
|
|
// Digester returns a new digester for the specified algorithm. If the algorithm
|
2017-02-07 21:33:23 +00:00
|
|
|
// does not have a digester implementation, nil will be returned. This can be
|
2017-07-06 14:28:13 +00:00
|
|
|
// checked by calling Available before calling Digester.
|
|
|
|
func (a Algorithm) Digester() Digester {
|
2017-02-07 21:33:23 +00:00
|
|
|
return &digester{
|
|
|
|
alg: a,
|
|
|
|
hash: a.Hash(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hash returns a new hash as used by the algorithm. If not available, the
|
|
|
|
// method will panic. Check Algorithm.Available() before calling.
|
|
|
|
func (a Algorithm) Hash() hash.Hash {
|
|
|
|
if !a.Available() {
|
2017-07-06 14:28:13 +00:00
|
|
|
// Empty algorithm string is invalid
|
|
|
|
if a == "" {
|
|
|
|
panic(fmt.Sprintf("empty digest algorithm, validate before calling Algorithm.Hash()"))
|
|
|
|
}
|
|
|
|
|
2017-02-07 21:33:23 +00:00
|
|
|
// NOTE(stevvooe): A missing hash is usually a programming error that
|
|
|
|
// must be resolved at compile time. We don't import in the digest
|
|
|
|
// package to allow users to choose their hash implementation (such as
|
|
|
|
// when using stevvooe/resumable or a hardware accelerated package).
|
|
|
|
//
|
|
|
|
// Applications that may want to resolve the hash at runtime should
|
|
|
|
// call Algorithm.Available before call Algorithm.Hash().
|
|
|
|
panic(fmt.Sprintf("%v not available (make sure it is imported)", a))
|
|
|
|
}
|
|
|
|
|
|
|
|
return algorithms[a].New()
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromReader returns the digest of the reader using the algorithm.
|
|
|
|
func (a Algorithm) FromReader(rd io.Reader) (Digest, error) {
|
2017-07-06 14:28:13 +00:00
|
|
|
digester := a.Digester()
|
2017-02-07 21:33:23 +00:00
|
|
|
|
|
|
|
if _, err := io.Copy(digester.Hash(), rd); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return digester.Digest(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromBytes digests the input and returns a Digest.
|
|
|
|
func (a Algorithm) FromBytes(p []byte) Digest {
|
2017-07-06 14:28:13 +00:00
|
|
|
digester := a.Digester()
|
2017-02-07 21:33:23 +00:00
|
|
|
|
|
|
|
if _, err := digester.Hash().Write(p); err != nil {
|
|
|
|
// Writes to a Hash should never fail. None of the existing
|
|
|
|
// hash implementations in the stdlib or hashes vendored
|
|
|
|
// here can return errors from Write. Having a panic in this
|
|
|
|
// condition instead of having FromBytes return an error value
|
|
|
|
// avoids unnecessary error handling paths in all callers.
|
|
|
|
panic("write to hash function returned error: " + err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return digester.Digest()
|
|
|
|
}
|
|
|
|
|
2017-07-06 14:28:13 +00:00
|
|
|
// FromString digests the string input and returns a Digest.
|
|
|
|
func (a Algorithm) FromString(s string) Digest {
|
|
|
|
return a.FromBytes([]byte(s))
|
2017-02-07 21:33:23 +00:00
|
|
|
}
|