2017-11-03 00:15:13 +01:00
|
|
|
package persistence
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
2018-03-03 18:09:49 +01:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"go.uber.org/zap"
|
2017-11-03 00:15:13 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
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)
|
2018-03-03 23:09:34 +01:00
|
|
|
// 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.
|
2018-03-03 18:09:49 +01:00
|
|
|
QueryTorrents(query string, discoveredOnBefore int64, orderBy orderingCriteria, ascending bool, page uint, pageSize uint) ([]TorrentMetadata, error)
|
2018-03-03 23:09:34 +01:00
|
|
|
// GetTorrents returns the TorrentExtMetadata for the torrent of the given InfoHash. Will return
|
2017-11-03 00:15:13 +01:00
|
|
|
// nil, nil if the torrent does not exist in the database.
|
|
|
|
GetTorrent(infoHash []byte) (*TorrentMetadata, error)
|
|
|
|
GetFiles(infoHash []byte) ([]File, error)
|
2018-03-03 18:09:49 +01:00
|
|
|
GetStatistics(n uint, granularity Granularity, to time.Time) (*Statistics, error)
|
2017-11-03 00:15:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type orderingCriteria uint8
|
|
|
|
const (
|
2018-03-03 18:09:49 +01:00
|
|
|
ByRelevance orderingCriteria = iota
|
|
|
|
BySize
|
|
|
|
ByDiscoveredOn
|
|
|
|
ByNFiles
|
2017-11-03 00:15:13 +01:00
|
|
|
)
|
|
|
|
|
2018-03-03 18:09:49 +01:00
|
|
|
type Granularity uint8
|
2017-11-03 00:15:13 +01:00
|
|
|
const (
|
2018-03-03 18:09:49 +01:00
|
|
|
Yearly Granularity = iota
|
|
|
|
Monthly
|
|
|
|
Weekly
|
|
|
|
Daily
|
|
|
|
Hourly
|
2017-11-03 00:15:13 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type databaseEngine uint8
|
|
|
|
const (
|
2018-03-03 18:09:49 +01:00
|
|
|
Sqlite3 databaseEngine = 1
|
2017-11-03 00:15:13 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Statistics struct {
|
2018-03-03 18:09:49 +01:00
|
|
|
N uint
|
2017-11-03 00:15:13 +01:00
|
|
|
|
|
|
|
// All these slices below have the exact length equal to the Period.
|
|
|
|
NTorrentsDiscovered []uint
|
|
|
|
NFilesDiscovered []uint
|
|
|
|
}
|
|
|
|
|
|
|
|
type File struct {
|
|
|
|
Size int64
|
|
|
|
Path string
|
|
|
|
}
|
|
|
|
|
|
|
|
type TorrentMetadata struct {
|
|
|
|
InfoHash []byte
|
|
|
|
Name string
|
2018-03-03 18:09:49 +01:00
|
|
|
TotalSize uint64
|
2017-11-03 00:15:13 +01:00
|
|
|
DiscoveredOn int64
|
|
|
|
NFiles uint
|
|
|
|
}
|
|
|
|
|
2018-03-03 18:11:13 +01:00
|
|
|
func MakeDatabase(rawURL string, logger *zap.Logger) (Database, error) {
|
2017-11-03 00:15:13 +01:00
|
|
|
if logger != nil {
|
|
|
|
zap.ReplaceGlobals(logger)
|
|
|
|
}
|
|
|
|
|
|
|
|
url_, err := url.Parse(rawURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch url_.Scheme {
|
|
|
|
case "sqlite3":
|
2018-03-03 18:11:13 +01:00
|
|
|
return makeSqlite3Database(url_)
|
2017-11-03 00:15:13 +01:00
|
|
|
|
|
|
|
case "postgresql":
|
|
|
|
return nil, fmt.Errorf("postgresql is not yet supported!")
|
|
|
|
|
|
|
|
case "mysql":
|
|
|
|
return nil, fmt.Errorf("mysql is not yet supported!")
|
|
|
|
|
2018-03-03 18:09:49 +01:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("unknown URI scheme (database engine)!")
|
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
}
|