mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-19 09:05:41 +03:00
55 lines
1.3 KiB
Go
55 lines
1.3 KiB
Go
|
package staticfiles
|
||
|
|
||
|
import (
|
||
|
"net/http"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
|
||
|
"bitbucket.org/lightcodelabs/caddy2"
|
||
|
"bitbucket.org/lightcodelabs/caddy2/modules/caddyhttp"
|
||
|
)
|
||
|
|
||
|
func init() {
|
||
|
caddy2.RegisterModule(caddy2.Module{
|
||
|
Name: "http.matchers.file",
|
||
|
New: func() (interface{}, error) { return new(FileMatcher), nil },
|
||
|
})
|
||
|
}
|
||
|
|
||
|
// TODO: Not sure how to do this well; we'd need the ability to
|
||
|
// hide files, etc...
|
||
|
// TODO: Also consider a feature to match directory that
|
||
|
// contains a certain filename (use filepath.Glob), useful
|
||
|
// if wanting to map directory-URI requests where the dir
|
||
|
// has index.php to PHP backends, for example (although this
|
||
|
// can effectively be done with rehandling already)
|
||
|
type FileMatcher struct {
|
||
|
Root string `json:"root"`
|
||
|
Path string `json:"path"`
|
||
|
Flags []string `json:"flags"`
|
||
|
}
|
||
|
|
||
|
func (m FileMatcher) Match(r *http.Request) bool {
|
||
|
// TODO: sanitize path
|
||
|
fullPath := filepath.Join(m.Root, m.Path)
|
||
|
var match bool
|
||
|
if len(m.Flags) > 0 {
|
||
|
match = true
|
||
|
fi, err := os.Stat(fullPath)
|
||
|
for _, f := range m.Flags {
|
||
|
switch f {
|
||
|
case "EXIST":
|
||
|
match = match && os.IsNotExist(err)
|
||
|
case "DIR":
|
||
|
match = match && err == nil && fi.IsDir()
|
||
|
default:
|
||
|
match = false
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return match
|
||
|
}
|
||
|
|
||
|
// Interface guard
|
||
|
var _ caddyhttp.RequestMatcher = (*FileMatcher)(nil)
|