无多路复用版本
This commit is contained in:
parent
8e8649db83
commit
1c49851d13
@ -1,9 +1,11 @@
|
|||||||
package shadow
|
package shadow
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"crypto/tls"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Client struct {
|
||||||
@ -22,8 +24,6 @@ func NewClient(listenAddress string, serverAddress string, fakeAddressSNI string
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Start() {
|
func (c *Client) Start() {
|
||||||
bridge := NewTLSBridge(c.ServerAddress, c.FakeAddressSNI)
|
|
||||||
|
|
||||||
listen, err := net.Listen("tcp", c.ListenAddress)
|
listen, err := net.Listen("tcp", c.ListenAddress)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("[Client] Start client error: %v\n", err)
|
fmt.Printf("[Client] Start client error: %v\n", err)
|
||||||
@ -37,15 +37,28 @@ func (c *Client) Start() {
|
|||||||
fmt.Printf("[Client] accept error: %v\n", err)
|
fmt.Printf("[Client] accept error: %v\n", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
go handlerClient(conn, c.ServerAddress, c.FakeAddressSNI)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
stream := bridge.GetStream()
|
func handlerClient(conn net.Conn, serverAddress string, fakeAddressSNI string) {
|
||||||
if stream == nil {
|
dial, err := tls.DialWithDialer(&net.Dialer{
|
||||||
conn.Close()
|
Timeout: time.Second * 5,
|
||||||
fmt.Println("[Client] connect to server error")
|
}, "tcp", serverAddress, &tls.Config{
|
||||||
continue
|
ServerName: fakeAddressSNI,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("[Client] Dial server error: %v\n", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
fmt.Printf("[Client] New TCP connection: %v <-> %v \n", conn.LocalAddr().String(), conn.RemoteAddr().String())
|
err = dial.Handshake()
|
||||||
go io.Copy(conn, stream)
|
if err != nil {
|
||||||
go io.Copy(stream, conn)
|
fmt.Printf("[Client] Handshake error: %v\n", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
dial.NetConn().SetDeadline(time.Now())
|
||||||
|
dial.NetConn().SetDeadline(time.Time{})
|
||||||
|
|
||||||
|
go io.Copy(conn, dial.NetConn())
|
||||||
|
go io.Copy(dial.NetConn(), conn)
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,6 @@ package shadow
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/xtaci/smux"
|
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"time"
|
"time"
|
||||||
@ -61,21 +60,16 @@ func handler(conn net.Conn, targetAddress string, fakeAddress string) {
|
|||||||
conn.SetDeadline(time.Now())
|
conn.SetDeadline(time.Now())
|
||||||
conn.SetDeadline(time.Time{})
|
conn.SetDeadline(time.Time{})
|
||||||
|
|
||||||
//Process real tcp connection
|
realConnection, err := net.Dial("tcp", targetAddress)
|
||||||
session, err := smux.Server(conn, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("[Server] smux error: %v\n", err)
|
fmt.Printf("[Server] Dial target error : %v\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for {
|
|
||||||
stream, err := session.AcceptStream()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("[Server] AcceptStream error: %v\n", err)
|
return
|
||||||
break
|
|
||||||
}
|
}
|
||||||
go handlerMux(stream, targetAddress)
|
go io.Copy(realConnection, conn)
|
||||||
}
|
go io.Copy(conn, realConnection)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func processHandshake(src net.Conn, dst net.Conn, waitCh chan int) {
|
func processHandshake(src net.Conn, dst net.Conn, waitCh chan int) {
|
||||||
@ -123,17 +117,3 @@ func processHandshake(src net.Conn, dst net.Conn, waitCh chan int) {
|
|||||||
}
|
}
|
||||||
waitCh <- 1
|
waitCh <- 1
|
||||||
}
|
}
|
||||||
|
|
||||||
func handlerMux(conn *smux.Stream, targetAddress string) {
|
|
||||||
|
|
||||||
realConnection, err := net.Dial("tcp", targetAddress)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("[Server] Dial target error : %v\n", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
go io.Copy(realConnection, conn)
|
|
||||||
go io.Copy(conn, realConnection)
|
|
||||||
}
|
|
||||||
|
@ -1,75 +0,0 @@
|
|||||||
package shadow
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto/tls"
|
|
||||||
"github.com/xtaci/smux"
|
|
||||||
"net"
|
|
||||||
"sync"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
type TLSBridge struct {
|
|
||||||
session *smux.Session
|
|
||||||
locker sync.Mutex
|
|
||||||
serverAddress string
|
|
||||||
fakeAddressSNI string
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewTLSBridge(serverAddress string, fakeAddressSNI string) *TLSBridge {
|
|
||||||
t := &TLSBridge{
|
|
||||||
session: nil,
|
|
||||||
locker: sync.Mutex{},
|
|
||||||
serverAddress: serverAddress,
|
|
||||||
fakeAddressSNI: fakeAddressSNI,
|
|
||||||
}
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TLSBridge) dial() error {
|
|
||||||
if t.session != nil {
|
|
||||||
t.session.Close()
|
|
||||||
}
|
|
||||||
dial, err := tls.DialWithDialer(&net.Dialer{
|
|
||||||
Timeout: time.Second * 5,
|
|
||||||
}, "tcp", t.serverAddress, &tls.Config{
|
|
||||||
ServerName: t.fakeAddressSNI,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = dial.Handshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
dial.NetConn().SetDeadline(time.Now())
|
|
||||||
dial.NetConn().SetDeadline(time.Time{})
|
|
||||||
time.Sleep(time.Millisecond * 100)
|
|
||||||
session, err := smux.Client(dial.NetConn(), nil)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
//force openStream to prevent first connection problem
|
|
||||||
session.OpenStream()
|
|
||||||
t.session = session
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TLSBridge) GetStream() *smux.Stream {
|
|
||||||
t.locker.Lock()
|
|
||||||
defer t.locker.Unlock()
|
|
||||||
|
|
||||||
if t.session == nil {
|
|
||||||
err := t.dial()
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
openStream, err := t.session.OpenStream()
|
|
||||||
if err != nil {
|
|
||||||
t.session.Close()
|
|
||||||
t.session = nil
|
|
||||||
}
|
|
||||||
return openStream
|
|
||||||
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user