123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264 |
- // Copyright 2017 fatedier, fatedier@gmail.com
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package visitor
- import (
- "fmt"
- "io"
- "net"
- "strconv"
- "sync"
- "time"
- "github.com/fatedier/golib/errors"
- libio "github.com/fatedier/golib/io"
- v1 "github.com/fatedier/frp/pkg/config/v1"
- "github.com/fatedier/frp/pkg/msg"
- "github.com/fatedier/frp/pkg/proto/udp"
- netpkg "github.com/fatedier/frp/pkg/util/net"
- "github.com/fatedier/frp/pkg/util/util"
- "github.com/fatedier/frp/pkg/util/xlog"
- )
- type SUDPVisitor struct {
- *BaseVisitor
- checkCloseCh chan struct{}
- // udpConn is the listener of udp packet
- udpConn *net.UDPConn
- readCh chan *msg.UDPPacket
- sendCh chan *msg.UDPPacket
- cfg *v1.SUDPVisitorConfig
- }
- // SUDP Run start listen a udp port
- func (sv *SUDPVisitor) Run() (err error) {
- xl := xlog.FromContextSafe(sv.ctx)
- addr, err := net.ResolveUDPAddr("udp", net.JoinHostPort(sv.cfg.BindAddr, strconv.Itoa(sv.cfg.BindPort)))
- if err != nil {
- return fmt.Errorf("sudp ResolveUDPAddr error: %v", err)
- }
- sv.udpConn, err = net.ListenUDP("udp", addr)
- if err != nil {
- return fmt.Errorf("listen udp port %s error: %v", addr.String(), err)
- }
- sv.sendCh = make(chan *msg.UDPPacket, 1024)
- sv.readCh = make(chan *msg.UDPPacket, 1024)
- xl.Infof("sudp start to work, listen on %s", addr)
- go sv.dispatcher()
- go udp.ForwardUserConn(sv.udpConn, sv.readCh, sv.sendCh, int(sv.clientCfg.UDPPacketSize))
- return
- }
- func (sv *SUDPVisitor) dispatcher() {
- xl := xlog.FromContextSafe(sv.ctx)
- var (
- visitorConn net.Conn
- err error
- firstPacket *msg.UDPPacket
- )
- for {
- select {
- case firstPacket = <-sv.sendCh:
- if firstPacket == nil {
- xl.Infof("frpc sudp visitor proxy is closed")
- return
- }
- case <-sv.checkCloseCh:
- xl.Infof("frpc sudp visitor proxy is closed")
- return
- }
- visitorConn, err = sv.getNewVisitorConn()
- if err != nil {
- xl.Warnf("newVisitorConn to frps error: %v, try to reconnect", err)
- continue
- }
- // visitorConn always be closed when worker done.
- sv.worker(visitorConn, firstPacket)
- select {
- case <-sv.checkCloseCh:
- return
- default:
- }
- }
- }
- func (sv *SUDPVisitor) worker(workConn net.Conn, firstPacket *msg.UDPPacket) {
- xl := xlog.FromContextSafe(sv.ctx)
- xl.Debugf("starting sudp proxy worker")
- wg := &sync.WaitGroup{}
- wg.Add(2)
- closeCh := make(chan struct{})
- // udp service -> frpc -> frps -> frpc visitor -> user
- workConnReaderFn := func(conn net.Conn) {
- defer func() {
- conn.Close()
- close(closeCh)
- wg.Done()
- }()
- for {
- var (
- rawMsg msg.Message
- errRet error
- )
- // frpc will send heartbeat in workConn to frpc visitor for keeping alive
- _ = conn.SetReadDeadline(time.Now().Add(60 * time.Second))
- if rawMsg, errRet = msg.ReadMsg(conn); errRet != nil {
- xl.Warnf("read from workconn for user udp conn error: %v", errRet)
- return
- }
- _ = conn.SetReadDeadline(time.Time{})
- switch m := rawMsg.(type) {
- case *msg.Ping:
- xl.Debugf("frpc visitor get ping message from frpc")
- continue
- case *msg.UDPPacket:
- if errRet := errors.PanicToError(func() {
- sv.readCh <- m
- xl.Tracef("frpc visitor get udp packet from workConn: %s", m.Content)
- }); errRet != nil {
- xl.Infof("reader goroutine for udp work connection closed")
- return
- }
- }
- }
- }
- // udp service <- frpc <- frps <- frpc visitor <- user
- workConnSenderFn := func(conn net.Conn) {
- defer func() {
- conn.Close()
- wg.Done()
- }()
- var errRet error
- if firstPacket != nil {
- if errRet = msg.WriteMsg(conn, firstPacket); errRet != nil {
- xl.Warnf("sender goroutine for udp work connection closed: %v", errRet)
- return
- }
- xl.Tracef("send udp package to workConn: %s", firstPacket.Content)
- }
- for {
- select {
- case udpMsg, ok := <-sv.sendCh:
- if !ok {
- xl.Infof("sender goroutine for udp work connection closed")
- return
- }
- if errRet = msg.WriteMsg(conn, udpMsg); errRet != nil {
- xl.Warnf("sender goroutine for udp work connection closed: %v", errRet)
- return
- }
- xl.Tracef("send udp package to workConn: %s", udpMsg.Content)
- case <-closeCh:
- return
- }
- }
- }
- go workConnReaderFn(workConn)
- go workConnSenderFn(workConn)
- wg.Wait()
- xl.Infof("sudp worker is closed")
- }
- func (sv *SUDPVisitor) getNewVisitorConn() (net.Conn, error) {
- xl := xlog.FromContextSafe(sv.ctx)
- visitorConn, err := sv.helper.ConnectServer()
- if err != nil {
- return nil, fmt.Errorf("frpc connect frps error: %v", err)
- }
- now := time.Now().Unix()
- newVisitorConnMsg := &msg.NewVisitorConn{
- RunID: sv.helper.RunID(),
- ProxyName: sv.cfg.ServerName,
- SignKey: util.GetAuthKey(sv.cfg.SecretKey, now),
- Timestamp: now,
- UseEncryption: sv.cfg.Transport.UseEncryption,
- UseCompression: sv.cfg.Transport.UseCompression,
- }
- err = msg.WriteMsg(visitorConn, newVisitorConnMsg)
- if err != nil {
- return nil, fmt.Errorf("frpc send newVisitorConnMsg to frps error: %v", err)
- }
- var newVisitorConnRespMsg msg.NewVisitorConnResp
- _ = visitorConn.SetReadDeadline(time.Now().Add(10 * time.Second))
- err = msg.ReadMsgInto(visitorConn, &newVisitorConnRespMsg)
- if err != nil {
- return nil, fmt.Errorf("frpc read newVisitorConnRespMsg error: %v", err)
- }
- _ = visitorConn.SetReadDeadline(time.Time{})
- if newVisitorConnRespMsg.Error != "" {
- return nil, fmt.Errorf("start new visitor connection error: %s", newVisitorConnRespMsg.Error)
- }
- var remote io.ReadWriteCloser
- remote = visitorConn
- if sv.cfg.Transport.UseEncryption {
- remote, err = libio.WithEncryption(remote, []byte(sv.cfg.SecretKey))
- if err != nil {
- xl.Errorf("create encryption stream error: %v", err)
- return nil, err
- }
- }
- if sv.cfg.Transport.UseCompression {
- remote = libio.WithCompression(remote)
- }
- return netpkg.WrapReadWriteCloserToConn(remote, visitorConn), nil
- }
- func (sv *SUDPVisitor) Close() {
- sv.mu.Lock()
- defer sv.mu.Unlock()
- select {
- case <-sv.checkCloseCh:
- return
- default:
- close(sv.checkCloseCh)
- }
- sv.BaseVisitor.Close()
- if sv.udpConn != nil {
- sv.udpConn.Close()
- }
- close(sv.readCh)
- close(sv.sendCh)
- }
|