pinboard.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package pinboard // import "miniflux.app/v2/internal/integration/pinboard"
  4. import (
  5. "encoding/xml"
  6. "errors"
  7. "fmt"
  8. "net/http"
  9. "net/url"
  10. "time"
  11. "miniflux.app/v2/internal/http/client"
  12. "miniflux.app/v2/internal/version"
  13. )
  14. var errPostNotFound = errors.New("pinboard: post not found")
  15. var errMissingCredentials = errors.New("pinboard: missing auth token")
  16. const defaultClientTimeout = 10 * time.Second
  17. type Client struct {
  18. authToken string
  19. }
  20. func NewClient(authToken string) *Client {
  21. return &Client{authToken: authToken}
  22. }
  23. func (c *Client) CreateBookmark(entryURL, entryTitle, pinboardTags string, markAsUnread bool) error {
  24. if c.authToken == "" {
  25. return errMissingCredentials
  26. }
  27. // We check if the url is already bookmarked to avoid overriding existing data.
  28. post, err := c.getBookmark(entryURL)
  29. if err != nil && errors.Is(err, errPostNotFound) {
  30. post = NewPost(entryURL, entryTitle)
  31. } else if err != nil {
  32. // In case of any other error, we return immediately to avoid overriding existing data.
  33. return err
  34. }
  35. post.addTag(pinboardTags)
  36. if markAsUnread {
  37. post.SetToread()
  38. }
  39. values := url.Values{}
  40. values.Add("auth_token", c.authToken)
  41. post.AddValues(values)
  42. apiEndpoint := "https://api.pinboard.in/v1/posts/add?" + values.Encode()
  43. request, err := http.NewRequest(http.MethodGet, apiEndpoint, nil)
  44. if err != nil {
  45. return fmt.Errorf("pinboard: unable to create request: %v", err)
  46. }
  47. request.Header.Set("User-Agent", "Miniflux/"+version.Version)
  48. httpClient := client.NewClientWithOptions(client.Options{Timeout: defaultClientTimeout})
  49. response, err := httpClient.Do(request)
  50. if err != nil {
  51. return fmt.Errorf("pinboard: unable to send request: %v", err)
  52. }
  53. defer response.Body.Close()
  54. if response.StatusCode >= 400 {
  55. return fmt.Errorf("pinboard: unable to create a bookmark: url=%s status=%d", apiEndpoint, response.StatusCode)
  56. }
  57. return nil
  58. }
  59. // getBookmark fetches a bookmark from Pinboard. https://www.pinboard.in/api/#posts_get
  60. func (c *Client) getBookmark(entryURL string) (*Post, error) {
  61. if c.authToken == "" {
  62. return nil, errMissingCredentials
  63. }
  64. values := url.Values{}
  65. values.Add("auth_token", c.authToken)
  66. values.Add("url", entryURL)
  67. apiEndpoint := "https://api.pinboard.in/v1/posts/get?" + values.Encode()
  68. request, err := http.NewRequest(http.MethodGet, apiEndpoint, nil)
  69. if err != nil {
  70. return nil, fmt.Errorf("pinboard: unable to create request: %v", err)
  71. }
  72. request.Header.Set("User-Agent", "Miniflux/"+version.Version)
  73. httpClient := client.NewClientWithOptions(client.Options{Timeout: defaultClientTimeout})
  74. response, err := httpClient.Do(request)
  75. if err != nil {
  76. return nil, fmt.Errorf("pinboard: unable fetch bookmark: %v", err)
  77. }
  78. defer response.Body.Close()
  79. if response.StatusCode >= 400 {
  80. return nil, fmt.Errorf("pinboard: unable to fetch bookmark, status=%d", response.StatusCode)
  81. }
  82. var results posts
  83. err = xml.NewDecoder(response.Body).Decode(&results)
  84. if err != nil {
  85. return nil, fmt.Errorf("pinboard: unable to decode XML: %v", err)
  86. }
  87. if len(results.Posts) == 0 {
  88. return nil, errPostNotFound
  89. }
  90. return &results.Posts[0], nil
  91. }