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
|
|
|
|
}
|
|
|
|
|
2019-10-27 16:02:23 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-07-27 03:57:55 +00:00
|
|
|
func (p *Port) ShouldResolveIP() bool {
|
|
|
|
return false
|
2019-10-27 16:02:23 +00:00
|
|
|
}
|
|
|
|
|
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 {
|
2019-10-27 16:02:23 +00:00
|
|
|
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,
|
2019-10-27 16:02:23 +00:00
|
|
|
}, nil
|
2019-05-09 13:00:29 +00:00
|
|
|
}
|