package main import ( "bytes" "encoding/hex" "encoding/json" "fmt" "io" "net/http" "os" "strconv" "strings" "time" "github.com/anacrolix/torrent" "github.com/anacrolix/torrent/storage" "github.com/gorilla/mux" "go.uber.org/zap" "golang.org/x/text/encoding/charmap" "github.com/boramalper/magnetico/pkg/persistence" ) type ApiReadmeHandler struct { client *torrent.Client tempdir string } func NewApiReadmeHandler() (*ApiReadmeHandler, error) { h := new(ApiReadmeHandler) var err error h.tempdir, err = os.MkdirTemp("", "magneticod_") if err != nil { return nil, err } config := torrent.NewDefaultClientConfig() config.ListenPort = 0 config.DefaultStorage = storage.NewFileByInfoHash(h.tempdir) h.client, err = torrent.NewClient(config) if err != nil { _ = os.RemoveAll(h.tempdir) return nil, err } return h, nil } func (h *ApiReadmeHandler) Close() { h.client.Close() _ = os.RemoveAll(h.tempdir) } func (h *ApiReadmeHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { infohashHex := mux.Vars(r)["infohash"] infohash, err := hex.DecodeString(infohashHex) if err != nil { respondError(w, http.StatusBadRequest, "couldn't decode infohash: %s", err.Error()) return } files, err := database.GetFiles(infohash) if err != nil { zap.L().Error("GetFiles error", zap.Error(err)) respondError(w, http.StatusInternalServerError, "Internal Server Error") } ok := false for _, file := range files { if strings.HasSuffix(file.Path, ".nfo") { ok = true break } else if strings.Contains(file.Path, "read") { ok = true break } } if !ok { w.WriteHeader(http.StatusNotFound) return } zap.L().Warn("README") t, err := h.client.AddMagnet("magnet:?xt=urn:btih:" + infohashHex) if err != nil { w.WriteHeader(http.StatusBadRequest) return } defer t.Drop() zap.L().Warn("WAITING FOR INFO") select { case <-t.GotInfo(): case <-time.After(30 * time.Second): respondError(w, http.StatusInternalServerError, "Timeout") return } zap.L().Warn("GOT INFO!") t.CancelPieces(0, t.NumPieces()) var file *torrent.File for _, file = range t.Files() { filePath := file.Path() if strings.HasSuffix(filePath, ".nfo") { break } else if strings.Contains(filePath, "read") { break } } if file == nil { w.WriteHeader(http.StatusNotFound) return } // Cancel if the file is larger than 50 KiB if file.Length() > 50*1024 { w.WriteHeader(http.StatusRequestEntityTooLarge) return } file.Download() reader := file.NewReader() // BEWARE: // ioutil.ReadAll(reader) // returns some adjancent garbage too, for reasons unknown... content := make([]byte, file.Length()) _, err = io.ReadFull(reader, content) if err != nil { w.WriteHeader(http.StatusInternalServerError) return } defer reader.Close() if strings.HasSuffix(file.Path(), ".nfo") { content, err = charmap.CodePage437.NewDecoder().Bytes(content) if err != nil { w.WriteHeader(http.StatusInternalServerError) return } } // Because .nfo files are right padded with \x00'es. content = bytes.TrimRight(content, "\x00") w.Header().Set("Content-Type", "text/plain;charset=UTF-8") _, _ = w.Write(content) } func apiTorrents(w http.ResponseWriter, r *http.Request) { // @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. if q := r.URL.Query(); !((q.Get("lastOrderedValue") != "" && q.Get("lastID") != "") || (q.Get("lastOrderedValue") == "" && q.Get("lastID") == "")) { respondError(w, 400, "`lastOrderedValue`, `lastID` must be supplied altogether, if supplied.") return } 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"` Limit *uint `schema:"limit"` } 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) *tq.Epoch = time.Now().Unix() // epoch, if not supplied, is NOW. } 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 } 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 } } if tq.Limit == nil { tq.Limit = new(uint) *tq.Limit = 20 } torrents, err := database.QueryTorrents( *tq.Query, *tq.Epoch, orderBy, *tq.Ascending, *tq.Limit, tq.LastOrderedValue, tq.LastID) if err != nil { respondError(w, 400, "query error: %s", err.Error()) return } 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)) } } func apiTorrent(w http.ResponseWriter, r *http.Request) { infohashHex := mux.Vars(r)["infohash"] 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 } else if torrent == nil { respondError(w, 404, "not found") return } 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)) } } func apiFilelist(w http.ResponseWriter, r *http.Request) { infohashHex := mux.Vars(r)["infohash"] 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 } else if files == nil { respondError(w, 404, "not found") return } 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)) } } func apiStatistics(w http.ResponseWriter, r *http.Request) { from := r.URL.Query().Get("from") // 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 } 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)) } } func parseOrderBy(s string) (persistence.OrderingCriteria, error) { switch s { case "RELEVANCE": return persistence.ByRelevance, nil 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) } }