兼容tls1.2握手

增加稳定性
This commit is contained in:
wenyifan 2022-09-01 17:23:45 +08:00
parent 7213e64b89
commit 0b338de152
5 changed files with 50 additions and 120 deletions

View File

@ -41,7 +41,7 @@ func (c *Client) Start() {
stream := bridge.GetStream() stream := bridge.GetStream()
if stream == nil { if stream == nil {
conn.Close() conn.Close()
fmt.Printf("[Client] connect to server error: %v\n", err) fmt.Printf("[Client] connect to server error")
continue continue
} }
fmt.Printf("[Client] New TCP connection: %v <-> %v \n", conn.LocalAddr().String(), conn.RemoteAddr().String()) fmt.Printf("[Client] New TCP connection: %v <-> %v \n", conn.LocalAddr().String(), conn.RemoteAddr().String())

View File

@ -1,128 +1,37 @@
package shadow package shadow
import ( import (
"crypto/tls"
"fmt" "fmt"
"github.com/xtaci/smux"
"io"
"net" "net"
"testing" "testing"
"time"
) )
func Test(t *testing.T) { func TestName(t *testing.T) {
listen, err := net.Listen("tcp", "0.0.0.0:11222") dial, err := tls.DialWithDialer(&net.Dialer{
Timeout: time.Second * 5,
}, "tcp", "www.baidu.com:443", &tls.Config{
ServerName: "www.baidu.com",
})
err = dial.Handshake()
if err != nil { if err != nil {
fmt.Printf("Start server failed : %v\n", err) fmt.Println(err)
return
}
defer listen.Close()
for {
conn, err := listen.Accept()
if err != nil {
fmt.Printf("Accept error: %v\n", err)
continue
}
fmt.Println("A")
go handler(conn)
} }
time.Sleep(time.Minute)
} }
func handler(conn net.Conn) { func TestName2(t *testing.T) {
dial, err := tls.DialWithDialer(&net.Dialer{
Timeout: time.Second * 5,
}, "tcp", "evan.run:443", &tls.Config{
ServerName: "evan.run",
})
fakeConn, err := net.Dial("tcp", "127.0.0.1:5900") err = dial.Handshake()
if err != nil { if err != nil {
fmt.Printf("Dial fake failed : %v\n", err) fmt.Println(err)
return
} }
time.Sleep(time.Minute)
if err != nil {
return
}
go io.Copy(fakeConn, conn)
go io.Copy(conn, fakeConn)
}
func TestSmuxServer(t *testing.T) {
listen, err := net.Listen("tcp", "0.0.0.0:7556")
if err != nil {
fmt.Printf("Start server failed : %v\n", err)
return
}
defer listen.Close()
for {
conn, err := listen.Accept()
if err != nil {
fmt.Printf("Accept error: %v\n", err)
continue
}
session, err := smux.Server(conn, nil)
if err != nil {
fmt.Printf("smux error: %v\n", err)
continue
}
go func() {
for {
stream, err := session.AcceptStream()
if err != nil {
fmt.Printf("AcceptStream error: %v\n", err)
continue
}
fmt.Println("A")
go handlerMuxTest(stream)
}
}()
}
}
func TestSmuxClient(t *testing.T) {
listen, err := net.Listen("tcp", "0.0.0.0:11222")
if err != nil {
fmt.Printf("Start server failed : %v\n", err)
return
}
defer listen.Close()
smuxConn, err := net.Dial("tcp", "127.0.0.1:7556")
if err != nil {
fmt.Printf("Start smuxConn failed : %v\n", err)
return
}
session, err := smux.Client(smuxConn, nil)
if err != nil {
fmt.Printf("Start smux.Client failed : %v\n", err)
return
}
for {
conn, err := listen.Accept()
if err != nil {
fmt.Printf("Accept error: %v\n", err)
continue
}
fmt.Println("A")
stream, err := session.OpenStream()
if err != nil {
fmt.Printf("OpenStream error: %v\n", err)
continue
}
go io.Copy(conn, stream)
go io.Copy(stream, conn)
}
}
func handlerMuxTest(conn *smux.Stream) {
fakeConn, err := net.Dial("tcp", "127.0.0.1:5900")
if err != nil {
fmt.Printf("Dial fake failed : %v\n", err)
return
}
fmt.Println("UUUUUUUUUUUUUU")
if err != nil {
return
}
go io.Copy(fakeConn, conn)
go io.Copy(conn, fakeConn)
} }

View File

@ -49,12 +49,11 @@ func handler(conn net.Conn, targetAddress string, fakeAddress string) {
fmt.Printf("[Server] Dial fake error : %v\n", err) fmt.Printf("[Server] Dial fake error : %v\n", err)
return return
} }
waitCh := make(chan int, 2) waitCh := make(chan int, 1)
go processHandshake(conn, fakeConn, waitCh) go processHandshake(conn, fakeConn, waitCh)
go processHandshake(fakeConn, conn, waitCh) go processHandshake(fakeConn, conn, waitCh)
<-waitCh
<-waitCh <-waitCh
//Process real tcp connection //Process real tcp connection
@ -83,8 +82,16 @@ func processHandshake(src net.Conn, dst net.Conn, waitCh chan int) {
header := ParseAndVerifyTLSHeader(buf[0:nr]) header := ParseAndVerifyTLSHeader(buf[0:nr])
nw, ew := dst.Write(buf[0:nr]) nw, ew := dst.Write(buf[0:nr])
if header != nil && header.Type == ChangeCipherSpec { if header != nil && header.Type == ChangeCipherSpec {
//fmt.Println(header.toString())
fmt.Println("[Server] handshake complete") fmt.Println("[Server] handshake complete")
dst.Close() if header.ChangeCipherSpecNext == AppData {
dst.Close()
waitCh <- 1
} else {
src.Close()
waitCh <- 1
return
}
break break
} }
if nw < 0 || nr < nw { if nw < 0 || nr < nw {

View File

@ -45,6 +45,8 @@ func (t *TLSBridge) dial() error {
if err != nil { if err != nil {
return err return err
} }
//force openStream to prevent first connection problem
session.OpenStream()
t.session = session t.session = session
return nil return nil
} }

View File

@ -22,10 +22,11 @@ const (
) )
type TLSHeader struct { type TLSHeader struct {
Type uint8 Type uint8
Version uint16 Version uint16
Length int Length int
HandshakeType uint8 HandshakeType uint8
ChangeCipherSpecNext uint8
} }
func (t *TLSHeader) toString() string { func (t *TLSHeader) toString() string {
@ -47,6 +48,14 @@ func (t *TLSHeader) toString() string {
break break
case ChangeCipherSpec: case ChangeCipherSpec:
desc += "Type=ChangeCipherSpec;" desc += "Type=ChangeCipherSpec;"
switch t.ChangeCipherSpecNext {
case Handshake:
desc += "ChangeCipherSpecNext=Handshake;"
break
case AppData:
desc += "ChangeCipherSpecNext=AppData;"
break
}
break break
case EncryptedAlert: case EncryptedAlert:
desc += "Type=EncryptedAlert;" desc += "Type=EncryptedAlert;"
@ -83,5 +92,8 @@ func ParseAndVerifyTLSHeader(data []byte) *TLSHeader {
return nil return nil
} }
} }
if header.Type == ChangeCipherSpec {
header.ChangeCipherSpecNext = data[6]
}
return header return header
} }