mirror of https://github.com/v2ray/v2ray-core
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
207 lines
4.0 KiB
207 lines
4.0 KiB
package router
|
|
|
|
import (
|
|
"net"
|
|
"regexp"
|
|
"strings"
|
|
|
|
v2net "v2ray.com/core/common/net"
|
|
"v2ray.com/core/proxy"
|
|
)
|
|
|
|
type Condition interface {
|
|
Apply(session *proxy.SessionInfo) bool
|
|
}
|
|
|
|
type ConditionChan []Condition
|
|
|
|
func NewConditionChan() *ConditionChan {
|
|
var condChan ConditionChan = make([]Condition, 0, 8)
|
|
return &condChan
|
|
}
|
|
|
|
func (this *ConditionChan) Add(cond Condition) *ConditionChan {
|
|
*this = append(*this, cond)
|
|
return this
|
|
}
|
|
|
|
func (this *ConditionChan) Apply(session *proxy.SessionInfo) bool {
|
|
for _, cond := range *this {
|
|
if !cond.Apply(session) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (this *ConditionChan) Len() int {
|
|
return len(*this)
|
|
}
|
|
|
|
type AnyCondition []Condition
|
|
|
|
func NewAnyCondition() *AnyCondition {
|
|
var anyCond AnyCondition = make([]Condition, 0, 8)
|
|
return &anyCond
|
|
}
|
|
|
|
func (this *AnyCondition) Add(cond Condition) *AnyCondition {
|
|
*this = append(*this, cond)
|
|
return this
|
|
}
|
|
|
|
func (this *AnyCondition) Apply(session *proxy.SessionInfo) bool {
|
|
for _, cond := range *this {
|
|
if cond.Apply(session) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (this *AnyCondition) Len() int {
|
|
return len(*this)
|
|
}
|
|
|
|
type PlainDomainMatcher struct {
|
|
pattern string
|
|
}
|
|
|
|
func NewPlainDomainMatcher(pattern string) *PlainDomainMatcher {
|
|
return &PlainDomainMatcher{
|
|
pattern: pattern,
|
|
}
|
|
}
|
|
|
|
func (this *PlainDomainMatcher) Apply(session *proxy.SessionInfo) bool {
|
|
dest := session.Destination
|
|
if !dest.Address.Family().IsDomain() {
|
|
return false
|
|
}
|
|
domain := dest.Address.Domain()
|
|
return strings.Contains(domain, this.pattern)
|
|
}
|
|
|
|
type RegexpDomainMatcher struct {
|
|
pattern *regexp.Regexp
|
|
}
|
|
|
|
func NewRegexpDomainMatcher(pattern string) (*RegexpDomainMatcher, error) {
|
|
r, err := regexp.Compile(pattern)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &RegexpDomainMatcher{
|
|
pattern: r,
|
|
}, nil
|
|
}
|
|
|
|
func (this *RegexpDomainMatcher) Apply(session *proxy.SessionInfo) bool {
|
|
dest := session.Destination
|
|
if !dest.Address.Family().IsDomain() {
|
|
return false
|
|
}
|
|
domain := dest.Address.Domain()
|
|
return this.pattern.MatchString(strings.ToLower(domain))
|
|
}
|
|
|
|
type CIDRMatcher struct {
|
|
cidr *net.IPNet
|
|
onSource bool
|
|
}
|
|
|
|
func NewCIDRMatcher(ip []byte, mask uint32, onSource bool) (*CIDRMatcher, error) {
|
|
cidr := &net.IPNet{
|
|
IP: net.IP(ip),
|
|
Mask: net.CIDRMask(int(mask), len(ip)),
|
|
}
|
|
return &CIDRMatcher{
|
|
cidr: cidr,
|
|
onSource: onSource,
|
|
}, nil
|
|
}
|
|
|
|
func (this *CIDRMatcher) Apply(session *proxy.SessionInfo) bool {
|
|
dest := session.Destination
|
|
if this.onSource {
|
|
dest = session.Source
|
|
}
|
|
if !dest.Address.Family().Either(v2net.AddressFamilyIPv4, v2net.AddressFamilyIPv6) {
|
|
return false
|
|
}
|
|
return this.cidr.Contains(dest.Address.IP())
|
|
}
|
|
|
|
type IPv4Matcher struct {
|
|
ipv4net *v2net.IPNet
|
|
onSource bool
|
|
}
|
|
|
|
func NewIPv4Matcher(ipnet *v2net.IPNet, onSource bool) *IPv4Matcher {
|
|
return &IPv4Matcher{
|
|
ipv4net: ipnet,
|
|
onSource: onSource,
|
|
}
|
|
}
|
|
|
|
func (this *IPv4Matcher) Apply(session *proxy.SessionInfo) bool {
|
|
dest := session.Destination
|
|
if this.onSource {
|
|
dest = session.Source
|
|
}
|
|
if !dest.Address.Family().Either(v2net.AddressFamilyIPv4) {
|
|
return false
|
|
}
|
|
return this.ipv4net.Contains(dest.Address.IP())
|
|
}
|
|
|
|
type PortMatcher struct {
|
|
port v2net.PortRange
|
|
}
|
|
|
|
func NewPortMatcher(portRange v2net.PortRange) *PortMatcher {
|
|
return &PortMatcher{
|
|
port: portRange,
|
|
}
|
|
}
|
|
|
|
func (this *PortMatcher) Apply(session *proxy.SessionInfo) bool {
|
|
return this.port.Contains(session.Destination.Port)
|
|
}
|
|
|
|
type NetworkMatcher struct {
|
|
network *v2net.NetworkList
|
|
}
|
|
|
|
func NewNetworkMatcher(network *v2net.NetworkList) *NetworkMatcher {
|
|
return &NetworkMatcher{
|
|
network: network,
|
|
}
|
|
}
|
|
|
|
func (this *NetworkMatcher) Apply(session *proxy.SessionInfo) bool {
|
|
return this.network.HasNetwork(session.Destination.Network)
|
|
}
|
|
|
|
type UserMatcher struct {
|
|
user []string
|
|
}
|
|
|
|
func NewUserMatcher(users []string) *UserMatcher {
|
|
return &UserMatcher{
|
|
user: users,
|
|
}
|
|
}
|
|
|
|
func (this *UserMatcher) Apply(session *proxy.SessionInfo) bool {
|
|
if session.User == nil {
|
|
return false
|
|
}
|
|
for _, u := range this.user {
|
|
if u == session.User.Email {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|