2020-04-27 15:46:46 -05:00
|
|
|
// Copyright 2015 Matthew Holt and The Caddy Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package fileserver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2021-09-16 15:40:31 -05:00
|
|
|
"os"
|
|
|
|
"runtime"
|
2020-04-27 15:46:46 -05:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
|
|
|
)
|
|
|
|
|
2020-09-16 19:09:28 -05:00
|
|
|
func TestFileMatcher(t *testing.T) {
|
2021-09-16 15:40:31 -05:00
|
|
|
|
|
|
|
// Windows doesn't like colons in files names
|
|
|
|
isWindows := runtime.GOOS == "windows"
|
|
|
|
if !isWindows {
|
|
|
|
filename := "with:in-name.txt"
|
|
|
|
f, err := os.Create("./testdata/" + filename)
|
|
|
|
if err != nil {
|
|
|
|
t.Fail()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
t.Cleanup(func() {
|
|
|
|
os.Remove("./testdata/" + filename)
|
|
|
|
})
|
|
|
|
f.WriteString(filename)
|
|
|
|
f.Close()
|
|
|
|
}
|
|
|
|
|
2020-09-16 19:09:28 -05:00
|
|
|
for i, tc := range []struct {
|
|
|
|
path string
|
|
|
|
expectedPath string
|
|
|
|
expectedType string
|
|
|
|
matched bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
path: "/foo.txt",
|
|
|
|
expectedPath: "/foo.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: "/foo.txt/",
|
|
|
|
expectedPath: "/foo.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: "/foodir",
|
|
|
|
expectedPath: "/foodir/",
|
|
|
|
expectedType: "directory",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: "/foodir/",
|
|
|
|
expectedPath: "/foodir/",
|
|
|
|
expectedType: "directory",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: "/foodir/foo.txt",
|
|
|
|
expectedPath: "/foodir/foo.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: "/missingfile.php",
|
|
|
|
matched: false,
|
|
|
|
},
|
2021-09-16 15:40:31 -05:00
|
|
|
{
|
|
|
|
path: "ملف.txt", // the path file name is not escaped
|
|
|
|
expectedPath: "ملف.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: url.PathEscape("ملف.txt"), // singly-escaped path
|
|
|
|
expectedPath: "ملف.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: url.PathEscape(url.PathEscape("ملف.txt")), // doubly-escaped path
|
|
|
|
expectedPath: "%D9%85%D9%84%D9%81.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
path: "./with:in-name.txt", // browsers send the request with the path as such
|
|
|
|
expectedPath: "with:in-name.txt",
|
|
|
|
expectedType: "file",
|
|
|
|
matched: !isWindows,
|
|
|
|
},
|
2020-09-16 19:09:28 -05:00
|
|
|
} {
|
|
|
|
m := &MatchFile{
|
|
|
|
Root: "./testdata",
|
|
|
|
TryFiles: []string{"{http.request.uri.path}", "{http.request.uri.path}/"},
|
|
|
|
}
|
|
|
|
|
|
|
|
u, err := url.Parse(tc.path)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Test %d: parsing path: %v", i, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
req := &http.Request{URL: u}
|
|
|
|
repl := caddyhttp.NewTestReplacer(req)
|
|
|
|
|
|
|
|
result := m.Match(req)
|
|
|
|
if result != tc.matched {
|
|
|
|
t.Fatalf("Test %d: expected match=%t, got %t", i, tc.matched, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
rel, ok := repl.Get("http.matchers.file.relative")
|
|
|
|
if !ok && result {
|
|
|
|
t.Fatalf("Test %d: expected replacer value", i)
|
|
|
|
}
|
|
|
|
if !result {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if rel != tc.expectedPath {
|
|
|
|
t.Fatalf("Test %d: actual path: %v, expected: %v", i, rel, tc.expectedPath)
|
|
|
|
}
|
|
|
|
|
2021-06-17 10:59:08 -05:00
|
|
|
fileType, _ := repl.Get("http.matchers.file.type")
|
2020-09-16 19:09:28 -05:00
|
|
|
if fileType != tc.expectedType {
|
|
|
|
t.Fatalf("Test %d: actual file type: %v, expected: %v", i, fileType, tc.expectedType)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:55:01 -05:00
|
|
|
func TestPHPFileMatcher(t *testing.T) {
|
2020-04-27 15:46:46 -05:00
|
|
|
for i, tc := range []struct {
|
2020-07-31 14:55:01 -05:00
|
|
|
path string
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath string
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType string
|
2020-07-31 14:55:01 -05:00
|
|
|
matched bool
|
2020-04-27 15:46:46 -05:00
|
|
|
}{
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/index.php",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/index.php",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/index.php/somewhere",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/index.php",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/remote.php",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/remote.php",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/remote.php/somewhere",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/remote.php",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/missingfile.php",
|
2020-04-27 15:46:46 -05:00
|
|
|
matched: false,
|
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/notphp.php.txt",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/notphp.php.txt",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/notphp.php.txt/",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/notphp.php.txt",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/notphp.php.txt.suffixed",
|
2020-04-27 15:46:46 -05:00
|
|
|
matched: false,
|
|
|
|
},
|
|
|
|
{
|
2020-07-31 14:55:01 -05:00
|
|
|
path: "/foo.php.php/index.php",
|
2020-04-27 15:46:46 -05:00
|
|
|
expectedPath: "/foo.php.php/index.php",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// See https://github.com/caddyserver/caddy/issues/3623
|
|
|
|
path: "/%E2%C3",
|
|
|
|
expectedPath: "/%E2%C3",
|
2020-09-16 19:09:28 -05:00
|
|
|
expectedType: "file",
|
2020-07-31 14:55:01 -05:00
|
|
|
matched: false,
|
2020-04-27 15:46:46 -05:00
|
|
|
},
|
|
|
|
} {
|
|
|
|
m := &MatchFile{
|
|
|
|
Root: "./testdata",
|
2020-07-31 14:55:01 -05:00
|
|
|
TryFiles: []string{"{http.request.uri.path}", "{http.request.uri.path}/index.php"},
|
2020-04-27 15:46:46 -05:00
|
|
|
SplitPath: []string{".php"},
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:55:01 -05:00
|
|
|
u, err := url.Parse(tc.path)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Test %d: parsing path: %v", i, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
req := &http.Request{URL: u}
|
2020-04-27 15:46:46 -05:00
|
|
|
repl := caddyhttp.NewTestReplacer(req)
|
|
|
|
|
|
|
|
result := m.Match(req)
|
|
|
|
if result != tc.matched {
|
2020-07-31 14:55:01 -05:00
|
|
|
t.Fatalf("Test %d: expected match=%t, got %t", i, tc.matched, result)
|
2020-04-27 15:46:46 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
rel, ok := repl.Get("http.matchers.file.relative")
|
|
|
|
if !ok && result {
|
|
|
|
t.Fatalf("Test %d: expected replacer value", i)
|
|
|
|
}
|
|
|
|
if !result {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if rel != tc.expectedPath {
|
|
|
|
t.Fatalf("Test %d: actual path: %v, expected: %v", i, rel, tc.expectedPath)
|
|
|
|
}
|
2020-09-16 19:09:28 -05:00
|
|
|
|
2021-06-17 10:59:08 -05:00
|
|
|
fileType, _ := repl.Get("http.matchers.file.type")
|
2020-09-16 19:09:28 -05:00
|
|
|
if fileType != tc.expectedType {
|
|
|
|
t.Fatalf("Test %d: actual file type: %v, expected: %v", i, fileType, tc.expectedType)
|
|
|
|
}
|
2020-04-27 15:46:46 -05:00
|
|
|
}
|
2020-07-31 14:55:01 -05:00
|
|
|
}
|
2020-08-01 13:43:30 -05:00
|
|
|
|
|
|
|
func TestFirstSplit(t *testing.T) {
|
|
|
|
m := MatchFile{SplitPath: []string{".php"}}
|
2020-12-04 19:12:13 -05:00
|
|
|
actual, remainder := m.firstSplit("index.PHP/somewhere")
|
2020-08-01 13:43:30 -05:00
|
|
|
expected := "index.PHP"
|
2020-12-04 19:12:13 -05:00
|
|
|
expectedRemainder := "/somewhere"
|
2020-08-01 13:43:30 -05:00
|
|
|
if actual != expected {
|
2020-12-04 19:12:13 -05:00
|
|
|
t.Errorf("Expected split %s but got %s", expected, actual)
|
|
|
|
}
|
|
|
|
if remainder != expectedRemainder {
|
|
|
|
t.Errorf("Expected remainder %s but got %s", expectedRemainder, remainder)
|
2020-08-01 13:43:30 -05:00
|
|
|
}
|
|
|
|
}
|