synthetic_master_test.go 25 KB

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