2017-12-20 17:52:21 -06:00
package api
import (
"testing"
"github.com/grafana/grafana/pkg/api/dtos"
"github.com/grafana/grafana/pkg/bus"
m "github.com/grafana/grafana/pkg/models"
"github.com/grafana/grafana/pkg/services/annotations"
. "github.com/smartystreets/goconvey/convey"
)
func TestAnnotationsApiEndpoint ( t * testing . T ) {
Convey ( "Given an annotation without a dashboard id" , t , func ( ) {
cmd := dtos . PostAnnotationsCmd {
Time : 1000 ,
Text : "annotation text" ,
Tags : [ ] string { "tag1" , "tag2" } ,
IsRegion : false ,
}
updateCmd := dtos . UpdateAnnotationsCmd {
Time : 1000 ,
Text : "annotation text" ,
Tags : [ ] string { "tag1" , "tag2" } ,
IsRegion : false ,
}
Convey ( "When user is an Org Viewer" , func ( ) {
role := m . ROLE_VIEWER
Convey ( "Should not be allowed to save an annotation" , func ( ) {
postAnnotationScenario ( "When calling POST on" , "/api/annotations" , "/api/annotations" , role , cmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
putAnnotationScenario ( "When calling PUT on" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , updateCmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "PUT" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , func ( sc * scenarioContext ) {
2018-03-22 16:13:46 -05:00
sc . handlerFunc = DeleteAnnotationByID
2017-12-20 17:52:21 -06:00
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/region/1" , "/api/annotations/region/:regionId" , role , func ( sc * scenarioContext ) {
sc . handlerFunc = DeleteAnnotationRegion
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
} )
} )
Convey ( "When user is an Org Editor" , func ( ) {
role := m . ROLE_EDITOR
Convey ( "Should be able to save an annotation" , func ( ) {
postAnnotationScenario ( "When calling POST on" , "/api/annotations" , "/api/annotations" , role , cmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
putAnnotationScenario ( "When calling PUT on" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , updateCmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "PUT" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , func ( sc * scenarioContext ) {
2018-03-22 16:13:46 -05:00
sc . handlerFunc = DeleteAnnotationByID
2017-12-20 17:52:21 -06:00
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/region/1" , "/api/annotations/region/:regionId" , role , func ( sc * scenarioContext ) {
sc . handlerFunc = DeleteAnnotationRegion
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
} )
} )
} )
Convey ( "Given an annotation with a dashboard id and the dashboard does not have an acl" , t , func ( ) {
cmd := dtos . PostAnnotationsCmd {
Time : 1000 ,
Text : "annotation text" ,
Tags : [ ] string { "tag1" , "tag2" } ,
IsRegion : false ,
DashboardId : 1 ,
PanelId : 1 ,
}
updateCmd := dtos . UpdateAnnotationsCmd {
Time : 1000 ,
Text : "annotation text" ,
Tags : [ ] string { "tag1" , "tag2" } ,
IsRegion : false ,
Id : 1 ,
}
2018-06-25 06:58:49 -05:00
deleteCmd := dtos . DeleteAnnotationsCmd {
DashboardId : 1 ,
PanelId : 1 ,
}
2017-12-20 17:52:21 -06:00
viewerRole := m . ROLE_VIEWER
editorRole := m . ROLE_EDITOR
aclMockResp := [ ] * m . DashboardAclInfoDTO {
{ Role : & viewerRole , Permission : m . PERMISSION_VIEW } ,
{ Role : & editorRole , Permission : m . PERMISSION_EDIT } ,
}
bus . AddHandler ( "test" , func ( query * m . GetDashboardAclInfoListQuery ) error {
query . Result = aclMockResp
return nil
} )
bus . AddHandler ( "test" , func ( query * m . GetTeamsByUserQuery ) error {
2018-07-11 13:23:07 -05:00
query . Result = [ ] * m . TeamDTO { }
2017-12-20 17:52:21 -06:00
return nil
} )
Convey ( "When user is an Org Viewer" , func ( ) {
role := m . ROLE_VIEWER
Convey ( "Should not be allowed to save an annotation" , func ( ) {
postAnnotationScenario ( "When calling POST on" , "/api/annotations" , "/api/annotations" , role , cmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
putAnnotationScenario ( "When calling PUT on" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , updateCmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "PUT" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , func ( sc * scenarioContext ) {
2018-03-22 16:13:46 -05:00
sc . handlerFunc = DeleteAnnotationByID
2017-12-20 17:52:21 -06:00
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/region/1" , "/api/annotations/region/:regionId" , role , func ( sc * scenarioContext ) {
sc . handlerFunc = DeleteAnnotationRegion
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 403 )
} )
} )
} )
Convey ( "When user is an Org Editor" , func ( ) {
role := m . ROLE_EDITOR
Convey ( "Should be able to save an annotation" , func ( ) {
postAnnotationScenario ( "When calling POST on" , "/api/annotations" , "/api/annotations" , role , cmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
putAnnotationScenario ( "When calling PUT on" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , updateCmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "PUT" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , func ( sc * scenarioContext ) {
2018-03-22 16:13:46 -05:00
sc . handlerFunc = DeleteAnnotationByID
2017-12-20 17:52:21 -06:00
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
loggedInUserScenarioWithRole ( "When calling DELETE on" , "DELETE" , "/api/annotations/region/1" , "/api/annotations/region/:regionId" , role , func ( sc * scenarioContext ) {
sc . handlerFunc = DeleteAnnotationRegion
sc . fakeReqWithParams ( "DELETE" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
} )
} )
2018-06-25 06:58:49 -05:00
Convey ( "When user is an Admin" , func ( ) {
role := m . ROLE_ADMIN
Convey ( "Should be able to do anything" , func ( ) {
postAnnotationScenario ( "When calling POST on" , "/api/annotations" , "/api/annotations" , role , cmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
putAnnotationScenario ( "When calling PUT on" , "/api/annotations/1" , "/api/annotations/:annotationId" , role , updateCmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "PUT" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
deleteAnnotationsScenario ( "When calling POST on" , "/api/annotations/mass-delete" , "/api/annotations/mass-delete" , role , deleteCmd , func ( sc * scenarioContext ) {
sc . fakeReqWithParams ( "POST" , sc . url , map [ string ] string { } ) . exec ( )
So ( sc . resp . Code , ShouldEqual , 200 )
} )
} )
} )
2017-12-20 17:52:21 -06:00
} )
}
type fakeAnnotationsRepo struct {
}
func ( repo * fakeAnnotationsRepo ) Delete ( params * annotations . DeleteParams ) error {
return nil
}
func ( repo * fakeAnnotationsRepo ) Save ( item * annotations . Item ) error {
item . Id = 1
return nil
}
func ( repo * fakeAnnotationsRepo ) Update ( item * annotations . Item ) error {
return nil
}
func ( repo * fakeAnnotationsRepo ) Find ( query * annotations . ItemQuery ) ( [ ] * annotations . ItemDTO , error ) {
2017-12-21 01:34:57 -06:00
annotations := [ ] * annotations . ItemDTO { { Id : 1 } }
2017-12-20 17:52:21 -06:00
return annotations , nil
}
var fakeAnnoRepo * fakeAnnotationsRepo
func postAnnotationScenario ( desc string , url string , routePattern string , role m . RoleType , cmd dtos . PostAnnotationsCmd , fn scenarioFunc ) {
Convey ( desc + " " + url , func ( ) {
defer bus . ClearBusHandlers ( )
2018-01-30 06:17:48 -06:00
sc := setupScenarioContext ( url )
2018-07-02 10:13:59 -05:00
sc . defaultHandler = Wrap ( func ( c * m . ReqContext ) Response {
2017-12-20 17:52:21 -06:00
sc . context = c
sc . context . UserId = TestUserID
sc . context . OrgId = TestOrgID
sc . context . OrgRole = role
return PostAnnotation ( c , cmd )
} )
fakeAnnoRepo = & fakeAnnotationsRepo { }
annotations . SetRepository ( fakeAnnoRepo )
sc . m . Post ( routePattern , sc . defaultHandler )
fn ( sc )
} )
}
func putAnnotationScenario ( desc string , url string , routePattern string , role m . RoleType , cmd dtos . UpdateAnnotationsCmd , fn scenarioFunc ) {
Convey ( desc + " " + url , func ( ) {
defer bus . ClearBusHandlers ( )
2018-01-30 06:17:48 -06:00
sc := setupScenarioContext ( url )
2018-07-02 10:13:59 -05:00
sc . defaultHandler = Wrap ( func ( c * m . ReqContext ) Response {
2017-12-20 17:52:21 -06:00
sc . context = c
sc . context . UserId = TestUserID
sc . context . OrgId = TestOrgID
sc . context . OrgRole = role
return UpdateAnnotation ( c , cmd )
} )
fakeAnnoRepo = & fakeAnnotationsRepo { }
annotations . SetRepository ( fakeAnnoRepo )
sc . m . Put ( routePattern , sc . defaultHandler )
fn ( sc )
} )
}
2018-06-25 06:58:49 -05:00
func deleteAnnotationsScenario ( desc string , url string , routePattern string , role m . RoleType , cmd dtos . DeleteAnnotationsCmd , fn scenarioFunc ) {
Convey ( desc + " " + url , func ( ) {
defer bus . ClearBusHandlers ( )
sc := setupScenarioContext ( url )
2018-07-02 10:13:59 -05:00
sc . defaultHandler = Wrap ( func ( c * m . ReqContext ) Response {
2018-06-25 06:58:49 -05:00
sc . context = c
sc . context . UserId = TestUserID
sc . context . OrgId = TestOrgID
sc . context . OrgRole = role
return DeleteAnnotations ( c , cmd )
} )
fakeAnnoRepo = & fakeAnnotationsRepo { }
annotations . SetRepository ( fakeAnnoRepo )
sc . m . Post ( routePattern , sc . defaultHandler )
fn ( sc )
} )
}