2020-04-20 09:20:45 -05:00
|
|
|
// package search_test contains integration tests for search
|
|
|
|
package searchstore_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-04-27 07:16:03 -05:00
|
|
|
"testing"
|
|
|
|
|
2022-05-24 08:24:55 -05:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
2020-04-20 09:20:45 -05:00
|
|
|
"github.com/grafana/grafana/pkg/components/simplejson"
|
|
|
|
"github.com/grafana/grafana/pkg/models"
|
2022-05-24 08:24:55 -05:00
|
|
|
"github.com/grafana/grafana/pkg/services/dashboards"
|
2022-08-10 04:56:48 -05:00
|
|
|
"github.com/grafana/grafana/pkg/services/org"
|
2020-04-20 09:20:45 -05:00
|
|
|
"github.com/grafana/grafana/pkg/services/sqlstore"
|
|
|
|
"github.com/grafana/grafana/pkg/services/sqlstore/permissions"
|
|
|
|
"github.com/grafana/grafana/pkg/services/sqlstore/searchstore"
|
2022-08-10 04:56:48 -05:00
|
|
|
"github.com/grafana/grafana/pkg/services/user"
|
2022-02-16 07:15:44 -06:00
|
|
|
"github.com/grafana/grafana/pkg/util"
|
2020-04-20 09:20:45 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
limit int64 = 15
|
|
|
|
page int64 = 1
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestBuilder_EqualResults_Basic(t *testing.T) {
|
2022-08-10 04:56:48 -05:00
|
|
|
user := &user.SignedInUser{
|
2022-08-11 06:28:55 -05:00
|
|
|
UserID: 1,
|
|
|
|
OrgID: 1,
|
2022-08-10 04:56:48 -05:00
|
|
|
OrgRole: org.RoleEditor,
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
db := setupTestEnvironment(t)
|
2022-08-11 06:28:55 -05:00
|
|
|
dashIds := createDashboards(t, db, 0, 1, user.OrgID)
|
2021-05-06 03:48:33 -05:00
|
|
|
require.Len(t, dashIds, 1)
|
2020-04-20 09:20:45 -05:00
|
|
|
|
|
|
|
// create one dashboard in another organization that shouldn't
|
|
|
|
// be listed in the results.
|
2021-03-17 10:06:10 -05:00
|
|
|
createDashboards(t, db, 1, 2, 2)
|
2020-04-20 09:20:45 -05:00
|
|
|
|
|
|
|
builder := &searchstore.Builder{
|
|
|
|
Filters: []interface{}{
|
2022-08-11 06:28:55 -05:00
|
|
|
searchstore.OrgFilter{OrgId: user.OrgID},
|
2020-04-20 09:20:45 -05:00
|
|
|
searchstore.TitleSorter{},
|
|
|
|
},
|
2021-08-25 08:11:22 -05:00
|
|
|
Dialect: db.Dialect,
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
2022-05-24 08:24:55 -05:00
|
|
|
res := []dashboards.DashboardSearchProjection{}
|
2021-03-17 10:06:10 -05:00
|
|
|
err := db.WithDbSession(context.Background(), func(sess *sqlstore.DBSession) error {
|
2020-11-10 23:21:08 -06:00
|
|
|
sql, params := builder.ToSQL(limit, page)
|
2020-04-27 07:16:03 -05:00
|
|
|
return sess.SQL(sql, params...).Find(&res)
|
2020-04-20 09:20:45 -05:00
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2020-04-27 07:16:03 -05:00
|
|
|
assert.Len(t, res, 1)
|
2021-02-11 01:49:16 -06:00
|
|
|
res[0].UID = ""
|
2022-05-24 08:24:55 -05:00
|
|
|
assert.EqualValues(t, []dashboards.DashboardSearchProjection{
|
2020-04-27 07:16:03 -05:00
|
|
|
{
|
2021-05-06 03:48:33 -05:00
|
|
|
ID: dashIds[0],
|
2020-04-27 07:16:03 -05:00
|
|
|
Title: "A",
|
|
|
|
Slug: "a",
|
|
|
|
Term: "templated",
|
|
|
|
},
|
|
|
|
}, res)
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestBuilder_Pagination(t *testing.T) {
|
2022-08-10 04:56:48 -05:00
|
|
|
user := &user.SignedInUser{
|
2022-08-11 06:28:55 -05:00
|
|
|
UserID: 1,
|
|
|
|
OrgID: 1,
|
2022-08-10 04:56:48 -05:00
|
|
|
OrgRole: org.RoleViewer,
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
db := setupTestEnvironment(t)
|
2022-08-11 06:28:55 -05:00
|
|
|
createDashboards(t, db, 0, 25, user.OrgID)
|
2020-04-20 09:20:45 -05:00
|
|
|
|
|
|
|
builder := &searchstore.Builder{
|
|
|
|
Filters: []interface{}{
|
2022-08-11 06:28:55 -05:00
|
|
|
searchstore.OrgFilter{OrgId: user.OrgID},
|
2020-04-20 09:20:45 -05:00
|
|
|
searchstore.TitleSorter{},
|
|
|
|
},
|
2021-08-25 08:11:22 -05:00
|
|
|
Dialect: db.Dialect,
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
2022-05-24 08:24:55 -05:00
|
|
|
resPg1 := []dashboards.DashboardSearchProjection{}
|
|
|
|
resPg2 := []dashboards.DashboardSearchProjection{}
|
|
|
|
resPg3 := []dashboards.DashboardSearchProjection{}
|
2021-03-17 10:06:10 -05:00
|
|
|
err := db.WithDbSession(context.Background(), func(sess *sqlstore.DBSession) error {
|
2020-11-10 23:21:08 -06:00
|
|
|
sql, params := builder.ToSQL(15, 1)
|
2020-04-20 09:20:45 -05:00
|
|
|
err := sess.SQL(sql, params...).Find(&resPg1)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-11-10 23:21:08 -06:00
|
|
|
sql, params = builder.ToSQL(15, 2)
|
2020-04-20 09:20:45 -05:00
|
|
|
err = sess.SQL(sql, params...).Find(&resPg2)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-11-10 23:21:08 -06:00
|
|
|
sql, params = builder.ToSQL(15, 3)
|
2020-04-20 09:20:45 -05:00
|
|
|
return sess.SQL(sql, params...).Find(&resPg3)
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Len(t, resPg1, 15)
|
|
|
|
assert.Len(t, resPg2, 10)
|
|
|
|
assert.Len(t, resPg3, 0, "sanity check: pages after last should be empty")
|
|
|
|
|
|
|
|
assert.Equal(t, "A", resPg1[0].Title, "page 1 should start with the first dashboard")
|
|
|
|
assert.Equal(t, "P", resPg2[0].Title, "page 2 should start with the 16th dashboard")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBuilder_Permissions(t *testing.T) {
|
2022-08-10 04:56:48 -05:00
|
|
|
user := &user.SignedInUser{
|
2022-08-11 06:28:55 -05:00
|
|
|
UserID: 1,
|
|
|
|
OrgID: 1,
|
2022-08-10 04:56:48 -05:00
|
|
|
OrgRole: org.RoleViewer,
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
db := setupTestEnvironment(t)
|
2022-08-11 06:28:55 -05:00
|
|
|
createDashboards(t, db, 0, 1, user.OrgID)
|
2020-04-20 09:20:45 -05:00
|
|
|
|
|
|
|
level := models.PERMISSION_EDIT
|
|
|
|
|
|
|
|
builder := &searchstore.Builder{
|
|
|
|
Filters: []interface{}{
|
2022-08-11 06:28:55 -05:00
|
|
|
searchstore.OrgFilter{OrgId: user.OrgID},
|
2020-04-20 09:20:45 -05:00
|
|
|
searchstore.TitleSorter{},
|
|
|
|
permissions.DashboardPermissionFilter{
|
2021-08-25 08:11:22 -05:00
|
|
|
Dialect: db.Dialect,
|
2020-04-20 09:20:45 -05:00
|
|
|
OrgRole: user.OrgRole,
|
2022-08-11 06:28:55 -05:00
|
|
|
OrgId: user.OrgID,
|
|
|
|
UserId: user.UserID,
|
2020-04-20 09:20:45 -05:00
|
|
|
PermissionLevel: level,
|
|
|
|
},
|
|
|
|
},
|
2021-08-25 08:11:22 -05:00
|
|
|
Dialect: db.Dialect,
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
2022-05-24 08:24:55 -05:00
|
|
|
res := []dashboards.DashboardSearchProjection{}
|
2021-03-17 10:06:10 -05:00
|
|
|
err := db.WithDbSession(context.Background(), func(sess *sqlstore.DBSession) error {
|
2020-11-10 23:21:08 -06:00
|
|
|
sql, params := builder.ToSQL(limit, page)
|
2020-04-27 07:16:03 -05:00
|
|
|
return sess.SQL(sql, params...).Find(&res)
|
2020-04-20 09:20:45 -05:00
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2020-04-27 07:16:03 -05:00
|
|
|
assert.Len(t, res, 0)
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
2020-11-10 23:21:08 -06:00
|
|
|
func setupTestEnvironment(t *testing.T) *sqlstore.SQLStore {
|
2020-04-20 09:20:45 -05:00
|
|
|
t.Helper()
|
|
|
|
store := sqlstore.InitTestDB(t)
|
|
|
|
return store
|
|
|
|
}
|
|
|
|
|
2021-05-06 03:48:33 -05:00
|
|
|
func createDashboards(t *testing.T, db *sqlstore.SQLStore, startID, endID int, orgID int64) []int64 {
|
2021-03-17 10:06:10 -05:00
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
require.GreaterOrEqual(t, endID, startID)
|
2020-04-20 09:20:45 -05:00
|
|
|
|
2021-05-06 03:48:33 -05:00
|
|
|
createdIds := []int64{}
|
2020-04-20 09:20:45 -05:00
|
|
|
for i := startID; i < endID; i++ {
|
|
|
|
dashboard, err := simplejson.NewJson([]byte(`{
|
|
|
|
"id": null,
|
|
|
|
"uid": null,
|
|
|
|
"title": "` + lexiCounter(i) + `",
|
|
|
|
"tags": [ "templated" ],
|
|
|
|
"timezone": "browser",
|
|
|
|
"schemaVersion": 16,
|
|
|
|
"version": 0
|
|
|
|
}`))
|
2021-03-17 10:06:10 -05:00
|
|
|
require.NoError(t, err)
|
2022-02-16 07:15:44 -06:00
|
|
|
|
|
|
|
var dash *models.Dashboard
|
|
|
|
err = db.WithDbSession(context.Background(), func(sess *sqlstore.DBSession) error {
|
|
|
|
dash = models.NewDashboardFromJson(dashboard)
|
|
|
|
dash.OrgId = orgID
|
|
|
|
dash.Uid = util.GenerateShortUID()
|
|
|
|
dash.CreatedBy = 1
|
|
|
|
dash.UpdatedBy = 1
|
|
|
|
_, err := sess.Insert(dash)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
tags := dash.GetTags()
|
|
|
|
if len(tags) > 0 {
|
|
|
|
for _, tag := range tags {
|
|
|
|
if _, err := sess.Insert(&sqlstore.DashboardTag{DashboardId: dash.Id, Term: tag}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2020-04-20 09:20:45 -05:00
|
|
|
})
|
2021-03-17 10:06:10 -05:00
|
|
|
require.NoError(t, err)
|
2021-05-06 03:48:33 -05:00
|
|
|
|
|
|
|
createdIds = append(createdIds, dash.Id)
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
2021-05-06 03:48:33 -05:00
|
|
|
|
|
|
|
return createdIds
|
2020-04-20 09:20:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// lexiCounter counts in a lexicographically sortable order.
|
|
|
|
func lexiCounter(n int) string {
|
|
|
|
alphabet := "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
|
|
value := string(alphabet[n%26])
|
|
|
|
|
|
|
|
if n >= 26 {
|
|
|
|
value = lexiCounter(n/26-1) + value
|
|
|
|
}
|
|
|
|
|
|
|
|
return value
|
|
|
|
}
|