133 lines
3.3 KiB
Go
133 lines
3.3 KiB
Go
package persistence
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"fmt"
|
|
"github.com/pkg/errors"
|
|
"net/url"
|
|
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
type Database interface {
|
|
Engine() databaseEngine
|
|
DoesTorrentExist(infoHash []byte) (bool, error)
|
|
AddNewTorrent(infoHash []byte, name string, files []File) error
|
|
Close() error
|
|
|
|
// GetNumberOfTorrents returns the number of torrents saved in the database. Might be an
|
|
// approximation.
|
|
GetNumberOfTorrents() (uint, error)
|
|
// QueryTorrents returns @pageSize amount of torrents,
|
|
// * that are discovered before @discoveredOnBefore
|
|
// * that match the @query if it's not empty, else all torrents
|
|
// * ordered by the @orderBy in ascending order if @ascending is true, else in descending order
|
|
// after skipping (@page * @pageSize) torrents that also fits the criteria above.
|
|
//
|
|
// On error, returns (nil, error), otherwise a non-nil slice of TorrentMetadata and nil.
|
|
QueryTorrents(
|
|
query string,
|
|
epoch int64,
|
|
orderBy OrderingCriteria,
|
|
ascending bool,
|
|
limit uint,
|
|
lastOrderedValue *float64,
|
|
lastID *uint64,
|
|
) ([]TorrentMetadata, error)
|
|
// GetTorrents returns the TorrentExtMetadata for the torrent of the given InfoHash. Will return
|
|
// nil, nil if the torrent does not exist in the database.
|
|
GetTorrent(infoHash []byte) (*TorrentMetadata, error)
|
|
GetFiles(infoHash []byte) ([]File, error)
|
|
GetStatistics(from string, n uint) (*Statistics, error)
|
|
}
|
|
|
|
type OrderingCriteria uint8
|
|
|
|
const (
|
|
ByRelevance OrderingCriteria = iota
|
|
ByTotalSize
|
|
ByDiscoveredOn
|
|
ByNFiles
|
|
ByNSeeders
|
|
ByNLeechers
|
|
ByUpdatedOn
|
|
)
|
|
|
|
// TODO: search `swtich (orderBy)` and see if all cases are covered all the time
|
|
|
|
type databaseEngine uint8
|
|
|
|
const (
|
|
Sqlite3 databaseEngine = 1
|
|
)
|
|
|
|
type Statistics struct {
|
|
NDiscovered map[string]uint64 `json:"nDiscovered"`
|
|
NFiles map[string]uint64 `json:"nFiles"`
|
|
TotalSize map[string]uint64 `json:"totalSize"`
|
|
|
|
// All these slices below have the exact length equal to the Period.
|
|
//NDiscovered []uint64 `json:"nDiscovered"`
|
|
|
|
}
|
|
|
|
type File struct {
|
|
Size int64 `json:"size"`
|
|
Path string `json:"path"`
|
|
}
|
|
|
|
type TorrentMetadata struct {
|
|
ID uint64 `json:"id"`
|
|
InfoHash []byte `json:"infoHash"` // marshalled differently
|
|
Name string `json:"name"`
|
|
Size uint64 `json:"size"`
|
|
DiscoveredOn int64 `json:"discoveredOn"`
|
|
NFiles uint `json:"nFiles"`
|
|
Relevance float64 `json:"relevance"`
|
|
}
|
|
|
|
func (tm *TorrentMetadata) MarshalJSON() ([]byte, error) {
|
|
type Alias TorrentMetadata
|
|
return json.Marshal(&struct {
|
|
InfoHash string `json:"infoHash"`
|
|
*Alias
|
|
}{
|
|
InfoHash: hex.EncodeToString(tm.InfoHash),
|
|
Alias: (*Alias)(tm),
|
|
})
|
|
}
|
|
|
|
func MakeDatabase(rawURL string, logger *zap.Logger) (Database, error) {
|
|
if logger != nil {
|
|
zap.ReplaceGlobals(logger)
|
|
}
|
|
|
|
url_, err := url.Parse(rawURL)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "url.Parse")
|
|
}
|
|
|
|
switch url_.Scheme {
|
|
case "sqlite3":
|
|
return makeSqlite3Database(url_)
|
|
|
|
case "postgresql":
|
|
return nil, fmt.Errorf("postgresql is not yet supported")
|
|
|
|
case "mysql":
|
|
return nil, fmt.Errorf("mysql is not yet supported")
|
|
|
|
default:
|
|
return nil, fmt.Errorf("unknown URI scheme: `%s`", url_.Scheme)
|
|
}
|
|
}
|
|
|
|
func NewStatistics() (s *Statistics) {
|
|
s = new(Statistics)
|
|
s.NDiscovered = make(map[string]uint64)
|
|
s.NFiles = make(map[string]uint64)
|
|
s.TotalSize = make(map[string]uint64)
|
|
return
|
|
}
|