dynamic_client_test.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. /*
  2. Copyright 2016 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 client
  14. import (
  15. "context"
  16. "fmt"
  17. "reflect"
  18. "testing"
  19. "time"
  20. "k8s.io/api/core/v1"
  21. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  22. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  23. "k8s.io/apimachinery/pkg/fields"
  24. "k8s.io/apimachinery/pkg/runtime"
  25. "k8s.io/apimachinery/pkg/runtime/schema"
  26. "k8s.io/apimachinery/pkg/util/wait"
  27. "k8s.io/apimachinery/pkg/watch"
  28. "k8s.io/client-go/dynamic"
  29. clientset "k8s.io/client-go/kubernetes"
  30. clientscheme "k8s.io/client-go/kubernetes/scheme"
  31. kubeapiservertesting "k8s.io/kubernetes/cmd/kube-apiserver/app/testing"
  32. "k8s.io/kubernetes/test/integration/framework"
  33. )
  34. func TestDynamicClient(t *testing.T) {
  35. result := kubeapiservertesting.StartTestServerOrDie(t, nil, []string{"--disable-admission-plugins", "ServiceAccount"}, framework.SharedEtcd())
  36. defer result.TearDownFn()
  37. client := clientset.NewForConfigOrDie(result.ClientConfig)
  38. dynamicClient, err := dynamic.NewForConfig(result.ClientConfig)
  39. if err != nil {
  40. t.Fatalf("unexpected error creating dynamic client: %v", err)
  41. }
  42. resource := schema.GroupVersionResource{Group: "", Version: "v1", Resource: "pods"}
  43. // Create a Pod with the normal client
  44. pod := &v1.Pod{
  45. ObjectMeta: metav1.ObjectMeta{
  46. GenerateName: "test",
  47. },
  48. Spec: v1.PodSpec{
  49. Containers: []v1.Container{
  50. {
  51. Name: "test",
  52. Image: "test-image",
  53. },
  54. },
  55. },
  56. }
  57. actual, err := client.CoreV1().Pods("default").Create(context.TODO(), pod, metav1.CreateOptions{})
  58. if err != nil {
  59. t.Fatalf("unexpected error when creating pod: %v", err)
  60. }
  61. // check dynamic list
  62. unstructuredList, err := dynamicClient.Resource(resource).Namespace("default").List(metav1.ListOptions{})
  63. if err != nil {
  64. t.Fatalf("unexpected error when listing pods: %v", err)
  65. }
  66. if len(unstructuredList.Items) != 1 {
  67. t.Fatalf("expected one pod, got %d", len(unstructuredList.Items))
  68. }
  69. got, err := unstructuredToPod(&unstructuredList.Items[0])
  70. if err != nil {
  71. t.Fatalf("unexpected error converting Unstructured to v1.Pod: %v", err)
  72. }
  73. if !reflect.DeepEqual(actual, got) {
  74. t.Fatalf("unexpected pod in list. wanted %#v, got %#v", actual, got)
  75. }
  76. // check dynamic get
  77. unstruct, err := dynamicClient.Resource(resource).Namespace("default").Get(actual.Name, metav1.GetOptions{})
  78. if err != nil {
  79. t.Fatalf("unexpected error when getting pod %q: %v", actual.Name, err)
  80. }
  81. got, err = unstructuredToPod(unstruct)
  82. if err != nil {
  83. t.Fatalf("unexpected error converting Unstructured to v1.Pod: %v", err)
  84. }
  85. if !reflect.DeepEqual(actual, got) {
  86. t.Fatalf("unexpected pod in list. wanted %#v, got %#v", actual, got)
  87. }
  88. // delete the pod dynamically
  89. err = dynamicClient.Resource(resource).Namespace("default").Delete(actual.Name, nil)
  90. if err != nil {
  91. t.Fatalf("unexpected error when deleting pod: %v", err)
  92. }
  93. list, err := client.CoreV1().Pods("default").List(context.TODO(), metav1.ListOptions{})
  94. if err != nil {
  95. t.Fatalf("unexpected error when listing pods: %v", err)
  96. }
  97. if len(list.Items) != 0 {
  98. t.Fatalf("expected zero pods, got %d", len(list.Items))
  99. }
  100. }
  101. func TestDynamicClientWatch(t *testing.T) {
  102. result := kubeapiservertesting.StartTestServerOrDie(t, nil, nil, framework.SharedEtcd())
  103. defer result.TearDownFn()
  104. client := clientset.NewForConfigOrDie(result.ClientConfig)
  105. dynamicClient, err := dynamic.NewForConfig(result.ClientConfig)
  106. if err != nil {
  107. t.Fatalf("unexpected error creating dynamic client: %v", err)
  108. }
  109. resource := v1.SchemeGroupVersion.WithResource("events")
  110. mkEvent := func(i int) *v1.Event {
  111. name := fmt.Sprintf("event-%v", i)
  112. return &v1.Event{
  113. ObjectMeta: metav1.ObjectMeta{
  114. Namespace: "default",
  115. Name: name,
  116. },
  117. InvolvedObject: v1.ObjectReference{
  118. Namespace: "default",
  119. Name: name,
  120. },
  121. Reason: fmt.Sprintf("event %v", i),
  122. }
  123. }
  124. rv1 := ""
  125. for i := 0; i < 10; i++ {
  126. event := mkEvent(i)
  127. got, err := client.CoreV1().Events("default").Create(context.TODO(), event, metav1.CreateOptions{})
  128. if err != nil {
  129. t.Fatalf("Failed creating event %#q: %v", event, err)
  130. }
  131. if rv1 == "" {
  132. rv1 = got.ResourceVersion
  133. if rv1 == "" {
  134. t.Fatal("did not get a resource version.")
  135. }
  136. }
  137. t.Logf("Created event %#v", got.ObjectMeta)
  138. }
  139. w, err := dynamicClient.Resource(resource).Namespace("default").Watch(metav1.ListOptions{
  140. ResourceVersion: rv1,
  141. Watch: true,
  142. FieldSelector: fields.OneTermEqualSelector("metadata.name", "event-9").String(),
  143. })
  144. if err != nil {
  145. t.Fatalf("Failed watch: %v", err)
  146. }
  147. defer w.Stop()
  148. select {
  149. case <-time.After(wait.ForeverTestTimeout):
  150. t.Fatalf("watch took longer than %s", wait.ForeverTestTimeout.String())
  151. case got, ok := <-w.ResultChan():
  152. if !ok {
  153. t.Fatal("Watch channel closed unexpectedly.")
  154. }
  155. // We expect to see an ADD of event-9 and only event-9. (This
  156. // catches a bug where all the events would have been sent down
  157. // the channel.)
  158. if e, a := watch.Added, got.Type; e != a {
  159. t.Errorf("Wanted %v, got %v", e, a)
  160. }
  161. unstructured, ok := got.Object.(*unstructured.Unstructured)
  162. if !ok {
  163. t.Fatalf("Unexpected watch event containing object %#q", got.Object)
  164. }
  165. event, err := unstructuredToEvent(unstructured)
  166. if err != nil {
  167. t.Fatalf("unexpected error converting Unstructured to v1.Event: %v", err)
  168. }
  169. if e, a := "event-9", event.Name; e != a {
  170. t.Errorf("Wanted %v, got %v", e, a)
  171. }
  172. }
  173. }
  174. func unstructuredToPod(obj *unstructured.Unstructured) (*v1.Pod, error) {
  175. json, err := runtime.Encode(unstructured.UnstructuredJSONScheme, obj)
  176. if err != nil {
  177. return nil, err
  178. }
  179. pod := new(v1.Pod)
  180. err = runtime.DecodeInto(clientscheme.Codecs.LegacyCodec(v1.SchemeGroupVersion), json, pod)
  181. pod.Kind = ""
  182. pod.APIVersion = ""
  183. return pod, err
  184. }
  185. func unstructuredToEvent(obj *unstructured.Unstructured) (*v1.Event, error) {
  186. json, err := runtime.Encode(unstructured.UnstructuredJSONScheme, obj)
  187. if err != nil {
  188. return nil, err
  189. }
  190. event := new(v1.Event)
  191. err = runtime.DecodeInto(clientscheme.Codecs.LegacyCodec(v1.SchemeGroupVersion), json, event)
  192. return event, err
  193. }