mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-06-28 00:24:19 +00:00
dcb923b4a1
Add a Links tab to the client form for attaching third-party share links and remote subscription URLs per client. They are merged into the client's raw/JSON/Clash subscription output: links are emitted verbatim and parsed for JSON/Clash; subscription URLs are fetched (cached, with a short timeout) and their configs merged in. i18n keys added across all 13 locales.
134 lines
3.5 KiB
Go
134 lines
3.5 KiB
Go
package sub
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// External subscription fetching: a "subscription" external link is a remote
|
|
// URL whose body is a (often base64-encoded) newline list of share links. We
|
|
// fetch it on demand, cache the decoded links briefly, and bound the request
|
|
// with a short timeout so a slow/dead provider can't stall a client's sub.
|
|
|
|
const (
|
|
subscriptionCacheTTL = 5 * time.Minute
|
|
subscriptionMaxBytes = 2 << 20 // 2 MiB
|
|
)
|
|
|
|
var subscriptionHTTPClient = &http.Client{Timeout: 6 * time.Second}
|
|
|
|
type subscriptionCacheEntry struct {
|
|
links []string
|
|
fetchedAt time.Time
|
|
}
|
|
|
|
var subscriptionCache = struct {
|
|
sync.Mutex
|
|
m map[string]subscriptionCacheEntry
|
|
}{m: make(map[string]subscriptionCacheEntry)}
|
|
|
|
// fetchSubscriptionLinks returns the share links contained in a remote
|
|
// subscription URL, using a short-lived cache. On any failure it returns the
|
|
// last cached value (if present) or nil — never an error, so the rest of the
|
|
// client's subscription still renders.
|
|
func fetchSubscriptionLinks(rawURL string) []string {
|
|
rawURL = strings.TrimSpace(rawURL)
|
|
if rawURL == "" {
|
|
return nil
|
|
}
|
|
|
|
subscriptionCache.Lock()
|
|
cached, ok := subscriptionCache.m[rawURL]
|
|
subscriptionCache.Unlock()
|
|
if ok && time.Since(cached.fetchedAt) < subscriptionCacheTTL {
|
|
return cached.links
|
|
}
|
|
|
|
links, err := doFetchSubscriptionLinks(rawURL)
|
|
if err != nil {
|
|
// Serve stale on error rather than dropping the client's configs.
|
|
if ok {
|
|
return cached.links
|
|
}
|
|
return nil
|
|
}
|
|
|
|
subscriptionCache.Lock()
|
|
subscriptionCache.m[rawURL] = subscriptionCacheEntry{links: links, fetchedAt: time.Now()}
|
|
subscriptionCache.Unlock()
|
|
return links
|
|
}
|
|
|
|
func doFetchSubscriptionLinks(rawURL string) ([]string, error) {
|
|
req, err := http.NewRequest(http.MethodGet, rawURL, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// Some providers gate the link body on a known client User-Agent.
|
|
req.Header.Set("User-Agent", "v2rayNG/1.8.5")
|
|
resp, err := subscriptionHTTPClient.Do(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer resp.Body.Close()
|
|
if resp.StatusCode < 200 || resp.StatusCode >= 300 {
|
|
return nil, errBadStatus
|
|
}
|
|
body, err := io.ReadAll(io.LimitReader(resp.Body, subscriptionMaxBytes))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return decodeSubscriptionBody(body), nil
|
|
}
|
|
|
|
var errBadStatus = &subError{"non-2xx subscription response"}
|
|
|
|
type subError struct{ msg string }
|
|
|
|
func (e *subError) Error() string { return e.msg }
|
|
|
|
// decodeSubscriptionBody handles the common base64-encoded newline list as well
|
|
// as a plain-text body, returning only the lines that look like share links.
|
|
func decodeSubscriptionBody(body []byte) []string {
|
|
text := strings.TrimSpace(string(body))
|
|
if text == "" {
|
|
return nil
|
|
}
|
|
if decoded, ok := tryDecodeBase64Body(text); ok {
|
|
text = strings.TrimSpace(decoded)
|
|
}
|
|
lines := strings.FieldsFunc(text, func(r rune) bool { return r == '\n' || r == '\r' })
|
|
out := make([]string, 0, len(lines))
|
|
for _, ln := range lines {
|
|
ln = strings.TrimSpace(ln)
|
|
if ln == "" || strings.HasPrefix(ln, "#") {
|
|
continue
|
|
}
|
|
if strings.Contains(ln, "://") {
|
|
out = append(out, ln)
|
|
}
|
|
}
|
|
return out
|
|
}
|
|
|
|
func tryDecodeBase64Body(s string) (string, bool) {
|
|
clean := strings.Map(func(r rune) rune {
|
|
switch r {
|
|
case ' ', '\n', '\r', '\t':
|
|
return -1
|
|
}
|
|
return r
|
|
}, s)
|
|
if b, err := base64.StdEncoding.DecodeString(padBase64Sub(clean)); err == nil {
|
|
return string(b), true
|
|
}
|
|
if b, err := base64.RawURLEncoding.DecodeString(strings.TrimRight(clean, "=")); err == nil {
|
|
return string(b), true
|
|
}
|
|
return "", false
|
|
}
|