request.go 4.1 KB

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