2014-07-28 19:26:10 +00:00
|
|
|
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"
|
2014-06-04 15:38:16 +00:00
|
|
|
"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"
|
2014-04-16 21:08:58 +00:00
|
|
|
"regexp"
|
2014-07-24 21:23:02 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2013-09-27 17:08:19 +00:00
|
|
|
)
|
|
|
|
|
2014-07-24 20:28:37 +00:00
|
|
|
var (
|
2015-04-22 15:44:06 +00:00
|
|
|
v1Aliases = []string{
|
2015-04-28 19:29:45 +00:00
|
|
|
"http://git-media.io/v/2", // alpha
|
2015-04-22 15:44:06 +00:00
|
|
|
"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")
|
2014-09-23 16:06:29 +00:00
|
|
|
pointerKeys = []string{"version", "oid", "size"}
|
2014-07-24 20:28:37 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Pointer struct {
|
|
|
|
Version string
|
|
|
|
Oid string
|
|
|
|
Size int64
|
2014-07-24 20:49:47 +00:00
|
|
|
OidType string
|
2014-07-24 20:28:37 +00:00
|
|
|
}
|
2013-09-27 17:08:53 +00:00
|
|
|
|
2014-07-24 20:34:19 +00:00
|
|
|
func NewPointer(oid string, size int64) *Pointer {
|
2014-07-24 21:23:02 +00:00
|
|
|
return &Pointer{latest, oid, size, oidType}
|
2014-07-24 20:34:19 +00:00
|
|
|
}
|
|
|
|
|
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)
|
2014-08-06 21:40:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2014-07-24 20:34:19 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2014-07-24 20:28:37 +00:00
|
|
|
func Decode(reader io.Reader) (*Pointer, error) {
|
2015-04-28 21:37:17 +00:00
|
|
|
_, 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 {
|
2015-04-28 21:37:17 +00:00
|
|
|
return buf, nil, err
|
2013-10-04 13:48:02 +00:00
|
|
|
}
|
2013-09-27 17:08:19 +00:00
|
|
|
|
2015-04-28 21:37:17 +00:00
|
|
|
output := buf[0:written]
|
|
|
|
p, err := decodeKV(bytes.TrimSpace(output))
|
|
|
|
return output, p, err
|
2015-04-22 15:44:06 +00:00
|
|
|
}
|
2013-09-27 17:08:19 +00:00
|
|
|
|
2015-04-22 15:44:06 +00:00
|
|
|
func verifyVersion(version string) error {
|
|
|
|
if len(version) == 0 {
|
|
|
|
return errors.New("Missing version")
|
2014-06-04 16:01:45 +00:00
|
|
|
}
|
2015-04-22 15:44:06 +00:00
|
|
|
|
|
|
|
for _, v := range v1Aliases {
|
|
|
|
if v == version {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors.New("Invalid version: " + version)
|
2014-07-24 20:49:47 +00:00
|
|
|
}
|
2014-06-04 16:01:45 +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
|
|
|
|
}
|
|
|
|
|
2015-04-22 15:44:06 +00:00
|
|
|
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()
|
|
|
|
}
|