| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 |
- package sqlstore
- import (
- "fmt"
- "time"
- "github.com/grafana/grafana/pkg/bus"
- m "github.com/grafana/grafana/pkg/models"
- )
- func init() {
- bus.AddHandler("sql", SetDashboardAcl)
- bus.AddHandler("sql", UpdateDashboardAcl)
- bus.AddHandler("sql", RemoveDashboardAcl)
- bus.AddHandler("sql", GetDashboardAclInfoList)
- }
- func UpdateDashboardAcl(cmd *m.UpdateDashboardAclCommand) error {
- return inTransaction(func(sess *DBSession) error {
- // delete existing items
- _, err := sess.Exec("DELETE FROM dashboard_acl WHERE dashboard_id=?", cmd.DashboardId)
- if err != nil {
- return err
- }
- for _, item := range cmd.Items {
- if item.UserId == 0 && item.TeamId == 0 && !item.Role.IsValid() {
- return m.ErrDashboardAclInfoMissing
- }
- if item.DashboardId == 0 {
- return m.ErrDashboardPermissionDashboardEmpty
- }
- sess.Nullable("user_id", "team_id")
- if _, err := sess.Insert(item); err != nil {
- return err
- }
- }
- // Update dashboard HasAcl flag
- dashboard := m.Dashboard{HasAcl: true}
- if _, err := sess.Cols("has_acl").Where("id=? OR folder_id=?", cmd.DashboardId, cmd.DashboardId).Update(&dashboard); err != nil {
- return err
- }
- return nil
- })
- }
- func SetDashboardAcl(cmd *m.SetDashboardAclCommand) error {
- return inTransaction(func(sess *DBSession) error {
- if cmd.UserId == 0 && cmd.TeamId == 0 {
- return m.ErrDashboardAclInfoMissing
- }
- if cmd.DashboardId == 0 {
- return m.ErrDashboardPermissionDashboardEmpty
- }
- if res, err := sess.Query("SELECT 1 from "+dialect.Quote("dashboard_acl")+" WHERE dashboard_id =? and (team_id=? or user_id=?)", cmd.DashboardId, cmd.TeamId, cmd.UserId); err != nil {
- return err
- } else if len(res) == 1 {
- entity := m.DashboardAcl{
- Permission: cmd.Permission,
- Updated: time.Now(),
- }
- if _, err := sess.Cols("updated", "permission").Where("dashboard_id =? and (team_id=? or user_id=?)", cmd.DashboardId, cmd.TeamId, cmd.UserId).Update(&entity); err != nil {
- return err
- }
- return nil
- }
- entity := m.DashboardAcl{
- OrgId: cmd.OrgId,
- TeamId: cmd.TeamId,
- UserId: cmd.UserId,
- Created: time.Now(),
- Updated: time.Now(),
- DashboardId: cmd.DashboardId,
- Permission: cmd.Permission,
- }
- cols := []string{"org_id", "created", "updated", "dashboard_id", "permission"}
- if cmd.UserId != 0 {
- cols = append(cols, "user_id")
- }
- if cmd.TeamId != 0 {
- cols = append(cols, "team_id")
- }
- _, err := sess.Cols(cols...).Insert(&entity)
- if err != nil {
- return err
- }
- cmd.Result = entity
- // Update dashboard HasAcl flag
- dashboard := m.Dashboard{
- HasAcl: true,
- }
- if _, err := sess.Cols("has_acl").Where("id=? OR folder_id=?", cmd.DashboardId, cmd.DashboardId).Update(&dashboard); err != nil {
- return err
- }
- return nil
- })
- }
- // RemoveDashboardAcl removes a specified permission from the dashboard acl
- func RemoveDashboardAcl(cmd *m.RemoveDashboardAclCommand) error {
- return inTransaction(func(sess *DBSession) error {
- var rawSQL = "DELETE FROM " + dialect.Quote("dashboard_acl") + " WHERE org_id =? and id=?"
- _, err := sess.Exec(rawSQL, cmd.OrgId, cmd.AclId)
- if err != nil {
- return err
- }
- return err
- })
- }
- // GetDashboardAclInfoList returns a list of permissions for a dashboard. They can be fetched from three
- // different places.
- // 1) Permissions for the dashboard
- // 2) permissions for its parent folder
- // 3) if no specific permissions have been set for the dashboard or its parent folder then get the default permissions
- func GetDashboardAclInfoList(query *m.GetDashboardAclInfoListQuery) error {
- var err error
- if query.DashboardId == 0 {
- sql := `SELECT
- da.id,
- da.org_id,
- da.dashboard_id,
- da.user_id,
- da.team_id,
- da.permission,
- da.role,
- da.created,
- da.updated,
- '' as user_login,
- '' as user_email,
- '' as team,
- '' as title,
- '' as slug,
- '' as uid,` +
- dialect.BooleanStr(false) + ` AS is_folder
- FROM dashboard_acl as da
- WHERE da.dashboard_id = -1`
- query.Result = make([]*m.DashboardAclInfoDTO, 0)
- err = x.SQL(sql).Find(&query.Result)
- } else {
- dashboardFilter := fmt.Sprintf(`IN (
- SELECT %d
- UNION
- SELECT folder_id from dashboard where id = %d
- )`, query.DashboardId, query.DashboardId)
- rawSQL := `
- -- get permissions for the dashboard and its parent folder
- SELECT
- da.id,
- da.org_id,
- da.dashboard_id,
- da.user_id,
- da.team_id,
- da.permission,
- da.role,
- da.created,
- da.updated,
- u.login AS user_login,
- u.email AS user_email,
- ug.name AS team,
- d.title,
- d.slug,
- d.uid,
- d.is_folder
- FROM` + dialect.Quote("dashboard_acl") + ` as da
- LEFT OUTER JOIN ` + dialect.Quote("user") + ` AS u ON u.id = da.user_id
- LEFT OUTER JOIN team ug on ug.id = da.team_id
- LEFT OUTER JOIN dashboard d on da.dashboard_id = d.id
- WHERE dashboard_id ` + dashboardFilter + ` AND da.org_id = ?
- -- Also include default permissions if folder or dashboard field "has_acl" is false
- UNION
- SELECT
- da.id,
- da.org_id,
- da.dashboard_id,
- da.user_id,
- da.team_id,
- da.permission,
- da.role,
- da.created,
- da.updated,
- '' as user_login,
- '' as user_email,
- '' as team,
- folder.title,
- folder.slug,
- folder.uid,
- folder.is_folder
- FROM dashboard_acl as da,
- dashboard as dash
- LEFT OUTER JOIN dashboard folder on dash.folder_id = folder.id
- WHERE
- dash.id = ? AND (
- dash.has_acl = ` + dialect.BooleanStr(false) + ` or
- folder.has_acl = ` + dialect.BooleanStr(false) + `
- ) AND
- da.dashboard_id = -1
- ORDER BY 1 ASC
- `
- query.Result = make([]*m.DashboardAclInfoDTO, 0)
- err = x.SQL(rawSQL, query.OrgId, query.DashboardId).Find(&query.Result)
- }
- for _, p := range query.Result {
- p.PermissionName = p.Permission.String()
- }
- return err
- }
|