Tool to help you manage your Grafana dashboards using Git.

client.go 3.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package grafana
  2. import (
  3. "bytes"
  4. "fmt"
  5. "io/ioutil"
  6. "net/http"
  7. "strings"
  8. )
  9. // Client implements a Grafana API client, and contains the instance's base URL
  10. // and API key, along with an HTTP client used to request the API.
  11. type Client struct {
  12. BaseURL string
  13. APIKey string
  14. httpClient *http.Client
  15. }
  16. // NewClient returns a new Grafana API client from a given base URL and API key.
  17. func NewClient(baseURL string, apiKey string) (c *Client) {
  18. // Grafana doesn't support double slashes in the API routes, so we strip the
  19. // last slash if there's one, because request() will append one anyway.
  20. if strings.HasSuffix(baseURL, "/") {
  21. baseURL = baseURL[:len(baseURL)-1]
  22. }
  23. return &Client{
  24. BaseURL: baseURL,
  25. APIKey: apiKey,
  26. httpClient: new(http.Client),
  27. }
  28. }
  29. // request preforms an HTTP request on a given endpoint, with a given method and
  30. // body. The endpoint is the Grafana API route to request, without the "/api/"
  31. // part. If the request doesn't require a body, the function has to be called
  32. // with "nil" as the "body" parameter.
  33. // Returns the response body (as a []byte containing JSON data).
  34. // Returns an error if there was an issue initialising the request, performing
  35. // it or reading the response body. Also returns an error on non-200 response
  36. // status codes. If the status code is 404, a standard error is returned, if the
  37. // status code is neither 200 nor 404 an error of type httpUnkownError is
  38. // returned.
  39. func (c *Client) request(method string, endpoint string, body []byte) ([]byte, error) {
  40. url := c.BaseURL + "/api/" + endpoint
  41. // Create the request
  42. req, err := http.NewRequest(method, url, bytes.NewBuffer(body))
  43. if err != nil {
  44. return nil, err
  45. }
  46. // Add the API key to the request as an Authorization HTTP header
  47. authHeader := fmt.Sprintf("Bearer %s", c.APIKey)
  48. req.Header.Add("Authorization", authHeader)
  49. // If the request isn't a GET, the body will be sent as JSON, so we need to
  50. // append the appropriate header
  51. if method != "GET" {
  52. req.Header.Add("Content-Type", "application/json")
  53. }
  54. // Perform the request
  55. resp, err := c.httpClient.Do(req)
  56. if err != nil {
  57. return nil, err
  58. }
  59. // Read the response body
  60. respBody, err := ioutil.ReadAll(resp.Body)
  61. if err != nil {
  62. return nil, err
  63. }
  64. // Return an error if the Grafana API responded with a non-200 status code.
  65. // We perform this here because http.Client.Do() doesn't return with an
  66. // error on non-200 status codes.
  67. if resp.StatusCode != http.StatusOK {
  68. if resp.StatusCode == http.StatusNotFound {
  69. err = fmt.Errorf("%s not found (404)", url)
  70. } else {
  71. // Return an httpUnkownError error if the status code is neither 200
  72. // nor 404
  73. err = newHttpUnknownError(resp.StatusCode)
  74. }
  75. }
  76. // Return the response body along with the error. This allows callers to
  77. // process httpUnkownError errors by displaying an error message located in
  78. // the response body along with the data contained in the error.
  79. return respBody, err
  80. }
  81. // httpUnkownError represents an HTTP error, created from an HTTP response where
  82. // the status code is neither 200 nor 404.
  83. type httpUnkownError struct {
  84. StatusCode int
  85. }
  86. // newHttpUnknownError creates and returns a new httpUnkownError error using
  87. // the provided status code.
  88. func newHttpUnknownError(statusCode int) *httpUnkownError {
  89. return &httpUnkownError{
  90. StatusCode: statusCode,
  91. }
  92. }
  93. // Error implements error.Error().
  94. func (e *httpUnkownError) Error() string {
  95. return fmt.Sprintf("Unknown HTTP error: %d", e.StatusCode)
  96. }