2011-01-28 07:07:18 +00:00
|
|
|
/*
|
|
|
|
Copyright 2011 Google Inc.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2011-01-17 05:00:36 +00:00
|
|
|
package blobref
|
|
|
|
|
|
|
|
import (
|
2011-05-04 00:55:57 +00:00
|
|
|
"crypto"
|
2011-01-17 05:00:36 +00:00
|
|
|
"fmt"
|
2011-03-25 02:20:47 +00:00
|
|
|
"io"
|
2011-05-05 23:31:19 +00:00
|
|
|
"io/ioutil"
|
2011-05-04 00:55:57 +00:00
|
|
|
"log"
|
2011-01-17 05:00:36 +00:00
|
|
|
"os"
|
2011-03-16 06:16:24 +00:00
|
|
|
"path/filepath"
|
2011-05-05 23:31:19 +00:00
|
|
|
"strings"
|
|
|
|
"sync"
|
2011-01-17 05:00:36 +00:00
|
|
|
)
|
|
|
|
|
2011-05-04 00:55:57 +00:00
|
|
|
var _ = log.Printf
|
|
|
|
|
2011-05-11 13:29:40 +00:00
|
|
|
// TODO: rename StreamingFetcher to be Fetch (the common case) and
|
|
|
|
// make a new interface for SeekingFetcher (the rare case)
|
|
|
|
|
2011-01-17 05:00:36 +00:00
|
|
|
type Fetcher interface {
|
2011-03-05 08:03:53 +00:00
|
|
|
// Fetch returns a blob. If the blob is not found then
|
|
|
|
// os.ENOENT should be returned for the error (not a wrapped
|
|
|
|
// error with a ENOENT inside)
|
2011-01-17 05:00:36 +00:00
|
|
|
Fetch(*BlobRef) (file ReadSeekCloser, size int64, err os.Error)
|
2011-03-25 02:20:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type StreamingFetcher interface {
|
|
|
|
// Fetch returns a blob. If the blob is not found then
|
|
|
|
// os.ENOENT should be returned for the error (not a wrapped
|
|
|
|
// error with a ENOENT inside)
|
|
|
|
FetchStreaming(*BlobRef) (file io.ReadCloser, size int64, err os.Error)
|
2011-01-17 05:00:36 +00:00
|
|
|
}
|
|
|
|
|
2011-03-16 06:16:24 +00:00
|
|
|
func NewSerialFetcher(fetchers ...Fetcher) Fetcher {
|
|
|
|
return &serialFetcher{fetchers}
|
|
|
|
}
|
|
|
|
|
2011-04-09 06:20:24 +00:00
|
|
|
func NewSerialStreamingFetcher(fetchers ...StreamingFetcher) StreamingFetcher {
|
|
|
|
return &serialStreamingFetcher{fetchers}
|
2011-01-17 05:00:36 +00:00
|
|
|
}
|
|
|
|
|
2011-04-09 06:20:24 +00:00
|
|
|
func NewSimpleDirectoryFetcher(dir string) *DirFetcher {
|
|
|
|
return &DirFetcher{dir, "camli"}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewConfigDirFetcher() *DirFetcher {
|
2011-03-16 06:16:24 +00:00
|
|
|
configDir := filepath.Join(os.Getenv("HOME"), ".camli", "keyblobs")
|
|
|
|
return NewSimpleDirectoryFetcher(configDir)
|
|
|
|
}
|
|
|
|
|
|
|
|
type serialFetcher struct {
|
|
|
|
fetchers []Fetcher
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sf *serialFetcher) Fetch(b *BlobRef) (file ReadSeekCloser, size int64, err os.Error) {
|
|
|
|
for _, fetcher := range sf.fetchers {
|
|
|
|
file, size, err = fetcher.Fetch(b)
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
2011-04-09 06:20:24 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
type serialStreamingFetcher struct {
|
|
|
|
fetchers []StreamingFetcher
|
2011-03-16 06:16:24 +00:00
|
|
|
}
|
|
|
|
|
2011-04-09 06:20:24 +00:00
|
|
|
func (sf *serialStreamingFetcher) FetchStreaming(b *BlobRef) (file io.ReadCloser, size int64, err os.Error) {
|
|
|
|
for _, fetcher := range sf.fetchers {
|
|
|
|
file, size, err = fetcher.FetchStreaming(b)
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
type DirFetcher struct {
|
2011-01-17 05:00:36 +00:00
|
|
|
directory, extension string
|
|
|
|
}
|
|
|
|
|
2011-04-09 06:20:24 +00:00
|
|
|
func (df *DirFetcher) FetchStreaming(b *BlobRef) (file io.ReadCloser, size int64, err os.Error) {
|
|
|
|
return df.Fetch(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (df *DirFetcher) Fetch(b *BlobRef) (file ReadSeekCloser, size int64, err os.Error) {
|
2011-01-17 05:00:36 +00:00
|
|
|
fileName := fmt.Sprintf("%s/%s.%s", df.directory, b.String(), df.extension)
|
|
|
|
var stat *os.FileInfo
|
|
|
|
stat, err = os.Stat(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2011-04-07 17:58:29 +00:00
|
|
|
file, err = os.Open(fileName)
|
2011-01-17 05:00:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
size = stat.Size
|
|
|
|
return
|
|
|
|
}
|
2011-05-04 00:55:57 +00:00
|
|
|
|
|
|
|
// MemoryStore stores blobs in memory and is a Fetcher and
|
|
|
|
// StreamingFetcher. Its zero value is usable.
|
|
|
|
type MemoryStore struct {
|
2011-05-05 23:31:19 +00:00
|
|
|
lk sync.Mutex
|
|
|
|
m map[string]string
|
2011-05-04 00:55:57 +00:00
|
|
|
}
|
|
|
|
|
2011-05-05 23:41:00 +00:00
|
|
|
func (s *MemoryStore) AddBlob(hashtype crypto.Hash, data string) (*BlobRef, os.Error) {
|
2011-05-04 00:55:57 +00:00
|
|
|
if hashtype != crypto.SHA1 {
|
2011-05-05 23:41:00 +00:00
|
|
|
return nil, os.NewError("blobref: unsupported hash type")
|
2011-05-04 00:55:57 +00:00
|
|
|
}
|
|
|
|
hash := hashtype.New()
|
2011-05-05 23:31:19 +00:00
|
|
|
hash.Write([]byte(data))
|
2011-05-04 00:55:57 +00:00
|
|
|
bstr := fmt.Sprintf("sha1-%x", hash.Sum())
|
2011-05-05 23:31:19 +00:00
|
|
|
s.lk.Lock()
|
|
|
|
defer s.lk.Unlock()
|
|
|
|
if s.m == nil {
|
|
|
|
s.m = make(map[string]string)
|
|
|
|
}
|
2011-05-04 00:55:57 +00:00
|
|
|
s.m[bstr] = data
|
2011-05-05 23:41:00 +00:00
|
|
|
return Parse(bstr), nil
|
2011-05-04 00:55:57 +00:00
|
|
|
}
|
|
|
|
|
2011-05-05 23:31:19 +00:00
|
|
|
func (s *MemoryStore) FetchStreaming(b *BlobRef) (file io.ReadCloser, size int64, err os.Error) {
|
|
|
|
s.lk.Lock()
|
|
|
|
defer s.lk.Unlock()
|
|
|
|
if s.m == nil {
|
|
|
|
return nil, 0, os.ENOENT
|
|
|
|
}
|
|
|
|
str, ok := s.m[b.String()]
|
|
|
|
if !ok {
|
|
|
|
return nil, 0, os.ENOENT
|
|
|
|
}
|
|
|
|
return ioutil.NopCloser(strings.NewReader(str)), int64(len(str)), nil
|
|
|
|
}
|