2018-11-21 13:47:46 +08:00
|
|
|
package executor
|
|
|
|
|
|
|
|
import (
|
2019-12-01 13:22:47 +08:00
|
|
|
"fmt"
|
|
|
|
"os"
|
2020-06-18 18:11:02 +08:00
|
|
|
"sync"
|
2019-12-01 13:22:47 +08:00
|
|
|
|
2021-06-10 14:05:56 +08:00
|
|
|
"github.com/Dreamacro/clash/adapter"
|
|
|
|
"github.com/Dreamacro/clash/adapter/outboundgroup"
|
2019-06-27 17:04:25 +08:00
|
|
|
"github.com/Dreamacro/clash/component/auth"
|
2020-02-15 21:42:46 +08:00
|
|
|
"github.com/Dreamacro/clash/component/dialer"
|
2021-09-06 23:07:34 +08:00
|
|
|
"github.com/Dreamacro/clash/component/iface"
|
2021-02-18 23:41:50 +08:00
|
|
|
"github.com/Dreamacro/clash/component/profile"
|
|
|
|
"github.com/Dreamacro/clash/component/profile/cachefile"
|
2020-02-15 21:42:46 +08:00
|
|
|
"github.com/Dreamacro/clash/component/resolver"
|
2020-05-28 12:13:05 +08:00
|
|
|
"github.com/Dreamacro/clash/component/trie"
|
2018-11-21 13:47:46 +08:00
|
|
|
"github.com/Dreamacro/clash/config"
|
|
|
|
C "github.com/Dreamacro/clash/constant"
|
2021-07-04 20:32:59 +08:00
|
|
|
"github.com/Dreamacro/clash/constant/provider"
|
2018-12-05 21:13:29 +08:00
|
|
|
"github.com/Dreamacro/clash/dns"
|
2022-11-18 22:57:33 +08:00
|
|
|
"github.com/Dreamacro/clash/listener"
|
2021-06-13 17:23:10 +08:00
|
|
|
authStore "github.com/Dreamacro/clash/listener/auth"
|
2018-11-21 13:47:46 +08:00
|
|
|
"github.com/Dreamacro/clash/log"
|
2020-02-15 21:42:46 +08:00
|
|
|
"github.com/Dreamacro/clash/tunnel"
|
2018-11-21 13:47:46 +08:00
|
|
|
)
|
|
|
|
|
2021-10-10 23:44:09 +08:00
|
|
|
var mux sync.Mutex
|
2020-06-18 18:11:02 +08:00
|
|
|
|
2019-12-01 13:22:47 +08:00
|
|
|
func readConfig(path string) ([]byte, error) {
|
|
|
|
if _, err := os.Stat(path); os.IsNotExist(err) {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-10-09 20:35:06 +08:00
|
|
|
data, err := os.ReadFile(path)
|
2019-12-01 13:22:47 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(data) == 0 {
|
2020-08-25 22:19:59 +08:00
|
|
|
return nil, fmt.Errorf("configuration file %s is empty", path)
|
2019-12-01 13:22:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return data, err
|
|
|
|
}
|
|
|
|
|
2018-11-21 13:47:46 +08:00
|
|
|
// Parse config with default config path
|
|
|
|
func Parse() (*config.Config, error) {
|
|
|
|
return ParseWithPath(C.Path.Config())
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseWithPath parse config with custom config path
|
|
|
|
func ParseWithPath(path string) (*config.Config, error) {
|
2019-12-01 13:22:47 +08:00
|
|
|
buf, err := readConfig(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-12-25 15:12:11 +08:00
|
|
|
return ParseWithBytes(buf)
|
2019-12-01 13:22:47 +08:00
|
|
|
}
|
|
|
|
|
2019-12-25 15:12:11 +08:00
|
|
|
// ParseWithBytes config with buffer
|
2019-12-01 13:22:47 +08:00
|
|
|
func ParseWithBytes(buf []byte) (*config.Config, error) {
|
2019-12-25 15:12:11 +08:00
|
|
|
return config.Parse(buf)
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// ApplyConfig dispatch configure to all parts
|
2018-11-30 17:42:40 +08:00
|
|
|
func ApplyConfig(cfg *config.Config, force bool) {
|
2020-06-18 18:11:02 +08:00
|
|
|
mux.Lock()
|
|
|
|
defer mux.Unlock()
|
|
|
|
|
2019-06-27 17:04:25 +08:00
|
|
|
updateUsers(cfg.Users)
|
2019-12-08 12:17:24 +08:00
|
|
|
updateProxies(cfg.Proxies, cfg.Providers)
|
2018-11-21 13:47:46 +08:00
|
|
|
updateRules(cfg.Rules)
|
2019-09-11 17:00:55 +08:00
|
|
|
updateHosts(cfg.Hosts)
|
2021-02-18 23:41:50 +08:00
|
|
|
updateProfile(cfg)
|
2021-07-10 17:01:40 +08:00
|
|
|
updateGeneral(cfg.General, force)
|
|
|
|
updateDNS(cfg.DNS)
|
|
|
|
updateExperimental(cfg)
|
2022-11-18 22:57:33 +08:00
|
|
|
updateTunnels(cfg.Tunnels)
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func GetGeneral() *config.General {
|
2022-11-18 22:57:33 +08:00
|
|
|
ports := listener.GetPorts()
|
2019-06-27 20:45:12 +08:00
|
|
|
authenticator := []string{}
|
|
|
|
if auth := authStore.Authenticator(); auth != nil {
|
|
|
|
authenticator = auth.Users()
|
|
|
|
}
|
|
|
|
|
|
|
|
general := &config.General{
|
2020-06-18 18:11:02 +08:00
|
|
|
Inbound: config.Inbound{
|
|
|
|
Port: ports.Port,
|
|
|
|
SocksPort: ports.SocksPort,
|
|
|
|
RedirPort: ports.RedirPort,
|
2020-11-09 10:46:10 +08:00
|
|
|
TProxyPort: ports.TProxyPort,
|
2020-06-18 18:11:02 +08:00
|
|
|
MixedPort: ports.MixedPort,
|
|
|
|
Authentication: authenticator,
|
2022-11-18 22:57:33 +08:00
|
|
|
AllowLan: listener.AllowLan(),
|
|
|
|
BindAddress: listener.BindAddress(),
|
2020-06-18 18:11:02 +08:00
|
|
|
},
|
|
|
|
Mode: tunnel.Mode(),
|
|
|
|
LogLevel: log.Level(),
|
2021-01-07 13:59:39 +08:00
|
|
|
IPv6: !resolver.DisableIPv6,
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
2019-06-27 20:45:12 +08:00
|
|
|
|
|
|
|
return general
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
|
|
|
|
2020-06-27 14:19:31 +08:00
|
|
|
func updateExperimental(c *config.Config) {}
|
2019-04-24 12:02:52 +08:00
|
|
|
|
2018-12-05 21:13:29 +08:00
|
|
|
func updateDNS(c *config.DNS) {
|
2020-08-25 22:19:59 +08:00
|
|
|
if !c.Enable {
|
2020-02-15 21:42:46 +08:00
|
|
|
resolver.DefaultResolver = nil
|
2020-09-17 10:48:42 +08:00
|
|
|
resolver.DefaultHostMapper = nil
|
|
|
|
dns.ReCreateServer("", nil, nil)
|
2018-12-05 21:13:29 +08:00
|
|
|
return
|
|
|
|
}
|
2020-09-17 10:48:42 +08:00
|
|
|
|
|
|
|
cfg := dns.Config{
|
2018-12-05 21:13:29 +08:00
|
|
|
Main: c.NameServer,
|
|
|
|
Fallback: c.Fallback,
|
|
|
|
IPv6: c.IPv6,
|
|
|
|
EnhancedMode: c.EnhancedMode,
|
2019-05-03 00:05:14 +08:00
|
|
|
Pool: c.FakeIPRange,
|
2020-08-11 10:28:17 +08:00
|
|
|
Hosts: c.Hosts,
|
2019-09-15 13:36:45 +08:00
|
|
|
FallbackFilter: dns.FallbackFilter{
|
2021-08-25 15:15:13 +08:00
|
|
|
GeoIP: c.FallbackFilter.GeoIP,
|
|
|
|
GeoIPCode: c.FallbackFilter.GeoIPCode,
|
|
|
|
IPCIDR: c.FallbackFilter.IPCIDR,
|
|
|
|
Domain: c.FallbackFilter.Domain,
|
2019-09-15 13:36:45 +08:00
|
|
|
},
|
2020-02-15 21:42:46 +08:00
|
|
|
Default: c.DefaultNameserver,
|
2021-05-19 11:17:35 +08:00
|
|
|
Policy: c.NameServerPolicy,
|
2020-09-17 10:48:42 +08:00
|
|
|
}
|
2020-08-31 00:32:18 +08:00
|
|
|
|
2022-06-14 11:26:04 +08:00
|
|
|
// deprecated warnning
|
|
|
|
if cfg.EnhancedMode == C.DNSMapping {
|
|
|
|
log.Warnln("[DNS] %s is deprecated, please use %s instead", cfg.EnhancedMode.String(), C.DNSFakeIP.String())
|
|
|
|
}
|
|
|
|
|
2020-09-17 10:48:42 +08:00
|
|
|
r := dns.NewResolver(cfg)
|
|
|
|
m := dns.NewEnhancer(cfg)
|
|
|
|
|
|
|
|
// reuse cache of old host mapper
|
|
|
|
if old := resolver.DefaultHostMapper; old != nil {
|
|
|
|
m.PatchFrom(old.(*dns.ResolverEnhancer))
|
2020-08-31 00:32:18 +08:00
|
|
|
}
|
|
|
|
|
2020-02-15 21:42:46 +08:00
|
|
|
resolver.DefaultResolver = r
|
2020-09-17 10:48:42 +08:00
|
|
|
resolver.DefaultHostMapper = m
|
|
|
|
|
2021-12-26 22:08:53 +08:00
|
|
|
dns.ReCreateServer(c.Listen, r, m)
|
2018-12-05 21:13:29 +08:00
|
|
|
}
|
|
|
|
|
2020-05-28 12:13:05 +08:00
|
|
|
func updateHosts(tree *trie.DomainTrie) {
|
2020-02-15 21:42:46 +08:00
|
|
|
resolver.DefaultHosts = tree
|
2019-09-11 17:00:55 +08:00
|
|
|
}
|
|
|
|
|
2019-12-08 12:17:24 +08:00
|
|
|
func updateProxies(proxies map[string]C.Proxy, providers map[string]provider.ProxyProvider) {
|
|
|
|
tunnel.UpdateProxies(proxies, providers)
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func updateRules(rules []C.Rule) {
|
2020-02-15 21:42:46 +08:00
|
|
|
tunnel.UpdateRules(rules)
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
|
|
|
|
2022-11-18 22:57:33 +08:00
|
|
|
func updateTunnels(tunnels []config.Tunnel) {
|
|
|
|
listener.PatchTunnel(tunnels, tunnel.TCPIn(), tunnel.UDPIn())
|
|
|
|
}
|
|
|
|
|
2020-06-18 18:11:02 +08:00
|
|
|
func updateGeneral(general *config.General, force bool) {
|
2018-12-03 23:41:40 +08:00
|
|
|
log.SetLevel(general.LogLevel)
|
2020-02-15 21:42:46 +08:00
|
|
|
tunnel.SetMode(general.Mode)
|
2020-06-18 18:11:02 +08:00
|
|
|
resolver.DisableIPv6 = !general.IPv6
|
|
|
|
|
2021-11-07 16:48:51 +08:00
|
|
|
dialer.DefaultInterface.Store(general.Interface)
|
2022-02-17 14:23:47 +08:00
|
|
|
dialer.DefaultRoutingMark.Store(int32(general.RoutingMark))
|
2020-06-27 14:19:31 +08:00
|
|
|
|
2021-09-06 23:07:34 +08:00
|
|
|
iface.FlushCache()
|
|
|
|
|
2020-06-18 18:11:02 +08:00
|
|
|
if !force {
|
|
|
|
return
|
|
|
|
}
|
2018-12-03 23:41:40 +08:00
|
|
|
|
2018-11-21 13:47:46 +08:00
|
|
|
allowLan := general.AllowLan
|
2022-11-18 22:57:33 +08:00
|
|
|
listener.SetAllowLan(allowLan)
|
2019-06-27 17:04:25 +08:00
|
|
|
|
2019-08-08 13:45:07 +08:00
|
|
|
bindAddress := general.BindAddress
|
2022-11-18 22:57:33 +08:00
|
|
|
listener.SetBindAddress(bindAddress)
|
2019-08-08 13:45:07 +08:00
|
|
|
|
2021-06-13 17:23:10 +08:00
|
|
|
tcpIn := tunnel.TCPIn()
|
|
|
|
udpIn := tunnel.UDPIn()
|
|
|
|
|
2022-11-18 22:57:33 +08:00
|
|
|
listener.ReCreateHTTP(general.Port, tcpIn)
|
|
|
|
listener.ReCreateSocks(general.SocksPort, tcpIn, udpIn)
|
|
|
|
listener.ReCreateRedir(general.RedirPort, tcpIn, udpIn)
|
|
|
|
listener.ReCreateTProxy(general.TProxyPort, tcpIn, udpIn)
|
|
|
|
listener.ReCreateMixed(general.MixedPort, tcpIn, udpIn)
|
2018-11-21 13:47:46 +08:00
|
|
|
}
|
2019-06-27 17:04:25 +08:00
|
|
|
|
|
|
|
func updateUsers(users []auth.AuthUser) {
|
|
|
|
authenticator := auth.NewAuthenticator(users)
|
|
|
|
authStore.SetAuthenticator(authenticator)
|
|
|
|
if authenticator != nil {
|
|
|
|
log.Infoln("Authentication of local server updated")
|
|
|
|
}
|
|
|
|
}
|
2021-02-18 23:41:50 +08:00
|
|
|
|
|
|
|
func updateProfile(cfg *config.Config) {
|
|
|
|
profileCfg := cfg.Profile
|
|
|
|
|
|
|
|
profile.StoreSelected.Store(profileCfg.StoreSelected)
|
|
|
|
if profileCfg.StoreSelected {
|
|
|
|
patchSelectGroup(cfg.Proxies)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func patchSelectGroup(proxies map[string]C.Proxy) {
|
|
|
|
mapping := cachefile.Cache().SelectedMap()
|
|
|
|
if mapping == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, proxy := range proxies {
|
2021-06-10 14:05:56 +08:00
|
|
|
outbound, ok := proxy.(*adapter.Proxy)
|
2021-02-18 23:41:50 +08:00
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
selector, ok := outbound.ProxyAdapter.(*outboundgroup.Selector)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
selected, exist := mapping[name]
|
|
|
|
if !exist {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
selector.Set(selected)
|
|
|
|
}
|
|
|
|
}
|