mirror of https://github.com/v2ray/v2ray-core
				
				
				
			
		
			
				
	
	
		
			277 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			277 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Go
		
	
	
| package socks
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"io/ioutil"
 | |
| 	"net"
 | |
| 	"testing"
 | |
| 
 | |
| 	"golang.org/x/net/proxy"
 | |
| 
 | |
| 	"github.com/v2ray/v2ray-core/app/point"
 | |
| 	v2proxy "github.com/v2ray/v2ray-core/proxy"
 | |
| 	"github.com/v2ray/v2ray-core/proxy/socks/config/json"
 | |
| 	"github.com/v2ray/v2ray-core/testing/mocks"
 | |
| 	"github.com/v2ray/v2ray-core/testing/unit"
 | |
| )
 | |
| 
 | |
| func TestSocksTcpConnect(t *testing.T) {
 | |
| 	assert := unit.Assert(t)
 | |
| 	port := uint16(12385)
 | |
| 
 | |
| 	och := &mocks.OutboundConnectionHandler{
 | |
| 		Data2Send:   bytes.NewBuffer(make([]byte, 0, 1024)),
 | |
| 		Data2Return: []byte("The data to be returned to socks server."),
 | |
| 	}
 | |
| 
 | |
| 	v2proxy.RegisterOutboundConnectionHandlerFactory("mock_och", och)
 | |
| 
 | |
| 	config := mocks.Config{
 | |
| 		PortValue: port,
 | |
| 		InboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "socks",
 | |
| 			SettingsValue: &json.SocksConfig{
 | |
| 				AuthMethod: "noauth",
 | |
| 			},
 | |
| 		},
 | |
| 		OutboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "mock_och",
 | |
| 			SettingsValue: nil,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	point, err := point.NewPoint(&config)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	err = point.Start()
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	socks5Client, err := proxy.SOCKS5("tcp", "127.0.0.1:12385", nil, proxy.Direct)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	targetServer := "google.com:80"
 | |
| 	conn, err := socks5Client.Dial("tcp", targetServer)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	data2Send := "The data to be sent to remote server."
 | |
| 	conn.Write([]byte(data2Send))
 | |
| 	if tcpConn, ok := conn.(*net.TCPConn); ok {
 | |
| 		tcpConn.CloseWrite()
 | |
| 	}
 | |
| 
 | |
| 	dataReturned, err := ioutil.ReadAll(conn)
 | |
| 	assert.Error(err).IsNil()
 | |
| 	conn.Close()
 | |
| 
 | |
| 	assert.Bytes([]byte(data2Send)).Equals(och.Data2Send.Bytes())
 | |
| 	assert.Bytes(dataReturned).Equals(och.Data2Return)
 | |
| 	assert.String(targetServer).Equals(och.Destination.Address().String())
 | |
| }
 | |
| 
 | |
