| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219 | 
							- package conf
 
- import (
 
- 	"encoding/json"
 
- 	"errors"
 
- 	"strconv"
 
- 	"strings"
 
- 	"v2ray.com/core/app/router"
 
- 	"v2ray.com/core/common/log"
 
- 	v2net "v2ray.com/core/common/net"
 
- 	"v2ray.com/core/tools/geoip"
 
- 	"github.com/golang/protobuf/proto"
 
- )
 
- type RouterRulesConfig struct {
 
- 	RuleList       []json.RawMessage `json:"rules"`
 
- 	DomainStrategy string            `json:"domainStrategy"`
 
- }
 
- type RouterConfig struct {
 
- 	Settings *RouterRulesConfig `json:"settings"`
 
- }
 
- func (this *RouterConfig) Build() (*router.Config, error) {
 
- 	if this.Settings == nil {
 
- 		return nil, errors.New("Router settings is not specified.")
 
- 	}
 
- 	config := new(router.Config)
 
- 	settings := this.Settings
 
- 	config.DomainStrategy = router.Config_AsIs
 
- 	config.Rule = make([]*router.RoutingRule, len(settings.RuleList))
 
- 	domainStrategy := strings.ToLower(settings.DomainStrategy)
 
- 	if domainStrategy == "alwaysip" {
 
- 		config.DomainStrategy = router.Config_UseIp
 
- 	} else if domainStrategy == "ipifnonmatch" {
 
- 		config.DomainStrategy = router.Config_IpIfNonMatch
 
- 	}
 
- 	for idx, rawRule := range settings.RuleList {
 
- 		rule := ParseRule(rawRule)
 
- 		config.Rule[idx] = rule
 
- 	}
 
- 	return config, nil
 
- }
 
- type RouterRule struct {
 
- 	Type        string `json:"type"`
 
- 	OutboundTag string `json:"outboundTag"`
 
- }
 
- func parseIP(s string) *router.IP {
 
- 	var addr, mask string
 
- 	i := strings.Index(s, "/")
 
- 	if i < 0 {
 
- 		addr = s
 
- 	} else {
 
- 		addr = s[:i]
 
- 		mask = s[i+1:]
 
- 	}
 
- 	ip := v2net.ParseAddress(addr)
 
- 	switch ip.Family() {
 
- 	case v2net.AddressFamilyIPv4:
 
- 		bits := uint32(32)
 
- 		if len(mask) > 0 {
 
- 			bits64, err := strconv.ParseUint(mask, 10, 32)
 
- 			if err != nil {
 
- 				return nil
 
- 			}
 
- 			bits = uint32(bits64)
 
- 		}
 
- 		if bits > 32 {
 
- 			log.Warning("Router: invalid network mask: ", bits)
 
- 			return nil
 
- 		}
 
- 		return &router.IP{
 
- 			Ip:             []byte(ip.IP()),
 
- 			UnmatchingBits: 32 - bits,
 
- 		}
 
- 	case v2net.AddressFamilyIPv6:
 
- 		bits := uint32(128)
 
- 		if len(mask) > 0 {
 
- 			bits64, err := strconv.ParseUint(mask, 10, 32)
 
- 			if err != nil {
 
- 				return nil
 
- 			}
 
- 			bits = uint32(bits64)
 
- 		}
 
- 		if bits > 128 {
 
- 			log.Warning("Router: invalid network mask: ", bits)
 
- 			return nil
 
- 		}
 
- 		return &router.IP{
 
- 			Ip:             []byte(ip.IP()),
 
- 			UnmatchingBits: 128 - bits,
 
- 		}
 
- 	default:
 
- 		log.Warning("Router: unsupported address: ", s)
 
- 		return nil
 
- 	}
 
- }
 
