connector.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package pocket // import "miniflux.app/v2/internal/integration/pocket"
  4. import (
  5. "bytes"
  6. "encoding/json"
  7. "errors"
  8. "fmt"
  9. "net/http"
  10. "miniflux.app/v2/internal/version"
  11. )
  12. // Connector manages the authorization flow with Pocket to get a personal access token.
  13. type Connector struct {
  14. consumerKey string
  15. }
  16. // NewConnector returns a new Pocket Connector.
  17. func NewConnector(consumerKey string) *Connector {
  18. return &Connector{consumerKey}
  19. }
  20. // RequestToken fetches a new request token from Pocket API.
  21. func (c *Connector) RequestToken(redirectURL string) (string, error) {
  22. apiEndpoint := "https://getpocket.com/v3/oauth/request"
  23. requestBody, err := json.Marshal(&createTokenRequest{ConsumerKey: c.consumerKey, RedirectURI: redirectURL})
  24. if err != nil {
  25. return "", fmt.Errorf("pocket: unable to encode request body: %v", err)
  26. }
  27. request, err := http.NewRequest(http.MethodPost, apiEndpoint, bytes.NewReader(requestBody))
  28. if err != nil {
  29. return "", fmt.Errorf("pocket: unable to create request: %v", err)
  30. }
  31. request.Header.Set("Content-Type", "application/json")
  32. request.Header.Set("X-Accept", "application/json")
  33. request.Header.Set("User-Agent", "Miniflux/"+version.Version)
  34. httpClient := &http.Client{Timeout: defaultClientTimeout}
  35. response, err := httpClient.Do(request)
  36. if err != nil {
  37. return "", fmt.Errorf("pocket: unable to send request: %v", err)
  38. }
  39. defer response.Body.Close()
  40. if response.StatusCode >= 400 {
  41. return "", fmt.Errorf("pocket: unable get request token: url=%s status=%d", apiEndpoint, response.StatusCode)
  42. }
  43. var result createTokenResponse
  44. if err := json.NewDecoder(response.Body).Decode(&result); err != nil {
  45. return "", fmt.Errorf("pocket: unable to decode response: %v", err)
  46. }
  47. if result.Code == "" {
  48. return "", errors.New("pocket: request token is empty")
  49. }
  50. return result.Code, nil
  51. }
  52. // AccessToken fetches a new access token once the end-user authorized the application.
  53. func (c *Connector) AccessToken(requestToken string) (string, error) {
  54. apiEndpoint := "https://getpocket.com/v3/oauth/authorize"
  55. requestBody, err := json.Marshal(&authorizeRequest{ConsumerKey: c.consumerKey, Code: requestToken})
  56. if err != nil {
  57. return "", fmt.Errorf("pocket: unable to encode request body: %v", err)
  58. }
  59. request, err := http.NewRequest(http.MethodPost, apiEndpoint, bytes.NewReader(requestBody))
  60. if err != nil {
  61. return "", fmt.Errorf("pocket: unable to create request: %v", err)
  62. }
  63. request.Header.Set("Content-Type", "application/json")
  64. request.Header.Set("X-Accept", "application/json")
  65. request.Header.Set("User-Agent", "Miniflux/"+version.Version)
  66. httpClient := &http.Client{Timeout: defaultClientTimeout}
  67. response, err := httpClient.Do(request)
  68. if err != nil {
  69. return "", fmt.Errorf("pocket: unable to send request: %v", err)
  70. }
  71. defer response.Body.Close()
  72. if response.StatusCode >= 400 {
  73. return "", fmt.Errorf("pocket: unable get access token: url=%s status=%d", apiEndpoint, response.StatusCode)
  74. }
  75. var result authorizeReponse
  76. if err := json.NewDecoder(response.Body).Decode(&result); err != nil {
  77. return "", fmt.Errorf("pocket: unable to decode response: %v", err)
  78. }
  79. if result.AccessToken == "" {
  80. return "", errors.New("pocket: access token is empty")
  81. }
  82. return result.AccessToken, nil
  83. }
  84. // AuthorizationURL returns the authorization URL for the end-user.
  85. func (c *Connector) AuthorizationURL(requestToken, redirectURL string) string {
  86. return fmt.Sprintf(
  87. "https://getpocket.com/auth/authorize?request_token=%s&redirect_uri=%s",
  88. requestToken,
  89. redirectURL,
  90. )
  91. }
  92. type createTokenRequest struct {
  93. ConsumerKey string `json:"consumer_key"`
  94. RedirectURI string `json:"redirect_uri"`
  95. }
  96. type createTokenResponse struct {
  97. Code string `json:"code"`
  98. }
  99. type authorizeRequest struct {
  100. ConsumerKey string `json:"consumer_key"`
  101. Code string `json:"code"`
  102. }
  103. type authorizeReponse struct {
  104. AccessToken string `json:"access_token"`
  105. Username string `json:"username"`
  106. }