2018-06-19 17:49:46 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2018-07-08 11:08:24 +02:00
|
|
|
"encoding/hex"
|
2018-06-19 17:49:46 +02:00
|
|
|
"encoding/json"
|
2018-06-29 19:08:00 +02:00
|
|
|
"fmt"
|
2018-06-19 17:49:46 +02:00
|
|
|
"net/http"
|
2018-07-07 13:56:34 +02:00
|
|
|
"strconv"
|
2018-06-19 17:49:46 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/boramalper/magnetico/pkg/persistence"
|
2018-07-08 11:08:24 +02:00
|
|
|
"github.com/gorilla/mux"
|
2018-06-19 17:49:46 +02:00
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
func apiTorrents(w http.ResponseWriter, r *http.Request) {
|
2018-06-19 17:49:46 +02:00
|
|
|
// @lastOrderedValue AND @lastID are either both supplied or neither of them should be supplied
|
|
|
|
// at all; and if that is NOT the case, then return an error.
|
2018-08-03 14:40:04 +02:00
|
|
|
if q := r.URL.Query(); !((q.Get("lastOrderedValue") != "" && q.Get("lastID") != "") ||
|
|
|
|
(q.Get("lastOrderedValue") == "" && q.Get("lastID") == "")) {
|
2018-06-19 17:49:46 +02:00
|
|
|
respondError(w, 400, "`lastOrderedValue`, `lastID` must be supplied altogether, if supplied.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-07-12 09:58:39 +02:00
|
|
|
var tq struct {
|
|
|
|
Epoch *int64 `schema:"epoch"`
|
|
|
|
Query *string `schema:"query"`
|
|
|
|
OrderBy *string `schema:"orderBy"`
|
|
|
|
Ascending *bool `schema:"ascending"`
|
|
|
|
LastOrderedValue *float64 `schema:"lastOrderedValue"`
|
|
|
|
LastID *uint64 `schema:"lastID"`
|
2019-05-21 14:31:51 +02:00
|
|
|
Limit *uint `schema:"limit"`
|
2018-07-12 09:58:39 +02:00
|
|
|
}
|
2018-06-19 17:49:46 +02:00
|
|
|
if err := decoder.Decode(&tq, r.URL.Query()); err != nil {
|
|
|
|
respondError(w, 400, "error while parsing the URL: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if tq.Query == nil {
|
|
|
|
tq.Query = new(string)
|
|
|
|
*tq.Query = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if tq.Epoch == nil {
|
|
|
|
tq.Epoch = new(int64)
|
2018-08-03 14:40:04 +02:00
|
|
|
*tq.Epoch = time.Now().Unix() // epoch, if not supplied, is NOW.
|
2018-06-19 17:49:46 +02:00
|
|
|
} else if *tq.Epoch <= 0 {
|
|
|
|
respondError(w, 400, "epoch must be greater than 0")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if tq.Ascending == nil {
|
|
|
|
tq.Ascending = new(bool)
|
|
|
|
*tq.Ascending = true
|
|
|
|
}
|
|
|
|
|
2018-07-01 16:30:06 +02:00
|
|
|
var orderBy persistence.OrderingCriteria
|
|
|
|
if tq.OrderBy == nil {
|
|
|
|
if *tq.Query == "" {
|
|
|
|
orderBy = persistence.ByDiscoveredOn
|
|
|
|
} else {
|
|
|
|
orderBy = persistence.ByRelevance
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
var err error
|
|
|
|
orderBy, err = parseOrderBy(*tq.OrderBy)
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 400, err.Error())
|
|
|
|
return
|
|
|
|
}
|
2018-06-29 19:08:00 +02:00
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
if tq.Limit == nil {
|
|
|
|
tq.Limit = new(uint)
|
|
|
|
*tq.Limit = 20
|
|
|
|
}
|
|
|
|
|
2018-06-19 17:49:46 +02:00
|
|
|
torrents, err := database.QueryTorrents(
|
2018-06-29 19:08:00 +02:00
|
|
|
*tq.Query, *tq.Epoch, orderBy,
|
2019-05-21 14:31:51 +02:00
|
|
|
*tq.Ascending, *tq.Limit, tq.LastOrderedValue, tq.LastID)
|
2018-06-19 17:49:46 +02:00
|
|
|
if err != nil {
|
|
|
|
respondError(w, 400, "query error: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
|
|
if err = json.NewEncoder(w).Encode(torrents); err != nil {
|
|
|
|
zap.L().Warn("JSON encode error", zap.Error(err))
|
2018-06-19 17:49:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
func apiTorrent(w http.ResponseWriter, r *http.Request) {
|
2018-07-08 11:08:24 +02:00
|
|
|
infohashHex := mux.Vars(r)["infohash"]
|
2018-06-19 17:49:46 +02:00
|
|
|
|
2018-07-08 11:08:24 +02:00
|
|
|
infohash, err := hex.DecodeString(infohashHex)
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 400, "couldn't decode infohash: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
torrent, err := database.GetTorrent(infohash)
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 500, "couldn't get torrent: %s", err.Error())
|
|
|
|
return
|
2018-08-27 19:25:17 +02:00
|
|
|
} else if torrent == nil {
|
|
|
|
respondError(w, 404, "not found")
|
|
|
|
return
|
2018-07-08 11:08:24 +02:00
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
|
|
if err = json.NewEncoder(w).Encode(torrent); err != nil {
|
|
|
|
zap.L().Warn("JSON encode error", zap.Error(err))
|
2018-07-08 11:08:24 +02:00
|
|
|
}
|
2018-06-19 17:49:46 +02:00
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
func apiFilelist(w http.ResponseWriter, r *http.Request) {
|
2018-07-08 11:08:24 +02:00
|
|
|
infohashHex := mux.Vars(r)["infohash"]
|
2018-06-19 17:49:46 +02:00
|
|
|
|
2018-07-08 11:08:24 +02:00
|
|
|
infohash, err := hex.DecodeString(infohashHex)
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 400, "couldn't decode infohash: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
files, err := database.GetFiles(infohash)
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 500, "couldn't get files: %s", err.Error())
|
|
|
|
return
|
2018-08-27 19:25:17 +02:00
|
|
|
} else if files == nil {
|
|
|
|
respondError(w, 404, "not found")
|
|
|
|
return
|
2018-07-08 11:08:24 +02:00
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
|
|
if err = json.NewEncoder(w).Encode(files); err != nil {
|
|
|
|
zap.L().Warn("JSON encode error", zap.Error(err))
|
2018-07-08 11:08:24 +02:00
|
|
|
}
|
2018-06-19 17:49:46 +02:00
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
func apiStatistics(w http.ResponseWriter, r *http.Request) {
|
2018-07-07 13:56:34 +02:00
|
|
|
from := r.URL.Query().Get("from")
|
2018-06-19 17:49:46 +02:00
|
|
|
|
2018-07-07 13:56:34 +02:00
|
|
|
// TODO: use gorilla?
|
|
|
|
var n int64
|
|
|
|
nStr := r.URL.Query().Get("n")
|
|
|
|
if nStr == "" {
|
|
|
|
n = 0
|
|
|
|
} else {
|
|
|
|
var err error
|
|
|
|
n, err = strconv.ParseInt(nStr, 10, 32)
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 400, "couldn't parse n: %s", err.Error())
|
|
|
|
return
|
|
|
|
} else if n <= 0 {
|
|
|
|
respondError(w, 400, "n must be a positive number")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stats, err := database.GetStatistics(from, uint(n))
|
|
|
|
if err != nil {
|
|
|
|
respondError(w, 400, "error while getting statistics: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-05-21 14:31:51 +02:00
|
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
|
|
if err = json.NewEncoder(w).Encode(stats); err != nil {
|
|
|
|
zap.L().Warn("JSON encode error", zap.Error(err))
|
2018-07-07 13:56:34 +02:00
|
|
|
}
|
2018-06-19 17:49:46 +02:00
|
|
|
}
|
2018-06-29 19:08:00 +02:00
|
|
|
|
2018-07-01 16:30:06 +02:00
|
|
|
func parseOrderBy(s string) (persistence.OrderingCriteria, error) {
|
|
|
|
switch s {
|
2018-07-07 13:56:34 +02:00
|
|
|
case "RELEVANCE":
|
|
|
|
return persistence.ByRelevance, nil
|
|
|
|
|
2018-06-29 19:08:00 +02:00
|
|
|
case "TOTAL_SIZE":
|
|
|
|
return persistence.ByTotalSize, nil
|
|
|
|
|
|
|
|
case "DISCOVERED_ON":
|
|
|
|
return persistence.ByDiscoveredOn, nil
|
|
|
|
|
|
|
|
case "N_FILES":
|
|
|
|
return persistence.ByNFiles, nil
|
|
|
|
|
|
|
|
case "UPDATED_ON":
|
|
|
|
return persistence.ByUpdatedOn, nil
|
|
|
|
|
|
|
|
case "N_SEEDERS":
|
|
|
|
return persistence.ByNSeeders, nil
|
|
|
|
|
|
|
|
case "N_LEECHERS":
|
|
|
|
return persistence.ByNLeechers, nil
|
|
|
|
|
|
|
|
default:
|
|
|
|
return persistence.ByDiscoveredOn, fmt.Errorf("unknown orderBy string: %s", s)
|
|
|
|
}
|
2018-08-03 14:40:04 +02:00
|
|
|
}
|