mihomo/adapters/outbound/util.go

107 lines
2.1 KiB
Go
Raw Normal View History

2019-12-08 12:17:24 +08:00
package outbound
2018-08-08 11:51:06 +08:00
import (
2019-04-25 13:48:47 +08:00
"bytes"
"crypto/tls"
2018-08-08 11:51:06 +08:00
"fmt"
"net"
"net/url"
2019-04-25 13:48:47 +08:00
"strconv"
"sync"
2018-08-08 11:51:06 +08:00
"time"
"github.com/Dreamacro/clash/component/resolver"
2019-04-25 13:48:47 +08:00
"github.com/Dreamacro/clash/component/socks5"
2018-08-08 11:51:06 +08:00
C "github.com/Dreamacro/clash/constant"
)
2018-10-22 21:14:22 +08:00
const (
tcpTimeout = 5 * time.Second
)
var (
globalClientSessionCache tls.ClientSessionCache
once sync.Once
)
2018-09-30 12:25:52 +08:00
func urlToMetadata(rawURL string) (addr C.Metadata, err error) {
2018-08-08 11:51:06 +08:00
u, err := url.Parse(rawURL)
if err != nil {
return
}
port := u.Port()
if port == "" {
2019-10-14 17:13:23 +08:00
switch u.Scheme {
case "https":
2018-08-08 11:51:06 +08:00
port = "443"
2019-10-14 17:13:23 +08:00
case "http":
2018-08-08 11:51:06 +08:00
port = "80"
2019-10-14 17:13:23 +08:00
default:
2018-08-08 11:51:06 +08:00
err = fmt.Errorf("%s scheme not Support", rawURL)
return
}
}
2018-09-30 12:25:52 +08:00
addr = C.Metadata{
2018-08-08 11:51:06 +08:00
AddrType: C.AtypDomainName,
Host: u.Hostname(),
2019-05-09 21:00:29 +08:00
DstIP: nil,
DstPort: port,
2018-08-08 11:51:06 +08:00
}
return
}
func tcpKeepAlive(c net.Conn) {
if tcp, ok := c.(*net.TCPConn); ok {
tcp.SetKeepAlive(true)
2018-10-01 19:42:15 +08:00
tcp.SetKeepAlivePeriod(30 * time.Second)
}
}
func getClientSessionCache() tls.ClientSessionCache {
once.Do(func() {
globalClientSessionCache = tls.NewLRUClientSessionCache(128)
})
return globalClientSessionCache
}
2019-04-25 13:48:47 +08:00
func serializesSocksAddr(metadata *C.Metadata) []byte {
var buf [][]byte
aType := uint8(metadata.AddrType)
2019-05-09 21:00:29 +08:00
p, _ := strconv.Atoi(metadata.DstPort)
2019-04-25 13:48:47 +08:00
port := []byte{uint8(p >> 8), uint8(p & 0xff)}
switch metadata.AddrType {
case socks5.AtypDomainName:
len := uint8(len(metadata.Host))
host := []byte(metadata.Host)
buf = [][]byte{{aType, len}, host, port}
case socks5.AtypIPv4:
2019-05-09 21:00:29 +08:00
host := metadata.DstIP.To4()
2019-04-25 13:48:47 +08:00
buf = [][]byte{{aType}, host, port}
case socks5.AtypIPv6:
2019-05-09 21:00:29 +08:00
host := metadata.DstIP.To16()
2019-04-25 13:48:47 +08:00
buf = [][]byte{{aType}, host, port}
}
return bytes.Join(buf, nil)
}
func resolveUDPAddr(network, address string) (*net.UDPAddr, error) {
host, port, err := net.SplitHostPort(address)
if err != nil {
return nil, err
}
ip, err := resolver.ResolveIP(host)
if err != nil {
return nil, err
}
return net.ResolveUDPAddr(network, net.JoinHostPort(ip.String(), port))
}
func safeConnClose(c net.Conn, err error) {
if err != nil {
c.Close()
}
}