2014-12-29 06:36:08 -06:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2015-02-03 08:04:35 -06:00
|
|
|
"encoding/json"
|
|
|
|
"os"
|
|
|
|
"path"
|
2015-06-19 13:42:55 -05:00
|
|
|
"strings"
|
2015-02-03 08:04:35 -06:00
|
|
|
|
2015-02-05 03:37:13 -06:00
|
|
|
"github.com/grafana/grafana/pkg/api/dtos"
|
|
|
|
"github.com/grafana/grafana/pkg/bus"
|
2016-05-24 00:39:58 -05:00
|
|
|
"github.com/grafana/grafana/pkg/log"
|
2015-03-22 14:14:00 -05:00
|
|
|
"github.com/grafana/grafana/pkg/metrics"
|
2015-02-05 03:37:13 -06:00
|
|
|
"github.com/grafana/grafana/pkg/middleware"
|
|
|
|
m "github.com/grafana/grafana/pkg/models"
|
2016-07-08 02:35:06 -05:00
|
|
|
"github.com/grafana/grafana/pkg/plugins"
|
2015-06-05 01:15:38 -05:00
|
|
|
"github.com/grafana/grafana/pkg/services/search"
|
2015-02-05 03:37:13 -06:00
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
|
|
"github.com/grafana/grafana/pkg/util"
|
2014-12-29 06:36:08 -06:00
|
|
|
)
|
|
|
|
|
2016-02-29 12:52:57 -06:00
|
|
|
func isDashboardStarredByUser(c *middleware.Context, dashId int64) (bool, error) {
|
2015-02-02 04:32:32 -06:00
|
|
|
if !c.IsSignedIn {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
query := m.IsStarredByUserQuery{UserId: c.UserId, DashboardId: dashId}
|
|
|
|
if err := bus.Dispatch(&query); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return query.Result, nil
|
|
|
|
}
|
|
|
|
|
2014-12-29 06:36:08 -06:00
|
|
|
func GetDashboard(c *middleware.Context) {
|
2015-06-19 13:42:55 -05:00
|
|
|
slug := strings.ToLower(c.Params(":slug"))
|
2014-12-29 06:36:08 -06:00
|
|
|
|
2015-02-23 13:07:49 -06:00
|
|
|
query := m.GetDashboardQuery{Slug: slug, OrgId: c.OrgId}
|
2014-12-29 06:58:06 -06:00
|
|
|
err := bus.Dispatch(&query)
|
2014-12-29 06:36:08 -06:00
|
|
|
if err != nil {
|
|
|
|
c.JsonApiErr(404, "Dashboard not found", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-02-29 12:52:57 -06:00
|
|
|
isStarred, err := isDashboardStarredByUser(c, query.Result.Id)
|
2015-02-02 04:32:32 -06:00
|
|
|
if err != nil {
|
|
|
|
c.JsonApiErr(500, "Error while checking if dashboard was starred by user", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-01-29 05:10:34 -06:00
|
|
|
dash := query.Result
|
2015-12-18 05:38:49 -06:00
|
|
|
|
2016-01-28 11:53:19 -06:00
|
|
|
// Finding creator and last updater of the dashboard
|
2016-01-27 23:55:54 -06:00
|
|
|
updater, creator := "Anonymous", "Anonymous"
|
|
|
|
if dash.UpdatedBy > 0 {
|
2016-01-28 00:00:24 -06:00
|
|
|
updater = getUserLogin(dash.UpdatedBy)
|
|
|
|
}
|
|
|
|
if dash.CreatedBy > 0 {
|
|
|
|
creator = getUserLogin(dash.CreatedBy)
|
|
|
|
}
|
2015-12-18 05:38:49 -06:00
|
|
|
|
2015-05-04 01:36:44 -05:00
|
|
|
dto := dtos.DashboardFullWithMeta{
|
|
|
|
Dashboard: dash.Data,
|
2015-05-12 10:39:56 -05:00
|
|
|
Meta: dtos.DashboardMeta{
|
2016-02-02 01:26:11 -06:00
|
|
|
IsStarred: isStarred,
|
|
|
|
Slug: slug,
|
|
|
|
Type: m.DashTypeDB,
|
|
|
|
CanStar: c.IsSignedIn,
|
|
|
|
CanSave: c.OrgRole == m.ROLE_ADMIN || c.OrgRole == m.ROLE_EDITOR,
|
|
|
|
CanEdit: canEditDashboard(c.OrgRole),
|
|
|
|
Created: dash.Created,
|
|
|
|
Updated: dash.Updated,
|
|
|
|
UpdatedBy: updater,
|
|
|
|
CreatedBy: creator,
|
|
|
|
Version: dash.Version,
|
2015-05-12 10:39:56 -05:00
|
|
|
},
|
2015-01-29 05:10:34 -06:00
|
|
|
}
|
2014-12-29 06:36:08 -06:00
|
|
|
|
2016-06-03 10:00:39 -05:00
|
|
|
c.TimeRequest(metrics.M_Api_Dashboard_Get)
|
2015-01-29 05:10:34 -06:00
|
|
|
c.JSON(200, dto)
|
2014-12-29 06:36:08 -06:00
|
|
|
}
|
|
|
|
|
2016-01-27 23:55:54 -06:00
|
|
|
func getUserLogin(userId int64) string {
|
2016-01-28 00:00:24 -06:00
|
|
|
query := m.GetUserByIdQuery{Id: userId}
|
|
|
|
err := bus.Dispatch(&query)
|
|
|
|
if err != nil {
|
|
|
|
return "Anonymous"
|
|
|
|
} else {
|
|
|
|
user := query.Result
|
|
|
|
return user.Login
|
|
|
|
}
|
2016-01-27 23:55:54 -06:00
|
|
|
}
|
|
|
|
|
2014-12-29 06:36:08 -06:00
|
|
|
func DeleteDashboard(c *middleware.Context) {
|
|
|
|
slug := c.Params(":slug")
|
|
|
|
|
2015-02-23 13:07:49 -06:00
|
|
|
query := m.GetDashboardQuery{Slug: slug, OrgId: c.OrgId}
|
2015-01-08 02:00:00 -06:00
|
|
|
if err := bus.Dispatch(&query); err != nil {
|
2014-12-29 06:36:08 -06:00
|
|
|
c.JsonApiErr(404, "Dashboard not found", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-02-23 13:07:49 -06:00
|
|
|
cmd := m.DeleteDashboardCommand{Slug: slug, OrgId: c.OrgId}
|
2015-01-08 02:00:00 -06:00
|
|
|
if err := bus.Dispatch(&cmd); err != nil {
|
2014-12-29 06:36:08 -06:00
|
|
|
c.JsonApiErr(500, "Failed to delete dashboard", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-12-29 06:58:06 -06:00
|
|
|
var resp = map[string]interface{}{"title": query.Result.Title}
|
2014-12-29 06:36:08 -06:00
|
|
|
|
|
|
|
c.JSON(200, resp)
|
|
|
|
}
|
|
|
|
|
2016-07-08 02:35:06 -05:00
|
|
|
func PostDashboard(c *middleware.Context, cmd m.SaveDashboardCommand) Response {
|
2015-02-23 13:07:49 -06:00
|
|
|
cmd.OrgId = c.OrgId
|
2015-12-18 05:38:49 -06:00
|
|
|
|
|
|
|
if !c.IsSignedIn {
|
2016-01-27 23:55:54 -06:00
|
|
|
cmd.UserId = -1
|
2015-12-18 05:38:49 -06:00
|
|
|
} else {
|
2016-01-28 00:00:24 -06:00
|
|
|
cmd.UserId = c.UserId
|
|
|
|
}
|
2014-12-29 06:36:08 -06:00
|
|
|
|
2015-07-20 09:38:25 -05:00
|
|
|
dash := cmd.GetDashboardModel()
|
|
|
|
if dash.Id == 0 {
|
2015-09-11 10:17:10 -05:00
|
|
|
limitReached, err := middleware.QuotaReached(c, "dashboard")
|
2015-07-20 09:38:25 -05:00
|
|
|
if err != nil {
|
2016-07-08 02:35:06 -05:00
|
|
|
return ApiError(500, "failed to get quota", err)
|
2015-07-20 09:38:25 -05:00
|
|
|
}
|
|
|
|
if limitReached {
|
2016-07-08 02:35:06 -05:00
|
|
|
return ApiError(403, "Quota reached", nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !cmd.Overwrite {
|
|
|
|
if autoUpdate, exists := dash.Data.CheckGet("autoUpdate"); exists {
|
|
|
|
message := "Dashboard marked as auto updated."
|
|
|
|
|
|
|
|
if pluginId, err := autoUpdate.Get("pluginId").String(); err == nil {
|
|
|
|
if pluginDef, ok := plugins.Plugins[pluginId]; ok {
|
|
|
|
message = "Dashboard updated automatically when plugin " + pluginDef.Name + " is updated."
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Json(412, util.DynMap{"status": "auto-update-dashboard", "message": message})
|
2015-07-20 09:38:25 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-29 06:36:08 -06:00
|
|
|
err := bus.Dispatch(&cmd)
|
|
|
|
if err != nil {
|
2015-01-05 10:04:29 -06:00
|
|
|
if err == m.ErrDashboardWithSameNameExists {
|
2016-07-08 02:35:06 -05:00
|
|
|
return Json(412, util.DynMap{"status": "name-exists", "message": err.Error()})
|
2015-03-02 15:24:01 -06:00
|
|
|
}
|
|
|
|
if err == m.ErrDashboardVersionMismatch {
|
2016-07-08 02:35:06 -05:00
|
|
|
return Json(412, util.DynMap{"status": "version-mismatch", "message": err.Error()})
|
2015-01-05 10:04:29 -06:00
|
|
|
}
|
2015-05-04 00:46:46 -05:00
|
|
|
if err == m.ErrDashboardNotFound {
|
2016-07-08 02:35:06 -05:00
|
|
|
return Json(404, util.DynMap{"status": "not-found", "message": err.Error()})
|
2015-05-04 00:46:46 -05:00
|
|
|
}
|
2016-07-08 02:35:06 -05:00
|
|
|
return ApiError(500, "Failed to save dashboard", err)
|
2014-12-29 06:36:08 -06:00
|
|
|
}
|
|
|
|
|
2016-06-03 10:00:39 -05:00
|
|
|
c.TimeRequest(metrics.M_Api_Dashboard_Save)
|
2016-07-08 02:35:06 -05:00
|
|
|
return Json(200, util.DynMap{"status": "success", "slug": cmd.Result.Slug, "version": cmd.Result.Version})
|
2014-12-29 06:36:08 -06:00
|
|
|
}
|
2015-02-03 08:04:35 -06:00
|
|
|
|
2015-10-26 12:54:32 -05:00
|
|
|
func canEditDashboard(role m.RoleType) bool {
|
|
|
|
return role == m.ROLE_ADMIN || role == m.ROLE_EDITOR || role == m.ROLE_READ_ONLY_EDITOR
|
|
|
|
}
|
|
|
|
|
2016-05-25 02:56:45 -05:00
|
|
|
func GetHomeDashboard(c *middleware.Context) Response {
|
2016-04-02 15:54:06 -05:00
|
|
|
prefsQuery := m.GetPreferencesWithDefaultsQuery{OrgId: c.OrgId, UserId: c.UserId}
|
|
|
|
if err := bus.Dispatch(&prefsQuery); err != nil {
|
2016-05-25 02:56:45 -05:00
|
|
|
return ApiError(500, "Failed to get preferences", err)
|
2016-03-17 04:29:34 -05:00
|
|
|
}
|
|
|
|
|
2016-04-02 15:54:06 -05:00
|
|
|
if prefsQuery.Result.HomeDashboardId != 0 {
|
|
|
|
slugQuery := m.GetDashboardSlugByIdQuery{Id: prefsQuery.Result.HomeDashboardId}
|
|
|
|
err := bus.Dispatch(&slugQuery)
|
2016-05-24 00:39:58 -05:00
|
|
|
if err == nil {
|
|
|
|
dashRedirect := dtos.DashboardRedirect{RedirectUri: "db/" + slugQuery.Result}
|
2016-05-25 02:56:45 -05:00
|
|
|
return Json(200, &dashRedirect)
|
2016-05-24 00:39:58 -05:00
|
|
|
} else {
|
|
|
|
log.Warn("Failed to get slug from database, %s", err.Error())
|
2016-03-17 04:29:34 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-03 15:06:07 -06:00
|
|
|
filePath := path.Join(setting.StaticRootPath, "dashboards/home.json")
|
2015-02-03 08:04:35 -06:00
|
|
|
file, err := os.Open(filePath)
|
|
|
|
if err != nil {
|
2016-05-25 02:56:45 -05:00
|
|
|
return ApiError(500, "Failed to load home dashboard", err)
|
2015-02-03 08:04:35 -06:00
|
|
|
}
|
|
|
|
|
2015-05-04 01:36:44 -05:00
|
|
|
dash := dtos.DashboardFullWithMeta{}
|
2015-02-03 08:04:35 -06:00
|
|
|
dash.Meta.IsHome = true
|
2015-10-26 12:54:32 -05:00
|
|
|
dash.Meta.CanEdit = canEditDashboard(c.OrgRole)
|
2015-02-03 08:04:35 -06:00
|
|
|
jsonParser := json.NewDecoder(file)
|
2015-05-04 01:36:44 -05:00
|
|
|
if err := jsonParser.Decode(&dash.Dashboard); err != nil {
|
2016-05-25 02:56:45 -05:00
|
|
|
return ApiError(500, "Failed to load home dashboard", err)
|
2015-02-03 08:04:35 -06:00
|
|
|
}
|
|
|
|
|
2016-05-25 02:56:45 -05:00
|
|
|
return Json(200, &dash)
|
2015-02-03 08:04:35 -06:00
|
|
|
}
|
2015-05-12 07:11:30 -05:00
|
|
|
|
|
|
|
func GetDashboardFromJsonFile(c *middleware.Context) {
|
|
|
|
file := c.Params(":file")
|
|
|
|
|
|
|
|
dashboard := search.GetDashboardFromJsonIndex(file)
|
|
|
|
if dashboard == nil {
|
|
|
|
c.JsonApiErr(404, "Dashboard not found", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
dash := dtos.DashboardFullWithMeta{Dashboard: dashboard.Data}
|
|
|
|
dash.Meta.Type = m.DashTypeJson
|
2015-10-26 12:54:32 -05:00
|
|
|
dash.Meta.CanEdit = canEditDashboard(c.OrgRole)
|
2015-05-12 07:11:30 -05:00
|
|
|
|
|
|
|
c.JSON(200, &dash)
|
|
|
|
}
|
2015-05-13 03:45:53 -05:00
|
|
|
|
|
|
|
func GetDashboardTags(c *middleware.Context) {
|
|
|
|
query := m.GetDashboardTagsQuery{OrgId: c.OrgId}
|
|
|
|
err := bus.Dispatch(&query)
|
|
|
|
if err != nil {
|
|
|
|
c.JsonApiErr(500, "Failed to get tags from database", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.JSON(200, query.Result)
|
|
|
|
}
|