alertdashboard.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. // Copyright 2019 Yunion
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package models
  15. import (
  16. "context"
  17. "time"
  18. "yunion.io/x/jsonutils"
  19. "yunion.io/x/log"
  20. "yunion.io/x/pkg/errors"
  21. "yunion.io/x/pkg/util/rbacscope"
  22. "yunion.io/x/sqlchemy"
  23. "yunion.io/x/onecloud/pkg/apis/monitor"
  24. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  25. "yunion.io/x/onecloud/pkg/httperrors"
  26. "yunion.io/x/onecloud/pkg/mcclient"
  27. "yunion.io/x/onecloud/pkg/util/stringutils2"
  28. )
  29. // +onecloud:swagger-gen-model-singular=alertdashboard
  30. // +onecloud:swagger-gen-model-plural=alertdashboards
  31. type SAlertDashBoardManager struct {
  32. db.SEnabledResourceBaseManager
  33. db.SStatusStandaloneResourceBaseManager
  34. db.SScopedResourceBaseManager
  35. }
  36. type SAlertDashBoard struct {
  37. //db.SVirtualResourceBase
  38. db.SEnabledResourceBase
  39. db.SStatusStandaloneResourceBase
  40. db.SScopedResourceBase
  41. Refresh string `nullable:"false" list:"user" create:"required" update:"user"`
  42. }
  43. var AlertDashBoardManager *SAlertDashBoardManager
  44. func init() {
  45. AlertDashBoardManager = &SAlertDashBoardManager{
  46. SStatusStandaloneResourceBaseManager: db.NewStatusStandaloneResourceBaseManager(
  47. SAlertDashBoard{},
  48. "alertdashboard_tbl",
  49. "alertdashboard",
  50. "alertdashboards",
  51. ),
  52. }
  53. AlertDashBoardManager.SetVirtualObject(AlertDashBoardManager)
  54. }
  55. func (manager *SAlertDashBoardManager) NamespaceScope() rbacscope.TRbacScope {
  56. return rbacscope.ScopeSystem
  57. }
  58. func (manager *SAlertDashBoardManager) ListItemExportKeys(ctx context.Context, q *sqlchemy.SQuery, userCred mcclient.TokenCredential, keys stringutils2.SSortedStrings) (*sqlchemy.SQuery, error) {
  59. q, err := manager.SStatusStandaloneResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
  60. if err != nil {
  61. return nil, errors.Wrap(err, "SStatusStandaloneResourceBaseManager.ListItemExportKeys")
  62. }
  63. q, err = manager.SScopedResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
  64. if err != nil {
  65. return nil, errors.Wrap(err, "SScopedResourceBaseManager.ListItemExportKeys")
  66. }
  67. return q, nil
  68. }
  69. func (man *SAlertDashBoardManager) OrderByExtraFields(
  70. ctx context.Context,
  71. q *sqlchemy.SQuery,
  72. userCred mcclient.TokenCredential,
  73. input monitor.AlertDashBoardListInput,
  74. ) (*sqlchemy.SQuery, error) {
  75. var err error
  76. q, err = man.SStatusStandaloneResourceBaseManager.OrderByExtraFields(ctx, q, userCred, input.StatusStandaloneResourceListInput)
  77. if err != nil {
  78. return nil, errors.Wrap(err, "SStandaloneResourceBaseManager.OrderByExtraFields")
  79. }
  80. q, err = man.SScopedResourceBaseManager.OrderByExtraFields(ctx, q, userCred, input.ScopedResourceBaseListInput)
  81. if err != nil {
  82. return nil, errors.Wrap(err, "SScopedResourceBaseManager.OrderByExtraFields")
  83. }
  84. return q, nil
  85. }
  86. func (man *SAlertDashBoardManager) ValidateCreateData(
  87. ctx context.Context, userCred mcclient.TokenCredential,
  88. ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject,
  89. data monitor.AlertDashBoardCreateInput) (monitor.AlertDashBoardCreateInput, error) {
  90. if len(data.Refresh) == 0 {
  91. data.Refresh = "1m"
  92. }
  93. if _, err := time.ParseDuration(data.Refresh); err != nil {
  94. return data, httperrors.NewInputParameterError("Invalid refresh format: %s", data.Refresh)
  95. }
  96. generateName, err := db.GenerateName(ctx, man, ownerId, data.Name)
  97. if err != nil {
  98. return data, err
  99. }
  100. data.Name = generateName
  101. return data, nil
  102. }
  103. func (dash *SAlertDashBoard) CustomizeCreate(
  104. ctx context.Context, userCred mcclient.TokenCredential,
  105. ownerId mcclient.IIdentityProvider,
  106. query jsonutils.JSONObject,
  107. data jsonutils.JSONObject,
  108. ) error {
  109. //return dash.SScopedResourceBase.CustomizeCreate(ctx, userCred, ownerId, query, data)
  110. return nil
  111. }
  112. func (dash *SAlertDashBoard) PostCreate(ctx context.Context,
  113. userCred mcclient.TokenCredential, ownerId mcclient.IIdentityProvider,
  114. query jsonutils.JSONObject, data jsonutils.JSONObject) {
  115. _, err := dash.PerformSetScope(ctx, userCred, query, data)
  116. if err != nil {
  117. log.Errorln(errors.Wrap(err, "dash PerformSetScope"))
  118. }
  119. }
  120. func (dash *SAlertDashBoard) PerformSetScope(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
  121. domainId := jsonutils.GetAnyString(data, []string{"domain_id", "domain", "project_domain_id", "project_domain"})
  122. projectId := jsonutils.GetAnyString(data, []string{"project_id", "project"})
  123. if len(domainId) == 0 && len(projectId) == 0 {
  124. scope, _ := data.GetString("scope")
  125. if len(scope) != 0 {
  126. switch rbacscope.TRbacScope(scope) {
  127. case rbacscope.ScopeSystem:
  128. case rbacscope.ScopeDomain:
  129. domainId = userCred.GetProjectDomainId()
  130. data.(*jsonutils.JSONDict).Set("domain_id", jsonutils.NewString(domainId))
  131. case rbacscope.ScopeProject:
  132. projectId = userCred.GetProjectId()
  133. data.(*jsonutils.JSONDict).Set("project_id", jsonutils.NewString(projectId))
  134. }
  135. }
  136. }
  137. return db.PerformSetScope(ctx, dash, userCred, data)
  138. }
  139. func (man *SAlertDashBoardManager) ListItemFilter(
  140. ctx context.Context, q *sqlchemy.SQuery,
  141. userCred mcclient.TokenCredential,
  142. query monitor.AlertDashBoardListInput,
  143. ) (*sqlchemy.SQuery, error) {
  144. q, err := AlertManager.ListItemFilter(ctx, q, userCred, query.AlertListInput)
  145. if err != nil {
  146. return nil, err
  147. }
  148. return q, nil
  149. }
  150. func (man *SAlertDashBoardManager) FetchCustomizeColumns(
  151. ctx context.Context,
  152. userCred mcclient.TokenCredential,
  153. query jsonutils.JSONObject,
  154. objs []interface{},
  155. fields stringutils2.SSortedStrings,
  156. isList bool,
  157. ) []monitor.AlertDashBoardDetails {
  158. rows := make([]monitor.AlertDashBoardDetails, len(objs))
  159. alertRows := AlertManager.FetchCustomizeColumns(ctx, userCred, query, objs, fields, isList)
  160. for i := range rows {
  161. rows[i].AlertDetails = alertRows[i]
  162. rows[i], _ = objs[i].(*SAlertDashBoard).GetMoreDetails(rows[i])
  163. }
  164. return rows
  165. }
  166. func (dash *SAlertDashBoard) GetMoreDetails(out monitor.AlertDashBoardDetails) (monitor.AlertDashBoardDetails, error) {
  167. panels, err := dash.getAttachPanels()
  168. if err != nil {
  169. return out, errors.Wrapf(err, "dashboard:%s GetMoreDetails error", dash.Name)
  170. }
  171. out.AlertPanelDetails = make([]monitor.AlertPanelDetail, len(panels))
  172. for i, panel := range panels {
  173. //out.AlertPanelDetail[i].PanelDetails = monitor.PanelDetails{}
  174. out.AlertPanelDetails[i].PanelDetails, err = panel.GetMoreDetails(out.AlertPanelDetails[i].PanelDetails)
  175. if err != nil {
  176. return out, errors.Wrapf(err, "dashboard:%s get panel:%s details error", dash.Name, panel.Name)
  177. }
  178. out.AlertPanelDetails[i].Setting = panel.Settings
  179. out.AlertPanelDetails[i].PanelId = panel.Id
  180. out.AlertPanelDetails[i].PanelName = panel.Name
  181. }
  182. return out, nil
  183. }
  184. func (dash *SAlertDashBoard) getJointPanels() ([]SAlertDashboardPanel, error) {
  185. joints := make([]SAlertDashboardPanel, 0)
  186. q := AlertDashBoardPanelManager.Query().Equals(AlertDashBoardPanelManager.GetMasterFieldName(), dash.Id)
  187. err := db.FetchModelObjects(AlertDashBoardPanelManager, q, &joints)
  188. if err != nil {
  189. return joints, errors.Wrapf(err, "get dash:%s joint panel err", dash.Name)
  190. }
  191. return joints, err
  192. }
  193. func (dash *SAlertDashBoard) getAttachPanels() ([]SAlertPanel, error) {
  194. panels := make([]SAlertPanel, 0)
  195. q := AlertPanelManager.Query()
  196. sq := AlertDashBoardPanelManager.Query().Equals("dashboard_id", dash.Id).SubQuery()
  197. q = q.Join(sq, sqlchemy.Equals(q.Field("id"), sq.Field("panel_id"))).Asc(sq.Field("index")).Desc("created_at")
  198. err := db.FetchModelObjects(AlertPanelManager, q, &panels)
  199. if err != nil {
  200. return panels, errors.Wrapf(err, "dashboard:%s get attach panels error", dash.Name)
  201. }
  202. return panels, nil
  203. }
  204. func (dash *SAlertDashBoard) ValidateUpdateData(
  205. ctx context.Context,
  206. userCred mcclient.TokenCredential,
  207. query jsonutils.JSONObject,
  208. data *jsonutils.JSONDict,
  209. ) (*jsonutils.JSONDict, error) {
  210. if refresh, err := data.GetString("refresh"); err == nil && len(refresh) != 0 {
  211. if _, err := time.ParseDuration(refresh); err != nil {
  212. return data, httperrors.NewInputParameterError("Invalid refresh format: %s", refresh)
  213. }
  214. }
  215. return data, nil
  216. }
  217. func (dash *SAlertDashBoard) CustomizeDelete(
  218. ctx context.Context, userCred mcclient.TokenCredential,
  219. query jsonutils.JSONObject, data jsonutils.JSONObject) error {
  220. panels, err := dash.getAttachPanels()
  221. if err != nil {
  222. return errors.Wrapf(err, "dash:%s,when exec customizedelete to get attach panels error", dash.Name)
  223. }
  224. for _, panel := range panels {
  225. if err := panel.CustomizeDelete(ctx, userCred, query, data); err != nil {
  226. return errors.Wrap(err, "dashboard exec panel CustomizeDelete error")
  227. }
  228. if err := panel.Delete(ctx, userCred); err != nil {
  229. return errors.Wrap(err, "dashboard exec panel Delete error")
  230. }
  231. }
  232. return nil
  233. }
  234. func (manager *SAlertDashBoardManager) getDashboardByid(id string) (*SAlertDashBoard, error) {
  235. iModel, err := db.FetchById(manager, id)
  236. if err != nil {
  237. return nil, err
  238. }
  239. return iModel.(*SAlertDashBoard), nil
  240. }
  241. func (dash *SAlertDashBoard) PerformClonePanel(ctx context.Context, userCred mcclient.TokenCredential,
  242. query jsonutils.JSONObject, input monitor.AlertClonePanelInput) (jsonutils.JSONObject, error) {
  243. panel, err := AlertPanelManager.getPanelByid(input.PanelId)
  244. if err != nil {
  245. return nil, errors.Wrapf(err, "getPanelByid:%s err", input.PanelId)
  246. }
  247. panelCloned, err := panel.ClonePanel(ctx, dash.GetId(), input)
  248. if err != nil {
  249. return nil, errors.Wrap(err, "ClonePanel err")
  250. }
  251. panelDetails := monitor.PanelDetails{}
  252. panelDetails, err = panelCloned.GetMoreDetails(panelDetails)
  253. if err != nil {
  254. return nil, errors.Wrapf(err, "panelCloned:%s GetMoreDetails err", panelCloned.Id)
  255. }
  256. output := jsonutils.Marshal(panelCloned)
  257. output.(*jsonutils.JSONDict).Update(jsonutils.Marshal(&panelDetails))
  258. return output, nil
  259. }
  260. func (dash *SAlertDashBoard) PerformSetPanelOrder(
  261. ctx context.Context,
  262. userCred mcclient.TokenCredential,
  263. query jsonutils.JSONObject,
  264. input monitor.AlertPanelSetOrderInput,
  265. ) (jsonutils.JSONObject, error) {
  266. panels, err := dash.getJointPanels()
  267. if err != nil {
  268. return nil, errors.Wrapf(err, "getJointPanels")
  269. }
  270. order := map[string]int{}
  271. for _, v := range input.Order {
  272. order[v.PanelId] = v.Index
  273. }
  274. for i := range panels {
  275. panel := panels[i]
  276. index, ok := order[panel.PanelId]
  277. if ok {
  278. _, err := db.Update(&panel, func() error {
  279. panel.Index = index
  280. return nil
  281. })
  282. if err != nil {
  283. return nil, errors.Wrapf(err, "update index")
  284. }
  285. }
  286. }
  287. return nil, nil
  288. }
  289. func (dash *SAlertDashBoard) PerformCloneDashboard(ctx context.Context, userCred mcclient.TokenCredential,
  290. query jsonutils.JSONObject, input monitor.AlertCloneDashboardInput) (jsonutils.JSONObject, error) {
  291. iModel, err := db.NewModelObject(AlertDashBoardManager)
  292. if err != nil {
  293. return nil, errors.Wrap(err, "AlertDashBoardManager NewModelObject err")
  294. }
  295. dashJson := jsonutils.Marshal(dash)
  296. dashJson.Unmarshal(iModel)
  297. iModel.(*SAlertDashBoard).Id = ""
  298. name, err := db.GenerateName2(ctx, AlertDashBoardManager, nil, input.CloneName, iModel, 1)
  299. if err != nil {
  300. return nil, errors.Wrap(err, "clonePanel GenerateName err")
  301. }
  302. iModel.(*SAlertDashBoard).Name = name
  303. err = AlertDashBoardManager.TableSpec().Insert(ctx, iModel)
  304. if err != nil {
  305. return nil, errors.Wrapf(err, "insert clone dashboard err by id:%s", dash.GetId())
  306. }
  307. alertPanels, err := dash.getAttachPanels()
  308. if err != nil {
  309. return nil, errors.Wrapf(err, "dashboard:%s getAttachPanels err", dash.Id)
  310. }
  311. for _, panel := range alertPanels {
  312. newDashId := iModel.(*SAlertDashBoard).GetId()
  313. _, err := panel.ClonePanel(ctx, newDashId, monitor.AlertClonePanelInput{})
  314. if err != nil {
  315. return nil, errors.Wrapf(err, "ClonePanel %s for dashboard %s", panel.GetId(), newDashId)
  316. }
  317. }
  318. boardDetails, err := iModel.(*SAlertDashBoard).GetMoreDetails(monitor.AlertDashBoardDetails{})
  319. if err != nil {
  320. return nil, errors.Wrap(err, "GetMoreDetails of dashboard")
  321. }
  322. output := jsonutils.Marshal(iModel)
  323. output.(*jsonutils.JSONDict).Update((jsonutils.Marshal(&boardDetails)))
  324. return output, nil
  325. }