Merge branch 'master' of github.com:didi/nightingale

This commit is contained in:
Ulric Qin 2021-08-21 21:52:47 +08:00
commit e288a3d3a9
3 changed files with 138 additions and 136 deletions

View File

@ -7,19 +7,20 @@ import (
"github.com/gin-contrib/gzip" "github.com/gin-contrib/gzip"
"github.com/gin-contrib/pprof" "github.com/gin-contrib/pprof"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/toolkits/csrf"
"github.com/didi/nightingale/v5/config" "github.com/didi/nightingale/v5/config"
) )
func configRoutes(r *gin.Engine) { func configRoutes(r *gin.Engine) {
csrfMid := csrf.Middleware(csrf.Options{ /*
Secret: config.Config.HTTP.CsrfSecret, csrfMid := csrf.Middleware(csrf.Options{
ErrorFunc: func(c *gin.Context) { Secret: config.Config.HTTP.CsrfSecret,
c.JSON(452, gin.H{"err": "csrf token mismatch"}) ErrorFunc: func(c *gin.Context) {
c.Abort() c.JSON(452, gin.H{"err": "csrf token mismatch"})
}, c.Abort()
}) },
})
*/
if config.Config.HTTP.Pprof { if config.Config.HTTP.Pprof {
pprof.Register(r, "/api/debug/pprof") pprof.Register(r, "/api/debug/pprof")
@ -51,12 +52,14 @@ func configRoutes(r *gin.Engine) {
} }
// for brower, expose location in nginx.conf // for brower, expose location in nginx.conf
pages := r.Group("/api/n9e", csrfMid) pages := r.Group("/api/n9e")
{ {
pages.GET("/csrf", func(c *gin.Context) { /*
renderData(c, csrf.GetToken(c), nil) pages.GET("/csrf", func(c *gin.Context) {
}) renderData(c, csrf.GetToken(c), nil)
})
*/
pages.GET("/roles", rolesGet) pages.GET("/roles", rolesGet)
pages.GET("/self/profile", selfProfileGet) pages.GET("/self/profile", selfProfileGet)
@ -83,6 +86,8 @@ func configRoutes(r *gin.Engine) {
pages.DELETE("/user-group/:id", login(), userGroupDel) pages.DELETE("/user-group/:id", login(), userGroupDel)
pages.GET("/classpaths", login(), classpathListGets) pages.GET("/classpaths", login(), classpathListGets)
pages.GET("/classpaths/tree", login(), classpathListNodeGets)
pages.GET("/classpaths/tree-node/:id", login(), classpathListNodeGetsById)
pages.POST("/classpaths", login(), classpathAdd) pages.POST("/classpaths", login(), classpathAdd)
pages.PUT("/classpath/:id", login(), classpathPut) pages.PUT("/classpath/:id", login(), classpathPut)
pages.DELETE("/classpath/:id", login(), classpathDel) pages.DELETE("/classpath/:id", login(), classpathDel)
@ -180,7 +185,7 @@ func configRoutes(r *gin.Engine) {
} }
// for brower, expose location in nginx.conf // for brower, expose location in nginx.conf
pagesV2 := r.Group("/api/n9e/v2", csrfMid) pagesV2 := r.Group("/api/n9e/v2")
{ {
pagesV2.POST("/collect-rules", login(), collectRulesAdd) pagesV2.POST("/collect-rules", login(), collectRulesAdd)
} }
@ -188,135 +193,19 @@ func configRoutes(r *gin.Engine) {
// for thirdparty, do not expose location in nginx.conf // for thirdparty, do not expose location in nginx.conf
v1 := r.Group("/v1/n9e") v1 := r.Group("/v1/n9e")
{ {
v1.GET("/roles", rolesGet)
v1.GET("/self/profile", selfProfileGet)
v1.PUT("/self/profile", selfProfilePut)
v1.PUT("/self/password", selfPasswordPut)
v1.GET("/self/token", selfTokenGets)
v1.POST("/self/token", selfTokenPost)
v1.PUT("/self/token", selfTokenPut)
v1.GET("/users", login(), userGets)
v1.POST("/users", admin(), userAddPost)
v1.GET("/user/:id/profile", login(), userProfileGet)
v1.PUT("/user/:id/profile", admin(), userProfilePut)
v1.PUT("/user/:id/status", admin(), userStatusPut)
v1.PUT("/user/:id/password", admin(), userPasswordPut)
v1.DELETE("/user/:id", admin(), userDel)
v1.GET("/user-groups", login(), userGroupListGet)
v1.GET("/user-groups/mine", login(), userGroupMineGet)
v1.POST("/user-groups", login(), userGroupAdd)
v1.PUT("/user-group/:id", login(), userGroupPut)
v1.GET("/user-group/:id", login(), userGroupGet)
v1.POST("/user-group/:id/members", login(), userGroupMemberAdd)
v1.DELETE("/user-group/:id/members", login(), userGroupMemberDel)
v1.DELETE("/user-group/:id", login(), userGroupDel)
v1.GET("/classpaths", login(), classpathListGets)
v1.POST("/classpaths", login(), classpathAdd)
v1.PUT("/classpath/:id", login(), classpathPut)
v1.DELETE("/classpath/:id", login(), classpathDel)
v1.POST("/classpath/:id/resources", login(), classpathAddResources)
v1.DELETE("/classpath/:id/resources", login(), classpathDelResources)
v1.GET("/classpath/:id/resources", login(), classpathGetsResources)
v1.GET("/classpaths/favorites", login(), classpathFavoriteGet)
v1.POST("/classpath/:id/favorites", login(), classpathFavoriteAdd)
v1.DELETE("/classpath/:id/favorites", login(), classpathFavoriteDel)
v1.GET("/resources", login(), resourcesQuery)
v1.PUT("/resources/note", resourceNotePut)
v1.PUT("/resources/tags", resourceTagsPut)
v1.PUT("/resources/classpaths", resourceClasspathsPut)
v1.PUT("/resources/mute", resourceMutePut)
v1.GET("/resource/:id", login(), resourceGet)
v1.DELETE("/resource/:id", login(), resourceDel)
v1.GET("/classpath/:id/collect-rules", login(), collectRuleGets)
v1.GET("/mutes", login(), muteGets)
v1.POST("/mutes", login(), muteAdd)
v1.GET("/mute/:id", login(), muteGet)
v1.DELETE("/mute/:id", login(), muteDel)
v1.GET("/dashboards", login(), dashboardGets)
v1.POST("/dashboards", login(), dashboardAdd)
v1.GET("/dashboard/:id", login(), dashboardGet)
v1.PUT("/dashboard/:id", login(), dashboardPut)
v1.DELETE("/dashboard/:id", login(), dashboardDel)
v1.POST("/dashboard/:id/favorites", login(), dashboardFavoriteAdd)
v1.DELETE("/dashboard/:id/favorites", login(), dashboardFavoriteDel)
v1.GET("/dashboard/:id/chart-groups", login(), chartGroupGets)
v1.POST("/dashboard/:id/chart-groups", login(), chartGroupAdd)
v1.PUT("/chart-groups", login(), chartGroupsPut)
v1.DELETE("/chart-group/:id", login(), chartGroupDel)
v1.GET("/chart-group/:id/charts", login(), chartGets)
v1.POST("/chart-group/:id/charts", login(), chartAdd)
v1.PUT("/chart/:id", login(), chartPut)
v1.DELETE("/chart/:id", login(), chartDel)
v1.PUT("/charts/configs", login(), chartConfigsPut)
v1.GET("/charts/tmps", login(), chartTmpGets)
v1.POST("/charts/tmps", login(), chartTmpAdd)
v1.GET("/alert-rule-groups", login(), alertRuleGroupGets)
v1.POST("/alert-rule-groups", login(), alertRuleGroupAdd)
v1.GET("/alert-rule-group/:id", login(), alertRuleGroupGet)
v1.PUT("/alert-rule-group/:id", login(), alertRuleGroupPut)
v1.DELETE("/alert-rule-group/:id", login(), alertRuleGroupDel)
v1.GET("/alert-rule-groups/favorites", login(), alertRuleGroupFavoriteGet)
v1.DELETE("/alert-rule-group/:id/favorites", login(), alertRuleGroupFavoriteDel)
v1.POST("/alert-rule-group/:id/favorites", login(), alertRuleGroupFavoriteAdd)
v1.GET("/alert-rule-group/:id/alert-rules", login(), alertRuleOfGroupGet)
v1.DELETE("/alert-rule-group/:id/alert-rules", login(), alertRuleOfGroupDel)
v1.POST("/alert-rules", login(), alertRuleAdd)
v1.PUT("/alert-rules/status", login(), alertRuleStatusPut)
v1.PUT("/alert-rules/notify-groups", login(), alertRuleNotifyGroupsPut)
v1.PUT("/alert-rules/notify-channels", login(), alertRuleNotifyChannelsPut)
v1.PUT("/alert-rules/append-tags", login(), alertRuleAppendTagsPut)
v1.GET("/alert-rule/:id", login(), alertRuleGet)
v1.PUT("/alert-rule/:id", login(), alertRulePut)
v1.DELETE("/alert-rule/:id", login(), alertRuleDel)
v1.GET("/alert-events", login(), alertEventGets)
v1.DELETE("/alert-events", login(), alertEventsDel)
v1.GET("/alert-event/:id", login(), alertEventGet)
v1.DELETE("/alert-event/:id", login(), alertEventDel)
v1.GET("/history-alert-events", login(), historyAlertEventGets)
v1.GET("/history-alert-event/:id", login(), historyAlertEventGet)
v1.POST("/collect-rules", login(), collectRuleAdd)
v1.DELETE("/collect-rules", login(), collectRuleDel)
v1.PUT("/collect-rule/:id", login(), collectRulePut)
v1.GET("/collect-rules-belong-to-ident", collectRuleGetsByIdent)
v1.GET("/collect-rules-summary", collectRuleSummaryGetByIdent)
v1.GET("/metric-descriptions", metricDescriptionGets)
v1.POST("/metric-descriptions", login(), metricDescriptionAdd)
v1.DELETE("/metric-descriptions", login(), metricDescriptionDel)
v1.PUT("/metric-description/:id", login(), metricDescriptionPut)
v1.GET("/contact-channels", contactChannelsGet)
v1.GET("/notify-channels", notifyChannelsGet)
v1.POST("/push", PushData)
v1.GET("/status", Status)
v1.POST("/query", GetData) v1.POST("/query", GetData)
v1.POST("/instant-query", GetDataInstant) v1.POST("/instant-query", GetDataInstant)
v1.POST("/tag-keys", GetTagKeys) v1.POST("/tag-keys", GetTagKeys)
v1.POST("/tag-values", GetTagValues) v1.POST("/tag-values", GetTagValues)
v1.POST("/tag-metrics", GetMetrics)
v1.POST("/tag-pairs", GetTagPairs) v1.POST("/tag-pairs", GetTagPairs)
v1.GET("/check-promql", checkPromeQl) v1.POST("/tag-metrics", GetMetrics)
v1.POST("/push", PushData)
v1.GET("/collect-rules-belong-to-ident", collectRuleGetsByIdent)
v1.GET("/collect-rules-summary", collectRuleSummaryGetByIdent)
v1.GET("/can-do-op-by-name", login(), canDoOpByName) v1.GET("/can-do-op-by-name", login(), canDoOpByName)
v1.GET("/can-do-op-by-token", login(), canDoOpByToken) v1.GET("/can-do-op-by-token", login(), canDoOpByToken)
} }
push := r.Group("/v1/n9e/series").Use(gzip.Gzip(gzip.DefaultCompression)) push := r.Group("/v1/n9e/series").Use(gzip.Gzip(gzip.DefaultCompression))

View File

@ -24,6 +24,21 @@ func classpathListGets(c *gin.Context) {
}, nil) }, nil)
} }
func classpathListNodeGets(c *gin.Context) {
query := queryStr(c, "query", "")
list, err := models.ClasspathNodeGets(query)
dangerous(err)
renderData(c, list, nil)
}
func classpathListNodeGetsById(c *gin.Context) {
cp := Classpath(urlParamInt64(c, "id"))
children, err := cp.DirectChildren()
renderData(c, children, err)
}
func classpathFavoriteGet(c *gin.Context) { func classpathFavoriteGet(c *gin.Context) {
lst, err := loginUser(c).FavoriteClasspaths() lst, err := loginUser(c).FavoriteClasspaths()
renderData(c, lst, err) renderData(c, lst, err)

View File

@ -19,6 +19,14 @@ type Classpath struct {
UpdateBy string `json:"update_by"` UpdateBy string `json:"update_by"`
} }
type ClasspathNode struct {
Id int64 `json:"id"`
Path string `json:"path"`
Note string `json:"note"`
Preset int `json:"preset"`
Children []*ClasspathNode `json:"children"`
}
func (c *Classpath) TableName() string { func (c *Classpath) TableName() string {
return "classpath" return "classpath"
} }
@ -104,7 +112,6 @@ func ClasspathGets(query string, limit, offset int) ([]Classpath, error) {
q := "%" + query + "%" q := "%" + query + "%"
session = session.Where("path like ?", q) session = session.Where("path like ?", q)
} }
var objs []Classpath var objs []Classpath
err := session.Find(&objs) err := session.Find(&objs)
if err != nil { if err != nil {
@ -151,7 +158,7 @@ func ClasspathGet(where string, args ...interface{}) (*Classpath, error) {
func ClasspathGetsByPrefix(prefix string) ([]Classpath, error) { func ClasspathGetsByPrefix(prefix string) ([]Classpath, error) {
var objs []Classpath var objs []Classpath
err := DB.Where("path like ?", prefix+"%").Find(&objs) err := DB.Where("path like ?", prefix+"%").OrderBy("path").Find(&objs)
if err != nil { if err != nil {
logger.Errorf("mysql.error: query classpath fail: %v", err) logger.Errorf("mysql.error: query classpath fail: %v", err)
return objs, internalServerError return objs, internalServerError
@ -218,3 +225,94 @@ func (c *Classpath) AddResources(idents []string) error {
func (c *Classpath) DelResources(idents []string) error { func (c *Classpath) DelResources(idents []string) error {
return ClasspathResourceDel(c.Id, idents) return ClasspathResourceDel(c.Id, idents)
} }
func ClasspathNodeGets(query string) ([]*ClasspathNode, error) {
session := DB.OrderBy("path")
if query != "" {
q := "%" + query + "%"
session = session.Where("path like ?", q)
}
var objs []Classpath
err := session.Find(&objs)
if err != nil {
logger.Errorf("mysql.error: query classpath fail: %v", err)
return []*ClasspathNode{}, internalServerError
}
if len(objs) == 0 {
return []*ClasspathNode{}, nil
}
pcs := ClasspathNodeAllChildren(objs)
return pcs, nil
}
func (cp *Classpath) DirectChildren() ([]Classpath, error) {
var pcs []Classpath
objs, err := ClasspathGetsByPrefix(cp.Path)
if err != nil {
logger.Errorf("mysql.error: query prefix classpath fail: %v", err)
return []Classpath{}, internalServerError
}
if len(objs) < 2 {
return []Classpath{}, nil
}
pre := objs[1]
path := pre.Path[len(objs[0].Path):]
pre.Path = path
pcs = append(pcs, pre)
for _, cp := range objs[2:] {
has := strings.HasPrefix(cp.Path, pre.Path)
if !has {
path := cp.Path[len(objs[0].Path):]
pre.Path = path
pcs = append(pcs, pre)
pre = cp
}
}
return pcs, nil
}
func ClasspathNodeAllChildren(cps []Classpath) []*ClasspathNode {
var node ClasspathNode
for _, cp := range cps {
ListInsert(cp, &node)
}
return node.Children
}
func ListInsert(obj Classpath, node *ClasspathNode) {
path := obj.Path
has := true
for {
if len(node.Children) == 0 {
break
}
children := node.Children[len(node.Children)-1]
prefix := children.Path
has = strings.HasPrefix(path, prefix)
if !has {
break
}
path = path[len(prefix):]
node = children
}
newNode := ToClasspathNode(obj, path)
node.Children = append(node.Children, &newNode)
}
func ToClasspathNode(cp Classpath, path string) ClasspathNode {
var obj ClasspathNode
obj.Id = cp.Id
obj.Path = path
obj.Note = cp.Note
obj.Preset = cp.Preset
obj.Children = []*ClasspathNode{}
return obj
}