package main import ( "fmt" "net/url" "reflect" "strings" "testing" ) func TestDropPrefixParts(t *testing.T) { f := func(path string, parts int, expectedResult string) { t.Helper() result := dropPrefixParts(path, parts) if result != expectedResult { t.Fatalf("unexpected result; got %q; want %q", result, expectedResult) } } f("", 0, "") f("", 1, "") f("", 10, "") f("foo", 0, "foo") f("foo", -1, "foo") f("foo", 1, "") f("/foo", 0, "/foo") f("/foo/bar", 0, "/foo/bar") f("/foo/bar/baz", 0, "/foo/bar/baz") f("foo", 0, "foo") f("foo/bar", 0, "foo/bar") f("foo/bar/baz", 0, "foo/bar/baz") f("/foo/", 0, "/foo/") f("/foo/bar/", 0, "/foo/bar/") f("/foo/bar/baz/", 0, "/foo/bar/baz/") f("/foo", 1, "") f("/foo/bar", 1, "/bar") f("/foo/bar/baz", 1, "/bar/baz") f("foo", 1, "") f("foo/bar", 1, "/bar") f("foo/bar/baz", 1, "/bar/baz") f("/foo/", 1, "/") f("/foo/bar/", 1, "/bar/") f("/foo/bar/baz/", 1, "/bar/baz/") f("/foo", 2, "") f("/foo/bar", 2, "") f("/foo/bar/baz", 2, "/baz") f("foo", 2, "") f("foo/bar", 2, "") f("foo/bar/baz", 2, "/baz") f("/foo/", 2, "") f("/foo/bar/", 2, "/") f("/foo/bar/baz/", 2, "/baz/") f("/foo", 3, "") f("/foo/bar", 3, "") f("/foo/bar/baz", 3, "") f("foo", 3, "") f("foo/bar", 3, "") f("foo/bar/baz", 3, "") f("/foo/", 3, "") f("/foo/bar/", 3, "") f("/foo/bar/baz/", 3, "/") f("/foo/", 4, "") f("/foo/bar/", 4, "") f("/foo/bar/baz/", 4, "") } func TestCreateTargetURLSuccess(t *testing.T) { f := func(ui *UserInfo, requestURI, expectedTarget, expectedRequestHeaders, expectedResponseHeaders string, expectedRetryStatusCodes []int, expectedLoadBalancingPolicy string, expectedDropSrcPathPrefixParts int) { t.Helper() if err := ui.initURLs(); err != nil { t.Fatalf("cannot initialize urls inside UserInfo: %s", err) } u, err := url.Parse(requestURI) if err != nil { t.Fatalf("cannot parse %q: %s", requestURI, err) } u = normalizeURL(u) up, hc := ui.getURLPrefixAndHeaders(u, nil) if up == nil { t.Fatalf("cannot determie backend: %s", err) } bu := up.getBackendURL() target := mergeURLs(bu.url, u, up.dropSrcPathPrefixParts) bu.put() if target.String() != expectedTarget { t.Fatalf("unexpected target; got %q; want %q", target, expectedTarget) } if s := headersToString(hc.RequestHeaders); s != expectedRequestHeaders { t.Fatalf("unexpected request headers; got %q; want %q", s, expectedRequestHeaders) } if s := headersToString(hc.ResponseHeaders); s != expectedResponseHeaders { t.Fatalf("unexpected response headers; got %q; want %q", s, expectedResponseHeaders) } if !reflect.DeepEqual(up.retryStatusCodes, expectedRetryStatusCodes) { t.Fatalf("unexpected retryStatusCodes; got %d; want %d", up.retryStatusCodes, expectedRetryStatusCodes) } if up.loadBalancingPolicy != expectedLoadBalancingPolicy { t.Fatalf("unexpected loadBalancingPolicy; got %q; want %q", up.loadBalancingPolicy, expectedLoadBalancingPolicy) } if up.dropSrcPathPrefixParts != expectedDropSrcPathPrefixParts { t.Fatalf("unexpected dropSrcPathPrefixParts; got %d; want %d", up.dropSrcPathPrefixParts, expectedDropSrcPathPrefixParts) } } // Simple routing with `url_prefix` f(&UserInfo{ URLPrefix: mustParseURL("http://foo.bar"), }, "", "http://foo.bar/.", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("http://foo.bar"), HeadersConf: HeadersConf{ RequestHeaders: []Header{ { Name: "bb", Value: "aaa", }, }, ResponseHeaders: []Header{ { Name: "x", Value: "y", }, }, }, RetryStatusCodes: []int{503, 501}, LoadBalancingPolicy: "first_available", DropSrcPathPrefixParts: intp(2), }, "/a/b/c", "http://foo.bar/c", `bb: aaa`, `x: y`, []int{503, 501}, "first_available", 2) f(&UserInfo{ URLPrefix: mustParseURL("http://foo.bar/federate"), }, "/", "http://foo.bar/federate", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("http://foo.bar"), }, "a/b?c=d", "http://foo.bar/a/b?c=d", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("https://sss:3894/x/y"), }, "/z", "https://sss:3894/x/y/z", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("https://sss:3894/x/y"), }, "/../../aaa", "https://sss:3894/x/y/aaa", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("https://sss:3894/x/y"), }, "/./asd/../../aaa?a=d&s=s/../d", "https://sss:3894/x/y/aaa?a=d&s=s%2F..%2Fd", "", "", nil, "least_loaded", 0) // Complex routing with `url_map` ui := &UserInfo{ URLMaps: []URLMap{ { SrcHosts: getRegexs([]string{"host42"}), SrcPaths: getRegexs([]string{"/vmsingle/api/v1/query"}), SrcQueryArgs: []QueryArg{ { Name: "db", Value: "foo", }, }, URLPrefix: mustParseURL("http://vmselect/0/prometheus"), HeadersConf: HeadersConf{ RequestHeaders: []Header{ { Name: "xx", Value: "aa", }, { Name: "yy", Value: "asdf", }, }, ResponseHeaders: []Header{ { Name: "qwe", Value: "rty", }, }, }, RetryStatusCodes: []int{503, 500, 501}, LoadBalancingPolicy: "first_available", DropSrcPathPrefixParts: intp(1), }, { SrcPaths: getRegexs([]string{"/api/v1/write"}), URLPrefix: mustParseURL("http://vminsert/0/prometheus"), RetryStatusCodes: []int{}, DropSrcPathPrefixParts: intp(0), }, }, URLPrefix: mustParseURL("http://default-server"), HeadersConf: HeadersConf{ RequestHeaders: []Header{{ Name: "bb", Value: "aaa", }}, ResponseHeaders: []Header{{ Name: "x", Value: "y", }}, }, RetryStatusCodes: []int{502}, DropSrcPathPrefixParts: intp(2), } f(ui, "http://host42/vmsingle/api/v1/query?query=up&db=foo", "http://vmselect/0/prometheus/api/v1/query?db=foo&query=up", "xx: aa\nyy: asdf", "qwe: rty", []int{503, 500, 501}, "first_available", 1) f(ui, "http://host123/vmsingle/api/v1/query?query=up", "http://default-server/v1/query?query=up", "bb: aaa", "x: y", []int{502}, "least_loaded", 2) f(ui, "https://foo-host/api/v1/write", "http://vminsert/0/prometheus/api/v1/write", "", "", []int{}, "least_loaded", 0) f(ui, "https://foo-host/foo/bar/api/v1/query_range", "http://default-server/api/v1/query_range", "bb: aaa", "x: y", []int{502}, "least_loaded", 2) // Complex routing regexp paths in `url_map` ui = &UserInfo{ URLMaps: []URLMap{ { SrcPaths: getRegexs([]string{"/api/v1/query(_range)?", "/api/v1/label/[^/]+/values"}), URLPrefix: mustParseURL("http://vmselect/0/prometheus"), }, { SrcPaths: getRegexs([]string{"/api/v1/write"}), URLPrefix: mustParseURL("http://vminsert/0/prometheus"), }, { SrcHosts: getRegexs([]string{"vmui\\..+"}), URLPrefix: mustParseURL("http://vmui.host:1234/vmui/"), }, }, URLPrefix: mustParseURL("http://default-server"), } f(ui, "/api/v1/query?query=up", "http://vmselect/0/prometheus/api/v1/query?query=up", "", "", nil, "least_loaded", 0) f(ui, "/api/v1/query_range?query=up", "http://vmselect/0/prometheus/api/v1/query_range?query=up", "", "", nil, "least_loaded", 0) f(ui, "/api/v1/label/foo/values", "http://vmselect/0/prometheus/api/v1/label/foo/values", "", "", nil, "least_loaded", 0) f(ui, "/api/v1/write", "http://vminsert/0/prometheus/api/v1/write", "", "", nil, "least_loaded", 0) f(ui, "/api/v1/foo/bar", "http://default-server/api/v1/foo/bar", "", "", nil, "least_loaded", 0) f(ui, "https://vmui.foobar.com/a/b?c=d", "http://vmui.host:1234/vmui/a/b?c=d", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("http://foo.bar?extra_label=team=dev"), }, "/api/v1/query", "http://foo.bar/api/v1/query?extra_label=team=dev", "", "", nil, "least_loaded", 0) f(&UserInfo{ URLPrefix: mustParseURL("http://foo.bar?extra_label=team=mobile"), }, "/api/v1/query?extra_label=team=dev", "http://foo.bar/api/v1/query?extra_label=team%3Dmobile", "", "", nil, "least_loaded", 0) } func TestCreateTargetURLFailure(t *testing.T) { f := func(ui *UserInfo, requestURI string) { t.Helper() u, err := url.Parse(requestURI) if err != nil { t.Fatalf("cannot parse %q: %s", requestURI, err) } u = normalizeURL(u) up, hc := ui.getURLPrefixAndHeaders(u, nil) if up != nil { t.Fatalf("unexpected non-empty up=%#v", up) } if hc.RequestHeaders != nil { t.Fatalf("unexpected non-empty request headers=%q", hc.RequestHeaders) } if hc.ResponseHeaders != nil { t.Fatalf("unexpected non-empty response headers=%q", hc.ResponseHeaders) } } f(&UserInfo{}, "/foo/bar") f(&UserInfo{ URLMaps: []URLMap{ { SrcPaths: getRegexs([]string{"/api/v1/query"}), URLPrefix: mustParseURL("http://foobar/baz"), }, }, }, "/api/v1/write") } func headersToString(hs []Header) string { a := make([]string, len(hs)) for i, h := range hs { a[i] = fmt.Sprintf("%s: %s", h.Name, h.Value) } return strings.Join(a, "\n") }