sqldb.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. package sqldb
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "io/ioutil"
  7. "strings"
  8. "github.com/grafana/grafana/pkg/log"
  9. "github.com/grafana/grafana/pkg/middleware"
  10. m "github.com/grafana/grafana/pkg/models"
  11. _ "github.com/go-sql-driver/mysql"
  12. "github.com/go-xorm/core"
  13. "github.com/go-xorm/xorm"
  14. _ "github.com/lib/pq"
  15. _ "github.com/mattn/go-sqlite3"
  16. )
  17. type sqlDataRequest struct {
  18. Query string `json:"query"`
  19. Body []byte `json:"-"`
  20. }
  21. type seriesStruct struct {
  22. Columns []string `json:"columns"`
  23. Name string `json:"name"`
  24. Values [][]interface{} `json:"values"`
  25. }
  26. type resultsStruct struct {
  27. Series []seriesStruct `json:"series"`
  28. }
  29. type dataStruct struct {
  30. Results []resultsStruct `json:"results"`
  31. }
  32. func getEngine(ds *m.DataSource) (*xorm.Engine, error) {
  33. dbms, err := ds.JsonData.Get("dbms").String()
  34. if err != nil {
  35. return nil, errors.New("Invalid DBMS")
  36. }
  37. host, err := ds.JsonData.Get("host").String()
  38. if err != nil {
  39. return nil, errors.New("Invalid host")
  40. }
  41. port, err := ds.JsonData.Get("port").String()
  42. if err != nil {
  43. return nil, errors.New("Invalid port")
  44. }
  45. constr := ""
  46. switch dbms {
  47. case "mysql":
  48. constr = fmt.Sprintf("%s:%s@tcp(%s:%s)/%s?charset=utf8",
  49. ds.User, ds.Password, host, port, ds.Database)
  50. case "postgres":
  51. sslEnabled, _ := ds.JsonData.Get("ssl").Bool()
  52. sslMode := "disable"
  53. if sslEnabled {
  54. sslMode = "require"
  55. }
  56. constr = fmt.Sprintf("user=%s password=%s host=%s port=%s dbname=%s sslmode=%s",
  57. ds.User, ds.Password, host, port, ds.Database, sslMode)
  58. default:
  59. return nil, fmt.Errorf("Unknown DBMS: %s", dbms)
  60. }
  61. return xorm.NewEngine(dbms, constr)
  62. }
  63. func getData(db *core.DB, req *sqlDataRequest) (interface{}, error) {
  64. queries := strings.Split(req.Query, ";")
  65. data := dataStruct{}
  66. data.Results = make([]resultsStruct, 1)
  67. data.Results[0].Series = make([]seriesStruct, 0)
  68. for i := range queries {
  69. if queries[i] == "" {
  70. continue
  71. }
  72. rows, err := db.Query(queries[i])
  73. if err != nil {
  74. return nil, err
  75. }
  76. defer rows.Close()
  77. name := fmt.Sprintf("table_%d", i+1)
  78. series, err := arrangeResult(rows, name)
  79. if err != nil {
  80. return nil, err
  81. }
  82. data.Results[0].Series = append(data.Results[0].Series, series.(seriesStruct))
  83. }
  84. return data, nil
  85. }
  86. func arrangeResult(rows *core.Rows, name string) (interface{}, error) {
  87. columnNames, err := rows.Columns()
  88. series := seriesStruct{}
  89. series.Columns = columnNames
  90. series.Name = name
  91. for rows.Next() {
  92. columnValues := make([]interface{}, len(columnNames))
  93. err = rows.ScanSlice(&columnValues)
  94. if err != nil {
  95. return nil, err
  96. }
  97. // bytes -> string
  98. for i := range columnValues {
  99. switch columnValues[i].(type) {
  100. case []byte:
  101. columnValues[i] = fmt.Sprintf("%s", columnValues[i])
  102. }
  103. }
  104. series.Values = append(series.Values, columnValues)
  105. }
  106. return series, err
  107. }
  108. func HandleRequest(c *middleware.Context, ds *m.DataSource) {
  109. var req sqlDataRequest
  110. req.Body, _ = ioutil.ReadAll(c.Req.Request.Body)
  111. json.Unmarshal(req.Body, &req)
  112. log.Debug("SQL request: query='%v'", req.Query)
  113. engine, err := getEngine(ds)
  114. if err != nil {
  115. c.JsonApiErr(500, "Unable to open SQL connection", err)
  116. return
  117. }
  118. defer engine.Close()
  119. session := engine.NewSession()
  120. defer session.Close()
  121. db := session.DB()
  122. result, err := getData(db, &req)
  123. if err != nil {
  124. c.JsonApiErr(500, fmt.Sprintf("Data error: %v, Query: %s", err.Error(), req.Query), err)
  125. return
  126. }
  127. c.JSON(200, result)
  128. }