synthetic_master_test.go 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906
  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 master
  14. import (
  15. "bytes"
  16. "encoding/json"
  17. "fmt"
  18. "io/ioutil"
  19. "net"
  20. "net/http"
  21. "os"
  22. "path"
  23. "strconv"
  24. "strings"
  25. "sync"
  26. "testing"
  27. "time"
  28. "sigs.k8s.io/yaml"
  29. appsv1 "k8s.io/api/apps/v1"
  30. corev1 "k8s.io/api/core/v1"
  31. "k8s.io/apimachinery/pkg/api/errors"
  32. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  33. "k8s.io/apimachinery/pkg/util/wait"
  34. "k8s.io/apiserver/pkg/authentication/group"
  35. "k8s.io/apiserver/pkg/authentication/request/bearertoken"
  36. "k8s.io/apiserver/pkg/authentication/user"
  37. "k8s.io/apiserver/pkg/authorization/authorizer"
  38. "k8s.io/apiserver/pkg/authorization/authorizerfactory"
  39. "k8s.io/apiserver/plugin/pkg/authenticator/token/tokentest"
  40. clientset "k8s.io/client-go/kubernetes"
  41. clienttypedv1 "k8s.io/client-go/kubernetes/typed/core/v1"
  42. restclient "k8s.io/client-go/rest"
  43. "k8s.io/kubernetes/pkg/master"
  44. "k8s.io/kubernetes/test/integration"
  45. "k8s.io/kubernetes/test/integration/framework"
  46. )
  47. const (
  48. AliceToken string = "abc123" // username: alice. Present in token file.
  49. BobToken string = "xyz987" // username: bob. Present in token file.
  50. )
  51. type allowAliceAuthorizer struct{}
  52. func (allowAliceAuthorizer) Authorize(a authorizer.Attributes) (authorizer.Decision, string, error) {
  53. if a.GetUser() != nil && a.GetUser().GetName() == "alice" {
  54. return authorizer.DecisionAllow, "", nil
  55. }
  56. return authorizer.DecisionNoOpinion, "I can't allow that. Go ask alice.", nil
  57. }
  58. func testPrefix(t *testing.T, prefix string) {
  59. _, s, closeFn := framework.RunAMaster(nil)
  60. defer closeFn()
  61. resp, err := http.Get(s.URL + prefix)
  62. if err != nil {
  63. t.Fatalf("unexpected error getting %s prefix: %v", prefix, err)
  64. }
  65. if resp.StatusCode != http.StatusOK {
  66. t.Fatalf("got status %v instead of 200 OK", resp.StatusCode)
  67. }
  68. }
  69. func TestAutoscalingPrefix(t *testing.T) {
  70. testPrefix(t, "/apis/autoscaling/")
  71. }
  72. func TestBatchPrefix(t *testing.T) {
  73. testPrefix(t, "/apis/batch/")
  74. }
  75. func TestAppsPrefix(t *testing.T) {
  76. testPrefix(t, "/apis/apps/")
  77. }
  78. func TestExtensionsPrefix(t *testing.T) {
  79. testPrefix(t, "/apis/extensions/")
  80. }
  81. func TestKubernetesService(t *testing.T) {
  82. config := framework.NewMasterConfig()
  83. _, _, closeFn := framework.RunAMaster(config)
  84. defer closeFn()
  85. coreClient := clientset.NewForConfigOrDie(config.GenericConfig.LoopbackClientConfig)
  86. err := wait.PollImmediate(time.Millisecond*100, wait.ForeverTestTimeout, func() (bool, error) {
  87. if _, err := coreClient.CoreV1().Services(metav1.NamespaceDefault).Get("kubernetes", metav1.GetOptions{}); err != nil && errors.IsNotFound(err) {
  88. return false, nil
  89. } else if err != nil {
  90. return false, err
  91. }
  92. return true, nil
  93. })
  94. if err != nil {
  95. t.Fatalf("Expected kubernetes service to exist, got: %v", err)
  96. }
  97. }
  98. func TestEmptyList(t *testing.T) {
  99. _, s, closeFn := framework.RunAMaster(nil)
  100. defer closeFn()
  101. u := s.URL + "/api/v1/namespaces/default/pods"
  102. resp, err := http.Get(u)
  103. if err != nil {
  104. t.Fatalf("unexpected error getting %s: %v", u, err)
  105. }
  106. if resp.StatusCode != http.StatusOK {
  107. t.Fatalf("got status %v instead of 200 OK", resp.StatusCode)
  108. }
  109. defer resp.Body.Close()
  110. data, _ := ioutil.ReadAll(resp.Body)
  111. decodedData := map[string]interface{}{}
  112. if err := json.Unmarshal(data, &decodedData); err != nil {
  113. t.Logf("body: %s", string(data))
  114. t.Fatalf("got error decoding data: %v", err)
  115. }
  116. if items, ok := decodedData["items"]; !ok {
  117. t.Logf("body: %s", string(data))
  118. t.Fatalf("missing items field in empty list (all lists should return an items field)")
  119. } else if items == nil {
  120. t.Logf("body: %s", string(data))
  121. t.Fatalf("nil items field from empty list (all lists should return non-nil empty items lists)")
  122. }
  123. }
  124. func initStatusForbiddenMasterCongfig() *master.Config {
  125. masterConfig := framework.NewIntegrationTestMasterConfig()
  126. masterConfig.GenericConfig.Authorization.Authorizer = authorizerfactory.NewAlwaysDenyAuthorizer()
  127. return masterConfig
  128. }
  129. func initUnauthorizedMasterCongfig() *master.Config {
  130. masterConfig := framework.NewIntegrationTestMasterConfig()
  131. tokenAuthenticator := tokentest.New()
  132. tokenAuthenticator.Tokens[AliceToken] = &user.DefaultInfo{Name: "alice", UID: "1"}
  133. tokenAuthenticator.Tokens[BobToken] = &user.DefaultInfo{Name: "bob", UID: "2"}
  134. masterConfig.GenericConfig.Authentication.Authenticator = group.NewGroupAdder(bearertoken.New(tokenAuthenticator), []string{user.AllAuthenticated})
  135. masterConfig.GenericConfig.Authorization.Authorizer = allowAliceAuthorizer{}
  136. return masterConfig
  137. }
  138. func TestStatus(t *testing.T) {
  139. testCases := []struct {
  140. name string
  141. masterConfig *master.Config
  142. statusCode int
  143. reqPath string
  144. reason string
  145. message string
  146. }{
  147. {
  148. name: "404",
  149. masterConfig: nil,
  150. statusCode: http.StatusNotFound,
  151. reqPath: "/apis/batch/v1/namespaces/default/jobs/foo",
  152. reason: "NotFound",
  153. message: `jobs.batch "foo" not found`,
  154. },
  155. {
  156. name: "403",
  157. masterConfig: initStatusForbiddenMasterCongfig(),
  158. statusCode: http.StatusForbidden,
  159. reqPath: "/apis",
  160. reason: "Forbidden",
  161. message: `forbidden: User "" cannot get path "/apis": Everything is forbidden.`,
  162. },
  163. {
  164. name: "401",
  165. masterConfig: initUnauthorizedMasterCongfig(),
  166. statusCode: http.StatusUnauthorized,
  167. reqPath: "/apis",
  168. reason: "Unauthorized",
  169. message: `Unauthorized`,
  170. },
  171. }
  172. for _, tc := range testCases {
  173. _, s, closeFn := framework.RunAMaster(tc.masterConfig)
  174. defer closeFn()
  175. u := s.URL + tc.reqPath
  176. resp, err := http.Get(u)
  177. if err != nil {
  178. t.Fatalf("unexpected error getting %s: %v", u, err)
  179. }
  180. if resp.StatusCode != tc.statusCode {
  181. t.Fatalf("got status %v instead of %s", resp.StatusCode, tc.name)
  182. }
  183. defer resp.Body.Close()
  184. data, _ := ioutil.ReadAll(resp.Body)
  185. decodedData := map[string]interface{}{}
  186. if err := json.Unmarshal(data, &decodedData); err != nil {
  187. t.Logf("body: %s", string(data))
  188. t.Fatalf("got error decoding data: %v", err)
  189. }
  190. t.Logf("body: %s", string(data))
  191. if got, expected := decodedData["apiVersion"], "v1"; got != expected {
  192. t.Errorf("unexpected apiVersion %q, expected %q", got, expected)
  193. }
  194. if got, expected := decodedData["kind"], "Status"; got != expected {
  195. t.Errorf("unexpected kind %q, expected %q", got, expected)
  196. }
  197. if got, expected := decodedData["status"], "Failure"; got != expected {
  198. t.Errorf("unexpected status %q, expected %q", got, expected)
  199. }
  200. if got, expected := decodedData["code"], float64(tc.statusCode); got != expected {
  201. t.Errorf("unexpected code %v, expected %v", got, expected)
  202. }
  203. if got, expected := decodedData["reason"], tc.reason; got != expected {
  204. t.Errorf("unexpected reason %v, expected %v", got, expected)
  205. }
  206. if got, expected := decodedData["message"], tc.message; got != expected {
  207. t.Errorf("unexpected message %v, expected %v", got, expected)
  208. }
  209. }
  210. }
  211. func constructBody(val string, size int, field string, t *testing.T) *appsv1.Deployment {
  212. var replicas int32 = 1
  213. deploymentObject := &appsv1.Deployment{
  214. TypeMeta: metav1.TypeMeta{
  215. Kind: "Deployment",
  216. APIVersion: "apps/v1",
  217. },
  218. ObjectMeta: metav1.ObjectMeta{
  219. Namespace: "default",
  220. Name: "test",
  221. },
  222. Spec: appsv1.DeploymentSpec{
  223. Replicas: &replicas,
  224. Selector: &metav1.LabelSelector{
  225. MatchLabels: map[string]string{
  226. "foo": "bar",
  227. },
  228. },
  229. Strategy: appsv1.DeploymentStrategy{
  230. Type: appsv1.RollingUpdateDeploymentStrategyType,
  231. },
  232. Template: corev1.PodTemplateSpec{
  233. ObjectMeta: metav1.ObjectMeta{
  234. Labels: map[string]string{"foo": "bar"},
  235. },
  236. Spec: corev1.PodSpec{
  237. Containers: []corev1.Container{
  238. {
  239. Name: "foo",
  240. Image: "foo",
  241. },
  242. },
  243. },
  244. },
  245. },
  246. }
  247. switch field {
  248. case "labels":
  249. labelsMap := map[string]string{}
  250. for i := 0; i < size; i++ {
  251. key := val + strconv.Itoa(i)
  252. labelsMap[key] = val
  253. }
  254. deploymentObject.ObjectMeta.Labels = labelsMap
  255. case "annotations":
  256. annotationsMap := map[string]string{}
  257. for i := 0; i < size; i++ {
  258. key := val + strconv.Itoa(i)
  259. annotationsMap[key] = val
  260. }
  261. deploymentObject.ObjectMeta.Annotations = annotationsMap
  262. case "finalizers":
  263. finalizerString := []string{}
  264. for i := 0; i < size; i++ {
  265. finalizerString = append(finalizerString, val)
  266. }
  267. deploymentObject.ObjectMeta.Finalizers = finalizerString
  268. default:
  269. t.Fatalf("Unexpected field: %s used for making large deployment object value", field)
  270. }
  271. return deploymentObject
  272. }
  273. func TestObjectSizeResponses(t *testing.T) {
  274. _, s, closeFn := framework.RunAMaster(nil)
  275. defer closeFn()
  276. client := clientset.NewForConfigOrDie(&restclient.Config{Host: s.URL})
  277. const DeploymentMegabyteSize = 100000
  278. const DeploymentTwoMegabyteSize = 1000000
  279. expectedMsgFor1MB := `etcdserver: request is too large`
  280. expectedMsgFor2MB := `rpc error: code = ResourceExhausted desc = trying to send message larger than max`
  281. expectedMsgForLargeAnnotation := `metadata.annotations: Too long: must have at most 262144 characters`
  282. deployment1 := constructBody("a", DeploymentMegabyteSize, "labels", t) // >1 MB file
  283. deployment2 := constructBody("a", DeploymentTwoMegabyteSize, "labels", t) // >2 MB file
  284. deployment3 := constructBody("a", DeploymentMegabyteSize, "annotations", t)
  285. deployment4 := constructBody("sample/sample", DeploymentMegabyteSize, "finalizers", t) // >1 MB file
  286. deployment5 := constructBody("sample/sample", DeploymentTwoMegabyteSize, "finalizers", t) // >2 MB file
  287. requests := []struct {
  288. size string
  289. deploymentObject *appsv1.Deployment
  290. expectedMessage string
  291. }{
  292. {"1 MB", deployment1, expectedMsgFor1MB},
  293. {"2 MB", deployment2, expectedMsgFor2MB},
  294. {"1 MB", deployment3, expectedMsgForLargeAnnotation},
  295. {"1 MB", deployment4, expectedMsgFor1MB},
  296. {"2 MB", deployment5, expectedMsgFor2MB},
  297. }
  298. for _, r := range requests {
  299. t.Run(r.size, func(t *testing.T) {
  300. _, err := client.AppsV1().Deployments(metav1.NamespaceDefault).Create(r.deploymentObject)
  301. if err != nil {
  302. if !strings.Contains(err.Error(), r.expectedMessage) {
  303. t.Errorf("got: %s;want: %s", err.Error(), r.expectedMessage)
  304. }
  305. }
  306. })
  307. }
  308. }
  309. func TestWatchSucceedsWithoutArgs(t *testing.T) {
  310. _, s, closeFn := framework.RunAMaster(nil)
  311. defer closeFn()
  312. resp, err := http.Get(s.URL + "/api/v1/namespaces?watch=1")
  313. if err != nil {
  314. t.Fatalf("unexpected error getting experimental prefix: %v", err)
  315. }
  316. if resp.StatusCode != http.StatusOK {
  317. t.Fatalf("got status %v instead of 200 OK", resp.StatusCode)
  318. }
  319. resp.Body.Close()
  320. }
  321. var hpaV1 = `
  322. {
  323. "apiVersion": "autoscaling/v1",
  324. "kind": "HorizontalPodAutoscaler",
  325. "metadata": {
  326. "name": "test-hpa",
  327. "namespace": "default"
  328. },
  329. "spec": {
  330. "scaleTargetRef": {
  331. "kind": "ReplicationController",
  332. "name": "test-hpa",
  333. "namespace": "default"
  334. },
  335. "minReplicas": 1,
  336. "maxReplicas": 10,
  337. "targetCPUUtilizationPercentage": 50
  338. }
  339. }
  340. `
  341. var deploymentExtensions = `
  342. {
  343. "apiVersion": "extensions/v1beta1",
  344. "kind": "Deployment",
  345. "metadata": {
  346. "name": "test-deployment1",
  347. "namespace": "default"
  348. },
  349. "spec": {
  350. "replicas": 1,
  351. "template": {
  352. "metadata": {
  353. "labels": {
  354. "app": "nginx0"
  355. }
  356. },
  357. "spec": {
  358. "containers": [{
  359. "name": "nginx",
  360. "image": "k8s.gcr.io/nginx:1.7.9"
  361. }]
  362. }
  363. }
  364. }
  365. }
  366. `
  367. var deploymentApps = `
  368. {
  369. "apiVersion": "apps/v1",
  370. "kind": "Deployment",
  371. "metadata": {
  372. "name": "test-deployment2",
  373. "namespace": "default"
  374. },
  375. "spec": {
  376. "replicas": 1,
  377. "selector": {
  378. "matchLabels": {
  379. "app": "nginx0"
  380. }
  381. },
  382. "template": {
  383. "metadata": {
  384. "labels": {
  385. "app": "nginx0"
  386. }
  387. },
  388. "spec": {
  389. "containers": [{
  390. "name": "nginx",
  391. "image": "k8s.gcr.io/nginx:1.7.9"
  392. }]
  393. }
  394. }
  395. }
  396. }
  397. `
  398. func autoscalingPath(resource, namespace, name string) string {
  399. if namespace != "" {
  400. namespace = path.Join("namespaces", namespace)
  401. }
  402. return path.Join("/apis/autoscaling/v1", namespace, resource, name)
  403. }
  404. func batchPath(resource, namespace, name string) string {
  405. if namespace != "" {
  406. namespace = path.Join("namespaces", namespace)
  407. }
  408. return path.Join("/apis/batch/v1", namespace, resource, name)
  409. }
  410. func extensionsPath(resource, namespace, name string) string {
  411. if namespace != "" {
  412. namespace = path.Join("namespaces", namespace)
  413. }
  414. return path.Join("/apis/extensions/v1beta1", namespace, resource, name)
  415. }
  416. func appsPath(resource, namespace, name string) string {
  417. if namespace != "" {
  418. namespace = path.Join("namespaces", namespace)
  419. }
  420. return path.Join("/apis/apps/v1", namespace, resource, name)
  421. }
  422. func TestAutoscalingGroupBackwardCompatibility(t *testing.T) {
  423. _, s, closeFn := framework.RunAMaster(nil)
  424. defer closeFn()
  425. transport := http.DefaultTransport
  426. requests := []struct {
  427. verb string
  428. URL string
  429. body string
  430. expectedStatusCodes map[int]bool
  431. expectedVersion string
  432. }{
  433. {"POST", autoscalingPath("horizontalpodautoscalers", metav1.NamespaceDefault, ""), hpaV1, integration.Code201, ""},
  434. {"GET", autoscalingPath("horizontalpodautoscalers", metav1.NamespaceDefault, ""), "", integration.Code200, "autoscaling/v1"},
  435. }
  436. for _, r := range requests {
  437. bodyBytes := bytes.NewReader([]byte(r.body))
  438. req, err := http.NewRequest(r.verb, s.URL+r.URL, bodyBytes)
  439. if err != nil {
  440. t.Logf("case %v", r)
  441. t.Fatalf("unexpected error: %v", err)
  442. }
  443. func() {
  444. resp, err := transport.RoundTrip(req)
  445. defer resp.Body.Close()
  446. if err != nil {
  447. t.Logf("case %v", r)
  448. t.Fatalf("unexpected error: %v", err)
  449. }
  450. b, _ := ioutil.ReadAll(resp.Body)
  451. body := string(b)
  452. if _, ok := r.expectedStatusCodes[resp.StatusCode]; !ok {
  453. t.Logf("case %v", r)
  454. t.Errorf("Expected status one of %v, but got %v", r.expectedStatusCodes, resp.StatusCode)
  455. t.Errorf("Body: %v", body)
  456. }
  457. if !strings.Contains(body, "\"apiVersion\":\""+r.expectedVersion) {
  458. t.Logf("case %v", r)
  459. t.Errorf("Expected version %v, got body %v", r.expectedVersion, body)
  460. }
  461. }()
  462. }
  463. }
  464. func TestAppsGroupBackwardCompatibility(t *testing.T) {
  465. _, s, closeFn := framework.RunAMaster(nil)
  466. defer closeFn()
  467. transport := http.DefaultTransport
  468. requests := []struct {
  469. verb string
  470. URL string
  471. body string
  472. expectedStatusCodes map[int]bool
  473. expectedVersion string
  474. }{
  475. // Post to extensions endpoint and get back from both: extensions and apps
  476. {"POST", extensionsPath("deployments", metav1.NamespaceDefault, ""), deploymentExtensions, integration.Code201, ""},
  477. {"GET", extensionsPath("deployments", metav1.NamespaceDefault, "test-deployment1"), "", integration.Code200, "extensions/v1beta1"},
  478. {"GET", appsPath("deployments", metav1.NamespaceDefault, "test-deployment1"), "", integration.Code200, "apps/v1"},
  479. {"DELETE", extensionsPath("deployments", metav1.NamespaceDefault, "test-deployment1"), "", integration.Code200, "extensions/v1beta1"},
  480. // Post to apps endpoint and get back from both: apps and extensions
  481. {"POST", appsPath("deployments", metav1.NamespaceDefault, ""), deploymentApps, integration.Code201, ""},
  482. {"GET", appsPath("deployments", metav1.NamespaceDefault, "test-deployment2"), "", integration.Code200, "apps/v1"},
  483. {"GET", extensionsPath("deployments", metav1.NamespaceDefault, "test-deployment2"), "", integration.Code200, "extensions/v1beta1"},
  484. // set propagationPolicy=Orphan to force the object to be returned so we can check the apiVersion (otherwise, we just get a status object back)
  485. {"DELETE", appsPath("deployments", metav1.NamespaceDefault, "test-deployment2") + "?propagationPolicy=Orphan", "", integration.Code200, "apps/v1"},
  486. }
  487. for _, r := range requests {
  488. bodyBytes := bytes.NewReader([]byte(r.body))
  489. req, err := http.NewRequest(r.verb, s.URL+r.URL, bodyBytes)
  490. if err != nil {
  491. t.Logf("case %v", r)
  492. t.Fatalf("unexpected error: %v", err)
  493. }
  494. func() {
  495. resp, err := transport.RoundTrip(req)
  496. defer resp.Body.Close()
  497. if err != nil {
  498. t.Logf("case %v", r)
  499. t.Fatalf("unexpected error: %v", err)
  500. }
  501. b, _ := ioutil.ReadAll(resp.Body)
  502. body := string(b)
  503. if _, ok := r.expectedStatusCodes[resp.StatusCode]; !ok {
  504. t.Logf("case %v", r)
  505. t.Errorf("Expected status one of %v, but got %v", r.expectedStatusCodes, resp.StatusCode)
  506. t.Errorf("Body: %v", body)
  507. }
  508. if !strings.Contains(body, "\"apiVersion\":\""+r.expectedVersion) {
  509. t.Logf("case %v", r)
  510. t.Errorf("Expected version %v, got body %v", r.expectedVersion, body)
  511. }
  512. }()
  513. }
  514. }
  515. func TestAccept(t *testing.T) {
  516. _, s, closeFn := framework.RunAMaster(nil)
  517. defer closeFn()
  518. resp, err := http.Get(s.URL + "/api/")
  519. if err != nil {
  520. t.Fatalf("unexpected error getting api: %v", err)
  521. }
  522. if resp.StatusCode != http.StatusOK {
  523. t.Fatalf("got status %v instead of 200 OK", resp.StatusCode)
  524. }
  525. body, _ := ioutil.ReadAll(resp.Body)
  526. if resp.Header.Get("Content-Type") != "application/json" {
  527. t.Errorf("unexpected content: %s", body)
  528. }
  529. if err := json.Unmarshal(body, &map[string]interface{}{}); err != nil {
  530. t.Fatal(err)
  531. }
  532. req, err := http.NewRequest("GET", s.URL+"/api/", nil)
  533. if err != nil {
  534. t.Fatal(err)
  535. }
  536. req.Header.Set("Accept", "application/yaml")
  537. resp, err = http.DefaultClient.Do(req)
  538. if err != nil {
  539. t.Fatal(err)
  540. }
  541. body, _ = ioutil.ReadAll(resp.Body)
  542. if resp.Header.Get("Content-Type") != "application/yaml" {
  543. t.Errorf("unexpected content: %s", body)
  544. }
  545. t.Logf("body: %s", body)
  546. if err := yaml.Unmarshal(body, &map[string]interface{}{}); err != nil {
  547. t.Fatal(err)
  548. }
  549. req, err = http.NewRequest("GET", s.URL+"/api/", nil)
  550. if err != nil {
  551. t.Fatal(err)
  552. }
  553. req.Header.Set("Accept", "application/json, application/yaml")
  554. resp, err = http.DefaultClient.Do(req)
  555. if err != nil {
  556. t.Fatal(err)
  557. }
  558. body, _ = ioutil.ReadAll(resp.Body)
  559. if resp.Header.Get("Content-Type") != "application/json" {
  560. t.Errorf("unexpected content: %s", body)
  561. }
  562. t.Logf("body: %s", body)
  563. if err := yaml.Unmarshal(body, &map[string]interface{}{}); err != nil {
  564. t.Fatal(err)
  565. }
  566. req, err = http.NewRequest("GET", s.URL+"/api/", nil)
  567. if err != nil {
  568. t.Fatal(err)
  569. }
  570. req.Header.Set("Accept", "application") // not a valid media type
  571. resp, err = http.DefaultClient.Do(req)
  572. if err != nil {
  573. t.Fatal(err)
  574. }
  575. if resp.StatusCode != http.StatusNotAcceptable {
  576. t.Errorf("unexpected error from the server")
  577. }
  578. }
  579. func countEndpoints(eps *corev1.Endpoints) int {
  580. count := 0
  581. for i := range eps.Subsets {
  582. count += len(eps.Subsets[i].Addresses) * len(eps.Subsets[i].Ports)
  583. }
  584. return count
  585. }
  586. func TestMasterService(t *testing.T) {
  587. _, s, closeFn := framework.RunAMaster(framework.NewIntegrationTestMasterConfig())
  588. defer closeFn()
  589. client := clientset.NewForConfigOrDie(&restclient.Config{Host: s.URL})
  590. err := wait.Poll(time.Second, time.Minute, func() (bool, error) {
  591. svcList, err := client.CoreV1().Services(metav1.NamespaceDefault).List(metav1.ListOptions{})
  592. if err != nil {
  593. t.Errorf("unexpected error: %v", err)
  594. return false, nil
  595. }
  596. found := false
  597. for i := range svcList.Items {
  598. if svcList.Items[i].Name == "kubernetes" {
  599. found = true
  600. break
  601. }
  602. }
  603. if found {
  604. ep, err := client.CoreV1().Endpoints(metav1.NamespaceDefault).Get("kubernetes", metav1.GetOptions{})
  605. if err != nil {
  606. return false, nil
  607. }
  608. if countEndpoints(ep) == 0 {
  609. return false, fmt.Errorf("no endpoints for kubernetes service: %v", ep)
  610. }
  611. return true, nil
  612. }
  613. return false, nil
  614. })
  615. if err != nil {
  616. t.Errorf("unexpected error: %v", err)
  617. }
  618. }
  619. func TestServiceAlloc(t *testing.T) {
  620. cfg := framework.NewIntegrationTestMasterConfig()
  621. _, cidr, err := net.ParseCIDR("192.168.0.0/29")
  622. if err != nil {
  623. t.Fatalf("bad cidr: %v", err)
  624. }
  625. cfg.ExtraConfig.ServiceIPRange = *cidr
  626. _, s, closeFn := framework.RunAMaster(cfg)
  627. defer closeFn()
  628. client := clientset.NewForConfigOrDie(&restclient.Config{Host: s.URL})
  629. svc := func(i int) *corev1.Service {
  630. return &corev1.Service{
  631. ObjectMeta: metav1.ObjectMeta{
  632. Name: fmt.Sprintf("svc-%v", i),
  633. },
  634. Spec: corev1.ServiceSpec{
  635. Type: corev1.ServiceTypeClusterIP,
  636. Ports: []corev1.ServicePort{
  637. {Port: 80},
  638. },
  639. },
  640. }
  641. }
  642. // Wait until the default "kubernetes" service is created.
  643. if err = wait.Poll(250*time.Millisecond, time.Minute, func() (bool, error) {
  644. _, err := client.CoreV1().Services(metav1.NamespaceDefault).Get("kubernetes", metav1.GetOptions{})
  645. if err != nil && !errors.IsNotFound(err) {
  646. return false, err
  647. }
  648. return !errors.IsNotFound(err), nil
  649. }); err != nil {
  650. t.Fatalf("creating kubernetes service timed out")
  651. }
  652. // make 5 more services to take up all IPs
  653. for i := 0; i < 5; i++ {
  654. if _, err := client.CoreV1().Services(metav1.NamespaceDefault).Create(svc(i)); err != nil {
  655. t.Error(err)
  656. }
  657. }
  658. // Make another service. It will fail because we're out of cluster IPs
  659. if _, err := client.CoreV1().Services(metav1.NamespaceDefault).Create(svc(8)); err != nil {
  660. if !strings.Contains(err.Error(), "range is full") {
  661. t.Errorf("unexpected error text: %v", err)
  662. }
  663. } else {
  664. svcs, err := client.CoreV1().Services(metav1.NamespaceAll).List(metav1.ListOptions{})
  665. if err != nil {
  666. t.Fatalf("unexpected success, and error getting the services: %v", err)
  667. }
  668. allIPs := []string{}
  669. for _, s := range svcs.Items {
  670. allIPs = append(allIPs, s.Spec.ClusterIP)
  671. }
  672. t.Fatalf("unexpected creation success. The following IPs exist: %#v. It should only be possible to allocate 2 IP addresses in this cluster.\n\n%#v", allIPs, svcs)
  673. }
  674. // Delete the first service.
  675. if err := client.CoreV1().Services(metav1.NamespaceDefault).Delete(svc(1).ObjectMeta.Name, nil); err != nil {
  676. t.Fatalf("got unexpected error: %v", err)
  677. }
  678. // This time creating the second service should work.
  679. if _, err := client.CoreV1().Services(metav1.NamespaceDefault).Create(svc(8)); err != nil {
  680. t.Fatalf("got unexpected error: %v", err)
  681. }
  682. }
  683. // TestUpdateNodeObjects represents a simple version of the behavior of node checkins at steady
  684. // state. This test allows for easy profiling of a realistic master scenario for baseline CPU
  685. // in very large clusters. It is disabled by default - start a kube-apiserver and pass
  686. // UPDATE_NODE_APISERVER as the host value.
  687. func TestUpdateNodeObjects(t *testing.T) {
  688. server := os.Getenv("UPDATE_NODE_APISERVER")
  689. if len(server) == 0 {
  690. t.Skip("UPDATE_NODE_APISERVER is not set")
  691. }
  692. c := clienttypedv1.NewForConfigOrDie(&restclient.Config{
  693. QPS: 10000,
  694. Host: server,
  695. ContentConfig: restclient.ContentConfig{
  696. AcceptContentTypes: "application/vnd.kubernetes.protobuf",
  697. ContentType: "application/vnd.kubernetes.protobuf",
  698. },
  699. })
  700. nodes := 400
  701. listers := 5
  702. watchers := 50
  703. iterations := 10000
  704. for i := 0; i < nodes*6; i++ {
  705. c.Nodes().Delete(fmt.Sprintf("node-%d", i), nil)
  706. _, err := c.Nodes().Create(&corev1.Node{
  707. ObjectMeta: metav1.ObjectMeta{
  708. Name: fmt.Sprintf("node-%d", i),
  709. },
  710. })
  711. if err != nil {
  712. t.Fatal(err)
  713. }
  714. }
  715. for k := 0; k < listers; k++ {
  716. go func(lister int) {
  717. for i := 0; i < iterations; i++ {
  718. _, err := c.Nodes().List(metav1.ListOptions{})
  719. if err != nil {
  720. fmt.Printf("[list:%d] error after %d: %v\n", lister, i, err)
  721. break
  722. }
  723. time.Sleep(time.Duration(lister)*10*time.Millisecond + 1500*time.Millisecond)
  724. }
  725. }(k)
  726. }
  727. for k := 0; k < watchers; k++ {
  728. go func(lister int) {
  729. w, err := c.Nodes().Watch(metav1.ListOptions{})
  730. if err != nil {
  731. fmt.Printf("[watch:%d] error: %v", lister, err)
  732. return
  733. }
  734. i := 0
  735. for r := range w.ResultChan() {
  736. i++
  737. if _, ok := r.Object.(*corev1.Node); !ok {
  738. fmt.Printf("[watch:%d] unexpected object after %d: %#v\n", lister, i, r)
  739. }
  740. if i%100 == 0 {
  741. fmt.Printf("[watch:%d] iteration %d ...\n", lister, i)
  742. }
  743. }
  744. fmt.Printf("[watch:%d] done\n", lister)
  745. }(k)
  746. }
  747. var wg sync.WaitGroup
  748. wg.Add(nodes - listers)
  749. for j := 0; j < nodes; j++ {
  750. go func(node int) {
  751. var lastCount int
  752. for i := 0; i < iterations; i++ {
  753. if i%100 == 0 {
  754. fmt.Printf("[%d] iteration %d ...\n", node, i)
  755. }
  756. if i%20 == 0 {
  757. _, err := c.Nodes().List(metav1.ListOptions{})
  758. if err != nil {
  759. fmt.Printf("[%d] error after %d: %v\n", node, i, err)
  760. break
  761. }
  762. }
  763. r, err := c.Nodes().List(metav1.ListOptions{
  764. FieldSelector: fmt.Sprintf("metadata.name=node-%d", node),
  765. ResourceVersion: "0",
  766. })
  767. if err != nil {
  768. fmt.Printf("[%d] error after %d: %v\n", node, i, err)
  769. break
  770. }
  771. if len(r.Items) != 1 {
  772. fmt.Printf("[%d] error after %d: unexpected list count\n", node, i)
  773. break
  774. }
  775. n, err := c.Nodes().Get(fmt.Sprintf("node-%d", node), metav1.GetOptions{})
  776. if err != nil {
  777. fmt.Printf("[%d] error after %d: %v\n", node, i, err)
  778. break
  779. }
  780. if len(n.Status.Conditions) != lastCount {
  781. fmt.Printf("[%d] worker set %d, read %d conditions\n", node, lastCount, len(n.Status.Conditions))
  782. break
  783. }
  784. previousCount := lastCount
  785. switch {
  786. case i%4 == 0:
  787. lastCount = 1
  788. n.Status.Conditions = []corev1.NodeCondition{
  789. {
  790. Type: corev1.NodeReady,
  791. Status: corev1.ConditionTrue,
  792. Reason: "foo",
  793. },
  794. }
  795. case i%4 == 1:
  796. lastCount = 2
  797. n.Status.Conditions = []corev1.NodeCondition{
  798. {
  799. Type: corev1.NodeReady,
  800. Status: corev1.ConditionFalse,
  801. Reason: "foo",
  802. },
  803. {
  804. Type: corev1.NodeDiskPressure,
  805. Status: corev1.ConditionTrue,
  806. Reason: "bar",
  807. },
  808. }
  809. case i%4 == 2:
  810. lastCount = 0
  811. n.Status.Conditions = nil
  812. }
  813. if _, err := c.Nodes().UpdateStatus(n); err != nil {
  814. if !errors.IsConflict(err) {
  815. fmt.Printf("[%d] error after %d: %v\n", node, i, err)
  816. break
  817. }
  818. lastCount = previousCount
  819. }
  820. }
  821. wg.Done()
  822. fmt.Printf("[%d] done\n", node)
  823. }(j)
  824. }
  825. wg.Wait()
  826. }