| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- package socks
- import (
- "bytes"
- "fmt"
- "io/ioutil"
- "net"
- "testing"
- "golang.org/x/net/proxy"
- v2nettesting "github.com/v2ray/v2ray-core/common/net/testing"
- "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/shell/point"
- "github.com/v2ray/v2ray-core/shell/point/config/testing/mocks"
- v2testing "github.com/v2ray/v2ray-core/testing"
- "github.com/v2ray/v2ray-core/testing/assert"
- )
- func TestSocksTcpConnect(t *testing.T) {
- v2testing.Current(t)
- port := v2nettesting.PickPort()
- 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", fmt.Sprintf("127.0.0.1:%d", port), 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.StringLiteral(targetServer).Equals(och.Destination.Address().String())
- }
- func TestSocksTcpConnectWithUserPass(t *testing.T) {
- v2testing.Current(t)
- port := v2nettesting.PickPort()
- 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.SocksAccountMap{
- "userx": "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", fmt.Sprintf("127.0.0.1:%d", port), &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.StringLiteral(targetServer).Equals(och.Destination.Address().String())
- }
- func TestSocksTcpConnectWithWrongUserPass(t *testing.T) {
- v2testing.Current(t)
- port := v2nettesting.PickPort()
- 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.SocksAccountMap{
- "userx": "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", fmt.Sprintf("127.0.0.1:%d", port), &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) {
- v2testing.Current(t)
- port := v2nettesting.PickPort()
- 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.SocksAccountMap{
- "userx": "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", fmt.Sprintf("127.0.0.1:%d", port), 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) {
- v2testing.Current(t)
- port := v2nettesting.PickPort()
- 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.StringLiteral(och.Destination.String()).Equals("udp:8.8.4.4:53")
- }
|