feat: add main command feat: add todos feat: add signup api feat: do not repeat code fix: user return feat: work out static box fix: setup static handlers feat: add share types License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com> feat: start static License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com> feat: bring back more features License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com> add feat: readd more files License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com> feat: add dockerignore License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com> feat: gitignore License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com> feat: readd submodule License: MIT Signed-off-by: Henrique Dias <hacdias@gmail.com>
146 lines
2.8 KiB
Go
146 lines
2.8 KiB
Go
package http
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"encoding/base64"
|
|
"net/http"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/filebrowser/filebrowser/types"
|
|
)
|
|
|
|
const apiSharePrefix = "/api/share"
|
|
|
|
func (e *Env) getShareData(w http.ResponseWriter, r *http.Request, prefix string) (string, bool) {
|
|
relPath, user, ok := e.getResourceData(w, r, apiSharePrefix)
|
|
if !ok {
|
|
return "", false
|
|
}
|
|
|
|
if !user.Perm.Share {
|
|
httpErr(w, http.StatusForbidden, nil)
|
|
return "", false
|
|
}
|
|
|
|
return filepath.Join(user.Scope, relPath), ok
|
|
}
|
|
|
|
func (e *Env) shareGetHandler(w http.ResponseWriter, r *http.Request) {
|
|
path, ok := e.getShareData(w, r, apiSharePrefix)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
s, err := e.Store.Share.GetByPath(path)
|
|
if err == types.ErrNotExist {
|
|
renderJSON(w, []*types.ShareLink{})
|
|
return
|
|
}
|
|
|
|
if err != nil {
|
|
httpErr(w, http.StatusInternalServerError, err)
|
|
return
|
|
}
|
|
|
|
for i, link := range s {
|
|
if link.Expires && link.ExpireDate.Before(time.Now()) {
|
|
e.Store.Share.Delete(link.Hash)
|
|
s = append(s[:i], s[i+1:]...)
|
|
}
|
|
}
|
|
|
|
renderJSON(w, s)
|
|
}
|
|
|
|
func (e *Env) shareDeleteHandler(w http.ResponseWriter, r *http.Request) {
|
|
user, ok := e.getUser(w, r)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
if !user.Perm.Share {
|
|
httpErr(w, http.StatusForbidden, nil)
|
|
return
|
|
}
|
|
|
|
hash := strings.TrimPrefix(r.URL.Path, apiSharePrefix)
|
|
hash = strings.TrimSuffix(hash, "/")
|
|
hash = strings.TrimPrefix(hash, "/")
|
|
if hash == "" {
|
|
return
|
|
}
|
|
|
|
err := e.Store.Share.Delete(hash)
|
|
if err != nil {
|
|
httpErr(w, http.StatusInternalServerError, err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (e *Env) sharePostHandler(w http.ResponseWriter, r *http.Request) {
|
|
path, ok := e.getShareData(w, r, apiSharePrefix)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
var s *types.ShareLink
|
|
expire := r.URL.Query().Get("expires")
|
|
unit := r.URL.Query().Get("unit")
|
|
|
|
if expire == "" {
|
|
var err error
|
|
s, err = e.Store.Share.GetPermanent(path)
|
|
if err == nil {
|
|
w.Write([]byte(e.Settings.BaseURL + "/share/" + s.Hash))
|
|
return
|
|
}
|
|
}
|
|
|
|
bytes := make([]byte, 6)
|
|
_, err := rand.Read(bytes)
|
|
if err != nil {
|
|
httpErr(w, http.StatusInternalServerError, err)
|
|
return
|
|
}
|
|
|
|
str := base64.URLEncoding.EncodeToString(bytes)
|
|
|
|
s = &types.ShareLink{
|
|
Path: path,
|
|
Hash: str,
|
|
Expires: expire != "",
|
|
}
|
|
|
|
if expire != "" {
|
|
num, err := strconv.Atoi(expire)
|
|
if err != nil {
|
|
httpErr(w, http.StatusInternalServerError, err)
|
|
return
|
|
}
|
|
|
|
var add time.Duration
|
|
switch unit {
|
|
case "seconds":
|
|
add = time.Second * time.Duration(num)
|
|
case "minutes":
|
|
add = time.Minute * time.Duration(num)
|
|
case "days":
|
|
add = time.Hour * 24 * time.Duration(num)
|
|
default:
|
|
add = time.Hour * time.Duration(num)
|
|
}
|
|
|
|
s.ExpireDate = time.Now().Add(add)
|
|
}
|
|
|
|
if err := e.Store.Share.Save(s); err != nil {
|
|
httpErr(w, http.StatusInternalServerError, err)
|
|
return
|
|
}
|
|
|
|
renderJSON(w, s)
|
|
}
|