257 lines
5 KiB
Go
257 lines
5 KiB
Go
package storage
|
|
|
|
import (
|
|
"crypto/sha256"
|
|
"encoding/hex"
|
|
"errors"
|
|
"hash"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/url"
|
|
"os"
|
|
"path"
|
|
|
|
"github.com/fd0/khepri/hashing"
|
|
)
|
|
|
|
const (
|
|
dirMode = 0700
|
|
objectPath = "objects"
|
|
refPath = "refs"
|
|
tempPath = "tmp"
|
|
)
|
|
|
|
type Repository interface {
|
|
Put(reader io.Reader) (ID, error)
|
|
PutFile(path string) (ID, error)
|
|
PutRaw([]byte) (ID, error)
|
|
Get(ID) (io.Reader, error)
|
|
Test(ID) (bool, error)
|
|
Remove(ID) error
|
|
Link(name string, id ID) error
|
|
Unlink(name string) error
|
|
Resolve(name string) (ID, error)
|
|
}
|
|
|
|
var (
|
|
ErrIDDoesNotExist = errors.New("ID does not exist")
|
|
)
|
|
|
|
// Name stands for the alias given to an ID.
|
|
type Name string
|
|
|
|
func (n Name) Encode() string {
|
|
return url.QueryEscape(string(n))
|
|
}
|
|
|
|
type DirRepository struct {
|
|
path string
|
|
hash func() hash.Hash
|
|
}
|
|
|
|
// NewDirRepository creates a new dir-baked repository at the given path.
|
|
func NewDirRepository(path string) (*DirRepository, error) {
|
|
d := &DirRepository{
|
|
path: path,
|
|
hash: sha256.New,
|
|
}
|
|
|
|
err := d.create()
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return d, nil
|
|
}
|
|
|
|
func (r *DirRepository) create() error {
|
|
dirs := []string{
|
|
r.path,
|
|
path.Join(r.path, objectPath),
|
|
path.Join(r.path, refPath),
|
|
path.Join(r.path, tempPath),
|
|
}
|
|
|
|
for _, dir := range dirs {
|
|
err := os.MkdirAll(dir, dirMode)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// SetHash changes the hash function used for deriving IDs. Default is SHA256.
|
|
func (r *DirRepository) SetHash(h func() hash.Hash) {
|
|
r.hash = h
|
|
}
|
|
|
|
// Path returns the directory used for this repository.
|
|
func (r *DirRepository) Path() string {
|
|
return r.path
|
|
}
|
|
|
|
// Put saves content and returns the ID.
|
|
func (r *DirRepository) Put(reader io.Reader) (ID, error) {
|
|
// save contents to tempfile, hash while writing
|
|
file, err := ioutil.TempFile(path.Join(r.path, tempPath), "temp-")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
rd := hashing.NewReader(reader, r.hash)
|
|
_, err = io.Copy(file, rd)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = file.Close()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// move file to final name using hash of contents
|
|
id := ID(rd.Hash())
|
|
filename := path.Join(r.path, objectPath, id.String())
|
|
err = os.Rename(file.Name(), filename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return id, nil
|
|
}
|
|
|
|
// PutFile saves a file's content to the repository and returns the ID.
|
|
func (r *DirRepository) PutFile(path string) (ID, error) {
|
|
f, err := os.Open(path)
|
|
defer f.Close()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return r.Put(f)
|
|
}
|
|
|
|
// PutRaw saves a []byte's content to the repository and returns the ID.
|
|
func (r *DirRepository) PutRaw(buf []byte) (ID, error) {
|
|
// save contents to tempfile, hash while writing
|
|
file, err := ioutil.TempFile(path.Join(r.path, tempPath), "temp-")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
wr := hashing.NewWriter(file, r.hash)
|
|
n, err := wr.Write(buf)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if n != len(buf) {
|
|
return nil, errors.New("not all bytes written")
|
|
}
|
|
|
|
err = file.Close()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// move file to final name using hash of contents
|
|
id := ID(wr.Hash())
|
|
filename := path.Join(r.path, objectPath, id.String())
|
|
err = os.Rename(file.Name(), filename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return id, nil
|
|
}
|
|
|
|
// Test returns true if the given ID exists in the repository.
|
|
func (r *DirRepository) Test(id ID) (bool, error) {
|
|
// try to open file
|
|
file, err := os.Open(path.Join(r.path, objectPath, id.String()))
|
|
defer func() {
|
|
file.Close()
|
|
}()
|
|
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
// Get returns a reader for the content stored under the given ID.
|
|
func (r *DirRepository) Get(id ID) (io.Reader, error) {
|
|
// try to open file
|
|
file, err := os.Open(path.Join(r.path, objectPath, id.String()))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return file, nil
|
|
}
|
|
|
|
// Remove removes the content stored at ID.
|
|
func (r *DirRepository) Remove(id ID) error {
|
|
return os.Remove(path.Join(r.path, objectPath, id.String()))
|
|
}
|
|
|
|
// Unlink removes a named ID.
|
|
func (r *DirRepository) Unlink(name string) error {
|
|
return os.Remove(path.Join(r.path, refPath, Name(name).Encode()))
|
|
}
|
|
|
|
// Link assigns a name to an ID. Name must be unique in this repository and ID must exist.
|
|
func (r *DirRepository) Link(name string, id ID) error {
|
|
exist, err := r.Test(id)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !exist {
|
|
return ErrIDDoesNotExist
|
|
}
|
|
|
|
// create file, write id
|
|
f, err := os.Create(path.Join(r.path, refPath, Name(name).Encode()))
|
|
defer f.Close()
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
f.Write([]byte(hex.EncodeToString(id)))
|
|
return nil
|
|
}
|
|
|
|
// Resolve returns the ID associated with the given name.
|
|
func (r *DirRepository) Resolve(name string) (ID, error) {
|
|
f, err := os.Open(path.Join(r.path, refPath, Name(name).Encode()))
|
|
defer f.Close()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// read hex string
|
|
l := r.hash().Size()
|
|
buf := make([]byte, l*2)
|
|
_, err = io.ReadFull(f, buf)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
id := make([]byte, l)
|
|
_, err = hex.Decode(id, buf)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return ID(id), nil
|
|
}
|