mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-05 18:44:58 +03:00
browse: Use helper functions in staticfiles to redirect (#1497)
* Use helper functions in staticfiles to redirect. Previously the browse package invoked staticfiles.Redirect when redirecting clients who requested a directory but with a Request-URI that did not contain a trailing '/'. staticfiles.Redirect only used a relative URI. This change defers the decision of how to format the Location header value to the helper methods in the staticfiles package. * Update const URLPathCtxKey in browse package.
This commit is contained in:
parent
a148b92381
commit
36d2027493
3 changed files with 96 additions and 26 deletions
|
@ -299,7 +299,7 @@ func (b Browse) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
|
|||
// Browsing navigation gets messed up if browsing a directory
|
||||
// that doesn't end in "/" (which it should, anyway)
|
||||
if !strings.HasSuffix(r.URL.Path, "/") {
|
||||
staticfiles.Redirect(w, r, r.URL.Path+"/", http.StatusTemporaryRedirect)
|
||||
staticfiles.RedirectToDir(w, r)
|
||||
return 0, nil
|
||||
}
|
||||
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
package browse
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
|
@ -362,3 +363,68 @@ func isReversed(data sort.Interface) bool {
|
|||
}
|
||||
return true
|
||||
}
|
||||
|
||||
func TestBrowseRedirect(t *testing.T) {
|
||||
testCases := []struct {
|
||||
url string
|
||||
statusCode int
|
||||
returnCode int
|
||||
location string
|
||||
}{
|
||||
{
|
||||
"http://www.example.com/photos",
|
||||
http.StatusMovedPermanently,
|
||||
0,
|
||||
"http://www.example.com/photos/",
|
||||
},
|
||||
{
|
||||
"/photos",
|
||||
http.StatusMovedPermanently,
|
||||
0,
|
||||
"/photos/",
|
||||
},
|
||||
}
|
||||
|
||||
for i, tc := range testCases {
|
||||
b := Browse{
|
||||
Next: httpserver.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) {
|
||||
t.Fatalf("Test %d - Next shouldn't be called", i)
|
||||
return 0, nil
|
||||
}),
|
||||
Configs: []Config{
|
||||
{
|
||||
PathScope: "/photos",
|
||||
Fs: staticfiles.FileServer{
|
||||
Root: http.Dir("./testdata"),
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
req, err := http.NewRequest("GET", tc.url, nil)
|
||||
u, _ := url.Parse(tc.url)
|
||||
ctx := context.WithValue(req.Context(), staticfiles.URLPathCtxKey, u.Path)
|
||||
req = req.WithContext(ctx)
|
||||
if err != nil {
|
||||
t.Fatalf("Test %d - could not create HTTP request: %v", i, err)
|
||||
}
|
||||
|
||||
rec := httptest.NewRecorder()
|
||||
|
||||
returnCode, _ := b.ServeHTTP(rec, req)
|
||||
if returnCode != tc.returnCode {
|
||||
t.Fatalf("Test %d - wrong return code, expected %d, got %d",
|
||||
i, tc.returnCode, returnCode)
|
||||
}
|
||||
|
||||
if got := rec.Code; got != tc.statusCode {
|
||||
t.Errorf("Test %d - wrong status, expected %d, got %d",
|
||||
i, tc.statusCode, got)
|
||||
}
|
||||
|
||||
if got := rec.Header().Get("Location"); got != tc.location {
|
||||
t.Errorf("Test %d - wrong Location header, expected %s, got %s",
|
||||
i, tc.location, got)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -96,30 +96,14 @@ func (fs FileServer) serveFile(w http.ResponseWriter, r *http.Request, name stri
|
|||
// Ensure / at end of directory url. If the original URL path is
|
||||
// used then ensure / exists as well.
|
||||
if !strings.HasSuffix(r.URL.Path, "/") {
|
||||
toURL, _ := url.Parse(r.URL.String())
|
||||
|
||||
path, ok := r.Context().Value(URLPathCtxKey).(string)
|
||||
if ok && !strings.HasSuffix(path, "/") {
|
||||
toURL.Path = path
|
||||
}
|
||||
toURL.Path += "/"
|
||||
|
||||
http.Redirect(w, r, toURL.String(), http.StatusMovedPermanently)
|
||||
RedirectToDir(w, r)
|
||||
return http.StatusMovedPermanently, nil
|
||||
}
|
||||
} else {
|
||||
// Ensure no / at end of file url. If the original URL path is
|
||||
// used then ensure no / exists as well.
|
||||
if strings.HasSuffix(r.URL.Path, "/") {
|
||||
toURL, _ := url.Parse(r.URL.String())
|
||||
|
||||
path, ok := r.Context().Value(URLPathCtxKey).(string)
|
||||
if ok && strings.HasSuffix(path, "/") {
|
||||
toURL.Path = path
|
||||
}
|
||||
toURL.Path = strings.TrimSuffix(toURL.Path, "/")
|
||||
|
||||
http.Redirect(w, r, toURL.String(), http.StatusMovedPermanently)
|
||||
RedirectToFile(w, r)
|
||||
return http.StatusMovedPermanently, nil
|
||||
}
|
||||
}
|
||||
|
@ -234,14 +218,34 @@ func (fs FileServer) IsHidden(d os.FileInfo) bool {
|
|||
return false
|
||||
}
|
||||
|
||||
// Redirect sends an HTTP redirect to the client but will preserve
|
||||
// the query string for the new path. Based on http.localRedirect
|
||||
// from the Go standard library.
|
||||
func Redirect(w http.ResponseWriter, r *http.Request, newPath string, statusCode int) {
|
||||
if q := r.URL.RawQuery; q != "" {
|
||||
newPath += "?" + q
|
||||
// RedirectToDir replies to the request with a redirect to the URL in r, which
|
||||
// has been transformed to indicate that the resource being requested is a
|
||||
// directory.
|
||||
func RedirectToDir(w http.ResponseWriter, r *http.Request) {
|
||||
toURL, _ := url.Parse(r.URL.String())
|
||||
|
||||
path, ok := r.Context().Value(URLPathCtxKey).(string)
|
||||
if ok && !strings.HasSuffix(path, "/") {
|
||||
toURL.Path = path
|
||||
}
|
||||
http.Redirect(w, r, newPath, statusCode)
|
||||
toURL.Path += "/"
|
||||
|
||||
http.Redirect(w, r, toURL.String(), http.StatusMovedPermanently)
|
||||
}
|
||||
|
||||
// RedirectToFile replies to the request with a redirect to the URL in r, which
|
||||
// has been transformed to indicate that the resource being requested is a
|
||||
// file.
|
||||
func RedirectToFile(w http.ResponseWriter, r *http.Request) {
|
||||
toURL, _ := url.Parse(r.URL.String())
|
||||
|
||||
path, ok := r.Context().Value(URLPathCtxKey).(string)
|
||||
if ok && strings.HasSuffix(path, "/") {
|
||||
toURL.Path = path
|
||||
}
|
||||
toURL.Path = strings.TrimSuffix(toURL.Path, "/")
|
||||
|
||||
http.Redirect(w, r, toURL.String(), http.StatusMovedPermanently)
|
||||
}
|
||||
|
||||
// IndexPages is a list of pages that may be understood as
|
||||
|
|
Loading…
Reference in a new issue