feat: geoUpdateInterval

This commit is contained in:
Toby 2023-11-13 20:27:08 -08:00
parent c62dc51017
commit e052f767db
4 changed files with 33 additions and 16 deletions

View file

@ -147,10 +147,11 @@ type serverConfigResolver struct {
} }
type serverConfigACL struct { type serverConfigACL struct {
File string `mapstructure:"file"` File string `mapstructure:"file"`
Inline []string `mapstructure:"inline"` Inline []string `mapstructure:"inline"`
GeoIP string `mapstructure:"geoip"` GeoIP string `mapstructure:"geoip"`
GeoSite string `mapstructure:"geosite"` GeoSite string `mapstructure:"geosite"`
GeoUpdateInterval time.Duration `mapstructure:"geoUpdateInterval"`
} }
type serverConfigOutboundDirect struct { type serverConfigOutboundDirect struct {
@ -469,6 +470,7 @@ func (c *serverConfig) fillOutboundConfig(hyConfig *server.Config) error {
gLoader := &utils.GeoLoader{ gLoader := &utils.GeoLoader{
GeoIPFilename: c.ACL.GeoIP, GeoIPFilename: c.ACL.GeoIP,
GeoSiteFilename: c.ACL.GeoSite, GeoSiteFilename: c.ACL.GeoSite,
UpdateInterval: c.ACL.GeoUpdateInterval,
DownloadFunc: geoDownloadFunc, DownloadFunc: geoDownloadFunc,
DownloadErrFunc: geoDownloadErrFunc, DownloadErrFunc: geoDownloadErrFunc,
} }

View file

@ -101,8 +101,9 @@ func TestServerConfig(t *testing.T) {
"lmao(ok)", "lmao(ok)",
"kek(cringe,boba,tea)", "kek(cringe,boba,tea)",
}, },
GeoIP: "some.dat", GeoIP: "some.dat",
GeoSite: "some_site.dat", GeoSite: "some_site.dat",
GeoUpdateInterval: 168 * time.Hour,
}, },
Outbounds: []serverConfigOutboundEntry{ Outbounds: []serverConfigOutboundEntry{
{ {

View file

@ -77,6 +77,7 @@ acl:
- kek(cringe,boba,tea) - kek(cringe,boba,tea)
geoip: some.dat geoip: some.dat
geosite: some_site.dat geosite: some_site.dat
geoUpdateInterval: 168h
outbounds: outbounds:
- name: goodstuff - name: goodstuff

View file

@ -4,6 +4,7 @@ import (
"io" "io"
"net/http" "net/http"
"os" "os"
"time"
"github.com/apernet/hysteria/extras/outbounds/acl" "github.com/apernet/hysteria/extras/outbounds/acl"
"github.com/apernet/hysteria/extras/outbounds/acl/v2geo" "github.com/apernet/hysteria/extras/outbounds/acl/v2geo"
@ -14,6 +15,8 @@ const (
geoipURL = "https://cdn.jsdelivr.net/gh/Loyalsoldier/v2ray-rules-dat@release/geoip.dat" geoipURL = "https://cdn.jsdelivr.net/gh/Loyalsoldier/v2ray-rules-dat@release/geoip.dat"
geositeFilename = "geosite.dat" geositeFilename = "geosite.dat"
geositeURL = "https://cdn.jsdelivr.net/gh/Loyalsoldier/v2ray-rules-dat@release/geosite.dat" geositeURL = "https://cdn.jsdelivr.net/gh/Loyalsoldier/v2ray-rules-dat@release/geosite.dat"
geoDefaultUpdateInterval = 7 * 24 * time.Hour // 7 days
) )
var _ acl.GeoLoader = (*GeoLoader)(nil) var _ acl.GeoLoader = (*GeoLoader)(nil)
@ -24,6 +27,7 @@ var _ acl.GeoLoader = (*GeoLoader)(nil)
type GeoLoader struct { type GeoLoader struct {
GeoIPFilename string GeoIPFilename string
GeoSiteFilename string GeoSiteFilename string
UpdateInterval time.Duration
DownloadFunc func(filename, url string) DownloadFunc func(filename, url string)
DownloadErrFunc func(err error) DownloadErrFunc func(err error)
@ -32,6 +36,19 @@ type GeoLoader struct {
geositeMap map[string]*v2geo.GeoSite geositeMap map[string]*v2geo.GeoSite
} }
func (l *GeoLoader) shouldDownload(filename string) bool {
info, err := os.Stat(filename)
if os.IsNotExist(err) {
return true
}
dt := time.Now().Sub(info.ModTime())
if l.UpdateInterval == 0 {
return dt > geoDefaultUpdateInterval
} else {
return dt > l.UpdateInterval
}
}
func (l *GeoLoader) download(filename, url string) error { func (l *GeoLoader) download(filename, url string) error {
l.DownloadFunc(filename, url) l.DownloadFunc(filename, url)
@ -64,15 +81,13 @@ func (l *GeoLoader) LoadGeoIP() (map[string]*v2geo.GeoIP, error) {
autoDL = true autoDL = true
filename = geoipFilename filename = geoipFilename
} }
m, err := v2geo.LoadGeoIP(filename) if autoDL && l.shouldDownload(filename) {
if os.IsNotExist(err) && autoDL { err := l.download(filename, geoipURL)
// It's ok, we will download it.
err = l.download(filename, geoipURL)
if err != nil { if err != nil {
return nil, err return nil, err
} }
m, err = v2geo.LoadGeoIP(filename)
} }
m, err := v2geo.LoadGeoIP(filename)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -90,15 +105,13 @@ func (l *GeoLoader) LoadGeoSite() (map[string]*v2geo.GeoSite, error) {
autoDL = true autoDL = true
filename = geositeFilename filename = geositeFilename
} }
m, err := v2geo.LoadGeoSite(filename) if autoDL && l.shouldDownload(filename) {
if os.IsNotExist(err) && autoDL { err := l.download(filename, geositeURL)
// It's ok, we will download it.
err = l.download(filename, geositeURL)
if err != nil { if err != nil {
return nil, err return nil, err
} }
m, err = v2geo.LoadGeoSite(filename)
} }
m, err := v2geo.LoadGeoSite(filename)
if err != nil { if err != nil {
return nil, err return nil, err
} }