Files
fnx_web/webcontroller/file_viewer.go

320 lines
8.5 KiB
Go
Raw Normal View History

2017-11-10 12:39:55 +01:00
package webcontroller
import (
2017-12-12 23:33:41 +01:00
"fmt"
2020-06-10 18:11:00 +02:00
"math/rand"
2017-11-10 12:39:55 +01:00
"net/http"
"strings"
2020-01-28 12:51:21 +01:00
"time"
2017-11-10 12:39:55 +01:00
2021-03-10 20:13:32 +01:00
"fornaxian.tech/pixeldrain_api_client/pixelapi"
"github.com/Fornaxian/log"
2017-11-10 12:39:55 +01:00
"github.com/julienschmidt/httprouter"
)
2020-02-21 14:32:06 +01:00
func (wc *WebController) viewTokenOrBust() (t string) {
2020-01-21 15:33:09 +01:00
var err error
2021-03-04 17:10:59 +01:00
if t, err = wc.api.GetMiscViewToken(); err != nil && !wc.proxyAPIRequests {
2020-01-21 15:33:09 +01:00
log.Error("Could not get viewtoken: %s", err)
}
return t
}
2020-01-28 12:51:21 +01:00
func browserCompat(ua string) bool {
return strings.Contains(ua, "MSIE") || strings.Contains(ua, "Trident/7.0")
}
2021-06-10 18:00:04 +02:00
type viewerData struct {
Type string // file or list
CaptchaKey string
ViewToken string
AdBannerType int
AdSkyscraperType string
AdFloaterType int
AdPopupType int
FileAdsEnabled bool
UserAdsEnabled bool
Embedded bool
APIResponse interface{}
2021-06-10 18:00:04 +02:00
}
func (vd *viewerData) adType(files []pixelapi.ListFile) {
2021-06-14 14:56:04 +02:00
if len(files) == 0 {
return
} else if !vd.FileAdsEnabled || !vd.UserAdsEnabled {
return
2021-06-14 14:56:04 +02:00
}
2021-06-10 18:00:04 +02:00
var avgSize int64
2021-09-15 22:56:53 +02:00
var nudity = false
2021-06-10 18:00:04 +02:00
for _, v := range files {
avgSize += v.Size
2021-09-15 22:56:53 +02:00
if strings.HasPrefix(v.MimeType, "video/") {
nudity = true
}
2021-06-10 18:00:04 +02:00
}
avgSize /= int64(len(files))
2020-10-27 08:25:03 +01:00
const (
2021-05-25 11:43:16 +02:00
// Banners
2021-06-10 18:00:04 +02:00
none = 0
aAds = 1
2020-10-27 08:25:03 +01:00
patreon = 2
soulStudio = 3
amarulaSolutions = 4
adMaven = 5
2021-05-25 11:43:16 +02:00
adSterra = 6
2021-05-05 20:20:34 +02:00
brave = 7
2021-01-11 22:31:26 +01:00
pdpro1 = 8
pdpro2 = 9
pdpro3 = 10
pdpro4 = 11
2021-06-08 16:43:20 +02:00
clickAduBanner = 12
2021-06-10 18:00:04 +02:00
amarulaElectronics = 13
2021-09-03 17:57:12 +02:00
adsPlus = 14
pixFuture = 15
publisherrest1 = 16
publisherrest2 = 17
publisherrest3 = 18
// Skyscrapers
aAdsSkyscraper = "a-ads"
pixfutureSkyscraper = "pixfuture"
2021-05-25 11:43:16 +02:00
// Floaters
// propellerFloat = 1
// adSterraFloat = 2
// adMavenFloat = 3
2021-06-10 18:00:04 +02:00
// Popunders
2021-09-21 21:39:28 +02:00
// clickAduPopup = 1
// propellerPopup = 2
2020-10-27 08:25:03 +01:00
)
2020-06-10 18:11:00 +02:00
vd.AdSkyscraperType = aAdsSkyscraper
2020-11-17 10:58:11 +01:00
// Intn returns a number up to n, but never n itself. So to get a random 0
2020-11-08 16:10:11 +01:00
// or 1 we need to give it n=2. We can use this function to make other
// splits like 1/3 1/4, etc
2021-09-15 22:56:53 +02:00
if nudity {
// Brave and a-ads don't care about nudity. I'm not sure about ads.plus
2021-09-23 20:38:17 +02:00
switch i := rand.Intn(10); i {
case 0, 1:
2021-09-15 22:56:53 +02:00
vd.AdBannerType = brave
case 2, 3, 4, 5, 6:
2021-09-20 12:10:32 +02:00
vd.AdBannerType = adsPlus
case 7:
vd.AdBannerType = publisherrest1
case 8:
vd.AdBannerType = publisherrest2
case 9:
vd.AdBannerType = publisherrest3
2021-09-15 22:56:53 +02:00
default:
panic(fmt.Errorf("random number generator returned unrecognised number: %d", i))
}
} else {
// PixFuture does not allow nudity, so that's what we'll show on all
// files which are safe
switch i := rand.Intn(10); i {
2021-09-23 20:38:17 +02:00
case 0, 1:
2021-09-15 22:56:53 +02:00
vd.AdBannerType = brave
case 2, 3:
2021-09-15 22:56:53 +02:00
vd.AdBannerType = adsPlus
case 4, 5, 6:
2021-09-15 22:56:53 +02:00
vd.AdBannerType = pixFuture
case 7:
vd.AdBannerType = publisherrest1
case 8:
vd.AdBannerType = publisherrest2
case 9:
vd.AdBannerType = publisherrest3
2021-09-15 22:56:53 +02:00
default:
panic(fmt.Errorf("random number generator returned unrecognised number: %d", i))
}
2021-05-25 11:43:16 +02:00
}
}
2017-11-10 12:39:55 +01:00
// ServeFileViewer controller for GET /u/:id
func (wc *WebController) serveFileViewer(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
2020-01-21 15:33:09 +01:00
var err error
2017-11-10 12:39:55 +01:00
if p.ByName("id") == "demo" {
wc.serveFileViewerDemo(w, r, 1, "a-ads") // Required for a-ads.com quality check
2021-09-03 17:57:12 +02:00
return
} else if p.ByName("id") == "adsplus" {
wc.serveFileViewerDemo(w, r, 14, "")
2021-09-03 17:57:12 +02:00
return
} else if p.ByName("id") == "pixfuture" {
wc.serveFileViewerDemo(w, r, 15, "")
2017-11-10 12:39:55 +01:00
return
}
// If the user agent is Wget we redirect it to the API so that the file can
// be downloaded directly
if strings.HasPrefix(r.UserAgent(), "Wget/") {
http.Redirect(w, r, "/api/file/"+p.ByName("id"), http.StatusSeeOther)
return
}
2020-01-28 12:51:21 +01:00
var ids = strings.Split(p.ByName("id"), ",")
2017-11-10 12:39:55 +01:00
2020-01-21 15:33:09 +01:00
templateData := wc.newTemplateData(w, r)
2021-03-10 20:13:32 +01:00
var files []pixelapi.ListFile
2017-11-10 12:39:55 +01:00
for _, id := range ids {
2020-01-21 15:43:09 +01:00
inf, err := templateData.PixelAPI.GetFileInfo(id)
if err != nil {
2021-03-10 20:13:32 +01:00
if pixelapi.ErrIsServerError(err) {
2020-07-20 23:24:49 +02:00
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
return
}
2017-11-10 12:39:55 +01:00
continue
}
2021-03-10 20:13:32 +01:00
files = append(files, pixelapi.ListFile{FileInfo: inf})
2017-11-10 12:39:55 +01:00
}
if len(files) == 0 {
w.WriteHeader(http.StatusNotFound)
wc.templates.Get().ExecuteTemplate(w, "file_not_found", templateData)
2017-11-10 12:39:55 +01:00
return
}
templateData.OGData = wc.metadataFromFile(files[0].FileInfo)
2020-10-27 11:26:41 +01:00
var vd = viewerData{
CaptchaKey: wc.captchaKey(),
ViewToken: wc.viewTokenOrBust(),
FileAdsEnabled: files[0].ShowAds,
2020-10-27 11:26:41 +01:00
UserAdsEnabled: !(templateData.Authenticated && templateData.User.Subscription.DisableAdDisplay),
}
2021-06-10 18:00:04 +02:00
vd.adType(files)
2021-03-10 20:13:32 +01:00
2020-01-28 12:51:21 +01:00
if len(ids) > 1 {
templateData.Title = fmt.Sprintf("%d files on pixeldrain", len(files))
2020-10-27 11:26:41 +01:00
vd.Type = "list"
2021-03-10 20:13:32 +01:00
vd.APIResponse = pixelapi.ListInfo{
2020-10-27 11:26:41 +01:00
Success: true,
Title: "Multiple files",
DateCreated: time.Now(),
Files: files,
2017-11-10 12:39:55 +01:00
}
} else {
templateData.Title = fmt.Sprintf("%s ~ pixeldrain", files[0].Name)
2020-10-27 11:26:41 +01:00
vd.Type = "file"
vd.APIResponse = files[0].FileInfo
2017-11-10 12:39:55 +01:00
}
2021-03-10 20:13:32 +01:00
if _, ok := r.URL.Query()["embed"]; ok {
vd.Embedded = true
}
2020-10-27 11:26:41 +01:00
templateData.Other = vd
2020-01-28 12:51:21 +01:00
var templateName = "file_viewer"
if browserCompat(r.UserAgent()) {
templateName = "file_viewer_compat"
}
for _, file := range files {
if file.AbuseType != "" {
w.WriteHeader(http.StatusUnavailableForLegalReasons)
break
}
}
2020-01-28 12:51:21 +01:00
err = wc.templates.Get().ExecuteTemplate(w, templateName, templateData)
2019-09-18 22:30:29 +02:00
if err != nil && !strings.Contains(err.Error(), "broken pipe") {
2017-11-10 12:39:55 +01:00
log.Error("Error executing template file_viewer: %s", err)
}
}
// ServeFileViewerDemo is a dummy API response that responds with info about a
// non-existent demo file. This is required by the a-ads ad network to allow for
// automatic checking of the presence of the ad unit on this page.
func (wc *WebController) serveFileViewerDemo(w http.ResponseWriter, r *http.Request, banner int, scraper string) {
templateData := wc.newTemplateData(w, r)
templateData.Other = viewerData{
Type: "file",
CaptchaKey: wc.captchaSiteKey,
AdBannerType: banner, // Always show a-ads on the demo page
AdSkyscraperType: scraper,
FileAdsEnabled: true,
UserAdsEnabled: true,
APIResponse: map[string]interface{}{
2019-03-26 20:53:19 +01:00
"id": "demo",
"name": "Demo file",
"date_upload": "2017-01-01 12:34:56",
"date_lastview": "2017-01-01 12:34:56",
"size": 123456789,
"views": 1,
"bandwidth_used": 123456789,
"mime_type": "text/demo",
"description": "A file to demonstrate the viewer page",
"mime_image": "/res/img/mime/text.png",
"thumbnail": "/res/img/mime/text.png",
2020-08-27 14:41:26 +02:00
"abuse_type": "",
},
}
err := wc.templates.Get().ExecuteTemplate(w, "file_viewer", templateData)
2019-09-18 22:30:29 +02:00
if err != nil && !strings.Contains(err.Error(), "broken pipe") {
log.Error("Error rendering demo file: %s", err)
}
}
2020-01-21 15:33:09 +01:00
// ServeListViewer controller for GET /l/:id
func (wc *WebController) serveListViewer(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
var templateData = wc.newTemplateData(w, r)
2021-03-10 20:13:32 +01:00
var list, err = templateData.PixelAPI.GetListID(p.ByName("id"))
2020-01-21 15:33:09 +01:00
if err != nil {
2021-03-10 20:13:32 +01:00
if err, ok := err.(pixelapi.Error); ok && err.Status == http.StatusNotFound {
2020-05-05 22:03:34 +02:00
w.WriteHeader(http.StatusNotFound)
wc.templates.Get().ExecuteTemplate(w, "list_not_found", templateData)
} else {
log.Error("API request error occurred: %s", err)
w.WriteHeader(http.StatusInternalServerError)
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
2020-01-21 15:33:09 +01:00
}
return
}
2020-06-07 21:12:48 +02:00
if len(list.Files) == 0 {
w.WriteHeader(http.StatusNotFound)
wc.templates.Get().ExecuteTemplate(w, "list_not_found", templateData)
2020-07-10 10:50:56 +02:00
return
2020-06-07 21:12:48 +02:00
}
2020-01-21 15:33:09 +01:00
templateData.Title = fmt.Sprintf("%s ~ pixeldrain", list.Title)
2021-01-12 23:20:32 +01:00
templateData.OGData = wc.metadataFromList(list)
2021-03-10 20:13:32 +01:00
var vd = viewerData{
2020-10-27 11:26:41 +01:00
Type: "list",
CaptchaKey: wc.captchaSiteKey,
ViewToken: wc.viewTokenOrBust(),
FileAdsEnabled: list.Files[0].ShowAds,
UserAdsEnabled: !(templateData.Authenticated && templateData.User.Subscription.DisableAdDisplay),
APIResponse: list,
2020-01-21 15:33:09 +01:00
}
2021-06-10 18:00:04 +02:00
vd.adType(list.Files)
2020-01-28 12:51:21 +01:00
2021-03-10 20:13:32 +01:00
if _, ok := r.URL.Query()["embed"]; ok {
vd.Embedded = true
}
templateData.Other = vd
2020-01-28 12:51:21 +01:00
var templateName = "file_viewer"
if browserCompat(r.UserAgent()) {
templateName = "file_viewer_compat"
}
for _, file := range list.Files {
if file.AbuseType != "" {
w.WriteHeader(http.StatusUnavailableForLegalReasons)
break
}
}
2020-01-28 12:51:21 +01:00
err = wc.templates.Get().ExecuteTemplate(w, templateName, templateData)
2020-01-21 15:33:09 +01:00
if err != nil && !strings.Contains(err.Error(), "broken pipe") {
log.Error("Error executing template file_viewer: %s", err)
}
}