// black-box testing // // see _examples/routing/main_test.go for the most common router tests that you may want to see, // this is a test for the new feature that I just coded: wildcard "/{p:path}" on root without conflicts package router_test import ( "net/http" "testing" "github.com/kataras/iris/v12" "github.com/kataras/iris/v12/context" "github.com/kataras/iris/v12/httptest" ) const ( same_as_request_path = "same" from_status_code = "from" staticPathPrefixBody = "from the static path: " prefix_static_path_following_by_request_path = "prefix_same" ) type testRouteRequest struct { method string subdomain string path string expectedStatusCode int expectedBody string } type testRoute struct { method string path string handler context.Handler requests []testRouteRequest } var h = func(ctx context.Context) { ctx.WriteString(ctx.Path()) } var h2 = func(ctx context.Context) { ctx.StatusCode(iris.StatusForbidden) // ! 200 but send the body as expected, // we need that kind of behavior to determinate which handler is executed for routes that // both having wildcard path but first one is registered on root level. ctx.WriteString(ctx.Path()) } func h3(ctx context.Context) { ctx.Writef(staticPathPrefixBody + ctx.Path()) } func TestRouterWildcardDifferentPrefixPath(t *testing.T) { tt := []testRoute{ {"GET", "/s/{p:path}", h, []testRouteRequest{ {"GET", "", "/s/that/is/wildcard", iris.StatusOK, same_as_request_path}, {"GET", "", "/s/ok", iris.StatusOK, same_as_request_path}, }}, {"GET", "/som/{p:path}", h, []testRouteRequest{ {"GET", "", "/som/that/is/wildcard", iris.StatusOK, same_as_request_path}, {"GET", "", "/som/ok", iris.StatusOK, same_as_request_path}, }}, {"GET", "/some/{p:path}", h, []testRouteRequest{ {"GET", "", "/some/that/is/wildcard", iris.StatusOK, same_as_request_path}, {"GET", "", "/some1/that/is/wildcard", iris.StatusNotFound, from_status_code}, }}, } testTheRoutes(t, tt, false) } func TestRouterWildcardAndStatic(t *testing.T) { tt := []testRoute{ {"GET", "/some/{p:path}", h2, []testRouteRequest{ {"GET", "", "/some/that/is/wildcard", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/some/did", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/some1/that/is/wildcard", iris.StatusNotFound, from_status_code}, }}, {"GET", "/some/static", h, []testRouteRequest{ {"GET", "", "/some/static", iris.StatusOK, same_as_request_path}, }}, {"GET", "/s/{p:path}", h2, []testRouteRequest{ {"GET", "", "/s/that/is/wildcard", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/s/did", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/s1/that/is/wildcard", iris.StatusNotFound, from_status_code}, }}, {"GET", "/s/static", h, []testRouteRequest{ {"GET", "", "/s/static", iris.StatusOK, same_as_request_path}, }}, } testTheRoutes(t, tt, false) } func TestRouterWildcardRootMany(t *testing.T) { tt := []testRoute{ // all routes will be handlded by "h" because we added wildcard to root, // this feature is very important and can remove noumerous of previous hacks on our apps. {"GET", "/{p:path}", h, []testRouteRequest{ {"GET", "", "/this/is/wildcard/on/root", iris.StatusOK, same_as_request_path}, }}, // mormally, order matters, root should be registered at last // but we change the front level order algorithm to put last these automatically // see handler.go {"GET", "/some/{p:path}", h2, []testRouteRequest{ {"GET", "", "/some/that/is/wildcard", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/some/did", iris.StatusForbidden, same_as_request_path}, }}, {"GET", "/some/static", h, []testRouteRequest{ {"GET", "", "/some/static", iris.StatusOK, same_as_request_path}, }}, {"GET", "/some1", h, []testRouteRequest{ {"GET", "", "/some1", iris.StatusOK, same_as_request_path}, // this will show up because of the first wildcard, as we wanted to do. {"GET", "", "/some1/that/is/wildcard", iris.StatusOK, same_as_request_path}, }}, } testTheRoutes(t, tt, false) } func TestRouterWildcardRootManyAndRootStatic(t *testing.T) { tt := []testRoute{ // routes that may return 404 will be handled by the below route ("h" handler) because we added wildcard to root, // this feature is very important and can remove noumerous of previous hacks on our apps. // // Static paths and parameters have priority over wildcard, all three types can be registered in the same path prefix. // // Remember, all of those routes are registered don't be tricked by the visual appearance of the below test blocks. {"GET", "/{p:path}", h, []testRouteRequest{ {"GET", "", "/other2almost/some", iris.StatusOK, same_as_request_path}, }}, {"GET", "/static/{p:path}", h, []testRouteRequest{ {"GET", "", "/static", iris.StatusOK, same_as_request_path}, // HERE<- IF NOT FOUND THEN BACKWARDS TO WILDCARD IF THERE IS ONE, HMM. {"GET", "", "/static/something/here", iris.StatusOK, same_as_request_path}, }}, {"GET", "/", h, []testRouteRequest{ {"GET", "", "/", iris.StatusOK, same_as_request_path}, }}, {"GET", "/other/{paramother:path}", h2, []testRouteRequest{ // OK and not h2 because of the root wildcard. {"GET", "", "/other", iris.StatusOK, same_as_request_path}, {"GET", "", "/other/wildcard", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/other/wildcard/here", iris.StatusForbidden, same_as_request_path}, }}, {"GET", "/other2/{paramothersecond:path}", h2, []testRouteRequest{ {"GET", "", "/other2/wildcard", iris.StatusForbidden, same_as_request_path}, {"GET", "", "/other2/more/than/one/path/parts", iris.StatusForbidden, same_as_request_path}, }}, {"GET", "/other2/static", h3, []testRouteRequest{ {"GET", "", "/other2/static", iris.StatusOK, prefix_static_path_following_by_request_path}, // h2(Forbiddenn) instead of h3 OK because it will be handled by the /other2/{paramothersecond:path}'s handler which gives 403. {"GET", "", "/other2/staticed", iris.StatusForbidden, same_as_request_path}, }}, } testTheRoutes(t, tt, false) } func testTheRoutes(t *testing.T, tests []testRoute, debug bool) { // build the api app := iris.New() for _, tt := range tests { app.Handle(tt.method, tt.path, tt.handler) } // setup the test suite e := httptest.New(t, app, httptest.Debug(debug)) // run the tests for _, tt := range tests { for _, req := range tt.requests { // t.Logf("req: %s:%s\n", tt.method, tt.path) method := req.method if method == "" { method = tt.method } ex := e.Request(method, req.path) if req.subdomain != "" { ex.WithURL("http://" + req.subdomain + ".localhost:8080") } expectedBody := req.expectedBody if req.expectedBody == same_as_request_path { expectedBody = req.path } if req.expectedBody == from_status_code { expectedBody = http.StatusText(req.expectedStatusCode) } if req.expectedBody == prefix_static_path_following_by_request_path { expectedBody = staticPathPrefixBody + req.path } ex.Expect().Status(req.expectedStatusCode).Body().Equal(expectedBody) } } }