router.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. package router
  2. //go:generate errorgen
  3. import (
  4. "context"
  5. "v2ray.com/core"
  6. "v2ray.com/core/common"
  7. "v2ray.com/core/common/net"
  8. "v2ray.com/core/common/session"
  9. "v2ray.com/core/features/dns"
  10. "v2ray.com/core/features/routing"
  11. "v2ray.com/core/proxy"
  12. )
  13. func init() {
  14. common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
  15. r := new(Router)
  16. if err := core.RequireFeatures(ctx, func(d dns.Client) error {
  17. return r.Init(config.(*Config), d)
  18. }); err != nil {
  19. return nil, err
  20. }
  21. return r, nil
  22. }))
  23. }
  24. // Router is an implementation of routing.Router.
  25. type Router struct {
  26. domainStrategy Config_DomainStrategy
  27. rules []Rule
  28. dns dns.Client
  29. }
  30. // Init initializes the Router.
  31. func (r *Router) Init(config *Config, d dns.Client) error {
  32. r.domainStrategy = config.DomainStrategy
  33. r.rules = make([]Rule, len(config.Rule))
  34. r.dns = d
  35. for idx, rule := range config.Rule {
  36. r.rules[idx].Tag = rule.Tag
  37. cond, err := rule.BuildCondition()
  38. if err != nil {
  39. return err
  40. }
  41. r.rules[idx].Condition = cond
  42. }
  43. return nil
  44. }
  45. type ipResolver struct {
  46. dns dns.Client
  47. ip []net.Address
  48. domain string
  49. resolved bool
  50. }
  51. func (r *ipResolver) Resolve() []net.Address {
  52. if r.resolved {
  53. return r.ip
  54. }
  55. newError("looking for IP for domain: ", r.domain).WriteToLog()
  56. r.resolved = true
  57. ips, err := r.dns.LookupIP(r.domain)
  58. if err != nil {
  59. newError("failed to get IP address").Base(err).WriteToLog()
  60. }
  61. if len(ips) == 0 {
  62. return nil
  63. }
  64. r.ip = make([]net.Address, len(ips))
  65. for i, ip := range ips {
  66. r.ip[i] = net.IPAddress(ip)
  67. }
  68. return r.ip
  69. }
  70. // PickRoute implements routing.Router.
  71. func (r *Router) PickRoute(ctx context.Context) (string, error) {
  72. resolver := &ipResolver{
  73. dns: r.dns,
  74. }
  75. outbound := session.OutboundFromContext(ctx)
  76. if r.domainStrategy == Config_IpOnDemand {
  77. if outbound != nil && outbound.Target.IsValid() && outbound.Target.Address.Family().IsDomain() {
  78. resolver.domain = outbound.Target.Address.Domain()
  79. ctx = proxy.ContextWithResolveIPs(ctx, resolver)
  80. }
  81. }
  82. for _, rule := range r.rules {
  83. if rule.Apply(ctx) {
  84. return rule.Tag, nil
  85. }
  86. }
  87. if outbound == nil || !outbound.Target.IsValid() {
  88. return "", common.ErrNoClue
  89. }
  90. dest := outbound.Target
  91. if r.domainStrategy == Config_IpIfNonMatch && dest.Address.Family().IsDomain() {
  92. resolver.domain = dest.Address.Domain()
  93. ips := resolver.Resolve()
  94. if len(ips) > 0 {
  95. ctx = proxy.ContextWithResolveIPs(ctx, resolver)
  96. for _, rule := range r.rules {
  97. if rule.Apply(ctx) {
  98. return rule.Tag, nil
  99. }
  100. }
  101. }
  102. }
  103. return "", common.ErrNoClue
  104. }
  105. // Start implements common.Runnable.
  106. func (*Router) Start() error {
  107. return nil
  108. }
  109. // Close implements common.Closable.
  110. func (*Router) Close() error {
  111. return nil
  112. }
  113. // Type implement common.HasType.
  114. func (*Router) Type() interface{} {
  115. return routing.RouterType()
  116. }