mirror of https://github.com/XTLS/Xray-core
				
				
				
			
		
			
				
	
	
		
			260 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			260 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Go
		
	
	
| package protocol
 | |
| 
 | |
| import (
 | |
| 	"io"
 | |
| 
 | |
| 	"github.com/xtls/xray-core/common"
 | |
| 	"github.com/xtls/xray-core/common/buf"
 | |
| 	"github.com/xtls/xray-core/common/errors"
 | |
| 	"github.com/xtls/xray-core/common/net"
 | |
| 	"github.com/xtls/xray-core/common/serial"
 | |
| )
 | |
| 
 | |
| type AddressOption func(*option)
 | |
| 
 | |
| func PortThenAddress() AddressOption {
 | |
| 	return func(p *option) {
 | |
| 		p.portFirst = true
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func AddressFamilyByte(b byte, f net.AddressFamily) AddressOption {
 | |
| 	if b >= 16 {
 | |
| 		panic("address family byte too big")
 | |
| 	}
 | |
| 	return func(p *option) {
 | |
| 		p.addrTypeMap[b] = f
 | |
| 		p.addrByteMap[f] = b
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type AddressTypeParser func(byte) byte
 | |
| 
 | |
| func WithAddressTypeParser(atp AddressTypeParser) AddressOption {
 | |
| 	return func(p *option) {
 | |
| 		p.typeParser = atp
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type AddressSerializer interface {
 | |
| 	ReadAddressPort(buffer *buf.Buffer, input io.Reader) (net.Address, net.Port, error)
 | |
| 
 | |
| 	WriteAddressPort(writer io.Writer, addr net.Address, port net.Port) error
 | |
| }
 | |
| 
 | |
| const afInvalid = 255
 | |
| 
 | |
| type option struct {
 | |
| 	addrTypeMap [16]net.AddressFamily
 | |
| 	addrByteMap [16]byte
 | |
| 	portFirst   bool
 | |
| 	typeParser  AddressTypeParser
 | |
| }
 | |
| 
 | |
| // NewAddressParser creates a new AddressParser
 | |
| func NewAddressParser(options ...AddressOption) AddressSerializer {
 | |
| 	var o option
 | |
| 	for i := range o.addrByteMap {
 | |
| 		o.addrByteMap[i] = afInvalid
 | |
| 	}
 | |
| 	for i := range o.addrTypeMap {
 | |
| 		o.addrTypeMap[i] = net.AddressFamily(afInvalid)
 | |
| 	}
 | |
| 	for _, opt := range options {
 | |
| 		opt(&o)
 | |
| 	}
 | |
| 
 | |
| 	ap := &addressParser{
 | |
| 		addrByteMap: o.addrByteMap,
 | |
| 		addrTypeMap: o.addrTypeMap,
 | |
| 	}
 | |
| 
 | |
| 	if o.typeParser != nil {
 | |
| 		ap.typeParser = o.typeParser
 | |
| 	}
 | |
| 
 | |
| 	if o.portFirst {
 | |
| 		return portFirstAddressParser{ap: ap}
 | |
| 	}
 | |
| 
 | |
| 	return portLastAddressParser{ap: ap}
 | |
| }
 | |
| 
 | |
| type portFirstAddressParser struct {
 | |
| 	ap *addressParser
 | |
| }
 | |
| 
 | |
| func (p portFirstAddressParser) ReadAddressPort(buffer *buf.Buffer, input io.Reader) (net.Address, net.Port, error) {
 | |
| 	if buffer == nil {
 | |
| 		buffer = buf.New()
 | |
| 		defer buffer.Release()
 | |
| 	}
 | |
| 
 | |
| 	port, err := readPort(buffer, input)
 | |
| 	if err != nil {
 | |
| 		return nil, 0, err
 | |
| 	}
 | |
| 
 | |
| 	addr, err := p.ap.readAddress(buffer, input)
 | |
| 	if err != nil {
 | |
| 		return nil, 0, err
 | |
| 	}
 | |
| 	return addr, port, nil
 | |
| }
 | |
| 
 | |
| func (p portFirstAddressParser) WriteAddressPort(writer io.Writer, addr net.Address, port net.Port) error {
 | |
| 	if err := writePort(writer, port); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	return p.ap.writeAddress(writer, addr)
 | |
| }
 | |
| 
 | |
| type portLastAddressParser struct {
 | |
| 	ap *addressParser
 | |
| }
 | |
| 
 | |
| func (p portLastAddressParser) ReadAddressPort(buffer *buf.Buffer, input io.Reader) (net.Address, net.Port, error) {
 | |
| 	if buffer == nil {
 | |
| 		buffer = buf.New()
 | |
| 		defer buffer.Release()
 | |
| 	}
 | |
| 
 | |
| 	addr, err := p.ap.readAddress(buffer, input)
 | |
| 	if err != nil {
 | |
| 		return nil, 0, err
 | |
| 	}
 | |
| 
 | |
| 	port, err := readPort(buffer, input)
 | |
| 	if err != nil {
 | |
| 		return nil, 0, err
 | |
| 	}
 | |
| 
 | |
| 	return addr, port, nil
 | |
| }
 | |
| 
 | |
| func (p portLastAddressParser) WriteAddressPort(writer io.Writer, addr net.Address, port net.Port) error {
 | |
| 	if err := p.ap.writeAddress(writer, addr); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	return writePort(writer, port)
 | |
| }
 | |
| 
 | |
| func readPort(b *buf.Buffer, reader io.Reader) (net.Port, error) {
 | |
| 	if _, err := b.ReadFullFrom(reader, 2); err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return net.PortFromBytes(b.BytesFrom(-2)), nil
 | |
| }
 | |
| 
 | |
| func writePort(writer io.Writer, port net.Port) error {
 | |
| 	return common.Error2(serial.WriteUint16(writer, port.Value()))
 | |
| }
 | |
| 
 | |
| func maybeIPPrefix(b byte) bool {
 | |
| 	return b == '[' || (b >= '0' && b <= '9')
 | |
| }
 | |
| 
 | |
| func isValidDomain(d string) bool {
 | |
| 	for _, c := range d {
 | |
| 		if !((c >= '0' && c <= '9') || (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c == '-' || c == '.' || c == '_') {
 | |
| 			return false
 | |
| 		}
 | |
| 	}
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| type addressParser struct {
 | |
| 	addrTypeMap [16]net.AddressFamily
 | |
| 	addrByteMap [16]byte
 | |
| 	typeParser  AddressTypeParser
 | |
| }
 | |
| 
 | |
| func (p *addressParser) readAddress(b *buf.Buffer, reader io.Reader) (net.Address, error) {
 | |
| 	if _, err := b.ReadFullFrom(reader, 1); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	addrType := b.Byte(b.Len() - 1)
 | |
| 	if p.typeParser != nil {
 | |
| 		addrType = p.typeParser(addrType)
 | |
| 	}
 | |
| 
 | |
| 	if addrType >= 16 {
 | |
| 		return nil, errors.New("unknown address type: ", addrType)
 | |
| 	}
 | |
| 
 | |
| 	addrFamily := p.addrTypeMap[addrType]
 | |
| 	if addrFamily == net.AddressFamily(afInvalid) {
 | |
| 		return nil, errors.New("unknown address type: ", addrType)
 | |
| 	}
 | |
| 
 | |
| 	switch addrFamily {
 | |
| 	case net.AddressFamilyIPv4:
 | |
| 		if _, err := b.ReadFullFrom(reader, 4); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		return net.IPAddress(b.BytesFrom(-4)), nil
 | |
| 	case net.AddressFamilyIPv6:
 | |
| 		if _, err := b.ReadFullFrom(reader, 16); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		return net.IPAddress(b.BytesFrom(-16)), nil
 | |
| 	case net.AddressFamilyDomain:
 | |
| 		if _, err := b.ReadFullFrom(reader, 1); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		domainLength := int32(b.Byte(b.Len() - 1))
 | |
| 		if _, err := b.ReadFullFrom(reader, domainLength); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		domain := string(b.BytesFrom(-domainLength))
 | |
| 		if maybeIPPrefix(domain[0]) {
 | |
| 			addr := net.ParseAddress(domain)
 | |
| 			if addr.Family().IsIP() {
 | |
| 				return addr, nil
 | |
| 			}
 | |
| 		}
 | |
| 		if !isValidDomain(domain) {
 | |
| 			return nil, errors.New("invalid domain name: ", domain)
 | |
| 		}
 | |
| 		return net.DomainAddress(domain), nil
 | |
| 	default:
 | |
| 		panic("impossible case")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *addressParser) writeAddress(writer io.Writer, address net.Address) error {
 | |
| 	tb := p.addrByteMap[address.Family()]
 | |
| 	if tb == afInvalid {
 | |
| 		return errors.New("unknown address family", address.Family())
 | |
| 	}
 | |
| 
 | |
| 	switch address.Family() {
 | |
| 	case net.AddressFamilyIPv4, net.AddressFamilyIPv6:
 | |
| 		if _, err := writer.Write([]byte{tb}); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if _, err := writer.Write(address.IP()); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	case net.AddressFamilyDomain:
 | |
| 		domain := address.Domain()
 | |
| 		if isDomainTooLong(domain) {
 | |
| 			return errors.New("Super long domain is not supported: ", domain)
 | |
| 		}
 | |
| 
 | |
| 		if _, err := writer.Write([]byte{tb, byte(len(domain))}); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if _, err := writer.Write([]byte(domain)); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	default:
 | |
| 		panic("Unknown family type.")
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 |