request.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package client // import "miniflux.app/v2/client"
  4. import (
  5. "bytes"
  6. "encoding/json"
  7. "errors"
  8. "fmt"
  9. "io"
  10. "log"
  11. "net/http"
  12. "net/url"
  13. "time"
  14. )
  15. const (
  16. userAgent = "Miniflux Client Library"
  17. defaultTimeout = 80
  18. )
  19. // List of exposed errors.
  20. var (
  21. ErrNotAuthorized = errors.New("miniflux: unauthorized (bad credentials)")
  22. ErrForbidden = errors.New("miniflux: access forbidden")
  23. ErrServerError = errors.New("miniflux: internal server error")
  24. ErrNotFound = errors.New("miniflux: resource not found")
  25. ErrBadRequest = errors.New("miniflux: bad request")
  26. ErrEmptyEndpoint = errors.New("miniflux: empty endpoint provided")
  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 any) (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 any) (io.ReadCloser, error) {
  47. return r.execute(http.MethodPut, path, data)
  48. }
  49. func (r *request) Delete(path string) error {
  50. _, err := r.execute(http.MethodDelete, path, nil)
  51. return err
  52. }
  53. func (r *request) execute(method, path string, data any) (io.ReadCloser, error) {
  54. if r.endpoint == "" {
  55. return nil, ErrEmptyEndpoint
  56. }
  57. if r.endpoint[len(r.endpoint)-1:] == "/" {
  58. r.endpoint = r.endpoint[:len(r.endpoint)-1]
  59. }
  60. u, err := url.Parse(r.endpoint + path)
  61. if err != nil {
  62. return nil, err
  63. }
  64. request := &http.Request{
  65. URL: u,
  66. Method: method,
  67. Header: r.buildHeaders(),
  68. }
  69. if r.username != "" && r.password != "" {
  70. request.SetBasicAuth(r.username, r.password)
  71. }
  72. if data != nil {
  73. switch data := data.(type) {
  74. case io.ReadCloser:
  75. request.Body = data
  76. default:
  77. request.Body = io.NopCloser(bytes.NewBuffer(r.toJSON(data)))
  78. }
  79. }
  80. client := r.buildClient()
  81. response, err := client.Do(request)
  82. if err != nil {
  83. return nil, err
  84. }
  85. switch response.StatusCode {
  86. case http.StatusUnauthorized:
  87. response.Body.Close()
  88. return nil, ErrNotAuthorized
  89. case http.StatusForbidden:
  90. response.Body.Close()
  91. return nil, ErrForbidden
  92. case http.StatusInternalServerError:
  93. defer response.Body.Close()
  94. var resp errorResponse
  95. decoder := json.NewDecoder(response.Body)
  96. // If we failed to decode, just return a generic ErrServerError
  97. if err := decoder.Decode(&resp); err != nil {
  98. return nil, ErrServerError
  99. }
  100. return nil, errors.New("miniflux: internal server error: " + resp.ErrorMessage)
  101. case http.StatusNotFound:
  102. response.Body.Close()
  103. return nil, ErrNotFound
  104. case http.StatusNoContent:
  105. response.Body.Close()
  106. return nil, nil
  107. case http.StatusBadRequest:
  108. defer response.Body.Close()
  109. var resp errorResponse
  110. decoder := json.NewDecoder(response.Body)
  111. if err := decoder.Decode(&resp); err != nil {
  112. return nil, fmt.Errorf("%w (%v)", ErrBadRequest, err)
  113. }
  114. return nil, fmt.Errorf("%w (%s)", ErrBadRequest, resp.ErrorMessage)
  115. }
  116. if response.StatusCode > 400 {
  117. response.Body.Close()
  118. return nil, fmt.Errorf("miniflux: status code=%d", response.StatusCode)
  119. }
  120. return response.Body, nil
  121. }
  122. func (r *request) buildClient() http.Client {
  123. return http.Client{
  124. Timeout: defaultTimeout * time.Second,
  125. }
  126. }
  127. func (r *request) buildHeaders() http.Header {
  128. headers := make(http.Header)
  129. headers.Add("User-Agent", userAgent)
  130. headers.Add("Content-Type", "application/json")
  131. headers.Add("Accept", "application/json")
  132. if r.apiKey != "" {
  133. headers.Add("X-Auth-Token", r.apiKey)
  134. }
  135. return headers
  136. }
  137. func (r *request) toJSON(v any) []byte {
  138. b, err := json.Marshal(v)
  139. if err != nil {
  140. log.Println("Unable to convert interface to JSON:", err)
  141. return []byte("")
  142. }
  143. return b
  144. }