mihomo/tunnel/tunnel.go

162 lines
3.4 KiB
Go
Raw Permalink Normal View History

2018-06-10 22:50:03 +08:00
package tunnel
import (
"sync"
2018-06-16 21:34:13 +08:00
"time"
2018-06-10 22:50:03 +08:00
2018-09-30 12:25:52 +08:00
InboundAdapter "github.com/Dreamacro/clash/adapters/inbound"
"github.com/Dreamacro/clash/common/observable"
cfg "github.com/Dreamacro/clash/config"
2018-06-10 22:50:03 +08:00
C "github.com/Dreamacro/clash/constant"
"gopkg.in/eapache/channels.v1"
)
var (
tunnel *Tunnel
once sync.Once
)
// Tunnel handle proxy socket and HTTP/SOCKS socket
2018-06-10 22:50:03 +08:00
type Tunnel struct {
queue *channels.InfiniteChannel
rules []C.Rule
2018-07-18 21:50:16 +08:00
proxies map[string]C.Proxy
2018-06-15 00:49:52 +08:00
configLock *sync.RWMutex
2018-06-17 22:41:32 +08:00
traffic *C.Traffic
// Outbound Rule
mode cfg.Mode
// Log
logCh chan interface{}
observable *observable.Observable
logLevel C.LogLevel
2018-06-10 22:50:03 +08:00
}
// Add request to queue
2018-06-10 22:50:03 +08:00
func (t *Tunnel) Add(req C.ServerAdapter) {
t.queue.In() <- req
}
// Traffic return traffic of all connections
2018-06-18 11:31:49 +08:00
func (t *Tunnel) Traffic() *C.Traffic {
return t.traffic
}
// Log return clash log stream
2018-06-18 11:31:49 +08:00
func (t *Tunnel) Log() *observable.Observable {
return t.observable
}
func (t *Tunnel) configMonitor(signal chan<- struct{}) {
sub := cfg.Instance().Subscribe()
signal <- struct{}{}
for elm := range sub {
event := elm.(*cfg.Event)
switch event.Type {
case "proxies":
proxies := event.Payload.(map[string]C.Proxy)
t.configLock.Lock()
t.proxies = proxies
t.configLock.Unlock()
case "rules":
rules := event.Payload.([]C.Rule)
t.configLock.Lock()
t.rules = rules
t.configLock.Unlock()
case "mode":
t.mode = event.Payload.(cfg.Mode)
case "log-level":
t.logLevel = event.Payload.(C.LogLevel)
2018-06-10 22:50:03 +08:00
}
}
}
func (t *Tunnel) process() {
queue := t.queue.Out()
for {
elm := <-queue
conn := elm.(C.ServerAdapter)
go t.handleConn(conn)
}
}
func (t *Tunnel) handleConn(localConn C.ServerAdapter) {
defer localConn.Close()
2018-09-30 12:25:52 +08:00
metadata := localConn.Metadata()
var proxy C.Proxy
switch t.mode {
case cfg.Direct:
2018-07-18 21:50:16 +08:00
proxy = t.proxies["DIRECT"]
case cfg.Global:
proxy = t.proxies["GLOBAL"]
// Rule
default:
2018-09-30 12:25:52 +08:00
proxy = t.match(metadata)
}
2018-09-30 12:25:52 +08:00
remoConn, err := proxy.Generator(metadata)
2018-06-10 22:50:03 +08:00
if err != nil {
t.logCh <- newLog(C.WARNING, "Proxy connect error: %s", err.Error())
2018-06-10 22:50:03 +08:00
return
}
defer remoConn.Close()
switch adapter := localConn.(type) {
2018-09-30 12:25:52 +08:00
case *InboundAdapter.HTTPAdapter:
t.handleHTTP(adapter, remoConn)
2018-09-30 12:25:52 +08:00
case *InboundAdapter.SocketAdapter:
t.handleSOCKS(adapter, remoConn)
}
2018-06-10 22:50:03 +08:00
}
2018-09-30 12:25:52 +08:00
func (t *Tunnel) match(metadata *C.Metadata) C.Proxy {
2018-06-15 00:49:52 +08:00
t.configLock.RLock()
defer t.configLock.RUnlock()
2018-06-10 22:50:03 +08:00
for _, rule := range t.rules {
2018-09-30 12:25:52 +08:00
if rule.IsMatch(metadata) {
2018-07-18 21:50:16 +08:00
a, ok := t.proxies[rule.Adapter()]
2018-06-10 22:50:03 +08:00
if !ok {
continue
}
2018-09-30 12:25:52 +08:00
t.logCh <- newLog(C.INFO, "%v match %s using %s", metadata.String(), rule.RuleType().String(), rule.Adapter())
2018-06-10 22:50:03 +08:00
return a
}
}
2018-09-30 12:25:52 +08:00
t.logCh <- newLog(C.INFO, "%v doesn't match any rule using DIRECT", metadata.String())
2018-07-18 21:50:16 +08:00
return t.proxies["DIRECT"]
2018-06-10 22:50:03 +08:00
}
// Run initial task
func (t *Tunnel) Run() {
go t.process()
go t.subscribeLogs()
signal := make(chan struct{})
go t.configMonitor(signal)
<-signal
}
2018-06-10 22:50:03 +08:00
func newTunnel() *Tunnel {
logCh := make(chan interface{})
return &Tunnel{
2018-06-10 22:50:03 +08:00
queue: channels.NewInfiniteChannel(),
2018-07-18 21:50:16 +08:00
proxies: make(map[string]C.Proxy),
2018-06-10 22:50:03 +08:00
observable: observable.NewObservable(logCh),
logCh: logCh,
2018-06-15 00:49:52 +08:00
configLock: &sync.RWMutex{},
2018-06-17 22:41:32 +08:00
traffic: C.NewTraffic(time.Second),
mode: cfg.Rule,
logLevel: C.INFO,
2018-06-10 22:50:03 +08:00
}
}
// Instance return singleton instance of Tunnel
func Instance() *Tunnel {
2018-06-10 22:50:03 +08:00
once.Do(func() {
tunnel = newTunnel()
})
return tunnel
}