| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 |
- package rules
- import (
- "errors"
- "time"
- "github.com/v2ray/v2ray-core/app"
- "github.com/v2ray/v2ray-core/app/dns"
- "github.com/v2ray/v2ray-core/app/router"
- "github.com/v2ray/v2ray-core/common/collect"
- "github.com/v2ray/v2ray-core/common/log"
- v2net "github.com/v2ray/v2ray-core/common/net"
- )
- var (
- ErrorInvalidRule = errors.New("Invalid Rule")
- ErrorNoRuleApplicable = errors.New("No rule applicable")
- )
- type cacheEntry struct {
- tag string
- err error
- validUntil time.Time
- }
- func newCacheEntry(tag string, err error) *cacheEntry {
- this := &cacheEntry{
- tag: tag,
- err: err,
- }
- this.Extend()
- return this
- }
- func (this *cacheEntry) IsValid() bool {
- return this.validUntil.Before(time.Now())
- }
- func (this *cacheEntry) Extend() {
- this.validUntil = time.Now().Add(time.Hour)
- }
- type Router struct {
- config *RouterRuleConfig
- cache *collect.ValidityMap
- dnsServer dns.Server
- }
- func NewRouter(config *RouterRuleConfig, space app.Space) *Router {
- r := &Router{
- config: config,
- cache: collect.NewValidityMap(3600),
- }
- space.InitializeApplication(func() error {
- if !space.HasApp(dns.APP_ID) {
- log.Error("DNS: Router is not found in the space.")
- return app.ErrorMissingApplication
- }
- r.dnsServer = space.GetApp(dns.APP_ID).(dns.Server)
- return nil
- })
- return r
- }
- func (this *Router) Release() {
- }
- // @Private
- func (this *Router) ResolveIP(dest v2net.Destination) []v2net.Destination {
- ips := this.dnsServer.Get(dest.Address().Domain())
- if len(ips) == 0 {
- return nil
- }
- dests := make([]v2net.Destination, len(ips))
- for idx, ip := range ips {
- if dest.IsTCP() {
- dests[idx] = v2net.TCPDestination(v2net.IPAddress(ip), dest.Port())
- } else {
- dests[idx] = v2net.UDPDestination(v2net.IPAddress(ip), dest.Port())
- }
- }
- return dests
- }
- func (this *Router) takeDetourWithoutCache(dest v2net.Destination) (string, error) {
- for _, rule := range this.config.Rules {
- if rule.Apply(dest) {
- return rule.Tag, nil
- }
- }
- if this.config.DomainStrategy == UseIPIfNonMatch && dest.Address().IsDomain() {
- log.Info("Router: Looking up IP for ", dest)
- ipDests := this.ResolveIP(dest)
- if ipDests != nil {
- for _, ipDest := range ipDests {
- log.Info("Router: Trying IP ", ipDest)
- for _, rule := range this.config.Rules {
- if rule.Apply(ipDest) {
- return rule.Tag, nil
- }
- }
- }
- }
- }
- return "", ErrorNoRuleApplicable
- }
- func (this *Router) TakeDetour(dest v2net.Destination) (string, error) {
- rawEntry := this.cache.Get(dest)
- if rawEntry == nil {
- tag, err := this.takeDetourWithoutCache(dest)
- this.cache.Set(dest, newCacheEntry(tag, err))
- return tag, err
- }
- entry := rawEntry.(*cacheEntry)
- return entry.tag, entry.err
- }
- type RouterFactory struct {
- }
- func (this *RouterFactory) Create(rawConfig interface{}, space app.Space) (router.Router, error) {
- return NewRouter(rawConfig.(*RouterRuleConfig), space), nil
- }
- func init() {
- router.RegisterRouter("rules", &RouterFactory{})
- }
|