magnetico/metainfo/info.go

217 lines
6.5 KiB
Go

package metainfo
import (
"errors"
"fmt"
"io"
"iter"
"os"
"path/filepath"
"slices"
"sort"
"strings"
)
// The info dictionary. See BEP 3 and BEP 52.
type Info struct {
PieceLength int64 `bencode:"piece length"` // BEP3
// BEP 3. This can be omitted because isn't needed in non-hybrid v2 infos. See BEP 52.
Pieces []byte `bencode:"pieces,omitempty"`
Name string `bencode:"name"` // BEP3
NameUtf8 string `bencode:"name.utf-8,omitempty"`
Length int64 `bencode:"length,omitempty"` // BEP3, mutually exclusive with Files
ExtendedFileAttrs
Private *bool `bencode:"private,omitempty"` // BEP27
// TODO: Document this field.
Source string `bencode:"source,omitempty"`
Files []FileInfo `bencode:"files,omitempty"` // BEP3, mutually exclusive with Length
// BEP 52 (BitTorrent v2)
MetaVersion int64 `bencode:"meta version,omitempty"`
FileTree FileTree `bencode:"file tree,omitempty"`
}
// The Info.Name field is "advisory". For multi-file torrents it's usually a suggested directory
// name. There are situations where we don't want a directory (like using the contents of a torrent
// as the immediate contents of a directory), or the name is invalid. Transmission will inject the
// name of the torrent file if it doesn't like the name, resulting in a different infohash
// (https://github.com/transmission/transmission/issues/1775). To work around these situations, we
// will use a sentinel name for compatibility with Transmission and to signal to our own client that
// we intended to have no directory name. By exposing it in the API we can check for references to
// this behaviour within this implementation.
const NoName = "-"
// This is a helper that sets Files and Pieces from a root path and its children.
func (info *Info) BuildFromFilePath(root string) (err error) {
info.Name = func() string {
b := filepath.Base(root)
switch b {
case ".", "..", string(filepath.Separator):
return NoName
default:
return b
}
}()
info.Files = nil
err = filepath.Walk(root, func(path string, fi os.FileInfo, err error) error {
if err != nil {
return err
}
if fi.IsDir() {
// Directories are implicit in torrent files.
return nil
} else if path == root {
// The root is a file.
info.Length = fi.Size()
return nil
}
relPath, err := filepath.Rel(root, path)
if err != nil {
return fmt.Errorf("error getting relative path: %s", err)
}
info.Files = append(info.Files, FileInfo{
Path: strings.Split(relPath, string(filepath.Separator)),
Length: fi.Size(),
})
return nil
})
if err != nil {
return
}
sort.Slice(info.Files, func(i, j int) bool {
return strings.Join(info.Files[i].BestPath(), "/") < strings.Join(info.Files[j].BestPath(), "/")
})
if info.PieceLength == 0 {
info.PieceLength = ChoosePieceLength(info.TotalLength())
}
err = info.GeneratePieces(func(fi FileInfo) (io.ReadCloser, error) {
return os.Open(filepath.Join(root, strings.Join(fi.BestPath(), string(filepath.Separator))))
})
if err != nil {
err = fmt.Errorf("error generating pieces: %s", err)
}
return
}
// Concatenates all the files in the torrent into w. open is a function that
// gets at the contents of the given file.
func (info *Info) writeFiles(w io.Writer, open func(fi FileInfo) (io.ReadCloser, error)) error {
for _, fi := range info.UpvertedFiles() {
r, err := open(fi)
if err != nil {
return fmt.Errorf("error opening %v: %s", fi, err)
}
wn, err := io.CopyN(w, r, fi.Length)
r.Close()
if wn != fi.Length {
return fmt.Errorf("error copying %v: %s", fi, err)
}
}
return nil
}
// Sets Pieces (the block of piece hashes in the Info) by using the passed
// function to get at the torrent data.
func (info *Info) GeneratePieces(open func(fi FileInfo) (io.ReadCloser, error)) (err error) {
if info.PieceLength == 0 {
return errors.New("piece length must be non-zero")
}
pr, pw := io.Pipe()
go func() {
err := info.writeFiles(pw, open)
pw.CloseWithError(err)
}()
defer pr.Close()
info.Pieces, err = GeneratePieces(pr, info.PieceLength, nil)
return
}
func (info *Info) TotalLength() (ret int64) {
for _, fi := range info.UpvertedFiles() {
ret += fi.Length
}
return
}
func (info *Info) NumPieces() (num int) {
if info.HasV2() {
info.FileTree.Walk(nil, func(path []string, ft *FileTree) {
num += int((ft.File.Length + info.PieceLength - 1) / info.PieceLength)
})
return
}
return len(info.Pieces) / 20
}
// Whether all files share the same top-level directory name. If they don't, Info.Name is usually used.
func (info *Info) IsDir() bool {
if info.HasV2() {
return info.FileTree.IsDir()
}
// I wonder if we should check for the existence of Info.Length here instead.
return len(info.Files) != 0
}
// The files field, converted up from the old single-file in the parent info dict if necessary. This
// is a helper to avoid having to conditionally handle single and multi-file torrent infos.
func (info *Info) UpvertedFiles() (files []FileInfo) {
return slices.Collect(info.UpvertedFilesIter())
}
// The files field, converted up from the old single-file in the parent info dict if necessary. This
// is a helper to avoid having to conditionally handle single and multi-file torrent infos.
func (info *Info) UpvertedFilesIter() iter.Seq[FileInfo] {
if info.HasV2() {
return info.FileTree.upvertedFiles(info.PieceLength)
}
return info.UpvertedV1Files()
}
// UpvertedFiles but specific to the files listed in the v1 info fields. This will include padding
// files for example that wouldn't appear in v2 file trees.
func (info *Info) UpvertedV1Files() iter.Seq[FileInfo] {
return func(yield func(FileInfo) bool) {
if len(info.Files) == 0 {
yield(FileInfo{
Length: info.Length,
// Callers should determine that Info.Name is the basename, and
// thus a regular file.
Path: nil,
})
}
var offset int64
for _, fi := range info.Files {
fi.TorrentOffset = offset
offset += fi.Length
if !yield(fi) {
return
}
}
}
}
func (info *Info) Piece(index int) Piece {
return Piece{info, index}
}
func (info *Info) BestName() string {
if info.NameUtf8 != "" {
return info.NameUtf8
}
return info.Name
}
// Whether the Info can be used as a v2 info dict, including having a V2 infohash.
func (info *Info) HasV2() bool {
return info.MetaVersion == 2
}
func (info *Info) HasV1() bool {
// See Upgrade Path in BEP 52.
return info.MetaVersion == 0 || info.MetaVersion == 1 || info.Files != nil || info.Length != 0 || len(info.Pieces) != 0
}
func (info *Info) FilesArePieceAligned() bool {
return info.HasV2()
}