clash/tunnel/tunnel.go

355 lines
8.3 KiB
Go
Raw Normal View History

2018-06-10 14:50:03 +00:00
package tunnel
import (
2019-02-02 12:47:38 +00:00
"fmt"
2018-12-05 13:13:29 +00:00
"net"
"runtime"
2018-06-10 14:50:03 +00:00
"sync"
2018-06-16 13:34:13 +00:00
"time"
2018-06-10 14:50:03 +00:00
2019-12-08 04:17:24 +00:00
"github.com/Dreamacro/clash/adapters/inbound"
"github.com/Dreamacro/clash/adapters/provider"
2019-10-11 12:11:18 +00:00
"github.com/Dreamacro/clash/component/nat"
"github.com/Dreamacro/clash/component/resolver"
2018-06-10 14:50:03 +00:00
C "github.com/Dreamacro/clash/constant"
2018-12-05 13:13:29 +00:00
"github.com/Dreamacro/clash/dns"
2018-11-21 05:47:46 +00:00
"github.com/Dreamacro/clash/log"
2018-06-10 14:50:03 +00:00
2019-02-02 12:47:38 +00:00
channels "gopkg.in/eapache/channels.v1"
2018-06-10 14:50:03 +00:00
)
var (
tcpQueue = channels.NewInfiniteChannel()
udpQueue = channels.NewInfiniteChannel()
natTable = nat.New()
rules []C.Rule
proxies = make(map[string]C.Proxy)
providers map[string]provider.ProxyProvider
configMux sync.RWMutex
enhancedMode *dns.Resolver
// experimental features
ignoreResolveFail bool
// Outbound Rule
mode = Rule
// default timeout for UDP session
udpTimeout = 60 * time.Second
)
func init() {
go process()
2018-06-10 14:50:03 +00:00
}
// Add request to queue
func Add(req C.ServerAdapter) {
tcpQueue.In() <- req
}
// AddPacket add udp Packet to queue
func AddPacket(packet *inbound.PacketAdapter) {
udpQueue.In() <- packet
2018-06-10 14:50:03 +00:00
}
2018-11-21 05:47:46 +00:00
// Rules return all rules
func Rules() []C.Rule {
return rules
2018-06-18 03:31:49 +00:00
}
2018-11-21 05:47:46 +00:00
// UpdateRules handle update rules
func UpdateRules(newRules []C.Rule) {
configMux.Lock()
rules = newRules
configMux.Unlock()
2018-11-21 05:47:46 +00:00
}
// Proxies return all proxies
func Proxies() map[string]C.Proxy {
return proxies
2018-11-21 05:47:46 +00:00
}
2019-12-08 04:17:24 +00:00
// Providers return all compatible providers
func Providers() map[string]provider.ProxyProvider {
return providers
2019-12-08 04:17:24 +00:00
}
2018-11-21 05:47:46 +00:00
// UpdateProxies handle update proxies
func UpdateProxies(newProxies map[string]C.Proxy, newProviders map[string]provider.ProxyProvider) {
configMux.Lock()
proxies = newProxies
providers = newProviders
configMux.Unlock()
}
// UpdateExperimental handle update experimental config
func UpdateExperimental(value bool) {
configMux.Lock()
ignoreResolveFail = value
configMux.Unlock()
2018-11-21 05:47:46 +00:00
}
// Mode return current mode
func Mode() TunnelMode {
return mode
2018-11-21 05:47:46 +00:00
}
// SetMode change the mode of tunnel
func SetMode(m TunnelMode) {
mode = m
}
// SetResolver set custom dns resolver for enhanced mode
func SetResolver(r *dns.Resolver) {
enhancedMode = r
2018-06-10 14:50:03 +00:00
}
// processUDP starts a loop to handle udp packet
func processUDP() {
queue := udpQueue.Out()
for elm := range queue {
conn := elm.(*inbound.PacketAdapter)
handleUDPConn(conn)
}
}
func process() {
numUDPWorkers := 4
if runtime.NumCPU() > numUDPWorkers {
numUDPWorkers = runtime.NumCPU()
}
for i := 0; i < numUDPWorkers; i++ {
go processUDP()
}
2019-10-11 12:11:18 +00:00
queue := tcpQueue.Out()
2019-10-11 12:11:18 +00:00
for elm := range queue {
2018-06-10 14:50:03 +00:00
conn := elm.(C.ServerAdapter)
go handleTCPConn(conn)
2018-06-10 14:50:03 +00:00
}
}
func needLookupIP(metadata *C.Metadata) bool {
return enhancedMode != nil && (enhancedMode.IsMapping() || enhancedMode.FakeIPEnabled()) && metadata.Host == "" && metadata.DstIP != nil
2019-02-11 07:44:42 +00:00
}
func preHandleMetadata(metadata *C.Metadata) error {
// handle IP string on host
if ip := net.ParseIP(metadata.Host); ip != nil {
metadata.DstIP = ip
}
2019-05-02 16:05:14 +00:00
// preprocess enhanced-mode metadata
if needLookupIP(metadata) {
host, exist := enhancedMode.IPToHost(metadata.DstIP)
2018-12-05 13:13:29 +00:00
if exist {
metadata.Host = host
metadata.AddrType = C.AtypDomainName
if enhancedMode.FakeIPEnabled() {
2019-05-09 13:00:29 +00:00
metadata.DstIP = nil
2019-05-02 16:05:14 +00:00
}
} else if enhancedMode.IsFakeIP(metadata.DstIP) {
return fmt.Errorf("fake DNS record %s missing", metadata.DstIP)
2018-12-05 13:13:29 +00:00
}
}
return nil
}
func resolveMetadata(metadata *C.Metadata) (C.Proxy, C.Rule, error) {
var proxy C.Proxy
var rule C.Rule
switch mode {
2018-11-21 05:47:46 +00:00
case Direct:
proxy = proxies["DIRECT"]
2018-11-21 05:47:46 +00:00
case Global:
proxy = proxies["GLOBAL"]
// Rule
default:
2019-02-02 12:47:38 +00:00
var err error
proxy, rule, err = match(metadata)
2019-02-02 12:47:38 +00:00
if err != nil {
2019-10-11 12:11:18 +00:00
return nil, nil, err
2019-02-02 12:47:38 +00:00
}
}
2019-10-11 12:11:18 +00:00
return proxy, rule, nil
}
2019-02-02 12:47:38 +00:00
func handleUDPConn(packet *inbound.PacketAdapter) {
metadata := packet.Metadata()
2019-10-11 12:11:18 +00:00
if !metadata.Valid() {
log.Warnln("[Metadata] not valid: %#v", metadata)
return
}
if err := preHandleMetadata(metadata); err != nil {
log.Debugln("[Metadata PreHandle] error: %s", err)
return
2020-01-31 11:26:33 +00:00
}
2020-01-31 06:43:54 +00:00
key := packet.LocalAddr().String()
pc := natTable.Get(key)
2019-10-11 12:11:18 +00:00
if pc != nil {
if !metadata.Resolved() {
ip, err := resolver.ResolveIP(metadata.Host)
if err != nil {
log.Warnln("[UDP] Resolve %s failed: %s, %#v", metadata.Host, err.Error(), metadata)
return
}
metadata.DstIP = ip
}
handleUDPToRemote(packet, pc, metadata.UDPAddr())
2019-10-11 12:11:18 +00:00
return
}
lockKey := key + "-lock"
wg, loaded := natTable.GetOrCreateLock(lockKey)
2019-10-11 12:11:18 +00:00
go func() {
if !loaded {
wg.Add(1)
proxy, rule, err := resolveMetadata(metadata)
2019-10-11 12:11:18 +00:00
if err != nil {
log.Warnln("[UDP] Parse metadata failed: %s", err.Error())
natTable.Delete(lockKey)
2019-10-11 12:11:18 +00:00
wg.Done()
return
}
2020-01-31 06:43:54 +00:00
rawPc, err := proxy.DialUDP(metadata)
2019-10-11 12:11:18 +00:00
if err != nil {
log.Warnln("[UDP] dial %s error: %s", proxy.Name(), err.Error())
natTable.Delete(lockKey)
2019-10-11 12:11:18 +00:00
wg.Done()
return
}
pc = newUDPTracker(rawPc, DefaultManager, metadata, rule)
2019-10-11 12:11:18 +00:00
2020-01-31 06:58:54 +00:00
switch true {
case rule != nil:
log.Infoln("[UDP] %s --> %v match %s using %s", metadata.SourceAddress(), metadata.String(), rule.RuleType().String(), rawPc.Chains().String())
case mode == Global:
2020-01-31 06:58:54 +00:00
log.Infoln("[UDP] %s --> %v using GLOBAL", metadata.SourceAddress(), metadata.String())
case mode == Direct:
2020-01-31 06:58:54 +00:00
log.Infoln("[UDP] %s --> %v using DIRECT", metadata.SourceAddress(), metadata.String())
default:
log.Infoln("[UDP] %s --> %v doesn't match any rule using DIRECT", metadata.SourceAddress(), metadata.String())
2019-10-11 12:11:18 +00:00
}
natTable.Set(key, pc)
natTable.Delete(lockKey)
2019-10-11 12:11:18 +00:00
wg.Done()
go handleUDPToLocal(packet.UDPPacket, pc, key)
2019-04-24 02:29:29 +00:00
}
2019-10-11 12:11:18 +00:00
wg.Wait()
pc := natTable.Get(key)
2019-10-11 12:11:18 +00:00
if pc != nil {
if !metadata.Resolved() {
ip, err := resolver.ResolveIP(metadata.Host)
if err != nil {
return
}
metadata.DstIP = ip
}
handleUDPToRemote(packet, pc, metadata.UDPAddr())
}
2019-10-11 12:11:18 +00:00
}()
}
func handleTCPConn(localConn C.ServerAdapter) {
2019-10-11 12:11:18 +00:00
defer localConn.Close()
metadata := localConn.Metadata()
if !metadata.Valid() {
log.Warnln("[Metadata] not valid: %#v", metadata)
return
2019-04-23 15:29:36 +00:00
}
if err := preHandleMetadata(metadata); err != nil {
log.Debugln("[Metadata PreHandle] error: %s", err)
return
}
proxy, rule, err := resolveMetadata(metadata)
2019-10-11 12:11:18 +00:00
if err != nil {
log.Warnln("Parse metadata failed: %v", err)
return
}
remoteConn, err := proxy.Dial(metadata)
2018-06-10 14:50:03 +00:00
if err != nil {
2019-08-10 12:14:24 +00:00
log.Warnln("dial %s error: %s", proxy.Name(), err.Error())
2018-06-10 14:50:03 +00:00
return
}
remoteConn = newTCPTracker(remoteConn, DefaultManager, metadata, rule)
defer remoteConn.Close()
2018-06-10 14:50:03 +00:00
2020-01-31 06:58:54 +00:00
switch true {
case rule != nil:
log.Infoln("[TCP] %s --> %v match %s using %s", metadata.SourceAddress(), metadata.String(), rule.RuleType().String(), remoteConn.Chains().String())
case mode == Global:
2020-01-31 06:58:54 +00:00
log.Infoln("[TCP] %s --> %v using GLOBAL", metadata.SourceAddress(), metadata.String())
case mode == Direct:
2020-01-31 06:58:54 +00:00
log.Infoln("[TCP] %s --> %v using DIRECT", metadata.SourceAddress(), metadata.String())
default:
log.Infoln("[TCP] %s --> %v doesn't match any rule using DIRECT", metadata.SourceAddress(), metadata.String())
}
switch adapter := localConn.(type) {
2019-12-08 04:17:24 +00:00
case *inbound.HTTPAdapter:
handleHTTP(adapter, remoteConn)
2019-12-08 04:17:24 +00:00
case *inbound.SocketAdapter:
handleSocket(adapter, remoteConn)
}
2018-06-10 14:50:03 +00:00
}
func shouldResolveIP(rule C.Rule, metadata *C.Metadata) bool {
return !rule.NoResolveIP() && metadata.Host != "" && metadata.DstIP == nil
2019-02-02 12:47:38 +00:00
}
func match(metadata *C.Metadata) (C.Proxy, C.Rule, error) {
configMux.RLock()
defer configMux.RUnlock()
2018-06-14 16:49:52 +00:00
var resolved bool
2019-09-11 09:00:55 +00:00
if node := resolver.DefaultHosts.Search(metadata.Host); node != nil {
2019-09-11 09:00:55 +00:00
ip := node.Data.(net.IP)
metadata.DstIP = ip
2019-09-11 09:00:55 +00:00
resolved = true
}
for _, rule := range rules {
if !resolved && shouldResolveIP(rule, metadata) {
ip, err := resolver.ResolveIP(metadata.Host)
2019-02-02 12:47:38 +00:00
if err != nil {
if !ignoreResolveFail {
return nil, nil, fmt.Errorf("[DNS] resolve %s error: %s", metadata.Host, err.Error())
}
log.Debugln("[DNS] resolve %s error: %s", metadata.Host, err.Error())
} else {
log.Debugln("[DNS] %s --> %s", metadata.Host, ip.String())
metadata.DstIP = ip
2019-02-02 12:47:38 +00:00
}
resolved = true
2019-02-02 12:47:38 +00:00
}
if rule.Match(metadata) {
adapter, ok := proxies[rule.Adapter()]
2019-04-23 15:29:36 +00:00
if !ok {
continue
2018-06-10 14:50:03 +00:00
}
2019-04-23 15:29:36 +00:00
if metadata.NetWork == C.UDP && !adapter.SupportUDP() {
log.Debugln("%v UDP is not supported", adapter.Name())
2019-04-23 15:29:36 +00:00
continue
}
return adapter, rule, nil
2018-06-10 14:50:03 +00:00
}
}
return proxies["DIRECT"], nil, nil
2018-06-10 14:50:03 +00:00
}