Force ipv6 setting
This commit is contained in:
@@ -42,7 +42,7 @@ func main() {
|
|||||||
|
|
||||||
cache := expirable.NewLRU[string, string](1e4, nil, time.Hour)
|
cache := expirable.NewLRU[string, string](1e4, nil, time.Hour)
|
||||||
|
|
||||||
client := zurghttp.NewHTTPClient(config.GetToken(), 5, nil)
|
client := zurghttp.NewHTTPClient(config.GetToken(), 5, config)
|
||||||
|
|
||||||
rd := realdebrid.NewRealDebrid(config.GetToken(), client, logutil.NewLogger().Named("realdebrid"))
|
rd := realdebrid.NewRealDebrid(config.GetToken(), client, logutil.NewLogger().Named("realdebrid"))
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ check_for_changes_every_secs: 15
|
|||||||
|
|
||||||
enable_repair: true # BEWARE! THERE CAN ONLY BE 1 INSTANCE OF ZURG THAT SHOULD REPAIR YOUR TORRENTS
|
enable_repair: true # BEWARE! THERE CAN ONLY BE 1 INSTANCE OF ZURG THAT SHOULD REPAIR YOUR TORRENTS
|
||||||
retain_folder_name_extension: false # if true, zurg won't modify the filenames from real-debrid
|
retain_folder_name_extension: false # if true, zurg won't modify the filenames from real-debrid
|
||||||
|
retain_rd_torrent_name: false # if true, it will strictly follow RD API returned torrent name which should make this more compatible with rdt-client
|
||||||
on_library_update: |
|
on_library_update: |
|
||||||
for arg in "$@"
|
for arg in "$@"
|
||||||
do
|
do
|
||||||
@@ -17,6 +18,8 @@ on_library_update: |
|
|||||||
done
|
done
|
||||||
|
|
||||||
network_buffer_size: 1048576 # 1 MiB
|
network_buffer_size: 1048576 # 1 MiB
|
||||||
|
serve_from_rclone: false
|
||||||
|
force_ipv6: true
|
||||||
# preferred_hosts: # Run the script here https://github.com/debridmediamanager/real-debrid-network-test
|
# preferred_hosts: # Run the script here https://github.com/debridmediamanager/real-debrid-network-test
|
||||||
# - 20.download.real-debrid.com
|
# - 20.download.real-debrid.com
|
||||||
# - 21.download.real-debrid.com
|
# - 21.download.real-debrid.com
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
response=$(curl -o /dev/null -s -w "%{http_code}" http://localhost:9999)
|
response=$(curl -o /dev/null -s -w "%{http_code}" http://zen.box:9999/__all__/)
|
||||||
if [ "$response" -eq 200 ]; then
|
if [ "$response" -eq 200 ]; then
|
||||||
exit 0
|
exit 0
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ type ConfigInterface interface {
|
|||||||
EnableRetainRDTorrentName() bool
|
EnableRetainRDTorrentName() bool
|
||||||
GetRandomPreferredHost() string
|
GetRandomPreferredHost() string
|
||||||
ShouldServeFromRclone() bool
|
ShouldServeFromRclone() bool
|
||||||
|
ShouldForceIPv6() bool
|
||||||
}
|
}
|
||||||
|
|
||||||
type ZurgConfig struct {
|
type ZurgConfig struct {
|
||||||
@@ -36,6 +37,7 @@ type ZurgConfig struct {
|
|||||||
RetainRDTorrentName bool `yaml:"retain_rd_torrent_name"`
|
RetainRDTorrentName bool `yaml:"retain_rd_torrent_name"`
|
||||||
PreferredHosts []string `yaml:"preferred_hosts"`
|
PreferredHosts []string `yaml:"preferred_hosts"`
|
||||||
ServeFromRclone bool `yaml:"serve_from_rclone"`
|
ServeFromRclone bool `yaml:"serve_from_rclone"`
|
||||||
|
ForceIPv6 bool `yaml:"force_ipv6"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (z *ZurgConfig) GetToken() string {
|
func (z *ZurgConfig) GetToken() string {
|
||||||
@@ -111,3 +113,7 @@ func (z *ZurgConfig) GetRandomPreferredHost() string {
|
|||||||
func (z *ZurgConfig) ShouldServeFromRclone() bool {
|
func (z *ZurgConfig) ShouldServeFromRclone() bool {
|
||||||
return z.ServeFromRclone
|
return z.ServeFromRclone
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (z *ZurgConfig) ShouldForceIPv6() bool {
|
||||||
|
return z.ForceIPv6
|
||||||
|
}
|
||||||
|
|||||||
@@ -117,6 +117,7 @@ func (gf *GetFile) HandleGetRequest(w http.ResponseWriter, r *http.Request, t *i
|
|||||||
} else {
|
} else {
|
||||||
gf.streamFileToResponse(file, resp.Download, w, r, t, c, log)
|
gf.streamFileToResponse(file, resp.Download, w, r, t, c, log)
|
||||||
}
|
}
|
||||||
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -176,9 +177,10 @@ func (gf *GetFile) playErrorVideo(link string, w http.ResponseWriter, r *http.Re
|
|||||||
}
|
}
|
||||||
if c.ShouldServeFromRclone() {
|
if c.ShouldServeFromRclone() {
|
||||||
redirect(w, r, resp.Download, c)
|
redirect(w, r, resp.Download, c)
|
||||||
}
|
} else {
|
||||||
gf.streamFileToResponse(nil, resp.Download, w, r, t, c, log)
|
gf.streamFileToResponse(nil, resp.Download, w, r, t, c, log)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func redirect(w http.ResponseWriter, r *http.Request, url string, c config.ConfigInterface) {
|
func redirect(w http.ResponseWriter, r *http.Request, url string, c config.ConfigInterface) {
|
||||||
prefHost := c.GetRandomPreferredHost()
|
prefHost := c.GetRandomPreferredHost()
|
||||||
|
|||||||
@@ -1,12 +1,15 @@
|
|||||||
package http
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/debridmediamanager.com/zurg/internal/config"
|
"github.com/debridmediamanager.com/zurg/internal/config"
|
||||||
"github.com/debridmediamanager.com/zurg/pkg/logutil"
|
"github.com/debridmediamanager.com/zurg/pkg/logutil"
|
||||||
|
cmap "github.com/orcaman/concurrent-map/v2"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -18,6 +21,7 @@ type HTTPClient struct {
|
|||||||
BearerToken string
|
BearerToken string
|
||||||
log *zap.SugaredLogger
|
log *zap.SugaredLogger
|
||||||
config config.ConfigInterface
|
config config.ConfigInterface
|
||||||
|
IPv6 cmap.ConcurrentMap[string, net.IP]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *HTTPClient) Do(req *http.Request) (*http.Response, error) {
|
func (r *HTTPClient) Do(req *http.Request) (*http.Response, error) {
|
||||||
@@ -30,6 +34,43 @@ func (r *HTTPClient) Do(req *http.Request) (*http.Response, error) {
|
|||||||
if r.BearerToken != "" {
|
if r.BearerToken != "" {
|
||||||
req.Header.Set("Authorization", "Bearer "+r.BearerToken)
|
req.Header.Set("Authorization", "Bearer "+r.BearerToken)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if r.config.ShouldForceIPv6() {
|
||||||
|
dialer := &net.Dialer{}
|
||||||
|
transport := &http.Transport{
|
||||||
|
DialContext: func(ctx context.Context, network, addr string) (net.Conn, error) {
|
||||||
|
host, _, err := net.SplitHostPort(addr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if ipv6, ok := r.IPv6.Get(host); !ok {
|
||||||
|
// Lookup IP address if not found in map
|
||||||
|
ips, err := net.LookupIP(host)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
for _, ip := range ips {
|
||||||
|
if ip.To4() == nil { // IPv6
|
||||||
|
ipv6 = ip
|
||||||
|
r.IPv6.Set(host, ipv6)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ipv6 == nil { // No IPv6 address found, fallback to default
|
||||||
|
r.log.Warnf("No IPv6 address found for %s, falling back to default", host)
|
||||||
|
addr = net.JoinHostPort(host, req.URL.Port())
|
||||||
|
} else {
|
||||||
|
addr = net.JoinHostPort(ipv6.String(), req.URL.Port())
|
||||||
|
}
|
||||||
|
} else if ipv6 != nil && host == req.URL.Hostname() {
|
||||||
|
addr = net.JoinHostPort(ipv6.String(), req.URL.Port())
|
||||||
|
}
|
||||||
|
return dialer.DialContext(ctx, network, addr)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
r.Client.Transport = transport
|
||||||
|
}
|
||||||
|
|
||||||
var resp *http.Response
|
var resp *http.Response
|
||||||
var err error
|
var err error
|
||||||
for attempt := 0; attempt < r.MaxRetries; attempt++ {
|
for attempt := 0; attempt < r.MaxRetries; attempt++ {
|
||||||
@@ -62,5 +103,6 @@ func NewHTTPClient(token string, maxRetries int, cfg config.ConfigInterface) *HT
|
|||||||
},
|
},
|
||||||
log: logutil.NewLogger().Named("client"),
|
log: logutil.NewLogger().Named("client"),
|
||||||
config: cfg,
|
config: cfg,
|
||||||
|
IPv6: cmap.New[net.IP](),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user