2018-06-10 22:50:03 +08:00
|
|
|
package constant
|
|
|
|
|
2018-06-11 18:36:39 +08:00
|
|
|
import (
|
2019-10-27 21:44:07 +08:00
|
|
|
"encoding/json"
|
2022-01-05 00:33:42 +08:00
|
|
|
"fmt"
|
2018-06-11 18:36:39 +08:00
|
|
|
"net"
|
2022-04-20 01:52:51 +08:00
|
|
|
"net/netip"
|
2020-01-31 14:43:54 +08:00
|
|
|
"strconv"
|
2018-06-11 18:36:39 +08:00
|
|
|
)
|
|
|
|
|
2018-06-10 22:50:03 +08:00
|
|
|
// Socks addr type
|
|
|
|
const (
|
|
|
|
AtypIPv4 = 1
|
|
|
|
AtypDomainName = 3
|
|
|
|
AtypIPv6 = 4
|
2018-06-14 01:00:58 +08:00
|
|
|
|
2018-07-26 00:04:59 +08:00
|
|
|
TCP NetWork = iota
|
2018-06-14 01:00:58 +08:00
|
|
|
UDP
|
2021-11-17 16:03:47 +08:00
|
|
|
ALLNet
|
2018-07-26 00:04:59 +08:00
|
|
|
|
2019-05-09 21:00:29 +08:00
|
|
|
HTTP Type = iota
|
2019-10-27 21:44:07 +08:00
|
|
|
HTTPCONNECT
|
2021-07-27 13:58:29 +08:00
|
|
|
SOCKS4
|
|
|
|
SOCKS5
|
2018-12-05 21:13:29 +08:00
|
|
|
REDIR
|
2020-11-09 10:46:10 +08:00
|
|
|
TPROXY
|
2021-11-17 16:03:47 +08:00
|
|
|
TUN
|
2022-01-18 21:09:36 +08:00
|
|
|
INNER
|
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"
|
2021-11-17 16:03:47 +08:00
|
|
|
} else if n == UDP {
|
|
|
|
return "udp"
|
2018-06-14 01:00:58 +08:00
|
|
|
}
|
2021-11-17 16:03:47 +08:00
|
|
|
return "all"
|
2018-06-14 01:00:58 +08:00
|
|
|
}
|
|
|
|
|
2019-10-27 21:44:07 +08:00
|
|
|
func (n NetWork) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(n.String())
|
|
|
|
}
|
|
|
|
|
2019-05-09 21:00:29 +08:00
|
|
|
type Type int
|
2018-07-26 00:04:59 +08:00
|
|
|
|
2019-10-27 21:44:07 +08:00
|
|
|
func (t Type) String() string {
|
|
|
|
switch t {
|
|
|
|
case HTTP:
|
|
|
|
return "HTTP"
|
|
|
|
case HTTPCONNECT:
|
|
|
|
return "HTTP Connect"
|
2021-07-27 13:58:29 +08:00
|
|
|
case SOCKS4:
|
|
|
|
return "Socks4"
|
|
|
|
case SOCKS5:
|
2019-10-27 21:44:07 +08:00
|
|
|
return "Socks5"
|
|
|
|
case REDIR:
|
|
|
|
return "Redir"
|
2020-11-09 10:46:10 +08:00
|
|
|
case TPROXY:
|
|
|
|
return "TProxy"
|
2021-11-17 16:03:47 +08:00
|
|
|
case TUN:
|
|
|
|
return "Tun"
|
2022-01-18 21:09:36 +08:00
|
|
|
case INNER:
|
|
|
|
return "Inner"
|
2019-10-27 21:44:07 +08:00
|
|
|
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-04-20 01:52:51 +08:00
|
|
|
NetWork NetWork `json:"network"`
|
|
|
|
Type Type `json:"type"`
|
|
|
|
SrcIP netip.Addr `json:"sourceIP"`
|
|
|
|
DstIP netip.Addr `json:"destinationIP"`
|
|
|
|
SrcPort string `json:"sourcePort"`
|
|
|
|
DstPort string `json:"destinationPort"`
|
|
|
|
AddrType int `json:"-"`
|
|
|
|
Host string `json:"host"`
|
|
|
|
DNSMode DNSMode `json:"dnsMode"`
|
|
|
|
Process string `json:"process"`
|
|
|
|
ProcessPath string `json:"processPath"`
|
|
|
|
UserAgent string `json:"userAgent"`
|
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 {
|
2022-04-21 03:54:34 +08:00
|
|
|
return net.JoinHostPort(m.String(), m.DstPort)
|
2019-10-11 20:11:18 +08:00
|
|
|
}
|
|
|
|
|
2020-01-31 14:58:54 +08:00
|
|
|
func (m *Metadata) SourceAddress() string {
|
|
|
|
return net.JoinHostPort(m.SrcIP.String(), m.SrcPort)
|
|
|
|
}
|
|
|
|
|
2022-01-05 00:33:42 +08:00
|
|
|
func (m *Metadata) SourceDetail() string {
|
|
|
|
if m.Process != "" {
|
|
|
|
return fmt.Sprintf("%s(%s)", m.SourceAddress(), m.Process)
|
|
|
|
} else {
|
2022-01-18 21:09:36 +08:00
|
|
|
if m.Type == INNER {
|
|
|
|
return fmt.Sprintf("[Clash]")
|
|
|
|
}
|
|
|
|
|
2022-01-05 00:33:42 +08:00
|
|
|
return fmt.Sprintf("%s", m.SourceAddress())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-07 20:53:43 +08:00
|
|
|
func (m *Metadata) Resolved() bool {
|
2022-04-20 01:52:51 +08:00
|
|
|
return m.DstIP.IsValid()
|
2020-02-07 20:53:43 +08:00
|
|
|
}
|
|
|
|
|
2021-10-18 21:08:27 +08:00
|
|
|
// Pure is used to solve unexpected behavior
|
|
|
|
// when dialing proxy connection in DNSMapping mode.
|
|
|
|
func (m *Metadata) Pure() *Metadata {
|
2022-04-20 01:52:51 +08:00
|
|
|
if m.DNSMode == DNSMapping && m.DstIP.IsValid() {
|
|
|
|
copyM := *m
|
|
|
|
copyM.Host = ""
|
|
|
|
if copyM.DstIP.Is4() {
|
|
|
|
copyM.AddrType = AtypIPv4
|
2021-10-18 21:08:27 +08:00
|
|
|
} else {
|
2022-04-20 01:52:51 +08:00
|
|
|
copyM.AddrType = AtypIPv6
|
2021-10-18 21:08:27 +08:00
|
|
|
}
|
2022-04-20 01:52:51 +08:00
|
|
|
return ©M
|
2021-10-18 21:08:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2020-01-31 14:43:54 +08:00
|
|
|
func (m *Metadata) UDPAddr() *net.UDPAddr {
|
2022-04-20 01:52:51 +08:00
|
|
|
if m.NetWork != UDP || !m.DstIP.IsValid() {
|
2020-01-31 14:43:54 +08:00
|
|
|
return nil
|
|
|
|
}
|
2022-01-02 01:09:29 +08:00
|
|
|
port, _ := strconv.ParseUint(m.DstPort, 10, 16)
|
2020-01-31 14:43:54 +08:00
|
|
|
return &net.UDPAddr{
|
2022-04-20 01:52:51 +08:00
|
|
|
IP: m.DstIP.AsSlice(),
|
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
|
2022-04-20 01:52:51 +08:00
|
|
|
} else if m.DstIP.IsValid() {
|
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 {
|
2022-04-20 01:52:51 +08:00
|
|
|
return m.Host != "" || m.DstIP.IsValid()
|
2019-02-02 20:47:38 +08:00
|
|
|
}
|