// Copyright 2015 Light Code Labs, LLC // // 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 rewrite import ( "context" "fmt" "log" "net/http" "net/http/httptest" "strings" "testing" "github.com/mholt/caddy/caddyhttp/httpserver" ) func TestRewrite(t *testing.T) { rw := Rewrite{ Next: httpserver.HandlerFunc(urlPrinter), Rules: []httpserver.HandlerConfig{ newSimpleRule(t, "^/from$", "/to"), newSimpleRule(t, "^/a$", "/b"), newSimpleRule(t, "^/b$", "/b{uri}"), newSimpleRule(t, "^/simplereggrp/([0-9]+)([a-z]*)$", "/{1}/{2}/{query}"), }, FileSys: http.Dir("."), } regexps := [][]string{ {"/reg/", ".*", "/to", ""}, {"/r/", "[a-z]+", "/toaz", "!.html|"}, {"/path/", "[a-z0-9]", "/to/{path}", ""}, {"/url/", "a([a-z0-9]*)s([A-Z]{2})", "/to/{rewrite_path}", ""}, {"/ab/", "ab", "/ab?{query}", ".txt|"}, {"/ab/", "ab", "/ab?type=html&{query}", ".html|"}, {"/abc/", "ab", "/abc/{file}", ".html|"}, {"/abcd/", "ab", "/a/{dir}/{file}", ".html|"}, {"/abcde/", "ab", "/a#{fragment}", ".html|"}, {"/ab/", `.*\.jpg`, "/ajpg", ""}, {"/reggrp", `/ad/([0-9]+)([a-z]*)`, "/a{1}/{2}", ""}, {"/reg2grp", `(.*)`, "/{1}", ""}, {"/reg3grp", `(.*)/(.*)/(.*)`, "/{1}{2}{3}", ""}, {"/hashtest", "(.*)", "/{1}", ""}, } for _, regexpRule := range regexps { var ext []string if s := strings.Split(regexpRule[3], "|"); len(s) > 1 { ext = s[:len(s)-1] } rule, err := NewComplexRule(regexpRule[0], regexpRule[1], regexpRule[2], ext, httpserver.IfMatcher{}) if err != nil { t.Fatal(err) } rw.Rules = append(rw.Rules, rule) } tests := []struct { from string expectedTo string }{ {"/from", "/to"}, {"/a", "/b"}, {"/b", "/b/b"}, {"/aa", "/aa"}, {"/", "/"}, {"/a?foo=bar", "/b?foo=bar"}, {"/asdf?foo=bar", "/asdf?foo=bar"}, {"/foo#bar", "/foo#bar"}, {"/a#foo", "/b#foo"}, {"/reg/foo", "/to"}, {"/re", "/re"}, {"/r/", "/r/"}, {"/r/123", "/r/123"}, {"/r/a123", "/toaz"}, {"/r/abcz", "/toaz"}, {"/r/z", "/toaz"}, {"/r/z.html", "/r/z.html"}, {"/r/z.js", "/toaz"}, {"/path/a1b2c", "/to/path/a1b2c"}, {"/path/d3e4f", "/to/path/d3e4f"}, {"/url/asAB", "/to/url/asAB"}, {"/url/aBsAB", "/url/aBsAB"}, {"/url/a00sAB", "/to/url/a00sAB"}, {"/url/a0z0sAB", "/to/url/a0z0sAB"}, {"/ab/aa", "/ab/aa"}, {"/ab/ab", "/ab/ab"}, {"/ab/ab.txt", "/ab"}, {"/ab/ab.txt?name=name", "/ab?name=name"}, {"/ab/ab.html?name=name", "/ab?type=html&name=name"}, {"/abc/ab.html", "/abc/ab.html"}, {"/abcd/abcd.html", "/a/abcd/abcd.html"}, {"/abcde/abcde.html", "/a"}, {"/abcde/abcde.html#1234", "/a#1234"}, {"/ab/ab.jpg", "/ajpg"}, {"/reggrp/ad/12", "/a12/"}, {"/reggrp/ad/124a", "/a124/a"}, {"/reggrp/ad/124abc", "/a124/abc"}, {"/reg2grp/ad/124abc", "/ad/124abc"}, {"/reg3grp/ad/aa/66", "/adaa66"}, {"/reg3grp/ad612/n1n/ab", "/ad612n1nab"}, {"/hashtest/a%20%23%20test", "/a%20%23%20test"}, {"/hashtest/a%20%3F%20test", "/a%20%3F%20test"}, {"/hashtest/a%20%3F%23test", "/a%20%3F%23test"}, {"/simplereggrp/123abc?q", "/123/abc/q?q"}, } for i, test := range tests { req, err := http.NewRequest("GET", test.from, nil) if err != nil { t.Fatalf("Test %d: Could not create HTTP request: %v", i, err) } ctx := context.WithValue(req.Context(), httpserver.OriginalURLCtxKey, *req.URL) req = req.WithContext(ctx) rec := httptest.NewRecorder() if _, err := rw.ServeHTTP(rec, req); err != nil { log.Println("[ERROR] failed to serve HTTP: ", err) } if got, want := rec.Body.String(), test.expectedTo; got != want { t.Errorf("Test %d: Expected URL '%s' to be rewritten to '%s' but was rewritten to '%s'", i, test.from, want, got) } } } // TestWordpress is a test for wordpress usecase. func TestWordpress(t *testing.T) { rw := Rewrite{ Next: httpserver.HandlerFunc(urlPrinter), Rules: []httpserver.HandlerConfig{ // both rules are same, thanks to Go regexp (confusion). newSimpleRule(t, "^/wp-admin", "{path} {path}/ /index.php?{query}", true), newSimpleRule(t, "^\\/wp-admin", "{path} {path}/ /index.php?{query}", true), }, FileSys: http.Dir("."), } tests := []struct { from string expectedTo string }{ {"/wp-admin", "/wp-admin"}, {"/wp-admin/login.php", "/wp-admin/login.php"}, {"/not-wp-admin/login.php?not=admin", "/index.php?not=admin"}, {"/loophole", "/index.php"}, {"/user?name=john", "/index.php?name=john"}, } for i, test := range tests { req, err := http.NewRequest("GET", test.from, nil) if err != nil { t.Fatalf("Test %d: Could not create HTTP request: %v", i, err) } ctx := context.WithValue(req.Context(), httpserver.OriginalURLCtxKey, *req.URL) req = req.WithContext(ctx) rec := httptest.NewRecorder() if _, err := rw.ServeHTTP(rec, req); err != nil { log.Println("[ERROR] failed to serve HTTP: ", err) } if got, want := rec.Body.String(), test.expectedTo; got != want { t.Errorf("Test %d: Expected URL to be '%s' but was '%s'", i, want, got) } } } func urlPrinter(w http.ResponseWriter, r *http.Request) (int, error) { _, _ = fmt.Fprint(w, r.URL.String()) return 0, nil }