add new file

This commit is contained in:
he liu
2022-01-23 17:30:38 +08:00
parent 05b2e55f39
commit c482967f8c
125 changed files with 9688 additions and 0 deletions

141
lib/common/addr.go Normal file
View File

@@ -0,0 +1,141 @@
package common
import (
"errors"
"io"
"net"
"strconv"
)
var AddrError = errors.New("addr error")
// SOCKS address types as defined in RFC 1928 section 5.
const (
AtypIPv4 = 1
AtypDomainName = 3
AtypIPv6 = 4
)
// MaxAddrLen is the maximum size of SOCKS address in bytes.
const MaxAddrLen = 1 + 1 + 255 + 2
// Addr represents a SOCKS address as defined in RFC 1928 section 5.
type Addr []byte
// String serializes SOCKS address a to string form.
func (a Addr) String() string {
var host, port string
switch a[0] { // address type
case AtypDomainName:
host = string(a[2 : 2+int(a[1])])
port = strconv.Itoa((int(a[2+int(a[1])]) << 8) | int(a[2+int(a[1])+1]))
case AtypIPv4:
host = net.IP(a[1 : 1+net.IPv4len]).String()
port = strconv.Itoa((int(a[1+net.IPv4len]) << 8) | int(a[1+net.IPv4len+1]))
case AtypIPv6:
host = net.IP(a[1 : 1+net.IPv6len]).String()
port = strconv.Itoa((int(a[1+net.IPv6len]) << 8) | int(a[1+net.IPv6len+1]))
}
return net.JoinHostPort(host, port)
}
func readAddr(r io.Reader, b []byte) (Addr, error) {
if len(b) < MaxAddrLen {
return nil, io.ErrShortBuffer
}
_, err := io.ReadFull(r, b[:1]) // read 1st byte for address type
if err != nil {
return nil, err
}
switch b[0] {
case AtypDomainName:
_, err = io.ReadFull(r, b[1:2]) // read 2nd byte for domain length
if err != nil {
return nil, err
}
_, err = io.ReadFull(r, b[2:2+int(b[1])+2])
return b[:1+1+int(b[1])+2], err
case AtypIPv4:
_, err = io.ReadFull(r, b[1:1+net.IPv4len+2])
return b[:1+net.IPv4len+2], err
case AtypIPv6:
_, err = io.ReadFull(r, b[1:1+net.IPv6len+2])
return b[:1+net.IPv6len+2], err
}
return nil, errors.New("addr type not supported")
}
// ReadAddr reads just enough bytes from r to get a valid Addr.
func ReadAddr(r io.Reader) (Addr, error) {
return readAddr(r, make([]byte, MaxAddrLen))
}
// SplitAddr slices a SOCKS address from beginning of b. Returns nil if failed.
func SplitAddr(b []byte) (Addr, error) {
addrLen := 1
if len(b) < addrLen {
return nil, AddrError
}
switch b[0] {
case AtypDomainName:
if len(b) < 2 {
return nil, AddrError
}
addrLen = 1 + 1 + int(b[1]) + 2
case AtypIPv4:
addrLen = 1 + net.IPv4len + 2
case AtypIPv6:
addrLen = 1 + net.IPv6len + 2
default:
return nil, AddrError
}
if len(b) < addrLen {
return nil, AddrError
}
return b[:addrLen], nil
}
// ParseAddr parses the address in string s. Returns nil if failed.
func ParseAddr(s string) (Addr, error) {
var addr Addr
host, port, err := net.SplitHostPort(s)
if err != nil {
return nil, AddrError
}
if ip := net.ParseIP(host); ip != nil {
if ip4 := ip.To4(); ip4 != nil {
addr = make([]byte, 1+net.IPv4len+2)
addr[0] = AtypIPv4
copy(addr[1:], ip4)
} else {
addr = make([]byte, 1+net.IPv6len+2)
addr[0] = AtypIPv6
copy(addr[1:], ip)
}
} else {
if len(host) > 255 {
return nil, AddrError
}
addr = make([]byte, 1+1+len(host)+2)
addr[0] = AtypDomainName
addr[1] = byte(len(host))
copy(addr[2:], host)
}
portnum, err := strconv.ParseUint(port, 10, 16)
if err != nil {
return nil, AddrError
}
addr[len(addr)-2], addr[len(addr)-1] = byte(portnum>>8), byte(portnum)
return addr, nil
}

75
lib/common/utils.go Normal file
View File

@@ -0,0 +1,75 @@
package common
import (
"encoding/binary"
"github.com/pkg/errors"
"io"
"strings"
)
// CopyBuffer is an implement of io.Copy with buffer pool
func CopyBuffer(dst io.Writer, src io.Reader, buf []byte) (written int64, err error) {
for {
nr, er := src.Read(buf)
if nr > 0 {
nw, ew := dst.Write(buf[0:nr])
if nw < 0 || nr < nw {
nw = 0
if ew == nil {
ew = errors.New("invalid write result")
}
}
written += int64(nw)
if ew != nil {
err = ew
break
}
if nr != nw {
err = errors.New("short write")
break
}
}
if er != nil {
if er != io.EOF {
err = er
}
break
}
}
return written, err
}
// HostContains tests whether the string host contained ruleHost
func HostContains(ruleHost string, host string) bool {
return strings.HasSuffix(host, strings.Replace(ruleHost, "*", "", -1))
}
// WriteLenBytes is used to write length and bytes to writer
func WriteLenBytes(w io.Writer, b []byte) (int, error) {
err := binary.Write(w, binary.LittleEndian, uint32(len(b)))
if err != nil {
return 0, errors.Wrap(err, "write len")
}
n, err := w.Write(b)
if err != nil {
return 0, errors.Wrap(err, "write bytes")
}
return n, nil
}
// ReadLenBytes is used to read bytes from reader
func ReadLenBytes(r io.Reader, b []byte) (int, error) {
var l int32
err := binary.Read(r, binary.LittleEndian, &l)
if err != nil {
return 0, errors.Wrap(err, "read len")
}
if int(l) > len(b) {
return 0, errors.Errorf("data is too long(%d)", l)
}
n, err := io.ReadAtLeast(r, b, int(l))
if err != nil {
return n, errors.Wrap(err, "read data error")
}
return n, nil
}