2018-12-09 21:06:22 +07:00
|
|
|
package proxy
|
2018-01-17 00:09:33 +07:00
|
|
|
|
|
|
|
import (
|
2019-10-12 19:13:12 +07:00
|
|
|
"context"
|
2018-01-17 00:09:33 +07:00
|
|
|
"fmt"
|
2019-10-12 19:13:12 +07:00
|
|
|
"net"
|
2018-01-17 00:09:33 +07:00
|
|
|
"sync"
|
|
|
|
|
2018-12-09 21:06:22 +07:00
|
|
|
"github.com/fatedier/frp/client/event"
|
2018-01-17 00:09:33 +07:00
|
|
|
"github.com/fatedier/frp/models/config"
|
|
|
|
"github.com/fatedier/frp/models/msg"
|
2019-10-12 19:13:12 +07:00
|
|
|
"github.com/fatedier/frp/utils/xlog"
|
2018-05-08 01:13:30 +07:00
|
|
|
|
|
|
|
"github.com/fatedier/golib/errors"
|
2018-01-17 00:09:33 +07:00
|
|
|
)
|
|
|
|
|
|
|
|
type ProxyManager struct {
|
2018-06-25 17:22:35 +07:00
|
|
|
sendCh chan (msg.Message)
|
2018-01-17 00:09:33 +07:00
|
|
|
proxies map[string]*ProxyWrapper
|
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
closed bool
|
|
|
|
mu sync.RWMutex
|
|
|
|
|
2019-08-21 03:53:27 +07:00
|
|
|
clientCfg config.ClientCommonConf
|
|
|
|
|
|
|
|
// The UDP port that the server is listening on
|
|
|
|
serverUDPPort int
|
|
|
|
|
2019-10-12 19:13:12 +07:00
|
|
|
ctx context.Context
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
|
2019-10-12 19:13:12 +07:00
|
|
|
func NewProxyManager(ctx context.Context, msgSendCh chan (msg.Message), clientCfg config.ClientCommonConf, serverUDPPort int) *ProxyManager {
|
2018-01-17 00:09:33 +07:00
|
|
|
return &ProxyManager{
|
2019-08-21 03:53:27 +07:00
|
|
|
sendCh: msgSendCh,
|
2019-10-12 19:13:12 +07:00
|
|
|
proxies: make(map[string]*ProxyWrapper),
|
2019-08-21 03:53:27 +07:00
|
|
|
closed: false,
|
|
|
|
clientCfg: clientCfg,
|
|
|
|
serverUDPPort: serverUDPPort,
|
2019-10-12 19:13:12 +07:00
|
|
|
ctx: ctx,
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-17 13:40:08 +07:00
|
|
|
func (pm *ProxyManager) StartProxy(name string, remoteAddr string, serverRespErr string) error {
|
2018-12-07 16:05:36 +07:00
|
|
|
pm.mu.RLock()
|
2018-01-17 00:09:33 +07:00
|
|
|
pxy, ok := pm.proxies[name]
|
2018-12-07 16:05:36 +07:00
|
|
|
pm.mu.RUnlock()
|
2018-01-17 00:09:33 +07:00
|
|
|
if !ok {
|
2018-12-07 16:05:36 +07:00
|
|
|
return fmt.Errorf("proxy [%s] not found", name)
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
err := pxy.SetRunningStatus(remoteAddr, serverRespErr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
func (pm *ProxyManager) Close() {
|
2018-12-11 14:06:54 +07:00
|
|
|
pm.mu.Lock()
|
|
|
|
defer pm.mu.Unlock()
|
2018-01-17 00:09:33 +07:00
|
|
|
for _, pxy := range pm.proxies {
|
2018-12-07 16:05:36 +07:00
|
|
|
pxy.Stop()
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
2018-12-11 14:06:54 +07:00
|
|
|
pm.proxies = make(map[string]*ProxyWrapper)
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
|
2019-10-12 19:13:12 +07:00
|
|
|
func (pm *ProxyManager) HandleWorkConn(name string, workConn net.Conn, m *msg.StartWorkConn) {
|
2018-01-17 00:09:33 +07:00
|
|
|
pm.mu.RLock()
|
2018-12-07 16:05:36 +07:00
|
|
|
pw, ok := pm.proxies[name]
|
|
|
|
pm.mu.RUnlock()
|
|
|
|
if ok {
|
2019-03-29 18:01:18 +07:00
|
|
|
pw.InWorkConn(workConn, m)
|
2018-12-07 16:05:36 +07:00
|
|
|
} else {
|
|
|
|
workConn.Close()
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
2018-12-07 16:05:36 +07:00
|
|
|
}
|
|
|
|
|
2018-12-09 21:06:22 +07:00
|
|
|
func (pm *ProxyManager) HandleEvent(evType event.EventType, payload interface{}) error {
|
2018-12-07 16:05:36 +07:00
|
|
|
var m msg.Message
|
2018-12-09 21:06:22 +07:00
|
|
|
switch e := payload.(type) {
|
|
|
|
case *event.StartProxyPayload:
|
|
|
|
m = e.NewProxyMsg
|
|
|
|
case *event.CloseProxyPayload:
|
|
|
|
m = e.CloseProxyMsg
|
2018-12-07 16:05:36 +07:00
|
|
|
default:
|
2018-12-09 21:06:22 +07:00
|
|
|
return event.ErrPayloadType
|
2018-12-07 16:05:36 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
err := errors.PanicToError(func() {
|
|
|
|
pm.sendCh <- m
|
|
|
|
})
|
|
|
|
return err
|
|
|
|
}
|
2018-01-17 00:09:33 +07:00
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
func (pm *ProxyManager) GetAllProxyStatus() []*ProxyStatus {
|
|
|
|
ps := make([]*ProxyStatus, 0)
|
|
|
|
pm.mu.RLock()
|
|
|
|
defer pm.mu.RUnlock()
|
2018-01-17 00:09:33 +07:00
|
|
|
for _, pxy := range pm.proxies {
|
2018-12-07 16:05:36 +07:00
|
|
|
ps = append(ps, pxy.GetStatus())
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
2018-12-07 16:05:36 +07:00
|
|
|
return ps
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
func (pm *ProxyManager) Reload(pxyCfgs map[string]config.ProxyConf) {
|
2019-10-12 19:13:12 +07:00
|
|
|
xl := xlog.FromContextSafe(pm.ctx)
|
2018-01-17 00:09:33 +07:00
|
|
|
pm.mu.Lock()
|
2018-12-07 16:05:36 +07:00
|
|
|
defer pm.mu.Unlock()
|
2018-01-17 00:09:33 +07:00
|
|
|
|
|
|
|
delPxyNames := make([]string, 0)
|
|
|
|
for name, pxy := range pm.proxies {
|
|
|
|
del := false
|
|
|
|
cfg, ok := pxyCfgs[name]
|
|
|
|
if !ok {
|
|
|
|
del = true
|
|
|
|
} else {
|
|
|
|
if !pxy.Cfg.Compare(cfg) {
|
|
|
|
del = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if del {
|
|
|
|
delPxyNames = append(delPxyNames, name)
|
|
|
|
delete(pm.proxies, name)
|
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
pxy.Stop()
|
2018-01-17 00:09:33 +07:00
|
|
|
}
|
|
|
|
}
|
2018-12-07 16:05:36 +07:00
|
|
|
if len(delPxyNames) > 0 {
|
2019-10-12 19:13:12 +07:00
|
|
|
xl.Info("proxy removed: %v", delPxyNames)
|
2018-12-07 16:05:36 +07:00
|
|
|
}
|
2018-01-17 00:09:33 +07:00
|
|
|
|
|
|
|
addPxyNames := make([]string, 0)
|
|
|
|
for name, cfg := range pxyCfgs {
|
|
|
|
if _, ok := pm.proxies[name]; !ok {
|
2019-10-12 19:13:12 +07:00
|
|
|
pxy := NewProxyWrapper(pm.ctx, cfg, pm.clientCfg, pm.HandleEvent, pm.serverUDPPort)
|
2018-01-17 00:09:33 +07:00
|
|
|
pm.proxies[name] = pxy
|
|
|
|
addPxyNames = append(addPxyNames, name)
|
2018-11-06 17:35:05 +07:00
|
|
|
|
2018-12-07 16:05:36 +07:00
|
|
|
pxy.Start()
|
|
|
|
}
|
2018-11-06 17:35:05 +07:00
|
|
|
}
|
2018-12-07 16:05:36 +07:00
|
|
|
if len(addPxyNames) > 0 {
|
2019-10-12 19:13:12 +07:00
|
|
|
xl.Info("proxy added: %v", addPxyNames)
|
2018-11-06 17:35:05 +07:00
|
|
|
}
|
|
|
|
}
|