metrics.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. package api
  2. import (
  3. "context"
  4. "encoding/json"
  5. "net/http"
  6. "github.com/grafana/grafana/pkg/api/dtos"
  7. "github.com/grafana/grafana/pkg/bus"
  8. "github.com/grafana/grafana/pkg/components/simplejson"
  9. "github.com/grafana/grafana/pkg/metrics"
  10. "github.com/grafana/grafana/pkg/middleware"
  11. "github.com/grafana/grafana/pkg/models"
  12. "github.com/grafana/grafana/pkg/tsdb"
  13. "github.com/grafana/grafana/pkg/tsdb/testdata"
  14. "github.com/grafana/grafana/pkg/util"
  15. )
  16. // POST /api/tsdb/query
  17. func QueryMetrics(c *middleware.Context, reqDto dtos.MetricRequest) Response {
  18. timeRange := tsdb.NewTimeRange(reqDto.From, reqDto.To)
  19. if len(reqDto.Queries) == 0 {
  20. return ApiError(400, "No queries found in query", nil)
  21. }
  22. dsId, err := reqDto.Queries[0].Get("datasourceId").Int64()
  23. if err != nil {
  24. return ApiError(400, "Query missing datasourceId", nil)
  25. }
  26. dsQuery := models.GetDataSourceByIdQuery{Id: dsId, OrgId: c.OrgId}
  27. if err := bus.Dispatch(&dsQuery); err != nil {
  28. return ApiError(500, "failed to fetch data source", err)
  29. }
  30. request := &tsdb.Request{TimeRange: timeRange}
  31. for _, query := range reqDto.Queries {
  32. request.Queries = append(request.Queries, &tsdb.Query{
  33. RefId: query.Get("refId").MustString("A"),
  34. MaxDataPoints: query.Get("maxDataPoints").MustInt64(100),
  35. IntervalMs: query.Get("intervalMs").MustInt64(1000),
  36. Model: query,
  37. DataSource: dsQuery.Result,
  38. })
  39. }
  40. resp, err := tsdb.HandleRequest(context.Background(), request)
  41. if err != nil {
  42. return ApiError(500, "Metric request error", err)
  43. }
  44. statusCode := 200
  45. for _, res := range resp.Results {
  46. if res.Error != nil {
  47. res.ErrorString = res.Error.Error()
  48. resp.Message = res.ErrorString
  49. statusCode = 500
  50. }
  51. }
  52. return Json(statusCode, &resp)
  53. }
  54. // GET /api/tsdb/testdata/scenarios
  55. func GetTestDataScenarios(c *middleware.Context) Response {
  56. result := make([]interface{}, 0)
  57. for _, scenario := range testdata.ScenarioRegistry {
  58. result = append(result, map[string]interface{}{
  59. "id": scenario.Id,
  60. "name": scenario.Name,
  61. "description": scenario.Description,
  62. "stringInput": scenario.StringInput,
  63. })
  64. }
  65. return Json(200, &result)
  66. }
  67. func GetInternalMetrics(c *middleware.Context) Response {
  68. if metrics.UseNilMetrics {
  69. return Json(200, util.DynMap{"message": "Metrics disabled"})
  70. }
  71. snapshots := metrics.MetricStats.GetSnapshots()
  72. resp := make(map[string]interface{})
  73. for _, m := range snapshots {
  74. metricName := m.Name() + m.StringifyTags()
  75. switch metric := m.(type) {
  76. case metrics.Gauge:
  77. resp[metricName] = map[string]interface{}{
  78. "value": metric.Value(),
  79. }
  80. case metrics.Counter:
  81. resp[metricName] = map[string]interface{}{
  82. "count": metric.Count(),
  83. }
  84. case metrics.Timer:
  85. percentiles := metric.Percentiles([]float64{0.25, 0.75, 0.90, 0.99})
  86. resp[metricName] = map[string]interface{}{
  87. "count": metric.Count(),
  88. "min": metric.Min(),
  89. "max": metric.Max(),
  90. "mean": metric.Mean(),
  91. "std": metric.StdDev(),
  92. "p25": percentiles[0],
  93. "p75": percentiles[1],
  94. "p90": percentiles[2],
  95. "p99": percentiles[3],
  96. }
  97. }
  98. }
  99. var b []byte
  100. var err error
  101. if b, err = json.MarshalIndent(resp, "", " "); err != nil {
  102. return ApiError(500, "body json marshal", err)
  103. }
  104. return &NormalResponse{
  105. body: b,
  106. status: 200,
  107. header: http.Header{
  108. "Content-Type": []string{"application/json"},
  109. },
  110. }
  111. }
  112. // Genereates a index out of range error
  113. func GenerateError(c *middleware.Context) Response {
  114. var array []string
  115. return Json(200, array[20])
  116. }
  117. // GET /api/tsdb/testdata/gensql
  118. func GenerateSqlTestData(c *middleware.Context) Response {
  119. if err := bus.Dispatch(&models.InsertSqlTestDataCommand{}); err != nil {
  120. return ApiError(500, "Failed to insert test data", err)
  121. }
  122. return Json(200, &util.DynMap{"message": "OK"})
  123. }
  124. // GET /api/tsdb/testdata/random-walk
  125. func GetTestDataRandomWalk(c *middleware.Context) Response {
  126. from := c.Query("from")
  127. to := c.Query("to")
  128. intervalMs := c.QueryInt64("intervalMs")
  129. timeRange := tsdb.NewTimeRange(from, to)
  130. request := &tsdb.Request{TimeRange: timeRange}
  131. request.Queries = append(request.Queries, &tsdb.Query{
  132. RefId: "A",
  133. IntervalMs: intervalMs,
  134. Model: simplejson.NewFromAny(&util.DynMap{
  135. "scenario": "random_walk",
  136. }),
  137. DataSource: &models.DataSource{Type: "grafana-testdata-datasource"},
  138. })
  139. resp, err := tsdb.HandleRequest(context.Background(), request)
  140. if err != nil {
  141. return ApiError(500, "Metric request error", err)
  142. }
  143. return Json(200, &resp)
  144. }