port.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. // Copyright 2019 Yunion
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package secrules
  15. import (
  16. "sort"
  17. )
  18. type ports []uint16
  19. func newPortsFromInts(ints ...int) ports {
  20. ps := make(ports, len(ints))
  21. for i := range ints {
  22. // panic on invalid value
  23. ps[i] = uint16(ints[i])
  24. }
  25. return ps
  26. }
  27. func (ps ports) IntSlice() []int {
  28. ints := make([]int, len(ps))
  29. for i := range ps {
  30. ints[i] = int(ps[i])
  31. }
  32. return ints
  33. }
  34. func (ps ports) Len() int {
  35. return len(ps)
  36. }
  37. func (ps ports) Swap(i, j int) {
  38. ps[i], ps[j] = ps[j], ps[i]
  39. }
  40. func (ps ports) Less(i, j int) bool {
  41. return ps[i] < ps[j]
  42. }
  43. func (ps ports) contains(p uint16) bool {
  44. for _, p0 := range ps {
  45. if p0 == p {
  46. return true
  47. }
  48. }
  49. return false
  50. }
  51. func (ps ports) containsPorts(ps1 ports) bool {
  52. for _, p := range ps1 {
  53. if !ps.contains(p) {
  54. return false
  55. }
  56. }
  57. return true
  58. }
  59. func (ps ports) dedup() ports {
  60. ps1 := make(ports, len(ps))
  61. copy(ps1, ps)
  62. sort.Sort(ps1)
  63. for i := len(ps1) - 1; i > 0; i-- {
  64. if ps1[i] == ps1[i-1] {
  65. ps1 = append(ps1[:i], ps1[i+1:]...)
  66. }
  67. }
  68. return ps1
  69. }
  70. func (ps ports) sameAs(ps1 ports) bool {
  71. if len(ps) != len(ps1) {
  72. return false
  73. }
  74. for i := range ps {
  75. if ps[i] != ps1[i] {
  76. return false
  77. }
  78. }
  79. return true
  80. }
  81. func (ps ports) substractPortRange(pr *portRange) (left, subs ports) {
  82. left = ports{}
  83. subs = ports{}
  84. for _, p := range ps {
  85. if pr.contains(p) {
  86. subs = append(subs, p)
  87. } else {
  88. left = append(left, p)
  89. }
  90. }
  91. return
  92. }
  93. func (ps ports) substractPorts(ps1 ports) (left, subs ports) {
  94. left = ports{}
  95. subs = ports{}
  96. for _, p0 := range ps {
  97. if ps1.contains(p0) {
  98. subs = append(subs, p0)
  99. } else {
  100. left = append(left, p0)
  101. }
  102. }
  103. return
  104. }
  105. type portRange struct {
  106. start uint16
  107. end uint16
  108. }
  109. func newPortRange(s, e uint16) *portRange {
  110. // panic on s > e
  111. return &portRange{
  112. start: s,
  113. end: e,
  114. }
  115. }
  116. func (pr *portRange) equals(pr1 *portRange) bool {
  117. return pr.start == pr1.start && pr.end == pr1.end
  118. }
  119. func (pr *portRange) contains(p uint16) bool {
  120. return p >= pr.start && p <= pr.end
  121. }
  122. func (pr *portRange) containsRange(pr1 *portRange) bool {
  123. return pr.start <= pr1.start && pr.end >= pr1.end
  124. }
  125. func (pr *portRange) count() uint16 {
  126. return pr.end - pr.start + 1
  127. }
  128. func (pr *portRange) substractPortRange(pr1 *portRange) (lefts []*portRange, sub *portRange) {
  129. // no intersection, no substract
  130. if pr.end < pr1.start || pr.start > pr1.end {
  131. l := *pr
  132. lefts = []*portRange{&l}
  133. return
  134. }
  135. // pr contains pr1
  136. if pr.containsRange(pr1) {
  137. nns := [][2]int32{
  138. [2]int32{int32(pr.start), int32(pr1.start) - 1},
  139. [2]int32{int32(pr1.end) + 1, int32(pr.end)},
  140. }
  141. lefts = []*portRange{}
  142. for _, nn := range nns {
  143. if nn[0] <= nn[1] {
  144. lefts = append(lefts, &portRange{
  145. start: uint16(nn[0]),
  146. end: uint16(nn[1]),
  147. })
  148. }
  149. }
  150. s := *pr1
  151. sub = &s
  152. return
  153. }
  154. // pr contained by pr1
  155. if pr1.containsRange(pr) {
  156. s := *pr
  157. sub = &s
  158. return
  159. }
  160. // intersect, pr on the left
  161. if pr.start < pr1.start && pr.end >= pr1.start {
  162. lefts = []*portRange{&portRange{start: pr.start, end: pr1.start - 1}}
  163. sub = &portRange{pr1.start, pr.end}
  164. return
  165. }
  166. // intersect, pr on the right
  167. if pr.start <= pr1.end && pr.end > pr1.end {
  168. lefts = []*portRange{&portRange{pr1.end + 1, pr.end}}
  169. sub = &portRange{pr.start, pr1.end}
  170. return
  171. }
  172. // no intersection
  173. return
  174. }
  175. func (pr *portRange) substractPorts(ps1 ports) (lefts []*portRange, subs ports) {
  176. // no duplicate
  177. // then ordered
  178. ps2 := make(ports, len(ps1))
  179. copy(ps2, ps1)
  180. sort.Sort(ps2)
  181. lefts = []*portRange{}
  182. subs = ports{}
  183. s := pr.start
  184. for _, p := range ps2 {
  185. if pr.contains(p) {
  186. if p > s {
  187. lefts = append(lefts, &portRange{
  188. start: s,
  189. end: p - 1,
  190. })
  191. }
  192. subs = append(subs, p)
  193. s = p + 1
  194. }
  195. }
  196. if s != 0 && s <= pr.end {
  197. lefts = append(lefts, &portRange{
  198. start: s,
  199. end: pr.end,
  200. })
  201. }
  202. return
  203. }