diff --git a/rest-dashboard.go b/rest-dashboard.go index ec7c7870..3a77c8f8 100644 --- a/rest-dashboard.go +++ b/rest-dashboard.go @@ -25,6 +25,7 @@ import ( "encoding/json" "fmt" "net/url" + "strconv" "strings" "time" @@ -170,7 +171,24 @@ type FoundBoard struct { // only starred dashboards and only for tags (logical OR applied to multiple tags). // // Reflects GET /api/search API call. +// Deprecated: This interface does not allow for API extension and is out of date. +// Please use Search(SearchType(SearchTypeDashboard)) func (r *Client) SearchDashboards(ctx context.Context, query string, starred bool, tags ...string) ([]FoundBoard, error) { + params := []SearchParam{ + SearchType(SearchTypeDashboard), + SearchQuery(query), + SearchStarred(starred), + } + for _, tag := range tags { + params = append(params, SearchTag(tag)) + } + return r.Search(ctx, params...) +} + +// Search searches folders and dashboards with query params specified. +// +// Reflects GET /api/search API call. +func (r *Client) Search(ctx context.Context, params ...SearchParam) ([]FoundBoard, error) { var ( raw []byte boards []FoundBoard @@ -179,14 +197,8 @@ func (r *Client) SearchDashboards(ctx context.Context, query string, starred boo ) u := url.URL{} q := u.Query() - if query != "" { - q.Set("query", query) - } - if starred { - q.Set("starred", "true") - } - for _, tag := range tags { - q.Add("tag", tag) + for _, p := range params { + p(&q) } if raw, code, err = r.get(ctx, "api/search", q); err != nil { return nil, err @@ -308,6 +320,95 @@ func (r *Client) DeleteDashboard(ctx context.Context, slug string) (StatusMessag return reply, err } +type ( + // SearchParam is a type for specifying Search params. + SearchParam func(*url.Values) + // SearchParamType is a type accepted by SearchType func. + SearchParamType string +) + +// Search entities to be used with SearchType(). +const ( + SearchTypeFolder SearchParamType = "dash-folder" + SearchTypeDashboard SearchParamType = "dash-db" +) + +// SearchQuery specifies Search search query. +// Empty query is silently ignored. +// Specifying it multiple times is futile, only last one will be sent. +func SearchQuery(query string) SearchParam { + return func(v *url.Values) { + if query != "" { + v.Set("query", query) + } + } +} + +// SearchTag specifies Search tag to search for. +// Empty tag is silently ignored. +// Can be specified multiple times, logical OR is applied. +func SearchTag(tag string) SearchParam { + return func(v *url.Values) { + if tag != "" { + v.Add("tag", tag) + } + } +} + +// SearchType specifies Search type to search for. +// Specifying it multiple times is futile, only last one will be sent. +func SearchType(searchType SearchParamType) SearchParam { + return func(v *url.Values) { + v.Set("type", string(searchType)) + } +} + +// SearchDashboardID specifies Search dashboard id's to search for. +// Can be specified multiple times, logical OR is applied. +func SearchDashboardID(dashboardID int) SearchParam { + return func(v *url.Values) { + v.Add("dashboardIds", strconv.Itoa(dashboardID)) + } +} + +// SearchFolderID specifies Search folder id's to search for. +// Can be specified multiple times, logical OR is applied. +func SearchFolderID(folderID int) SearchParam { + return func(v *url.Values) { + v.Add("folderIds", strconv.Itoa(folderID)) + } +} + +// SearchStarred specifies if Search should search for starred dashboards only. +// Specifying it multiple times is futile, only last one will be sent. +func SearchStarred(starred bool) SearchParam { + return func(v *url.Values) { + v.Set("starred", strconv.FormatBool(starred)) + } +} + +// SearchLimit specifies maximum number of results from Search query. +// As of grafana 6.7 it has to be <= 5000. 0 stands for absence of parameter in a query. +// Specifying it multiple times is futile, only last one will be sent. +func SearchLimit(limit uint) SearchParam { + return func(v *url.Values) { + if limit > 0 { + v.Set("limit", strconv.FormatUint(uint64(limit), 10)) + } + } +} + +// SearchPage specifies Search page number to be queried for. +// Zero page is silently ignored, page numbers start from one. +// Specifying it multiple times is futile, only last one will be sent. +func SearchPage(page uint) SearchParam { + return func(v *url.Values) { + if page > 0 { + v.Set("page", strconv.FormatUint(uint64(page), 10)) + } + } +} + // implicitly use dashboards from Grafana DB not from a file system func setPrefix(slug string) string { if strings.HasPrefix(slug, "db") { diff --git a/rest-dashboard_integration_test.go b/rest-dashboard_integration_test.go index 1deced44..5637edb2 100644 --- a/rest-dashboard_integration_test.go +++ b/rest-dashboard_integration_test.go @@ -20,6 +20,7 @@ func Test_Dashboard_CRUD(t *testing.T) { client := getClient(t) var board sdk.Board + raw, _ := ioutil.ReadFile("testdata/new-empty-dashboard-2.6.json") if err = json.Unmarshal(raw, &board); err != nil { @@ -38,10 +39,14 @@ func Test_Dashboard_CRUD(t *testing.T) { t.Fatal(err) } - if boardLinks, err = client.SearchDashboards(ctx, "", false); err != nil { + if boardLinks, err = client.Search(ctx, sdk.SearchType(sdk.SearchTypeDashboard)); err != nil { t.Fatal(err) } + if len(boardLinks) == 0 { + t.Fatal("search query returned empty dashboard list") + } + for _, link := range boardLinks { _, _, err = client.GetDashboardByUID(ctx, link.UID) if err != nil { diff --git a/rest-dashboard_test.go b/rest-dashboard_test.go new file mode 100644 index 00000000..006d98a5 --- /dev/null +++ b/rest-dashboard_test.go @@ -0,0 +1,285 @@ +package sdk_test + +import ( + "context" + "net/http" + "net/http/httptest" + "net/url" + "reflect" + "strconv" + "testing" + + "github.com/grafana-tools/sdk" +) + +func TestClient_SearchDashboards(t *testing.T) { + type in struct { + Query string + Starred bool + Tags []string + } + type testCase struct { + In in + Out url.Values + } + dashboard := string(sdk.SearchTypeDashboard) + for i, tc := range []testCase{ + { + In: in{}, + Out: url.Values{"type": []string{dashboard}, "starred": []string{"false"}}, + }, + { + In: in{Starred: true}, + Out: url.Values{"type": []string{dashboard}, "starred": []string{"true"}}, + }, + { + In: in{Query: "Foo"}, + Out: url.Values{"type": []string{dashboard}, "query": []string{"Foo"}, "starred": []string{"false"}}, + }, + { + In: in{Tags: []string{"Foo", "Bar"}}, + Out: url.Values{"type": []string{dashboard}, "starred": []string{"false"}, "tag": []string{"Foo", "Bar"}}, + }, + } { + ts := httptest.NewServer(http.HandlerFunc(testSearchQuery(t, i, tc.Out))) + client := sdk.NewClient(ts.URL, "", ts.Client()) + ctx := context.Background() + _, err := client.SearchDashboards(ctx, tc.In.Query, tc.In.Starred, tc.In.Tags...) + ts.Close() + if err != nil { + t.Fatalf("SearchDashboards test %d failed: %s", i, err) + } + } +} + +func TestClient_Search(t *testing.T) { + type testCase struct { + In []sdk.SearchParam + Out url.Values + } + for i, tc := range []testCase{ + { + In: []sdk.SearchParam{}, + Out: url.Values{}, + }, + { + // Test all options given their correct usage. + In: []sdk.SearchParam{ + sdk.SearchDashboardID(234), + sdk.SearchDashboardID(432), + sdk.SearchFolderID(123), + sdk.SearchFolderID(321), + sdk.SearchLimit(10), + sdk.SearchPage(99), + sdk.SearchQuery("Q"), + sdk.SearchStarred(true), + sdk.SearchTag("Foo"), + sdk.SearchTag("Bar"), + sdk.SearchType(sdk.SearchTypeFolder), + }, + Out: url.Values{ + "dashboardIds": []string{"234", "432"}, + "folderIds": []string{"123", "321"}, + "limit": []string{"10"}, + "page": []string{"99"}, + "query": []string{"Q"}, + "starred": []string{"true"}, + "tag": []string{"Foo", "Bar"}, + "type": []string{string(sdk.SearchTypeFolder)}, + }, + }, + { + // Test non-repeatable options. + In: []sdk.SearchParam{ + sdk.SearchLimit(10), + sdk.SearchLimit(100), + sdk.SearchPage(88), + sdk.SearchPage(99), + sdk.SearchQuery("Q1"), + sdk.SearchQuery("Q2"), + sdk.SearchStarred(true), + sdk.SearchStarred(false), + sdk.SearchType(sdk.SearchTypeFolder), + sdk.SearchType(sdk.SearchTypeDashboard), + }, + Out: url.Values{ + "limit": []string{"100"}, + "page": []string{"99"}, + "query": []string{"Q2"}, + "starred": []string{"false"}, + "type": []string{string(sdk.SearchTypeDashboard)}, + }, + }, + } { + ts := httptest.NewServer(http.HandlerFunc(testSearchQuery(t, i, tc.Out))) + client := sdk.NewClient(ts.URL, "", ts.Client()) + ctx := context.Background() + _, err := client.Search(ctx, tc.In...) + ts.Close() + if err != nil { + t.Fatalf("SearchDashboards test %d failed: %s", i, err) + } + + } +} + +func testSearchQuery(t *testing.T, testID int, exp url.Values) func(w http.ResponseWriter, r *http.Request) { + return func(w http.ResponseWriter, r *http.Request) { + if m := r.Method; m != http.MethodGet { + t.Fatalf("unexpected http method for case %d: expected %s, got %s", testID, http.MethodGet, m) + } + if e := "/api/search"; r.URL.Path != e { + t.Fatalf("unexpected http handler called for case %d: expected %s, got %s", testID, r.URL.Path, e) + } + qv, err := url.ParseQuery(r.URL.RawQuery) + if err != nil { + t.Fatalf("failed to parse query for case %d: %s, got err %s", testID, r.URL.RawQuery, err) + } + if !reflect.DeepEqual(exp, qv) { + t.Fatalf("unexpected query arguments for case %d: expected %v, got %v", testID, exp, qv) + } + if _, err := w.Write([]byte("[]")); err != nil { + t.Fatalf("failed to write http answer for case %d: %s", testID, err) + } + } +} + +func TestSearchQuery(t *testing.T) { + testStringSearchParam(t, sdk.SearchQuery, "query", []string{"foo", "bar"}) +} + +func TestSearchTag(t *testing.T) { + testRepeatableStringSearchParam(t, sdk.SearchTag, "tag", []string{"foo", "bar"}) +} + +func TestSearchDashboardID(t *testing.T) { + testRepeatableIntSearchParam(t, sdk.SearchDashboardID, "dashboardIds", []int{100, 200}) +} + +func TestSearchFolderID(t *testing.T) { + testRepeatableIntSearchParam(t, sdk.SearchFolderID, "folderIds", []int{100, 200}) +} + +func TestSearchPage(t *testing.T) { + testNonZeroUIntSearchParam(t, sdk.SearchPage, "page", []uint{100, 200}) +} + +func TestSearchLimit(t *testing.T) { + testNonZeroUIntSearchParam(t, sdk.SearchLimit, "limit", []uint{100, 200}) +} + +func TestSearchStarred(t *testing.T) { + testBoolSearchParam(t, sdk.SearchStarred, "starred", []bool{true, false}) +} + +func TestSearchType(t *testing.T) { + var ( + sp = sdk.SearchType + key = "type" + testValues = []sdk.SearchParamType{sdk.SearchTypeFolder, sdk.SearchTypeDashboard} + ) + + v := make(url.Values) + for _, testValue := range testValues { + sp(testValue)(&v) + expectedLen := 1 + gotLen := len(v[key]) + if gotLen != expectedLen { + t.Errorf("expected length of %s to be %d, but was %d", key, expectedLen, gotLen) + } + value := v.Get(key) + if value != string(testValue) { + t.Errorf("expected value of %s to be %s, but was %s", key, testValue, value) + } + } +} + +func testRepeatableStringSearchParam(t *testing.T, sp func(string) sdk.SearchParam, key string, testValues []string) { + v := make(url.Values) + for i, testValue := range testValues { + sp(testValue)(&v) + expectedLen := i + 1 + gotLen := len(v[key]) + if gotLen != expectedLen { + t.Errorf("expected length of %s to be %d, but was %d", key, expectedLen, gotLen) + } + last := v[key][i] + if last != testValue { + t.Errorf("expected last %s to be %s, but was %s", key, testValue, last) + } + } +} + +func testStringSearchParam(t *testing.T, sp func(string) sdk.SearchParam, key string, testValues []string) { + v := make(url.Values) + for _, testValue := range testValues { + sp(testValue)(&v) + expectedLen := 1 + gotLen := len(v[key]) + if gotLen != expectedLen { + t.Errorf("expected length of %s to be %d, but was %d", key, expectedLen, gotLen) + } + value := v.Get(key) + if value != testValue { + t.Errorf("expected value of %s to be %s, but was %s", key, testValue, value) + } + } +} + +func testBoolSearchParam(t *testing.T, sp func(bool) sdk.SearchParam, key string, testValues []bool) { + v := make(url.Values) + for _, testValue := range testValues { + sp(testValue)(&v) + expectedLen := 1 + gotLen := len(v[key]) + if gotLen != expectedLen { + t.Errorf("expected length of %s to be %d, but was %d", key, expectedLen, gotLen) + } + value := v.Get(key) + if value != strconv.FormatBool(testValue) { + t.Errorf("expected value of %s to be %t, but was %s", key, testValue, value) + } + } +} + +func testRepeatableIntSearchParam(t *testing.T, sp func(int) sdk.SearchParam, key string, testValues []int) { + v := make(url.Values) + for i, testValue := range testValues { + sp(testValue)(&v) + expectedLen := i + 1 + gotLen := len(v[key]) + if gotLen != expectedLen { + t.Errorf("expected length of %s to be %d, but was %d", key, expectedLen, gotLen) + } + last := v[key][i] + if last != strconv.Itoa(testValue) { + t.Errorf("expected last %s to be %d, but was %s", key, testValue, last) + } + } +} + +func testNonZeroUIntSearchParam(t *testing.T, sp func(uint) sdk.SearchParam, key string, testValues []uint) { + v := make(url.Values) + sp(0)(&v) + value := v.Get(key) + if value != "" { + t.Errorf("expected value of %s to be unset, but was %s", key, value) + } + testUIntSearchParam(t, sp, key, testValues) +} + +func testUIntSearchParam(t *testing.T, sp func(uint) sdk.SearchParam, key string, testValues []uint) { + v := make(url.Values) + for _, testValue := range testValues { + sp(testValue)(&v) + expectedLen := 1 + gotLen := len(v[key]) + if gotLen != expectedLen { + t.Errorf("expected length of %s to be %d, but was %d", key, expectedLen, gotLen) + } + value := v.Get(key) + if value != strconv.FormatUint(uint64(testValue), 10) { + t.Errorf("expected value of %s to be %d, but was %s", key, testValue, value) + } + } +}