node_label_test.go 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. /*
  2. Copyright 2019 The Kubernetes Authors.
  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. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package nodelabel
  14. import (
  15. "context"
  16. "testing"
  17. v1 "k8s.io/api/core/v1"
  18. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  19. "k8s.io/apimachinery/pkg/runtime"
  20. framework "k8s.io/kubernetes/pkg/scheduler/framework/v1alpha1"
  21. "k8s.io/kubernetes/pkg/scheduler/internal/cache"
  22. schedulernodeinfo "k8s.io/kubernetes/pkg/scheduler/nodeinfo"
  23. )
  24. func TestValidateNodeLabelArgs(t *testing.T) {
  25. tests := []struct {
  26. name string
  27. args string
  28. err bool
  29. }{
  30. {
  31. name: "happy case",
  32. args: `{"presentLabels" : ["foo", "bar"], "absentLabels" : ["baz"], "presentLabelsPreference" : ["foo", "bar"], "absentLabelsPreference" : ["baz"]}`,
  33. },
  34. {
  35. name: "label presence conflict",
  36. // "bar" exists in both present and absent labels therefore validation should fail.
  37. args: `{"presentLabels" : ["foo", "bar"], "absentLabels" : ["bar", "baz"], "presentLabelsPreference" : ["foo", "bar"], "absentLabelsPreference" : ["baz"]}`,
  38. err: true,
  39. },
  40. {
  41. name: "label preference conflict",
  42. // "bar" exists in both present and absent labels preferences therefore validation should fail.
  43. args: `{"presentLabels" : ["foo", "bar"], "absentLabels" : ["baz"], "presentLabelsPreference" : ["foo", "bar"], "absentLabelsPreference" : ["bar", "baz"]}`,
  44. err: true,
  45. },
  46. {
  47. name: "both label presence and preference conflict",
  48. args: `{"presentLabels" : ["foo", "bar"], "absentLabels" : ["bar", "baz"], "presentLabelsPreference" : ["foo", "bar"], "absentLabelsPreference" : ["bar", "baz"]}`,
  49. err: true,
  50. },
  51. }
  52. for _, test := range tests {
  53. t.Run(test.name, func(t *testing.T) {
  54. args := &runtime.Unknown{Raw: []byte(test.args)}
  55. _, err := New(args, nil)
  56. if test.err && err == nil {
  57. t.Fatal("Plugin initialization should fail.")
  58. }
  59. if !test.err && err != nil {
  60. t.Fatalf("Plugin initialization shouldn't fail: %v", err)
  61. }
  62. })
  63. }
  64. }
  65. func TestNodeLabelFilter(t *testing.T) {
  66. label := map[string]string{"foo": "any value", "bar": "any value"}
  67. var pod *v1.Pod
  68. tests := []struct {
  69. name string
  70. rawArgs string
  71. res framework.Code
  72. }{
  73. {
  74. name: "present label does not match",
  75. rawArgs: `{"presentLabels" : ["baz"]}`,
  76. res: framework.UnschedulableAndUnresolvable,
  77. },
  78. {
  79. name: "absent label does not match",
  80. rawArgs: `{"absentLabels" : ["baz"]}`,
  81. res: framework.Success,
  82. },
  83. {
  84. name: "one of two present labels matches",
  85. rawArgs: `{"presentLabels" : ["foo", "baz"]}`,
  86. res: framework.UnschedulableAndUnresolvable,
  87. },
  88. {
  89. name: "one of two absent labels matches",
  90. rawArgs: `{"absentLabels" : ["foo", "baz"]}`,
  91. res: framework.UnschedulableAndUnresolvable,
  92. },
  93. {
  94. name: "all present abels match",
  95. rawArgs: `{"presentLabels" : ["foo", "bar"]}`,
  96. res: framework.Success,
  97. },
  98. {
  99. name: "all absent labels match",
  100. rawArgs: `{"absentLabels" : ["foo", "bar"]}`,
  101. res: framework.UnschedulableAndUnresolvable,
  102. },
  103. {
  104. name: "both present and absent label matches",
  105. rawArgs: `{"presentLabels" : ["foo"], "absentLabels" : ["bar"]}`,
  106. res: framework.UnschedulableAndUnresolvable,
  107. },
  108. {
  109. name: "neither present nor absent label matches",
  110. rawArgs: `{"presentLabels" : ["foz"], "absentLabels" : ["baz"]}`,
  111. res: framework.UnschedulableAndUnresolvable,
  112. },
  113. {
  114. name: "present label matches and absent label doesn't match",
  115. rawArgs: `{"presentLabels" : ["foo"], "absentLabels" : ["baz"]}`,
  116. res: framework.Success,
  117. },
  118. {
  119. name: "present label doesn't match and absent label matches",
  120. rawArgs: `{"presentLabels" : ["foz"], "absentLabels" : ["bar"]}`,
  121. res: framework.UnschedulableAndUnresolvable,
  122. },
  123. }
  124. for _, test := range tests {
  125. t.Run(test.name, func(t *testing.T) {
  126. node := v1.Node{ObjectMeta: metav1.ObjectMeta{Labels: label}}
  127. nodeInfo := schedulernodeinfo.NewNodeInfo()
  128. nodeInfo.SetNode(&node)
  129. args := &runtime.Unknown{Raw: []byte(test.rawArgs)}
  130. p, err := New(args, nil)
  131. if err != nil {
  132. t.Fatalf("Failed to create plugin: %v", err)
  133. }
  134. status := p.(framework.FilterPlugin).Filter(context.TODO(), nil, pod, nodeInfo)
  135. if status.Code() != test.res {
  136. t.Errorf("Status mismatch. got: %v, want: %v", status.Code(), test.res)
  137. }
  138. })
  139. }
  140. }
  141. func TestNodeLabelScore(t *testing.T) {
  142. tests := []struct {
  143. rawArgs string
  144. want int64
  145. name string
  146. }{
  147. {
  148. want: framework.MaxNodeScore,
  149. rawArgs: `{"presentLabelsPreference" : ["foo"]}`,
  150. name: "one present label match",
  151. },
  152. {
  153. want: 0,
  154. rawArgs: `{"presentLabelsPreference" : ["somelabel"]}`,
  155. name: "one present label mismatch",
  156. },
  157. {
  158. want: framework.MaxNodeScore,
  159. rawArgs: `{"presentLabelsPreference" : ["foo", "bar"]}`,
  160. name: "two present labels match",
  161. },
  162. {
  163. want: 0,
  164. rawArgs: `{"presentLabelsPreference" : ["somelabel1", "somelabel2"]}`,
  165. name: "two present labels mismatch",
  166. },
  167. {
  168. want: framework.MaxNodeScore / 2,
  169. rawArgs: `{"presentLabelsPreference" : ["foo", "somelabel"]}`,
  170. name: "two present labels only one matches",
  171. },
  172. {
  173. want: 0,
  174. rawArgs: `{"absentLabelsPreference" : ["foo"]}`,
  175. name: "one absent label match",
  176. },
  177. {
  178. want: framework.MaxNodeScore,
  179. rawArgs: `{"absentLabelsPreference" : ["somelabel"]}`,
  180. name: "one absent label mismatch",
  181. },
  182. {
  183. want: 0,
  184. rawArgs: `{"absentLabelsPreference" : ["foo", "bar"]}`,
  185. name: "two absent labels match",
  186. },
  187. {
  188. want: framework.MaxNodeScore,
  189. rawArgs: `{"absentLabelsPreference" : ["somelabel1", "somelabel2"]}`,
  190. name: "two absent labels mismatch",
  191. },
  192. {
  193. want: framework.MaxNodeScore / 2,
  194. rawArgs: `{"absentLabelsPreference" : ["foo", "somelabel"]}`,
  195. name: "two absent labels only one matches",
  196. },
  197. {
  198. want: framework.MaxNodeScore,
  199. rawArgs: `{"presentLabelsPreference" : ["foo", "bar"], "absentLabelsPreference" : ["somelabel1", "somelabel2"]}`,
  200. name: "two present labels match, two absent labels mismatch",
  201. },
  202. {
  203. want: 0,
  204. rawArgs: `{"absentLabelsPreference" : ["foo", "bar"], "presentLabelsPreference" : ["somelabel1", "somelabel2"]}`,
  205. name: "two present labels both mismatch, two absent labels both match",
  206. },
  207. {
  208. want: 3 * framework.MaxNodeScore / 4,
  209. rawArgs: `{"presentLabelsPreference" : ["foo", "somelabel"], "absentLabelsPreference" : ["somelabel1", "somelabel2"]}`,
  210. name: "two present labels one matches, two absent labels mismatch",
  211. },
  212. }
  213. for _, test := range tests {
  214. t.Run(test.name, func(t *testing.T) {
  215. state := framework.NewCycleState()
  216. node := &v1.Node{ObjectMeta: metav1.ObjectMeta{Name: "machine1", Labels: map[string]string{"foo": "", "bar": ""}}}
  217. fh, _ := framework.NewFramework(nil, nil, nil, framework.WithSnapshotSharedLister(cache.NewSnapshot(nil, []*v1.Node{node})))
  218. args := &runtime.Unknown{Raw: []byte(test.rawArgs)}
  219. p, err := New(args, fh)
  220. if err != nil {
  221. t.Fatalf("Failed to create plugin: %+v", err)
  222. }
  223. nodeName := node.ObjectMeta.Name
  224. score, status := p.(framework.ScorePlugin).Score(context.Background(), state, nil, nodeName)
  225. if !status.IsSuccess() {
  226. t.Errorf("unexpected error: %v", status)
  227. }
  228. if test.want != score {
  229. t.Errorf("Wrong score. got %#v, want %#v", score, test.want)
  230. }
  231. })
  232. }
  233. }
  234. func TestNodeLabelFilterWithoutNode(t *testing.T) {
  235. var pod *v1.Pod
  236. t.Run("node does not exist", func(t *testing.T) {
  237. nodeInfo := schedulernodeinfo.NewNodeInfo()
  238. p, err := New(nil, nil)
  239. if err != nil {
  240. t.Fatalf("Failed to create plugin: %v", err)
  241. }
  242. status := p.(framework.FilterPlugin).Filter(context.TODO(), nil, pod, nodeInfo)
  243. if status.Code() != framework.Error {
  244. t.Errorf("Status mismatch. got: %v, want: %v", status.Code(), framework.Error)
  245. }
  246. })
  247. }
  248. func TestNodeLabelScoreWithoutNode(t *testing.T) {
  249. t.Run("node does not exist", func(t *testing.T) {
  250. fh, _ := framework.NewFramework(nil, nil, nil, framework.WithSnapshotSharedLister(cache.NewEmptySnapshot()))
  251. p, err := New(nil, fh)
  252. if err != nil {
  253. t.Fatalf("Failed to create plugin: %+v", err)
  254. }
  255. _, status := p.(framework.ScorePlugin).Score(context.Background(), nil, nil, "")
  256. if status.Code() != framework.Error {
  257. t.Errorf("Status mismatch. got: %v, want: %v", status.Code(), framework.Error)
  258. }
  259. })
  260. }