mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-25 07:49:19 -04:00 
			
		
		
		
	Merge branch 'master' into letsencrypt
Conflicts: caddy/parse/parse.go caddy/parse/parsing.go config/config.go config/setup/controller.go main.go server/server.go
This commit is contained in:
		
						commit
						ad057ab873
					
				| @ -9,7 +9,6 @@ environment: | ||||
| 
 | ||||
| install: | ||||
|   - go get golang.org/x/tools/cmd/vet | ||||
|   - echo %PATH% | ||||
|   - echo %GOPATH% | ||||
|   - go version | ||||
|   - go env | ||||
|  | ||||
| @ -1,11 +1,27 @@ | ||||
| package caddy | ||||
| 
 | ||||
| import ( | ||||
| 	"reflect" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 
 | ||||
| 	"github.com/mholt/caddy/server" | ||||
| ) | ||||
| 
 | ||||
| func TestNewDefault(t *testing.T) { | ||||
| 	config := NewDefault() | ||||
| 
 | ||||
| 	if actual, expected := config.Root, DefaultRoot; actual != expected { | ||||
| 		t.Errorf("Root was %s but expected %s", actual, expected) | ||||
| 	} | ||||
| 	if actual, expected := config.Host, DefaultHost; actual != expected { | ||||
| 		t.Errorf("Host was %s but expected %s", actual, expected) | ||||
| 	} | ||||
| 	if actual, expected := config.Port, DefaultPort; actual != expected { | ||||
| 		t.Errorf("Port was %s but expected %s", actual, expected) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestResolveAddr(t *testing.T) { | ||||
| 	// NOTE: If tests fail due to comparing to string "127.0.0.1", | ||||
| 	// it's possible that system env resolves with IPv6, or ::1. | ||||
| @ -62,3 +78,61 @@ func TestResolveAddr(t *testing.T) { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestMakeOnces(t *testing.T) { | ||||
| 	directives := []directive{ | ||||
| 		{"dummy", nil}, | ||||
| 		{"dummy2", nil}, | ||||
| 	} | ||||
| 	directiveOrder = directives | ||||
| 	onces := makeOnces() | ||||
| 	if len(onces) != len(directives) { | ||||
| 		t.Errorf("onces had len %d , expected %d", len(onces), len(directives)) | ||||
| 	} | ||||
| 	expected := map[string]*sync.Once{ | ||||
| 		"dummy":  new(sync.Once), | ||||
| 		"dummy2": new(sync.Once), | ||||
| 	} | ||||
| 	if !reflect.DeepEqual(onces, expected) { | ||||
| 		t.Errorf("onces was %v, expected %v", onces, expected) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestMakeStorages(t *testing.T) { | ||||
| 	directives := []directive{ | ||||
| 		{"dummy", nil}, | ||||
| 		{"dummy2", nil}, | ||||
| 	} | ||||
| 	directiveOrder = directives | ||||
| 	storages := makeStorages() | ||||
| 	if len(storages) != len(directives) { | ||||
| 		t.Errorf("storages had len %d , expected %d", len(storages), len(directives)) | ||||
| 	} | ||||
| 	expected := map[string]interface{}{ | ||||
| 		"dummy":  nil, | ||||
| 		"dummy2": nil, | ||||
| 	} | ||||
| 	if !reflect.DeepEqual(storages, expected) { | ||||
| 		t.Errorf("storages was %v, expected %v", storages, expected) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestValidDirective(t *testing.T) { | ||||
| 	directives := []directive{ | ||||
| 		{"dummy", nil}, | ||||
| 		{"dummy2", nil}, | ||||
| 	} | ||||
| 	directiveOrder = directives | ||||
| 	for i, test := range []struct { | ||||
| 		directive string | ||||
| 		valid     bool | ||||
| 	}{ | ||||
| 		{"dummy", true}, | ||||
| 		{"dummy2", true}, | ||||
| 		{"dummy3", false}, | ||||
| 	} { | ||||
| 		if actual, expected := validDirective(test.directive), test.valid; actual != expected { | ||||
| 			t.Errorf("Test %d: valid was %t, expected %t", i, actual, expected) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -58,6 +58,9 @@ func NewTestController(input string) *Controller { | ||||
| 			Root: ".", | ||||
| 		}, | ||||
| 		Dispenser: parse.NewDispenser("Testfile", strings.NewReader(input)), | ||||
| 		OncePerServerBlock: func(f func() error) error { | ||||
| 			return f() | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										58
									
								
								caddy/setup/startupshutdown_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								caddy/setup/startupshutdown_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | ||||
| package setup | ||||
| 
 | ||||
| import ( | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // The Startup function's tests are symmetrical to Shutdown tests, | ||||
| // because the Startup and Shutdown functions share virtually the | ||||
| // same functionality | ||||
| func TestStartup(t *testing.T) { | ||||
| 
 | ||||
| 	tempDirPath, err := getTempDirPath() | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("BeforeTest: Failed to find an existing directory for testing! Error was: %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	testDir := filepath.Join(tempDirPath, "temp_dir_for_testing_startupshutdown.go") | ||||
| 	osSenitiveTestDir := filepath.FromSlash(testDir) | ||||
| 
 | ||||
| 	exec.Command("rm", "-r", osSenitiveTestDir).Run() // removes osSenitiveTestDir from the OS's temp directory, if the osSenitiveTestDir already exists | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		input              string | ||||
| 		shouldExecutionErr bool | ||||
| 		shouldRemoveErr    bool | ||||
| 	}{ | ||||
| 		// test case #0 tests proper functionality blocking commands | ||||
| 		{"startup mkdir " + osSenitiveTestDir, false, false}, | ||||
| 
 | ||||
| 		// test case #1 tests proper functionality of non-blocking commands | ||||
| 		{"startup mkdir " + osSenitiveTestDir + " &", false, true}, | ||||
| 
 | ||||
| 		// test case #2 tests handling of non-existant commands | ||||
| 		{"startup " + strconv.Itoa(int(time.Now().UnixNano())), true, true}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		c := NewTestController(test.input) | ||||
| 		_, err = Startup(c) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("Expected no errors, got: %v", err) | ||||
| 		} | ||||
| 		err = c.Startup[0]() | ||||
| 		if err != nil && !test.shouldExecutionErr { | ||||
| 			t.Errorf("Test %d recieved an error of:\n%v", i, err) | ||||
| 		} | ||||
| 		err = os.Remove(osSenitiveTestDir) | ||||
| 		if err != nil && !test.shouldRemoveErr { | ||||
| 			t.Errorf("Test %d recieved an error of:\n%v", i, err) | ||||
| 		} | ||||
| 
 | ||||
| 	} | ||||
| } | ||||
| @ -32,18 +32,48 @@ func templatesParse(c *Controller) ([]templates.Rule, error) { | ||||
| 	for c.Next() { | ||||
| 		var rule templates.Rule | ||||
| 
 | ||||
| 		if c.NextArg() { | ||||
| 		rule.Path = defaultTemplatePath | ||||
| 		rule.Extensions = defaultTemplateExtensions | ||||
| 
 | ||||
| 		args := c.RemainingArgs() | ||||
| 
 | ||||
| 		switch len(args) { | ||||
| 		case 0: | ||||
| 			// Optional block | ||||
| 			for c.NextBlock() { | ||||
| 				switch c.Val() { | ||||
| 				case "path": | ||||
| 					args := c.RemainingArgs() | ||||
| 					if len(args) != 1 { | ||||
| 						return nil, c.ArgErr() | ||||
| 					} | ||||
| 					rule.Path = args[0] | ||||
| 
 | ||||
| 				case "ext": | ||||
| 					args := c.RemainingArgs() | ||||
| 					if len(args) == 0 { | ||||
| 						return nil, c.ArgErr() | ||||
| 					} | ||||
| 					rule.Extensions = args | ||||
| 
 | ||||
| 				case "between": | ||||
| 					args := c.RemainingArgs() | ||||
| 					if len(args) != 2 { | ||||
| 						return nil, c.ArgErr() | ||||
| 					} | ||||
| 					rule.Delims[0] = args[0] | ||||
| 					rule.Delims[1] = args[1] | ||||
| 				} | ||||
| 			} | ||||
| 		default: | ||||
| 			// First argument would be the path | ||||
| 			rule.Path = c.Val() | ||||
| 			rule.Path = args[0] | ||||
| 
 | ||||
| 			// Any remaining arguments are extensions | ||||
| 			rule.Extensions = c.RemainingArgs() | ||||
| 			rule.Extensions = args[1:] | ||||
| 			if len(rule.Extensions) == 0 { | ||||
| 				rule.Extensions = defaultTemplateExtensions | ||||
| 			} | ||||
| 		} else { | ||||
| 			rule.Path = defaultTemplatePath | ||||
| 			rule.Extensions = defaultTemplateExtensions | ||||
| 		} | ||||
| 
 | ||||
| 		for _, ext := range rule.Extensions { | ||||
| @ -52,7 +82,6 @@ func templatesParse(c *Controller) ([]templates.Rule, error) { | ||||
| 
 | ||||
| 		rules = append(rules, rule) | ||||
| 	} | ||||
| 
 | ||||
| 	return rules, nil | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -2,8 +2,9 @@ package setup | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/mholt/caddy/middleware/templates" | ||||
| 	"testing" | ||||
| 
 | ||||
| 	"github.com/mholt/caddy/middleware/templates" | ||||
| ) | ||||
| 
 | ||||
| func TestTemplates(t *testing.T) { | ||||
| @ -40,7 +41,11 @@ func TestTemplates(t *testing.T) { | ||||
| 	if fmt.Sprint(myHandler.Rules[0].IndexFiles) != fmt.Sprint(indexFiles) { | ||||
| 		t.Errorf("Expected %v to be the Default Index files", indexFiles) | ||||
| 	} | ||||
| 	if myHandler.Rules[0].Delims != [2]string{} { | ||||
| 		t.Errorf("Expected %v to be the Default Delims", [2]string{}) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestTemplatesParse(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		inputTemplateConfig    string | ||||
| @ -50,19 +55,32 @@ func TestTemplatesParse(t *testing.T) { | ||||
| 		{`templates /api1`, false, []templates.Rule{{ | ||||
| 			Path:       "/api1", | ||||
| 			Extensions: defaultTemplateExtensions, | ||||
| 			Delims:     [2]string{}, | ||||
| 		}}}, | ||||
| 		{`templates /api2 .txt .htm`, false, []templates.Rule{{ | ||||
| 			Path:       "/api2", | ||||
| 			Extensions: []string{".txt", ".htm"}, | ||||
| 			Delims:     [2]string{}, | ||||
| 		}}}, | ||||
| 
 | ||||
| 		{`templates /api3 .htm .html | ||||
| 		  templates /api4 .txt .tpl `, false, []templates.Rule{{ | ||||
| 			Path:       "/api3", | ||||
| 			Extensions: []string{".htm", ".html"}, | ||||
| 			Delims:     [2]string{}, | ||||
| 		}, { | ||||
| 			Path:       "/api4", | ||||
| 			Extensions: []string{".txt", ".tpl"}, | ||||
| 			Delims:     [2]string{}, | ||||
| 		}}}, | ||||
| 		{`templates { | ||||
| 				path /api5 | ||||
| 				ext .html | ||||
| 				between {% %} | ||||
| 			}`, false, []templates.Rule{{ | ||||
| 			Path:       "/api5", | ||||
| 			Extensions: []string{".html"}, | ||||
| 			Delims:     [2]string{"{%", "%}"}, | ||||
| 		}}}, | ||||
| 	} | ||||
| 	for i, test := range tests { | ||||
|  | ||||
| @ -16,6 +16,12 @@ func TLS(c *Controller) (middleware.Middleware, error) { | ||||
| 			"specify port 80 explicitly (https://%s:80).", c.Port, c.Host, c.Host) | ||||
| 	} | ||||
| 
 | ||||
| 	if c.Port == "http" { | ||||
| 		c.TLS.Enabled = false | ||||
| 		log.Printf("Warning: TLS disabled for %s://%s. To force TLS over the plaintext HTTP port, "+ | ||||
| 			"specify port 80 explicitly (https://%s:80).", c.Port, c.Host, c.Host) | ||||
| 	} | ||||
| 
 | ||||
| 	for c.Next() { | ||||
| 		args := c.RemainingArgs() | ||||
| 		switch len(args) { | ||||
|  | ||||
| @ -54,6 +54,25 @@ func TestWebSocketParse(t *testing.T) { | ||||
| 			Path:    "/api4", | ||||
| 			Command: "cat", | ||||
| 		}}}, | ||||
| 
 | ||||
| 		{`websocket /api5 "cmd arg1 arg2 arg3"`, false, []websocket.Config{{ | ||||
| 			Path:      "/api5", | ||||
| 			Command:   "cmd", | ||||
| 			Arguments: []string{"arg1", "arg2", "arg3"}, | ||||
| 		}}}, | ||||
| 
 | ||||
| 		// accept respawn | ||||
| 		{`websocket /api6 cat { | ||||
| 			respawn | ||||
| 		}`, false, []websocket.Config{{ | ||||
| 			Path:    "/api6", | ||||
| 			Command: "cat", | ||||
| 		}}}, | ||||
| 
 | ||||
| 		// invalid configuration | ||||
| 		{`websocket /api7 cat { | ||||
| 			invalid | ||||
| 		}`, true, []websocket.Config{}}, | ||||
| 	} | ||||
| 	for i, test := range tests { | ||||
| 		c := NewTestController(test.inputWebSocketConfig) | ||||
|  | ||||
							
								
								
									
										40
									
								
								main_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								main_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,40 @@ | ||||
| package main | ||||
| 
 | ||||
| import ( | ||||
| 	"runtime" | ||||
| 	"testing" | ||||
| ) | ||||
| 
 | ||||
| func TestSetCPU(t *testing.T) { | ||||
| 	currentCPU := runtime.GOMAXPROCS(-1) | ||||
| 	maxCPU := runtime.NumCPU() | ||||
| 	for i, test := range []struct { | ||||
| 		input     string | ||||
| 		output    int | ||||
| 		shouldErr bool | ||||
| 	}{ | ||||
| 		{"1", 1, false}, | ||||
| 		{"-1", currentCPU, true}, | ||||
| 		{"0", currentCPU, true}, | ||||
| 		{"100%", maxCPU, false}, | ||||
| 		{"50%", int(0.5 * float32(maxCPU)), false}, | ||||
| 		{"110%", currentCPU, true}, | ||||
| 		{"-10%", currentCPU, true}, | ||||
| 		{"invalid input", currentCPU, true}, | ||||
| 		{"invalid input%", currentCPU, true}, | ||||
| 		{"9999", maxCPU, false}, // over available CPU | ||||
| 	} { | ||||
| 		err := setCPU(test.input) | ||||
| 		if test.shouldErr && err == nil { | ||||
| 			t.Errorf("Test %d: Expected error, but there wasn't any", i) | ||||
| 		} | ||||
| 		if !test.shouldErr && err != nil { | ||||
| 			t.Errorf("Test %d: Expected no error, but there was one: %v", i, err) | ||||
| 		} | ||||
| 		if actual, expected := runtime.GOMAXPROCS(-1), test.output; actual != expected { | ||||
| 			t.Errorf("Test %d: GOMAXPROCS was %d but expected %d", i, actual, expected) | ||||
| 		} | ||||
| 		// teardown | ||||
| 		runtime.GOMAXPROCS(currentCPU) | ||||
| 	} | ||||
| } | ||||
| @ -2,18 +2,30 @@ package middleware | ||||
| 
 | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"runtime" | ||||
| 	"unicode" | ||||
| 
 | ||||
| 	"github.com/flynn/go-shlex" | ||||
| ) | ||||
| 
 | ||||
| var runtimeGoos = runtime.GOOS | ||||
| 
 | ||||
| // SplitCommandAndArgs takes a command string and parses it | ||||
| // shell-style into the command and its separate arguments. | ||||
| func SplitCommandAndArgs(command string) (cmd string, args []string, err error) { | ||||
| 	parts, err := shlex.Split(command) | ||||
| 	var parts []string | ||||
| 
 | ||||
| 	if runtimeGoos == "windows" { | ||||
| 		parts = parseWindowsCommand(command) // parse it Windows-style | ||||
| 	} else { | ||||
| 		parts, err = parseUnixCommand(command) // parse it Unix-style | ||||
| 		if err != nil { | ||||
| 			err = errors.New("error parsing command: " + err.Error()) | ||||
| 			return | ||||
| 	} else if len(parts) == 0 { | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if len(parts) == 0 { | ||||
| 		err = errors.New("no command contained in '" + command + "'") | ||||
| 		return | ||||
| 	} | ||||
| @ -25,3 +37,84 @@ func SplitCommandAndArgs(command string) (cmd string, args []string, err error) | ||||
| 
 | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| // parseUnixCommand parses a unix style command line and returns the | ||||
| // command and its arguments or an error | ||||
| func parseUnixCommand(cmd string) ([]string, error) { | ||||
| 	return shlex.Split(cmd) | ||||
| } | ||||
| 
 | ||||
| // parseWindowsCommand parses windows command lines and | ||||
| // returns the command and the arguments as an array. It | ||||
| // should be able to parse commonly used command lines. | ||||
| // Only basic syntax is supported: | ||||
| //  - spaces in double quotes are not token delimiters | ||||
| //  - double quotes are escaped by either backspace or another double quote | ||||
| //  - except for the above case backspaces are path separators (not special) | ||||
| // | ||||
| // Many sources point out that escaping quotes using backslash can be unsafe. | ||||
| // Use two double quotes when possible. (Source: http://stackoverflow.com/a/31413730/2616179 ) | ||||
| // | ||||
| // This function has to be used on Windows instead | ||||
| // of the shlex package because this function treats backslash | ||||
| // characters properly. | ||||
| func parseWindowsCommand(cmd string) []string { | ||||
| 	const backslash = '\\' | ||||
| 	const quote = '"' | ||||
| 
 | ||||
| 	var parts []string | ||||
| 	var part string | ||||
| 	var inQuotes bool | ||||
| 	var lastRune rune | ||||
| 
 | ||||
| 	for i, ch := range cmd { | ||||
| 
 | ||||
| 		if i != 0 { | ||||
| 			lastRune = rune(cmd[i-1]) | ||||
| 		} | ||||
| 
 | ||||
| 		if ch == backslash { | ||||
| 			// put it in the part - for now we don't know if it's an | ||||
| 			// escaping char or path separator | ||||
| 			part += string(ch) | ||||
| 			continue | ||||
| 		} | ||||
| 
 | ||||
| 		if ch == quote { | ||||
| 			if lastRune == backslash { | ||||
| 				// remove the backslash from the part and add the escaped quote instead | ||||
| 				part = part[:len(part)-1] | ||||
| 				part += string(ch) | ||||
| 				continue | ||||
| 			} | ||||
| 
 | ||||
| 			if lastRune == quote { | ||||
| 				// revert the last change of the inQuotes state | ||||
| 				// it was an escaping quote | ||||
| 				inQuotes = !inQuotes | ||||
| 				part += string(ch) | ||||
| 				continue | ||||
| 			} | ||||
| 
 | ||||
| 			// normal escaping quotes | ||||
| 			inQuotes = !inQuotes | ||||
| 			continue | ||||
| 
 | ||||
| 		} | ||||
| 
 | ||||
| 		if unicode.IsSpace(ch) && !inQuotes && len(part) > 0 { | ||||
| 			parts = append(parts, part) | ||||
| 			part = "" | ||||
| 			continue | ||||
| 		} | ||||
| 
 | ||||
| 		part += string(ch) | ||||
| 	} | ||||
| 
 | ||||
| 	if len(part) > 0 { | ||||
| 		parts = append(parts, part) | ||||
| 		part = "" | ||||
| 	} | ||||
| 
 | ||||
| 	return parts | ||||
| } | ||||
|  | ||||
| @ -2,11 +2,176 @@ package middleware | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| ) | ||||
| 
 | ||||
| func TestParseUnixCommand(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		input    string | ||||
| 		expected []string | ||||
| 	}{ | ||||
| 		// 0 - emtpy command | ||||
| 		{ | ||||
| 			input:    ``, | ||||
| 			expected: []string{}, | ||||
| 		}, | ||||
| 		// 1 - command without arguments | ||||
| 		{ | ||||
| 			input:    `command`, | ||||
| 			expected: []string{`command`}, | ||||
| 		}, | ||||
| 		// 2 - command with single argument | ||||
| 		{ | ||||
| 			input:    `command arg1`, | ||||
| 			expected: []string{`command`, `arg1`}, | ||||
| 		}, | ||||
| 		// 3 - command with multiple arguments | ||||
| 		{ | ||||
| 			input:    `command arg1 arg2`, | ||||
| 			expected: []string{`command`, `arg1`, `arg2`}, | ||||
| 		}, | ||||
| 		// 4 - command with single argument with space character - in quotes | ||||
| 		{ | ||||
| 			input:    `command "arg1 arg1"`, | ||||
| 			expected: []string{`command`, `arg1 arg1`}, | ||||
| 		}, | ||||
| 		// 5 - command with multiple spaces and tab character | ||||
| 		{ | ||||
| 			input:    "command arg1    arg2\targ3", | ||||
| 			expected: []string{`command`, `arg1`, `arg2`, `arg3`}, | ||||
| 		}, | ||||
| 		// 6 - command with single argument with space character - escaped with backspace | ||||
| 		{ | ||||
| 			input:    `command arg1\ arg2`, | ||||
| 			expected: []string{`command`, `arg1 arg2`}, | ||||
| 		}, | ||||
| 		// 7 - single quotes should escape special chars | ||||
| 		{ | ||||
| 			input:    `command 'arg1\ arg2'`, | ||||
| 			expected: []string{`command`, `arg1\ arg2`}, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		errorPrefix := fmt.Sprintf("Test [%d]: ", i) | ||||
| 		errorSuffix := fmt.Sprintf(" Command to parse: [%s]", test.input) | ||||
| 		actual, _ := parseUnixCommand(test.input) | ||||
| 		if len(actual) != len(test.expected) { | ||||
| 			t.Errorf(errorPrefix+"Expected %d parts, got %d: %#v."+errorSuffix, len(test.expected), len(actual), actual) | ||||
| 			continue | ||||
| 		} | ||||
| 		for j := 0; j < len(actual); j++ { | ||||
| 			if expectedPart, actualPart := test.expected[j], actual[j]; expectedPart != actualPart { | ||||
| 				t.Errorf(errorPrefix+"Expected: %v Actual: %v (index %d)."+errorSuffix, expectedPart, actualPart, j) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestParseWindowsCommand(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		input    string | ||||
| 		expected []string | ||||
| 	}{ | ||||
| 		{ // 0 - empty command - do not fail | ||||
| 			input:    ``, | ||||
| 			expected: []string{}, | ||||
| 		}, | ||||
| 		{ // 1 - cmd without args | ||||
| 			input:    `cmd`, | ||||
| 			expected: []string{`cmd`}, | ||||
| 		}, | ||||
| 		{ // 2 - multiple args | ||||
| 			input:    `cmd arg1 arg2`, | ||||
| 			expected: []string{`cmd`, `arg1`, `arg2`}, | ||||
| 		}, | ||||
| 		{ // 3 - multiple args with space | ||||
| 			input:    `cmd "combined arg" arg2`, | ||||
| 			expected: []string{`cmd`, `combined arg`, `arg2`}, | ||||
| 		}, | ||||
| 		{ // 4 - path without spaces | ||||
| 			input:    `mkdir C:\Windows\foo\bar`, | ||||
| 			expected: []string{`mkdir`, `C:\Windows\foo\bar`}, | ||||
| 		}, | ||||
| 		{ // 5 - command with space in quotes | ||||
| 			input:    `"command here"`, | ||||
| 			expected: []string{`command here`}, | ||||
| 		}, | ||||
| 		{ // 6 - argument with escaped quotes (two quotes) | ||||
| 			input:    `cmd ""arg""`, | ||||
| 			expected: []string{`cmd`, `"arg"`}, | ||||
| 		}, | ||||
| 		{ // 7 - argument with escaped quotes (backslash) | ||||
| 			input:    `cmd \"arg\"`, | ||||
| 			expected: []string{`cmd`, `"arg"`}, | ||||
| 		}, | ||||
| 		{ // 8 - two quotes (escaped) inside an inQuote element | ||||
| 			input:    `cmd "a ""quoted value"`, | ||||
| 			expected: []string{`cmd`, `a "quoted value`}, | ||||
| 		}, | ||||
| 		// TODO - see how many quotes are dislayed if we use "", """, """"""" | ||||
| 		{ // 9 - two quotes outside an inQuote element | ||||
| 			input:    `cmd a ""quoted value`, | ||||
| 			expected: []string{`cmd`, `a`, `"quoted`, `value`}, | ||||
| 		}, | ||||
| 		{ // 10 - path with space in quotes | ||||
| 			input:    `mkdir "C:\directory name\foobar"`, | ||||
| 			expected: []string{`mkdir`, `C:\directory name\foobar`}, | ||||
| 		}, | ||||
| 		{ // 11 - space without quotes | ||||
| 			input:    `mkdir C:\ space`, | ||||
| 			expected: []string{`mkdir`, `C:\`, `space`}, | ||||
| 		}, | ||||
| 		{ // 12 - space in quotes | ||||
| 			input:    `mkdir "C:\ space"`, | ||||
| 			expected: []string{`mkdir`, `C:\ space`}, | ||||
| 		}, | ||||
| 		{ // 13 - UNC | ||||
| 			input:    `mkdir \\?\C:\Users`, | ||||
| 			expected: []string{`mkdir`, `\\?\C:\Users`}, | ||||
| 		}, | ||||
| 		{ // 14 - UNC with space | ||||
| 			input:    `mkdir "\\?\C:\Program Files"`, | ||||
| 			expected: []string{`mkdir`, `\\?\C:\Program Files`}, | ||||
| 		}, | ||||
| 
 | ||||
| 		{ // 15 - unclosed quotes - treat as if the path ends with quote | ||||
| 			input:    `mkdir "c:\Program files`, | ||||
| 			expected: []string{`mkdir`, `c:\Program files`}, | ||||
| 		}, | ||||
| 		{ // 16 - quotes used inside the argument | ||||
| 			input:    `mkdir "c:\P"rogra"m f"iles`, | ||||
| 			expected: []string{`mkdir`, `c:\Program files`}, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		errorPrefix := fmt.Sprintf("Test [%d]: ", i) | ||||
| 		errorSuffix := fmt.Sprintf(" Command to parse: [%s]", test.input) | ||||
| 
 | ||||
| 		actual := parseWindowsCommand(test.input) | ||||
| 		if len(actual) != len(test.expected) { | ||||
| 			t.Errorf(errorPrefix+"Expected %d parts, got %d: %#v."+errorSuffix, len(test.expected), len(actual), actual) | ||||
| 			continue | ||||
| 		} | ||||
| 		for j := 0; j < len(actual); j++ { | ||||
| 			if expectedPart, actualPart := test.expected[j], actual[j]; expectedPart != actualPart { | ||||
| 				t.Errorf(errorPrefix+"Expected: %v Actual: %v (index %d)."+errorSuffix, expectedPart, actualPart, j) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestSplitCommandAndArgs(t *testing.T) { | ||||
| 
 | ||||
| 	// force linux parsing. It's more robust and covers error cases | ||||
| 	runtimeGoos = "linux" | ||||
| 	defer func() { | ||||
| 		runtimeGoos = runtime.GOOS | ||||
| 	}() | ||||
| 
 | ||||
| 	var parseErrorContent = "error parsing command:" | ||||
| 	var noCommandErrContent = "no command contained in" | ||||
| 
 | ||||
| @ -16,84 +181,42 @@ func TestSplitCommandAndArgs(t *testing.T) { | ||||
| 		expectedArgs       []string | ||||
| 		expectedErrContent string | ||||
| 	}{ | ||||
| 		// Test case 0 - emtpy command | ||||
| 		// 0 - emtpy command | ||||
| 		{ | ||||
| 			input:              ``, | ||||
| 			expectedCommand:    ``, | ||||
| 			expectedArgs:       nil, | ||||
| 			expectedErrContent: noCommandErrContent, | ||||
| 		}, | ||||
| 		// Test case 1 - command without arguments | ||||
| 		// 1 - command without arguments | ||||
| 		{ | ||||
| 			input:              `command`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       nil, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 2 - command with single argument | ||||
| 		// 2 - command with single argument | ||||
| 		{ | ||||
| 			input:              `command arg1`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1`}, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 3 - command with multiple arguments | ||||
| 		// 3 - command with multiple arguments | ||||
| 		{ | ||||
| 			input:              `command arg1 arg2`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1`, `arg2`}, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 4 - command with single argument with space character - in quotes | ||||
| 		{ | ||||
| 			input:              `command "arg1 arg1"`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1 arg1`}, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 4 - command with single argument with space character - escaped | ||||
| 		{ | ||||
| 			input:              `command arg1\ arg1`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1 arg1`}, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 6 - command with escaped quote character | ||||
| 		{ | ||||
| 			input:              `command "arg1 \" arg1"`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1 " arg1`}, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 7 - command with escaped backslash | ||||
| 		{ | ||||
| 			input:              `command '\arg1'`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`\arg1`}, | ||||
| 			expectedErrContent: ``, | ||||
| 		}, | ||||
| 		// Test case 8 - command with comments | ||||
| 		{ | ||||
| 			input:              `command arg1 #comment1 comment2`, | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1`}, | ||||
| 			expectedErrContent: "", | ||||
| 		}, | ||||
| 		// Test case 9 - command with multiple spaces and tab character | ||||
| 		{ | ||||
| 			input:              "command arg1    arg2\targ3", | ||||
| 			expectedCommand:    `command`, | ||||
| 			expectedArgs:       []string{`arg1`, `arg2`, "arg3"}, | ||||
| 			expectedErrContent: "", | ||||
| 		}, | ||||
| 		// Test case 10 - command with unclosed quotes | ||||
| 		// 4 - command with unclosed quotes | ||||
| 		{ | ||||
| 			input:              `command "arg1 arg2`, | ||||
| 			expectedCommand:    "", | ||||
| 			expectedArgs:       nil, | ||||
| 			expectedErrContent: parseErrorContent, | ||||
| 		}, | ||||
| 		// Test case 11 - command with unclosed quotes | ||||
| 		// 5 - command with unclosed quotes | ||||
| 		{ | ||||
| 			input:              `command 'arg1 arg2"`, | ||||
| 			expectedCommand:    "", | ||||
| @ -120,14 +243,14 @@ func TestSplitCommandAndArgs(t *testing.T) { | ||||
| 
 | ||||
| 		// test if command matches | ||||
| 		if test.expectedCommand != actualCommand { | ||||
| 			t.Errorf("Expected command: [%s], actual: [%s]."+errorSuffix, test.expectedCommand, actualCommand) | ||||
| 			t.Errorf(errorPrefix+"Expected command: [%s], actual: [%s]."+errorSuffix, test.expectedCommand, actualCommand) | ||||
| 		} | ||||
| 
 | ||||
| 		// test if arguments match | ||||
| 		if len(test.expectedArgs) != len(actualArgs) { | ||||
| 			t.Errorf("Wrong number of arguments! Expected [%v], actual [%v]."+errorSuffix, test.expectedArgs, actualArgs) | ||||
| 		} | ||||
| 
 | ||||
| 			t.Errorf(errorPrefix+"Wrong number of arguments! Expected [%v], actual [%v]."+errorSuffix, test.expectedArgs, actualArgs) | ||||
| 		} else { | ||||
| 			// test args only if the count matches. | ||||
| 			for j, actualArg := range actualArgs { | ||||
| 				expectedArg := test.expectedArgs[j] | ||||
| 				if actualArg != expectedArg { | ||||
| @ -136,3 +259,33 @@ func TestSplitCommandAndArgs(t *testing.T) { | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func ExampleSplitCommandAndArgs() { | ||||
| 	var commandLine string | ||||
| 	var command string | ||||
| 	var args []string | ||||
| 
 | ||||
| 	// just for the test - change GOOS and reset it at the end of the test | ||||
| 	runtimeGoos = "windows" | ||||
| 	defer func() { | ||||
| 		runtimeGoos = runtime.GOOS | ||||
| 	}() | ||||
| 
 | ||||
| 	commandLine = `mkdir /P "C:\Program Files"` | ||||
| 	command, args, _ = SplitCommandAndArgs(commandLine) | ||||
| 
 | ||||
| 	fmt.Printf("Windows: %s: %s [%s]\n", commandLine, command, strings.Join(args, ",")) | ||||
| 
 | ||||
| 	// set GOOS to linux | ||||
| 	runtimeGoos = "linux" | ||||
| 
 | ||||
| 	commandLine = `mkdir -p /path/with\ space` | ||||
| 	command, args, _ = SplitCommandAndArgs(commandLine) | ||||
| 
 | ||||
| 	fmt.Printf("Linux: %s: %s [%s]\n", commandLine, command, strings.Join(args, ",")) | ||||
| 
 | ||||
| 	// Output: | ||||
| 	// Windows: mkdir /P "C:\Program Files": mkdir [/P,C:\Program Files] | ||||
| 	// Linux: mkdir -p /path/with\ space: mkdir [-p,/path/with space] | ||||
| } | ||||
|  | ||||
| @ -97,6 +97,10 @@ func (c Context) URI() string { | ||||
| func (c Context) Host() (string, error) { | ||||
| 	host, _, err := net.SplitHostPort(c.Req.Host) | ||||
| 	if err != nil { | ||||
| 		if !strings.Contains(c.Req.Host, ":") { | ||||
| 			// common with sites served on the default port 80 | ||||
| 			return c.Req.Host, nil | ||||
| 		} | ||||
| 		return "", err | ||||
| 	} | ||||
| 	return host, nil | ||||
|  | ||||
							
								
								
									
										545
									
								
								middleware/context_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										545
									
								
								middleware/context_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,545 @@ | ||||
| package middleware | ||||
| 
 | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| func TestInclude(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	inputFilename := "test_file" | ||||
| 	absInFilePath := filepath.Join(fmt.Sprintf("%s", context.Root), inputFilename) | ||||
| 	defer func() { | ||||
| 		err := os.Remove(absInFilePath) | ||||
| 		if err != nil && !os.IsNotExist(err) { | ||||
| 			t.Fatalf("Failed to clean test file!") | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		fileContent          string | ||||
| 		expectedContent      string | ||||
| 		shouldErr            bool | ||||
| 		expectedErrorContent string | ||||
| 	}{ | ||||
| 		// Test 0 - all good | ||||
| 		{ | ||||
| 			fileContent:          `str1 {{ .Root }} str2`, | ||||
| 			expectedContent:      fmt.Sprintf("str1 %s str2", context.Root), | ||||
| 			shouldErr:            false, | ||||
| 			expectedErrorContent: "", | ||||
| 		}, | ||||
| 		// Test 1 - failure on template.Parse | ||||
| 		{ | ||||
| 			fileContent:          `str1 {{ .Root } str2`, | ||||
| 			expectedContent:      "", | ||||
| 			shouldErr:            true, | ||||
| 			expectedErrorContent: `unexpected "}" in operand`, | ||||
| 		}, | ||||
| 		// Test 3 - failure on template.Execute | ||||
| 		{ | ||||
| 			fileContent:          `str1 {{ .InvalidField }} str2`, | ||||
| 			expectedContent:      "", | ||||
| 			shouldErr:            true, | ||||
| 			expectedErrorContent: `InvalidField is not a field of struct type middleware.Context`, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		testPrefix := getTestPrefix(i) | ||||
| 
 | ||||
| 		// WriteFile truncates the contentt | ||||
| 		err := ioutil.WriteFile(absInFilePath, []byte(test.fileContent), os.ModePerm) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(testPrefix+"Failed to create test file. Error was: %v", err) | ||||
| 		} | ||||
| 
 | ||||
| 		content, err := context.Include(inputFilename) | ||||
| 		if err != nil { | ||||
| 			if !test.shouldErr { | ||||
| 				t.Errorf(testPrefix+"Expected no error, found [%s]", test.expectedErrorContent, err.Error()) | ||||
| 			} | ||||
| 			if !strings.Contains(err.Error(), test.expectedErrorContent) { | ||||
| 				t.Errorf(testPrefix+"Expected error content [%s], found [%s]", test.expectedErrorContent, err.Error()) | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if err == nil && test.shouldErr { | ||||
| 			t.Errorf(testPrefix+"Expected error [%s] but found nil. Input file was: %s", test.expectedErrorContent, inputFilename) | ||||
| 		} | ||||
| 
 | ||||
| 		if content != test.expectedContent { | ||||
| 			t.Errorf(testPrefix+"Expected content [%s] but found [%s]. Input file was: %s", test.expectedContent, content, inputFilename) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestIncludeNotExisting(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	_, err := context.Include("not_existing") | ||||
| 	if err == nil { | ||||
| 		t.Errorf("Expected error but found nil!") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestCookie(t *testing.T) { | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		cookie        *http.Cookie | ||||
| 		cookieName    string | ||||
| 		expectedValue string | ||||
| 	}{ | ||||
| 		// Test 0 - happy path | ||||
| 		{ | ||||
| 			cookie:        &http.Cookie{Name: "cookieName", Value: "cookieValue"}, | ||||
| 			cookieName:    "cookieName", | ||||
| 			expectedValue: "cookieValue", | ||||
| 		}, | ||||
| 		// Test 1 - try to get a non-existing cookie | ||||
| 		{ | ||||
| 			cookie:        &http.Cookie{Name: "cookieName", Value: "cookieValue"}, | ||||
| 			cookieName:    "notExisting", | ||||
| 			expectedValue: "", | ||||
| 		}, | ||||
| 		// Test 2 - partial name match | ||||
| 		{ | ||||
| 			cookie:        &http.Cookie{Name: "cookie", Value: "cookieValue"}, | ||||
| 			cookieName:    "cook", | ||||
| 			expectedValue: "", | ||||
| 		}, | ||||
| 		// Test 3 - cookie with optional fields | ||||
| 		{ | ||||
| 			cookie:        &http.Cookie{Name: "cookie", Value: "cookieValue", Path: "/path", Domain: "https://localhost", Expires: (time.Now().Add(10 * time.Minute)), MaxAge: 120}, | ||||
| 			cookieName:    "cookie", | ||||
| 			expectedValue: "cookieValue", | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		testPrefix := getTestPrefix(i) | ||||
| 
 | ||||
| 		// reinitialize the context for each test | ||||
| 		context := getContextOrFail(t) | ||||
| 
 | ||||
| 		context.Req.AddCookie(test.cookie) | ||||
| 
 | ||||
| 		actualCookieVal := context.Cookie(test.cookieName) | ||||
| 
 | ||||
| 		if actualCookieVal != test.expectedValue { | ||||
| 			t.Errorf(testPrefix+"Expected cookie value [%s] but found [%s] for cookie with name %s", test.expectedValue, actualCookieVal, test.cookieName) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestCookieMultipleCookies(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	cookieNameBase, cookieValueBase := "cookieName", "cookieValue" | ||||
| 
 | ||||
| 	// make sure that there's no state and multiple requests for different cookies return the correct result | ||||
| 	for i := 0; i < 10; i++ { | ||||
| 		context.Req.AddCookie(&http.Cookie{Name: fmt.Sprintf("%s%d", cookieNameBase, i), Value: fmt.Sprintf("%s%d", cookieValueBase, i)}) | ||||
| 	} | ||||
| 
 | ||||
| 	for i := 0; i < 10; i++ { | ||||
| 		expectedCookieVal := fmt.Sprintf("%s%d", cookieValueBase, i) | ||||
| 		actualCookieVal := context.Cookie(fmt.Sprintf("%s%d", cookieNameBase, i)) | ||||
| 		if actualCookieVal != expectedCookieVal { | ||||
| 			t.Fatalf("Expected cookie value %s, found %s", expectedCookieVal, actualCookieVal) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestHeader(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	headerKey, headerVal := "Header1", "HeaderVal1" | ||||
| 	context.Req.Header.Add(headerKey, headerVal) | ||||
| 
 | ||||
| 	actualHeaderVal := context.Header(headerKey) | ||||
| 	if actualHeaderVal != headerVal { | ||||
| 		t.Errorf("Expected header %s, found %s", headerVal, actualHeaderVal) | ||||
| 	} | ||||
| 
 | ||||
| 	missingHeaderVal := context.Header("not-existing") | ||||
| 	if missingHeaderVal != "" { | ||||
| 		t.Errorf("Expected empty header value, found %s", missingHeaderVal) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestIP(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		inputRemoteAddr string | ||||
| 		expectedIP      string | ||||
| 	}{ | ||||
| 		// Test 0 - ipv4 with port | ||||
| 		{"1.1.1.1:1111", "1.1.1.1"}, | ||||
| 		// Test 1 - ipv4 without port | ||||
| 		{"1.1.1.1", "1.1.1.1"}, | ||||
| 		// Test 2 - ipv6 with port | ||||
| 		{"[::1]:11", "::1"}, | ||||
| 		// Test 3 - ipv6 without port and brackets | ||||
| 		{"[2001:db8:a0b:12f0::1]", "[2001:db8:a0b:12f0::1]"}, | ||||
| 		// Test 4 - ipv6 with zone and port | ||||
| 		{`[fe80:1::3%eth0]:44`, `fe80:1::3%eth0`}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		testPrefix := getTestPrefix(i) | ||||
| 
 | ||||
| 		context.Req.RemoteAddr = test.inputRemoteAddr | ||||
| 		actualIP := context.IP() | ||||
| 
 | ||||
| 		if actualIP != test.expectedIP { | ||||
| 			t.Errorf(testPrefix+"Expected IP %s, found %s", test.expectedIP, actualIP) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestURL(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	inputURL := "http://localhost" | ||||
| 	context.Req.RequestURI = inputURL | ||||
| 
 | ||||
| 	if inputURL != context.URI() { | ||||
| 		t.Errorf("Expected url %s, found %s", inputURL, context.URI()) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestHost(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		input        string | ||||
| 		expectedHost string | ||||
| 		shouldErr    bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			input:        "localhost:123", | ||||
| 			expectedHost: "localhost", | ||||
| 			shouldErr:    false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			input:        "localhost", | ||||
| 			expectedHost: "localhost", | ||||
| 			shouldErr:    false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			input:        "[::]", | ||||
| 			expectedHost: "", | ||||
| 			shouldErr:    true, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for _, test := range tests { | ||||
| 		testHostOrPort(t, true, test.input, test.expectedHost, test.shouldErr) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestPort(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		input        string | ||||
| 		expectedPort string | ||||
| 		shouldErr    bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			input:        "localhost:123", | ||||
| 			expectedPort: "123", | ||||
| 			shouldErr:    false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			input:        "localhost", | ||||
| 			expectedPort: "", | ||||
| 			shouldErr:    true, // missing port in address | ||||
| 		}, | ||||
| 		{ | ||||
| 			input:        ":8080", | ||||
| 			expectedPort: "8080", | ||||
| 			shouldErr:    false, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for _, test := range tests { | ||||
| 		testHostOrPort(t, false, test.input, test.expectedPort, test.shouldErr) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func testHostOrPort(t *testing.T, isTestingHost bool, input, expectedResult string, shouldErr bool) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	context.Req.Host = input | ||||
| 	var actualResult, testedObject string | ||||
| 	var err error | ||||
| 
 | ||||
| 	if isTestingHost { | ||||
| 		actualResult, err = context.Host() | ||||
| 		testedObject = "host" | ||||
| 	} else { | ||||
| 		actualResult, err = context.Port() | ||||
| 		testedObject = "port" | ||||
| 	} | ||||
| 
 | ||||
| 	if shouldErr && err == nil { | ||||
| 		t.Errorf("Expected error, found nil!") | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if !shouldErr && err != nil { | ||||
| 		t.Errorf("Expected no error, found %s", err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if actualResult != expectedResult { | ||||
| 		t.Errorf("Expected %s %s, found %s", testedObject, expectedResult, actualResult) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestMethod(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	method := "POST" | ||||
| 	context.Req.Method = method | ||||
| 
 | ||||
| 	if method != context.Method() { | ||||
| 		t.Errorf("Expected method %s, found %s", method, context.Method()) | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| func TestPathMatches(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		urlStr      string | ||||
| 		pattern     string | ||||
| 		shouldMatch bool | ||||
| 	}{ | ||||
| 		// Test 0 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost/", | ||||
| 			pattern:     "", | ||||
| 			shouldMatch: true, | ||||
| 		}, | ||||
| 		// Test 1 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost", | ||||
| 			pattern:     "", | ||||
| 			shouldMatch: true, | ||||
| 		}, | ||||
| 		// Test 1 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost/", | ||||
| 			pattern:     "/", | ||||
| 			shouldMatch: true, | ||||
| 		}, | ||||
| 		// Test 3 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost/?param=val", | ||||
| 			pattern:     "/", | ||||
| 			shouldMatch: true, | ||||
| 		}, | ||||
| 		// Test 4 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost/dir1/dir2", | ||||
| 			pattern:     "/dir2", | ||||
| 			shouldMatch: false, | ||||
| 		}, | ||||
| 		// Test 5 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost/dir1/dir2", | ||||
| 			pattern:     "/dir1", | ||||
| 			shouldMatch: true, | ||||
| 		}, | ||||
| 		// Test 6 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost:444/dir1/dir2", | ||||
| 			pattern:     "/dir1", | ||||
| 			shouldMatch: true, | ||||
| 		}, | ||||
| 		// Test 7 | ||||
| 		{ | ||||
| 			urlStr:      "http://localhost/dir1/dir2", | ||||
| 			pattern:     "*/dir2", | ||||
| 			shouldMatch: false, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		testPrefix := getTestPrefix(i) | ||||
| 		var err error | ||||
| 		context.Req.URL, err = url.Parse(test.urlStr) | ||||
| 		if err != nil { | ||||
| 			t.Fatalf("Failed to prepare test URL from string %s! Error was: %s", test.urlStr, err) | ||||
| 		} | ||||
| 
 | ||||
| 		matches := context.PathMatches(test.pattern) | ||||
| 		if matches != test.shouldMatch { | ||||
| 			t.Errorf(testPrefix+"Expected and actual result differ: expected to match [%t], actual matches [%t]", test.shouldMatch, matches) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestTruncate(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 	tests := []struct { | ||||
| 		inputString string | ||||
| 		inputLength int | ||||
| 		expected    string | ||||
| 	}{ | ||||
| 		// Test 0 - small length | ||||
| 		{ | ||||
| 			inputString: "string", | ||||
| 			inputLength: 1, | ||||
| 			expected:    "s", | ||||
| 		}, | ||||
| 		// Test 1 - exact length | ||||
| 		{ | ||||
| 			inputString: "string", | ||||
| 			inputLength: 6, | ||||
| 			expected:    "string", | ||||
| 		}, | ||||
| 		// Test 2 - bigger length | ||||
| 		{ | ||||
| 			inputString: "string", | ||||
| 			inputLength: 10, | ||||
| 			expected:    "string", | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		actual := context.Truncate(test.inputString, test.inputLength) | ||||
| 		if actual != test.expected { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected %s, found %s. Input was Truncate(%q, %d)", test.expected, actual, test.inputString, test.inputLength) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestStripHTML(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 	tests := []struct { | ||||
| 		input    string | ||||
| 		expected string | ||||
| 	}{ | ||||
| 		// Test 0 - no tags | ||||
| 		{ | ||||
| 			input:    `h1`, | ||||
| 			expected: `h1`, | ||||
| 		}, | ||||
| 		// Test 1 - happy path | ||||
| 		{ | ||||
| 			input:    `<h1>h1</h1>`, | ||||
| 			expected: `h1`, | ||||
| 		}, | ||||
| 		// Test 2 - tag in quotes | ||||
| 		{ | ||||
| 			input:    `<h1">">h1</h1>`, | ||||
| 			expected: `h1`, | ||||
| 		}, | ||||
| 		// Test 3 - multiple tags | ||||
| 		{ | ||||
| 			input:    `<h1><b>h1</b></h1>`, | ||||
| 			expected: `h1`, | ||||
| 		}, | ||||
| 		// Test 4 - tags not closed | ||||
| 		{ | ||||
| 			input:    `<h1`, | ||||
| 			expected: `<h1`, | ||||
| 		}, | ||||
| 		// Test 5 - false start | ||||
| 		{ | ||||
| 			input:    `<h1<b>hi`, | ||||
| 			expected: `<h1hi`, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		actual := context.StripHTML(test.input) | ||||
| 		if actual != test.expected { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected %s, found %s. Input was StripHTML(%s)", test.expected, actual, test.input) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestStripExt(t *testing.T) { | ||||
| 	context := getContextOrFail(t) | ||||
| 	tests := []struct { | ||||
| 		input    string | ||||
| 		expected string | ||||
| 	}{ | ||||
| 		// Test 0 - empty input | ||||
| 		{ | ||||
| 			input:    "", | ||||
| 			expected: "", | ||||
| 		}, | ||||
| 		// Test 1 - relative file with ext | ||||
| 		{ | ||||
| 			input:    "file.ext", | ||||
| 			expected: "file", | ||||
| 		}, | ||||
| 		// Test 2 - relative file without ext | ||||
| 		{ | ||||
| 			input:    "file", | ||||
| 			expected: "file", | ||||
| 		}, | ||||
| 		// Test 3 - absolute file without ext | ||||
| 		{ | ||||
| 			input:    "/file", | ||||
| 			expected: "/file", | ||||
| 		}, | ||||
| 		// Test 4 - absolute file with ext | ||||
| 		{ | ||||
| 			input:    "/file.ext", | ||||
| 			expected: "/file", | ||||
| 		}, | ||||
| 		// Test 5 - with ext but ends with / | ||||
| 		{ | ||||
| 			input:    "/dir.ext/", | ||||
| 			expected: "/dir.ext/", | ||||
| 		}, | ||||
| 		// Test 6 - file with ext under dir with ext | ||||
| 		{ | ||||
| 			input:    "/dir.ext/file.ext", | ||||
| 			expected: "/dir.ext/file", | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		actual := context.StripExt(test.input) | ||||
| 		if actual != test.expected { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected %s, found %s. Input was StripExt(%q)", test.expected, actual, test.input) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func initTestContext() (Context, error) { | ||||
| 	body := bytes.NewBufferString("request body") | ||||
| 	request, err := http.NewRequest("GET", "https://localhost", body) | ||||
| 	if err != nil { | ||||
| 		return Context{}, err | ||||
| 	} | ||||
| 
 | ||||
| 	return Context{Root: http.Dir(os.TempDir()), Req: request}, nil | ||||
| } | ||||
| 
 | ||||
| func getContextOrFail(t *testing.T) Context { | ||||
| 	context, err := initTestContext() | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to prepare test context") | ||||
| 	} | ||||
| 	return context | ||||
| } | ||||
| 
 | ||||
| func getTestPrefix(testN int) string { | ||||
| 	return fmt.Sprintf("Test [%d]: ", testN) | ||||
| } | ||||
| @ -4,6 +4,7 @@ | ||||
| package fastcgi | ||||
| 
 | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| @ -46,10 +47,21 @@ func (h Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) | ||||
| 		fpath := r.URL.Path | ||||
| 		if idx, ok := middleware.IndexFile(h.FileSys, fpath, rule.IndexFiles); ok { | ||||
| 			fpath = idx | ||||
| 			// Index file present. | ||||
| 			// If request path cannot be split, return error. | ||||
| 			if !h.canSplit(fpath, rule) { | ||||
| 				return http.StatusInternalServerError, ErrIndexMissingSplit | ||||
| 			} | ||||
| 		} else { | ||||
| 			// No index file present. | ||||
| 			// If request path cannot be split, ignore request. | ||||
| 			if !h.canSplit(fpath, rule) { | ||||
| 				continue | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		// These criteria work well in this order for PHP sites | ||||
| 		if fpath[len(fpath)-1] == '/' || strings.HasSuffix(fpath, rule.Ext) || !h.exists(fpath) { | ||||
| 		if !h.exists(fpath) || fpath[len(fpath)-1] == '/' || strings.HasSuffix(fpath, rule.Ext) { | ||||
| 
 | ||||
| 			// Create environment for CGI script | ||||
| 			env, err := h.buildEnv(r, rule, fpath) | ||||
| @ -137,6 +149,10 @@ func (h Handler) exists(path string) bool { | ||||
| 	return false | ||||
| } | ||||
| 
 | ||||
| func (h Handler) canSplit(path string, rule Rule) bool { | ||||
| 	return strings.Contains(path, rule.SplitPath) | ||||
| } | ||||
| 
 | ||||
| // buildEnv returns a set of CGI environment variables for the request. | ||||
| func (h Handler) buildEnv(r *http.Request, rule Rule, fpath string) (map[string]string, error) { | ||||
| 	var env map[string]string | ||||
| @ -153,22 +169,15 @@ func (h Handler) buildEnv(r *http.Request, rule Rule, fpath string) (map[string] | ||||
| 		ip = r.RemoteAddr | ||||
| 	} | ||||
| 
 | ||||
| 	// Split path in preparation for env variables | ||||
| 	// Split path in preparation for env variables. | ||||
| 	// Previous h.canSplit checks ensure this can never be -1. | ||||
| 	splitPos := strings.Index(fpath, rule.SplitPath) | ||||
| 	var docURI, scriptName, scriptFilename, pathInfo string | ||||
| 	if splitPos == -1 { | ||||
| 		// Request doesn't have the extension, so assume index file in root | ||||
| 		docURI = "/" + rule.IndexFiles[0] | ||||
| 		scriptName = "/" + rule.IndexFiles[0] | ||||
| 		scriptFilename = filepath.Join(h.AbsRoot, rule.IndexFiles[0]) | ||||
| 		pathInfo = fpath | ||||
| 	} else { | ||||
| 
 | ||||
| 	// Request has the extension; path was split successfully | ||||
| 		docURI = fpath[:splitPos+len(rule.SplitPath)] | ||||
| 		pathInfo = fpath[splitPos+len(rule.SplitPath):] | ||||
| 		scriptName = fpath | ||||
| 		scriptFilename = absPath | ||||
| 	} | ||||
| 	docURI := fpath[:splitPos+len(rule.SplitPath)] | ||||
| 	pathInfo := fpath[splitPos+len(rule.SplitPath):] | ||||
| 	scriptName := fpath | ||||
| 	scriptFilename := absPath | ||||
| 
 | ||||
| 	// Strip PATH_INFO from SCRIPT_NAME | ||||
| 	scriptName = strings.TrimSuffix(scriptName, pathInfo) | ||||
| @ -267,4 +276,8 @@ type Rule struct { | ||||
| 	EnvVars [][2]string | ||||
| } | ||||
| 
 | ||||
| var headerNameReplacer = strings.NewReplacer(" ", "_", "-", "_") | ||||
| var ( | ||||
| 	headerNameReplacer = strings.NewReplacer(" ", "_", "-", "_") | ||||
| 
 | ||||
| 	ErrIndexMissingSplit = errors.New("configured index file(s) must include split value") | ||||
| ) | ||||
|  | ||||
| @ -62,8 +62,8 @@ func (fh *fileHandler) serveFile(w http.ResponseWriter, r *http.Request, name st | ||||
| 	} | ||||
| 	defer f.Close() | ||||
| 
 | ||||
| 	d, err1 := f.Stat() | ||||
| 	if err1 != nil { | ||||
| 	d, err := f.Stat() | ||||
| 	if err != nil { | ||||
| 		if os.IsNotExist(err) { | ||||
| 			return http.StatusNotFound, nil | ||||
| 		} else if os.IsPermission(err) { | ||||
|  | ||||
							
								
								
									
										325
									
								
								middleware/fileserver_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										325
									
								
								middleware/fileserver_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,325 @@ | ||||
| package middleware | ||||
| 
 | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"net/http" | ||||
| 	"net/http/httptest" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| ) | ||||
| 
 | ||||
| var testDir = filepath.Join(os.TempDir(), "caddy_testdir") | ||||
| var customErr = errors.New("Custom Error") | ||||
| 
 | ||||
| // testFiles is a map with relative paths to test files as keys and file content as values. | ||||
| // The map represents the following structure: | ||||
| // - $TEMP/caddy_testdir/ | ||||
| // '-- file1.html | ||||
| // '-- dirwithindex/ | ||||
| // '---- index.html | ||||
| // '-- dir/ | ||||
| // '---- file2.html | ||||
| // '---- hidden.html | ||||
| var testFiles = map[string]string{ | ||||
| 	"file1.html":                                "<h1>file1.html</h1>", | ||||
| 	filepath.Join("dirwithindex", "index.html"): "<h1>dirwithindex/index.html</h1>", | ||||
| 	filepath.Join("dir", "file2.html"):          "<h1>dir/file2.html</h1>", | ||||
| 	filepath.Join("dir", "hidden.html"):         "<h1>dir/hidden.html</h1>", | ||||
| } | ||||
| 
 | ||||
| // TestServeHTTP covers positive scenarios when serving files. | ||||
| func TestServeHTTP(t *testing.T) { | ||||
| 
 | ||||
| 	beforeServeHttpTest(t) | ||||
| 	defer afterServeHttpTest(t) | ||||
| 
 | ||||
| 	fileserver := FileServer(http.Dir(testDir), []string{"hidden.html"}) | ||||
| 
 | ||||
| 	movedPermanently := "Moved Permanently" | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		url string | ||||
| 
 | ||||
| 		expectedStatus      int | ||||
| 		expectedBodyContent string | ||||
| 	}{ | ||||
| 		// Test 0 - access withoutt any path | ||||
| 		{ | ||||
| 			url:            "https://foo", | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 		}, | ||||
| 		// Test 1 - access root (without index.html) | ||||
| 		{ | ||||
| 			url:            "https://foo/", | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 		}, | ||||
| 		// Test 2 - access existing file | ||||
| 		{ | ||||
| 			url:                 "https://foo/file1.html", | ||||
| 			expectedStatus:      http.StatusOK, | ||||
| 			expectedBodyContent: testFiles["file1.html"], | ||||
| 		}, | ||||
| 		// Test 3 - access folder with index file with trailing slash | ||||
| 		{ | ||||
| 			url:                 "https://foo/dirwithindex/", | ||||
| 			expectedStatus:      http.StatusOK, | ||||
| 			expectedBodyContent: testFiles[filepath.Join("dirwithindex", "index.html")], | ||||
| 		}, | ||||
| 		// Test 4 - access folder with index file without trailing slash | ||||
| 		{ | ||||
| 			url:                 "https://foo/dirwithindex", | ||||
| 			expectedStatus:      http.StatusMovedPermanently, | ||||
| 			expectedBodyContent: movedPermanently, | ||||
| 		}, | ||||
| 		// Test 5 - access folder without index file | ||||
| 		{ | ||||
| 			url:            "https://foo/dir/", | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 		}, | ||||
| 		// Test 6 - access folder withtout trailing slash | ||||
| 		{ | ||||
| 			url:                 "https://foo/dir", | ||||
| 			expectedStatus:      http.StatusMovedPermanently, | ||||
| 			expectedBodyContent: movedPermanently, | ||||
| 		}, | ||||
| 		// Test 6 - access file with trailing slash | ||||
| 		{ | ||||
| 			url:                 "https://foo/file1.html/", | ||||
| 			expectedStatus:      http.StatusMovedPermanently, | ||||
| 			expectedBodyContent: movedPermanently, | ||||
| 		}, | ||||
| 		// Test 7 - access not existing path | ||||
| 		{ | ||||
| 			url:            "https://foo/not_existing", | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 		}, | ||||
| 		// Test 8 - access a file, marked as hidden | ||||
| 		{ | ||||
| 			url:            "https://foo/dir/hidden.html", | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 		}, | ||||
| 		// Test 9 - access a index file directly | ||||
| 		{ | ||||
| 			url:                 "https://foo/dirwithindex/index.html", | ||||
| 			expectedStatus:      http.StatusOK, | ||||
| 			expectedBodyContent: testFiles[filepath.Join("dirwithindex", "index.html")], | ||||
| 		}, | ||||
| 		// Test 10 - send a request with query params | ||||
| 		{ | ||||
| 			url:                 "https://foo/dir?param1=val", | ||||
| 			expectedStatus:      http.StatusMovedPermanently, | ||||
| 			expectedBodyContent: movedPermanently, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		responseRecorder := httptest.NewRecorder() | ||||
| 		request, err := http.NewRequest("GET", test.url, strings.NewReader("")) | ||||
| 		status, err := fileserver.ServeHTTP(responseRecorder, request) | ||||
| 
 | ||||
| 		// check if error matches expectations | ||||
| 		if err != nil { | ||||
| 			t.Errorf(getTestPrefix(i)+"Serving file at %s failed. Error was: %v", test.url, err) | ||||
| 		} | ||||
| 
 | ||||
| 		// check status code | ||||
| 		if test.expectedStatus != status { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected status %d, found %d", test.expectedStatus, status) | ||||
| 		} | ||||
| 
 | ||||
| 		// check body content | ||||
| 		if !strings.Contains(responseRecorder.Body.String(), test.expectedBodyContent) { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected body to contain %q, found %q", test.expectedBodyContent, responseRecorder.Body.String()) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| // beforeServeHttpTest creates a test directory with the structure, defined in the variable testFiles | ||||
| func beforeServeHttpTest(t *testing.T) { | ||||
| 	// make the root test dir | ||||
| 	err := os.Mkdir(testDir, os.ModePerm) | ||||
| 	if err != nil { | ||||
| 		if !os.IsExist(err) { | ||||
| 			t.Fatalf("Failed to create test dir. Error was: %v", err) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	for relFile, fileContent := range testFiles { | ||||
| 		absFile := filepath.Join(testDir, relFile) | ||||
| 
 | ||||
| 		// make sure the parent directories exist | ||||
| 		parentDir := filepath.Dir(absFile) | ||||
| 		_, err = os.Stat(parentDir) | ||||
| 		if err != nil { | ||||
| 			os.MkdirAll(parentDir, os.ModePerm) | ||||
| 		} | ||||
| 
 | ||||
| 		// now create the test files | ||||
| 		f, err := os.Create(absFile) | ||||
| 		if err != nil { | ||||
| 			t.Fatalf("Failed to create test file %s. Error was: %v", absFile, err) | ||||
| 			return | ||||
| 		} | ||||
| 
 | ||||
| 		// and fill them with content | ||||
| 		_, err = f.WriteString(fileContent) | ||||
| 		if err != nil { | ||||
| 			t.Fatalf("Failed to write to %s. Error was: %v", absFile, err) | ||||
| 			return | ||||
| 		} | ||||
| 		f.Close() | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| // afterServeHttpTest removes the test dir and all its content | ||||
| func afterServeHttpTest(t *testing.T) { | ||||
| 	// cleans up everything under the test dir. No need to clean the individual files. | ||||
| 	err := os.RemoveAll(testDir) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to clean up test dir %s. Error was: %v", testDir, err) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // failingFS implements the http.FileSystem interface. The Open method always returns the error, assigned to err | ||||
| type failingFS struct { | ||||
| 	err      error     // the error to return when Open is called | ||||
| 	fileImpl http.File // inject the file implementation | ||||
| } | ||||
| 
 | ||||
| // Open returns the assigned failingFile and error | ||||
| func (f failingFS) Open(path string) (http.File, error) { | ||||
| 	return f.fileImpl, f.err | ||||
| } | ||||
| 
 | ||||
| // failingFile implements http.File but returns a predefined error on every Stat() method call. | ||||
| type failingFile struct { | ||||
| 	http.File | ||||
| 	err error | ||||
| } | ||||
| 
 | ||||
| // Stat returns nil FileInfo and the provided error on every call | ||||
| func (ff failingFile) Stat() (os.FileInfo, error) { | ||||
| 	return nil, ff.err | ||||
| } | ||||
| 
 | ||||
| // Close is noop and returns no error | ||||
| func (ff failingFile) Close() error { | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // TestServeHTTPFailingFS tests error cases where the Open function fails with various errors. | ||||
| func TestServeHTTPFailingFS(t *testing.T) { | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		fsErr           error | ||||
| 		expectedStatus  int | ||||
| 		expectedErr     error | ||||
| 		expectedHeaders map[string]string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			fsErr:          os.ErrNotExist, | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 			expectedErr:    nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			fsErr:          os.ErrPermission, | ||||
| 			expectedStatus: http.StatusForbidden, | ||||
| 			expectedErr:    os.ErrPermission, | ||||
| 		}, | ||||
| 		{ | ||||
| 			fsErr:           customErr, | ||||
| 			expectedStatus:  http.StatusServiceUnavailable, | ||||
| 			expectedErr:     customErr, | ||||
| 			expectedHeaders: map[string]string{"Retry-After": "5"}, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		// initialize a file server with the failing FileSystem | ||||
| 		fileserver := FileServer(failingFS{err: test.fsErr}, nil) | ||||
| 
 | ||||
| 		// prepare the request and response | ||||
| 		request, err := http.NewRequest("GET", "https://foo/", nil) | ||||
| 		if err != nil { | ||||
| 			t.Fatalf("Failed to build request. Error was: %v", err) | ||||
| 		} | ||||
| 		responseRecorder := httptest.NewRecorder() | ||||
| 
 | ||||
| 		status, actualErr := fileserver.ServeHTTP(responseRecorder, request) | ||||
| 
 | ||||
| 		// check the status | ||||
| 		if status != test.expectedStatus { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected status %d, found %d", test.expectedStatus, status) | ||||
| 		} | ||||
| 
 | ||||
| 		// check the error | ||||
| 		if actualErr != test.expectedErr { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected err %v, found %v", test.expectedErr, actualErr) | ||||
| 		} | ||||
| 
 | ||||
| 		// check the headers - a special case for server under load | ||||
| 		if test.expectedHeaders != nil && len(test.expectedHeaders) > 0 { | ||||
| 			for expectedKey, expectedVal := range test.expectedHeaders { | ||||
| 				actualVal := responseRecorder.Header().Get(expectedKey) | ||||
| 				if expectedVal != actualVal { | ||||
| 					t.Errorf(getTestPrefix(i)+"Expected header %s: %s, found %s", expectedKey, expectedVal, actualVal) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // TestServeHTTPFailingStat tests error cases where the initial Open function succeeds, but the Stat method on the opened file fails. | ||||
| func TestServeHTTPFailingStat(t *testing.T) { | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		statErr        error | ||||
| 		expectedStatus int | ||||
| 		expectedErr    error | ||||
| 	}{ | ||||
| 		{ | ||||
| 			statErr:        os.ErrNotExist, | ||||
| 			expectedStatus: http.StatusNotFound, | ||||
| 			expectedErr:    nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			statErr:        os.ErrPermission, | ||||
| 			expectedStatus: http.StatusForbidden, | ||||
| 			expectedErr:    os.ErrPermission, | ||||
| 		}, | ||||
| 		{ | ||||
| 			statErr:        customErr, | ||||
| 			expectedStatus: http.StatusInternalServerError, | ||||
| 			expectedErr:    customErr, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		// initialize a file server. The FileSystem will not fail, but calls to the Stat method of the returned File object will | ||||
| 		fileserver := FileServer(failingFS{err: nil, fileImpl: failingFile{err: test.statErr}}, nil) | ||||
| 
 | ||||
| 		// prepare the request and response | ||||
| 		request, err := http.NewRequest("GET", "https://foo/", nil) | ||||
| 		if err != nil { | ||||
| 			t.Fatalf("Failed to build request. Error was: %v", err) | ||||
| 		} | ||||
| 		responseRecorder := httptest.NewRecorder() | ||||
| 
 | ||||
| 		status, actualErr := fileserver.ServeHTTP(responseRecorder, request) | ||||
| 
 | ||||
| 		// check the status | ||||
| 		if status != test.expectedStatus { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected status %d, found %d", test.expectedStatus, status) | ||||
| 		} | ||||
| 
 | ||||
| 		// check the error | ||||
| 		if actualErr != test.expectedErr { | ||||
| 			t.Errorf(getTestPrefix(i)+"Expected err %v, found %v", test.expectedErr, actualErr) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @ -17,6 +17,8 @@ import ( | ||||
| // It only generates static files if it is enabled (cfg.StaticDir | ||||
| // must be set). | ||||
| func GenerateStatic(md Markdown, cfg *Config) error { | ||||
| 	// If static site generation is enabled. | ||||
| 	if cfg.StaticDir != "" { | ||||
| 		generated, err := generateLinks(md, cfg) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| @ -27,8 +29,6 @@ func GenerateStatic(md Markdown, cfg *Config) error { | ||||
| 			return nil | ||||
| 		} | ||||
| 
 | ||||
| 	// If static site generation is enabled. | ||||
| 	if cfg.StaticDir != "" { | ||||
| 		if err := generateStaticHTML(md, cfg); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| @ -136,6 +136,7 @@ func (md Markdown) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error | ||||
| 						// generation, serve the static page | ||||
| 						if fs.ModTime().Before(fs1.ModTime()) { | ||||
| 							if html, err := ioutil.ReadFile(filepath); err == nil { | ||||
| 								middleware.SetLastModifiedHeader(w, fs1.ModTime()) | ||||
| 								w.Write(html) | ||||
| 								return http.StatusOK, nil | ||||
| 							} | ||||
| @ -162,6 +163,7 @@ func (md Markdown) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error | ||||
| 					return http.StatusInternalServerError, err | ||||
| 				} | ||||
| 
 | ||||
| 				middleware.SetLastModifiedHeader(w, fs.ModTime()) | ||||
| 				w.Write(html) | ||||
| 				return http.StatusOK, nil | ||||
| 			} | ||||
|  | ||||
| @ -92,7 +92,7 @@ func TestMarkdown(t *testing.T) { | ||||
| 	expectedBody := `<!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
| <title>Markdown test</title> | ||||
| <title>Markdown test 1</title> | ||||
| </head> | ||||
| <body> | ||||
| <h1>Header</h1> | ||||
| @ -102,11 +102,10 @@ Welcome to A Caddy website! | ||||
| 
 | ||||
| <p>Body</p> | ||||
| 
 | ||||
| <p><code>go | ||||
| func getTrue() bool { | ||||
| <pre><code class="language-go">func getTrue() bool { | ||||
|     return true | ||||
| } | ||||
| </code></p> | ||||
| </code></pre> | ||||
| 
 | ||||
| </body> | ||||
| </html> | ||||
| @ -129,7 +128,7 @@ func getTrue() bool { | ||||
| 	expectedBody = `<!DOCTYPE html> | ||||
| <html> | ||||
| 	<head> | ||||
| 		<title>Markdown test</title> | ||||
| 		<title>Markdown test 2</title> | ||||
| 		<meta charset="utf-8"> | ||||
| 		<link rel="stylesheet" href="/resources/css/log.css"> | ||||
| <link rel="stylesheet" href="/resources/css/default.css"> | ||||
| @ -143,11 +142,10 @@ func getTrue() bool { | ||||
| 
 | ||||
| <p>Body</p> | ||||
| 
 | ||||
| <p><code>go | ||||
| func getTrue() bool { | ||||
| <pre><code class="language-go">func getTrue() bool { | ||||
|     return true | ||||
| } | ||||
| </code></p> | ||||
| </code></pre> | ||||
| 
 | ||||
| 	</body> | ||||
| </html>` | ||||
|  | ||||
| @ -65,7 +65,8 @@ func (md Markdown) Process(c *Config, requestPath string, b []byte, ctx middlewa | ||||
| 	} | ||||
| 
 | ||||
| 	// process markdown | ||||
| 	markdown = blackfriday.Markdown(markdown, c.Renderer, 0) | ||||
| 	extns := blackfriday.EXTENSION_TABLES | blackfriday.EXTENSION_FENCED_CODE | blackfriday.EXTENSION_STRIKETHROUGH | ||||
| 	markdown = blackfriday.Markdown(markdown, c.Renderer, extns) | ||||
| 
 | ||||
| 	// set it as body for template | ||||
| 	metadata.Variables["body"] = string(markdown) | ||||
|  | ||||
							
								
								
									
										2
									
								
								middleware/markdown/testdata/blog/test.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								middleware/markdown/testdata/blog/test.md
									
									
									
									
										vendored
									
									
								
							| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: Markdown test | ||||
| title: Markdown test 1 | ||||
| sitename: A Caddy website | ||||
| --- | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										2
									
								
								middleware/markdown/testdata/log/test.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								middleware/markdown/testdata/log/test.md
									
									
									
									
										vendored
									
									
								
							| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: Markdown test | ||||
| title: Markdown test 2 | ||||
| sitename: A Caddy website | ||||
| --- | ||||
| 
 | ||||
|  | ||||
| @ -4,6 +4,7 @@ package middleware | ||||
| import ( | ||||
| 	"net/http" | ||||
| 	"path" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| type ( | ||||
| @ -78,3 +79,30 @@ func IndexFile(root http.FileSystem, fpath string, indexFiles []string) (string, | ||||
| 	} | ||||
| 	return "", false | ||||
| } | ||||
| 
 | ||||
| // SetLastModifiedHeader checks if the provided modTime is valid and if it is sets it | ||||
| // as a Last-Modified header to the ResponseWriter. If the modTime is in the future | ||||
| // the current time is used instead. | ||||
| func SetLastModifiedHeader(w http.ResponseWriter, modTime time.Time) { | ||||
| 	if modTime.IsZero() || modTime.Equal(time.Unix(0, 0)) { | ||||
| 		// the time does not appear to be valid. Don't put it in the response | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// RFC 2616 - Section 14.29 - Last-Modified: | ||||
| 	// An origin server MUST NOT send a Last-Modified date which is later than the | ||||
| 	// server's time of message origination. In such cases, where the resource's last | ||||
| 	// modification would indicate some time in the future, the server MUST replace | ||||
| 	// that date with the message origination date. | ||||
| 	now := currentTime() | ||||
| 	if modTime.After(now) { | ||||
| 		modTime = now | ||||
| 	} | ||||
| 
 | ||||
| 	w.Header().Set("Last-Modified", modTime.UTC().Format(http.TimeFormat)) | ||||
| } | ||||
| 
 | ||||
| // currentTime returns time.Now() everytime it's called. It's used for mocking in tests. | ||||
| var currentTime = func() time.Time { | ||||
| 	return time.Now() | ||||
| } | ||||
|  | ||||
| @ -1,8 +1,11 @@ | ||||
| package middleware | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"net/http/httptest" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| func TestIndexfile(t *testing.T) { | ||||
| @ -42,3 +45,64 @@ func TestIndexfile(t *testing.T) { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestSetLastModified(t *testing.T) { | ||||
| 	nowTime := time.Now() | ||||
| 
 | ||||
| 	// ovewrite the function to return reliable time | ||||
| 	originalGetCurrentTimeFunc := currentTime | ||||
| 	currentTime = func() time.Time { | ||||
| 		return nowTime | ||||
| 	} | ||||
| 	defer func() { | ||||
| 		currentTime = originalGetCurrentTimeFunc | ||||
| 	}() | ||||
| 
 | ||||
| 	pastTime := nowTime.Truncate(1 * time.Hour) | ||||
| 	futureTime := nowTime.Add(1 * time.Hour) | ||||
| 
 | ||||
| 	tests := []struct { | ||||
| 		inputModTime         time.Time | ||||
| 		expectedIsHeaderSet  bool | ||||
| 		expectedLastModified string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			inputModTime:         pastTime, | ||||
| 			expectedIsHeaderSet:  true, | ||||
| 			expectedLastModified: pastTime.UTC().Format(http.TimeFormat), | ||||
| 		}, | ||||
| 		{ | ||||
| 			inputModTime:         nowTime, | ||||
| 			expectedIsHeaderSet:  true, | ||||
| 			expectedLastModified: nowTime.UTC().Format(http.TimeFormat), | ||||
| 		}, | ||||
| 		{ | ||||
| 			inputModTime:         futureTime, | ||||
| 			expectedIsHeaderSet:  true, | ||||
| 			expectedLastModified: nowTime.UTC().Format(http.TimeFormat), | ||||
| 		}, | ||||
| 		{ | ||||
| 			inputModTime:        time.Time{}, | ||||
| 			expectedIsHeaderSet: false, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for i, test := range tests { | ||||
| 		responseRecorder := httptest.NewRecorder() | ||||
| 		errorPrefix := fmt.Sprintf("Test [%d]: ", i) | ||||
| 		SetLastModifiedHeader(responseRecorder, test.inputModTime) | ||||
| 		actualLastModifiedHeader := responseRecorder.Header().Get("Last-Modified") | ||||
| 
 | ||||
| 		if test.expectedIsHeaderSet && actualLastModifiedHeader == "" { | ||||
| 			t.Fatalf(errorPrefix + "Expected to find Last-Modified header, but found nothing") | ||||
| 		} | ||||
| 
 | ||||
| 		if !test.expectedIsHeaderSet && actualLastModifiedHeader != "" { | ||||
| 			t.Fatalf(errorPrefix+"Did not expect to find Last-Modified header, but found one [%s].", actualLastModifiedHeader) | ||||
| 		} | ||||
| 
 | ||||
| 		if test.expectedLastModified != actualLastModifiedHeader { | ||||
| 			t.Errorf(errorPrefix+"Expected Last-Modified content [%s], found [%s}", test.expectedLastModified, actualLastModifiedHeader) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -33,8 +33,18 @@ func (t Templates) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error | ||||
| 				// Create execution context | ||||
| 				ctx := middleware.Context{Root: t.FileSys, Req: r, URL: r.URL} | ||||
| 
 | ||||
| 				// New template | ||||
| 				templateName := filepath.Base(fpath) | ||||
| 				tpl := template.New(templateName) | ||||
| 
 | ||||
| 				// Set delims | ||||
| 				if rule.Delims != [2]string{} { | ||||
| 					tpl.Delims(rule.Delims[0], rule.Delims[1]) | ||||
| 				} | ||||
| 
 | ||||
| 				// Build the template | ||||
| 				tpl, err := template.ParseFiles(filepath.Join(t.Root, fpath)) | ||||
| 				templatePath := filepath.Join(t.Root, fpath) | ||||
| 				tpl, err := tpl.ParseFiles(templatePath) | ||||
| 				if err != nil { | ||||
| 					if os.IsNotExist(err) { | ||||
| 						return http.StatusNotFound, nil | ||||
| @ -50,6 +60,12 @@ func (t Templates) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error | ||||
| 				if err != nil { | ||||
| 					return http.StatusInternalServerError, err | ||||
| 				} | ||||
| 
 | ||||
| 				templateInfo, err := os.Stat(templatePath) | ||||
| 				if err == nil { | ||||
| 					// add the Last-Modified header if we were able to optain the information | ||||
| 					middleware.SetLastModifiedHeader(w, templateInfo.ModTime()) | ||||
| 				} | ||||
| 				buf.WriteTo(w) | ||||
| 
 | ||||
| 				return http.StatusOK, nil | ||||
| @ -75,4 +91,5 @@ type Rule struct { | ||||
| 	Path       string | ||||
| 	Extensions []string | ||||
| 	IndexFiles []string | ||||
| 	Delims     [2]string | ||||
| } | ||||
|  | ||||
| @ -23,6 +23,7 @@ func Test(t *testing.T) { | ||||
| 				Extensions: []string{".html", ".htm"}, | ||||
| 				IndexFiles: []string{"index.html", "index.htm"}, | ||||
| 				Path:       "/images", | ||||
| 				Delims:     [2]string{"{%", "%}"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		Root:    "./testdata", | ||||
| @ -94,6 +95,30 @@ func Test(t *testing.T) { | ||||
| 		t.Fatalf("Test: the expected body %v is different from the response one: %v", expectedBody, respBody) | ||||
| 	} | ||||
| 
 | ||||
| 	/* | ||||
| 	* Test tmpl on /images/img2.htm | ||||
| 	 */ | ||||
| 	req, err = http.NewRequest("GET", "/images/img2.htm", nil) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Could not create HTTP request: %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	rec = httptest.NewRecorder() | ||||
| 
 | ||||
| 	tmpl.ServeHTTP(rec, req) | ||||
| 
 | ||||
| 	if rec.Code != http.StatusOK { | ||||
| 		t.Fatalf("Test: Wrong response code: %d, should be %d", rec.Code, http.StatusOK) | ||||
| 	} | ||||
| 
 | ||||
| 	respBody = rec.Body.String() | ||||
| 	expectedBody = `<!DOCTYPE html><html><head><title>img</title></head><body>{{.Include "header.html"}}</body></html> | ||||
| ` | ||||
| 
 | ||||
| 	if respBody != expectedBody { | ||||
| 		t.Fatalf("Test: the expected body %v is different from the response one: %v", expectedBody, respBody) | ||||
| 	} | ||||
| 
 | ||||
| 	/* | ||||
| 	* Test tmplroot on /root.html | ||||
| 	 */ | ||||
|  | ||||
							
								
								
									
										2
									
								
								middleware/templates/testdata/images/img.htm
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								middleware/templates/testdata/images/img.htm
									
									
									
									
										vendored
									
									
								
							| @ -1 +1 @@ | ||||
| <!DOCTYPE html><html><head><title>img</title></head><body>{{.Include "header.html"}}</body></html> | ||||
| <!DOCTYPE html><html><head><title>img</title></head><body>{%.Include "header.html"%}</body></html> | ||||
|  | ||||
							
								
								
									
										1
									
								
								middleware/templates/testdata/images/img2.htm
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								middleware/templates/testdata/images/img2.htm
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| <!DOCTYPE html><html><head><title>img</title></head><body>{{.Include "header.html"}}</body></html> | ||||
| @ -172,7 +172,7 @@ func reader(conn *websocket.Conn, stdout io.ReadCloser, stdin io.WriteCloser) { | ||||
| 	conn.SetReadDeadline(time.Now().Add(pongWait)) | ||||
| 	conn.SetPongHandler(func(string) error { conn.SetReadDeadline(time.Now().Add(pongWait)); return nil }) | ||||
| 	tickerChan := make(chan bool) | ||||
| 	defer func() { tickerChan <- true }() // make sure to close the ticker when we are done. | ||||
| 	defer close(tickerChan) // make sure to close the ticker when we are done. | ||||
| 	go ticker(conn, tickerChan) | ||||
| 
 | ||||
| 	for { | ||||
| @ -213,10 +213,7 @@ func reader(conn *websocket.Conn, stdout io.ReadCloser, stdin io.WriteCloser) { | ||||
| // between the server and client to keep it alive with ping messages. | ||||
| func ticker(conn *websocket.Conn, c chan bool) { | ||||
| 	ticker := time.NewTicker(pingPeriod) | ||||
| 	defer func() { | ||||
| 		ticker.Stop() | ||||
| 		close(c) | ||||
| 	}() | ||||
| 	defer ticker.Stop() | ||||
| 
 | ||||
| 	for { // blocking loop with select to wait for stimulation. | ||||
| 		select { | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user