2018-10-28 22:08:20 +08:00
|
|
|
package serve
|
|
|
|
|
|
|
|
import (
|
2019-09-23 04:31:11 +08:00
|
|
|
"bytes"
|
2021-12-09 00:14:45 +08:00
|
|
|
"context"
|
2018-10-28 22:08:20 +08:00
|
|
|
"fmt"
|
|
|
|
"html/template"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"path"
|
2020-05-01 02:24:11 +08:00
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"time"
|
2018-10-28 22:08:20 +08:00
|
|
|
|
2019-07-29 01:47:38 +08:00
|
|
|
"github.com/rclone/rclone/fs"
|
|
|
|
"github.com/rclone/rclone/fs/accounting"
|
|
|
|
"github.com/rclone/rclone/lib/rest"
|
2018-10-28 22:08:20 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// DirEntry is a directory entry
|
|
|
|
type DirEntry struct {
|
2020-05-01 02:24:11 +08:00
|
|
|
remote string
|
|
|
|
URL string
|
|
|
|
Leaf string
|
|
|
|
IsDir bool
|
|
|
|
Size int64
|
|
|
|
ModTime time.Time
|
2018-10-28 22:08:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Directory represents a directory
|
|
|
|
type Directory struct {
|
2018-12-23 08:16:50 +08:00
|
|
|
DirRemote string
|
|
|
|
Title string
|
2020-05-01 02:24:11 +08:00
|
|
|
Name string
|
2018-12-23 08:16:50 +08:00
|
|
|
Entries []DirEntry
|
|
|
|
Query string
|
|
|
|
HTMLTemplate *template.Template
|
2020-05-01 02:24:11 +08:00
|
|
|
Breadcrumb []Crumb
|
|
|
|
Sort string
|
|
|
|
Order string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Crumb is a breadcrumb entry
|
|
|
|
type Crumb struct {
|
|
|
|
Link string
|
|
|
|
Text string
|
2018-10-28 22:08:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewDirectory makes an empty Directory
|
2018-12-23 08:16:50 +08:00
|
|
|
func NewDirectory(dirRemote string, htmlTemplate *template.Template) *Directory {
|
2020-05-01 02:24:11 +08:00
|
|
|
var breadcrumb []Crumb
|
|
|
|
|
|
|
|
// skip trailing slash
|
|
|
|
lpath := "/" + dirRemote
|
|
|
|
if lpath[len(lpath)-1] == '/' {
|
|
|
|
lpath = lpath[:len(lpath)-1]
|
|
|
|
}
|
|
|
|
|
|
|
|
parts := strings.Split(lpath, "/")
|
|
|
|
for i := range parts {
|
|
|
|
txt := parts[i]
|
|
|
|
if i == 0 && parts[i] == "" {
|
|
|
|
txt = "/"
|
|
|
|
}
|
|
|
|
lnk := strings.Repeat("../", len(parts)-i-1)
|
|
|
|
breadcrumb = append(breadcrumb, Crumb{Link: lnk, Text: txt})
|
|
|
|
}
|
|
|
|
|
2018-10-28 22:08:20 +08:00
|
|
|
d := &Directory{
|
2018-12-23 08:16:50 +08:00
|
|
|
DirRemote: dirRemote,
|
|
|
|
Title: fmt.Sprintf("Directory listing of /%s", dirRemote),
|
2020-05-01 02:24:11 +08:00
|
|
|
Name: fmt.Sprintf("/%s", dirRemote),
|
2018-12-23 08:16:50 +08:00
|
|
|
HTMLTemplate: htmlTemplate,
|
2020-05-01 02:24:11 +08:00
|
|
|
Breadcrumb: breadcrumb,
|
2018-10-28 22:08:20 +08:00
|
|
|
}
|
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetQuery sets the query parameters for each URL
|
|
|
|
func (d *Directory) SetQuery(queryParams url.Values) *Directory {
|
|
|
|
d.Query = ""
|
|
|
|
if len(queryParams) > 0 {
|
|
|
|
d.Query = "?" + queryParams.Encode()
|
|
|
|
}
|
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
2020-05-01 02:24:11 +08:00
|
|
|
// AddHTMLEntry adds an entry to that directory
|
|
|
|
func (d *Directory) AddHTMLEntry(remote string, isDir bool, size int64, modTime time.Time) {
|
|
|
|
leaf := path.Base(remote)
|
|
|
|
if leaf == "." {
|
|
|
|
leaf = ""
|
|
|
|
}
|
|
|
|
urlRemote := leaf
|
|
|
|
if isDir {
|
|
|
|
leaf += "/"
|
|
|
|
urlRemote += "/"
|
|
|
|
}
|
|
|
|
d.Entries = append(d.Entries, DirEntry{
|
|
|
|
remote: remote,
|
|
|
|
URL: rest.URLPathEscape(urlRemote) + d.Query,
|
|
|
|
Leaf: leaf,
|
|
|
|
IsDir: isDir,
|
|
|
|
Size: size,
|
|
|
|
ModTime: modTime,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-10-28 22:08:20 +08:00
|
|
|
// AddEntry adds an entry to that directory
|
|
|
|
func (d *Directory) AddEntry(remote string, isDir bool) {
|
|
|
|
leaf := path.Base(remote)
|
|
|
|
if leaf == "." {
|
|
|
|
leaf = ""
|
|
|
|
}
|
|
|
|
urlRemote := leaf
|
|
|
|
if isDir {
|
|
|
|
leaf += "/"
|
|
|
|
urlRemote += "/"
|
|
|
|
}
|
|
|
|
d.Entries = append(d.Entries, DirEntry{
|
|
|
|
remote: remote,
|
|
|
|
URL: rest.URLPathEscape(urlRemote) + d.Query,
|
|
|
|
Leaf: leaf,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-05-20 18:39:20 +08:00
|
|
|
// Error logs the error and if a ResponseWriter is given it writes an http.StatusInternalServerError
|
2021-12-09 00:14:45 +08:00
|
|
|
func Error(ctx context.Context, what interface{}, w http.ResponseWriter, text string, err error) {
|
|
|
|
err = fs.CountError(ctx, err)
|
2018-10-28 22:08:20 +08:00
|
|
|
fs.Errorf(what, "%s: %v", text, err)
|
2019-09-23 04:31:11 +08:00
|
|
|
if w != nil {
|
|
|
|
http.Error(w, text+".", http.StatusInternalServerError)
|
|
|
|
}
|
2018-10-28 22:08:20 +08:00
|
|
|
}
|
|
|
|
|
Spelling fixes
Fix spelling of: above, already, anonymous, associated,
authentication, bandwidth, because, between, blocks, calculate,
candidates, cautious, changelog, cleaner, clipboard, command,
completely, concurrently, considered, constructs, corrupt, current,
daemon, dependencies, deprecated, directory, dispatcher, download,
eligible, ellipsis, encrypter, endpoint, entrieslist, essentially,
existing writers, existing, expires, filesystem, flushing, frequently,
hierarchy, however, implementation, implements, inaccurate,
individually, insensitive, longer, maximum, metadata, modified,
multipart, namedirfirst, nextcloud, obscured, opened, optional,
owncloud, pacific, passphrase, password, permanently, persimmon,
positive, potato, protocol, quota, receiving, recommends, referring,
requires, revisited, satisfied, satisfies, satisfy, semver,
serialized, session, storage, strategies, stringlist, successful,
supported, surprise, temporarily, temporary, transactions, unneeded,
update, uploads, wrapped
Signed-off-by: Josh Soref <jsoref@users.noreply.github.com>
2020-10-09 08:17:24 +08:00
|
|
|
// ProcessQueryParams takes and sorts/orders based on the request sort/order parameters and default is namedirfirst/asc
|
2020-05-01 02:24:11 +08:00
|
|
|
func (d *Directory) ProcessQueryParams(sortParm string, orderParm string) *Directory {
|
|
|
|
d.Sort = sortParm
|
|
|
|
d.Order = orderParm
|
|
|
|
|
|
|
|
var toSort sort.Interface
|
|
|
|
|
|
|
|
switch d.Sort {
|
|
|
|
case sortByName:
|
|
|
|
toSort = byName(*d)
|
|
|
|
case sortByNameDirFirst:
|
|
|
|
toSort = byNameDirFirst(*d)
|
|
|
|
case sortBySize:
|
|
|
|
toSort = bySize(*d)
|
|
|
|
case sortByTime:
|
|
|
|
toSort = byTime(*d)
|
|
|
|
default:
|
|
|
|
toSort = byNameDirFirst(*d)
|
|
|
|
}
|
|
|
|
if d.Order == "desc" && toSort != nil {
|
|
|
|
toSort = sort.Reverse(toSort)
|
|
|
|
}
|
|
|
|
if toSort != nil {
|
|
|
|
sort.Sort(toSort)
|
|
|
|
}
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
type byName Directory
|
|
|
|
type byNameDirFirst Directory
|
|
|
|
type bySize Directory
|
|
|
|
type byTime Directory
|
|
|
|
|
|
|
|
func (d byName) Len() int { return len(d.Entries) }
|
|
|
|
func (d byName) Swap(i, j int) { d.Entries[i], d.Entries[j] = d.Entries[j], d.Entries[i] }
|
|
|
|
|
|
|
|
func (d byName) Less(i, j int) bool {
|
|
|
|
return strings.ToLower(d.Entries[i].Leaf) < strings.ToLower(d.Entries[j].Leaf)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d byNameDirFirst) Len() int { return len(d.Entries) }
|
|
|
|
func (d byNameDirFirst) Swap(i, j int) { d.Entries[i], d.Entries[j] = d.Entries[j], d.Entries[i] }
|
|
|
|
|
|
|
|
func (d byNameDirFirst) Less(i, j int) bool {
|
|
|
|
// sort by name if both are dir or file
|
|
|
|
if d.Entries[i].IsDir == d.Entries[j].IsDir {
|
|
|
|
return strings.ToLower(d.Entries[i].Leaf) < strings.ToLower(d.Entries[j].Leaf)
|
|
|
|
}
|
|
|
|
// sort dir ahead of file
|
|
|
|
return d.Entries[i].IsDir
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d bySize) Len() int { return len(d.Entries) }
|
|
|
|
func (d bySize) Swap(i, j int) { d.Entries[i], d.Entries[j] = d.Entries[j], d.Entries[i] }
|
|
|
|
|
|
|
|
func (d bySize) Less(i, j int) bool {
|
|
|
|
const directoryOffset = -1 << 31 // = -math.MinInt32
|
|
|
|
|
|
|
|
iSize, jSize := d.Entries[i].Size, d.Entries[j].Size
|
|
|
|
|
|
|
|
// directory sizes depend on the file system; to
|
|
|
|
// provide a consistent experience, put them up front
|
|
|
|
// and sort them by name
|
|
|
|
if d.Entries[i].IsDir {
|
|
|
|
iSize = directoryOffset
|
|
|
|
}
|
|
|
|
if d.Entries[j].IsDir {
|
|
|
|
jSize = directoryOffset
|
|
|
|
}
|
|
|
|
if d.Entries[i].IsDir && d.Entries[j].IsDir {
|
|
|
|
return strings.ToLower(d.Entries[i].Leaf) < strings.ToLower(d.Entries[j].Leaf)
|
|
|
|
}
|
|
|
|
|
|
|
|
return iSize < jSize
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d byTime) Len() int { return len(d.Entries) }
|
|
|
|
func (d byTime) Swap(i, j int) { d.Entries[i], d.Entries[j] = d.Entries[j], d.Entries[i] }
|
|
|
|
func (d byTime) Less(i, j int) bool { return d.Entries[i].ModTime.Before(d.Entries[j].ModTime) }
|
|
|
|
|
|
|
|
const (
|
|
|
|
sortByName = "name"
|
|
|
|
sortByNameDirFirst = "namedirfirst"
|
|
|
|
sortBySize = "size"
|
|
|
|
sortByTime = "time"
|
|
|
|
)
|
|
|
|
|
2018-10-28 22:08:20 +08:00
|
|
|
// Serve serves a directory
|
|
|
|
func (d *Directory) Serve(w http.ResponseWriter, r *http.Request) {
|
2021-12-09 00:14:45 +08:00
|
|
|
ctx := r.Context()
|
2018-10-28 22:08:20 +08:00
|
|
|
// Account the transfer
|
2024-01-19 00:44:13 +08:00
|
|
|
tr := accounting.Stats(r.Context()).NewTransferRemoteSize(d.DirRemote, -1, nil, nil)
|
2020-11-06 00:59:59 +08:00
|
|
|
defer tr.Done(r.Context(), nil)
|
2018-10-28 22:08:20 +08:00
|
|
|
|
|
|
|
fs.Infof(d.DirRemote, "%s: Serving directory", r.RemoteAddr)
|
2018-12-23 08:16:50 +08:00
|
|
|
|
2019-09-23 04:31:11 +08:00
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
err := d.HTMLTemplate.Execute(buf, d)
|
2018-10-28 22:08:20 +08:00
|
|
|
if err != nil {
|
2021-12-09 00:14:45 +08:00
|
|
|
Error(ctx, d.DirRemote, w, "Failed to render template", err)
|
2018-10-28 22:08:20 +08:00
|
|
|
return
|
|
|
|
}
|
2024-05-14 00:24:54 +08:00
|
|
|
w.Header().Set("Content-Length", fmt.Sprintf("%d", buf.Len()))
|
2019-09-23 04:31:11 +08:00
|
|
|
_, err = buf.WriteTo(w)
|
|
|
|
if err != nil {
|
2021-12-09 00:14:45 +08:00
|
|
|
Error(ctx, d.DirRemote, nil, "Failed to drain template buffer", err)
|
2019-09-23 04:31:11 +08:00
|
|
|
}
|
2018-10-28 22:08:20 +08:00
|
|
|
}
|