mirror of
https://github.com/ehang-io/nps.git
synced 2025-08-31 17:56:56 +00:00
P2p、install、log package
This commit is contained in:
@@ -17,10 +17,14 @@ type Service interface {
|
||||
Close() error
|
||||
}
|
||||
|
||||
type NetBridge interface {
|
||||
SendLinkInfo(clientId int, link *conn.Link, t *file.Tunnel) (target net.Conn, err error)
|
||||
}
|
||||
|
||||
//BaseServer struct
|
||||
type BaseServer struct {
|
||||
id int
|
||||
bridge *bridge.Bridge
|
||||
bridge NetBridge
|
||||
task *file.Tunnel
|
||||
errorContent []byte
|
||||
sync.Mutex
|
||||
|
@@ -147,7 +147,7 @@ func (s *httpServer) httpHandle(c *conn.Conn, r *http.Request) {
|
||||
logs.Warn(err.Error())
|
||||
break
|
||||
}
|
||||
lk = conn.NewLink(common.CONN_TCP, targetAddr, host.Client.Cnf.Crypt, host.Client.Cnf.Compress, r.RemoteAddr, host.Target.LocalProxy)
|
||||
lk = conn.NewLink("http", targetAddr, host.Client.Cnf.Crypt, host.Client.Cnf.Compress, r.RemoteAddr, host.Target.LocalProxy)
|
||||
if target, err = s.bridge.SendLinkInfo(host.Client.Id, lk, nil); err != nil {
|
||||
logs.Notice("connect to target %s error %s", lk.Host, err)
|
||||
break
|
||||
|
@@ -1,7 +1,6 @@
|
||||
package proxy
|
||||
|
||||
import (
|
||||
"github.com/cnlh/nps/bridge"
|
||||
"github.com/cnlh/nps/lib/cache"
|
||||
"github.com/cnlh/nps/lib/common"
|
||||
"github.com/cnlh/nps/lib/conn"
|
||||
@@ -22,7 +21,7 @@ type HttpsServer struct {
|
||||
httpsListenerMap sync.Map
|
||||
}
|
||||
|
||||
func NewHttpsServer(l net.Listener, bridge *bridge.Bridge, useCache bool, cacheLen int) *HttpsServer {
|
||||
func NewHttpsServer(l net.Listener, bridge NetBridge, useCache bool, cacheLen int) *HttpsServer {
|
||||
https := &HttpsServer{listener: l}
|
||||
https.bridge = bridge
|
||||
https.useCache = useCache
|
||||
|
@@ -1,28 +1,24 @@
|
||||
package proxy
|
||||
|
||||
import (
|
||||
"encoding/binary"
|
||||
"github.com/cnlh/nps/lib/common"
|
||||
"github.com/cnlh/nps/lib/conn"
|
||||
"github.com/cnlh/nps/lib/pool"
|
||||
"github.com/cnlh/nps/vender/github.com/astaxie/beego/logs"
|
||||
"net"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
type P2PServer struct {
|
||||
BaseServer
|
||||
p2pPort int
|
||||
p2p map[string]*p2p
|
||||
p2pPort int
|
||||
p2p map[string]*p2p
|
||||
listener *net.UDPConn
|
||||
}
|
||||
|
||||
type p2p struct {
|
||||
provider *conn.Conn
|
||||
visitor *conn.Conn
|
||||
visitorAddr string
|
||||
providerAddr string
|
||||
providerNatType int32
|
||||
visitorNatType int32
|
||||
visitorAddr *net.UDPAddr
|
||||
providerAddr *net.UDPAddr
|
||||
}
|
||||
|
||||
func NewP2PServer(p2pPort int) *P2PServer {
|
||||
@@ -33,62 +29,52 @@ func NewP2PServer(p2pPort int) *P2PServer {
|
||||
}
|
||||
|
||||
func (s *P2PServer) Start() error {
|
||||
return conn.NewKcpListenerAndProcess(":"+strconv.Itoa(s.p2pPort), func(c net.Conn) {
|
||||
s.p2pProcess(conn.NewConn(c))
|
||||
})
|
||||
logs.Info("start p2p server port", s.p2pPort)
|
||||
var err error
|
||||
s.listener, err = net.ListenUDP("udp", &net.UDPAddr{net.ParseIP("0.0.0.0"), s.p2pPort, ""})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
for {
|
||||
buf := pool.BufPoolUdp.Get().([]byte)
|
||||
n, addr, err := s.listener.ReadFromUDP(buf)
|
||||
if err != nil {
|
||||
if strings.Contains(err.Error(), "use of closed network connection") {
|
||||
break
|
||||
}
|
||||
continue
|
||||
}
|
||||
go s.handleP2P(addr, string(buf[:n]))
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *P2PServer) p2pProcess(c *conn.Conn) {
|
||||
func (s *P2PServer) handleP2P(addr *net.UDPAddr, str string) {
|
||||
var (
|
||||
f string
|
||||
b []byte
|
||||
err error
|
||||
v *p2p
|
||||
ok bool
|
||||
natType int32
|
||||
v *p2p
|
||||
ok bool
|
||||
)
|
||||
if b, err = c.GetShortContent(32); err != nil {
|
||||
arr := strings.Split(str, common.CONN_DATA_SEQ)
|
||||
if len(arr) < 2 {
|
||||
return
|
||||
}
|
||||
//get role
|
||||
if f, err = c.ReadFlag(); err != nil {
|
||||
return
|
||||
}
|
||||
//get nat type
|
||||
if err := binary.Read(c, binary.LittleEndian, &natType); err != nil {
|
||||
return
|
||||
}
|
||||
if v, ok = s.p2p[string(b)]; !ok {
|
||||
if v, ok = s.p2p[arr[0]]; !ok {
|
||||
v = new(p2p)
|
||||
s.p2p[string(b)] = v
|
||||
s.p2p[arr[0]] = v
|
||||
}
|
||||
logs.Trace("new p2p connection ,role %s , password %s, nat type %s ,local address %s", f, string(b), strconv.Itoa(int(natType)), c.RemoteAddr().String())
|
||||
//存储
|
||||
if f == common.WORK_P2P_VISITOR {
|
||||
v.visitorAddr = c.Conn.RemoteAddr().String()
|
||||
v.visitorNatType = natType
|
||||
v.visitor = c
|
||||
logs.Trace("new p2p connection ,role %s , password %s ,local address %s", arr[1], arr[0], addr.String())
|
||||
if arr[1] == common.WORK_P2P_VISITOR {
|
||||
v.visitorAddr = addr
|
||||
for i := 20; i > 0; i-- {
|
||||
if v.provider != nil {
|
||||
v.provider.WriteLenContent([]byte(v.visitorAddr))
|
||||
binary.Write(v.provider, binary.LittleEndian, v.visitorNatType)
|
||||
if v.providerAddr != nil {
|
||||
s.listener.WriteTo([]byte(v.providerAddr.String()), v.visitorAddr)
|
||||
s.listener.WriteTo([]byte(v.visitorAddr.String()), v.providerAddr)
|
||||
break
|
||||
}
|
||||
time.Sleep(time.Second)
|
||||
}
|
||||
v.provider = nil
|
||||
delete(s.p2p, arr[0])
|
||||
} else {
|
||||
v.providerAddr = c.Conn.RemoteAddr().String()
|
||||
v.providerNatType = natType
|
||||
v.provider = c
|
||||
for i := 20; i > 0; i-- {
|
||||
if v.visitor != nil {
|
||||
v.visitor.WriteLenContent([]byte(v.providerAddr))
|
||||
binary.Write(v.visitor, binary.LittleEndian, v.providerNatType)
|
||||
break
|
||||
}
|
||||
time.Sleep(time.Second)
|
||||
}
|
||||
v.visitor = nil
|
||||
v.providerAddr = addr
|
||||
}
|
||||
}
|
||||
|
@@ -3,7 +3,6 @@ package proxy
|
||||
import (
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"github.com/cnlh/nps/bridge"
|
||||
"github.com/cnlh/nps/lib/common"
|
||||
"github.com/cnlh/nps/lib/conn"
|
||||
"github.com/cnlh/nps/lib/file"
|
||||
@@ -264,7 +263,7 @@ func (s *Sock5ModeServer) Start() error {
|
||||
}
|
||||
|
||||
//new
|
||||
func NewSock5ModeServer(bridge *bridge.Bridge, task *file.Tunnel) *Sock5ModeServer {
|
||||
func NewSock5ModeServer(bridge NetBridge, task *file.Tunnel) *Sock5ModeServer {
|
||||
s := new(Sock5ModeServer)
|
||||
s.bridge = bridge
|
||||
s.task = task
|
||||
@@ -274,4 +273,4 @@ func NewSock5ModeServer(bridge *bridge.Bridge, task *file.Tunnel) *Sock5ModeServ
|
||||
//close
|
||||
func (s *Sock5ModeServer) Close() error {
|
||||
return s.listener.Close()
|
||||
}
|
||||
}
|
@@ -13,6 +13,7 @@ import (
|
||||
"net/http"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"syscall"
|
||||
)
|
||||
|
||||
type TunnelModeServer struct {
|
||||
@@ -22,7 +23,7 @@ type TunnelModeServer struct {
|
||||
}
|
||||
|
||||
//tcp|http|host
|
||||
func NewTunnelModeServer(process process, bridge *bridge.Bridge, task *file.Tunnel) *TunnelModeServer {
|
||||
func NewTunnelModeServer(process process, bridge NetBridge, task *file.Tunnel) *TunnelModeServer {
|
||||
s := new(TunnelModeServer)
|
||||
s.bridge = bridge
|
||||
s.process = process
|
||||
@@ -114,3 +115,35 @@ func ProcessHttp(c *conn.Conn, s *TunnelModeServer) error {
|
||||
}
|
||||
return s.DealClient(c, s.task.Client, addr, rb, common.CONN_TCP, nil, s.task.Flow, s.task.Target.LocalProxy)
|
||||
}
|
||||
|
||||
func HandleTrans(c *conn.Conn, s *TunnelModeServer) error {
|
||||
if addr, err := getAddress(c.Conn); err != nil {
|
||||
return err
|
||||
} else {
|
||||
return s.DealClient(c, s.task.Client, addr, nil, common.CONN_TCP, nil, s.task.Flow, s.task.Target.LocalProxy)
|
||||
}
|
||||
}
|
||||
|
||||
const SO_ORIGINAL_DST = 80
|
||||
|
||||
func getAddress(conn net.Conn) (string, error) {
|
||||
sysrawconn, f := conn.(syscall.Conn)
|
||||
if !f {
|
||||
return "", nil
|
||||
}
|
||||
rawConn, err := sysrawconn.SyscallConn()
|
||||
if err != nil {
|
||||
return "", nil
|
||||
}
|
||||
var ip string
|
||||
var port uint16
|
||||
err = rawConn.Control(func(fd uintptr) {
|
||||
addr, err := syscall.GetsockoptIPv6Mreq(int(fd), syscall.IPPROTO_IP, SO_ORIGINAL_DST)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
ip = net.IP(addr.Multiaddr[4:8]).String()
|
||||
port = uint16(addr.Multiaddr[2])<<8 + uint16(addr.Multiaddr[3])
|
||||
})
|
||||
return ip + ":" + strconv.Itoa(int(port)), nil
|
||||
}
|
||||
|
@@ -90,8 +90,9 @@ func StartNewServer(bridgePort int, cnf *file.Tunnel, bridgeType string) {
|
||||
}
|
||||
}()
|
||||
if p, err := beego.AppConfig.Int("p2p_port"); err == nil {
|
||||
logs.Info("start p2p server port", p)
|
||||
go proxy.NewP2PServer(p).Start()
|
||||
go proxy.NewP2PServer(p + 1).Start()
|
||||
go proxy.NewP2PServer(p + 2).Start()
|
||||
}
|
||||
go DealBridgeTask()
|
||||
go dealClientFlow()
|
||||
@@ -125,6 +126,8 @@ func NewMode(Bridge *bridge.Bridge, c *file.Tunnel) proxy.Service {
|
||||
service = proxy.NewSock5ModeServer(Bridge, c)
|
||||
case "httpProxy":
|
||||
service = proxy.NewTunnelModeServer(proxy.ProcessHttp, Bridge, c)
|
||||
case "tcpTrans":
|
||||
service = proxy.NewTunnelModeServer(proxy.HandleTrans, Bridge, c)
|
||||
case "udp":
|
||||
service = proxy.NewUdpModeServer(Bridge, c)
|
||||
case "webServer":
|
||||
|
Reference in New Issue
Block a user