rssbridge.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  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. )
  13. const defaultClientTimeout = 30 * time.Second
  14. type Bridge struct {
  15. URL string `json:"url"`
  16. BridgeMeta BridgeMeta `json:"bridgeMeta"`
  17. }
  18. type BridgeMeta struct {
  19. Name string `json:"name"`
  20. }
  21. func DetectBridges(rssBridgeURL, rssBridgeToken, websiteURL string) ([]*Bridge, error) {
  22. endpointURL, err := url.Parse(rssBridgeURL)
  23. if err != nil {
  24. return nil, fmt.Errorf("RSS-Bridge: unable to parse bridge URL: %w", err)
  25. }
  26. values := endpointURL.Query()
  27. if rssBridgeToken != "" {
  28. values.Add("token", rssBridgeToken)
  29. }
  30. values.Add("action", "findfeed")
  31. values.Add("format", "atom")
  32. values.Add("url", websiteURL)
  33. endpointURL.RawQuery = values.Encode()
  34. slog.Debug("Detecting RSS bridges", slog.String("url", endpointURL.String()))
  35. request, err := http.NewRequest(http.MethodGet, endpointURL.String(), nil)
  36. if err != nil {
  37. return nil, fmt.Errorf("RSS-Bridge: unable to create request: %w", err)
  38. }
  39. httpClient := &http.Client{Timeout: defaultClientTimeout}
  40. response, err := httpClient.Do(request)
  41. if err != nil {
  42. return nil, fmt.Errorf("RSS-Bridge: unable to execute request: %w", err)
  43. }
  44. defer response.Body.Close()
  45. if response.StatusCode == http.StatusNotFound {
  46. return nil, nil
  47. }
  48. if response.StatusCode > 400 {
  49. return nil, fmt.Errorf("RSS-Bridge: unexpected status code %d", response.StatusCode)
  50. }
  51. var bridgeResponse []*Bridge
  52. if err := json.NewDecoder(response.Body).Decode(&bridgeResponse); err != nil {
  53. return nil, fmt.Errorf("RSS-Bridge: unable to decode bridge response: %w", err)
  54. }
  55. for _, bridge := range bridgeResponse {
  56. slog.Debug("Found RSS bridge",
  57. slog.String("name", bridge.BridgeMeta.Name),
  58. slog.String("url", bridge.URL),
  59. )
  60. if strings.HasPrefix(bridge.URL, "./") {
  61. bridge.URL = rssBridgeURL + bridge.URL[2:]
  62. slog.Debug("Rewrited relative RSS bridge URL",
  63. slog.String("name", bridge.BridgeMeta.Name),
  64. slog.String("url", bridge.URL),
  65. )
  66. }
  67. if rssBridgeToken != "" {
  68. bridge.URL = bridge.URL + "&token=" + rssBridgeToken
  69. slog.Debug("Appended token to RSS bridge URL",
  70. slog.String("name", bridge.BridgeMeta.Name),
  71. slog.String("url", bridge.URL),
  72. )
  73. }
  74. }
  75. return bridgeResponse, nil
  76. }