2016-04-10 20:56:10 -07:00
|
|
|
package middleware
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/labstack/echo"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
// StaticConfig defines the config for static middleware.
|
|
|
|
StaticConfig struct {
|
|
|
|
// Root is the directory from where the static content is served.
|
|
|
|
// Required.
|
|
|
|
Root string `json:"root"`
|
|
|
|
|
|
|
|
// Index is the list of index files to be searched and used when serving
|
|
|
|
// a directory.
|
2016-04-20 07:32:51 -07:00
|
|
|
// Optional, with default value as []string{"index.html"}.
|
2016-04-10 20:56:10 -07:00
|
|
|
Index []string `json:"index"`
|
|
|
|
|
|
|
|
// Browse is a flag to enable/disable directory browsing.
|
2016-04-20 07:32:51 -07:00
|
|
|
// Optional, with default value as false.
|
2016-04-10 20:56:10 -07:00
|
|
|
Browse bool `json:"browse"`
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// DefaultStaticConfig is the default static middleware config.
|
|
|
|
DefaultStaticConfig = StaticConfig{
|
|
|
|
Index: []string{"index.html"},
|
|
|
|
Browse: false,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// Static returns a static middleware to serves static content from the provided
|
|
|
|
// root directory.
|
|
|
|
func Static(root string) echo.MiddlewareFunc {
|
|
|
|
c := DefaultStaticConfig
|
|
|
|
c.Root = root
|
|
|
|
return StaticWithConfig(c)
|
|
|
|
}
|
|
|
|
|
|
|
|
// StaticWithConfig returns a static middleware from config.
|
|
|
|
// See `Static()`.
|
|
|
|
func StaticWithConfig(config StaticConfig) echo.MiddlewareFunc {
|
|
|
|
// Defaults
|
|
|
|
if config.Index == nil {
|
|
|
|
config.Index = DefaultStaticConfig.Index
|
|
|
|
}
|
|
|
|
|
|
|
|
return func(next echo.HandlerFunc) echo.HandlerFunc {
|
|
|
|
return func(c echo.Context) error {
|
|
|
|
fs := http.Dir(config.Root)
|
|
|
|
p := c.Request().URL().Path()
|
|
|
|
if strings.Contains(c.Path(), "*") { // If serving from a group, e.g. `/static*`.
|
|
|
|
p = c.P(0)
|
|
|
|
}
|
|
|
|
file := path.Clean(p)
|
|
|
|
f, err := fs.Open(file)
|
|
|
|
if err != nil {
|
|
|
|
return next(c)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
fi, err := f.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if fi.IsDir() {
|
|
|
|
/* NOTE:
|
|
|
|
Not checking the Last-Modified header as it caches the response `304` when
|
|
|
|
changing different directories for the same path.
|
|
|
|
*/
|
|
|
|
d := f
|
|
|
|
|
|
|
|
// Index file
|
|
|
|
// TODO: search all files
|
|
|
|
file = path.Join(file, config.Index[0])
|
|
|
|
f, err = fs.Open(file)
|
2016-05-01 06:56:52 -07:00
|
|
|
if err == nil {
|
|
|
|
// Index file
|
|
|
|
if fi, err = f.Stat(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else if err != nil && config.Browse {
|
2016-04-10 20:56:10 -07:00
|
|
|
dirs, err := d.Readdir(-1)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a directory index
|
2016-04-24 10:21:23 -07:00
|
|
|
res := c.Response()
|
|
|
|
res.Header().Set(echo.HeaderContentType, echo.MIMETextHTMLCharsetUTF8)
|
|
|
|
if _, err = fmt.Fprintf(res, "<pre>\n"); err != nil {
|
2016-04-10 20:56:10 -07:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, d := range dirs {
|
|
|
|
name := d.Name()
|
|
|
|
color := "#212121"
|
|
|
|
if d.IsDir() {
|
|
|
|
color = "#e91e63"
|
|
|
|
name += "/"
|
|
|
|
}
|
2016-04-24 10:21:23 -07:00
|
|
|
if _, err = fmt.Fprintf(res, "<a href=\"%s\" style=\"color: %s;\">%s</a>\n", name, color, name); err != nil {
|
2016-04-10 20:56:10 -07:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2016-04-24 10:21:23 -07:00
|
|
|
_, err = fmt.Fprintf(res, "</pre>\n")
|
2016-04-10 20:56:10 -07:00
|
|
|
return err
|
2016-05-01 06:56:52 -07:00
|
|
|
} else {
|
|
|
|
return next(c)
|
2016-04-10 20:56:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return c.ServeContent(f, fi.Name(), fi.ModTime())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|