| func TestSocksTcpConnectWithUserPass(t *testing.T) {
 | |
| 	assert := unit.Assert(t)
 | |
| 	port := uint16(12386)
 | |
| 
 | |
| 	och := &mocks.OutboundConnectionHandler{
 | |
| 		Data2Send:   bytes.NewBuffer(make([]byte, 0, 1024)),
 | |
| 		Data2Return: []byte("The data to be returned to socks server."),
 | |
| 	}
 | |
| 
 | |
| 	v2proxy.RegisterOutboundConnectionHandlerFactory("mock_och", och)
 | |
| 
 | |
| 	config := mocks.Config{
 | |
| 		PortValue: port,
 | |
| 		InboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "socks",
 | |
| 			SettingsValue: &json.SocksConfig{
 | |
| 				AuthMethod: "password",
 | |
| 				Accounts: []json.SocksAccount{
 | |
| 					json.SocksAccount{
 | |
| 						Username: "userx",
 | |
| 						Password: "passy",
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		OutboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "mock_och",
 | |
| 			SettingsValue: nil,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	point, err := point.NewPoint(&config)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	err = point.Start()
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	socks5Client, err := proxy.SOCKS5("tcp", "127.0.0.1:12386", &proxy.Auth{"userx", "passy"}, proxy.Direct)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	targetServer := "1.2.3.4:443"
 | |
| 	conn, err := socks5Client.Dial("tcp", targetServer)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	data2Send := "The data to be sent to remote server."
 | |
| 	conn.Write([]byte(data2Send))
 | |
| 	if tcpConn, ok := conn.(*net.TCPConn); ok {
 | |
| 		tcpConn.CloseWrite()
 | |
| 	}
 | |
| 
 | |
| 	dataReturned, err := ioutil.ReadAll(conn)
 | |
| 	assert.Error(err).IsNil()
 | |
| 	conn.Close()
 | |
| 
 | |
| 	assert.Bytes([]byte(data2Send)).Equals(och.Data2Send.Bytes())
 | |
| 	assert.Bytes(dataReturned).Equals(och.Data2Return)
 | |
| 	assert.String(targetServer).Equals(och.Destination.Address().String())
 | |
| }
 | |
| 
 | |
| func TestSocksTcpConnectWithWrongUserPass(t *testing.T) {
 | |
| 	assert := unit.Assert(t)
 | |
| 	port := uint16(12389)
 | |
| 
 | |
| 	och := &mocks.OutboundConnectionHandler{
 | |
| 		Data2Send:   bytes.NewBuffer(make([]byte, 0, 1024)),
 | |
| 		Data2Return: []byte("The data to be returned to socks server."),
 | |
| 	}
 | |
| 
 | |
| 	v2proxy.RegisterOutboundConnectionHandlerFactory("mock_och", och)
 | |
| 
 | |
| 	config := mocks.Config{
 | |
| 		PortValue: port,
 | |
| 		InboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "socks",
 | |
| 			SettingsValue: &json.SocksConfig{
 | |
| 				AuthMethod: "password",
 | |
| 				Accounts: []json.SocksAccount{
 | |
| 					json.SocksAccount{
 | |
| 						Username: "userx",
 | |
| 						Password: "passy",
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		OutboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "mock_och",
 | |
| 			SettingsValue: nil,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	point, err := point.NewPoint(&config)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	err = point.Start()
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	socks5Client, err := proxy.SOCKS5("tcp", "127.0.0.1:12389", &proxy.Auth{"userx", "passz"}, proxy.Direct)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	targetServer := "1.2.3.4:443"
 | |
| 	_, err = socks5Client.Dial("tcp", targetServer)
 | |
| 	assert.Error(err).IsNotNil()
 | |
| }
 | |
| 
 | |
| func TestSocksTcpConnectWithWrongAuthMethod(t *testing.T) {
 | |
| 	assert := unit.Assert(t)
 | |
| 	port := uint16(38405)
 | |
| 
 | |
| 	och := &mocks.OutboundConnectionHandler{
 | |
| 		Data2Send:   bytes.NewBuffer(make([]byte, 0, 1024)),
 | |
| 		Data2Return: []byte("The data to be returned to socks server."),
 | |
| 	}
 | |
| 
 | |
| 	v2proxy.RegisterOutboundConnectionHandlerFactory("mock_och", och)
 | |
| 
 | |
| 	config := mocks.Config{
 | |
| 		PortValue: port,
 | |
| 		InboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "socks",
 | |
| 			SettingsValue: &json.SocksConfig{
 | |
| 				AuthMethod: "password",
 | |
| 				Accounts: []json.SocksAccount{
 | |
| 					json.SocksAccount{
 | |
| 						Username: "userx",
 | |
| 						Password: "passy",
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		OutboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "mock_och",
 | |
| 			SettingsValue: nil,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	point, err := point.NewPoint(&config)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	err = point.Start()
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	socks5Client, err := proxy.SOCKS5("tcp", "127.0.0.1:38405", nil, proxy.Direct)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	targetServer := "1.2.3.4:443"
 | |
| 	_, err = socks5Client.Dial("tcp", targetServer)
 | |
| 	assert.Error(err).IsNotNil()
 | |
| }
 | |
| 
 | |
| func TestSocksUdpSend(t *testing.T) {
 | |
| 	assert := unit.Assert(t)
 | |
| 	port := uint16(12372)
 | |
| 
 | |
| 	och := &mocks.OutboundConnectionHandler{
 | |
| 		Data2Send:   bytes.NewBuffer(make([]byte, 0, 1024)),
 | |
| 		Data2Return: []byte("The data to be returned to socks server."),
 | |
| 	}
 | |
| 
 | |
| 	v2proxy.RegisterOutboundConnectionHandlerFactory("mock_och", och)
 | |
| 
 | |
| 	config := mocks.Config{
 | |
| 		PortValue: port,
 | |
| 		InboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "socks",
 | |
| 			SettingsValue: &json.SocksConfig{
 | |
| 				AuthMethod: "noauth",
 | |
| 				UDPEnabled: true,
 | |
| 			},
 | |
| 		},
 | |
| 		OutboundConfigValue: &mocks.ConnectionConfig{
 | |
| 			ProtocolValue: "mock_och",
 | |
| 			SettingsValue: nil,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	point, err := point.NewPoint(&config)
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	err = point.Start()
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	conn, err := net.DialUDP("udp", nil, &net.UDPAddr{
 | |
| 		IP:   []byte{127, 0, 0, 1},
 | |
| 		Port: int(port),
 | |
| 		Zone: "",
 | |
| 	})
 | |
| 
 | |
| 	assert.Error(err).IsNil()
 | |
| 
 | |
| 	data2Send := []byte("Fake DNS request")
 | |
| 
 | |
| 	buffer := make([]byte, 0, 1024)
 | |
| 	buffer = append(buffer, 0, 0, 0)
 | |
| 	buffer = append(buffer, 1, 8, 8, 4, 4, 0, 53)
 | |
| 	buffer = append(buffer, data2Send...)
 | |
| 
 | |
| 	conn.Write(buffer)
 | |
| 
 | |
| 	response := make([]byte, 1024)
 | |
| 	nBytes, err := conn.Read(response)
 | |
| 
 | |
| 	assert.Error(err).IsNil()
 | |
| 	assert.Bytes(response[10:nBytes]).Equals(och.Data2Return)
 | |
| 	assert.Bytes(data2Send).Equals(och.Data2Send.Bytes())
 | |
| 	assert.String(och.Destination.String()).Equals("udp:8.8.4.4:53")
 | |
| }
 |