request.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. // Copyright 2018 Frédéric Guillot. All rights reserved.
  2. // Use of this source code is governed by the Apache 2.0
  3. // license that can be found in the LICENSE file.
  4. package client // import "miniflux.app/client"
  5. import (
  6. "bytes"
  7. "encoding/json"
  8. "errors"
  9. "fmt"
  10. "io"
  11. "io/ioutil"
  12. "log"
  13. "net/http"
  14. "net/url"
  15. "time"
  16. )
  17. const (
  18. userAgent = "Miniflux Client Library"
  19. defaultTimeout = 80
  20. )
  21. // List of exposed errors.
  22. var (
  23. ErrNotAuthorized = errors.New("miniflux: unauthorized (bad credentials)")
  24. ErrForbidden = errors.New("miniflux: access forbidden")
  25. ErrServerError = errors.New("miniflux: internal server error")
  26. ErrNotFound = errors.New("miniflux: resource not found")
  27. )
  28. type errorResponse struct {
  29. ErrorMessage string `json:"error_message"`
  30. }
  31. type request struct {
  32. endpoint string
  33. username string
  34. password string
  35. apiKey string
  36. }
  37. func (r *request) Get(path string) (io.ReadCloser, error) {
  38. return r.execute(http.MethodGet, path, nil)
  39. }
  40. func (r *request) Post(path string, data interface{}) (io.ReadCloser, error) {
  41. return r.execute(http.MethodPost, path, data)
  42. }
  43. func (r *request) PostFile(path string, f io.ReadCloser) (io.ReadCloser, error) {
  44. return r.execute(http.MethodPost, path, f)
  45. }
  46. func (r *request) Put(path string, data interface{}) (io.ReadCloser, error) {
  47. return r.execute(http.MethodPut, path, data)
  48. }
  49. func (r *request) Delete(path string) (io.ReadCloser, error) {
  50. return r.execute(http.MethodDelete, path, nil)
  51. }
  52. func (r *request) execute(method, path string, data interface{}) (io.ReadCloser, error) {
  53. if r.endpoint[len(r.endpoint)-1:] == "/" {
  54. r.endpoint = r.endpoint[:len(r.endpoint)-1]
  55. }
  56. u, err := url.Parse(r.endpoint + path)
  57. if err != nil {
  58. return nil, err
  59. }
  60. request := &http.Request{
  61. URL: u,
  62. Method: method,
  63. Header: r.buildHeaders(),
  64. }
  65. if r.username != "" && r.password != "" {
  66. request.SetBasicAuth(r.username, r.password)
  67. }
  68. if data != nil {
  69. switch data.(type) {
  70. case io.ReadCloser:
  71. request.Body = data.(io.ReadCloser)
  72. default:
  73. request.Body = ioutil.NopCloser(bytes.NewBuffer(r.toJSON(data)))
  74. }
  75. }
  76. client := r.buildClient()
  77. response, err := client.Do(request)
  78. if err != nil {
  79. return nil, err
  80. }
  81. switch response.StatusCode {
  82. case http.StatusUnauthorized:
  83. return nil, ErrNotAuthorized
  84. case http.StatusForbidden:
  85. return nil, ErrForbidden
  86. case http.StatusInternalServerError:
  87. return nil, ErrServerError
  88. case http.StatusNotFound:
  89. return nil, ErrNotFound
  90. case http.StatusBadRequest:
  91. defer response.Body.Close()
  92. var resp errorResponse
  93. decoder := json.NewDecoder(response.Body)
  94. if err := decoder.Decode(&resp); err != nil {
  95. return nil, fmt.Errorf("miniflux: bad request error (%v)", err)
  96. }
  97. return nil, fmt.Errorf("miniflux: bad request (%s)", resp.ErrorMessage)
  98. }
  99. if response.StatusCode > 400 {
  100. return nil, fmt.Errorf("miniflux: status code=%d", response.StatusCode)
  101. }
  102. return response.Body, nil
  103. }
  104. func (r *request) buildClient() http.Client {
  105. return http.Client{
  106. Timeout: time.Duration(defaultTimeout * time.Second),
  107. }
  108. }
  109. func (r *request) buildHeaders() http.Header {
  110. headers := make(http.Header)
  111. headers.Add("User-Agent", userAgent)
  112. headers.Add("Content-Type", "application/json")
  113. headers.Add("Accept", "application/json")
  114. if r.apiKey != "" {
  115. headers.Add("X-Auth-Token", r.apiKey)
  116. }
  117. return headers
  118. }
  119. func (r *request) toJSON(v interface{}) []byte {
  120. b, err := json.Marshal(v)
  121. if err != nil {
  122. log.Println("Unable to convert interface to JSON:", err)
  123. return []byte("")
  124. }
  125. return b
  126. }