2015-09-25 01:58:50 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2019-01-31 07:52:43 +01:00
|
|
|
"fmt"
|
2019-01-25 08:33:11 +01:00
|
|
|
"io"
|
2015-09-25 01:58:50 +02:00
|
|
|
"net/http"
|
2015-10-14 04:52:55 +02:00
|
|
|
"net/url"
|
2019-01-25 08:33:11 +01:00
|
|
|
"strconv"
|
2015-09-30 01:28:10 +02:00
|
|
|
"strings"
|
2019-01-31 07:52:43 +01:00
|
|
|
"time"
|
2015-09-25 01:58:50 +02:00
|
|
|
|
2017-05-02 06:25:56 +02:00
|
|
|
"github.com/andreimarcu/linx-server/backends"
|
2019-01-09 05:28:01 +01:00
|
|
|
"github.com/andreimarcu/linx-server/expiry"
|
2019-01-31 07:52:43 +01:00
|
|
|
"github.com/andreimarcu/linx-server/httputil"
|
2015-09-25 01:58:50 +02:00
|
|
|
"github.com/zenazn/goji/web"
|
|
|
|
)
|
|
|
|
|
|
|
|
func fileServeHandler(c web.C, w http.ResponseWriter, r *http.Request) {
|
2015-09-25 04:20:44 +02:00
|
|
|
fileName := c.URLParams["name"]
|
2015-09-25 01:58:50 +02:00
|
|
|
|
2019-01-09 05:28:01 +01:00
|
|
|
metadata, err := checkFile(fileName)
|
2019-01-25 08:33:11 +01:00
|
|
|
if err == backends.NotFoundErr {
|
2015-09-25 18:00:14 +02:00
|
|
|
notFoundHandler(c, w, r)
|
2015-09-25 01:58:50 +02:00
|
|
|
return
|
2019-01-09 05:28:01 +01:00
|
|
|
} else if err != nil {
|
2019-01-25 08:33:11 +01:00
|
|
|
oopsHandler(c, w, r, RespAUTO, "Corrupt metadata.")
|
2019-01-09 05:28:01 +01:00
|
|
|
return
|
2015-09-25 01:58:50 +02:00
|
|
|
}
|
|
|
|
|
2015-09-30 01:28:10 +02:00
|
|
|
if !Config.allowHotlink {
|
|
|
|
referer := r.Header.Get("Referer")
|
2015-10-14 04:52:55 +02:00
|
|
|
u, _ := url.Parse(referer)
|
2016-06-04 10:22:01 +02:00
|
|
|
p, _ := url.Parse(getSiteURL(r))
|
2015-10-14 04:52:55 +02:00
|
|
|
if referer != "" && !sameOrigin(u, p) {
|
2015-11-12 06:56:22 +01:00
|
|
|
http.Redirect(w, r, Config.sitePath+fileName, 303)
|
2015-09-30 01:28:10 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-04 23:58:00 +02:00
|
|
|
w.Header().Set("Content-Security-Policy", Config.fileContentSecurityPolicy)
|
2019-01-08 20:56:09 +01:00
|
|
|
w.Header().Set("Referrer-Policy", Config.fileReferrerPolicy)
|
2015-10-04 23:58:00 +02:00
|
|
|
|
2019-01-25 08:33:11 +01:00
|
|
|
w.Header().Set("Content-Type", metadata.Mimetype)
|
|
|
|
w.Header().Set("Content-Length", strconv.FormatInt(metadata.Size, 10))
|
2019-01-31 07:52:43 +01:00
|
|
|
w.Header().Set("Etag", fmt.Sprintf("\"%s\"", metadata.Sha256sum))
|
|
|
|
w.Header().Set("Cache-Control", "public, no-cache")
|
|
|
|
|
|
|
|
modtime := time.Unix(0, 0)
|
|
|
|
if done := httputil.CheckPreconditions(w, r, modtime); done == true {
|
|
|
|
return
|
|
|
|
}
|
2019-01-09 05:28:01 +01:00
|
|
|
|
2019-01-25 08:33:11 +01:00
|
|
|
if r.Method != "HEAD" {
|
2019-01-31 07:52:43 +01:00
|
|
|
_, reader, err := storageBackend.Get(fileName)
|
|
|
|
if err != nil {
|
|
|
|
oopsHandler(c, w, r, RespAUTO, "Unable to open file.")
|
|
|
|
return
|
|
|
|
}
|
2019-01-25 08:33:11 +01:00
|
|
|
defer reader.Close()
|
|
|
|
|
|
|
|
if _, err = io.CopyN(w, reader, metadata.Size); err != nil {
|
|
|
|
oopsHandler(c, w, r, RespAUTO, err.Error())
|
|
|
|
}
|
|
|
|
}
|
2015-09-28 06:25:57 +02:00
|
|
|
}
|
2015-09-28 04:17:12 +02:00
|
|
|
|
2015-09-30 21:54:30 +02:00
|
|
|
func staticHandler(c web.C, w http.ResponseWriter, r *http.Request) {
|
|
|
|
path := r.URL.Path
|
|
|
|
if path[len(path)-1:] == "/" {
|
|
|
|
notFoundHandler(c, w, r)
|
|
|
|
return
|
|
|
|
} else {
|
2015-10-04 18:58:30 +02:00
|
|
|
if path == "/favicon.ico" {
|
2015-10-30 23:36:47 +01:00
|
|
|
path = Config.sitePath + "/static/images/favicon.gif"
|
2015-10-04 18:58:30 +02:00
|
|
|
}
|
|
|
|
|
2015-10-30 23:36:47 +01:00
|
|
|
filePath := strings.TrimPrefix(path, Config.sitePath+"static/")
|
2015-10-14 20:58:27 +02:00
|
|
|
file, err := staticBox.Open(filePath)
|
|
|
|
if err != nil {
|
|
|
|
notFoundHandler(c, w, r)
|
|
|
|
return
|
2015-09-30 21:54:30 +02:00
|
|
|
}
|
|
|
|
|
2019-01-31 07:52:43 +01:00
|
|
|
w.Header().Set("Etag", fmt.Sprintf("\"%s\"", timeStartedStr))
|
|
|
|
w.Header().Set("Cache-Control", "public, max-age=86400")
|
2015-10-14 20:58:27 +02:00
|
|
|
http.ServeContent(w, r, filePath, timeStarted, file)
|
2015-09-30 21:54:30 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-09 05:28:01 +01:00
|
|
|
func checkFile(filename string) (metadata backends.Metadata, err error) {
|
2019-01-25 08:33:11 +01:00
|
|
|
metadata, err = storageBackend.Head(filename)
|
2015-10-08 04:45:34 +02:00
|
|
|
if err != nil {
|
2019-01-09 05:28:01 +01:00
|
|
|
return
|
2015-09-28 04:17:12 +02:00
|
|
|
}
|
2015-09-25 01:58:50 +02:00
|
|
|
|
2019-01-09 05:28:01 +01:00
|
|
|
if expiry.IsTsExpired(metadata.Expiry) {
|
2019-01-25 08:33:11 +01:00
|
|
|
storageBackend.Delete(filename)
|
|
|
|
err = backends.NotFoundErr
|
2019-01-09 05:28:01 +01:00
|
|
|
return
|
2015-09-28 06:25:57 +02:00
|
|
|
}
|
|
|
|
|
2019-01-09 05:28:01 +01:00
|
|
|
return
|
2015-09-25 01:58:50 +02:00
|
|
|
}
|