2023-10-22 04:50:29 +02:00
|
|
|
// SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-03-24 20:19:03 +01:00
|
|
|
package rssbridge // import "miniflux.app/v2/internal/integration/rssbridge"
|
2023-10-22 20:10:56 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2024-01-13 23:31:45 +01:00
|
|
|
"log/slog"
|
2023-10-22 20:10:56 +02:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2024-01-13 23:31:45 +01:00
|
|
|
"strings"
|
|
|
|
"time"
|
2023-10-22 20:10:56 +02:00
|
|
|
)
|
|
|
|
|
2024-01-13 23:31:45 +01:00
|
|
|
const defaultClientTimeout = 30 * time.Second
|
|
|
|
|
2023-10-22 20:10:56 +02:00
|
|
|
type Bridge struct {
|
|
|
|
URL string `json:"url"`
|
|
|
|
BridgeMeta BridgeMeta `json:"bridgeMeta"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type BridgeMeta struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
}
|
|
|
|
|
2024-01-13 23:31:45 +01:00
|
|
|
func DetectBridges(rssBridgeURL, websiteURL string) ([]*Bridge, error) {
|
|
|
|
endpointURL, err := url.Parse(rssBridgeURL)
|
2023-10-22 20:10:56 +02:00
|
|
|
if err != nil {
|
2024-01-13 23:31:45 +01:00
|
|
|
return nil, fmt.Errorf("RSS-Bridge: unable to parse bridge URL: %w", err)
|
2023-10-22 20:10:56 +02:00
|
|
|
}
|
2024-01-13 23:31:45 +01:00
|
|
|
|
|
|
|
values := endpointURL.Query()
|
2023-10-22 20:10:56 +02:00
|
|
|
values.Add("action", "findfeed")
|
|
|
|
values.Add("format", "atom")
|
|
|
|
values.Add("url", websiteURL)
|
2024-01-13 23:31:45 +01:00
|
|
|
endpointURL.RawQuery = values.Encode()
|
|
|
|
|
|
|
|
slog.Debug("Detecting RSS bridges", slog.String("url", endpointURL.String()))
|
|
|
|
|
|
|
|
request, err := http.NewRequest(http.MethodGet, endpointURL.String(), nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("RSS-Bridge: unable to create request: %w", err)
|
|
|
|
}
|
2023-10-22 20:10:56 +02:00
|
|
|
|
2024-01-13 23:31:45 +01:00
|
|
|
httpClient := &http.Client{Timeout: defaultClientTimeout}
|
|
|
|
|
|
|
|
response, err := httpClient.Do(request)
|
2023-10-22 20:10:56 +02:00
|
|
|
if err != nil {
|
2024-01-13 23:31:45 +01:00
|
|
|
return nil, fmt.Errorf("RSS-Bridge: unable to execute request: %w", err)
|
2023-10-22 20:10:56 +02:00
|
|
|
}
|
|
|
|
defer response.Body.Close()
|
2024-01-13 23:31:45 +01:00
|
|
|
|
2023-10-22 20:10:56 +02:00
|
|
|
if response.StatusCode == http.StatusNotFound {
|
2024-01-13 23:31:45 +01:00
|
|
|
return nil, nil
|
2023-10-22 20:10:56 +02:00
|
|
|
}
|
2024-01-13 23:31:45 +01:00
|
|
|
|
2023-10-22 20:10:56 +02:00
|
|
|
if response.StatusCode > 400 {
|
2023-10-22 04:50:29 +02:00
|
|
|
return nil, fmt.Errorf("RSS-Bridge: unexpected status code %d", response.StatusCode)
|
2023-10-22 20:10:56 +02:00
|
|
|
}
|
2024-01-13 23:31:45 +01:00
|
|
|
|
|
|
|
var bridgeResponse []*Bridge
|
2023-10-22 20:10:56 +02:00
|
|
|
if err := json.NewDecoder(response.Body).Decode(&bridgeResponse); err != nil {
|
|
|
|
return nil, fmt.Errorf("RSS-Bridge: unable to decode bridge response: %w", err)
|
|
|
|
}
|
2024-01-13 23:31:45 +01:00
|
|
|
|
|
|
|
for _, bridge := range bridgeResponse {
|
|
|
|
slog.Debug("Found RSS bridge",
|
|
|
|
slog.String("name", bridge.BridgeMeta.Name),
|
|
|
|
slog.String("url", bridge.URL),
|
|
|
|
)
|
|
|
|
|
|
|
|
if strings.HasPrefix(bridge.URL, "./") {
|
|
|
|
bridge.URL = rssBridgeURL + bridge.URL[2:]
|
|
|
|
|
|
|
|
slog.Debug("Rewrited relative RSS bridge URL",
|
|
|
|
slog.String("name", bridge.BridgeMeta.Name),
|
|
|
|
slog.String("url", bridge.URL),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bridgeResponse, nil
|
2023-10-22 20:10:56 +02:00
|
|
|
}
|