-
-
Notifications
You must be signed in to change notification settings - Fork 141
/
Copy pathdirhandler.go
173 lines (145 loc) · 5.11 KB
/
dirhandler.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
package engine
// Directory Index
import (
"bytes"
"net/http"
"path/filepath"
"strings"
"github.com/go-gcfg/gcfg"
"github.com/sirupsen/logrus"
"github.com/xyproto/algernon/platformdep"
"github.com/xyproto/algernon/themes"
"github.com/xyproto/algernon/utils"
)
const (
dotSlash = "." + utils.Pathsep /* ./ */
doubleP = utils.Pathsep + utils.Pathsep /* // */
)
// List of filenames that should be displayed instead of a directory listing
var indexFilenames = []string{"index.lua", "index.html", "index.md", "index.txt", "index.pongo2", "index.tmpl", "index.po2", "index.amber", "index.happ", "index.hyper", "index.hyper.js", "index.hyper.jsx", "index.tl", "index.prompt"}
// DirConfig keeps a directory listing configuration
type DirConfig struct {
Main struct {
Title string
Theme string
}
}
// DirectoryListing serves the given directory as a web page with links the the contents
func (ac *Config) DirectoryListing(w http.ResponseWriter, req *http.Request, rootdir, dirname, theme string) {
var (
buf bytes.Buffer
fullFilename string
URLpath string
title = dirname
)
// Remove a trailing slash after the root directory, if present
rootdir = strings.TrimSuffix(rootdir, "/")
// Read ignore patterns if ignore.txt is present
var ignorePatterns []string
ignoreFilePath := filepath.Join(dirname, platformdep.IgnoreFilename)
if ac.fs.Exists(ignoreFilePath) {
patterns, err := ReadIgnoreFile(ignoreFilePath)
if err != nil {
logrus.Warn("Could not read ignore.txt: ", err)
} else {
ignorePatterns = patterns
}
}
// Fill the coming HTML body with a list of all the filenames in `dirname`
for _, filename := range utils.GetFilenames(dirname) {
if filename == platformdep.DirConfFilename || filename == platformdep.IgnoreFilename {
// Skip
continue
}
// Check if the filename matches any of the ignore patterns
if ShouldIgnore(filename, ignorePatterns) {
continue
}
// Find the full name
fullFilename = dirname
// Add a "/" after the directory name, if missing
if !strings.HasSuffix(fullFilename, utils.Pathsep) {
fullFilename += utils.Pathsep
}
// Add the filename at the end
fullFilename += filename
// Remove the root directory from the link path
URLpath = fullFilename[len(rootdir)+1:]
// Output different entries for files and directories
buf.WriteString(themes.HTMLLink(filename, URLpath, ac.fs.IsDir(fullFilename)))
}
// Read directory configuration, if present
if fullDirConfFilename := filepath.Join(dirname, platformdep.DirConfFilename); ac.fs.Exists(fullDirConfFilename) {
var dirConf DirConfig
if err := gcfg.ReadFileInto(&dirConf, fullDirConfFilename); err == nil { // if no error
if dirConf.Main.Title != "" {
title = dirConf.Main.Title
}
if dirConf.Main.Theme != "" {
theme = dirConf.Main.Theme
}
}
} else {
// Strip the leading "./" from the current directory
title = strings.TrimPrefix(title, dotSlash)
// Replace "//" with just "/"
title = strings.ReplaceAll(title, doubleP, utils.Pathsep)
}
// Check if the current page contents are empty
if buf.Len() == 0 {
buf.WriteString("Empty directory")
}
htmldata := themes.MessagePageBytes(title, buf.Bytes(), theme)
// If the auto-refresh feature has been enabled
if ac.autoRefresh {
// Insert JavaScript for refreshing the page into the generated HTML
htmldata = ac.InsertAutoRefresh(req, htmldata)
}
// Serve the page
w.Header().Add(contentType, htmlUTF8)
ac.DataToClient(w, req, dirname, htmldata)
}
// DirPage serves a directory, using index.* files, if present.
// The directory must exist.
// rootdir is the base directory (can be ".")
// dirname is the specific directory that is to be served (should never be ".")
func (ac *Config) DirPage(w http.ResponseWriter, req *http.Request, rootdir, dirname, theme, luaDataFilename string) {
// Check if we are instructed to quit after serving the first file
if ac.quitAfterFirstRequest {
go ac.quitSoon("Quit after first request", defaultSoonDuration)
}
// If the URL does not end with a slash, redirect to an URL that does
if !strings.HasSuffix(req.URL.Path, "/") {
if req.Method == "POST" {
logrus.Warn("Redirecting a POST request: " + req.URL.Path + " -> " + req.URL.Path + "/.")
logrus.Warn("Header data may be lost! Please add the missing slash.")
}
http.Redirect(w, req, req.URL.Path+"/", http.StatusMovedPermanently)
return
}
// Handle the serving of index files, if needed
var filename string
for _, indexfile := range indexFilenames {
filename = filepath.Join(dirname, indexfile)
if ac.fs.Exists(filename) {
ac.FilePage(w, req, filename, luaDataFilename)
return
}
}
// Serve handler.lua, if found in parent directories
var ancestor string
ancestor = filepath.Dir(dirname)
for x := 0; x < 100; x++ { // a maximum of 100 directories deep
filename = filepath.Join(ancestor, "handler.lua")
if ac.fs.Exists(filename) {
ac.FilePage(w, req, filename, luaDataFilename)
return
}
if ancestor == "." {
break
}
ancestor = filepath.Dir(ancestor)
}
// Serve a directory listing if no index file is found
ac.DirectoryListing(w, req, rootdir, dirname, theme)
}