rssbridge.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package rssbridge // import "miniflux.app/v2/internal/integration/rssbridge"
  4. import (
  5. "encoding/json"
  6. "fmt"
  7. "log/slog"
  8. "net/http"
  9. "net/url"
  10. "strings"
  11. "time"
  12. "miniflux.app/v2/internal/version"
  13. )
  14. const defaultClientTimeout = 30 * time.Second
  15. type Bridge struct {
  16. URL string `json:"url"`
  17. BridgeMeta BridgeMeta `json:"bridgeMeta"`
  18. }
  19. type BridgeMeta struct {
  20. Name string `json:"name"`
  21. }
  22. func DetectBridges(rssBridgeURL, rssBridgeToken, websiteURL string) ([]*Bridge, error) {
  23. endpointURL, err := url.Parse(rssBridgeURL)
  24. if err != nil {
  25. return nil, fmt.Errorf("rssbridge: unable to parse bridge URL: %w", err)
  26. }
  27. values := endpointURL.Query()
  28. if rssBridgeToken != "" {
  29. values.Add("token", rssBridgeToken)
  30. }
  31. values.Add("action", "findfeed")
  32. values.Add("format", "atom")
  33. values.Add("url", websiteURL)
  34. endpointURL.RawQuery = values.Encode()
  35. slog.Debug("Detecting RSS bridges", slog.String("url", endpointURL.String()))
  36. request, err := http.NewRequest(http.MethodGet, endpointURL.String(), nil)
  37. if err != nil {
  38. return nil, fmt.Errorf("rssbridge: unable to create request: %w", err)
  39. }
  40. request.Header.Set("User-Agent", "Miniflux/"+version.Version)
  41. httpClient := &http.Client{Timeout: defaultClientTimeout}
  42. response, err := httpClient.Do(request)
  43. if err != nil {
  44. return nil, fmt.Errorf("rssbridge: unable to execute request: %w", err)
  45. }
  46. defer response.Body.Close()
  47. if response.StatusCode == http.StatusNotFound {
  48. return nil, nil
  49. }
  50. if response.StatusCode >= 400 {
  51. return nil, fmt.Errorf("rssbridge: unexpected status code %d", response.StatusCode)
  52. }
  53. var bridgeResponse []*Bridge
  54. if err := json.NewDecoder(response.Body).Decode(&bridgeResponse); err != nil {
  55. return nil, fmt.Errorf("rssbridge: unable to decode bridge response: %w", err)
  56. }
  57. for _, bridge := range bridgeResponse {
  58. slog.Debug("Found RSS bridge",
  59. slog.String("name", bridge.BridgeMeta.Name),
  60. slog.String("url", bridge.URL),
  61. )
  62. if strings.HasPrefix(bridge.URL, "./") {
  63. bridge.URL = rssBridgeURL + bridge.URL[2:]
  64. slog.Debug("Rewrote relative RSS bridge URL",
  65. slog.String("name", bridge.BridgeMeta.Name),
  66. slog.String("url", bridge.URL),
  67. )
  68. }
  69. if rssBridgeToken != "" {
  70. bridge.URL = bridge.URL + "&token=" + rssBridgeToken
  71. slog.Debug("Appended token to RSS bridge URL",
  72. slog.String("name", bridge.BridgeMeta.Name),
  73. slog.String("url", bridge.URL),
  74. )
  75. }
  76. }
  77. return bridgeResponse, nil
  78. }