2021-07-04 20:32:59 +08:00
|
|
|
package provider
|
|
|
|
|
|
|
|
import (
|
2024-08-27 11:04:42 +08:00
|
|
|
"context"
|
2024-07-26 22:30:42 +08:00
|
|
|
"fmt"
|
|
|
|
|
2023-11-03 21:01:45 +08:00
|
|
|
"github.com/metacubex/mihomo/common/utils"
|
|
|
|
"github.com/metacubex/mihomo/constant"
|
2021-07-04 20:32:59 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// Vehicle Type
|
|
|
|
const (
|
|
|
|
File VehicleType = iota
|
|
|
|
HTTP
|
|
|
|
Compatible
|
|
|
|
)
|
|
|
|
|
|
|
|
// VehicleType defined
|
|
|
|
type VehicleType int
|
|
|
|
|
|
|
|
func (v VehicleType) String() string {
|
|
|
|
switch v {
|
|
|
|
case File:
|
|
|
|
return "File"
|
|
|
|
case HTTP:
|
|
|
|
return "HTTP"
|
|
|
|
case Compatible:
|
|
|
|
return "Compatible"
|
|
|
|
default:
|
|
|
|
return "Unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type Vehicle interface {
|
2024-09-22 11:36:31 +08:00
|
|
|
Read(ctx context.Context, oldHash HashType) (buf []byte, hash HashType, err error)
|
2021-07-04 20:32:59 +08:00
|
|
|
Path() string
|
2024-09-22 00:24:49 +08:00
|
|
|
Url() string
|
2024-04-12 04:58:07 +08:00
|
|
|
Proxy() string
|
2021-07-04 20:32:59 +08:00
|
|
|
Type() VehicleType
|
|
|
|
}
|
|
|
|
|
|
|
|
// Provider Type
|
|
|
|
const (
|
|
|
|
Proxy ProviderType = iota
|
|
|
|
Rule
|
|
|
|
)
|
|
|
|
|
|
|
|
// ProviderType defined
|
|
|
|
type ProviderType int
|
|
|
|
|
|
|
|
func (pt ProviderType) String() string {
|
|
|
|
switch pt {
|
|
|
|
case Proxy:
|
|
|
|
return "Proxy"
|
|
|
|
case Rule:
|
|
|
|
return "Rule"
|
|
|
|
default:
|
|
|
|
return "Unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Provider interface
|
|
|
|
type Provider interface {
|
|
|
|
Name() string
|
|
|
|
VehicleType() VehicleType
|
|
|
|
Type() ProviderType
|
|
|
|
Initial() error
|
|
|
|
Update() error
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProxyProvider interface
|
|
|
|
type ProxyProvider interface {
|
|
|
|
Provider
|
|
|
|
Proxies() []constant.Proxy
|
2024-09-09 09:15:37 +08:00
|
|
|
Count() int
|
2022-11-04 13:11:01 +08:00
|
|
|
// Touch is used to inform the provider that the proxy is actually being used while getting the list of proxies.
|
2021-10-15 21:44:53 +08:00
|
|
|
// Commonly used in DialContext and DialPacketConn
|
2022-06-07 17:19:25 +08:00
|
|
|
Touch()
|
2021-07-04 20:32:59 +08:00
|
|
|
HealthCheck()
|
2022-07-20 08:53:54 +08:00
|
|
|
Version() uint32
|
2023-06-04 11:51:30 +08:00
|
|
|
RegisterHealthCheckTask(url string, expectedStatus utils.IntRanges[uint16], filter string, interval uint)
|
2024-02-24 14:52:42 +08:00
|
|
|
HealthCheckURL() string
|
2021-07-04 20:32:59 +08:00
|
|
|
}
|
|
|
|
|
2023-04-14 13:51:26 +08:00
|
|
|
// RuleProvider interface
|
|
|
|
type RuleProvider interface {
|
|
|
|
Provider
|
|
|
|
Behavior() RuleBehavior
|
2024-08-27 11:04:42 +08:00
|
|
|
Count() int
|
2023-04-14 13:51:26 +08:00
|
|
|
Match(*constant.Metadata) bool
|
|
|
|
ShouldResolveIP() bool
|
|
|
|
ShouldFindProcess() bool
|
2024-06-17 22:04:51 +08:00
|
|
|
Strategy() any
|
2023-04-14 13:51:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Rule Behavior
|
2021-07-04 20:32:59 +08:00
|
|
|
const (
|
2023-04-14 13:51:26 +08:00
|
|
|
Domain RuleBehavior = iota
|
2021-07-04 20:32:59 +08:00
|
|
|
IPCIDR
|
|
|
|
Classical
|
|
|
|
)
|
|
|
|
|
2023-04-14 13:51:26 +08:00
|
|
|
// RuleBehavior defined
|
|
|
|
type RuleBehavior int
|
2021-07-04 20:32:59 +08:00
|
|
|
|
2023-04-14 13:51:26 +08:00
|
|
|
func (rt RuleBehavior) String() string {
|
2021-07-04 20:32:59 +08:00
|
|
|
switch rt {
|
|
|
|
case Domain:
|
|
|
|
return "Domain"
|
|
|
|
case IPCIDR:
|
|
|
|
return "IPCIDR"
|
|
|
|
case Classical:
|
|
|
|
return "Classical"
|
|
|
|
default:
|
|
|
|
return "Unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-27 10:36:11 +08:00
|
|
|
func (rt RuleBehavior) Byte() byte {
|
|
|
|
switch rt {
|
|
|
|
case Domain:
|
|
|
|
return 0
|
|
|
|
case IPCIDR:
|
|
|
|
return 1
|
|
|
|
case Classical:
|
|
|
|
return 2
|
|
|
|
default:
|
|
|
|
return 255
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-26 22:30:42 +08:00
|
|
|
func ParseBehavior(s string) (behavior RuleBehavior, err error) {
|
|
|
|
switch s {
|
|
|
|
case "domain":
|
|
|
|
behavior = Domain
|
|
|
|
case "ipcidr":
|
|
|
|
behavior = IPCIDR
|
|
|
|
case "classical":
|
|
|
|
behavior = Classical
|
|
|
|
default:
|
|
|
|
err = fmt.Errorf("unsupported behavior type: %s", s)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-04-14 13:51:26 +08:00
|
|
|
const (
|
|
|
|
YamlRule RuleFormat = iota
|
|
|
|
TextRule
|
2024-07-26 22:30:42 +08:00
|
|
|
MrsRule
|
2023-04-14 13:51:26 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type RuleFormat int
|
|
|
|
|
|
|
|
func (rf RuleFormat) String() string {
|
|
|
|
switch rf {
|
|
|
|
case YamlRule:
|
|
|
|
return "YamlRule"
|
|
|
|
case TextRule:
|
|
|
|
return "TextRule"
|
2024-07-26 22:30:42 +08:00
|
|
|
case MrsRule:
|
|
|
|
return "MrsRule"
|
2023-04-14 13:51:26 +08:00
|
|
|
default:
|
|
|
|
return "Unknown"
|
|
|
|
}
|
2021-07-04 20:32:59 +08:00
|
|
|
}
|
2024-06-17 22:04:51 +08:00
|
|
|
|
2024-07-26 22:30:42 +08:00
|
|
|
func ParseRuleFormat(s string) (format RuleFormat, err error) {
|
|
|
|
switch s {
|
|
|
|
case "", "yaml":
|
|
|
|
format = YamlRule
|
|
|
|
case "text":
|
|
|
|
format = TextRule
|
|
|
|
case "mrs":
|
|
|
|
format = MrsRule
|
|
|
|
default:
|
|
|
|
err = fmt.Errorf("unsupported format type: %s", s)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-06-17 22:04:51 +08:00
|
|
|
type Tunnel interface {
|
|
|
|
Providers() map[string]ProxyProvider
|
|
|
|
RuleProviders() map[string]RuleProvider
|
|
|
|
RuleUpdateCallback() *utils.Callback[RuleProvider]
|
|
|
|
}
|