1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-05 00:47:51 -05:00
v2fly/app/router/condition.go

302 lines
5.9 KiB
Go
Raw Normal View History

2016-10-12 10:11:13 -04:00
package router
2016-01-17 10:20:49 -05:00
import (
"context"
2018-07-16 07:47:00 -04:00
"strings"
2016-01-17 10:20:49 -05:00
2018-07-16 07:47:00 -04:00
"v2ray.com/core/app/dispatcher"
2017-08-29 08:32:54 -04:00
"v2ray.com/core/common/net"
2018-10-22 14:59:01 -04:00
"v2ray.com/core/common/session"
2018-06-26 15:57:41 -04:00
"v2ray.com/core/common/strmatcher"
2016-01-17 10:20:49 -05:00
)
type Condition interface {
Apply(ctx context.Context) bool
2016-01-17 10:20:49 -05:00
}
type ConditionChan []Condition
func NewConditionChan() *ConditionChan {
var condChan ConditionChan = make([]Condition, 0, 8)
return &condChan
}
2016-11-27 15:39:09 -05:00
func (v *ConditionChan) Add(cond Condition) *ConditionChan {
*v = append(*v, cond)
return v
2016-01-17 10:20:49 -05:00
}
func (v *ConditionChan) Apply(ctx context.Context) bool {
2016-11-27 15:39:09 -05:00
for _, cond := range *v {
if !cond.Apply(ctx) {
2016-01-17 10:20:49 -05:00
return false
}
}
return true
}
2016-11-27 15:39:09 -05:00
func (v *ConditionChan) Len() int {
return len(*v)
2016-01-17 10:20:49 -05:00
}
2018-06-26 15:57:41 -04:00
var matcherTypeMap = map[Domain_Type]strmatcher.Type{
Domain_Plain: strmatcher.Substr,
Domain_Regex: strmatcher.Regex,
Domain_Domain: strmatcher.Domain,
2018-08-21 15:30:32 -04:00
Domain_Full: strmatcher.Full,
2018-06-26 15:57:41 -04:00
}
2018-08-19 15:04:15 -04:00
func domainToMatcher(domain *Domain) (strmatcher.Matcher, error) {
2018-06-26 15:57:41 -04:00
matcherType, f := matcherTypeMap[domain.Type]
if !f {
2018-08-19 15:04:15 -04:00
return nil, newError("unsupported domain type", domain.Type)
2017-11-06 15:12:28 -05:00
}
2018-06-26 15:57:41 -04:00
matcher, err := matcherType.New(domain.Value)
if err != nil {
2018-08-19 15:04:15 -04:00
return nil, newError("failed to create domain matcher").Base(err)
2018-06-26 15:57:41 -04:00
}
2018-08-19 15:04:15 -04:00
return matcher, nil
2017-11-06 16:30:56 -05:00
}
2018-08-19 15:04:15 -04:00
type DomainMatcher struct {
matchers strmatcher.IndexMatcher
2017-11-06 16:30:56 -05:00
}
2018-08-19 15:04:15 -04:00
func NewDomainMatcher(domains []*Domain) (*DomainMatcher, error) {
2018-08-20 09:39:58 -04:00
g := new(strmatcher.MatcherGroup)
2018-08-19 15:04:15 -04:00
for _, d := range domains {
m, err := domainToMatcher(d)
if err != nil {
return nil, err
}
g.Add(m)
2017-11-06 16:30:56 -05:00
}
2018-08-19 15:04:15 -04:00
return &DomainMatcher{
matchers: g,
2018-08-19 15:04:15 -04:00
}, nil
}
2017-11-06 16:30:56 -05:00
2018-08-19 15:04:15 -04:00
func (m *DomainMatcher) ApplyDomain(domain string) bool {
return m.matchers.Match(domain) > 0
2017-11-06 16:30:56 -05:00
}
2018-08-19 15:04:15 -04:00
func (m *DomainMatcher) Apply(ctx context.Context) bool {
outbound := session.OutboundFromContext(ctx)
if outbound == nil || !outbound.Target.IsValid() {
2017-02-09 16:49:38 -05:00
return false
}
dest := outbound.Target
2016-09-20 05:53:05 -04:00
if !dest.Address.Family().IsDomain() {
2016-01-17 10:20:49 -05:00
return false
}
2017-11-06 16:30:56 -05:00
return m.ApplyDomain(dest.Address.Domain())
2017-11-06 15:12:28 -05:00
}
func sourceFromContext(ctx context.Context) net.Destination {
inbound := session.InboundFromContext(ctx)
if inbound == nil {
return net.Destination{}
}
return inbound.Source
}
func targetFromContent(ctx context.Context) net.Destination {
outbound := session.OutboundFromContext(ctx)
if outbound == nil {
return net.Destination{}
}
return outbound.Target
}
type MultiGeoIPMatcher struct {
matchers []*GeoIPMatcher
2018-11-07 17:57:06 -05:00
destFunc func(context.Context) net.Destination
}
func NewMultiGeoIPMatcher(geoips []*GeoIP, onSource bool) (*MultiGeoIPMatcher, error) {
var matchers []*GeoIPMatcher
for _, geoip := range geoips {
matcher, err := globalGeoIPContainer.Add(geoip)
if err != nil {
return nil, err
}
matchers = append(matchers, matcher)
}
2018-11-07 17:57:06 -05:00
var destFunc func(context.Context) net.Destination
if onSource {
destFunc = sourceFromContext
} else {
destFunc = targetFromContent
}
return &MultiGeoIPMatcher{
matchers: matchers,
2018-11-07 17:57:06 -05:00
destFunc: destFunc,
}, nil
}
func (m *MultiGeoIPMatcher) Apply(ctx context.Context) bool {
ips := make([]net.IP, 0, 4)
2018-11-07 17:57:06 -05:00
dest := m.destFunc(ctx)
2018-11-16 05:13:36 -05:00
if dest.IsValid() && dest.Address.Family().IsIP() {
ips = append(ips, dest.Address.IP())
2018-11-07 17:57:06 -05:00
} else if resolver, ok := ResolvedIPsFromContext(ctx); ok {
resolvedIPs := resolver.Resolve()
for _, rip := range resolvedIPs {
ips = append(ips, rip.IP())
}
}
for _, ip := range ips {
for _, matcher := range m.matchers {
if matcher.Match(ip) {
return true
}
}
}
return false
}
2016-01-17 10:20:49 -05:00
type PortMatcher struct {
2017-08-29 08:32:54 -04:00
port net.PortRange
2016-01-17 10:20:49 -05:00
}
2017-08-29 08:32:54 -04:00
func NewPortMatcher(portRange net.PortRange) *PortMatcher {
2016-01-17 10:20:49 -05:00
return &PortMatcher{
port: portRange,
}
}
func (v *PortMatcher) Apply(ctx context.Context) bool {
outbound := session.OutboundFromContext(ctx)
if outbound == nil || !outbound.Target.IsValid() {
2017-02-09 16:49:38 -05:00
return false
}
return v.port.Contains(outbound.Target.Port)
2016-01-17 10:20:49 -05:00
}
type NetworkMatcher struct {
2018-11-20 06:25:56 -05:00
list [8]bool
2016-01-17 10:20:49 -05:00
}
func NewNetworkMatcher(network []net.Network) NetworkMatcher {
2018-11-20 06:25:56 -05:00
var matcher NetworkMatcher
for _, n := range network {
2018-11-20 06:25:56 -05:00
matcher.list[int(n)] = true
2016-01-17 10:20:49 -05:00
}
2018-11-20 06:25:56 -05:00
return matcher
2016-01-17 10:20:49 -05:00
}
2018-11-20 06:25:56 -05:00
func (v NetworkMatcher) Apply(ctx context.Context) bool {
outbound := session.OutboundFromContext(ctx)
if outbound == nil || !outbound.Target.IsValid() {
2017-02-09 16:49:38 -05:00
return false
}
2018-11-20 06:25:56 -05:00
return v.list[int(outbound.Target.Network)]
2016-10-18 17:01:39 -04:00
}
type UserMatcher struct {
user []string
}
func NewUserMatcher(users []string) *UserMatcher {
2017-05-08 05:48:41 -04:00
usersCopy := make([]string, 0, len(users))
for _, user := range users {
if len(user) > 0 {
usersCopy = append(usersCopy, user)
}
}
2016-10-18 17:01:39 -04:00
return &UserMatcher{
2017-05-08 05:48:41 -04:00
user: usersCopy,
2016-10-18 17:01:39 -04:00
}
}
func (v *UserMatcher) Apply(ctx context.Context) bool {
2018-10-15 02:36:50 -04:00
inbound := session.InboundFromContext(ctx)
if inbound == nil {
return false
}
user := inbound.User
if user == nil {
2016-10-18 17:01:39 -04:00
return false
}
2016-11-27 15:39:09 -05:00
for _, u := range v.user {
if u == user.Email {
2016-10-18 17:01:39 -04:00
return true
}
}
return false
2016-01-17 10:20:49 -05:00
}
2016-11-13 15:23:34 -05:00
type InboundTagMatcher struct {
tags []string
}
func NewInboundTagMatcher(tags []string) *InboundTagMatcher {
2017-05-08 05:48:41 -04:00
tagsCopy := make([]string, 0, len(tags))
for _, tag := range tags {
if len(tag) > 0 {
tagsCopy = append(tagsCopy, tag)
}
}
2016-11-13 15:23:34 -05:00
return &InboundTagMatcher{
2017-05-08 05:48:41 -04:00
tags: tagsCopy,
2016-11-13 15:23:34 -05:00
}
}
func (v *InboundTagMatcher) Apply(ctx context.Context) bool {
inbound := session.InboundFromContext(ctx)
if inbound == nil || len(inbound.Tag) == 0 {
2016-11-13 15:23:34 -05:00
return false
}
tag := inbound.Tag
2016-11-27 15:39:09 -05:00
for _, t := range v.tags {
if t == tag {
2016-11-13 15:23:34 -05:00
return true
}
}
return false
}
2018-07-16 07:47:00 -04:00
type ProtocolMatcher struct {
protocols []string
}
func NewProtocolMatcher(protocols []string) *ProtocolMatcher {
pCopy := make([]string, 0, len(protocols))
for _, p := range protocols {
if len(p) > 0 {
pCopy = append(pCopy, p)
}
}
return &ProtocolMatcher{
protocols: pCopy,
}
}
func (m *ProtocolMatcher) Apply(ctx context.Context) bool {
result := dispatcher.SniffingResultFromContext(ctx)
if result == nil {
return false
}
protocol := result.Protocol()
for _, p := range m.protocols {
if strings.HasPrefix(protocol, p) {
return true
}
}
return false
}