Albirew/nyaa-pantsu
Albirew
/
nyaa-pantsu
Archivé
1
0
Bifurcation 0
Ce dépôt a été archivé le 2022-05-07. Vous pouvez voir ses fichiers ou le cloner, mais pas ouvrir de ticket ou de demandes d'ajout, ni soumettre de changements.
nyaa-pantsu/main.go

342 lignes
9.5 KiB
Go
Brut Vue normale Historique

2017-05-02 12:39:53 +02:00
package main
import (
2017-05-05 03:57:08 +02:00
"bytes"
"compress/zlib"
2017-05-02 12:39:53 +02:00
"encoding/json"
2017-05-04 21:48:40 +02:00
"github.com/gorilla/feeds"
2017-05-02 12:39:53 +02:00
"github.com/gorilla/mux"
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/sqlite"
2017-05-02 12:39:53 +02:00
"html"
"html/template"
2017-05-05 03:57:08 +02:00
"io/ioutil"
2017-05-02 12:39:53 +02:00
"log"
"net/http"
"strconv"
2017-05-03 06:59:27 +02:00
"strings"
2017-05-02 12:39:53 +02:00
"time"
)
2017-05-05 13:37:22 +02:00
type SearchParam struct {
Category string
Order string
Query string
Max int
Status string
Sort string
}
var db *gorm.DB
var router *mux.Router
2017-05-02 12:39:53 +02:00
var debugLogger *log.Logger
var trackers = "&tr=udp://tracker.coppersurfer.tk:6969&tr=udp://zer0day.to:1337/announce&tr=udp://tracker.leechers-paradise.org:6969&tr=udp://explodie.org:6969&tr=udp://tracker.opentrackr.org:1337&tr=http://tracker.baka-sub.cf/announce"
2017-05-02 12:39:53 +02:00
func getDBHandle() *gorm.DB {
dbInit, err := gorm.Open("sqlite3", "./nyaa.db")
2017-05-02 12:39:53 +02:00
// Migrate the schema of Torrents
2017-05-05 06:24:58 +02:00
dbInit.AutoMigrate(&Torrents{}, &Categories{}, &Sub_Categories{}, &Statuses{})
2017-05-02 12:39:53 +02:00
checkErr(err)
return dbInit
2017-05-02 12:39:53 +02:00
}
func checkErr(err error) {
if err != nil {
debugLogger.Println(" " + err.Error())
}
}
2017-05-05 03:57:08 +02:00
func unZlib(description []byte) string {
if len(description) > 0 {
b := bytes.NewReader(description)
2017-05-05 11:04:28 +02:00
//log.Println(b)
z, err := zlib.NewReader(b)
checkErr(err)
defer z.Close()
p, err := ioutil.ReadAll(z)
checkErr(err)
return string(p)
}
return ""
2017-05-05 03:57:08 +02:00
}
2017-05-02 12:39:53 +02:00
func apiHandler(w http.ResponseWriter, r *http.Request) {
vars := mux.Vars(r)
page := vars["page"]
pagenum, _ := strconv.Atoi(html.EscapeString(page))
b := CategoryJson{Torrents: []TorrentsJson{}}
maxPerPage := 50
nbTorrents := 0
torrents, nbTorrents := getAllTorrents(maxPerPage, maxPerPage*(pagenum-1))
for i, _ := range torrents {
res := torrents[i].toJson()
b.Torrents = append(b.Torrents, res)
2017-05-02 12:39:53 +02:00
}
b.QueryRecordCount = maxPerPage
b.TotalRecordCount = nbTorrents
2017-05-02 12:39:53 +02:00
w.Header().Set("Content-Type", "application/json")
err := json.NewEncoder(w).Encode(b)
2017-05-02 12:39:53 +02:00
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
}
2017-05-05 02:54:37 +02:00
func apiViewHandler(w http.ResponseWriter, r *http.Request) {
2017-05-02 12:39:53 +02:00
vars := mux.Vars(r)
id := vars["id"]
b := CategoryJson{Torrents: []TorrentsJson{}}
torrent, err := getTorrentById(id)
res := torrent.toJson()
b.Torrents = append(b.Torrents, res)
2017-05-02 12:39:53 +02:00
b.QueryRecordCount = 1
b.TotalRecordCount = 1
2017-05-02 12:39:53 +02:00
w.Header().Set("Content-Type", "application/json")
err = json.NewEncoder(w).Encode(b)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
}
func searchHandler(w http.ResponseWriter, r *http.Request) {
var templates = template.Must(template.New("home").Funcs(funcMap).ParseFiles("templates/index.html", "templates/home.html"))
templates.ParseGlob("templates/_*.html") // common
2017-05-02 12:39:53 +02:00
vars := mux.Vars(r)
page := vars["page"]
// db params url
2017-05-02 12:39:53 +02:00
pagenum, _ := strconv.Atoi(html.EscapeString(page))
if pagenum == 0 {
pagenum = 1
}
2017-05-05 13:37:22 +02:00
b := []TorrentsJson{}
search_param, torrents, nbTorrents := searchByQuery( r, pagenum )
2017-05-05 06:24:58 +02:00
2017-05-05 13:37:22 +02:00
for i, _ := range torrents {
res := torrents[i].toJson()
b = append(b, res)
}
navigationTorrents := Navigation{nbTorrents, search_param.Max, pagenum, "search_page"}
searchForm := SearchForm{
search_param.Query,
search_param.Status,
search_param.Category,
search_param.Sort,
search_param.Order,
}
htv := HomeTemplateVariables{b, getAllCategories(false), searchForm, navigationTorrents, r.URL, mux.CurrentRoute(r)}
err := templates.ExecuteTemplate(w, "index.html", htv)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
func searchByQuery(r *http.Request, pagenum int) (SearchParam, []Torrents, int) {
maxPerPage, errConv := strconv.Atoi(r.URL.Query().Get("max"))
if errConv != nil {
maxPerPage = 50 // default Value maxPerPage
}
search_param := SearchParam{}
search_param.Max = maxPerPage
search_param.Query = r.URL.Query().Get("q")
search_param.Category = r.URL.Query().Get("c")
search_param.Status = r.URL.Query().Get("s")
search_param.Sort = r.URL.Query().Get("sort")
search_param.Order = r.URL.Query().Get("order")
catsSplit := strings.Split(search_param.Category, "_")
// need this to prevent out of index panics
var searchCatId, searchSubCatId string
if len(catsSplit) == 2 {
searchCatId = html.EscapeString(catsSplit[0])
searchSubCatId = html.EscapeString(catsSplit[1])
}
2017-05-05 13:37:22 +02:00
if search_param.Sort == "" {
search_param.Sort = "torrent_id"
2017-05-05 06:24:58 +02:00
}
2017-05-05 13:37:22 +02:00
if search_param.Order == "" {
search_param.Order = "desc"
2017-05-05 06:24:58 +02:00
}
2017-05-05 13:37:22 +02:00
order_by := search_param.Sort + " " + search_param.Order
2017-05-04 00:54:07 +02:00
2017-05-05 11:04:28 +02:00
parameters := WhereParams{}
conditions := []string{}
if searchCatId != "" {
conditions = append(conditions, "category_id = ?")
parameters.params = append(parameters.params, searchCatId)
}
if searchSubCatId != "" {
conditions = append(conditions, "sub_category_id = ?")
parameters.params = append(parameters.params, searchSubCatId)
}
2017-05-05 13:37:22 +02:00
if search_param.Status != "" {
2017-05-05 11:04:28 +02:00
conditions = append(conditions, "status_id = ?")
2017-05-05 13:37:22 +02:00
parameters.params = append(parameters.params, search_param.Status)
2017-05-05 11:04:28 +02:00
}
2017-05-05 13:37:22 +02:00
searchQuerySplit := strings.Split(search_param.Query, " ")
2017-05-05 11:04:28 +02:00
for i, _ := range searchQuerySplit {
conditions = append(conditions, "torrent_name LIKE ?")
parameters.params = append(parameters.params, "%"+searchQuerySplit[i]+"%")
}
parameters.conditions = strings.Join(conditions[:], " AND ")
log.Printf("SQL query is :: %s\n", parameters.conditions)
2017-05-05 13:37:22 +02:00
torrents, n := getTorrentsOrderBy(&parameters, order_by, maxPerPage, maxPerPage*(pagenum-1))
return search_param, torrents, n
2017-05-02 12:39:53 +02:00
}
2017-05-05 13:37:22 +02:00
2017-05-03 09:33:39 +02:00
func safe(s string) template.URL {
return template.URL(s)
}
2017-05-02 12:39:53 +02:00
2017-05-04 01:15:20 +02:00
func faqHandler(w http.ResponseWriter, r *http.Request) {
var templates = template.Must(template.New("FAQ").Funcs(funcMap).ParseFiles("templates/index.html", "templates/FAQ.html"))
templates.ParseGlob("templates/_*.html") // common
err := templates.ExecuteTemplate(w, "index.html", FaqTemplateVariables{Navigation{}, NewSearchForm(), r.URL, mux.CurrentRoute(r)})
2017-05-04 01:15:20 +02:00
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
2017-05-04 21:48:40 +02:00
func rssHandler(w http.ResponseWriter, r *http.Request) {
2017-05-05 13:37:22 +02:00
_, torrents, _ := searchByQuery( r, 1 )
created_as_time := time.Now()
2017-05-05 06:24:58 +02:00
if len(torrents) > 0 {
2017-05-05 13:37:22 +02:00
created_as_time = time.Unix(torrents[0].Date, 0)
2017-05-04 21:48:40 +02:00
}
feed := &feeds.Feed{
2017-05-05 06:24:58 +02:00
Title: "Nyaa Pantsu",
Link: &feeds.Link{Href: "https://nyaa.pantsu.cat/"},
Created: created_as_time,
2017-05-04 21:48:40 +02:00
}
feed.Items = []*feeds.Item{}
2017-05-05 06:24:58 +02:00
feed.Items = make([]*feeds.Item, len(torrents))
2017-05-04 21:48:40 +02:00
for i, _ := range torrents {
2017-05-05 13:37:22 +02:00
timestamp_as_time := time.Unix(torrents[0].Date, 0)
torrent_json := torrents[i].toJson()
feed.Items[i] = &feeds.Item{
// need a torrent view first
//Id: URL + torrents[i].Hash,
Title: torrents[i].Name,
Link: &feeds.Link{Href: string(torrent_json.Magnet)},
Description: "",
Created: timestamp_as_time,
Updated: timestamp_as_time,
2017-05-04 21:48:40 +02:00
}
}
rss, err := feed.ToRss()
if err == nil {
2017-05-05 06:24:58 +02:00
w.Write([]byte(rss))
2017-05-04 21:48:40 +02:00
} else {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
2017-05-05 13:37:22 +02:00
2017-05-05 02:54:37 +02:00
func viewHandler(w http.ResponseWriter, r *http.Request) {
var templates = template.Must(template.ParseFiles("templates/index.html", "templates/view.html"))
templates.ParseGlob("templates/_*.html") // common
2017-05-05 02:54:37 +02:00
vars := mux.Vars(r)
id := vars["id"]
torrent, err := getTorrentById(id)
b := torrent.toJson()
2017-05-05 02:54:37 +02:00
htv := ViewTemplateVariables{b, NewSearchForm(), Navigation{}, r.URL, mux.CurrentRoute(r)}
2017-05-05 02:54:37 +02:00
err = templates.ExecuteTemplate(w, "index.html", htv)
2017-05-04 01:15:20 +02:00
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
2017-05-02 12:39:53 +02:00
func rootHandler(w http.ResponseWriter, r *http.Request) {
var templates = template.Must(template.New("home").Funcs(funcMap).ParseFiles("templates/index.html", "templates/home.html"))
templates.ParseGlob("templates/_*.html") // common
2017-05-02 12:39:53 +02:00
vars := mux.Vars(r)
page := vars["page"]
2017-05-04 00:20:50 +02:00
// db params url
maxPerPage, errConv := strconv.Atoi(r.URL.Query().Get("max"))
if errConv != nil {
maxPerPage = 50 // default Value maxPerPage
2017-05-04 00:20:50 +02:00
}
nbTorrents := 0
2017-05-02 12:39:53 +02:00
pagenum, _ := strconv.Atoi(html.EscapeString(page))
if pagenum == 0 {
pagenum = 1
}
b := []TorrentsJson{}
torrents, nbTorrents := getAllTorrents(maxPerPage, maxPerPage*(pagenum-1))
for i, _ := range torrents {
res := torrents[i].toJson()
b = append(b, res)
2017-05-02 12:39:53 +02:00
}
navigationTorrents := Navigation{nbTorrents, maxPerPage, pagenum, "search_page"}
htv := HomeTemplateVariables{b, getAllCategories(false), NewSearchForm(), navigationTorrents, r.URL, mux.CurrentRoute(r)}
err := templates.ExecuteTemplate(w, "index.html", htv)
2017-05-02 12:39:53 +02:00
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
func main() {
db = getDBHandle()
router = mux.NewRouter()
2017-05-02 12:39:53 +02:00
cssHandler := http.FileServer(http.Dir("./css/"))
jsHandler := http.FileServer(http.Dir("./js/"))
imgHandler := http.FileServer(http.Dir("./img/"))
2017-05-04 00:20:50 +02:00
http.Handle("/css/", http.StripPrefix("/css/", cssHandler))
http.Handle("/js/", http.StripPrefix("/js/", jsHandler))
http.Handle("/img/", http.StripPrefix("/img/", imgHandler))
2017-05-02 12:39:53 +02:00
// Routes,
router.HandleFunc("/", rootHandler).Name("home")
router.HandleFunc("/page/{page:[0-9]+}", rootHandler).Name("home_page")
router.HandleFunc("/search", searchHandler).Name("search")
router.HandleFunc("/search/{page}", searchHandler).Name("search_page")
2017-05-02 12:39:53 +02:00
router.HandleFunc("/api/{page}", apiHandler).Methods("GET")
2017-05-05 02:54:37 +02:00
router.HandleFunc("/api/view/{id}", apiViewHandler).Methods("GET")
router.HandleFunc("/faq", faqHandler).Name("faq")
2017-05-04 21:48:40 +02:00
router.HandleFunc("/feed.xml", rssHandler)
router.HandleFunc("/view/{id}", viewHandler).Name("view_torrent")
http.Handle("/", router)
2017-05-02 12:39:53 +02:00
// Set up server,
srv := &http.Server{
Addr: "localhost:9999",
WriteTimeout: 15 * time.Second,
ReadTimeout: 15 * time.Second,
}
err := srv.ListenAndServe()
checkErr(err)
}