rssbridge.go 2.7 KB

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