git-lfs/pointer/pointer.go

166 lines
3.3 KiB
Go
Raw Normal View History

package pointer
2013-09-27 17:08:19 +00:00
import (
2014-07-24 21:23:02 +00:00
"bufio"
2013-09-27 17:08:19 +00:00
"bytes"
"errors"
2014-07-24 22:02:27 +00:00
"fmt"
2015-03-19 19:30:55 +00:00
"github.com/github/git-lfs/lfs"
2013-09-27 17:08:19 +00:00
"io"
"regexp"
2014-07-24 21:23:02 +00:00
"strconv"
"strings"
2013-09-27 17:08:19 +00:00
)
var (
v1Aliases = []string{
2015-04-28 19:29:45 +00:00
"http://git-media.io/v/2", // alpha
"https://hawser.github.com/spec/v1", // pre-release
"https://git-lfs.github.com/spec/v1", // public launch
}
latest = "https://git-lfs.github.com/spec/v1"
oidType = "sha256"
oidRE = regexp.MustCompile(`\A[0-9a-fA-F]{64}`)
template = `version %s
2014-08-14 17:15:22 +00:00
oid sha256:%s
size %d
2014-07-24 22:02:27 +00:00
`
2015-03-19 19:30:55 +00:00
matcherRE = regexp.MustCompile("git-media|hawser|git-lfs")
pointerKeys = []string{"version", "oid", "size"}
)
type Pointer struct {
Version string
Oid string
Size int64
2014-07-24 20:49:47 +00:00
OidType string
}
2013-09-27 17:08:53 +00:00
func NewPointer(oid string, size int64) *Pointer {
2014-07-24 21:23:02 +00:00
return &Pointer{latest, oid, size, oidType}
}
2015-03-22 16:01:26 +00:00
func (p *Pointer) Smudge(writer io.Writer, workingfile string, cb lfs.CopyCallback) error {
return Smudge(writer, p, workingfile, cb)
}
func (p *Pointer) Encode(writer io.Writer) (int, error) {
return Encode(writer, p)
}
2014-09-18 18:38:10 +00:00
func (p *Pointer) Encoded() string {
return fmt.Sprintf(template, latest, p.Oid, p.Size)
}
func Encode(writer io.Writer, pointer *Pointer) (int, error) {
2014-09-18 18:38:10 +00:00
return writer.Write([]byte(pointer.Encoded()))
2013-09-27 17:08:19 +00:00
}
func Decode(reader io.Reader) (*Pointer, error) {
_, p, err := DecodeFrom(reader)
return p, err
}
func DecodeFrom(reader io.Reader) ([]byte, *Pointer, error) {
buf := make([]byte, 512)
2013-10-04 13:48:02 +00:00
written, err := reader.Read(buf)
if err != nil {
return buf, nil, err
2013-10-04 13:48:02 +00:00
}
2013-09-27 17:08:19 +00:00
output := buf[0:written]
p, err := decodeKV(bytes.TrimSpace(output))
return output, p, err
}
2013-09-27 17:08:19 +00:00
func verifyVersion(version string) error {
if len(version) == 0 {
return errors.New("Missing version")
}
for _, v := range v1Aliases {
if v == version {
return nil
}
}
return errors.New("Invalid version: " + version)
2014-07-24 20:49:47 +00:00
}
2014-08-14 17:15:22 +00:00
func decodeKV(data []byte) (*Pointer, error) {
parsed, err := decodeKVData(data)
2014-07-24 21:23:02 +00:00
if err != nil {
return nil, err
}
if err := verifyVersion(parsed["version"]); err != nil {
return nil, err
2014-07-24 21:23:02 +00:00
}
2014-08-14 17:15:22 +00:00
oidValue, ok := parsed["oid"]
2014-07-24 21:23:02 +00:00
if !ok {
return nil, errors.New("Invalid Oid")
}
oidParts := strings.SplitN(oidValue, ":", 2)
if len(oidParts) != 2 {
return nil, errors.New("Invalid Oid type in" + oidValue)
}
if oidParts[0] != oidType {
return nil, errors.New("Invalid Oid type: " + oidParts[0])
}
oid := oidParts[1]
var size int64
2014-08-14 17:15:22 +00:00
sizeStr, ok := parsed["size"]
2014-07-24 21:23:02 +00:00
if !ok {
return nil, errors.New("Invalid Oid")
} else {
size, err = strconv.ParseInt(sizeStr, 10, 0)
if err != nil {
return nil, errors.New("Invalid size: " + sizeStr)
}
}
return NewPointer(oid, size), nil
}
2014-08-14 17:15:22 +00:00
func decodeKVData(data []byte) (map[string]string, error) {
m := make(map[string]string)
2014-08-14 20:23:19 +00:00
2015-01-30 16:56:12 +00:00
if !matcherRE.Match(data) {
2015-03-19 19:30:55 +00:00
return m, fmt.Errorf("Not a valid Git LFS pointer file.")
2014-08-14 20:23:19 +00:00
}
2014-08-14 17:15:22 +00:00
scanner := bufio.NewScanner(bytes.NewBuffer(data))
2014-08-14 20:23:19 +00:00
line := 0
2014-08-14 20:30:33 +00:00
numKeys := len(pointerKeys)
2014-08-14 17:15:22 +00:00
for scanner.Scan() {
2014-08-14 20:30:33 +00:00
text := scanner.Text()
if len(text) == 0 {
continue
}
parts := strings.SplitN(text, " ", 2)
key := parts[0]
if numKeys <= line {
return m, fmt.Errorf("Extra line: %s", text)
}
if expected := pointerKeys[line]; key != expected {
return m, fmt.Errorf("Expected key %s, got %s", expected, key)
}
2014-08-14 20:23:19 +00:00
line += 1
if len(parts) < 2 {
2014-08-14 20:30:33 +00:00
return m, fmt.Errorf("Error reading line %d: %s", line, text)
2014-08-14 17:15:22 +00:00
}
2014-08-14 20:30:33 +00:00
m[key] = parts[1]
2014-08-14 17:15:22 +00:00
}
return m, scanner.Err()
}