radix_test.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  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 appsrv
  15. import (
  16. "strings"
  17. "testing"
  18. )
  19. func TestRadixNode(t *testing.T) {
  20. root := NewRadix()
  21. root.Add([]string{}, "root")
  22. root.Add([]string{"layer1"}, "layer1")
  23. if root.Add([]string{"layer1"}, "layer1") == nil {
  24. t.Fatal("add duplicate data should fail")
  25. }
  26. root.Add([]string{"layer1", "layer1.1", "layer1.1.1", "layer1.1.1.1"}, "layer1.1.1.1")
  27. root.Add([]string{"layer1", "layer1.2", "layer1.2.1"}, "layer1.2.1")
  28. root.Add([]string{"layer1", "<layer1.x>"}, "layer1.*")
  29. root.Add([]string{"layer1", "layer1.0"}, "layer1.0")
  30. root.Add([]string{"layer1", "<layer1.x>", "layer1.*.1"}, "layer1.*.1")
  31. root.Add([]string{"layer1", "<phone_number:^1[0-9-]{10}$>", "layer1.2.1"}, "layer1.*.1_CHINA_MOBILE_REG")
  32. root.Add([]string{"layer1", "layer1.0", "<phone_number:^1[0-9-]{10}$>"}, "layer1.1.*_CHINA_MOBILE_REG")
  33. root.Add([]string{"<phone_number:^1[0-9-]{10}$>"}, "phonelayer1")
  34. f := func(path string, data interface{}) {
  35. t.Logf("%s %s", path, data)
  36. }
  37. root.Walk(f)
  38. cases := []struct {
  39. caseName string
  40. segments []string
  41. want string
  42. }{
  43. {"case1", []string{"layer1", "layer1.0", "layer2.0"}, "layer1.0"},
  44. {"case2", []string{"layer1", "layer1.0"}, "layer1.0"},
  45. {"case3", []string{"layer1", "layer1.1"}, "layer1.*"},
  46. {"case4", []string{"layer1", "layer1.4"}, "layer1.*"},
  47. {
  48. "case5", []string{"layer1", "layer1.1", "layer1.1.1", "layer1.1.1.1", "layer1.1.1.1.1", "layer1.1.1.1.1.1"},
  49. "layer1.1.1.1",
  50. },
  51. {"case6", []string{"layer1", "layer1.2", "layer1.2.1"}, "layer1.2.1"},
  52. {"case7", []string{"layer1", "layer1.2"}, "layer1.*"},
  53. {"case8", []string{"layer1", "layer1.3"}, "layer1.*"},
  54. {"case9", []string{"layer1", "layer1.3", "layer1.*.1"}, "layer1.*.1"},
  55. {"case10", []string{"layer1", "layer1.3", "layer1.*.1", "layer1.*.1.1"}, "layer1.*.1"},
  56. {"case11", []string{"layer1", "12345678901", "layer1.2.1"}, "layer1.*.1_CHINA_MOBILE_REG"},
  57. {"case12", []string{"layer1", "layer1.0", "12345678901"}, "layer1.1.*_CHINA_MOBILE_REG"},
  58. {"case13", []string{}, "root"},
  59. {"case14", []string{"12345678900"}, "phonelayer1"},
  60. }
  61. for _, c := range cases {
  62. t.Run(c.caseName, func(t *testing.T) {
  63. params := make(map[string]string)
  64. got := root.Match(c.segments, params)
  65. if got.(string) != c.want {
  66. t.Error("unexpect result:", got, "!=", c.want)
  67. }
  68. })
  69. }
  70. }
  71. func TestRadixMatchParams(t *testing.T) {
  72. r := NewRadix()
  73. r.Add([]string{"POST", "clouds", "<cls_action>"}, "classAction")
  74. r.Add([]string{"POST", "clouds", "<resid>", "sync"}, "objectSyncAction")
  75. r.Add([]string{"POST", "clouds", "<resid>", "<obj_action>"}, "objectAction")
  76. r.Add([]string{"POST", "clouds", "<resid2:.*>", "<obj_action2:.*>", "over"}, "objectAction2")
  77. f := func(path string, data interface{}) {
  78. t.Logf("%s %s", path, data)
  79. }
  80. r.Walk(f)
  81. cases := []struct {
  82. in []string
  83. out interface{}
  84. outParams map[string]string
  85. }{
  86. {
  87. in: []string{"POST", "clouds", "myid", "sync"},
  88. out: "objectSyncAction",
  89. outParams: map[string]string{
  90. "<resid>": "myid",
  91. },
  92. },
  93. {
  94. in: []string{"POST", "clouds", "myid", "start"},
  95. out: "objectAction",
  96. outParams: map[string]string{
  97. "<resid>": "myid",
  98. "<obj_action>": "start",
  99. },
  100. },
  101. {
  102. in: []string{"POST", "clouds", "start"},
  103. out: "classAction",
  104. outParams: map[string]string{
  105. "<cls_action>": "start",
  106. },
  107. },
  108. {
  109. in: []string{"POST", "clouds", "start", "test", "over"},
  110. out: "objectAction2",
  111. outParams: map[string]string{
  112. "<resid2>": "start",
  113. "<obj_action2>": "test",
  114. },
  115. },
  116. }
  117. for _, c := range cases {
  118. t.Run(strings.Join(c.in, "_"), func(t *testing.T) {
  119. gotParams := map[string]string{}
  120. got := r.Match(c.in, gotParams)
  121. if got != c.out {
  122. t.Fatalf("want %s, got %s", c.out, c.out)
  123. }
  124. if len(gotParams) != len(c.outParams) {
  125. t.Fatalf("params length mismatch\nwant %#v\ngot %#v",
  126. c.outParams, gotParams,
  127. )
  128. }
  129. for k, v := range gotParams {
  130. v1 := c.outParams[k]
  131. if v == v1 {
  132. continue
  133. }
  134. t.Fatalf("params key %s has mismatch value\nwant %#v\ngot %#v",
  135. k, c.outParams, gotParams,
  136. )
  137. }
  138. })
  139. }
  140. }