2017-11-10 12:39:55 +01:00
|
|
|
package webcontroller
|
|
|
|
|
|
|
|
import (
|
2017-12-12 23:33:41 +01:00
|
|
|
"fmt"
|
2022-06-07 14:43:01 +02:00
|
|
|
"html/template"
|
2023-05-31 00:26:51 +02:00
|
|
|
"io"
|
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-11-16 15:20:15 +01:00
|
|
|
"fornaxian.tech/log"
|
2021-03-10 20:13:32 +01:00
|
|
|
"fornaxian.tech/pixeldrain_api_client/pixelapi"
|
2023-05-10 15:08:29 +02:00
|
|
|
"fornaxian.tech/util"
|
2017-11-10 12:39:55 +01:00
|
|
|
"github.com/julienschmidt/httprouter"
|
2023-05-31 00:26:51 +02:00
|
|
|
"github.com/microcosm-cc/bluemonday"
|
|
|
|
blackfriday "github.com/russross/blackfriday/v2"
|
2017-11-10 12:39:55 +01:00
|
|
|
)
|
|
|
|
|
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-11-02 10:23:22 +01:00
|
|
|
type fileViewerData struct {
|
2022-06-07 14:43:01 +02:00
|
|
|
Type string `json:"type"` // file or list
|
|
|
|
APIResponse interface{} `json:"api_response"`
|
|
|
|
CaptchaKey string `json:"captcha_key"`
|
|
|
|
Embedded bool `json:"embedded"`
|
|
|
|
UserAdsEnabled bool `json:"user_ads_enabled"`
|
|
|
|
ThemeURI template.URL `json:"theme_uri"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (vd *fileViewerData) themeOverride(r *http.Request, files []pixelapi.ListFile) {
|
|
|
|
vd.ThemeURI = "/theme.css"
|
|
|
|
var theme = r.URL.Query().Get("style")
|
|
|
|
var hue = r.URL.Query().Get("hue")
|
|
|
|
|
|
|
|
if files[0].Branding != nil {
|
|
|
|
if theme == "" {
|
|
|
|
theme = files[0].Branding["theme"]
|
|
|
|
}
|
|
|
|
if hue == "" {
|
|
|
|
hue = files[0].Branding["hue"]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if theme != "" {
|
|
|
|
vd.ThemeURI += template.URL("?style=" + theme)
|
|
|
|
if hue != "" {
|
|
|
|
vd.ThemeURI += template.URL("&hue=" + hue)
|
|
|
|
}
|
|
|
|
}
|
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) {
|
2021-03-23 11:31:11 +01:00
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
2024-08-27 01:29:09 +02:00
|
|
|
// Prevent search engines from indexing this page for privacy reasons
|
|
|
|
w.Header().Set("X-Robots-Tag", "noindex, nofollow")
|
|
|
|
|
2021-11-02 10:23:22 +01:00
|
|
|
var err error
|
2020-01-28 12:51:21 +01:00
|
|
|
var ids = strings.Split(p.ByName("id"), ",")
|
2021-11-02 10:23:22 +01:00
|
|
|
var templateData = wc.newTemplateData(w, r)
|
2020-01-21 15:33:09 +01:00
|
|
|
|
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)
|
2018-06-20 23:47:47 +02:00
|
|
|
if err != nil {
|
2021-03-10 20:13:32 +01:00
|
|
|
if pixelapi.ErrIsServerError(err) {
|
2024-09-24 00:12:41 +02:00
|
|
|
wc.templates.Run(w, r, "500", templateData)
|
2020-07-20 23:24:49 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2021-03-04 15:42:46 +01:00
|
|
|
if len(files) == 0 {
|
2019-07-28 22:34:15 +02:00
|
|
|
w.WriteHeader(http.StatusNotFound)
|
2024-09-24 00:12:41 +02:00
|
|
|
wc.templates.Run(w, r, "file_not_found", templateData)
|
2017-11-10 12:39:55 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-01 16:56:46 +01:00
|
|
|
if files[0].SkipFileViewer {
|
|
|
|
http.Redirect(w, r, "/api/file/"+p.ByName("id"), http.StatusSeeOther)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-09 14:48:10 +02:00
|
|
|
templateData.OGData = wc.metadataFromFile(r, files[0].FileInfo)
|
2020-10-27 11:26:41 +01:00
|
|
|
|
2021-11-02 10:23:22 +01:00
|
|
|
var vd = fileViewerData{
|
2020-10-27 11:26:41 +01:00
|
|
|
CaptchaKey: wc.captchaKey(),
|
2021-11-22 21:34:14 +01:00
|
|
|
UserAdsEnabled: templateData.User.Subscription.ID == "",
|
2020-10-27 11:26:41 +01:00
|
|
|
}
|
2021-03-10 20:13:32 +01:00
|
|
|
|
2020-01-28 12:51:21 +01:00
|
|
|
if len(ids) > 1 {
|
2021-03-04 15:42:46 +01:00
|
|
|
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
|
|
|
Title: "Multiple files",
|
|
|
|
DateCreated: time.Now(),
|
2021-03-04 15:42:46 +01:00
|
|
|
Files: files,
|
2017-11-10 12:39:55 +01:00
|
|
|
}
|
|
|
|
} else {
|
2021-03-04 15:42:46 +01:00
|
|
|
templateData.Title = fmt.Sprintf("%s ~ pixeldrain", files[0].Name)
|
2020-10-27 11:26:41 +01:00
|
|
|
vd.Type = "file"
|
2021-03-04 15:42:46 +01:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2022-06-07 14:43:01 +02:00
|
|
|
vd.themeOverride(r, files)
|
2020-10-27 11:26:41 +01:00
|
|
|
templateData.Other = vd
|
2020-01-28 12:51:21 +01:00
|
|
|
|
2021-03-04 15:42:46 +01:00
|
|
|
for _, file := range files {
|
|
|
|
if file.AbuseType != "" {
|
|
|
|
w.WriteHeader(http.StatusUnavailableForLegalReasons)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-02 10:23:22 +01:00
|
|
|
var templateName = "file_viewer_svelte"
|
|
|
|
if browserCompat(r.UserAgent()) {
|
|
|
|
templateName = "file_viewer_compat"
|
|
|
|
}
|
|
|
|
|
2024-09-24 00:12:41 +02:00
|
|
|
err = wc.templates.Run(w, r, templateName, templateData)
|
2023-05-10 15:08:29 +02:00
|
|
|
if err != nil && !util.IsNetError(err) {
|
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
|
|
|
|
2021-11-02 10:23:22 +01:00
|
|
|
func (wc *WebController) serveListViewer(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
|
|
// 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/list/"+p.ByName("id")+"/zip", http.StatusSeeOther)
|
|
|
|
return
|
2019-02-19 23:10:04 +01:00
|
|
|
}
|
2020-01-21 15:33:09 +01:00
|
|
|
|
2024-08-27 01:29:09 +02:00
|
|
|
// Prevent search engines from indexing this page for privacy reasons
|
|
|
|
w.Header().Set("X-Robots-Tag", "noindex, nofollow")
|
|
|
|
|
2020-01-21 15:33:09 +01:00
|
|
|
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 {
|
2024-04-24 18:48:00 +02:00
|
|
|
if apiErr, ok := err.(pixelapi.Error); ok && apiErr.Status == http.StatusNotFound {
|
2020-05-05 22:03:34 +02:00
|
|
|
w.WriteHeader(http.StatusNotFound)
|
2024-09-24 00:12:41 +02:00
|
|
|
wc.templates.Run(w, r, "list_not_found", templateData)
|
2024-11-14 17:56:01 +01:00
|
|
|
} else if strings.HasSuffix(err.Error(), "invalid control character in URL") {
|
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
wc.templates.Run(w, r, "list_not_found", templateData)
|
2020-05-05 22:03:34 +02:00
|
|
|
} else {
|
|
|
|
log.Error("API request error occurred: %s", err)
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
2024-09-24 00:12:41 +02:00
|
|
|
wc.templates.Run(w, r, "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)
|
2024-09-24 00:12:41 +02:00
|
|
|
wc.templates.Run(w, r, "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)
|
2024-09-09 14:48:10 +02:00
|
|
|
templateData.OGData = wc.metadataFromList(r, list)
|
2021-11-02 10:23:22 +01:00
|
|
|
var vd = fileViewerData{
|
2020-10-27 11:26:41 +01:00
|
|
|
Type: "list",
|
|
|
|
CaptchaKey: wc.captchaSiteKey,
|
2021-11-22 21:34:14 +01:00
|
|
|
UserAdsEnabled: templateData.User.Subscription.ID == "",
|
2020-10-27 11:26:41 +01:00
|
|
|
APIResponse: list,
|
2020-01-21 15:33:09 +01:00
|
|
|
}
|
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
|
|
|
|
}
|
|
|
|
|
2022-06-07 14:43:01 +02:00
|
|
|
vd.themeOverride(r, list.Files)
|
|
|
|
templateData.Other = vd
|
2022-02-07 12:00:22 +01:00
|
|
|
|
2021-03-04 15:42:46 +01:00
|
|
|
for _, file := range list.Files {
|
|
|
|
if file.AbuseType != "" {
|
|
|
|
w.WriteHeader(http.StatusUnavailableForLegalReasons)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-02 10:23:22 +01:00
|
|
|
var templateName = "file_viewer_svelte"
|
|
|
|
if browserCompat(r.UserAgent()) {
|
|
|
|
templateName = "file_viewer_compat"
|
|
|
|
}
|
|
|
|
|
2024-09-24 00:12:41 +02:00
|
|
|
err = wc.templates.Run(w, r, templateName, templateData)
|
2023-05-10 15:08:29 +02:00
|
|
|
if err != nil && !util.IsNetError(err) {
|
2020-01-21 15:33:09 +01:00
|
|
|
log.Error("Error executing template file_viewer: %s", err)
|
|
|
|
}
|
|
|
|
}
|
2023-05-31 00:26:51 +02:00
|
|
|
|
|
|
|
func (wc *WebController) serveFilePreview(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
|
|
apiKey, _ := wc.getAPIKey(r)
|
|
|
|
api := wc.api.Login(apiKey).RealIP(util.RemoteAddress(r)).RealAgent(r.UserAgent())
|
|
|
|
|
|
|
|
file, err := api.GetFileInfo(p.ByName("id")) // TODO: Error handling
|
|
|
|
if err != nil {
|
|
|
|
wc.serveNotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.HasPrefix(file.MimeType, "text") &&
|
|
|
|
(strings.HasSuffix(file.Name, ".md") || strings.HasSuffix(file.Name, ".markdown")) {
|
|
|
|
if file.Size > 1<<22 { // Prevent out of memory errors
|
|
|
|
w.Write([]byte("File is too large to view online.\nPlease download and view it locally."))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
body, err := api.GetFile(file.ID)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Can't download text file for preview: %s", err)
|
|
|
|
w.Write([]byte("An error occurred while downloading this file."))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer body.Close()
|
|
|
|
|
|
|
|
bodyBytes, err := io.ReadAll(body)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Can't read text file for preview: %s", err)
|
|
|
|
w.Write([]byte("An error occurred while reading this file."))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Write(bluemonday.UGCPolicy().SanitizeBytes(blackfriday.Run(bodyBytes)))
|
|
|
|
}
|
|
|
|
}
|