clash/constant/metadata.go

140 lines
2.5 KiB
Go
Raw Normal View History

2018-06-10 22:50:03 +08:00
package constant
2018-06-11 18:36:39 +08:00
import (
"encoding/json"
2018-06-11 18:36:39 +08:00
"net"
"net/netip"
2020-01-31 14:43:54 +08:00
"strconv"
"github.com/Dreamacro/clash/transport/socks5"
2018-06-11 18:36:39 +08:00
)
2018-06-10 22:50:03 +08:00
// Socks addr type
const (
TCP NetWork = iota
2018-06-14 01:00:58 +08:00
UDP
2019-05-09 21:00:29 +08:00
HTTP Type = iota
HTTPCONNECT
SOCKS4
SOCKS5
2018-12-05 21:13:29 +08:00
REDIR
TPROXY
2022-11-18 22:57:33 +08:00
TUNNEL
2018-06-10 22:50:03 +08:00
)
2018-06-14 01:00:58 +08:00
type NetWork int
2020-06-07 16:54:41 +08:00
func (n NetWork) String() string {
if n == TCP {
2018-06-14 01:00:58 +08:00
return "tcp"
}
return "udp"
}
func (n NetWork) MarshalJSON() ([]byte, error) {
return json.Marshal(n.String())
}
2019-05-09 21:00:29 +08:00
type Type int
func (t Type) String() string {
switch t {
case HTTP:
return "HTTP"
case HTTPCONNECT:
return "HTTP Connect"
case SOCKS4:
return "Socks4"
case SOCKS5:
return "Socks5"
case REDIR:
return "Redir"
case TPROXY:
return "TProxy"
default:
return "Unknown"
}
}
func (t Type) MarshalJSON() ([]byte, error) {
return json.Marshal(t.String())
}
2018-09-30 12:25:52 +08:00
// Metadata is used to store connection address
type Metadata struct {
2022-11-18 22:57:33 +08:00
NetWork NetWork `json:"network"`
Type Type `json:"type"`
SrcIP net.IP `json:"sourceIP"`
DstIP net.IP `json:"destinationIP"`
SrcPort string `json:"sourcePort"`
DstPort string `json:"destinationPort"`
InboundPort string `json:"inboundPort"`
2022-11-18 22:57:33 +08:00
Host string `json:"host"`
DNSMode DNSMode `json:"dnsMode"`
ProcessPath string `json:"processPath"`
SpecialProxy string `json:"specialProxy"`
OriginDst netip.AddrPort `json:"-"`
2018-06-10 22:50:03 +08:00
}
2018-06-11 18:36:39 +08:00
2019-10-11 20:11:18 +08:00
func (m *Metadata) RemoteAddress() string {
return net.JoinHostPort(m.String(), m.DstPort)
}
2020-01-31 14:58:54 +08:00
func (m *Metadata) SourceAddress() string {
return net.JoinHostPort(m.SrcIP.String(), m.SrcPort)
}
func (m *Metadata) AddrType() int {
switch true {
case m.Host != "" || m.DstIP == nil:
return socks5.AtypDomainName
case m.DstIP.To4() != nil:
return socks5.AtypIPv4
default:
return socks5.AtypIPv6
}
}
func (m *Metadata) Resolved() bool {
return m.DstIP != nil
}
// Pure is used to solve unexpected behavior
// when dialing proxy connection in DNSMapping mode.
func (m *Metadata) Pure() *Metadata {
if m.DNSMode == DNSMapping && m.DstIP != nil {
copy := *m
copy.Host = ""
return &copy
}
return m
}
2020-01-31 14:43:54 +08:00
func (m *Metadata) UDPAddr() *net.UDPAddr {
if m.NetWork != UDP || m.DstIP == nil {
return nil
}
port, _ := strconv.ParseUint(m.DstPort, 10, 16)
2020-01-31 14:43:54 +08:00
return &net.UDPAddr{
IP: m.DstIP,
2021-11-08 00:31:08 +08:00
Port: int(port),
2020-01-31 14:43:54 +08:00
}
}
2019-02-02 20:47:38 +08:00
func (m *Metadata) String() string {
2019-10-11 20:11:18 +08:00
if m.Host != "" {
return m.Host
} else if m.DstIP != nil {
2019-05-09 21:00:29 +08:00
return m.DstIP.String()
2019-10-11 20:11:18 +08:00
} else {
return "<nil>"
2018-06-11 18:36:39 +08:00
}
2019-02-02 20:47:38 +08:00
}
func (m *Metadata) Valid() bool {
2019-05-09 21:00:29 +08:00
return m.Host != "" || m.DstIP != nil
2019-02-02 20:47:38 +08:00
}