support custom error pages
This commit is contained in:
parent
0be3507b63
commit
9ed8108ed7
44
main.go
44
main.go
@ -1,8 +1,13 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@ -21,6 +26,7 @@ type PageProvider interface {
|
|||||||
type RootHandler struct {
|
type RootHandler struct {
|
||||||
Sessions SessionProvider
|
Sessions SessionProvider
|
||||||
Pages PageProvider
|
Pages PageProvider
|
||||||
|
StaticDir string
|
||||||
}
|
}
|
||||||
|
|
||||||
type AdminHandler struct {
|
type AdminHandler struct {
|
||||||
@ -47,25 +53,57 @@ func (h *RootHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|||||||
// fall back to serving out of the static directory, but:
|
// fall back to serving out of the static directory, but:
|
||||||
// 1. prevent the generated indexes from rendering
|
// 1. prevent the generated indexes from rendering
|
||||||
if strings.HasSuffix(r.URL.Path, "/") {
|
if strings.HasSuffix(r.URL.Path, "/") {
|
||||||
http.NotFound(w, r)
|
h.ErrorHandle(404, w)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// 2. prevent hidden paths from rendering
|
// 2. prevent hidden paths from rendering
|
||||||
for _, seg := range strings.Split(r.URL.Path, "/") {
|
for _, seg := range strings.Split(r.URL.Path, "/") {
|
||||||
if strings.HasPrefix(seg, ".") {
|
if strings.HasPrefix(seg, ".") {
|
||||||
http.NotFound(w, r)
|
h.ErrorHandle(404, w)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// 3. catch files that would 404 and serve our own 404 page
|
||||||
|
if !staticFileExists(filepath.Join(h.StaticDir, r.URL.Path)) {
|
||||||
|
h.ErrorHandle(404, w)
|
||||||
|
return
|
||||||
|
}
|
||||||
// finally, use the built-in fileserver to serve
|
// finally, use the built-in fileserver to serve
|
||||||
fs := http.FileServer(http.Dir("./static"))
|
fs := http.FileServer(http.Dir(h.StaticDir))
|
||||||
fs.ServeHTTP(w, r)
|
fs.ServeHTTP(w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *RootHandler) ErrorHandle(status int, w http.ResponseWriter) {
|
||||||
|
f, err := os.Open(filepath.Join(h.StaticDir, strconv.Itoa(status)+".html"))
|
||||||
|
if err == nil {
|
||||||
|
w.WriteHeader(status)
|
||||||
|
_, err = io.Copy(w, f)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(w, "Internal Server Error while loading %d page\n", status)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "text/plain; charset=utf-8")
|
||||||
|
w.Header().Set("X-Content-Type-Options", "nosniff")
|
||||||
|
w.WriteHeader(404)
|
||||||
|
fmt.Fprintf(w, "%d\n", status)
|
||||||
|
}
|
||||||
|
|
||||||
|
func staticFileExists(name string) bool {
|
||||||
|
f, err := os.Open(name)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
_, err = f.Stat()
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
handler := &RootHandler{
|
handler := &RootHandler{
|
||||||
Sessions: &Sessions{},
|
Sessions: &Sessions{},
|
||||||
Pages: &Index{},
|
Pages: &Index{},
|
||||||
|
StaticDir: "./static",
|
||||||
}
|
}
|
||||||
handler.Pages.Save("foo", &Page{
|
handler.Pages.Save("foo", &Page{
|
||||||
Contents: []byte("foobar"),
|
Contents: []byte("foobar"),
|
||||||
|
Loading…
Reference in New Issue
Block a user