clash/rule/port.go

126 lines
2.3 KiB
Go
Raw Normal View History

2019-05-09 13:00:29 +00:00
package rules
import (
2021-11-17 08:03:47 +00:00
"fmt"
2019-05-09 13:00:29 +00:00
"strconv"
2021-11-17 08:03:47 +00:00
"strings"
2019-05-09 13:00:29 +00:00
C "github.com/Dreamacro/clash/constant"
)
2021-11-17 08:03:47 +00:00
type portReal struct {
portStart int
portEnd int
}
2019-05-09 13:00:29 +00:00
type Port struct {
2021-11-17 08:03:47 +00:00
adapter string
port string
isSource bool
portList []portReal
ruleExtra *C.RuleExtra
2019-05-09 13:00:29 +00:00
}
func (p *Port) RuleType() C.RuleType {
if p.isSource {
return C.SrcPort
}
return C.DstPort
}
func (p *Port) Match(metadata *C.Metadata) bool {
2019-05-09 13:00:29 +00:00
if p.isSource {
2021-11-17 08:03:47 +00:00
return p.matchPortReal(metadata.SrcPort)
2019-05-09 13:00:29 +00:00
}
2021-11-17 08:03:47 +00:00
return p.matchPortReal(metadata.DstPort)
2019-05-09 13:00:29 +00:00
}
func (p *Port) Adapter() string {
return p.adapter
}
func (p *Port) Payload() string {
return p.port
}
func (p *Port) ShouldResolveIP() bool {
return false
}
2021-11-17 08:03:47 +00:00
func (p *Port) RuleExtra() *C.RuleExtra {
return p.ruleExtra
}
func (p *Port) matchPortReal(portRef string) bool {
port, err := strconv.Atoi(portRef)
2019-05-09 13:00:29 +00:00
if err != nil {
2021-11-17 08:03:47 +00:00
return false
}
var rs bool
for _, pr := range p.portList {
if pr.portEnd == -1 {
rs = port == pr.portStart
} else {
rs = port >= pr.portStart && port <= pr.portEnd
}
if rs {
return true
}
}
return false
}
func NewPort(port string, adapter string, isSource bool, ruleExtra *C.RuleExtra) (*Port, error) {
ports := strings.Split(port, "/")
if len(ports) > 28 {
return nil, fmt.Errorf("%s, too many ports to use, maximum support 28 ports", errPayload.Error())
}
var portList []portReal
for _, p := range ports {
if p == "" {
continue
}
subPorts := strings.Split(p, "-")
subPortsLen := len(subPorts)
if subPortsLen > 2 {
return nil, errPayload
}
portStart, err := strconv.Atoi(strings.Trim(subPorts[0], "[ ]"))
if err != nil || portStart < 0 || portStart > 65535 {
return nil, errPayload
}
if subPortsLen == 1 {
portList = append(portList, portReal{portStart, -1})
} else if subPortsLen == 2 {
portEnd, err1 := strconv.Atoi(strings.Trim(subPorts[1], "[ ]"))
if err1 != nil || portEnd < 0 || portEnd > 65535 {
return nil, errPayload
}
shouldReverse := portStart > portEnd
if shouldReverse {
portList = append(portList, portReal{portEnd, portStart})
} else {
portList = append(portList, portReal{portStart, portEnd})
}
}
}
if len(portList) == 0 {
return nil, errPayload
2019-05-09 13:00:29 +00:00
}
2021-11-17 08:03:47 +00:00
2019-05-09 13:00:29 +00:00
return &Port{
2021-11-17 08:03:47 +00:00
adapter: adapter,
port: port,
isSource: isSource,
portList: portList,
ruleExtra: ruleExtra,
}, nil
2019-05-09 13:00:29 +00:00
}