2017-06-12 08:48:55 -05:00
package api
import (
2021-06-15 09:08:27 -05:00
"context"
2017-06-12 08:48:55 -05:00
"encoding/json"
2021-05-26 09:20:13 -05:00
"errors"
2018-02-19 04:12:56 -06:00
"fmt"
2020-06-22 11:00:39 -05:00
"io/ioutil"
2021-07-16 05:40:03 -05:00
"net/http"
2017-06-12 08:48:55 -05:00
"testing"
2022-03-10 11:19:50 -06:00
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
2017-06-12 08:48:55 -05:00
"github.com/grafana/grafana/pkg/api/dtos"
2021-01-15 07:43:20 -06:00
"github.com/grafana/grafana/pkg/api/response"
"github.com/grafana/grafana/pkg/api/routing"
2017-06-12 16:05:32 -05:00
"github.com/grafana/grafana/pkg/components/simplejson"
2022-06-30 08:31:54 -05:00
"github.com/grafana/grafana/pkg/framework/coremodel/registry"
2021-09-22 09:28:40 -05:00
"github.com/grafana/grafana/pkg/infra/usagestats"
2020-03-04 05:57:20 -06:00
"github.com/grafana/grafana/pkg/models"
2022-03-10 05:58:18 -06:00
accesscontrolmock "github.com/grafana/grafana/pkg/services/accesscontrol/mock"
2018-10-13 00:53:28 -05:00
"github.com/grafana/grafana/pkg/services/alerting"
2017-12-12 09:15:24 -06:00
"github.com/grafana/grafana/pkg/services/dashboards"
2022-02-16 07:15:44 -06:00
"github.com/grafana/grafana/pkg/services/dashboards/database"
2022-05-21 19:44:12 -05:00
"github.com/grafana/grafana/pkg/services/dashboards/service"
2022-05-25 03:41:51 -05:00
dashver "github.com/grafana/grafana/pkg/services/dashboardversion"
"github.com/grafana/grafana/pkg/services/dashboardversion/dashvertest"
2022-01-26 11:44:20 -06:00
"github.com/grafana/grafana/pkg/services/featuremgmt"
2022-03-21 04:49:49 -05:00
"github.com/grafana/grafana/pkg/services/guardian"
2021-05-12 01:48:17 -05:00
"github.com/grafana/grafana/pkg/services/libraryelements"
2020-10-28 03:36:57 -05:00
"github.com/grafana/grafana/pkg/services/live"
2022-04-21 08:03:17 -05:00
pref "github.com/grafana/grafana/pkg/services/preference"
"github.com/grafana/grafana/pkg/services/preference/preftest"
2019-04-30 06:32:18 -05:00
"github.com/grafana/grafana/pkg/services/provisioning"
2020-12-15 12:09:04 -06:00
"github.com/grafana/grafana/pkg/services/quota"
2021-08-25 08:11:22 -05:00
"github.com/grafana/grafana/pkg/services/sqlstore"
2022-02-03 02:20:20 -06:00
"github.com/grafana/grafana/pkg/services/sqlstore/mockstore"
2017-12-15 07:19:49 -06:00
"github.com/grafana/grafana/pkg/setting"
2021-10-11 07:30:59 -05:00
"github.com/grafana/grafana/pkg/web"
2017-06-12 08:48:55 -05:00
)
2020-06-22 11:00:39 -05:00
func TestGetHomeDashboard ( t * testing . T ) {
2021-07-16 05:40:03 -05:00
httpReq , err := http . NewRequest ( http . MethodGet , "" , nil )
require . NoError ( t , err )
2022-02-09 06:44:38 -06:00
httpReq . Header . Add ( "Content-Type" , "application/json" )
2021-10-11 07:30:59 -05:00
req := & models . ReqContext { SignedInUser : & models . SignedInUser { } , Context : & web . Context { Req : httpReq } }
2020-06-22 11:00:39 -05:00
cfg := setting . NewCfg ( )
cfg . StaticRootPath = "../../public/"
2022-04-21 08:03:17 -05:00
prefService := preftest . NewPreferenceServiceFake ( )
2022-05-25 03:41:51 -05:00
dashboardVersionService := dashvertest . NewDashboardVersionServiceFake ( )
2020-06-22 11:00:39 -05:00
2021-03-17 10:06:10 -05:00
hs := & HTTPServer {
2022-05-25 03:41:51 -05:00
Cfg : cfg ,
pluginStore : & fakePluginStore { } ,
SQLStore : mockstore . NewSQLStoreMock ( ) ,
preferenceService : prefService ,
dashboardVersionService : dashboardVersionService ,
2021-03-17 10:06:10 -05:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2020-06-22 11:00:39 -05:00
tests := [ ] struct {
name string
defaultSetting string
expectedDashboardPath string
} {
{ name : "using default config" , defaultSetting : "" , expectedDashboardPath : "../../public/dashboards/home.json" } ,
{ name : "custom path" , defaultSetting : "../../public/dashboards/default.json" , expectedDashboardPath : "../../public/dashboards/default.json" } ,
}
for _ , tc := range tests {
t . Run ( tc . name , func ( t * testing . T ) {
dash := dtos . DashboardFullWithMeta { }
dash . Meta . IsHome = true
dash . Meta . FolderTitle = "General"
homeDashJSON , err := ioutil . ReadFile ( tc . expectedDashboardPath )
require . NoError ( t , err , "must be able to read expected dashboard file" )
hs . Cfg . DefaultHomeDashboardPath = tc . defaultSetting
bytes , err := simplejson . NewJson ( homeDashJSON )
require . NoError ( t , err , "must be able to encode file as JSON" )
2022-04-21 08:03:17 -05:00
prefService . ExpectedPreference = & pref . Preference { }
2020-06-22 11:00:39 -05:00
dash . Dashboard = bytes
b , err := json . Marshal ( dash )
require . NoError ( t , err , "must be able to marshal object to JSON" )
res := hs . GetHomeDashboard ( req )
2021-01-15 07:43:20 -06:00
nr , ok := res . ( * response . NormalResponse )
2020-06-22 11:00:39 -05:00
require . True ( t , ok , "should return *NormalResponse" )
2021-01-15 07:43:20 -06:00
require . Equal ( t , b , nr . Body ( ) , "default home dashboard should equal content on disk" )
2020-06-22 11:00:39 -05:00
} )
}
}
2022-03-22 14:48:32 -05:00
func newTestLive ( t * testing . T , store * sqlstore . SQLStore ) * live . GrafanaLive {
2022-01-26 11:44:20 -06:00
features := featuremgmt . WithFeatures ( )
2021-08-25 08:11:22 -05:00
cfg := & setting . Cfg { AppURL : "http://localhost:3000/" }
2022-01-26 11:44:20 -06:00
cfg . IsFeatureToggleEnabled = features . IsEnabled
gLive , err := live . ProvideService ( nil , cfg ,
routing . NewRouteRegister ( ) ,
nil , nil , nil ,
2022-03-22 14:48:32 -05:00
store ,
2022-01-26 11:44:20 -06:00
nil ,
& usagestats . UsageStatsMock { T : t } ,
nil ,
2022-05-17 13:52:22 -05:00
features , accesscontrolmock . New ( ) , & dashboards . FakeDashboardService { } )
2021-05-04 10:44:55 -05:00
require . NoError ( t , err )
return gLive
}
2018-02-19 04:12:56 -06:00
// This tests three main scenarios.
// If a user has access to execute an action on a dashboard:
// 1. and the dashboard is in a folder which does not have an acl
// 2. and the dashboard is in a folder which does have an acl
// 3. Post dashboard response tests
2018-01-30 07:09:30 -06:00
2020-11-13 02:52:38 -06:00
func TestDashboardAPIEndpoint ( t * testing . T ) {
t . Run ( "Given a dashboard with a parent folder which does not have an ACL" , func ( t * testing . T ) {
2022-02-07 05:43:43 -06:00
fakeDash := models . NewDashboard ( "Child dash" )
fakeDash . Id = 1
fakeDash . FolderId = 1
fakeDash . HasAcl = false
2022-05-25 03:41:51 -05:00
fakeDashboardVersionService := dashvertest . NewDashboardVersionServiceFake ( )
fakeDashboardVersionService . ExpectedDashboardVersion = & dashver . DashboardVersion { }
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = fakeDash
} ) . Return ( nil )
2022-02-07 05:43:43 -06:00
mockSQLStore := mockstore . NewSQLStoreMock ( )
2018-01-31 09:51:06 -06:00
2022-02-07 05:43:43 -06:00
hs := & HTTPServer {
2022-05-25 03:41:51 -05:00
Cfg : setting . NewCfg ( ) ,
pluginStore : & fakePluginStore { } ,
SQLStore : mockSQLStore ,
AccessControl : accesscontrolmock . New ( ) ,
Features : featuremgmt . WithFeatures ( ) ,
2022-07-06 13:42:39 -05:00
DashboardService : dashboardService ,
2022-05-25 03:41:51 -05:00
dashboardVersionService : fakeDashboardVersionService ,
2022-02-07 05:43:43 -06:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2018-01-29 14:23:07 -06:00
2022-02-07 05:43:43 -06:00
setUp := func ( ) {
2020-11-13 02:52:38 -06:00
viewerRole := models . ROLE_VIEWER
editorRole := models . ROLE_EDITOR
2022-06-01 13:16:26 -05:00
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO {
{ Role : & viewerRole , Permission : models . PERMISSION_VIEW } ,
{ Role : & editorRole , Permission : models . PERMISSION_EDIT } ,
}
} ) . Return ( nil )
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2020-11-13 02:52:38 -06:00
}
2017-06-22 16:43:55 -05:00
2018-01-30 07:09:30 -06:00
// This tests two scenarios:
// 1. user is an org viewer
// 2. user is an org editor
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Viewer" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_VIEWER
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUp ( )
sc . sqlStore = mockSQLStore
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , nil , nil , dashboardService )
2020-11-13 02:52:38 -06:00
assert . False ( t , dash . Meta . CanEdit )
assert . False ( t , dash . Meta . CanSave )
assert . False ( t , dash . Meta . CanAdmin )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" ,
"/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
sc . sqlStore = mockSQLStore
2018-01-31 09:46:31 -06:00
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersion ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" ,
"/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
sc . sqlStore = mockSQLStore
2017-06-13 17:28:34 -05:00
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 08:48:55 -05:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Editor" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_EDITOR
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUp ( )
sc . sqlStore = mockSQLStore
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , nil , nil , dashboardService )
2017-06-12 08:48:55 -05:00
2020-11-13 02:52:38 -06:00
assert . True ( t , dash . Meta . CanEdit )
assert . True ( t , dash . Meta . CanSave )
assert . False ( t , dash . Meta . CanAdmin )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" ,
"/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
sc . sqlStore = mockSQLStore
hs . callGetDashboardVersion ( sc )
2018-01-31 09:46:31 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" ,
"/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2017-06-13 17:28:34 -05:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 08:48:55 -05:00
} )
} )
2020-11-13 02:52:38 -06:00
t . Run ( "Given a dashboard with a parent folder which has an ACL" , func ( t * testing . T ) {
2022-02-07 05:43:43 -06:00
fakeDash := models . NewDashboard ( "Child dash" )
fakeDash . Id = 1
fakeDash . FolderId = 1
fakeDash . HasAcl = true
2022-05-25 03:41:51 -05:00
fakeDashboardVersionService := dashvertest . NewDashboardVersionServiceFake ( )
fakeDashboardVersionService . ExpectedDashboardVersion = & dashver . DashboardVersion { }
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = fakeDash
} ) . Return ( nil )
2022-06-01 13:16:26 -05:00
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO {
{
DashboardId : 1 ,
Permission : models . PERMISSION_EDIT ,
UserId : 200 ,
} ,
}
} ) . Return ( nil )
2022-02-07 05:43:43 -06:00
mockSQLStore := mockstore . NewSQLStoreMock ( )
2022-03-10 05:58:18 -06:00
cfg := setting . NewCfg ( )
2022-03-22 14:48:32 -05:00
sql := sqlstore . InitTestDB ( t )
2022-05-17 13:52:22 -05:00
2020-11-13 02:52:38 -06:00
hs := & HTTPServer {
2022-05-25 03:41:51 -05:00
Cfg : cfg ,
Live : newTestLive ( t , sql ) ,
LibraryPanelService : & mockLibraryPanelService { } ,
LibraryElementService : & mockLibraryElementService { } ,
SQLStore : mockSQLStore ,
AccessControl : accesscontrolmock . New ( ) ,
2022-07-06 13:42:39 -05:00
DashboardService : dashboardService ,
2022-05-25 03:41:51 -05:00
dashboardVersionService : fakeDashboardVersionService ,
2020-11-13 02:52:38 -06:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2017-06-12 08:48:55 -05:00
2022-02-07 05:43:43 -06:00
setUp := func ( ) {
2020-11-13 02:52:38 -06:00
origCanEdit := setting . ViewersCanEdit
t . Cleanup ( func ( ) {
setting . ViewersCanEdit = origCanEdit
} )
setting . ViewersCanEdit = false
2022-06-01 13:16:26 -05:00
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2019-04-30 06:32:18 -05:00
}
2018-01-30 07:09:30 -06:00
// This tests six scenarios:
// 1. user is an org viewer AND has no permissions for this dashboard
// 2. user is an org editor AND has no permissions for this dashboard
// 3. user is an org viewer AND has been granted edit permission for the dashboard
// 4. user is an org viewer AND all viewers have edit permission for this dashboard
// 5. user is an org viewer AND has been granted an admin permission
// 6. user is an org editor AND has been granted a view permission
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Viewer and has no permissions for this dashboard" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_VIEWER
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUp ( )
sc . sqlStore = mockSQLStore
2020-11-13 02:52:38 -06:00
sc . handlerFunc = hs . GetDashboard
sc . fakeReqWithParams ( "GET" , sc . url , map [ string ] string { } ) . exec ( )
2022-02-07 05:43:43 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling DELETE on" , "DELETE" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUp ( )
sc . sqlStore = mockSQLStore
2022-05-23 10:14:27 -05:00
hs . callDeleteDashboardByUID ( t , sc , dashboardService )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" ,
"/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
sc . sqlStore = mockSQLStore
hs . callGetDashboardVersion ( sc )
2018-01-31 09:46:31 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" ,
"/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2017-06-13 17:28:34 -05:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 08:48:55 -05:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Editor and has no permissions for this dashboard" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_EDITOR
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUp ( )
sc . sqlStore = mockSQLStore
2020-11-13 02:52:38 -06:00
sc . handlerFunc = hs . GetDashboard
sc . fakeReqWithParams ( "GET" , sc . url , map [ string ] string { } ) . exec ( )
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling DELETE on" , "DELETE" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUp ( )
2022-05-23 10:14:27 -05:00
hs . callDeleteDashboardByUID ( t , sc , dashboardService )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" ,
"/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersion ( sc )
2018-01-31 09:46:31 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" ,
"/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUp ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2017-06-13 17:28:34 -05:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 08:48:55 -05:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Viewer but has an edit permission" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_VIEWER
2017-06-12 16:05:32 -05:00
2022-02-07 05:43:43 -06:00
setUpInner := func ( ) {
2022-06-01 13:16:26 -05:00
origCanEdit := setting . ViewersCanEdit
t . Cleanup ( func ( ) {
setting . ViewersCanEdit = origCanEdit
} )
setting . ViewersCanEdit = false
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO {
{ OrgId : 1 , DashboardId : 2 , UserId : 1 , Permission : models . PERMISSION_EDIT } ,
}
} ) . Return ( nil )
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2020-11-13 02:52:38 -06:00
}
2022-02-07 05:43:43 -06:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" ,
"/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUpInner ( )
sc . sqlStore = mockSQLStore
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , nil , nil , dashboardService )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
assert . True ( t , dash . Meta . CanEdit )
assert . True ( t , dash . Meta . CanSave )
assert . False ( t , dash . Meta . CanAdmin )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-22 17:34:19 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling DELETE on" , "DELETE" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUpInner ( )
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = models . NewDashboard ( "test" )
} ) . Return ( nil )
dashboardService . On ( "DeleteDashboard" , mock . Anything , mock . AnythingOfType ( "int64" ) , mock . AnythingOfType ( "int64" ) ) . Return ( nil )
2022-02-07 05:43:43 -06:00
2022-05-23 10:14:27 -05:00
hs . callDeleteDashboardByUID ( t , sc , dashboardService )
2022-02-07 05:43:43 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-22 17:34:19 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" , "/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUpInner ( )
2022-02-07 05:43:43 -06:00
sc . sqlStore = mockSQLStore
2022-05-25 03:41:51 -05:00
sc . dashboardVersionService = fakeDashboardVersionService
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersion ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-22 17:34:19 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" , "/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUpInner ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-22 17:34:19 -05:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Viewer and viewers can edit" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_VIEWER
2017-12-15 07:19:49 -06:00
2022-02-07 05:43:43 -06:00
setUpInner := func ( ) {
2020-11-13 02:52:38 -06:00
origCanEdit := setting . ViewersCanEdit
t . Cleanup ( func ( ) {
setting . ViewersCanEdit = origCanEdit
2017-12-15 07:19:49 -06:00
} )
2020-11-13 02:52:38 -06:00
setting . ViewersCanEdit = true
2022-06-01 13:16:26 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO {
{ OrgId : 1 , DashboardId : 2 , UserId : 1 , Permission : models . PERMISSION_VIEW } ,
}
} ) . Return ( nil )
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2020-11-13 02:52:38 -06:00
}
2022-02-07 05:43:43 -06:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUpInner ( )
2017-12-15 07:19:49 -06:00
2020-11-13 02:52:38 -06:00
require . True ( t , setting . ViewersCanEdit )
2022-02-07 05:43:43 -06:00
sc . sqlStore = mockSQLStore
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , nil , nil , dashboardService )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
assert . True ( t , dash . Meta . CanEdit )
assert . False ( t , dash . Meta . CanSave )
assert . False ( t , dash . Meta . CanAdmin )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-12-15 07:19:49 -06:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling DELETE on" , "DELETE" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUpInner ( )
2018-01-31 09:46:31 -06:00
2022-05-17 13:52:22 -05:00
hs . callDeleteDashboardByUID ( t , sc , dashboardService )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-12-15 07:19:49 -06:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Viewer but has an admin permission" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_VIEWER
2017-06-22 17:34:19 -05:00
2022-02-07 05:43:43 -06:00
setUpInner := func ( ) {
2022-06-01 13:16:26 -05:00
origCanEdit := setting . ViewersCanEdit
t . Cleanup ( func ( ) {
setting . ViewersCanEdit = origCanEdit
} )
setting . ViewersCanEdit = true
2020-11-13 02:52:38 -06:00
2022-06-01 13:16:26 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO {
{ OrgId : 1 , DashboardId : 2 , UserId : 1 , Permission : models . PERMISSION_ADMIN } ,
}
} ) . Return ( nil )
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2017-06-22 17:34:19 -05:00
}
2022-02-07 05:43:43 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
setUpInner ( )
sc . sqlStore = mockSQLStore
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , nil , nil , dashboardService )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
assert . True ( t , dash . Meta . CanEdit )
assert . True ( t , dash . Meta . CanSave )
assert . True ( t , dash . Meta . CanAdmin )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling DELETE on" , "DELETE" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUpInner ( )
sc . sqlStore = mockSQLStore
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = models . NewDashboard ( "test" )
} ) . Return ( nil )
dashboardService . On ( "DeleteDashboard" , mock . Anything , mock . AnythingOfType ( "int64" ) , mock . AnythingOfType ( "int64" ) ) . Return ( nil )
hs . callDeleteDashboardByUID ( t , sc , dashboardService )
2018-01-29 14:23:07 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" , "/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUpInner ( )
2018-01-31 09:46:31 -06:00
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersion ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" , "/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUpInner ( )
2017-06-13 17:28:34 -05:00
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 08:48:55 -05:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "When user is an Org Editor but has a view permission" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_EDITOR
2017-06-12 16:05:32 -05:00
2022-02-07 05:43:43 -06:00
setUpInner := func ( ) {
2022-06-01 13:16:26 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO {
{ OrgId : 1 , DashboardId : 2 , UserId : 1 , Permission : models . PERMISSION_VIEW } ,
}
} ) . Return ( nil )
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2020-11-13 02:52:38 -06:00
}
2017-06-12 08:48:55 -05:00
2022-02-07 05:43:43 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
setUpInner ( )
sc . sqlStore = mockSQLStore
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , nil , nil , dashboardService )
2022-02-07 05:43:43 -06:00
2020-11-13 02:52:38 -06:00
assert . False ( t , dash . Meta . CanEdit )
assert . False ( t , dash . Meta . CanSave )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling DELETE on" , "DELETE" , "/api/dashboards/uid/abcdefghi" , "/api/dashboards/uid/:uid" , role , func ( sc * scenarioContext ) {
2022-02-07 05:43:43 -06:00
setUpInner ( )
2022-05-17 13:52:22 -05:00
hs . callDeleteDashboardByUID ( t , sc , dashboardService )
2018-01-31 09:46:31 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions/1" , "/api/dashboards/id/:dashboardId/versions/:id" , role , func ( sc * scenarioContext ) {
setUpInner ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersion ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-13 17:28:34 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/id/2/versions" , "/api/dashboards/id/:dashboardId/versions" , role , func ( sc * scenarioContext ) {
setUpInner ( )
2022-02-07 05:43:43 -06:00
hs . callGetDashboardVersions ( sc )
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 403 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2017-06-12 08:48:55 -05:00
} )
} )
2018-01-31 09:51:06 -06:00
2020-11-13 02:52:38 -06:00
t . Run ( "Given two dashboards with the same title in different folders" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
dashOne := models . NewDashboard ( "dash" )
2018-01-31 09:51:06 -06:00
dashOne . Id = 2
dashOne . FolderId = 1
dashOne . HasAcl = false
2020-03-04 05:57:20 -06:00
dashTwo := models . NewDashboard ( "dash" )
2018-01-31 09:51:06 -06:00
dashTwo . Id = 4
dashTwo . FolderId = 3
dashTwo . HasAcl = false
} )
2018-02-19 04:12:56 -06:00
2020-11-13 02:52:38 -06:00
t . Run ( "Post dashboard response tests" , func ( t * testing . T ) {
2022-03-10 11:19:50 -06:00
dashboardStore := & dashboards . FakeDashboardStore { }
2022-02-16 07:15:44 -06:00
defer dashboardStore . AssertExpectations ( t )
2018-02-19 04:12:56 -06:00
// This tests that a valid request returns correct response
2020-11-13 02:52:38 -06:00
t . Run ( "Given a correct request for creating a dashboard" , func ( t * testing . T ) {
const folderID int64 = 3
const dashID int64 = 2
2018-02-19 04:12:56 -06:00
2020-03-04 05:57:20 -06:00
cmd := models . SaveDashboardCommand {
2018-02-19 04:12:56 -06:00
OrgId : 1 ,
UserId : 5 ,
Dashboard : simplejson . NewFromAny ( map [ string ] interface { } {
"title" : "Dash" ,
} ) ,
Overwrite : true ,
2020-11-13 02:52:38 -06:00
FolderId : folderID ,
2018-02-19 04:12:56 -06:00
IsFolder : false ,
Message : "msg" ,
}
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "SaveDashboard" , mock . Anything , mock . AnythingOfType ( "*dashboards.SaveDashboardDTO" ) , mock . AnythingOfType ( "bool" ) ) .
Return ( & models . Dashboard { Id : dashID , Uid : "uid" , Title : "Dash" , Slug : "dash" , Version : 2 } , nil )
2018-02-19 04:12:56 -06:00
2022-05-23 10:14:27 -05:00
postDashboardScenario ( t , "When calling POST on" , "/api/dashboards" , "/api/dashboards" , cmd , dashboardService , nil , func ( sc * scenarioContext ) {
2020-11-13 02:52:38 -06:00
callPostDashboardShouldReturnSuccess ( sc )
2018-02-19 04:12:56 -06:00
2020-11-13 02:52:38 -06:00
result := sc . ToJSON ( )
assert . Equal ( t , "success" , result . Get ( "status" ) . MustString ( ) )
assert . Equal ( t , dashID , result . Get ( "id" ) . MustInt64 ( ) )
assert . Equal ( t , "uid" , result . Get ( "uid" ) . MustString ( ) )
assert . Equal ( t , "dash" , result . Get ( "slug" ) . MustString ( ) )
assert . Equal ( t , "/d/uid/dash" , result . Get ( "url" ) . MustString ( ) )
2018-02-19 04:12:56 -06:00
} )
} )
2021-05-26 09:20:13 -05:00
t . Run ( "Given a correct request for creating a dashboard with folder uid" , func ( t * testing . T ) {
const folderUid string = "folderUID"
const dashID int64 = 2
cmd := models . SaveDashboardCommand {
OrgId : 1 ,
UserId : 5 ,
Dashboard : simplejson . NewFromAny ( map [ string ] interface { } {
"title" : "Dash" ,
} ) ,
Overwrite : true ,
FolderUid : folderUid ,
IsFolder : false ,
Message : "msg" ,
}
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "SaveDashboard" , mock . Anything , mock . AnythingOfType ( "*dashboards.SaveDashboardDTO" ) , mock . AnythingOfType ( "bool" ) ) .
Return ( & models . Dashboard { Id : dashID , Uid : "uid" , Title : "Dash" , Slug : "dash" , Version : 2 } , nil )
2021-05-26 09:20:13 -05:00
mockFolder := & fakeFolderService {
GetFolderByUIDResult : & models . Folder { Id : 1 , Uid : "folderUID" , Title : "Folder" } ,
}
2022-05-23 10:14:27 -05:00
postDashboardScenario ( t , "When calling POST on" , "/api/dashboards" , "/api/dashboards" , cmd , dashboardService , mockFolder , func ( sc * scenarioContext ) {
2021-05-26 09:20:13 -05:00
callPostDashboardShouldReturnSuccess ( sc )
result := sc . ToJSON ( )
assert . Equal ( t , "success" , result . Get ( "status" ) . MustString ( ) )
assert . Equal ( t , dashID , result . Get ( "id" ) . MustInt64 ( ) )
assert . Equal ( t , "uid" , result . Get ( "uid" ) . MustString ( ) )
assert . Equal ( t , "dash" , result . Get ( "slug" ) . MustString ( ) )
assert . Equal ( t , "/d/uid/dash" , result . Get ( "url" ) . MustString ( ) )
} )
} )
t . Run ( "Given a request with incorrect folder uid for creating a dashboard with" , func ( t * testing . T ) {
cmd := models . SaveDashboardCommand {
OrgId : 1 ,
UserId : 5 ,
Dashboard : simplejson . NewFromAny ( map [ string ] interface { } {
"title" : "Dash" ,
} ) ,
Overwrite : true ,
2022-05-23 10:14:27 -05:00
FolderUid : "folderUID" ,
2021-05-26 09:20:13 -05:00
IsFolder : false ,
Message : "msg" ,
}
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
2021-05-26 09:20:13 -05:00
mockFolder := & fakeFolderService {
GetFolderByUIDError : errors . New ( "Error while searching Folder ID" ) ,
}
2022-05-23 10:14:27 -05:00
postDashboardScenario ( t , "When calling POST on" , "/api/dashboards" , "/api/dashboards" , cmd , dashboardService , mockFolder , func ( sc * scenarioContext ) {
2021-05-26 09:20:13 -05:00
callPostDashboard ( sc )
assert . Equal ( t , 500 , sc . resp . Code )
} )
} )
2018-02-19 04:12:56 -06:00
// This tests that invalid requests returns expected error responses
2020-11-13 02:52:38 -06:00
t . Run ( "Given incorrect requests for creating a dashboard" , func ( t * testing . T ) {
2018-02-19 04:12:56 -06:00
testCases := [ ] struct {
SaveError error
ExpectedStatusCode int
} {
2022-06-30 08:31:54 -05:00
{ SaveError : dashboards . ErrDashboardNotFound , ExpectedStatusCode : 404 } ,
{ SaveError : dashboards . ErrFolderNotFound , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardWithSameUIDExists , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardWithSameNameInFolderExists , ExpectedStatusCode : 412 } ,
{ SaveError : dashboards . ErrDashboardVersionMismatch , ExpectedStatusCode : 412 } ,
{ SaveError : dashboards . ErrDashboardTitleEmpty , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardFolderCannotHaveParent , ExpectedStatusCode : 400 } ,
2018-10-13 00:53:28 -05:00
{ SaveError : alerting . ValidationError { Reason : "Mu" } , ExpectedStatusCode : 422 } ,
2022-06-30 08:31:54 -05:00
{ SaveError : dashboards . ErrDashboardFailedGenerateUniqueUid , ExpectedStatusCode : 500 } ,
{ SaveError : dashboards . ErrDashboardTypeMismatch , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardFolderWithSameNameAsDashboard , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardWithSameNameAsFolder , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardFolderNameExists , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardUpdateAccessDenied , ExpectedStatusCode : 403 } ,
{ SaveError : dashboards . ErrDashboardInvalidUid , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardUidTooLong , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . ErrDashboardCannotSaveProvisionedDashboard , ExpectedStatusCode : 400 } ,
{ SaveError : dashboards . UpdatePluginDashboardError { PluginId : "plug" } , ExpectedStatusCode : 412 } ,
2018-02-19 04:12:56 -06:00
}
2020-03-04 05:57:20 -06:00
cmd := models . SaveDashboardCommand {
2018-02-19 04:12:56 -06:00
OrgId : 1 ,
Dashboard : simplejson . NewFromAny ( map [ string ] interface { } {
"title" : "" ,
} ) ,
}
for _ , tc := range testCases {
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "SaveDashboard" , mock . Anything , mock . AnythingOfType ( "*dashboards.SaveDashboardDTO" ) , mock . AnythingOfType ( "bool" ) ) . Return ( nil , tc . SaveError )
2018-02-19 04:12:56 -06:00
2020-11-13 02:52:38 -06:00
postDashboardScenario ( t , fmt . Sprintf ( "Expect '%s' error when calling POST on" , tc . SaveError . Error ( ) ) ,
2022-05-23 10:14:27 -05:00
"/api/dashboards" , "/api/dashboards" , cmd , dashboardService , nil , func ( sc * scenarioContext ) {
2020-11-13 02:52:38 -06:00
callPostDashboard ( sc )
assert . Equal ( t , tc . ExpectedStatusCode , sc . resp . Code )
} )
2018-02-19 04:12:56 -06:00
}
} )
} )
2018-02-27 10:53:30 -06:00
2020-11-13 02:52:38 -06:00
t . Run ( "Given two dashboards being compared" , func ( t * testing . T ) {
2022-05-25 03:41:51 -05:00
fakeDashboardVersionService := dashvertest . NewDashboardVersionServiceFake ( )
fakeDashboardVersionService . ExpectedDashboardVersions = [ ] * dashver . DashboardVersion {
{
DashboardID : 1 ,
Version : 1 ,
Data : simplejson . NewFromAny ( map [ string ] interface { } {
"title" : "Dash1" ,
} ) ,
} ,
{
DashboardID : 2 ,
Version : 2 ,
Data : simplejson . NewFromAny ( map [ string ] interface { } {
"title" : "Dash2" ,
} ) ,
} ,
}
2022-06-08 05:22:55 -05:00
sqlmock := mockstore . SQLStoreMock { }
2020-11-13 02:52:38 -06:00
setUp := func ( ) {
2022-06-01 13:16:26 -05:00
dashSvc := dashboards . NewFakeDashboardService ( t )
dashSvc . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Return ( nil )
guardian . InitLegacyGuardian ( & sqlmock , dashSvc )
2020-11-13 02:52:38 -06:00
}
2018-02-27 10:53:30 -06:00
cmd := dtos . CalculateDiffOptions {
Base : dtos . CalculateDiffTarget {
DashboardId : 1 ,
Version : 1 ,
} ,
New : dtos . CalculateDiffTarget {
DashboardId : 2 ,
Version : 2 ,
} ,
DiffType : "basic" ,
}
2020-11-13 02:52:38 -06:00
t . Run ( "when user does not have permission" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_VIEWER
2020-11-13 02:52:38 -06:00
postDiffScenario ( t , "When calling POST on" , "/api/dashboards/calculate-diff" , "/api/dashboards/calculate-diff" , cmd , role , func ( sc * scenarioContext ) {
setUp ( )
callPostDashboard ( sc )
assert . Equal ( t , 403 , sc . resp . Code )
2022-05-25 03:41:51 -05:00
} , & sqlmock , fakeDashboardVersionService )
2018-02-27 10:53:30 -06:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "when user does have permission" , func ( t * testing . T ) {
2020-03-04 05:57:20 -06:00
role := models . ROLE_ADMIN
2020-11-13 02:52:38 -06:00
postDiffScenario ( t , "When calling POST on" , "/api/dashboards/calculate-diff" , "/api/dashboards/calculate-diff" , cmd , role , func ( sc * scenarioContext ) {
2022-06-01 13:16:26 -05:00
// This test shouldn't hit GetDashboardAclInfoList, so no setup needed
2022-05-25 03:41:51 -05:00
sc . dashboardVersionService = fakeDashboardVersionService
2020-11-13 02:52:38 -06:00
callPostDashboard ( sc )
assert . Equal ( t , 200 , sc . resp . Code )
2022-05-25 03:41:51 -05:00
} , & sqlmock , fakeDashboardVersionService )
2018-02-27 10:53:30 -06:00
} )
} )
2019-03-06 07:38:40 -06:00
2020-11-13 02:52:38 -06:00
t . Run ( "Given dashboard in folder being restored should restore to folder" , func ( t * testing . T ) {
const folderID int64 = 1
2022-02-07 05:43:43 -06:00
fakeDash := models . NewDashboard ( "Child dash" )
fakeDash . Id = 2
fakeDash . FolderId = folderID
fakeDash . HasAcl = false
2019-03-06 07:38:40 -06:00
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = fakeDash
} ) . Return ( nil )
dashboardService . On ( "SaveDashboard" , mock . Anything , mock . AnythingOfType ( "*dashboards.SaveDashboardDTO" ) , mock . AnythingOfType ( "bool" ) ) . Run ( func ( args mock . Arguments ) {
cmd := args . Get ( 1 ) . ( * dashboards . SaveDashboardDTO )
cmd . Dashboard = & models . Dashboard {
Id : 2 , Uid : "uid" , Title : "Dash" , Slug : "dash" , Version : 1 ,
}
} ) . Return ( nil , nil )
2019-03-06 07:38:40 -06:00
cmd := dtos . RestoreDashboardVersionCommand {
Version : 1 ,
}
2022-05-25 03:41:51 -05:00
fakeDashboardVersionService := dashvertest . NewDashboardVersionServiceFake ( )
fakeDashboardVersionService . ExpectedDashboardVersions = [ ] * dashver . DashboardVersion {
2022-02-10 02:58:52 -06:00
{
2022-05-25 03:41:51 -05:00
DashboardID : 2 ,
2022-02-10 02:58:52 -06:00
Version : 1 ,
Data : fakeDash . Data ,
} }
2022-05-25 03:41:51 -05:00
mockSQLStore := mockstore . NewSQLStoreMock ( )
2020-11-13 02:52:38 -06:00
restoreDashboardVersionScenario ( t , "When calling POST on" , "/api/dashboards/id/1/restore" ,
2022-05-25 03:41:51 -05:00
"/api/dashboards/id/:dashboardId/restore" , dashboardService , fakeDashboardVersionService , cmd , func ( sc * scenarioContext ) {
sc . dashboardVersionService = fakeDashboardVersionService
2020-11-13 02:52:38 -06:00
callRestoreDashboardVersion ( sc )
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2019-03-06 07:38:40 -06:00
} )
2020-11-13 02:52:38 -06:00
t . Run ( "Given dashboard in general folder being restored should restore to general folder" , func ( t * testing . T ) {
2022-02-07 05:43:43 -06:00
fakeDash := models . NewDashboard ( "Child dash" )
fakeDash . Id = 2
fakeDash . HasAcl = false
2019-03-06 07:38:40 -06:00
2022-05-23 10:14:27 -05:00
dashboardService := dashboards . NewFakeDashboardService ( t )
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = fakeDash
} ) . Return ( nil )
dashboardService . On ( "SaveDashboard" , mock . Anything , mock . AnythingOfType ( "*dashboards.SaveDashboardDTO" ) , mock . AnythingOfType ( "bool" ) ) . Run ( func ( args mock . Arguments ) {
cmd := args . Get ( 1 ) . ( * dashboards . SaveDashboardDTO )
cmd . Dashboard = & models . Dashboard {
Id : 2 , Uid : "uid" , Title : "Dash" , Slug : "dash" , Version : 1 ,
}
} ) . Return ( nil , nil )
2019-03-06 07:38:40 -06:00
2022-05-25 03:41:51 -05:00
fakeDashboardVersionService := dashvertest . NewDashboardVersionServiceFake ( )
fakeDashboardVersionService . ExpectedDashboardVersions = [ ] * dashver . DashboardVersion {
{
DashboardID : 2 ,
Version : 1 ,
Data : fakeDash . Data ,
} }
2019-03-06 07:38:40 -06:00
cmd := dtos . RestoreDashboardVersionCommand {
Version : 1 ,
}
2022-02-07 05:43:43 -06:00
mockSQLStore := mockstore . NewSQLStoreMock ( )
2020-11-13 02:52:38 -06:00
restoreDashboardVersionScenario ( t , "When calling POST on" , "/api/dashboards/id/1/restore" ,
2022-05-25 03:41:51 -05:00
"/api/dashboards/id/:dashboardId/restore" , dashboardService , fakeDashboardVersionService , cmd , func ( sc * scenarioContext ) {
2020-11-13 02:52:38 -06:00
callRestoreDashboardVersion ( sc )
assert . Equal ( t , 200 , sc . resp . Code )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2019-03-06 07:38:40 -06:00
} )
2019-04-10 06:29:10 -05:00
2020-11-13 02:52:38 -06:00
t . Run ( "Given provisioned dashboard" , func ( t * testing . T ) {
2022-03-21 04:49:49 -05:00
mockSQLStore := mockstore . NewSQLStoreMock ( )
2022-06-01 13:16:26 -05:00
dashboardStore := dashboards . NewFakeDashboardStore ( t )
dashboardStore . On ( "GetProvisionedDataByDashboardID" , mock . Anything ) . Return ( & models . DashboardProvisioning { ExternalId : "/dashboard1.json" } , nil ) . Once ( )
dashboardService := dashboards . NewFakeDashboardService ( t )
2022-02-16 07:15:44 -06:00
2022-02-07 05:43:43 -06:00
dataValue , err := simplejson . NewJson ( [ ] byte ( ` { "id": 1, "editable": true, "style": "dark"} ` ) )
require . NoError ( t , err )
2022-05-23 10:14:27 -05:00
dashboardService . On ( "GetDashboard" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardQuery )
q . Result = & models . Dashboard { Id : 1 , Data : dataValue }
} ) . Return ( nil )
2022-06-01 13:16:26 -05:00
dashboardService . On ( "GetDashboardAclInfoList" , mock . Anything , mock . AnythingOfType ( "*models.GetDashboardAclInfoListQuery" ) ) . Run ( func ( args mock . Arguments ) {
q := args . Get ( 1 ) . ( * models . GetDashboardAclInfoListQuery )
q . Result = [ ] * models . DashboardAclInfoDTO { { OrgId : testOrgID , DashboardId : 1 , UserId : testUserID , Permission : models . PERMISSION_EDIT } }
} ) . Return ( nil )
guardian . InitLegacyGuardian ( mockSQLStore , dashboardService )
2019-04-30 06:32:18 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When calling GET on" , "GET" , "/api/dashboards/uid/dash" , "/api/dashboards/uid/:uid" , models . ROLE_EDITOR , func ( sc * scenarioContext ) {
2022-02-16 07:15:44 -06:00
fakeProvisioningService := provisioning . NewProvisioningServiceMock ( context . Background ( ) )
fakeProvisioningService . GetDashboardProvisionerResolvedPathFunc = func ( name string ) string {
2019-04-30 06:32:18 -05:00
return "/tmp/grafana/dashboards"
}
2022-05-17 13:52:22 -05:00
dash := getDashboardShouldReturn200WithConfig ( t , sc , fakeProvisioningService , dashboardStore , dashboardService )
2022-02-16 07:15:44 -06:00
assert . Equal ( t , "../../../dashboard1.json" , dash . Meta . ProvisionedExternalId , mockSQLStore )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2019-10-31 08:27:31 -05:00
2020-11-13 02:52:38 -06:00
loggedInUserScenarioWithRole ( t , "When allowUiUpdates is true and calling GET on" , "GET" , "/api/dashboards/uid/dash" , "/api/dashboards/uid/:uid" , models . ROLE_EDITOR , func ( sc * scenarioContext ) {
2022-02-16 07:15:44 -06:00
fakeProvisioningService := provisioning . NewProvisioningServiceMock ( context . Background ( ) )
fakeProvisioningService . GetDashboardProvisionerResolvedPathFunc = func ( name string ) string {
2019-10-31 08:27:31 -05:00
return "/tmp/grafana/dashboards"
}
2022-02-16 07:15:44 -06:00
fakeProvisioningService . GetAllowUIUpdatesFromConfigFunc = func ( name string ) bool {
2019-10-31 08:27:31 -05:00
return true
}
hs := & HTTPServer {
2022-02-16 07:15:44 -06:00
Cfg : setting . NewCfg ( ) ,
ProvisioningService : fakeProvisioningService ,
LibraryPanelService : & mockLibraryPanelService { } ,
LibraryElementService : & mockLibraryElementService { } ,
dashboardProvisioningService : mockDashboardProvisioningService { } ,
SQLStore : mockSQLStore ,
2022-04-04 09:57:43 -05:00
AccessControl : accesscontrolmock . New ( ) ,
2022-07-06 13:42:39 -05:00
DashboardService : dashboardService ,
2019-10-31 08:27:31 -05:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2022-02-07 05:43:43 -06:00
hs . callGetDashboard ( sc )
2019-10-31 08:27:31 -05:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , 200 , sc . resp . Code )
2019-10-31 08:27:31 -05:00
dash := dtos . DashboardFullWithMeta { }
err := json . NewDecoder ( sc . resp . Body ) . Decode ( & dash )
2020-11-13 02:52:38 -06:00
require . NoError ( t , err )
2019-10-31 08:27:31 -05:00
2020-11-13 02:52:38 -06:00
assert . Equal ( t , false , dash . Meta . Provisioned )
2022-02-07 05:43:43 -06:00
} , mockSQLStore )
2019-04-10 06:29:10 -05:00
} )
2017-06-12 08:48:55 -05:00
}
2017-06-12 16:05:32 -05:00
2022-05-17 13:52:22 -05:00
func getDashboardShouldReturn200WithConfig ( t * testing . T , sc * scenarioContext , provisioningService provisioning . ProvisioningService , dashboardStore dashboards . Store , dashboardService dashboards . DashboardService ) dtos . DashboardFullWithMeta {
2022-02-16 07:15:44 -06:00
t . Helper ( )
2019-04-30 06:32:18 -05:00
if provisioningService == nil {
2021-11-03 05:31:56 -05:00
provisioningService = provisioning . NewProvisioningServiceMock ( context . Background ( ) )
2019-04-30 06:32:18 -05:00
}
2022-02-16 07:15:44 -06:00
if dashboardStore == nil {
sql := sqlstore . InitTestDB ( t )
dashboardStore = database . ProvideDashboardStore ( sql )
}
2021-05-12 01:48:17 -05:00
libraryPanelsService := mockLibraryPanelService { }
libraryElementsService := mockLibraryElementService { }
2022-03-10 05:58:18 -06:00
cfg := setting . NewCfg ( )
2022-06-07 04:02:20 -05:00
ac := accesscontrolmock . New ( )
folderPermissions := accesscontrolmock . NewMockedPermissionsService ( )
dashboardPermissions := accesscontrolmock . NewMockedPermissionsService ( )
2022-03-10 05:58:18 -06:00
features := featuremgmt . WithFeatures ( )
2021-05-12 01:48:17 -05:00
2022-05-23 10:14:27 -05:00
if dashboardService == nil {
2022-06-07 04:02:20 -05:00
dashboardService = service . ProvideDashboardService (
cfg , dashboardStore , nil , features ,
folderPermissions , dashboardPermissions , ac ,
)
2022-05-23 10:14:27 -05:00
}
2019-04-30 06:32:18 -05:00
hs := & HTTPServer {
2022-03-10 05:58:18 -06:00
Cfg : cfg ,
LibraryPanelService : & libraryPanelsService ,
LibraryElementService : & libraryElementsService ,
SQLStore : sc . sqlStore ,
ProvisioningService : provisioningService ,
2022-04-04 09:57:43 -05:00
AccessControl : accesscontrolmock . New ( ) ,
2022-03-10 05:58:18 -06:00
dashboardProvisioningService : service . ProvideDashboardService (
2022-05-10 08:48:47 -05:00
cfg , dashboardStore , nil , features ,
2022-06-07 04:02:20 -05:00
folderPermissions , dashboardPermissions , ac ,
2022-03-10 05:58:18 -06:00
) ,
2022-07-06 13:42:39 -05:00
DashboardService : dashboardService ,
2019-04-30 06:32:18 -05:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2021-05-12 01:48:17 -05:00
2022-02-07 05:43:43 -06:00
hs . callGetDashboard ( sc )
2017-06-12 16:05:32 -05:00
2020-11-13 02:52:38 -06:00
require . Equal ( sc . t , 200 , sc . resp . Code )
2017-06-12 16:05:32 -05:00
dash := dtos . DashboardFullWithMeta { }
err := json . NewDecoder ( sc . resp . Body ) . Decode ( & dash )
2020-11-13 02:52:38 -06:00
require . NoError ( sc . t , err )
2017-06-12 16:05:32 -05:00
return dash
}
2022-02-07 05:43:43 -06:00
func ( hs * HTTPServer ) callGetDashboard ( sc * scenarioContext ) {
2019-04-30 06:32:18 -05:00
sc . handlerFunc = hs . GetDashboard
2018-01-29 06:51:01 -06:00
sc . fakeReqWithParams ( "GET" , sc . url , map [ string ] string { } ) . exec ( )
}
2022-02-07 05:43:43 -06:00
func ( hs * HTTPServer ) callGetDashboardVersion ( sc * scenarioContext ) {
sc . handlerFunc = hs . GetDashboardVersion
2017-06-13 17:28:34 -05:00
sc . fakeReqWithParams ( "GET" , sc . url , map [ string ] string { } ) . exec ( )
}
2022-02-07 05:43:43 -06:00
func ( hs * HTTPServer ) callGetDashboardVersions ( sc * scenarioContext ) {
sc . handlerFunc = hs . GetDashboardVersions
2017-06-13 17:28:34 -05:00
sc . fakeReqWithParams ( "GET" , sc . url , map [ string ] string { } ) . exec ( )
}
2022-02-16 07:15:44 -06:00
func ( hs * HTTPServer ) callDeleteDashboardByUID ( t * testing . T ,
sc * scenarioContext , mockDashboard * dashboards . FakeDashboardService ) {
2022-07-06 13:42:39 -05:00
hs . DashboardService = mockDashboard
2021-01-20 02:28:10 -06:00
sc . handlerFunc = hs . DeleteDashboardByUID
2018-01-31 09:46:31 -06:00
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
}
2020-11-13 02:52:38 -06:00
func callPostDashboard ( sc * scenarioContext ) {
2017-06-12 16:05:32 -05:00
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
}
2020-11-13 02:52:38 -06:00
func callRestoreDashboardVersion ( sc * scenarioContext ) {
2019-03-06 07:38:40 -06:00
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
}
2020-11-13 02:52:38 -06:00
func callPostDashboardShouldReturnSuccess ( sc * scenarioContext ) {
callPostDashboard ( sc )
2018-01-30 16:37:54 -06:00
2020-11-13 02:52:38 -06:00
assert . Equal ( sc . t , 200 , sc . resp . Code )
2018-01-30 16:37:54 -06:00
}
2022-02-16 07:15:44 -06:00
func postDashboardScenario ( t * testing . T , desc string , url string , routePattern string , cmd models . SaveDashboardCommand , dashboardService dashboards . DashboardService , folderService dashboards . FolderService , fn scenarioFunc ) {
2020-11-13 02:52:38 -06:00
t . Run ( fmt . Sprintf ( "%s %s" , desc , url ) , func ( t * testing . T ) {
2020-12-15 12:09:04 -06:00
cfg := setting . NewCfg ( )
2019-02-11 14:12:01 -06:00
hs := HTTPServer {
2020-12-15 12:09:04 -06:00
Cfg : cfg ,
2021-11-03 05:31:56 -05:00
ProvisioningService : provisioning . NewProvisioningServiceMock ( context . Background ( ) ) ,
2022-03-22 14:48:32 -05:00
Live : newTestLive ( t , sqlstore . InitTestDB ( t ) ) ,
2020-12-15 12:09:04 -06:00
QuotaService : & quota . QuotaService {
Cfg : cfg ,
} ,
2021-11-01 04:53:33 -05:00
pluginStore : & fakePluginStore { } ,
2021-05-12 01:48:17 -05:00
LibraryPanelService : & mockLibraryPanelService { } ,
LibraryElementService : & mockLibraryElementService { } ,
2022-07-06 13:42:39 -05:00
DashboardService : dashboardService ,
2022-02-16 07:15:44 -06:00
folderService : folderService ,
2022-03-03 08:05:47 -06:00
Features : featuremgmt . WithFeatures ( ) ,
2019-02-11 14:12:01 -06:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2019-02-11 14:12:01 -06:00
2020-11-13 02:52:38 -06:00
sc := setupScenarioContext ( t , url )
2021-01-15 07:43:20 -06:00
sc . defaultHandler = routing . Wrap ( func ( c * models . ReqContext ) response . Response {
2021-11-29 03:18:01 -06:00
c . Req . Body = mockRequestBody ( cmd )
2022-02-09 06:44:38 -06:00
c . Req . Header . Add ( "Content-Type" , "application/json" )
2017-06-12 16:05:32 -05:00
sc . context = c
2020-03-04 05:57:20 -06:00
sc . context . SignedInUser = & models . SignedInUser { OrgId : cmd . OrgId , UserId : cmd . UserId }
2017-06-12 16:05:32 -05:00
2021-11-29 03:18:01 -06:00
return hs . PostDashboard ( c )
2017-06-12 16:05:32 -05:00
} )
sc . m . Post ( routePattern , sc . defaultHandler )
fn ( sc )
} )
}
2018-01-29 12:27:53 -06:00
2022-05-25 03:41:51 -05:00
func postDiffScenario ( t * testing . T , desc string , url string , routePattern string , cmd dtos . CalculateDiffOptions ,
role models . RoleType , fn scenarioFunc , sqlmock sqlstore . Store , fakeDashboardVersionService * dashvertest . FakeDashboardVersionService ) {
2020-11-13 02:52:38 -06:00
t . Run ( fmt . Sprintf ( "%s %s" , desc , url ) , func ( t * testing . T ) {
2022-02-10 02:58:52 -06:00
cfg := setting . NewCfg ( )
hs := HTTPServer {
2022-05-25 03:41:51 -05:00
Cfg : cfg ,
ProvisioningService : provisioning . NewProvisioningServiceMock ( context . Background ( ) ) ,
Live : newTestLive ( t , sqlstore . InitTestDB ( t ) ) ,
QuotaService : & quota . QuotaService { Cfg : cfg } ,
LibraryPanelService : & mockLibraryPanelService { } ,
LibraryElementService : & mockLibraryElementService { } ,
SQLStore : sqlmock ,
dashboardVersionService : fakeDashboardVersionService ,
2022-02-10 02:58:52 -06:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2022-02-10 02:58:52 -06:00
2020-11-13 02:52:38 -06:00
sc := setupScenarioContext ( t , url )
2021-01-15 07:43:20 -06:00
sc . defaultHandler = routing . Wrap ( func ( c * models . ReqContext ) response . Response {
2021-11-29 03:18:01 -06:00
c . Req . Body = mockRequestBody ( cmd )
2022-02-09 06:44:38 -06:00
c . Req . Header . Add ( "Content-Type" , "application/json" )
2018-02-27 10:53:30 -06:00
sc . context = c
2020-03-04 05:57:20 -06:00
sc . context . SignedInUser = & models . SignedInUser {
2020-11-13 02:52:38 -06:00
OrgId : testOrgID ,
UserId : testUserID ,
2018-02-27 10:53:30 -06:00
}
sc . context . OrgRole = role
2022-02-10 02:58:52 -06:00
return hs . CalculateDashboardDiff ( c )
2018-02-27 10:53:30 -06:00
} )
sc . m . Post ( routePattern , sc . defaultHandler )
fn ( sc )
} )
}
2022-05-25 03:41:51 -05:00
func restoreDashboardVersionScenario ( t * testing . T , desc string , url string , routePattern string ,
mock * dashboards . FakeDashboardService , fakeDashboardVersionService * dashvertest . FakeDashboardVersionService ,
cmd dtos . RestoreDashboardVersionCommand , fn scenarioFunc , sqlStore sqlstore . Store ) {
2020-11-13 02:52:38 -06:00
t . Run ( fmt . Sprintf ( "%s %s" , desc , url ) , func ( t * testing . T ) {
2020-12-15 12:09:04 -06:00
cfg := setting . NewCfg ( )
2019-03-06 07:38:40 -06:00
hs := HTTPServer {
2022-05-25 03:41:51 -05:00
Cfg : cfg ,
ProvisioningService : provisioning . NewProvisioningServiceMock ( context . Background ( ) ) ,
Live : newTestLive ( t , sqlstore . InitTestDB ( t ) ) ,
QuotaService : & quota . QuotaService { Cfg : cfg } ,
LibraryPanelService : & mockLibraryPanelService { } ,
LibraryElementService : & mockLibraryElementService { } ,
2022-07-06 13:42:39 -05:00
DashboardService : mock ,
2022-05-25 03:41:51 -05:00
SQLStore : sqlStore ,
Features : featuremgmt . WithFeatures ( ) ,
dashboardVersionService : fakeDashboardVersionService ,
2019-03-06 07:38:40 -06:00
}
2022-06-15 08:47:04 -05:00
hs . CoremodelStaticRegistry , hs . CoremodelRegistry = setupDashboardCoremodel ( t )
2019-03-06 07:38:40 -06:00
2020-11-13 02:52:38 -06:00
sc := setupScenarioContext ( t , url )
2022-05-17 11:57:27 -05:00
sc . sqlStore = sqlStore
2022-05-25 03:41:51 -05:00
sc . dashboardVersionService = fakeDashboardVersionService
2021-01-15 07:43:20 -06:00
sc . defaultHandler = routing . Wrap ( func ( c * models . ReqContext ) response . Response {
2021-11-29 03:18:01 -06:00
c . Req . Body = mockRequestBody ( cmd )
2022-02-09 06:44:38 -06:00
c . Req . Header . Add ( "Content-Type" , "application/json" )
2019-03-06 07:38:40 -06:00
sc . context = c
2020-03-04 05:57:20 -06:00
sc . context . SignedInUser = & models . SignedInUser {
2020-11-13 02:52:38 -06:00
OrgId : testOrgID ,
UserId : testUserID ,
2019-03-06 07:38:40 -06:00
}
2020-03-04 05:57:20 -06:00
sc . context . OrgRole = models . ROLE_ADMIN
2019-03-06 07:38:40 -06:00
2021-11-29 03:18:01 -06:00
return hs . RestoreDashboardVersion ( c )
2019-03-06 07:38:40 -06:00
} )
sc . m . Post ( routePattern , sc . defaultHandler )
fn ( sc )
} )
}
2022-06-15 08:47:04 -05:00
func setupDashboardCoremodel ( t * testing . T ) ( * registry . Static , * registry . Generic ) {
2022-05-21 19:44:12 -05:00
// TODO abstract and generalize this further for wider reuse
t . Helper ( )
2022-06-15 08:47:04 -05:00
sreg , err := registry . ProvideStatic ( )
2022-05-21 19:44:12 -05:00
require . NoError ( t , err )
2022-06-15 08:47:04 -05:00
greg , err := registry . ProvideGeneric ( )
2022-05-21 19:44:12 -05:00
require . NoError ( t , err )
2022-06-15 08:47:04 -05:00
return sreg , greg
2022-05-21 19:44:12 -05:00
}
2018-03-22 16:13:46 -05:00
func ( sc * scenarioContext ) ToJSON ( ) * simplejson . Json {
2022-04-12 02:30:34 -05:00
result := simplejson . New ( )
err := json . NewDecoder ( sc . resp . Body ) . Decode ( result )
2020-11-13 02:52:38 -06:00
require . NoError ( sc . t , err )
2018-01-29 12:27:53 -06:00
return result
}
2019-10-31 08:27:31 -05:00
type mockDashboardProvisioningService struct {
2021-03-12 04:51:02 -06:00
dashboards . DashboardProvisioningService
2019-10-31 08:27:31 -05:00
}
2021-03-17 10:06:10 -05:00
func ( s mockDashboardProvisioningService ) GetProvisionedDashboardDataByDashboardID ( dashboardID int64 ) (
* models . DashboardProvisioning , error ) {
return nil , nil
2019-10-31 08:27:31 -05:00
}
2021-05-12 01:48:17 -05:00
type mockLibraryPanelService struct {
}
2021-09-27 02:04:36 -05:00
func ( m * mockLibraryPanelService ) LoadLibraryPanelsForDashboard ( c context . Context , dash * models . Dashboard ) error {
2021-05-12 01:48:17 -05:00
return nil
}
func ( m * mockLibraryPanelService ) CleanLibraryPanelsForDashboard ( dash * models . Dashboard ) error {
return nil
}
2021-09-27 02:04:36 -05:00
func ( m * mockLibraryPanelService ) ConnectLibraryPanelsForDashboard ( c context . Context , signedInUser * models . SignedInUser , dash * models . Dashboard ) error {
2021-05-12 01:48:17 -05:00
return nil
}
2022-06-29 10:18:27 -05:00
func ( m * mockLibraryPanelService ) ImportLibraryPanelsForDashboard ( c context . Context , signedInUser * models . SignedInUser , libraryPanels * simplejson . Json , panels [ ] interface { } , folderID int64 ) error {
2021-09-20 03:58:24 -05:00
return nil
}
2021-05-12 01:48:17 -05:00
type mockLibraryElementService struct {
}
2021-09-27 02:04:36 -05:00
func ( l * mockLibraryElementService ) CreateElement ( c context . Context , signedInUser * models . SignedInUser , cmd libraryelements . CreateLibraryElementCommand ) ( libraryelements . LibraryElementDTO , error ) {
2021-09-20 03:58:24 -05:00
return libraryelements . LibraryElementDTO { } , nil
}
// GetElement gets an element from a UID.
2021-09-27 02:04:36 -05:00
func ( l * mockLibraryElementService ) GetElement ( c context . Context , signedInUser * models . SignedInUser , UID string ) ( libraryelements . LibraryElementDTO , error ) {
2021-05-12 01:48:17 -05:00
return libraryelements . LibraryElementDTO { } , nil
}
// GetElementsForDashboard gets all connected elements for a specific dashboard.
2021-09-27 02:04:36 -05:00
func ( l * mockLibraryElementService ) GetElementsForDashboard ( c context . Context , dashboardID int64 ) ( map [ string ] libraryelements . LibraryElementDTO , error ) {
2021-05-12 01:48:17 -05:00
return map [ string ] libraryelements . LibraryElementDTO { } , nil
}
// ConnectElementsToDashboard connects elements to a specific dashboard.
2021-09-27 02:04:36 -05:00
func ( l * mockLibraryElementService ) ConnectElementsToDashboard ( c context . Context , signedInUser * models . SignedInUser , elementUIDs [ ] string , dashboardID int64 ) error {
2021-05-12 01:48:17 -05:00
return nil
}
// DisconnectElementsFromDashboard disconnects elements from a specific dashboard.
2021-09-27 02:04:36 -05:00
func ( l * mockLibraryElementService ) DisconnectElementsFromDashboard ( c context . Context , dashboardID int64 ) error {
2021-05-12 01:48:17 -05:00
return nil
}
// DeleteLibraryElementsInFolder deletes all elements for a specific folder.
2021-09-27 02:04:36 -05:00
func ( l * mockLibraryElementService ) DeleteLibraryElementsInFolder ( c context . Context , signedInUser * models . SignedInUser , folderUID string ) error {
2021-05-12 01:48:17 -05:00
return nil
}