2015-10-02 00:09:04 +10:00
|
|
|
// Implements torrent data storage as per-piece files.
|
2015-03-11 02:41:21 +11:00
|
|
|
package blob
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/sha1"
|
|
|
|
"encoding/hex"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2015-03-18 18:12:46 +11:00
|
|
|
"sort"
|
2015-03-30 23:10:37 +11:00
|
|
|
"sync"
|
2015-03-18 18:12:46 +11:00
|
|
|
"time"
|
2015-03-11 02:41:21 +11:00
|
|
|
|
2015-06-01 18:26:32 +10:00
|
|
|
"github.com/anacrolix/missinggo"
|
|
|
|
|
2015-04-30 00:31:34 +10:00
|
|
|
"github.com/anacrolix/torrent/metainfo"
|
2015-03-11 02:41:21 +11:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
filePerm = 0640
|
|
|
|
dirPerm = 0750
|
|
|
|
)
|
|
|
|
|
|
|
|
type store struct {
|
2015-03-30 23:10:37 +11:00
|
|
|
baseDir string
|
|
|
|
capacity int64
|
|
|
|
|
|
|
|
mu sync.Mutex
|
2015-03-18 18:34:35 +11:00
|
|
|
completed map[[20]byte]struct{}
|
2015-03-11 02:41:21 +11:00
|
|
|
}
|
|
|
|
|
2015-10-02 00:16:25 +10:00
|
|
|
func (me *store) OpenTorrent(info *metainfo.Info) *data {
|
2015-03-11 02:41:21 +11:00
|
|
|
return &data{info, me}
|
|
|
|
}
|
|
|
|
|
|
|
|
type StoreOption func(*store)
|
|
|
|
|
|
|
|
func Capacity(bytes int64) StoreOption {
|
|
|
|
return func(s *store) {
|
|
|
|
s.capacity = bytes
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-02 00:09:04 +10:00
|
|
|
func NewStore(baseDir string, opt ...StoreOption) *store {
|
2015-03-30 23:10:37 +11:00
|
|
|
s := &store{baseDir, -1, sync.Mutex{}, nil}
|
2015-03-11 02:41:21 +11:00
|
|
|
for _, o := range opt {
|
|
|
|
o(s)
|
|
|
|
}
|
|
|
|
s.initCompleted()
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2015-05-16 10:52:35 +10:00
|
|
|
// Turns 40 byte hex string into its equivalent binary byte array.
|
|
|
|
func hexStringPieceHashArray(s string) (ret [20]byte, ok bool) {
|
2015-03-18 18:34:35 +11:00
|
|
|
if len(s) != 40 {
|
2015-05-16 10:52:35 +10:00
|
|
|
return
|
2015-03-18 18:34:35 +11:00
|
|
|
}
|
|
|
|
n, err := hex.Decode(ret[:], []byte(s))
|
|
|
|
if err != nil {
|
2015-05-16 10:52:35 +10:00
|
|
|
return
|
2015-03-18 18:34:35 +11:00
|
|
|
}
|
|
|
|
if n != 20 {
|
|
|
|
panic(n)
|
|
|
|
}
|
2015-05-16 10:52:35 +10:00
|
|
|
ok = true
|
2015-03-18 18:34:35 +11:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-03-11 02:41:21 +11:00
|
|
|
func (me *store) initCompleted() {
|
|
|
|
fis, err := me.readCompletedDir()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Lock()
|
2015-03-18 18:34:35 +11:00
|
|
|
me.completed = make(map[[20]byte]struct{}, len(fis))
|
2015-03-11 02:41:21 +11:00
|
|
|
for _, fi := range fis {
|
2015-05-16 10:52:35 +10:00
|
|
|
binHash, ok := hexStringPieceHashArray(fi.Name())
|
|
|
|
if !ok {
|
2015-03-18 18:34:35 +11:00
|
|
|
continue
|
|
|
|
}
|
2015-05-16 10:52:35 +10:00
|
|
|
me.completed[binHash] = struct{}{}
|
2015-03-11 02:41:21 +11:00
|
|
|
}
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Unlock()
|
2015-03-11 02:41:21 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (me *store) completePieceDirPath() string {
|
2015-10-01 15:41:30 +10:00
|
|
|
return filepath.Join(me.baseDir, "completed")
|
2015-03-11 02:41:21 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (me *store) path(p metainfo.Piece, completed bool) string {
|
|
|
|
return filepath.Join(me.baseDir, func() string {
|
|
|
|
if completed {
|
2015-10-01 15:41:30 +10:00
|
|
|
return "completed"
|
2015-03-11 02:41:21 +11:00
|
|
|
} else {
|
|
|
|
return "incomplete"
|
|
|
|
}
|
|
|
|
}(), fmt.Sprintf("%x", p.Hash()))
|
|
|
|
}
|
|
|
|
|
2015-03-18 18:34:35 +11:00
|
|
|
func sliceToPieceHashArray(b []byte) (ret [20]byte) {
|
|
|
|
n := copy(ret[:], b)
|
|
|
|
if n != 20 {
|
|
|
|
panic(n)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-03-11 02:41:21 +11:00
|
|
|
func (me *store) pieceComplete(p metainfo.Piece) bool {
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Lock()
|
|
|
|
defer me.mu.Unlock()
|
2015-03-18 18:34:35 +11:00
|
|
|
_, ok := me.completed[sliceToPieceHashArray(p.Hash())]
|
2015-03-11 02:41:21 +11:00
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me *store) pieceWrite(p metainfo.Piece) (f *os.File) {
|
|
|
|
if me.pieceComplete(p) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
name := me.path(p, false)
|
|
|
|
os.MkdirAll(filepath.Dir(name), dirPerm)
|
|
|
|
f, err := os.OpenFile(name, os.O_CREATE|os.O_WRONLY, filePerm)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-20 22:21:20 +10:00
|
|
|
// Returns the file for the given piece, if it exists. It could be completed,
|
|
|
|
// or incomplete.
|
2015-03-11 02:41:21 +11:00
|
|
|
func (me *store) pieceRead(p metainfo.Piece) (f *os.File) {
|
|
|
|
f, err := os.Open(me.path(p, true))
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !os.IsNotExist(err) {
|
|
|
|
panic(err)
|
|
|
|
}
|
2015-05-20 22:21:20 +10:00
|
|
|
// Mark the file not completed, in case we thought it was. TODO: Trigger
|
|
|
|
// an asynchronous initCompleted to reinitialize the entire completed map
|
|
|
|
// as there are likely other files missing.
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Lock()
|
2015-03-18 18:35:22 +11:00
|
|
|
delete(me.completed, sliceToPieceHashArray(p.Hash()))
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Unlock()
|
2015-03-11 02:41:21 +11:00
|
|
|
f, err = os.Open(me.path(p, false))
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !os.IsNotExist(err) {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me *store) readCompletedDir() (fis []os.FileInfo, err error) {
|
|
|
|
f, err := os.Open(me.completePieceDirPath())
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
err = nil
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fis, err = f.Readdir(-1)
|
|
|
|
f.Close()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me *store) removeCompleted(name string) (err error) {
|
|
|
|
err = os.Remove(filepath.Join(me.completePieceDirPath(), name))
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
err = nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-05-16 10:52:35 +10:00
|
|
|
binHash, ok := hexStringPieceHashArray(name)
|
|
|
|
if ok {
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Lock()
|
2015-05-16 10:52:35 +10:00
|
|
|
delete(me.completed, binHash)
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Unlock()
|
2015-05-16 10:52:35 +10:00
|
|
|
}
|
2015-03-11 02:41:21 +11:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-03-18 18:12:46 +11:00
|
|
|
type fileInfoSorter struct {
|
|
|
|
fis []os.FileInfo
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me fileInfoSorter) Len() int {
|
|
|
|
return len(me.fis)
|
|
|
|
}
|
|
|
|
|
|
|
|
func lastTime(fi os.FileInfo) (ret time.Time) {
|
|
|
|
ret = fi.ModTime()
|
2015-06-01 18:26:32 +10:00
|
|
|
atime := missinggo.FileInfoAccessTime(fi)
|
2015-03-18 18:12:46 +11:00
|
|
|
if atime.After(ret) {
|
|
|
|
ret = atime
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me fileInfoSorter) Less(i, j int) bool {
|
|
|
|
return lastTime(me.fis[i]).Before(lastTime(me.fis[j]))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me fileInfoSorter) Swap(i, j int) {
|
|
|
|
me.fis[i], me.fis[j] = me.fis[j], me.fis[i]
|
|
|
|
}
|
|
|
|
|
|
|
|
func sortFileInfos(fis []os.FileInfo) {
|
|
|
|
sorter := fileInfoSorter{fis}
|
|
|
|
sort.Sort(sorter)
|
|
|
|
}
|
|
|
|
|
2015-03-11 02:41:21 +11:00
|
|
|
func (me *store) makeSpace(space int64) error {
|
|
|
|
if me.capacity < 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if space > me.capacity {
|
|
|
|
return errors.New("space requested exceeds capacity")
|
|
|
|
}
|
|
|
|
fis, err := me.readCompletedDir()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var size int64
|
|
|
|
for _, fi := range fis {
|
|
|
|
size += fi.Size()
|
|
|
|
}
|
2015-03-18 18:12:46 +11:00
|
|
|
sortFileInfos(fis)
|
2015-03-11 02:41:21 +11:00
|
|
|
for size > me.capacity-space {
|
2015-03-18 18:12:46 +11:00
|
|
|
me.removeCompleted(fis[0].Name())
|
|
|
|
size -= fis[0].Size()
|
|
|
|
fis = fis[1:]
|
2015-03-11 02:41:21 +11:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me *store) PieceCompleted(p metainfo.Piece) (err error) {
|
|
|
|
err = me.makeSpace(p.Length())
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var (
|
|
|
|
incompletePiecePath = me.path(p, false)
|
|
|
|
completedPiecePath = me.path(p, true)
|
|
|
|
)
|
|
|
|
fSrc, err := os.Open(incompletePiecePath)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer fSrc.Close()
|
|
|
|
os.MkdirAll(filepath.Dir(completedPiecePath), dirPerm)
|
|
|
|
fDst, err := os.OpenFile(completedPiecePath, os.O_EXCL|os.O_CREATE|os.O_WRONLY, filePerm)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer fDst.Close()
|
|
|
|
hasher := sha1.New()
|
|
|
|
r := io.TeeReader(io.LimitReader(fSrc, p.Length()), hasher)
|
|
|
|
_, err = io.Copy(fDst, r)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !bytes.Equal(hasher.Sum(nil), p.Hash()) {
|
|
|
|
err = errors.New("piece incomplete")
|
|
|
|
os.Remove(completedPiecePath)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
os.Remove(incompletePiecePath)
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Lock()
|
2015-03-18 18:34:35 +11:00
|
|
|
me.completed[sliceToPieceHashArray(p.Hash())] = struct{}{}
|
2015-08-03 14:24:59 +10:00
|
|
|
me.mu.Unlock()
|
2015-03-11 02:41:21 +11:00
|
|
|
return
|
|
|
|
}
|