2020-12-03 22:18:10 +08:00
|
|
|
package librarypanels
|
|
|
|
|
|
|
|
|
|
import (
|
2021-01-20 16:28:10 +08:00
|
|
|
"fmt"
|
|
|
|
|
|
2020-12-09 20:10:18 +08:00
|
|
|
"github.com/grafana/grafana/pkg/api/routing"
|
2021-01-20 16:28:10 +08:00
|
|
|
"github.com/grafana/grafana/pkg/components/simplejson"
|
2020-12-03 22:18:10 +08:00
|
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
2021-01-20 16:28:10 +08:00
|
|
|
"github.com/grafana/grafana/pkg/models"
|
2020-12-03 22:18:10 +08:00
|
|
|
"github.com/grafana/grafana/pkg/registry"
|
2021-05-11 13:10:19 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/libraryelements"
|
2020-12-09 20:10:18 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/sqlstore"
|
2020-12-03 22:18:10 +08:00
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// LibraryPanelService is the service for the Panel Library feature.
|
|
|
|
|
type LibraryPanelService struct {
|
2021-05-11 13:10:19 +08:00
|
|
|
Cfg *setting.Cfg `inject:""`
|
|
|
|
|
SQLStore *sqlstore.SQLStore `inject:""`
|
|
|
|
|
RouteRegister routing.RouteRegister `inject:""`
|
|
|
|
|
LibraryElementService *libraryelements.LibraryElementService `inject:""`
|
|
|
|
|
log log.Logger
|
2020-12-03 22:18:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
|
registry.RegisterService(&LibraryPanelService{})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Init initializes the LibraryPanel service
|
2020-12-09 20:10:18 +08:00
|
|
|
func (lps *LibraryPanelService) Init() error {
|
2021-05-11 13:10:19 +08:00
|
|
|
lps.log = log.New("library-panels")
|
2020-12-03 22:18:10 +08:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IsEnabled returns true if the Panel Library feature is enabled for this instance.
|
2020-12-09 20:10:18 +08:00
|
|
|
func (lps *LibraryPanelService) IsEnabled() bool {
|
|
|
|
|
if lps.Cfg == nil {
|
2020-12-03 22:18:10 +08:00
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-09 20:10:18 +08:00
|
|
|
return lps.Cfg.IsPanelLibraryEnabled()
|
2020-12-03 22:18:10 +08:00
|
|
|
}
|
|
|
|
|
|
2021-01-20 16:28:10 +08:00
|
|
|
// LoadLibraryPanelsForDashboard loops through all panels in dashboard JSON and replaces any library panel JSON
|
|
|
|
|
// with JSON stored for library panel in db.
|
2021-02-16 20:00:56 +08:00
|
|
|
func (lps *LibraryPanelService) LoadLibraryPanelsForDashboard(c *models.ReqContext, dash *models.Dashboard) error {
|
2021-01-20 16:28:10 +08:00
|
|
|
if !lps.IsEnabled() {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-11 13:10:19 +08:00
|
|
|
elements, err := lps.LibraryElementService.GetElementsForDashboard(c, dash.Id)
|
2021-01-20 16:28:10 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
panels := dash.Data.Get("panels").MustArray()
|
|
|
|
|
for i, panel := range panels {
|
|
|
|
|
panelAsJSON := simplejson.NewFromAny(panel)
|
|
|
|
|
libraryPanel := panelAsJSON.Get("libraryPanel")
|
|
|
|
|
if libraryPanel.Interface() == nil {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// we have a library panel
|
|
|
|
|
uid := libraryPanel.Get("uid").MustString()
|
|
|
|
|
if len(uid) == 0 {
|
|
|
|
|
return errLibraryPanelHeaderUIDMissing
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-11 13:10:19 +08:00
|
|
|
elementInDB, ok := elements[uid]
|
2021-01-20 16:28:10 +08:00
|
|
|
if !ok {
|
2021-01-29 13:11:13 +08:00
|
|
|
name := libraryPanel.Get("name").MustString()
|
|
|
|
|
elem := dash.Data.Get("panels").GetIndex(i)
|
|
|
|
|
elem.Set("gridPos", panelAsJSON.Get("gridPos").MustMap())
|
|
|
|
|
elem.Set("id", panelAsJSON.Get("id").MustInt64())
|
|
|
|
|
elem.Set("type", fmt.Sprintf("Name: \"%s\", UID: \"%s\"", name, uid))
|
|
|
|
|
elem.Set("libraryPanel", map[string]interface{}{
|
|
|
|
|
"uid": uid,
|
|
|
|
|
"name": name,
|
|
|
|
|
})
|
|
|
|
|
continue
|
2021-01-20 16:28:10 +08:00
|
|
|
}
|
|
|
|
|
|
2021-05-11 13:10:19 +08:00
|
|
|
if libraryelements.LibraryElementKind(elementInDB.Kind) != libraryelements.Panel {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-20 16:28:10 +08:00
|
|
|
// we have a match between what is stored in db and in dashboard json
|
2021-05-11 13:10:19 +08:00
|
|
|
libraryPanelModel, err := elementInDB.Model.MarshalJSON()
|
2021-01-20 16:28:10 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("could not marshal library panel JSON: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
libraryPanelModelAsJSON, err := simplejson.NewJson(libraryPanelModel)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("could not convert library panel to simplejson model: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// set the library panel json as the new panel json in dashboard json
|
|
|
|
|
dash.Data.Get("panels").SetIndex(i, libraryPanelModelAsJSON.Interface())
|
|
|
|
|
|
|
|
|
|
// set dashboard specific props
|
|
|
|
|
elem := dash.Data.Get("panels").GetIndex(i)
|
|
|
|
|
elem.Set("gridPos", panelAsJSON.Get("gridPos").MustMap())
|
|
|
|
|
elem.Set("id", panelAsJSON.Get("id").MustInt64())
|
|
|
|
|
elem.Set("libraryPanel", map[string]interface{}{
|
2021-05-11 13:10:19 +08:00
|
|
|
"uid": elementInDB.UID,
|
|
|
|
|
"name": elementInDB.Name,
|
|
|
|
|
"type": elementInDB.Type,
|
|
|
|
|
"description": elementInDB.Description,
|
|
|
|
|
"version": elementInDB.Version,
|
2021-02-02 13:25:35 +08:00
|
|
|
"meta": map[string]interface{}{
|
2021-05-11 13:10:19 +08:00
|
|
|
"folderName": elementInDB.Meta.FolderName,
|
|
|
|
|
"folderUid": elementInDB.Meta.FolderUID,
|
|
|
|
|
"connectedDashboards": elementInDB.Meta.Connections,
|
|
|
|
|
"created": elementInDB.Meta.Created,
|
|
|
|
|
"updated": elementInDB.Meta.Updated,
|
2021-02-02 13:25:35 +08:00
|
|
|
"createdBy": map[string]interface{}{
|
2021-05-11 13:10:19 +08:00
|
|
|
"id": elementInDB.Meta.CreatedBy.ID,
|
|
|
|
|
"name": elementInDB.Meta.CreatedBy.Name,
|
|
|
|
|
"avatarUrl": elementInDB.Meta.CreatedBy.AvatarURL,
|
2021-02-02 13:25:35 +08:00
|
|
|
},
|
|
|
|
|
"updatedBy": map[string]interface{}{
|
2021-05-11 13:10:19 +08:00
|
|
|
"id": elementInDB.Meta.UpdatedBy.ID,
|
|
|
|
|
"name": elementInDB.Meta.UpdatedBy.Name,
|
|
|
|
|
"avatarUrl": elementInDB.Meta.UpdatedBy.AvatarURL,
|
2021-02-02 13:25:35 +08:00
|
|
|
},
|
|
|
|
|
},
|
2021-01-20 16:28:10 +08:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// CleanLibraryPanelsForDashboard loops through all panels in dashboard JSON and cleans up any library panel JSON so that
|
|
|
|
|
// only the necessary JSON properties remain when storing the dashboard JSON.
|
|
|
|
|
func (lps *LibraryPanelService) CleanLibraryPanelsForDashboard(dash *models.Dashboard) error {
|
|
|
|
|
if !lps.IsEnabled() {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
panels := dash.Data.Get("panels").MustArray()
|
|
|
|
|
for i, panel := range panels {
|
|
|
|
|
panelAsJSON := simplejson.NewFromAny(panel)
|
|
|
|
|
libraryPanel := panelAsJSON.Get("libraryPanel")
|
|
|
|
|
if libraryPanel.Interface() == nil {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// we have a library panel
|
|
|
|
|
uid := libraryPanel.Get("uid").MustString()
|
|
|
|
|
if len(uid) == 0 {
|
|
|
|
|
return errLibraryPanelHeaderUIDMissing
|
|
|
|
|
}
|
|
|
|
|
name := libraryPanel.Get("name").MustString()
|
|
|
|
|
if len(name) == 0 {
|
|
|
|
|
return errLibraryPanelHeaderNameMissing
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// keep only the necessary JSON properties, the rest of the properties should be safely stored in library_panels table
|
|
|
|
|
gridPos := panelAsJSON.Get("gridPos").MustMap()
|
|
|
|
|
id := panelAsJSON.Get("id").MustInt64(int64(i))
|
|
|
|
|
dash.Data.Get("panels").SetIndex(i, map[string]interface{}{
|
|
|
|
|
"id": id,
|
|
|
|
|
"gridPos": gridPos,
|
|
|
|
|
"libraryPanel": map[string]interface{}{
|
|
|
|
|
"uid": uid,
|
|
|
|
|
"name": name,
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ConnectLibraryPanelsForDashboard loops through all panels in dashboard JSON and connects any library panels to the dashboard.
|
|
|
|
|
func (lps *LibraryPanelService) ConnectLibraryPanelsForDashboard(c *models.ReqContext, dash *models.Dashboard) error {
|
|
|
|
|
if !lps.IsEnabled() {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
panels := dash.Data.Get("panels").MustArray()
|
|
|
|
|
var libraryPanels []string
|
|
|
|
|
for _, panel := range panels {
|
|
|
|
|
panelAsJSON := simplejson.NewFromAny(panel)
|
|
|
|
|
libraryPanel := panelAsJSON.Get("libraryPanel")
|
|
|
|
|
if libraryPanel.Interface() == nil {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// we have a library panel
|
|
|
|
|
uid := libraryPanel.Get("uid").MustString()
|
|
|
|
|
if len(uid) == 0 {
|
|
|
|
|
return errLibraryPanelHeaderUIDMissing
|
|
|
|
|
}
|
|
|
|
|
libraryPanels = append(libraryPanels, uid)
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-11 13:10:19 +08:00
|
|
|
return lps.LibraryElementService.ConnectElementsToDashboard(c, libraryPanels, dash.Id)
|
2020-12-03 22:18:10 +08:00
|
|
|
}
|