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