2019-12-08 12:17:24 +08:00
|
|
|
package provider
|
|
|
|
|
|
|
|
import (
|
2024-09-19 14:24:35 +08:00
|
|
|
"encoding"
|
2019-12-08 12:17:24 +08:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
|
2023-11-03 21:01:45 +08:00
|
|
|
"github.com/metacubex/mihomo/common/structure"
|
|
|
|
"github.com/metacubex/mihomo/common/utils"
|
|
|
|
"github.com/metacubex/mihomo/component/resource"
|
|
|
|
C "github.com/metacubex/mihomo/constant"
|
|
|
|
types "github.com/metacubex/mihomo/constant/provider"
|
2024-09-19 14:24:35 +08:00
|
|
|
|
|
|
|
"github.com/dlclark/regexp2"
|
2019-12-08 12:17:24 +08:00
|
|
|
)
|
|
|
|
|
2023-06-06 09:45:05 +08:00
|
|
|
var (
|
|
|
|
errVehicleType = errors.New("unsupport vehicle type")
|
|
|
|
errSubPath = errors.New("path is not subpath of home directory")
|
|
|
|
)
|
2019-12-08 12:17:24 +08:00
|
|
|
|
|
|
|
type healthCheckSchema struct {
|
2023-06-04 14:00:24 +08:00
|
|
|
Enable bool `provider:"enable"`
|
|
|
|
URL string `provider:"url"`
|
|
|
|
Interval int `provider:"interval"`
|
2024-01-24 12:45:35 +08:00
|
|
|
TestTimeout int `provider:"timeout,omitempty"`
|
2023-06-04 14:00:24 +08:00
|
|
|
Lazy bool `provider:"lazy,omitempty"`
|
|
|
|
ExpectedStatus string `provider:"expected-status,omitempty"`
|
2019-12-08 12:17:24 +08:00
|
|
|
}
|
|
|
|
|
2024-09-18 22:41:06 +08:00
|
|
|
type OverrideProxyNameSchema struct {
|
|
|
|
// matching expression for regex replacement
|
2024-09-19 14:24:35 +08:00
|
|
|
Pattern *regexp2.Regexp `provider:"pattern"`
|
2024-09-18 22:41:06 +08:00
|
|
|
// the new content after regex matching
|
|
|
|
Target string `provider:"target"`
|
|
|
|
}
|
|
|
|
|
2024-09-19 14:24:35 +08:00
|
|
|
var _ encoding.TextUnmarshaler = (*regexp2.Regexp)(nil) // ensure *regexp2.Regexp can decode direct by structure package
|
|
|
|
|
2023-11-17 00:37:54 +08:00
|
|
|
type OverrideSchema struct {
|
2024-06-16 18:19:04 +08:00
|
|
|
TFO *bool `provider:"tfo,omitempty"`
|
|
|
|
MPTcp *bool `provider:"mptcp,omitempty"`
|
2024-02-20 15:15:22 +08:00
|
|
|
UDP *bool `provider:"udp,omitempty"`
|
2024-06-16 18:19:04 +08:00
|
|
|
UDPOverTCP *bool `provider:"udp-over-tcp,omitempty"`
|
2024-02-20 15:15:22 +08:00
|
|
|
Up *string `provider:"up,omitempty"`
|
|
|
|
Down *string `provider:"down,omitempty"`
|
|
|
|
DialerProxy *string `provider:"dialer-proxy,omitempty"`
|
|
|
|
SkipCertVerify *bool `provider:"skip-cert-verify,omitempty"`
|
|
|
|
Interface *string `provider:"interface-name,omitempty"`
|
|
|
|
RoutingMark *int `provider:"routing-mark,omitempty"`
|
|
|
|
IPVersion *string `provider:"ip-version,omitempty"`
|
|
|
|
AdditionalPrefix *string `provider:"additional-prefix,omitempty"`
|
|
|
|
AdditionalSuffix *string `provider:"additional-suffix,omitempty"`
|
2024-09-18 22:41:06 +08:00
|
|
|
|
|
|
|
ProxyName []OverrideProxyNameSchema `provider:"proxy-name,omitempty"`
|
2023-11-17 00:37:54 +08:00
|
|
|
}
|
|
|
|
|
2019-12-08 12:17:24 +08:00
|
|
|
type proxyProviderSchema struct {
|
2024-12-25 10:23:55 +08:00
|
|
|
Type string `provider:"type"`
|
|
|
|
Path string `provider:"path,omitempty"`
|
|
|
|
URL string `provider:"url,omitempty"`
|
|
|
|
Proxy string `provider:"proxy,omitempty"`
|
|
|
|
Interval int `provider:"interval,omitempty"`
|
|
|
|
Filter string `provider:"filter,omitempty"`
|
|
|
|
ExcludeFilter string `provider:"exclude-filter,omitempty"`
|
|
|
|
ExcludeType string `provider:"exclude-type,omitempty"`
|
|
|
|
DialerProxy string `provider:"dialer-proxy,omitempty"`
|
|
|
|
SizeLimit int64 `provider:"size-limit,omitempty"`
|
|
|
|
Payload []map[string]any `provider:"payload,omitempty"`
|
2023-11-17 00:37:54 +08:00
|
|
|
|
2024-04-07 05:28:22 +08:00
|
|
|
HealthCheck healthCheckSchema `provider:"health-check,omitempty"`
|
|
|
|
Override OverrideSchema `provider:"override,omitempty"`
|
|
|
|
Header map[string][]string `provider:"header,omitempty"`
|
2019-12-08 12:17:24 +08:00
|
|
|
}
|
|
|
|
|
2022-03-16 12:10:13 +08:00
|
|
|
func ParseProxyProvider(name string, mapping map[string]any) (types.ProxyProvider, error) {
|
2019-12-08 12:17:24 +08:00
|
|
|
decoder := structure.NewDecoder(structure.Option{TagName: "provider", WeaklyTypedInput: true})
|
|
|
|
|
2020-11-19 00:53:22 +08:00
|
|
|
schema := &proxyProviderSchema{
|
|
|
|
HealthCheck: healthCheckSchema{
|
|
|
|
Lazy: true,
|
|
|
|
},
|
|
|
|
}
|
2019-12-08 12:17:24 +08:00
|
|
|
if err := decoder.Decode(mapping, schema); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-01-20 11:00:06 +08:00
|
|
|
expectedStatus, err := utils.NewUnsignedRanges[uint16](schema.HealthCheck.ExpectedStatus)
|
2023-06-04 14:00:24 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-03-24 01:00:21 +08:00
|
|
|
var hcInterval uint
|
2019-12-08 12:17:24 +08:00
|
|
|
if schema.HealthCheck.Enable {
|
2023-12-31 07:39:17 +08:00
|
|
|
if schema.HealthCheck.Interval == 0 {
|
|
|
|
schema.HealthCheck.Interval = 300
|
|
|
|
}
|
2019-12-26 18:41:06 +08:00
|
|
|
hcInterval = uint(schema.HealthCheck.Interval)
|
2019-12-08 12:17:24 +08:00
|
|
|
}
|
2024-01-24 12:45:35 +08:00
|
|
|
hc := NewHealthCheck([]C.Proxy{}, schema.HealthCheck.URL, uint(schema.HealthCheck.TestTimeout), hcInterval, schema.HealthCheck.Lazy, expectedStatus)
|
2019-12-08 12:17:24 +08:00
|
|
|
|
2024-12-25 10:23:55 +08:00
|
|
|
parser, err := NewProxiesParser(schema.Filter, schema.ExcludeFilter, schema.ExcludeType, schema.DialerProxy, schema.Override)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-07-04 20:32:59 +08:00
|
|
|
var vehicle types.Vehicle
|
2019-12-08 12:17:24 +08:00
|
|
|
switch schema.Type {
|
|
|
|
case "file":
|
2023-06-15 22:45:02 +08:00
|
|
|
path := C.Path.Resolve(schema.Path)
|
2022-07-11 21:30:34 +08:00
|
|
|
vehicle = resource.NewFileVehicle(path)
|
2019-12-08 12:17:24 +08:00
|
|
|
case "http":
|
2024-04-07 05:28:22 +08:00
|
|
|
path := C.Path.GetPathByHash("proxies", schema.URL)
|
2023-06-15 22:45:02 +08:00
|
|
|
if schema.Path != "" {
|
2024-04-07 05:28:22 +08:00
|
|
|
path = C.Path.Resolve(schema.Path)
|
2024-09-11 16:10:35 +08:00
|
|
|
if !C.Path.IsSafePath(path) {
|
2023-06-15 22:45:02 +08:00
|
|
|
return nil, fmt.Errorf("%w: %s", errSubPath, path)
|
|
|
|
}
|
2023-06-06 09:45:05 +08:00
|
|
|
}
|
2024-11-27 09:28:38 +08:00
|
|
|
vehicle = resource.NewHTTPVehicle(schema.URL, path, schema.Proxy, schema.Header, resource.DefaultHttpTimeout, schema.SizeLimit)
|
2024-12-25 10:23:55 +08:00
|
|
|
case "inline":
|
|
|
|
return NewInlineProvider(name, schema.Payload, parser, hc)
|
2019-12-08 12:17:24 +08:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("%w: %s", errVehicleType, schema.Type)
|
|
|
|
}
|
|
|
|
|
|
|
|
interval := time.Duration(uint(schema.Interval)) * time.Second
|
2023-01-03 21:27:07 +08:00
|
|
|
|
2024-12-25 10:23:55 +08:00
|
|
|
return NewProxySetProvider(name, interval, parser, vehicle, hc)
|
2019-12-08 12:17:24 +08:00
|
|
|
}
|