2017-11-10 12:39:55 +01:00
|
|
|
package webcontroller
|
|
|
|
|
|
|
|
import (
|
2017-12-12 23:33:41 +01:00
|
|
|
"fmt"
|
2020-02-04 19:37:46 +01:00
|
|
|
"io/ioutil"
|
2020-06-10 18:11:00 +02:00
|
|
|
"math/rand"
|
2020-02-04 22:29:48 +01:00
|
|
|
"mime"
|
2017-11-10 12:39:55 +01:00
|
|
|
"net/http"
|
2020-02-04 19:37:46 +01:00
|
|
|
"strconv"
|
2017-11-10 12:39:55 +01:00
|
|
|
"strings"
|
2020-01-28 12:51:21 +01:00
|
|
|
"time"
|
2017-11-10 12:39:55 +01:00
|
|
|
|
2020-07-29 17:27:36 +02:00
|
|
|
"fornaxian.tech/pixeldrain_server/api/restapi/apiclient"
|
|
|
|
"fornaxian.tech/pixeldrain_server/api/restapi/apitype"
|
2018-06-17 16:15:58 +02:00
|
|
|
"github.com/Fornaxian/log"
|
2020-06-07 21:12:48 +02:00
|
|
|
pdmimetype "github.com/Fornaxian/pd_mime_type"
|
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-01-05 00:00:46 +01:00
|
|
|
if t, err = wc.api.GetMiscViewToken(); err != nil {
|
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")
|
|
|
|
}
|
|
|
|
|
2020-11-08 16:10:11 +01:00
|
|
|
func adType() int {
|
2020-10-27 08:25:03 +01:00
|
|
|
const (
|
2021-01-26 19:01:00 +01:00
|
|
|
aAds = 0
|
2020-10-27 08:25:03 +01:00
|
|
|
amarulaElectronics = 1
|
|
|
|
patreon = 2
|
|
|
|
soulStudio = 3
|
|
|
|
amarulaSolutions = 4
|
|
|
|
adMaven = 5
|
2020-11-26 11:13:27 +01:00
|
|
|
propellerAds = 7
|
2021-01-11 22:31:26 +01:00
|
|
|
pdpro1 = 8
|
|
|
|
pdpro2 = 9
|
|
|
|
pdpro3 = 10
|
|
|
|
pdpro4 = 11
|
2020-10-27 08:25:03 +01:00
|
|
|
)
|
2020-06-10 18:11:00 +02:00
|
|
|
|
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-01-26 19:01:00 +01:00
|
|
|
switch i := rand.Intn(20); i {
|
2021-02-14 10:06:48 +01:00
|
|
|
case 0, 1: // 10%
|
2020-10-27 08:25:03 +01:00
|
|
|
return amarulaSolutions
|
2021-02-16 19:13:15 +01:00
|
|
|
case 2: // 5%, also shows propellerads
|
2021-01-11 22:31:26 +01:00
|
|
|
return pdpro1
|
2021-02-16 19:13:15 +01:00
|
|
|
case 3: // 5%, also shows propellerads
|
2021-01-11 22:31:26 +01:00
|
|
|
return pdpro2
|
2021-02-16 19:13:15 +01:00
|
|
|
case 4: // 5%, also shows propellerads
|
2021-01-11 22:31:26 +01:00
|
|
|
return pdpro3
|
2021-02-16 19:13:15 +01:00
|
|
|
case 5: // 5%, also shows propellerads
|
2021-01-11 22:31:26 +01:00
|
|
|
return pdpro4
|
2021-02-16 19:13:15 +01:00
|
|
|
default: // 70%, also shows a-ads
|
2020-11-26 11:13:27 +01:00
|
|
|
return propellerAds
|
2021-02-14 10:06:48 +01:00
|
|
|
// default:
|
|
|
|
// panic(fmt.Errorf("random number generator returned unrecognised number: %d", i))
|
2020-06-10 18:11:00 +02:00
|
|
|
}
|
2020-05-24 14:23:03 +02:00
|
|
|
}
|
|
|
|
|
2019-02-19 23:10:04 +01:00
|
|
|
type viewerData struct {
|
2020-10-27 11:26:41 +01:00
|
|
|
Type string // file or list
|
|
|
|
CaptchaKey string
|
|
|
|
ViewToken string
|
|
|
|
AdType int
|
|
|
|
FileAdsEnabled bool
|
|
|
|
UserAdsEnabled bool
|
|
|
|
APIResponse interface{}
|
2019-02-19 23:10:04 +01:00
|
|
|
}
|
|
|
|
|
2017-11-10 12:39:55 +01:00
|
|
|
// ServeFileViewer controller for GET /u/:id
|
2018-06-17 16:15:58 +02:00
|
|
|
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" {
|
2019-02-19 23:10:04 +01:00
|
|
|
wc.serveFileViewerDemo(w, r) // Required for a-ads.com quality check
|
2017-11-10 12:39:55 +01:00
|
|
|
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)
|
|
|
|
|
2020-06-07 21:12:48 +02:00
|
|
|
var finfo []apitype.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)
|
2018-06-20 23:47:47 +02:00
|
|
|
if err != nil {
|
2020-07-20 23:24:49 +02:00
|
|
|
if apiclient.ErrIsServerError(err) {
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
|
|
|
|
return
|
|
|
|
}
|
2017-11-10 12:39:55 +01:00
|
|
|
continue
|
|
|
|
}
|
2020-06-07 21:12:48 +02:00
|
|
|
finfo = append(finfo, apitype.ListFile{FileInfo: inf})
|
2017-11-10 12:39:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(finfo) == 0 {
|
2019-07-28 22:34:15 +02:00
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "file_not_found", templateData)
|
2017-11-10 12:39:55 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-01-12 23:20:32 +01:00
|
|
|
templateData.OGData = wc.metadataFromFile(finfo[0].FileInfo)
|
2020-10-27 11:26:41 +01:00
|
|
|
|
|
|
|
var vd = viewerData{
|
|
|
|
CaptchaKey: wc.captchaKey(),
|
|
|
|
ViewToken: wc.viewTokenOrBust(),
|
|
|
|
AdType: adType(),
|
|
|
|
FileAdsEnabled: finfo[0].ShowAds,
|
|
|
|
UserAdsEnabled: !(templateData.Authenticated && templateData.User.Subscription.DisableAdDisplay),
|
|
|
|
}
|
2020-01-28 12:51:21 +01:00
|
|
|
if len(ids) > 1 {
|
2020-01-17 20:32:21 +01:00
|
|
|
templateData.Title = fmt.Sprintf("%d files on pixeldrain", len(finfo))
|
2020-10-27 11:26:41 +01:00
|
|
|
vd.Type = "list"
|
|
|
|
vd.APIResponse = apitype.ListInfo{
|
|
|
|
Success: true,
|
|
|
|
Title: "Multiple files",
|
|
|
|
DateCreated: time.Now(),
|
|
|
|
Files: finfo,
|
2017-11-10 12:39:55 +01:00
|
|
|
}
|
|
|
|
} else {
|
2020-01-17 20:32:21 +01:00
|
|
|
templateData.Title = fmt.Sprintf("%s ~ pixeldrain", finfo[0].Name)
|
2020-10-27 11:26:41 +01:00
|
|
|
vd.Type = "file"
|
|
|
|
vd.APIResponse = finfo[0].FileInfo
|
2017-11-10 12:39:55 +01:00
|
|
|
}
|
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"
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
2018-06-17 16:15:58 +02:00
|
|
|
|
|
|
|
// 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.
|
2019-02-19 23:10:04 +01:00
|
|
|
func (wc *WebController) serveFileViewerDemo(w http.ResponseWriter, r *http.Request) {
|
|
|
|
templateData := wc.newTemplateData(w, r)
|
|
|
|
templateData.Other = viewerData{
|
2020-10-27 11:26:41 +01:00
|
|
|
Type: "file",
|
|
|
|
CaptchaKey: wc.captchaSiteKey,
|
|
|
|
AdType: 0, // Always show a-ads on the demo page
|
|
|
|
FileAdsEnabled: true,
|
|
|
|
UserAdsEnabled: true,
|
2019-02-19 23:10:04 +01:00
|
|
|
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": "",
|
2018-06-17 16:15:58 +02:00
|
|
|
},
|
2019-02-19 23:10:04 +01:00
|
|
|
}
|
|
|
|
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") {
|
2019-02-19 23:10:04 +01:00
|
|
|
log.Error("Error rendering demo file: %s", err)
|
|
|
|
}
|
2018-06-17 16:15:58 +02:00
|
|
|
}
|
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)
|
2020-01-28 12:51:21 +01:00
|
|
|
var list, err = templateData.PixelAPI.GetList(p.ByName("id"))
|
2020-01-21 15:33:09 +01:00
|
|
|
if err != nil {
|
2020-06-07 21:12:48 +02:00
|
|
|
if err, ok := err.(apiclient.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)
|
2020-01-21 15:33:09 +01:00
|
|
|
templateData.Other = viewerData{
|
2020-10-27 11:26:41 +01:00
|
|
|
Type: "list",
|
|
|
|
CaptchaKey: wc.captchaSiteKey,
|
|
|
|
ViewToken: wc.viewTokenOrBust(),
|
|
|
|
AdType: adType(),
|
|
|
|
FileAdsEnabled: list.Files[0].ShowAds,
|
|
|
|
UserAdsEnabled: !(templateData.Authenticated && templateData.User.Subscription.DisableAdDisplay),
|
|
|
|
APIResponse: list,
|
2020-01-21 15:33:09 +01:00
|
|
|
}
|
2020-01-28 12:51:21 +01:00
|
|
|
|
|
|
|
var templateName = "file_viewer"
|
|
|
|
if browserCompat(r.UserAgent()) {
|
|
|
|
templateName = "file_viewer_compat"
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
2020-02-04 19:37:46 +01:00
|
|
|
|
|
|
|
// ServeFileViewer controller for GET /s/:id
|
|
|
|
func (wc *WebController) serveSkynetViewer(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
|
|
var err error
|
|
|
|
templateData := wc.newTemplateData(w, r)
|
|
|
|
|
|
|
|
// Get the first few bytes from the file to probe the content type and
|
|
|
|
// length
|
2020-02-18 14:57:27 +01:00
|
|
|
rq, err := http.NewRequest("GET", "https://skydrain.net/file/"+p.ByName("id"), nil)
|
2020-02-04 19:37:46 +01:00
|
|
|
if err != nil {
|
2020-02-04 22:29:48 +01:00
|
|
|
log.Warn("Failed to make request to sia portal: %s", err)
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
|
|
|
|
return
|
2020-02-04 19:37:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Range header limits the number of bytes which will be read
|
|
|
|
rq.Header.Set("Range", "bytes=0-1023")
|
|
|
|
|
|
|
|
resp, err := wc.httpClient.Do(rq)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode >= 500 {
|
2020-02-11 21:52:14 +01:00
|
|
|
head, _ := ioutil.ReadAll(resp.Body)
|
|
|
|
log.Warn("Sia portal returned error: %s", head)
|
2020-02-04 19:37:46 +01:00
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
|
|
|
|
return
|
|
|
|
} else if resp.StatusCode >= 400 {
|
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "file_not_found", templateData)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
head, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
2020-02-04 22:29:48 +01:00
|
|
|
log.Warn("Failed to read file header from Sia portal: %s", err)
|
2020-02-04 19:37:46 +01:00
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var fileType = resp.Header.Get("Content-Type")
|
|
|
|
if fileType == "application/octet-stream" || fileType == "" {
|
|
|
|
fileType = pdmimetype.Detect(head)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now get the size of the file from the content-range header
|
|
|
|
contentRange := resp.Header.Get("Content-Range")
|
|
|
|
if contentRange == "" {
|
2020-02-04 22:29:48 +01:00
|
|
|
log.Warn("Sia portal didn't send Content-Range")
|
2020-02-04 19:37:46 +01:00
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
wc.templates.Get().ExecuteTemplate(w, "500", templateData)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
contentRange = strings.TrimPrefix(contentRange, "bytes ")
|
2020-06-07 21:12:48 +02:00
|
|
|
size, err := strconv.ParseInt(strings.Split(contentRange, "/")[1], 10, 64)
|
2020-02-04 19:37:46 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2020-02-04 22:29:48 +01:00
|
|
|
var name string
|
|
|
|
_, params, err := mime.ParseMediaType(resp.Header.Get("Content-Disposition"))
|
|
|
|
if err != nil {
|
|
|
|
name = "skynet_file"
|
|
|
|
} else {
|
|
|
|
name = params["filename"]
|
|
|
|
}
|
|
|
|
|
|
|
|
templateData.Title = fmt.Sprintf("name ~ Skynet")
|
2020-02-04 19:37:46 +01:00
|
|
|
templateData.Other = viewerData{
|
2020-05-24 14:23:03 +02:00
|
|
|
Type: "skylink",
|
|
|
|
AdType: adType(),
|
2020-06-07 21:12:48 +02:00
|
|
|
APIResponse: apitype.FileInfo{
|
|
|
|
Success: true,
|
|
|
|
ID: p.ByName("id"),
|
|
|
|
Name: name,
|
|
|
|
Size: size,
|
|
|
|
DateUpload: time.Now(),
|
|
|
|
DateLastView: time.Now(),
|
|
|
|
MimeType: fileType,
|
2020-02-04 19:37:46 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var templateName = "file_viewer"
|
|
|
|
if browserCompat(r.UserAgent()) {
|
|
|
|
templateName = "file_viewer_compat"
|
|
|
|
}
|
|
|
|
|
|
|
|
err = wc.templates.Get().ExecuteTemplate(w, templateName, templateData)
|
|
|
|
if err != nil && !strings.Contains(err.Error(), "broken pipe") {
|
|
|
|
log.Error("Error executing template file_viewer: %s", err)
|
|
|
|
}
|
|
|
|
}
|