package fastcgi import ( "net" "net/http" "net/http/fcgi" "net/http/httptest" "net/url" "strconv" "testing" ) func TestServeHTTP(t *testing.T) { body := "This is some test body content" bodyLenStr := strconv.Itoa(len(body)) listener, err := net.Listen("tcp", "127.0.0.1:0") if err != nil { t.Fatalf("Unable to create listener for test: %v", err) } defer listener.Close() go fcgi.Serve(listener, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Length", bodyLenStr) w.Write([]byte(body)) })) handler := Handler{ Next: nil, Rules: []Rule{{Path: "/", Address: listener.Addr().String()}}, } r, err := http.NewRequest("GET", "/", nil) if err != nil { t.Fatalf("Unable to create request: %v", err) } w := httptest.NewRecorder() status, err := handler.ServeHTTP(w, r) if got, want := status, 0; got != want { t.Errorf("Expected returned status code to be %d, got %d", want, got) } if err != nil { t.Errorf("Expected nil error, got: %v", err) } if got, want := w.Header().Get("Content-Length"), bodyLenStr; got != want { t.Errorf("Expected Content-Length to be '%s', got: '%s'", want, got) } if got, want := w.Body.String(), body; got != want { t.Errorf("Expected response body to be '%s', got: '%s'", want, got) } } func TestRuleParseAddress(t *testing.T) { getClientTestTable := []struct { rule *Rule expectednetwork string expectedaddress string }{ {&Rule{Address: "tcp://172.17.0.1:9000"}, "tcp", "172.17.0.1:9000"}, {&Rule{Address: "fastcgi://localhost:9000"}, "tcp", "localhost:9000"}, {&Rule{Address: "172.17.0.15"}, "tcp", "172.17.0.15"}, {&Rule{Address: "/my/unix/socket"}, "unix", "/my/unix/socket"}, {&Rule{Address: "unix:/second/unix/socket"}, "unix", "/second/unix/socket"}, } for _, entry := range getClientTestTable { if actualnetwork, _ := entry.rule.parseAddress(); actualnetwork != entry.expectednetwork { t.Errorf("Unexpected network for address string %v. Got %v, expected %v", entry.rule.Address, actualnetwork, entry.expectednetwork) } if _, actualaddress := entry.rule.parseAddress(); actualaddress != entry.expectedaddress { t.Errorf("Unexpected parsed address for address string %v. Got %v, expected %v", entry.rule.Address, actualaddress, entry.expectedaddress) } } } func TestRuleIgnoredPath(t *testing.T) { rule := &Rule{ Path: "/fastcgi", IgnoredSubPaths: []string{"/download", "/static"}, } tests := []struct { url string expected bool }{ {"/fastcgi", true}, {"/fastcgi/dl", true}, {"/fastcgi/download", false}, {"/fastcgi/download/static", false}, {"/fastcgi/static", false}, {"/fastcgi/static/download", false}, {"/fastcgi/something/download", true}, {"/fastcgi/something/static", true}, {"/fastcgi//static", false}, {"/fastcgi//static//download", false}, {"/fastcgi//download", false}, } for i, test := range tests { allowed := rule.AllowedPath(test.url) if test.expected != allowed { t.Errorf("Test %d: expected %v found %v", i, test.expected, allowed) } } } func TestBuildEnv(t *testing.T) { testBuildEnv := func(r *http.Request, rule Rule, fpath string, envExpected map[string]string) { var h Handler env, err := h.buildEnv(r, rule, fpath) if err != nil { t.Error("Unexpected error:", err.Error()) } for k, v := range envExpected { if env[k] != v { t.Errorf("Unexpected %v. Got %v, expected %v", k, env[k], v) } } } rule := Rule{} url, err := url.Parse("http://localhost:2015/fgci_test.php?test=blabla") if err != nil { t.Error("Unexpected error:", err.Error()) } var newReq = func() *http.Request { return &http.Request{ Method: "GET", URL: url, Proto: "HTTP/1.1", ProtoMajor: 1, ProtoMinor: 1, Host: "localhost:2015", RemoteAddr: "[2b02:1810:4f2d:9400:70ab:f822:be8a:9093]:51688", RequestURI: "/fgci_test.php", } } fpath := "/fgci_test.php" var newEnv = func() map[string]string { return map[string]string{ "REMOTE_ADDR": "2b02:1810:4f2d:9400:70ab:f822:be8a:9093", "REMOTE_PORT": "51688", "SERVER_PROTOCOL": "HTTP/1.1", "QUERY_STRING": "test=blabla", "REQUEST_METHOD": "GET", "HTTP_HOST": "localhost:2015", } } // request var r *http.Request // expected environment variables var envExpected map[string]string // 1. Test for full canonical IPv6 address r = newReq() testBuildEnv(r, rule, fpath, envExpected) // 2. Test for shorthand notation of IPv6 address r = newReq() r.RemoteAddr = "[::1]:51688" envExpected = newEnv() envExpected["REMOTE_ADDR"] = "::1" testBuildEnv(r, rule, fpath, envExpected) // 3. Test for IPv4 address r = newReq() r.RemoteAddr = "192.168.0.10:51688" envExpected = newEnv() envExpected["REMOTE_ADDR"] = "192.168.0.10" testBuildEnv(r, rule, fpath, envExpected) // 4. Test for environment variable r = newReq() rule.EnvVars = [][2]string{ {"HTTP_HOST", "localhost:2016"}, {"REQUEST_METHOD", "POST"}, } envExpected = newEnv() envExpected["HTTP_HOST"] = "localhost:2016" envExpected["REQUEST_METHOD"] = "POST" testBuildEnv(r, rule, fpath, envExpected) // 5. Test for environment variable placeholders r = newReq() rule.EnvVars = [][2]string{ {"HTTP_HOST", "{host}"}, {"CUSTOM_URI", "custom_uri{uri}"}, {"CUSTOM_QUERY", "custom=true&{query}"}, } envExpected = newEnv() envExpected["HTTP_HOST"] = "localhost:2015" envExpected["CUSTOM_URI"] = "custom_uri/fgci_test.php?test=blabla" envExpected["CUSTOM_QUERY"] = "custom=true&test=blabla" testBuildEnv(r, rule, fpath, envExpected) }