| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287 | 
							- package socks
 
- import (
 
- 	"bytes"
 
- 	"io/ioutil"
 
- 	"net"
 
- 	"testing"
 
- 	"golang.org/x/net/proxy"
 
- 	"github.com/v2ray/v2ray-core/app/point"
 
- 	"github.com/v2ray/v2ray-core/proxy/common/connhandler"
 
- 	"github.com/v2ray/v2ray-core/proxy/socks/config/json"
 
- 	proxymocks "github.com/v2ray/v2ray-core/proxy/testing/mocks"
 
- 	"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)
 
- 	connInput := []byte("The data to be returned to socks server.")
 
- 	connOutput := bytes.NewBuffer(make([]byte, 0, 1024))
 
- 	och := &proxymocks.OutboundConnectionHandler{
 
- 		ConnOutput: connOutput,
 
- 		ConnInput:  bytes.NewReader(connInput),
 
- 	}
 
- 	connhandler.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(connOutput.Bytes())
 
- 	assert.Bytes(dataReturned).Equals(connInput)
 
- 	assert.String(targetServer).Equals(och.Destination.Address().String())
 
- }
 
- func TestSocksTcpConnectWithUserPass(t *testing.T) {
 
- 	assert := unit.Assert(t)
 
- 	port := uint16(12386)
 
- 	connInput := []byte("The data to be returned to socks server.")
 
- 	connOutput := bytes.NewBuffer(make([]byte, 0, 1024))
 
- 	och := &proxymocks.OutboundConnectionHandler{
 
- 		ConnInput:  bytes.NewReader(connInput),
 
- 		ConnOutput: connOutput,
 
- 	}
 
- 	connhandler.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(connOutput.Bytes())
 
- 	assert.Bytes(dataReturned).Equals(connInput)
 
- 	assert.String(targetServer).Equals(och.Destination.Address().String())
 
- }
 
- func TestSocksTcpConnectWithWrongUserPass(t *testing.T) {
 
- 	assert := unit.Assert(t)
 
- 	port := uint16(12389)
 
- 	connInput := []byte("The data to be returned to socks server.")
 
- 	connOutput := bytes.NewBuffer(make([]byte, 0, 1024))
 
- 	och := &proxymocks.OutboundConnectionHandler{
 
- 		ConnInput:  bytes.NewReader(connInput),
 
- 		ConnOutput: connOutput,
 
- 	}
 
- 	connhandler.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)
 
- 	connInput := []byte("The data to be returned to socks server.")
 
- 	connOutput := bytes.NewBuffer(make([]byte, 0, 1024))
 
- 	och := &proxymocks.OutboundConnectionHandler{
 
- 		ConnInput:  bytes.NewReader(connInput),
 
- 		ConnOutput: connOutput,
 
- 	}
 
- 	connhandler.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)
 
- 	connInput := []byte("The data to be returned to socks server.")
 
- 	connOutput := bytes.NewBuffer(make([]byte, 0, 1024))
 
- 	och := &proxymocks.OutboundConnectionHandler{
 
- 		ConnInput:  bytes.NewReader(connInput),
 
- 		ConnOutput: connOutput,
 
- 	}
 
- 	connhandler.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(connInput)
 
- 	assert.Bytes(data2Send).Equals(connOutput.Bytes())
 
- 	assert.String(och.Destination.String()).Equals("udp:8.8.4.4:53")
 
- }
 
 
  |