- func parseFieldRule(msg json.RawMessage) (*router.RoutingRule, error) {
 
- 	type RawFieldRule struct {
 
- 		RouterRule
 
- 		Domain  *StringList  `json:"domain"`
 
- 		IP      *StringList  `json:"ip"`
 
- 		Port    *PortRange   `json:"port"`
 
- 		Network *NetworkList `json:"network"`
 
- 	}
 
- 	rawFieldRule := new(RawFieldRule)
 
- 	err := json.Unmarshal(msg, rawFieldRule)
 
- 	if err != nil {
 
- 		return nil, err
 
- 	}
 
- 	rule := new(router.RoutingRule)
 
- 	rule.Tag = rawFieldRule.OutboundTag
 
- 	if rawFieldRule.Domain != nil {
 
- 		for _, domain := range *rawFieldRule.Domain {
 
- 			domainRule := new(router.Domain)
 
- 			if strings.HasPrefix(domain, "regexp:") {
 
- 				domainRule.Type = router.Domain_Regex
 
- 				domainRule.Value = domain[7:]
 
- 			} else {
 
- 				domainRule.Type = router.Domain_Plain
 
- 				domainRule.Value = domain
 
- 			}
 
- 			rule.Domain = append(rule.Domain, domainRule)
 
- 		}
 
- 	}
 
- 	if rawFieldRule.IP != nil {
 
- 		for _, ip := range *rawFieldRule.IP {
 
- 			ipRule := parseIP(ip)
 
- 			if ipRule != nil {
 
- 				rule.Ip = append(rule.Ip, ipRule)
 
- 			}
 
- 		}
 
- 	}
 
- 	if rawFieldRule.Port != nil {
 
- 		rule.PortRange = rawFieldRule.Port.Build()
 
- 	}
 
- 	if rawFieldRule.Network != nil {
 
- 		rule.NetworkList = rawFieldRule.Network.Build()
 
- 	}
 
- 	return rule, nil
 
- }
 
- func ParseRule(msg json.RawMessage) *router.RoutingRule {
 
- 	rawRule := new(RouterRule)
 
- 	err := json.Unmarshal(msg, rawRule)
 
- 	if err != nil {
 
- 		log.Error("Router: Invalid router rule: ", err)
 
- 		return nil
 
- 	}
 
- 	if rawRule.Type == "field" {
 
- 		fieldrule, err := parseFieldRule(msg)
 
- 		if err != nil {
 
- 			log.Error("Invalid field rule: ", err)
 
- 			return nil
 
- 		}
 
- 		return fieldrule
 
- 	}
 
- 	if rawRule.Type == "chinaip" {
 
- 		chinaiprule, err := parseChinaIPRule(msg)
 
- 		if err != nil {
 
- 			log.Error("Router: Invalid chinaip rule: ", err)
 
- 			return nil
 
- 		}
 
- 		return chinaiprule
 
- 	}
 
- 	if rawRule.Type == "chinasites" {
 
- 		chinasitesrule, err := parseChinaSitesRule(msg)
 
- 		if err != nil {
 
- 			log.Error("Invalid chinasites rule: ", err)
 
- 			return nil
 
- 		}
 
- 		return chinasitesrule
 
- 	}
 
- 	log.Error("Unknown router rule type: ", rawRule.Type)
 
- 	return nil
 
- }
 
- func parseChinaIPRule(data []byte) (*router.RoutingRule, error) {
 
- 	rawRule := new(RouterRule)
 
- 	err := json.Unmarshal(data, rawRule)
 
- 	if err != nil {
 
- 		log.Error("Router: Invalid router rule: ", err)
 
- 		return nil, err
 
- 	}
 
- 	var chinaIPs geoip.CountryIPRange
 
- 	if err := proto.Unmarshal(geoip.ChinaIPs, &chinaIPs); err != nil {
 
- 		return nil, err
 
- 	}
 
- 	return &router.RoutingRule{
 
- 		Tag: rawRule.OutboundTag,
 
- 		Ip:  chinaIPs.Ips,
 
- 	}, nil
 
- }
 
- func parseChinaSitesRule(data []byte) (*router.RoutingRule, error) {
 
- 	rawRule := new(RouterRule)
 
- 	err := json.Unmarshal(data, rawRule)
 
- 	if err != nil {
 
- 		log.Error("Router: Invalid router rule: ", err)
 
- 		return nil, err
 
- 	}
 
- 	return &router.RoutingRule{
 
- 		Tag:    rawRule.OutboundTag,
 
- 		Domain: chinaSitesDomains,
 
- 	}, nil
 
- }
 
 
  |