storage_test.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. /*
  2. Copyright 2015 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 storage
  14. import (
  15. "fmt"
  16. "net/http"
  17. "reflect"
  18. "testing"
  19. apiequality "k8s.io/apimachinery/pkg/api/equality"
  20. "k8s.io/apimachinery/pkg/api/errors"
  21. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  22. "k8s.io/apimachinery/pkg/fields"
  23. "k8s.io/apimachinery/pkg/labels"
  24. "k8s.io/apimachinery/pkg/runtime"
  25. "k8s.io/apimachinery/pkg/util/diff"
  26. "k8s.io/apimachinery/pkg/util/intstr"
  27. genericapirequest "k8s.io/apiserver/pkg/endpoints/request"
  28. "k8s.io/apiserver/pkg/registry/generic"
  29. genericregistrytest "k8s.io/apiserver/pkg/registry/generic/testing"
  30. "k8s.io/apiserver/pkg/registry/rest"
  31. storeerr "k8s.io/apiserver/pkg/storage/errors"
  32. etcdtesting "k8s.io/apiserver/pkg/storage/etcd/testing"
  33. "k8s.io/kubernetes/pkg/apis/apps"
  34. "k8s.io/kubernetes/pkg/apis/autoscaling"
  35. api "k8s.io/kubernetes/pkg/apis/core"
  36. "k8s.io/kubernetes/pkg/registry/registrytest"
  37. )
  38. const defaultReplicas = 100
  39. func newStorage(t *testing.T) (*DeploymentStorage, *etcdtesting.EtcdTestServer) {
  40. etcdStorage, server := registrytest.NewEtcdStorage(t, apps.GroupName)
  41. restOptions := generic.RESTOptions{StorageConfig: etcdStorage, Decorator: generic.UndecoratedStorage, DeleteCollectionWorkers: 1, ResourcePrefix: "deployments"}
  42. deploymentStorage := NewStorage(restOptions)
  43. return &deploymentStorage, server
  44. }
  45. var namespace = "foo-namespace"
  46. var name = "foo-deployment"
  47. func validNewDeployment() *apps.Deployment {
  48. return &apps.Deployment{
  49. ObjectMeta: metav1.ObjectMeta{
  50. Name: name,
  51. Namespace: namespace,
  52. },
  53. Spec: apps.DeploymentSpec{
  54. Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"a": "b"}},
  55. Strategy: apps.DeploymentStrategy{
  56. Type: apps.RollingUpdateDeploymentStrategyType,
  57. RollingUpdate: &apps.RollingUpdateDeployment{
  58. MaxSurge: intstr.FromInt(1),
  59. MaxUnavailable: intstr.FromInt(1),
  60. },
  61. },
  62. Template: api.PodTemplateSpec{
  63. ObjectMeta: metav1.ObjectMeta{
  64. Labels: map[string]string{"a": "b"},
  65. },
  66. Spec: api.PodSpec{
  67. Containers: []api.Container{
  68. {
  69. Name: "test",
  70. Image: "test_image",
  71. ImagePullPolicy: api.PullIfNotPresent,
  72. TerminationMessagePolicy: api.TerminationMessageReadFile,
  73. },
  74. },
  75. RestartPolicy: api.RestartPolicyAlways,
  76. DNSPolicy: api.DNSClusterFirst,
  77. },
  78. },
  79. Replicas: 7,
  80. },
  81. Status: apps.DeploymentStatus{
  82. Replicas: 5,
  83. },
  84. }
  85. }
  86. var validDeployment = *validNewDeployment()
  87. func TestCreate(t *testing.T) {
  88. storage, server := newStorage(t)
  89. defer server.Terminate(t)
  90. defer storage.Deployment.Store.DestroyFunc()
  91. test := genericregistrytest.New(t, storage.Deployment.Store)
  92. deployment := validNewDeployment()
  93. deployment.ObjectMeta = metav1.ObjectMeta{}
  94. test.TestCreate(
  95. // valid
  96. deployment,
  97. // invalid (invalid selector)
  98. &apps.Deployment{
  99. Spec: apps.DeploymentSpec{
  100. Selector: &metav1.LabelSelector{MatchLabels: map[string]string{}},
  101. Template: validDeployment.Spec.Template,
  102. },
  103. },
  104. )
  105. }
  106. func TestUpdate(t *testing.T) {
  107. storage, server := newStorage(t)
  108. defer server.Terminate(t)
  109. defer storage.Deployment.Store.DestroyFunc()
  110. test := genericregistrytest.New(t, storage.Deployment.Store)
  111. test.TestUpdate(
  112. // valid
  113. validNewDeployment(),
  114. // updateFunc
  115. func(obj runtime.Object) runtime.Object {
  116. object := obj.(*apps.Deployment)
  117. object.Spec.Template.Spec.NodeSelector = map[string]string{"c": "d"}
  118. return object
  119. },
  120. // invalid updateFunc
  121. func(obj runtime.Object) runtime.Object {
  122. object := obj.(*apps.Deployment)
  123. object.Name = ""
  124. return object
  125. },
  126. func(obj runtime.Object) runtime.Object {
  127. object := obj.(*apps.Deployment)
  128. object.Spec.Template.Spec.RestartPolicy = api.RestartPolicyOnFailure
  129. return object
  130. },
  131. func(obj runtime.Object) runtime.Object {
  132. object := obj.(*apps.Deployment)
  133. object.Spec.Selector = &metav1.LabelSelector{MatchLabels: map[string]string{}}
  134. return object
  135. },
  136. )
  137. }
  138. func TestDelete(t *testing.T) {
  139. storage, server := newStorage(t)
  140. defer server.Terminate(t)
  141. defer storage.Deployment.Store.DestroyFunc()
  142. test := genericregistrytest.New(t, storage.Deployment.Store)
  143. test.TestDelete(validNewDeployment())
  144. }
  145. func TestGet(t *testing.T) {
  146. storage, server := newStorage(t)
  147. defer server.Terminate(t)
  148. defer storage.Deployment.Store.DestroyFunc()
  149. test := genericregistrytest.New(t, storage.Deployment.Store)
  150. test.TestGet(validNewDeployment())
  151. }
  152. func TestList(t *testing.T) {
  153. storage, server := newStorage(t)
  154. defer server.Terminate(t)
  155. defer storage.Deployment.Store.DestroyFunc()
  156. test := genericregistrytest.New(t, storage.Deployment.Store)
  157. test.TestList(validNewDeployment())
  158. }
  159. func TestWatch(t *testing.T) {
  160. storage, server := newStorage(t)
  161. defer server.Terminate(t)
  162. defer storage.Deployment.Store.DestroyFunc()
  163. test := genericregistrytest.New(t, storage.Deployment.Store)
  164. test.TestWatch(
  165. validNewDeployment(),
  166. // matching labels
  167. []labels.Set{},
  168. // not matching labels
  169. []labels.Set{
  170. {"a": "c"},
  171. {"foo": "bar"},
  172. },
  173. // matching fields
  174. []fields.Set{
  175. {"metadata.name": name},
  176. },
  177. // not matching fields
  178. []fields.Set{
  179. {"metadata.name": "bar"},
  180. {"name": name},
  181. },
  182. )
  183. }
  184. func TestScaleGet(t *testing.T) {
  185. storage, server := newStorage(t)
  186. defer server.Terminate(t)
  187. defer storage.Deployment.Store.DestroyFunc()
  188. var deployment apps.Deployment
  189. ctx := genericapirequest.WithNamespace(genericapirequest.NewContext(), namespace)
  190. key := "/deployments/" + namespace + "/" + name
  191. if err := storage.Deployment.Storage.Create(ctx, key, &validDeployment, &deployment, 0, false); err != nil {
  192. t.Fatalf("error setting new deployment (key: %s) %v: %v", key, validDeployment, err)
  193. }
  194. selector, err := metav1.LabelSelectorAsSelector(validDeployment.Spec.Selector)
  195. if err != nil {
  196. t.Fatal(err)
  197. }
  198. want := &autoscaling.Scale{
  199. ObjectMeta: metav1.ObjectMeta{
  200. Name: name,
  201. Namespace: namespace,
  202. UID: deployment.UID,
  203. ResourceVersion: deployment.ResourceVersion,
  204. CreationTimestamp: deployment.CreationTimestamp,
  205. },
  206. Spec: autoscaling.ScaleSpec{
  207. Replicas: validDeployment.Spec.Replicas,
  208. },
  209. Status: autoscaling.ScaleStatus{
  210. Replicas: validDeployment.Status.Replicas,
  211. Selector: selector.String(),
  212. },
  213. }
  214. obj, err := storage.Scale.Get(ctx, name, &metav1.GetOptions{})
  215. if err != nil {
  216. t.Fatalf("error fetching scale for %s: %v", name, err)
  217. }
  218. got := obj.(*autoscaling.Scale)
  219. if !apiequality.Semantic.DeepEqual(want, got) {
  220. t.Errorf("unexpected scale: %s", diff.ObjectDiff(want, got))
  221. }
  222. }
  223. func TestScaleUpdate(t *testing.T) {
  224. storage, server := newStorage(t)
  225. defer server.Terminate(t)
  226. defer storage.Deployment.Store.DestroyFunc()
  227. var deployment apps.Deployment
  228. ctx := genericapirequest.WithNamespace(genericapirequest.NewContext(), namespace)
  229. key := "/deployments/" + namespace + "/" + name
  230. if err := storage.Deployment.Storage.Create(ctx, key, &validDeployment, &deployment, 0, false); err != nil {
  231. t.Fatalf("error setting new deployment (key: %s) %v: %v", key, validDeployment, err)
  232. }
  233. replicas := int32(12)
  234. update := autoscaling.Scale{
  235. ObjectMeta: metav1.ObjectMeta{Name: name, Namespace: namespace},
  236. Spec: autoscaling.ScaleSpec{
  237. Replicas: replicas,
  238. },
  239. }
  240. if _, _, err := storage.Scale.Update(ctx, update.Name, rest.DefaultUpdatedObjectInfo(&update), rest.ValidateAllObjectFunc, rest.ValidateAllObjectUpdateFunc, false, &metav1.UpdateOptions{}); err != nil {
  241. t.Fatalf("error updating scale %v: %v", update, err)
  242. }
  243. obj, err := storage.Scale.Get(ctx, name, &metav1.GetOptions{})
  244. if err != nil {
  245. t.Fatalf("error fetching scale for %s: %v", name, err)
  246. }
  247. scale := obj.(*autoscaling.Scale)
  248. if scale.Spec.Replicas != replicas {
  249. t.Errorf("wrong replicas count expected: %d got: %d", replicas, deployment.Spec.Replicas)
  250. }
  251. update.ResourceVersion = deployment.ResourceVersion
  252. update.Spec.Replicas = 15
  253. if _, _, err = storage.Scale.Update(ctx, update.Name, rest.DefaultUpdatedObjectInfo(&update), rest.ValidateAllObjectFunc, rest.ValidateAllObjectUpdateFunc, false, &metav1.UpdateOptions{}); err != nil && !errors.IsConflict(err) {
  254. t.Fatalf("unexpected error, expecting an update conflict but got %v", err)
  255. }
  256. }
  257. func TestStatusUpdate(t *testing.T) {
  258. storage, server := newStorage(t)
  259. defer server.Terminate(t)
  260. defer storage.Deployment.Store.DestroyFunc()
  261. ctx := genericapirequest.WithNamespace(genericapirequest.NewContext(), namespace)
  262. key := "/deployments/" + namespace + "/" + name
  263. if err := storage.Deployment.Storage.Create(ctx, key, &validDeployment, nil, 0, false); err != nil {
  264. t.Fatalf("unexpected error: %v", err)
  265. }
  266. update := apps.Deployment{
  267. ObjectMeta: validDeployment.ObjectMeta,
  268. Spec: apps.DeploymentSpec{
  269. Replicas: defaultReplicas,
  270. },
  271. Status: apps.DeploymentStatus{
  272. Replicas: defaultReplicas,
  273. },
  274. }
  275. if _, _, err := storage.Status.Update(ctx, update.Name, rest.DefaultUpdatedObjectInfo(&update), rest.ValidateAllObjectFunc, rest.ValidateAllObjectUpdateFunc, false, &metav1.UpdateOptions{}); err != nil {
  276. t.Fatalf("unexpected error: %v", err)
  277. }
  278. obj, err := storage.Deployment.Get(ctx, name, &metav1.GetOptions{})
  279. if err != nil {
  280. t.Fatalf("unexpected error: %v", err)
  281. }
  282. deployment := obj.(*apps.Deployment)
  283. if deployment.Spec.Replicas != 7 {
  284. t.Errorf("we expected .spec.replicas to not be updated but it was updated to %v", deployment.Spec.Replicas)
  285. }
  286. if deployment.Status.Replicas != defaultReplicas {
  287. t.Errorf("we expected .status.replicas to be updated to %d but it was %v", defaultReplicas, deployment.Status.Replicas)
  288. }
  289. }
  290. func TestEtcdCreateDeploymentRollback(t *testing.T) {
  291. ctx := genericapirequest.WithNamespace(genericapirequest.NewContext(), namespace)
  292. testCases := map[string]struct {
  293. rollback apps.DeploymentRollback
  294. errOK func(error) bool
  295. }{
  296. "normal": {
  297. rollback: apps.DeploymentRollback{
  298. Name: name,
  299. UpdatedAnnotations: map[string]string{},
  300. RollbackTo: apps.RollbackConfig{Revision: 1},
  301. },
  302. errOK: func(err error) bool { return err == nil },
  303. },
  304. "noAnnotation": {
  305. rollback: apps.DeploymentRollback{
  306. Name: name,
  307. RollbackTo: apps.RollbackConfig{Revision: 1},
  308. },
  309. errOK: func(err error) bool { return err == nil },
  310. },
  311. "noName": {
  312. rollback: apps.DeploymentRollback{
  313. UpdatedAnnotations: map[string]string{},
  314. RollbackTo: apps.RollbackConfig{Revision: 1},
  315. },
  316. errOK: func(err error) bool { return err != nil },
  317. },
  318. }
  319. for k, test := range testCases {
  320. storage, server := newStorage(t)
  321. rollbackStorage := storage.Rollback
  322. if _, err := storage.Deployment.Create(ctx, validNewDeployment(), rest.ValidateAllObjectFunc, &metav1.CreateOptions{}); err != nil {
  323. t.Fatalf("%s: unexpected error: %v", k, err)
  324. }
  325. rollbackRespStatus, err := rollbackStorage.Create(ctx, test.rollback.Name, &test.rollback, rest.ValidateAllObjectFunc, &metav1.CreateOptions{})
  326. if !test.errOK(err) {
  327. t.Errorf("%s: unexpected error: %v", k, err)
  328. } else if err == nil {
  329. // If rollback succeeded, verify Rollback response and Rollback field of deployment
  330. status, ok := rollbackRespStatus.(*metav1.Status)
  331. if !ok {
  332. t.Errorf("%s: unexpected response format", k)
  333. }
  334. if status.Code != http.StatusOK || status.Status != metav1.StatusSuccess {
  335. t.Errorf("%s: unexpected response, code: %d, status: %s", k, status.Code, status.Status)
  336. }
  337. d, err := storage.Deployment.Get(ctx, validNewDeployment().ObjectMeta.Name, &metav1.GetOptions{})
  338. if err != nil {
  339. t.Errorf("%s: unexpected error: %v", k, err)
  340. } else if !reflect.DeepEqual(*d.(*apps.Deployment).Spec.RollbackTo, test.rollback.RollbackTo) {
  341. t.Errorf("%s: expected: %v, got: %v", k, *d.(*apps.Deployment).Spec.RollbackTo, test.rollback.RollbackTo)
  342. }
  343. }
  344. storage.Deployment.Store.DestroyFunc()
  345. server.Terminate(t)
  346. }
  347. }
  348. func TestCreateDeploymentRollbackValidation(t *testing.T) {
  349. storage, server := newStorage(t)
  350. rollbackStorage := storage.Rollback
  351. rollback := apps.DeploymentRollback{
  352. Name: name,
  353. UpdatedAnnotations: map[string]string{},
  354. RollbackTo: apps.RollbackConfig{Revision: 1},
  355. }
  356. ctx := genericapirequest.WithNamespace(genericapirequest.NewContext(), namespace)
  357. if _, err := storage.Deployment.Create(ctx, validNewDeployment(), rest.ValidateAllObjectFunc, &metav1.CreateOptions{}); err != nil {
  358. t.Fatalf("Unexpected error: %v", err)
  359. }
  360. validationError := fmt.Errorf("admission deny")
  361. alwaysDenyValidationFunc := func(obj runtime.Object) error { return validationError }
  362. _, err := rollbackStorage.Create(ctx, rollback.Name, &rollback, alwaysDenyValidationFunc, &metav1.CreateOptions{})
  363. if err == nil || validationError != err {
  364. t.Errorf("expected: %v, got: %v", validationError, err)
  365. }
  366. storage.Deployment.Store.DestroyFunc()
  367. server.Terminate(t)
  368. }
  369. // Ensure that when a deploymentRollback is created for a deployment that has already been deleted
  370. // by the API server, API server returns not-found error.
  371. func TestEtcdCreateDeploymentRollbackNoDeployment(t *testing.T) {
  372. storage, server := newStorage(t)
  373. defer server.Terminate(t)
  374. defer storage.Deployment.Store.DestroyFunc()
  375. rollbackStorage := storage.Rollback
  376. ctx := genericapirequest.WithNamespace(genericapirequest.NewContext(), namespace)
  377. _, err := rollbackStorage.Create(ctx, name, &apps.DeploymentRollback{
  378. Name: name,
  379. UpdatedAnnotations: map[string]string{},
  380. RollbackTo: apps.RollbackConfig{Revision: 1},
  381. }, rest.ValidateAllObjectFunc, &metav1.CreateOptions{})
  382. if err == nil {
  383. t.Fatalf("Expected not-found-error but got nothing")
  384. }
  385. if !errors.IsNotFound(storeerr.InterpretGetError(err, apps.Resource("deployments"), name)) {
  386. t.Fatalf("Unexpected error returned: %#v", err)
  387. }
  388. _, err = storage.Deployment.Get(ctx, name, &metav1.GetOptions{})
  389. if err == nil {
  390. t.Fatalf("Expected not-found-error but got nothing")
  391. }
  392. if !errors.IsNotFound(storeerr.InterpretGetError(err, apps.Resource("deployments"), name)) {
  393. t.Fatalf("Unexpected error: %v", err)
  394. }
  395. }
  396. func TestShortNames(t *testing.T) {
  397. storage, server := newStorage(t)
  398. defer server.Terminate(t)
  399. defer storage.Deployment.Store.DestroyFunc()
  400. expected := []string{"deploy"}
  401. registrytest.AssertShortNames(t, storage.Deployment, expected)
  402. }
  403. func TestCategories(t *testing.T) {
  404. storage, server := newStorage(t)
  405. defer server.Terminate(t)
  406. defer storage.Deployment.Store.DestroyFunc()
  407. expected := []string{"all"}
  408. registrytest.AssertCategories(t, storage.Deployment, expected)
  409. }