mysql.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. package mysql
  2. import (
  3. "context"
  4. "fmt"
  5. "sync"
  6. "github.com/go-xorm/xorm"
  7. "github.com/grafana/grafana/pkg/log"
  8. "github.com/grafana/grafana/pkg/models"
  9. "github.com/grafana/grafana/pkg/tsdb"
  10. )
  11. type MysqlExecutor struct {
  12. *models.DataSource
  13. engine *xorm.Engine
  14. log log.Logger
  15. }
  16. type engineCacheType struct {
  17. cache map[int64]*xorm.Engine
  18. versions map[int64]int
  19. sync.Mutex
  20. }
  21. var engineCache = engineCacheType{
  22. cache: make(map[int64]*xorm.Engine),
  23. versions: make(map[int64]int),
  24. }
  25. func NewMysqlExecutor(datasource *models.DataSource) (tsdb.Executor, error) {
  26. engine, err := getEngineFor(datasource)
  27. if err != nil {
  28. return nil, err
  29. }
  30. return &MysqlExecutor{
  31. log: log.New("tsdb.mysql"),
  32. engine: engine,
  33. }, nil
  34. }
  35. func getEngineFor(ds *models.DataSource) (*xorm.Engine, error) {
  36. engineCache.Lock()
  37. defer engineCache.Unlock()
  38. if engine, present := engineCache.cache[ds.Id]; present {
  39. if version, _ := engineCache.versions[ds.Id]; version == ds.Version {
  40. return engine, nil
  41. }
  42. }
  43. cnnstr := fmt.Sprintf("%s:%s@%s(%s)/%s?charset=utf8mb4", ds.User, ds.Password, "tcp", ds.Url, ds.Database)
  44. engine, err := xorm.NewEngine("mysql", cnnstr)
  45. engine.SetMaxConns(10)
  46. engine.SetMaxIdleConns(10)
  47. if err != nil {
  48. return nil, err
  49. }
  50. engineCache.cache[ds.Id] = engine
  51. return engine, nil
  52. }
  53. func init() {
  54. tsdb.RegisterExecutor("graphite", NewMysqlExecutor)
  55. }
  56. func (e *MysqlExecutor) Execute(ctx context.Context, queries tsdb.QuerySlice, context *tsdb.QueryContext) *tsdb.BatchResult {
  57. result := &tsdb.BatchResult{}
  58. session := engine.NewSession()
  59. defer session.Close()
  60. db := session.DB()
  61. result, err := getData(db, &req)
  62. if err != nil {
  63. return
  64. }
  65. }
  66. // func getData(db *core.DB, req *sqlDataRequest) (interface{}, error) {
  67. // queries := strings.Split(req.Query, ";")
  68. //
  69. // data := dataStruct{}
  70. // data.Results = make([]resultsStruct, 1)
  71. // data.Results[0].Series = make([]seriesStruct, 0)
  72. //
  73. // for i := range queries {
  74. // if queries[i] == "" {
  75. // continue
  76. // }
  77. //
  78. // rows, err := db.Query(queries[i])
  79. // if err != nil {
  80. // return nil, err
  81. // }
  82. // defer rows.Close()
  83. //
  84. // name := fmt.Sprintf("table_%d", i+1)
  85. // series, err := arrangeResult(rows, name)
  86. // if err != nil {
  87. // return nil, err
  88. // }
  89. // data.Results[0].Series = append(data.Results[0].Series, series.(seriesStruct))
  90. // }
  91. //
  92. // return data, nil
  93. // }
  94. //
  95. // func arrangeResult(rows *core.Rows, name string) (interface{}, error) {
  96. // columnNames, err := rows.Columns()
  97. //
  98. // series := seriesStruct{}
  99. // series.Columns = columnNames
  100. // series.Name = name
  101. //
  102. // for rows.Next() {
  103. // columnValues := make([]interface{}, len(columnNames))
  104. //
  105. // err = rows.ScanSlice(&columnValues)
  106. // if err != nil {
  107. // return nil, err
  108. // }
  109. //
  110. // // bytes -> string
  111. // for i := range columnValues {
  112. // switch columnValues[i].(type) {
  113. // case []byte:
  114. // columnValues[i] = fmt.Sprintf("%s", columnValues[i])
  115. // }
  116. // }
  117. //
  118. // series.Values = append(series.Values, columnValues)
  119. // }
  120. //
  121. // return series, err
  122. // }
  123. //
  124. // type sqlDataRequest struct {
  125. // Query string `json:"query"`
  126. // Body []byte `json:"-"`
  127. // }
  128. //
  129. // type seriesStruct struct {
  130. // Columns []string `json:"columns"`
  131. // Name string `json:"name"`
  132. // Values [][]interface{} `json:"values"`
  133. // }
  134. //
  135. // type resultsStruct struct {
  136. // Series []seriesStruct `json:"series"`
  137. // }
  138. //
  139. // type dataStruct struct {
  140. // Results []resultsStruct `json:"results"`
  141. // }