1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2024-09-27 14:26:11 -04:00

Full implementation of Socks5 protocol, proxying is on the way

This commit is contained in:
V2Ray 2015-09-08 13:18:55 +02:00
parent 5317b7d112
commit fb1ad8b81b
7 changed files with 121 additions and 10 deletions

View File

@ -68,12 +68,15 @@ func (r *Socks5AuthenticationResponse) ToBytes() []byte {
func WriteAuthentication(writer io.Writer, response Socks5AuthenticationResponse) error {
_, err := writer.Write(response.ToBytes())
if err != nil {
return err
}
return nil
return err
}
const (
AddrTypeIPv4 = byte(0x01)
AddrTypeIPv6 = byte(0x04)
AddrTypeDomain = byte(0x03)
)
type Socks5Request struct {
version byte
command byte
@ -149,3 +152,51 @@ func ReadRequest(reader io.Reader) (request *Socks5Request, err error) {
request.port = binary.BigEndian.Uint16(buffer)
return
}
const (
ErrorSuccess = byte(0x00)
ErrorGeneralFailure = byte(0x01)
ErrorConnectionNotAllowed = byte(0x02)
ErrorNetworkUnreachable = byte(0x03)
ErrorHostUnUnreachable = byte(0x04)
ErrorConnectionRefused = byte(0x05)
ErrorTTLExpired = byte(0x06)
ErrorCommandNotSupported = byte(0x07)
ErrorAddressTypeNotSupported = byte(0x08)
)
type Socks5Response struct {
Version byte
Error byte
AddrType byte
IPv4 [4]byte
Domain string
IPv6 [16]byte
Port uint16
}
func (r Socks5Response) toBytes() []byte {
buffer := make([]byte, 0, 300)
buffer = append(buffer, r.Version)
buffer = append(buffer, r.Error)
buffer = append(buffer, 0x00) // reserved
buffer = append(buffer, r.AddrType)
switch r.AddrType {
case 0x01:
buffer = append(buffer, r.IPv4[:]...)
case 0x03:
buffer = append(buffer, byte(len(r.Domain)))
buffer = append(buffer, []byte(r.Domain)...)
case 0x04:
buffer = append(buffer, r.IPv6[:]...)
}
portBuffer := make([]byte, 2)
binary.BigEndian.PutUint16(portBuffer, r.Port)
buffer = append(buffer, portBuffer...)
return buffer
}
func WriteResponse(writer io.Writer, response Socks5Response) error {
_, err := writer.Write(response.toBytes())
return err
}

View File

@ -68,3 +68,27 @@ func TestRequestRead(t *testing.T) {
t.Errorf("Expected port 53, but got %d", request.port)
}
}
func TestResponseToBytes(t *testing.T) {
response := Socks5Response{
socksVersion,
ErrorSuccess,
AddrTypeIPv4,
[4]byte{0x72, 0x72, 0x72, 0x72},
"",
[16]byte{},
uint16(53),
}
rawResponse := response.toBytes()
expectedBytes := []byte{
socksVersion,
ErrorSuccess,
byte(0x00),
AddrTypeIPv4,
0x72, 0x72, 0x72, 0x72,
byte(0x00), byte(0x035),
}
if !bytes.Equal(rawResponse, expectedBytes) {
t.Errorf("Expected response %v, but got %v", expectedBytes, rawResponse)
}
}

View File

@ -1,4 +1,4 @@
package net
package socks
import (
"net"

View File

@ -1,17 +1,17 @@
package core
// User account that is used for connection to a VPoint
// VUser is the user account that is used for connection to a VPoint
type VUser struct {
id VID // The ID of this VUser.
Id VID // The ID of this VUser.
}
// The next VPoint server in the connection chain.
// VNext is the next VPoint server in the connection chain.
type VNext struct {
ServerAddress string // Address of VNext server, in the form of "IP:Port"
User []VUser // User accounts for accessing VNext.
}
// The config for VPoint server.
// VConfig is the config for VPoint server.
type VConfig struct {
Port uint16 // Port of this VPoint server.
AllowedClients []VUser

1
vid.go
View File

@ -9,6 +9,7 @@ import (
// The ID of en entity, in the form of an UUID.
type VID [16]byte
// Hash generates a MD5 hash based on current VID and a suffix string.
func (v VID) Hash(suffix []byte) []byte {
md5 := md5.New()
md5.Write(v[:])

View File

@ -4,6 +4,7 @@ import (
"fmt"
)
// VPoint is an single server in V2Ray system.
type VPoint struct {
config VConfig
connHandler ConnectionHandler

View File

@ -1,5 +1,39 @@
package core
import (
"encoding/base64"
)
type VUserSet struct {
validUserIds [][]byte
validUserIds []VID
userIdsAskHash map[string]int
}
func NewVUserSet() *VUserSet {
vuSet := new(VUserSet)
vuSet.validUserIds = make([]VID, 0, 16)
return vuSet
}
func hashBytesToString(hash []byte) string {
return base64.StdEncoding.EncodeToString(hash)
}
func (us *VUserSet) AddUser(user VUser) error {
id := user.Id
us.validUserIds = append(us.validUserIds, id)
idBase64 := hashBytesToString(id.Hash([]byte("ASK")))
us.userIdsAskHash[idBase64] = len(us.validUserIds) - 1
return nil
}
func (us VUserSet) IsValidUserId(askHash []byte) (*VID, bool) {
askBase64 := hashBytesToString(askHash)
idIndex, found := us.userIdsAskHash[askBase64]
if found {
return &us.validUserIds[idIndex], true
}
return nil, false
}