rule_headless.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. package rule
  2. import (
  3. "context"
  4. "github.com/sagernet/sing-box/adapter"
  5. C "github.com/sagernet/sing-box/constant"
  6. "github.com/sagernet/sing-box/option"
  7. "github.com/sagernet/sing/common"
  8. E "github.com/sagernet/sing/common/exceptions"
  9. "github.com/sagernet/sing/service"
  10. )
  11. func NewHeadlessRule(ctx context.Context, options option.HeadlessRule) (adapter.HeadlessRule, error) {
  12. switch options.Type {
  13. case "", C.RuleTypeDefault:
  14. if !options.DefaultOptions.IsValid() {
  15. return nil, E.New("missing conditions")
  16. }
  17. return NewDefaultHeadlessRule(ctx, options.DefaultOptions)
  18. case C.RuleTypeLogical:
  19. if !options.LogicalOptions.IsValid() {
  20. return nil, E.New("missing conditions")
  21. }
  22. return NewLogicalHeadlessRule(ctx, options.LogicalOptions)
  23. default:
  24. return nil, E.New("unknown rule type: ", options.Type)
  25. }
  26. }
  27. var _ adapter.HeadlessRule = (*DefaultHeadlessRule)(nil)
  28. type DefaultHeadlessRule struct {
  29. abstractDefaultRule
  30. }
  31. func NewDefaultHeadlessRule(ctx context.Context, options option.DefaultHeadlessRule) (*DefaultHeadlessRule, error) {
  32. networkManager := service.FromContext[adapter.NetworkManager](ctx)
  33. rule := &DefaultHeadlessRule{
  34. abstractDefaultRule{
  35. invert: options.Invert,
  36. },
  37. }
  38. if len(options.Network) > 0 {
  39. item := NewNetworkItem(options.Network)
  40. rule.items = append(rule.items, item)
  41. rule.allItems = append(rule.allItems, item)
  42. }
  43. if len(options.Domain) > 0 || len(options.DomainSuffix) > 0 {
  44. item := NewDomainItem(options.Domain, options.DomainSuffix)
  45. rule.destinationAddressItems = append(rule.destinationAddressItems, item)
  46. rule.allItems = append(rule.allItems, item)
  47. } else if options.DomainMatcher != nil {
  48. item := NewRawDomainItem(options.DomainMatcher)
  49. rule.destinationAddressItems = append(rule.destinationAddressItems, item)
  50. rule.allItems = append(rule.allItems, item)
  51. }
  52. if len(options.DomainKeyword) > 0 {
  53. item := NewDomainKeywordItem(options.DomainKeyword)
  54. rule.destinationAddressItems = append(rule.destinationAddressItems, item)
  55. rule.allItems = append(rule.allItems, item)
  56. }
  57. if len(options.DomainRegex) > 0 {
  58. item, err := NewDomainRegexItem(options.DomainRegex)
  59. if err != nil {
  60. return nil, E.Cause(err, "domain_regex")
  61. }
  62. rule.destinationAddressItems = append(rule.destinationAddressItems, item)
  63. rule.allItems = append(rule.allItems, item)
  64. }
  65. if len(options.SourceIPCIDR) > 0 {
  66. item, err := NewIPCIDRItem(true, options.SourceIPCIDR)
  67. if err != nil {
  68. return nil, E.Cause(err, "source_ip_cidr")
  69. }
  70. rule.sourceAddressItems = append(rule.sourceAddressItems, item)
  71. rule.allItems = append(rule.allItems, item)
  72. } else if options.SourceIPSet != nil {
  73. item := NewRawIPCIDRItem(true, options.SourceIPSet)
  74. rule.sourceAddressItems = append(rule.sourceAddressItems, item)
  75. rule.allItems = append(rule.allItems, item)
  76. }
  77. if len(options.IPCIDR) > 0 {
  78. item, err := NewIPCIDRItem(false, options.IPCIDR)
  79. if err != nil {
  80. return nil, E.Cause(err, "ipcidr")
  81. }
  82. rule.destinationIPCIDRItems = append(rule.destinationIPCIDRItems, item)
  83. rule.allItems = append(rule.allItems, item)
  84. } else if options.IPSet != nil {
  85. item := NewRawIPCIDRItem(false, options.IPSet)
  86. rule.destinationIPCIDRItems = append(rule.destinationIPCIDRItems, item)
  87. rule.allItems = append(rule.allItems, item)
  88. }
  89. if len(options.SourcePort) > 0 {
  90. item := NewPortItem(true, options.SourcePort)
  91. rule.sourcePortItems = append(rule.sourcePortItems, item)
  92. rule.allItems = append(rule.allItems, item)
  93. }
  94. if len(options.SourcePortRange) > 0 {
  95. item, err := NewPortRangeItem(true, options.SourcePortRange)
  96. if err != nil {
  97. return nil, E.Cause(err, "source_port_range")
  98. }
  99. rule.sourcePortItems = append(rule.sourcePortItems, item)
  100. rule.allItems = append(rule.allItems, item)
  101. }
  102. if len(options.Port) > 0 {
  103. item := NewPortItem(false, options.Port)
  104. rule.destinationPortItems = append(rule.destinationPortItems, item)
  105. rule.allItems = append(rule.allItems, item)
  106. }
  107. if len(options.PortRange) > 0 {
  108. item, err := NewPortRangeItem(false, options.PortRange)
  109. if err != nil {
  110. return nil, E.Cause(err, "port_range")
  111. }
  112. rule.destinationPortItems = append(rule.destinationPortItems, item)
  113. rule.allItems = append(rule.allItems, item)
  114. }
  115. if len(options.ProcessName) > 0 {
  116. item := NewProcessItem(options.ProcessName)
  117. rule.items = append(rule.items, item)
  118. rule.allItems = append(rule.allItems, item)
  119. }
  120. if len(options.ProcessPath) > 0 {
  121. item := NewProcessPathItem(options.ProcessPath)
  122. rule.items = append(rule.items, item)
  123. rule.allItems = append(rule.allItems, item)
  124. }
  125. if len(options.ProcessPathRegex) > 0 {
  126. item, err := NewProcessPathRegexItem(options.ProcessPathRegex)
  127. if err != nil {
  128. return nil, E.Cause(err, "process_path_regex")
  129. }
  130. rule.items = append(rule.items, item)
  131. rule.allItems = append(rule.allItems, item)
  132. }
  133. if len(options.PackageName) > 0 {
  134. item := NewPackageNameItem(options.PackageName)
  135. rule.items = append(rule.items, item)
  136. rule.allItems = append(rule.allItems, item)
  137. }
  138. if networkManager != nil {
  139. if len(options.NetworkType) > 0 {
  140. item := NewNetworkTypeItem(networkManager, common.Map(options.NetworkType, option.InterfaceType.Build))
  141. rule.items = append(rule.items, item)
  142. rule.allItems = append(rule.allItems, item)
  143. }
  144. if options.NetworkIsExpensive {
  145. item := NewNetworkIsExpensiveItem(networkManager)
  146. rule.items = append(rule.items, item)
  147. rule.allItems = append(rule.allItems, item)
  148. }
  149. if options.NetworkIsConstrained {
  150. item := NewNetworkIsConstrainedItem(networkManager)
  151. rule.items = append(rule.items, item)
  152. rule.allItems = append(rule.allItems, item)
  153. }
  154. if len(options.WIFISSID) > 0 {
  155. item := NewWIFISSIDItem(networkManager, options.WIFISSID)
  156. rule.items = append(rule.items, item)
  157. rule.allItems = append(rule.allItems, item)
  158. }
  159. if len(options.WIFIBSSID) > 0 {
  160. item := NewWIFIBSSIDItem(networkManager, options.WIFIBSSID)
  161. rule.items = append(rule.items, item)
  162. rule.allItems = append(rule.allItems, item)
  163. }
  164. }
  165. if len(options.AdGuardDomain) > 0 {
  166. item := NewAdGuardDomainItem(options.AdGuardDomain)
  167. rule.destinationAddressItems = append(rule.destinationAddressItems, item)
  168. rule.allItems = append(rule.allItems, item)
  169. } else if options.AdGuardDomainMatcher != nil {
  170. item := NewRawAdGuardDomainItem(options.AdGuardDomainMatcher)
  171. rule.destinationAddressItems = append(rule.destinationAddressItems, item)
  172. rule.allItems = append(rule.allItems, item)
  173. }
  174. return rule, nil
  175. }
  176. var _ adapter.HeadlessRule = (*LogicalHeadlessRule)(nil)
  177. type LogicalHeadlessRule struct {
  178. abstractLogicalRule
  179. }
  180. func NewLogicalHeadlessRule(ctx context.Context, options option.LogicalHeadlessRule) (*LogicalHeadlessRule, error) {
  181. r := &LogicalHeadlessRule{
  182. abstractLogicalRule{
  183. rules: make([]adapter.HeadlessRule, len(options.Rules)),
  184. invert: options.Invert,
  185. },
  186. }
  187. switch options.Mode {
  188. case C.LogicalTypeAnd:
  189. r.mode = C.LogicalTypeAnd
  190. case C.LogicalTypeOr:
  191. r.mode = C.LogicalTypeOr
  192. default:
  193. return nil, E.New("unknown logical mode: ", options.Mode)
  194. }
  195. for i, subRule := range options.Rules {
  196. rule, err := NewHeadlessRule(ctx, subRule)
  197. if err != nil {
  198. return nil, E.Cause(err, "sub rule[", i, "]")
  199. }
  200. r.rules[i] = rule
  201. }
  202. return r, nil
  203. }