mirror of https://github.com/go-gitea/gitea.git
135 lines
3.0 KiB
Go
135 lines
3.0 KiB
Go
// Copyright 2023 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package git
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/hex"
|
|
"errors"
|
|
"fmt"
|
|
"hash"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
type ObjectID interface {
|
|
String() string
|
|
IsZero() bool
|
|
RawValue() []byte
|
|
Type() ObjectFormat
|
|
}
|
|
|
|
type Sha1Hash [20]byte
|
|
|
|
func (h *Sha1Hash) String() string {
|
|
return hex.EncodeToString(h[:])
|
|
}
|
|
|
|
func (h *Sha1Hash) IsZero() bool {
|
|
empty := Sha1Hash{}
|
|
return bytes.Equal(empty[:], h[:])
|
|
}
|
|
func (h *Sha1Hash) RawValue() []byte { return h[:] }
|
|
func (*Sha1Hash) Type() ObjectFormat { return Sha1ObjectFormat }
|
|
|
|
var _ ObjectID = &Sha1Hash{}
|
|
|
|
// EmptyObjectID creates a new ObjectID from an object format hash name
|
|
func EmptyObjectID(objectFormatName string) (ObjectID, error) {
|
|
objectFormat := ObjectFormatFromName(objectFormatName)
|
|
if objectFormat != nil {
|
|
return objectFormat.EmptyObjectID(), nil
|
|
}
|
|
|
|
return nil, errors.New("unsupported hash type")
|
|
}
|
|
|
|
func IDFromRaw(h ObjectFormat, b []byte) (ObjectID, error) {
|
|
if len(b) != h.FullLength()/2 {
|
|
return h.EmptyObjectID(), fmt.Errorf("length must be %d: %v", h.FullLength(), b)
|
|
}
|
|
return h.MustID(b), nil
|
|
}
|
|
|
|
func MustIDFromString(h ObjectFormat, s string) ObjectID {
|
|
b, _ := hex.DecodeString(s)
|
|
return h.MustID(b)
|
|
}
|
|
|
|
func genericIDFromString(h ObjectFormat, s string) (ObjectID, error) {
|
|
s = strings.TrimSpace(s)
|
|
if len(s) != h.FullLength() {
|
|
return h.EmptyObjectID(), fmt.Errorf("length must be %d: %s", h.FullLength(), s)
|
|
}
|
|
b, err := hex.DecodeString(s)
|
|
if err != nil {
|
|
return h.EmptyObjectID(), err
|
|
}
|
|
return h.NewID(b)
|
|
}
|
|
|
|
func IDFromString(hexHash string) (ObjectID, error) {
|
|
for _, objectFormat := range SupportedObjectFormats {
|
|
if len(hexHash) == objectFormat.FullLength() {
|
|
return objectFormat.NewIDFromString(hexHash)
|
|
}
|
|
}
|
|
|
|
return nil, fmt.Errorf("invalid hash hex string: '%s' len: %d", hexHash, len(hexHash))
|
|
}
|
|
|
|
func IsEmptyCommitID(commitID string) bool {
|
|
if commitID == "" {
|
|
return true
|
|
}
|
|
|
|
id, err := IDFromString(commitID)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
|
|
return id.IsZero()
|
|
}
|
|
|
|
// HasherInterface is a struct that will generate a Hash
|
|
type HasherInterface interface {
|
|
hash.Hash
|
|
|
|
HashSum() ObjectID
|
|
}
|
|
|
|
type Sha1Hasher struct {
|
|
hash.Hash
|
|
}
|
|
|
|
// ComputeBlobHash compute the hash for a given blob content
|
|
func ComputeBlobHash(hashType ObjectFormat, content []byte) ObjectID {
|
|
return ComputeHash(hashType, ObjectBlob, content)
|
|
}
|
|
|
|
// ComputeHash compute the hash for a given ObjectType and content
|
|
func ComputeHash(hashType ObjectFormat, t ObjectType, content []byte) ObjectID {
|
|
h := hashType.NewHasher()
|
|
_, _ = h.Write(t.Bytes())
|
|
_, _ = h.Write([]byte(" "))
|
|
_, _ = h.Write([]byte(strconv.FormatInt(int64(len(content)), 10)))
|
|
_, _ = h.Write([]byte{0})
|
|
return h.HashSum()
|
|
}
|
|
|
|
// HashSum generates a SHA1 for the provided hash
|
|
func (h *Sha1Hasher) HashSum() ObjectID {
|
|
var sha1 Sha1Hash
|
|
copy(sha1[:], h.Hash.Sum(nil))
|
|
return &sha1
|
|
}
|
|
|
|
type ErrInvalidSHA struct {
|
|
SHA string
|
|
}
|
|
|
|
func (err ErrInvalidSHA) Error() string {
|
|
return fmt.Sprintf("invalid sha: %s", err.SHA)
|
|
}
|