2017-10-04 15:07:48 +02:00
|
|
|
package persistence
|
|
|
|
|
|
|
|
import (
|
2018-04-21 11:05:12 +02:00
|
|
|
"bytes"
|
2017-11-03 00:15:13 +01:00
|
|
|
"database/sql"
|
|
|
|
"fmt"
|
2017-10-04 15:07:48 +02:00
|
|
|
"net/url"
|
|
|
|
"os"
|
2017-11-03 00:15:13 +01:00
|
|
|
"path"
|
2018-07-07 13:56:34 +02:00
|
|
|
"text/template"
|
2017-11-03 00:15:13 +01:00
|
|
|
"time"
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
_ "github.com/mattn/go-sqlite3"
|
2017-10-04 15:07:48 +02:00
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
// Close your rows lest you get "database table is locked" error(s)!
|
|
|
|
// See https://github.com/mattn/go-sqlite3/issues/2741
|
|
|
|
|
2017-10-04 15:07:48 +02:00
|
|
|
type sqlite3Database struct {
|
|
|
|
conn *sql.DB
|
|
|
|
}
|
|
|
|
|
2018-03-04 12:07:53 +01:00
|
|
|
func makeSqlite3Database(url_ *url.URL) (Database, error) {
|
2017-10-04 15:07:48 +02:00
|
|
|
db := new(sqlite3Database)
|
|
|
|
|
|
|
|
dbDir, _ := path.Split(url_.Path)
|
|
|
|
if err := os.MkdirAll(dbDir, 0755); err != nil {
|
|
|
|
return nil, fmt.Errorf("for directory `%s`: %s", dbDir, err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
db.conn, err = sql.Open("sqlite3", url_.Path)
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return nil, fmt.Errorf("sql.Open: %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// > Open may just validate its arguments without creating a connection to the database. To
|
2017-11-03 00:15:13 +01:00
|
|
|
// > verify that the data source Name is valid, call Ping.
|
2017-10-04 15:07:48 +02:00
|
|
|
// https://golang.org/pkg/database/sql/#Open
|
|
|
|
if err = db.conn.Ping(); err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return nil, fmt.Errorf("sql.DB.Ping: %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := db.setupDatabase(); err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return nil, fmt.Errorf("setupDatabase: %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return db, nil
|
|
|
|
}
|
|
|
|
|
2018-03-04 12:07:53 +01:00
|
|
|
func (db *sqlite3Database) Engine() databaseEngine {
|
|
|
|
return Sqlite3
|
|
|
|
}
|
|
|
|
|
2017-10-04 15:07:48 +02:00
|
|
|
func (db *sqlite3Database) DoesTorrentExist(infoHash []byte) (bool, error) {
|
|
|
|
rows, err := db.conn.Query("SELECT 1 FROM torrents WHERE info_hash = ?;", infoHash)
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return false, err
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2017-10-04 15:07:48 +02:00
|
|
|
|
|
|
|
// If rows.Next() returns true, meaning that the torrent is in the database, return true; else
|
|
|
|
// return false.
|
|
|
|
exists := rows.Next()
|
2018-07-24 14:41:13 +02:00
|
|
|
if rows.Err() != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return false, err
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
return exists, nil
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *sqlite3Database) AddNewTorrent(infoHash []byte, name string, files []File) error {
|
|
|
|
tx, err := db.conn.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// If everything goes as planned and no error occurs, we will commit the transaction before
|
|
|
|
// returning from the function so the tx.Rollback() call will fail, trying to rollback a
|
|
|
|
// committed transaction. BUT, if an error occurs, we'll get our transaction rollback'ed, which
|
|
|
|
// is nice.
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
var totalSize uint64 = 0
|
2017-10-04 15:07:48 +02:00
|
|
|
for _, file := range files {
|
2018-04-21 11:05:12 +02:00
|
|
|
totalSize += uint64(file.Size)
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
// This is a workaround for a bug: the database will not accept total_size to be zero.
|
2018-03-04 12:07:53 +01:00
|
|
|
if totalSize == 0 {
|
2017-11-03 00:15:13 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-03-04 12:07:53 +01:00
|
|
|
// Although we check whether the torrent exists in the database before asking MetadataSink to
|
|
|
|
// fetch its metadata, the torrent can also exists in the Sink before that. Now, if a torrent in
|
|
|
|
// the sink is still being fetched, that's still not a problem as we just add the new peer for
|
|
|
|
// the torrent and exit, but if the torrent is complete (i.e. its metadata) and if its waiting
|
|
|
|
// in the channel to be received, a race condition arises when we query the database and seeing
|
|
|
|
// that it doesn't exists there, add it to the sink.
|
|
|
|
// Hence INSERT OR IGNORE.
|
2017-10-04 15:07:48 +02:00
|
|
|
res, err := tx.Exec(`
|
2018-03-04 12:07:53 +01:00
|
|
|
INSERT OR IGNORE INTO torrents (
|
2017-10-04 15:07:48 +02:00
|
|
|
info_hash,
|
|
|
|
name,
|
|
|
|
total_size,
|
2017-11-03 00:15:13 +01:00
|
|
|
discovered_on
|
|
|
|
) VALUES (?, ?, ?, ?);
|
2018-03-04 12:07:53 +01:00
|
|
|
`, infoHash, name, totalSize, time.Now().Unix())
|
2017-10-04 15:07:48 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var lastInsertId int64
|
|
|
|
if lastInsertId, err = res.LastInsertId(); err != nil {
|
|
|
|
return fmt.Errorf("sql.Result.LastInsertId()! %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, file := range files {
|
2018-04-21 11:05:12 +02:00
|
|
|
_, err = tx.Exec("INSERT INTO files (torrent_id, size, path) VALUES (?, ?, ?);",
|
2017-10-04 15:07:48 +02:00
|
|
|
lastInsertId, file.Size, file.Path,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = tx.Commit()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *sqlite3Database) Close() error {
|
|
|
|
return db.conn.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *sqlite3Database) GetNumberOfTorrents() (uint, error) {
|
2018-03-04 12:07:53 +01:00
|
|
|
// COUNT(1) is much more inefficient since it scans the whole table, so use MAX(ROWID).
|
|
|
|
// Keep in mind that the value returned by GetNumberOfTorrents() might be an approximation.
|
2017-10-04 15:07:48 +02:00
|
|
|
rows, err := db.conn.Query("SELECT MAX(ROWID) FROM torrents;")
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2017-10-04 15:07:48 +02:00
|
|
|
|
|
|
|
if rows.Next() != true {
|
2018-04-21 11:05:12 +02:00
|
|
|
fmt.Errorf("No rows returned from `SELECT MAX(ROWID)`")
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var n uint
|
|
|
|
if err = rows.Scan(&n); err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
func (db *sqlite3Database) QueryTorrents(
|
|
|
|
query string,
|
|
|
|
epoch int64,
|
2018-06-29 19:08:00 +02:00
|
|
|
orderBy OrderingCriteria,
|
2018-04-21 11:05:12 +02:00
|
|
|
ascending bool,
|
|
|
|
limit uint,
|
2018-06-19 17:49:46 +02:00
|
|
|
lastOrderedValue *float64,
|
2018-04-25 22:33:50 +02:00
|
|
|
lastID *uint64,
|
2018-04-21 11:05:12 +02:00
|
|
|
) ([]TorrentMetadata, error) {
|
2018-03-04 12:07:53 +01:00
|
|
|
if query == "" && orderBy == ByRelevance {
|
2018-04-21 11:05:12 +02:00
|
|
|
return nil, fmt.Errorf("torrents cannot be ordered by relevance when the query is empty")
|
|
|
|
}
|
|
|
|
if (lastOrderedValue == nil) != (lastID == nil) {
|
|
|
|
return nil, fmt.Errorf("lastOrderedValue and lastID should be supplied together, if supplied")
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
2018-04-25 22:33:50 +02:00
|
|
|
doJoin := query != ""
|
2018-06-19 17:49:46 +02:00
|
|
|
firstPage := lastID == nil
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
// executeTemplate is used to prepare the SQL query, WITH PLACEHOLDERS FOR USER INPUT.
|
|
|
|
sqlQuery := executeTemplate(`
|
2018-06-19 17:49:46 +02:00
|
|
|
SELECT id
|
|
|
|
, info_hash
|
2018-04-21 11:05:12 +02:00
|
|
|
, name
|
|
|
|
, total_size
|
|
|
|
, discovered_on
|
|
|
|
, (SELECT COUNT(*) FROM files WHERE torrents.id = files.torrent_id) AS n_files
|
2018-06-19 17:49:46 +02:00
|
|
|
{{ if .DoJoin }}
|
|
|
|
, idx.rank
|
|
|
|
{{ else }}
|
|
|
|
, 0
|
|
|
|
{{ end }}
|
2018-04-21 11:05:12 +02:00
|
|
|
FROM torrents
|
|
|
|
{{ if .DoJoin }}
|
|
|
|
INNER JOIN (
|
|
|
|
SELECT rowid AS id
|
|
|
|
, bm25(torrents_idx) AS rank
|
|
|
|
FROM torrents_idx
|
|
|
|
WHERE torrents_idx MATCH ?
|
|
|
|
) AS idx USING(id)
|
|
|
|
{{ end }}
|
|
|
|
WHERE modified_on <= ?
|
2018-04-25 22:33:50 +02:00
|
|
|
{{ if not .FirstPage }}
|
2018-07-01 16:29:42 +02:00
|
|
|
AND ( {{.OrderOn}}, id ) {{GTEorLTE .Ascending}} (?, ?) -- https://www.sqlite.org/rowvalue.html#row_value_comparisons
|
2018-04-21 11:05:12 +02:00
|
|
|
{{ end }}
|
2018-07-01 16:29:42 +02:00
|
|
|
ORDER BY {{.OrderOn}} {{AscOrDesc .Ascending}}, id {{AscOrDesc .Ascending}}
|
2018-04-21 11:05:12 +02:00
|
|
|
LIMIT ?;
|
|
|
|
`, struct {
|
|
|
|
DoJoin bool
|
|
|
|
FirstPage bool
|
|
|
|
OrderOn string
|
|
|
|
Ascending bool
|
|
|
|
}{
|
2018-04-25 22:33:50 +02:00
|
|
|
DoJoin: doJoin,
|
|
|
|
FirstPage: firstPage,
|
2018-04-21 11:05:12 +02:00
|
|
|
OrderOn: orderOn(orderBy),
|
|
|
|
Ascending: ascending,
|
|
|
|
}, template.FuncMap{
|
|
|
|
"GTEorLTE": func(ascending bool) string {
|
|
|
|
if ascending {
|
|
|
|
return ">"
|
2018-04-25 22:33:50 +02:00
|
|
|
} else {
|
|
|
|
return "<"
|
2018-04-21 11:05:12 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
"AscOrDesc": func(ascending bool) string {
|
|
|
|
if ascending {
|
|
|
|
return "ASC"
|
|
|
|
} else {
|
|
|
|
return "DESC"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
// Prepare query
|
|
|
|
queryArgs := make([]interface{}, 0)
|
|
|
|
if doJoin {
|
|
|
|
queryArgs = append(queryArgs, query)
|
|
|
|
}
|
|
|
|
queryArgs = append(queryArgs, epoch)
|
2018-04-25 22:33:50 +02:00
|
|
|
if !firstPage {
|
2018-04-21 11:05:12 +02:00
|
|
|
queryArgs = append(queryArgs, lastOrderedValue)
|
2018-07-01 16:29:42 +02:00
|
|
|
queryArgs = append(queryArgs, lastID)
|
2018-04-21 11:05:12 +02:00
|
|
|
}
|
|
|
|
queryArgs = append(queryArgs, limit)
|
|
|
|
|
|
|
|
rows, err := db.conn.Query(sqlQuery, queryArgs...)
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2018-04-21 11:05:12 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error while querying torrents: %s", err.Error())
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2018-04-25 22:33:50 +02:00
|
|
|
torrents := make([]TorrentMetadata, 0)
|
2018-04-21 11:05:12 +02:00
|
|
|
for rows.Next() {
|
|
|
|
var torrent TorrentMetadata
|
2018-06-19 17:49:46 +02:00
|
|
|
err = rows.Scan(
|
|
|
|
&torrent.ID,
|
|
|
|
&torrent.InfoHash,
|
|
|
|
&torrent.Name,
|
|
|
|
&torrent.Size,
|
|
|
|
&torrent.DiscoveredOn,
|
|
|
|
&torrent.NFiles,
|
|
|
|
&torrent.Relevance,
|
|
|
|
)
|
|
|
|
if err != nil {
|
2018-04-21 11:05:12 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
torrents = append(torrents, torrent)
|
|
|
|
}
|
|
|
|
|
|
|
|
return torrents, nil
|
|
|
|
}
|
|
|
|
|
2018-06-29 19:08:00 +02:00
|
|
|
func orderOn(orderBy OrderingCriteria) string {
|
2018-04-21 11:05:12 +02:00
|
|
|
switch orderBy {
|
|
|
|
case ByRelevance:
|
|
|
|
return "idx.rank"
|
|
|
|
|
2018-06-29 19:08:00 +02:00
|
|
|
case ByTotalSize:
|
2018-04-21 11:05:12 +02:00
|
|
|
return "total_size"
|
|
|
|
|
|
|
|
case ByDiscoveredOn:
|
|
|
|
return "discovered_on"
|
|
|
|
|
|
|
|
case ByNFiles:
|
|
|
|
return "n_files"
|
|
|
|
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unknown orderBy: %v", orderBy))
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *sqlite3Database) GetTorrent(infoHash []byte) (*TorrentMetadata, error) {
|
2018-04-21 11:05:12 +02:00
|
|
|
rows, err := db.conn.Query(`
|
|
|
|
SELECT
|
2017-10-04 15:07:48 +02:00
|
|
|
info_hash,
|
|
|
|
name,
|
2017-11-03 00:15:13 +01:00
|
|
|
total_size,
|
2017-10-04 15:07:48 +02:00
|
|
|
discovered_on,
|
2018-04-21 11:05:12 +02:00
|
|
|
(SELECT COUNT(*) FROM files WHERE torrent_id = torrents.id) AS n_files
|
2017-10-04 15:07:48 +02:00
|
|
|
FROM torrents
|
|
|
|
WHERE info_hash = ?`,
|
|
|
|
infoHash,
|
|
|
|
)
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2017-10-04 15:07:48 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if rows.Next() != true {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
var tm TorrentMetadata
|
2018-04-21 11:05:12 +02:00
|
|
|
if err = rows.Scan(&tm.InfoHash, &tm.Name, &tm.Size, &tm.DiscoveredOn, &tm.NFiles); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
return &tm, nil
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *sqlite3Database) GetFiles(infoHash []byte) ([]File, error) {
|
2018-06-29 17:58:57 +02:00
|
|
|
rows, err := db.conn.Query(
|
|
|
|
"SELECT size, path FROM files, torrents WHERE files.torrent_id = torrents.id AND torrents.info_hash = ?;",
|
|
|
|
infoHash)
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2017-11-03 00:15:13 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
var files []File
|
|
|
|
for rows.Next() {
|
|
|
|
var file File
|
2018-04-21 11:05:12 +02:00
|
|
|
if err = rows.Scan(&file.Size, &file.Path); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
files = append(files, file)
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
return files, nil
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2018-07-07 13:56:34 +02:00
|
|
|
func (db *sqlite3Database) GetStatistics(from string, n uint) (*Statistics, error) {
|
|
|
|
fromTime, gran, err := ParseISO8601(from)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error while parsing from: %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
var toTime time.Time
|
|
|
|
var timef string // time format: https://www.sqlite.org/lang_datefunc.html
|
|
|
|
|
|
|
|
switch gran {
|
|
|
|
case Year:
|
|
|
|
toTime = fromTime.AddDate(int(n), 0, 0)
|
|
|
|
timef = "%Y"
|
|
|
|
case Month:
|
|
|
|
toTime = fromTime.AddDate(0, int(n), 0)
|
|
|
|
timef = "%Y-%m"
|
|
|
|
case Week:
|
|
|
|
toTime = fromTime.AddDate(0, 0, int(n) * 7)
|
|
|
|
timef = "%Y-%W"
|
|
|
|
case Day:
|
|
|
|
toTime = fromTime.AddDate(0, 0, int(n))
|
|
|
|
timef = "%Y-%m-%d"
|
|
|
|
case Hour:
|
|
|
|
toTime = fromTime.Add(time.Duration(n) * time.Hour)
|
|
|
|
timef = "%Y-%m-%dT%H"
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: make it faster!
|
|
|
|
rows, err := db.conn.Query(fmt.Sprintf(`
|
|
|
|
SELECT strftime('%s', discovered_on, 'unixepoch') AS dT
|
|
|
|
, sum(files.size) AS tS
|
|
|
|
, count(DISTINCT torrents.id) AS nD
|
|
|
|
, count(DISTINCT files.id) AS nF
|
|
|
|
FROM torrents, files
|
|
|
|
WHERE torrents.id = files.torrent_id
|
|
|
|
AND discovered_on >= ?
|
|
|
|
AND discovered_on <= ?
|
|
|
|
GROUP BY dt;`,
|
|
|
|
timef),
|
|
|
|
fromTime.Unix(), toTime.Unix())
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2018-04-21 11:05:12 +02:00
|
|
|
if err != nil {
|
2018-07-07 13:56:34 +02:00
|
|
|
return nil, err
|
2018-04-21 11:05:12 +02:00
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
|
2018-07-07 13:56:34 +02:00
|
|
|
stats := NewStatistics()
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
var dT string
|
|
|
|
var tS, nD, nF uint64
|
|
|
|
if err := rows.Scan(&dT, &tS, &nD, &nF); err != nil {
|
|
|
|
if err := rows.Close(); err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
stats.NDiscovered[dT] = nD
|
|
|
|
stats.TotalSize[dT] = tS
|
|
|
|
stats.NFiles[dT] = nF
|
|
|
|
}
|
2018-04-21 11:05:12 +02:00
|
|
|
|
2018-07-07 13:56:34 +02:00
|
|
|
return stats, nil
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *sqlite3Database) setupDatabase() error {
|
|
|
|
// Enable Write-Ahead Logging for SQLite as "WAL provides more concurrency as readers do not
|
|
|
|
// block writers and a writer does not block readers. Reading and writing can proceed
|
|
|
|
// concurrently."
|
|
|
|
// Caveats:
|
|
|
|
// * Might be unsupported by OSes other than Windows and UNIXes.
|
|
|
|
// * Does not work over a network filesystem.
|
|
|
|
// * Transactions that involve changes against multiple ATTACHed databases are not atomic
|
|
|
|
// across all databases as a set.
|
|
|
|
// See: https://www.sqlite.org/wal.html
|
|
|
|
//
|
|
|
|
// Force SQLite to use disk, instead of memory, for all temporary files to reduce the memory
|
|
|
|
// footprint.
|
|
|
|
//
|
|
|
|
// Enable foreign key constraints in SQLite which are crucial to prevent programmer errors on
|
|
|
|
// our side.
|
|
|
|
_, err := db.conn.Exec(`
|
|
|
|
PRAGMA journal_mode=WAL;
|
|
|
|
PRAGMA temp_store=1;
|
|
|
|
PRAGMA foreign_keys=ON;
|
2018-03-04 12:07:53 +01:00
|
|
|
PRAGMA encoding='UTF-8';
|
2017-10-04 15:07:48 +02:00
|
|
|
`)
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.DB.Exec (PRAGMAs): %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
tx, err := db.conn.Begin()
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.DB.Begin: %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
// If everything goes as planned and no error occurs, we will commit the transaction before
|
|
|
|
// returning from the function so the tx.Rollback() call will fail, trying to rollback a
|
|
|
|
// committed transaction. BUT, if an error occurs, we'll get our transaction rollback'ed, which
|
|
|
|
// is nice.
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
// Initial Setup for `user_version` 0:
|
|
|
|
// FROZEN.
|
|
|
|
// TODO: "torrent_id" column of the "files" table can be NULL, how can we fix this in a new
|
|
|
|
// version schema?
|
2017-10-04 15:07:48 +02:00
|
|
|
_, err = tx.Exec(`
|
|
|
|
CREATE TABLE IF NOT EXISTS torrents (
|
|
|
|
id INTEGER PRIMARY KEY,
|
|
|
|
info_hash BLOB NOT NULL UNIQUE,
|
|
|
|
name TEXT NOT NULL,
|
|
|
|
total_size INTEGER NOT NULL CHECK(total_size > 0),
|
|
|
|
discovered_on INTEGER NOT NULL CHECK(discovered_on > 0)
|
|
|
|
);
|
|
|
|
CREATE TABLE IF NOT EXISTS files (
|
|
|
|
id INTEGER PRIMARY KEY,
|
|
|
|
torrent_id INTEGER REFERENCES torrents ON DELETE CASCADE ON UPDATE RESTRICT,
|
|
|
|
size INTEGER NOT NULL,
|
|
|
|
path TEXT NOT NULL
|
|
|
|
);
|
|
|
|
`)
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.Tx.Exec (v0): %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the user_version:
|
2017-11-03 00:15:13 +01:00
|
|
|
rows, err := tx.Query("PRAGMA user_version;")
|
2017-10-04 15:07:48 +02:00
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.Tx.Query (user_version): %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
2018-07-24 14:41:13 +02:00
|
|
|
defer rows.Close()
|
2017-11-03 00:15:13 +01:00
|
|
|
var userVersion int
|
|
|
|
if rows.Next() != true {
|
|
|
|
return fmt.Errorf("sql.Rows.Next (user_version): PRAGMA user_version did not return any rows!")
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
2017-11-03 00:15:13 +01:00
|
|
|
if err = rows.Scan(&userVersion); err != nil {
|
|
|
|
return fmt.Errorf("sql.Rows.Scan (user_version): %s", err.Error())
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
|
|
|
|
switch userVersion {
|
2018-04-21 11:05:12 +02:00
|
|
|
case 0: // FROZEN.
|
2017-11-03 00:15:13 +01:00
|
|
|
// Upgrade from user_version 0 to 1
|
|
|
|
// Changes:
|
|
|
|
// * `info_hash_index` is recreated as UNIQUE.
|
|
|
|
zap.L().Warn("Updating database schema from 0 to 1... (this might take a while)")
|
2017-10-04 15:07:48 +02:00
|
|
|
_, err = tx.Exec(`
|
2018-04-21 11:05:12 +02:00
|
|
|
DROP INDEX IF EXISTS info_hash_index;
|
2017-10-04 15:07:48 +02:00
|
|
|
CREATE UNIQUE INDEX info_hash_index ON torrents (info_hash);
|
|
|
|
PRAGMA user_version = 1;
|
|
|
|
`)
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.Tx.Exec (v0 -> v1): %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
fallthrough
|
2017-11-03 00:15:13 +01:00
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
case 1: // FROZEN.
|
2017-11-03 00:15:13 +01:00
|
|
|
// Upgrade from user_version 1 to 2
|
|
|
|
// Changes:
|
|
|
|
// * Added `n_seeders`, `n_leechers`, and `updated_on` columns to the `torrents` table, and
|
|
|
|
// the constraints they entail.
|
|
|
|
// * Added `is_readme` and `content` columns to the `files` table, and the constraints & the
|
|
|
|
// the indices they entail.
|
|
|
|
// * Added unique index `readme_index` on `files` table.
|
|
|
|
zap.L().Warn("Updating database schema from 1 to 2... (this might take a while)")
|
|
|
|
// We introduce two new columns in `files`: content BLOB, and is_readme INTEGER which we
|
|
|
|
// treat as a bool (NULL for false, and 1 for true; see the CHECK statement).
|
2017-10-04 15:07:48 +02:00
|
|
|
// The reason for the change is that as we introduce the new "readme" feature which
|
|
|
|
// downloads a readme file as a torrent descriptor, we needed to store it somewhere in the
|
|
|
|
// database with the following conditions:
|
|
|
|
//
|
|
|
|
// 1. There can be one and only one readme (content) for a given torrent; hence the
|
|
|
|
// UNIQUE INDEX on (torrent_id, is_description) (remember that SQLite treats each NULL
|
|
|
|
// value as distinct [UNIQUE], see https://sqlite.org/nulls.html).
|
|
|
|
// 2. We would like to keep the readme (content) associated with the file it came from;
|
|
|
|
// hence we modify the files table instead of the torrents table.
|
|
|
|
//
|
|
|
|
// Regarding the implementation details, following constraints arise:
|
|
|
|
//
|
2017-11-03 00:15:13 +01:00
|
|
|
// 1. The column is_readme is either NULL or 1, and if it is 1, then column content cannot
|
|
|
|
// be NULL (but might be an empty BLOB). Vice versa, if column content of a row is,
|
|
|
|
// NULL then column is_readme must be NULL.
|
2017-10-04 15:07:48 +02:00
|
|
|
//
|
|
|
|
// This is to prevent unused content fields filling up the database, and to catch
|
|
|
|
// programmers' errors.
|
|
|
|
_, err = tx.Exec(`
|
2017-11-03 00:15:13 +01:00
|
|
|
ALTER TABLE torrents ADD COLUMN updated_on INTEGER CHECK (updated_on > 0) DEFAULT NULL;
|
|
|
|
ALTER TABLE torrents ADD COLUMN n_seeders INTEGER CHECK ((updated_on IS NOT NULL AND n_seeders >= 0) OR (updated_on IS NULL AND n_seeders IS NULL)) DEFAULT NULL;
|
|
|
|
ALTER TABLE torrents ADD COLUMN n_leechers INTEGER CHECK ((updated_on IS NOT NULL AND n_leechers >= 0) OR (updated_on IS NULL AND n_leechers IS NULL)) DEFAULT NULL;
|
|
|
|
|
2017-10-04 15:07:48 +02:00
|
|
|
ALTER TABLE files ADD COLUMN is_readme INTEGER CHECK (is_readme IS NULL OR is_readme=1) DEFAULT NULL;
|
2017-11-03 00:15:13 +01:00
|
|
|
ALTER TABLE files ADD COLUMN content TEXT CHECK ((content IS NULL AND is_readme IS NULL) OR (content IS NOT NULL AND is_readme=1)) DEFAULT NULL;
|
2017-10-04 15:07:48 +02:00
|
|
|
CREATE UNIQUE INDEX readme_index ON files (torrent_id, is_readme);
|
2017-11-03 00:15:13 +01:00
|
|
|
|
2017-10-04 15:07:48 +02:00
|
|
|
PRAGMA user_version = 2;
|
|
|
|
`)
|
|
|
|
if err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.Tx.Exec (v1 -> v2): %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
2018-03-04 12:07:53 +01:00
|
|
|
fallthrough
|
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
case 2: // NOT FROZEN! (subject to change or complete removal)
|
2018-03-04 12:07:53 +01:00
|
|
|
// Upgrade from user_version 2 to 3
|
|
|
|
// Changes:
|
|
|
|
// * Created `torrents_idx` FTS5 virtual table.
|
2018-04-21 11:05:12 +02:00
|
|
|
//
|
|
|
|
// See:
|
|
|
|
// * https://sqlite.org/fts5.html
|
|
|
|
// * https://sqlite.org/fts3.html
|
|
|
|
//
|
|
|
|
// * Added `n_files` column to the `torrents` table.
|
2018-03-04 12:07:53 +01:00
|
|
|
zap.L().Warn("Updating database schema from 2 to 3... (this might take a while)")
|
2018-04-25 22:33:50 +02:00
|
|
|
_, err = tx.Exec(`
|
|
|
|
CREATE VIRTUAL TABLE IF NOT EXISTS torrents_idx USING fts5(name, content='torrents', content_rowid='id', tokenize="porter unicode61 separators ' !""#$%&''()*+,-./:;<=>?@[\]^_` + "`" + `{|}~'");
|
|
|
|
|
2018-03-04 12:07:53 +01:00
|
|
|
-- Populate the index
|
|
|
|
INSERT INTO torrents_idx(rowid, name) SELECT id, name FROM torrents;
|
|
|
|
|
|
|
|
-- Triggers to keep the FTS index up to date.
|
|
|
|
CREATE TRIGGER torrents_ai AFTER INSERT ON torrents BEGIN
|
|
|
|
INSERT INTO torrents_idx(rowid, name) VALUES (new.id, new.name);
|
|
|
|
END;
|
|
|
|
CREATE TRIGGER torrents_ad AFTER DELETE ON torrents BEGIN
|
|
|
|
INSERT INTO torrents_idx(torrents_idx, rowid, name) VALUES('delete', old.id, old.name);
|
|
|
|
END;
|
|
|
|
CREATE TRIGGER torrents_au AFTER UPDATE ON torrents BEGIN
|
|
|
|
INSERT INTO torrents_idx(torrents_idx, rowid, name) VALUES('delete', old.id, old.name);
|
|
|
|
INSERT INTO torrents_idx(rowid, name) VALUES (new.id, new.name);
|
|
|
|
END;
|
|
|
|
|
2018-04-21 11:05:12 +02:00
|
|
|
-- Add column modified_on
|
|
|
|
ALTER TABLE torrents ADD COLUMN modified_on INTEGER;
|
2018-04-25 22:33:50 +02:00
|
|
|
UPDATE torrents SET modified_on = (SELECT discovered_on);
|
2018-04-21 11:05:12 +02:00
|
|
|
CREATE INDEX modified_on_index ON torrents (modified_on);
|
|
|
|
|
2018-03-04 12:07:53 +01:00
|
|
|
PRAGMA user_version = 3;
|
|
|
|
`)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("sql.Tx.Exec (v2 -> v3): %s", err.Error())
|
|
|
|
}
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if err = tx.Commit(); err != nil {
|
2017-11-03 00:15:13 +01:00
|
|
|
return fmt.Errorf("sql.Tx.Commit: %s", err.Error())
|
2017-10-04 15:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2017-11-03 00:15:13 +01:00
|
|
|
}
|
2018-04-21 11:05:12 +02:00
|
|
|
|
|
|
|
func executeTemplate(text string, data interface{}, funcs template.FuncMap) string {
|
|
|
|
t := template.Must(template.New("anon").Funcs(funcs).Parse(text))
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
err := t.Execute(&buf, data)
|
|
|
|
if err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
return buf.String()
|
|
|
|
}
|