You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

159 lines
6.9 KiB

  1. HA endpoints for K8s
  2. ====================
  3. The following components require a highly available endpoints:
  4. * etcd cluster,
  5. * kube-apiserver service instances.
  6. The latter relies on a 3rd side reverse proxies, like Nginx or HAProxy, to
  7. achieve the same goal.
  8. Etcd
  9. ----
  10. In order to use an external loadbalancing (L4/TCP or L7 w/ SSL Passthrough VIP), the following variables need to be overriden in group_vars
  11. * `etcd_access_addresses`
  12. * `etcd_client_url`
  13. * `etcd_cert_alt_names`
  14. * `etcd_cert_alt_ips`
  15. ### Example of a VIP w/ FQDN
  16. ```yaml
  17. etcd_access_addresses: https://etcd.example.com:2379
  18. etcd_client_url: https://etcd.example.com:2379
  19. etcd_cert_alt_names:
  20. - "etcd.kube-system.svc.{{ dns_domain }}"
  21. - "etcd.kube-system.svc"
  22. - "etcd.kube-system"
  23. - "etcd"
  24. - "etcd.example.com" # This one needs to be added to the default etcd_cert_alt_names
  25. ```
  26. ### Example of a VIP w/o FQDN (IP only)
  27. ```yaml
  28. etcd_access_addresses: https://2.3.7.9:2379
  29. etcd_client_url: https://2.3.7.9:2379
  30. etcd_cert_alt_ips:
  31. - "2.3.7.9"
  32. ```
  33. Kube-apiserver
  34. --------------
  35. K8s components require a loadbalancer to access the apiservers via a reverse
  36. proxy. Kubespray includes support for an nginx-based proxy that resides on each
  37. non-master Kubernetes node. This is referred to as localhost loadbalancing. It
  38. is less efficient than a dedicated load balancer because it creates extra
  39. health checks on the Kubernetes apiserver, but is more practical for scenarios
  40. where an external LB or virtual IP management is inconvenient. This option is
  41. configured by the variable `loadbalancer_apiserver_localhost` (defaults to
  42. `True`. Or `False`, if there is an external `loadbalancer_apiserver` defined).
  43. You may also define the port the local internal loadbalancer uses by changing,
  44. `nginx_kube_apiserver_port`. This defaults to the value of
  45. `kube_apiserver_port`. It is also important to note that Kubespray will only
  46. configure kubelet and kube-proxy on non-master nodes to use the local internal
  47. loadbalancer.
  48. If you choose to NOT use the local internal loadbalancer, you will need to
  49. configure your own loadbalancer to achieve HA. Note that deploying a
  50. loadbalancer is up to a user and is not covered by ansible roles in Kubespray.
  51. By default, it only configures a non-HA endpoint, which points to the
  52. `access_ip` or IP address of the first server node in the `kube-master` group.
  53. It can also configure clients to use endpoints for a given loadbalancer type.
  54. The following diagram shows how traffic to the apiserver is directed.
  55. ![Image](figures/loadbalancer_localhost.png?raw=true)
  56. Note: Kubernetes master nodes still use insecure localhost access because
  57. there are bugs in Kubernetes <1.5.0 in using TLS auth on master role
  58. services. This makes backends receiving unencrypted traffic and may be a
  59. security issue when interconnecting different nodes, or maybe not, if those
  60. belong to the isolated management network without external access.
  61. A user may opt to use an external loadbalancer (LB) instead. An external LB
  62. provides access for external clients, while the internal LB accepts client
  63. connections only to the localhost.
  64. Given a frontend `VIP` address and `IP1, IP2` addresses of backends, here is
  65. an example configuration for a HAProxy service acting as an external LB:
  66. ```
  67. listen kubernetes-apiserver-https
  68. bind <VIP>:8383
  69. option ssl-hello-chk
  70. mode tcp
  71. timeout client 3h
  72. timeout server 3h
  73. server master1 <IP1>:6443
  74. server master2 <IP2>:6443
  75. balance roundrobin
  76. ```
  77. Note: That's an example config managed elsewhere outside of Kubespray.
  78. And the corresponding example global vars for such a "cluster-aware"
  79. external LB with the cluster API access modes configured in Kubespray:
  80. ```
  81. apiserver_loadbalancer_domain_name: "my-apiserver-lb.example.com"
  82. loadbalancer_apiserver:
  83. address: <VIP>
  84. port: 8383
  85. ```
  86. Note: The default kubernetes apiserver configuration binds to all interfaces,
  87. so you will need to use a different port for the vip from that the API is
  88. listening on, or set the `kube_apiserver_bind_address` so that the API only
  89. listens on a specific interface (to avoid conflict with haproxy binding the
  90. port on the VIP adddress)
  91. This domain name, or default "lb-apiserver.kubernetes.local", will be inserted
  92. into the `/etc/hosts` file of all servers in the `k8s-cluster` group and wired
  93. into the generated self-signed TLS/SSL certificates as well. Note that
  94. the HAProxy service should as well be HA and requires a VIP management, which
  95. is out of scope of this doc.
  96. There is a special case for an internal and an externally configured (not with
  97. Kubespray) LB used simultaneously. Keep in mind that the cluster is not aware
  98. of such an external LB and you need no to specify any configuration variables
  99. for it.
  100. Note: TLS/SSL termination for externally accessed API endpoints' will **not**
  101. be covered by Kubespray for that case. Make sure your external LB provides it.
  102. Alternatively you may specify an externally load balanced VIPs in the
  103. `supplementary_addresses_in_ssl_keys` list. Then, kubespray will add them into
  104. the generated cluster certifactes as well.
  105. Aside of that specific case, the `loadbalancer_apiserver` considered mutually
  106. exclusive to `loadbalancer_apiserver_localhost`.
  107. Access API endpoints are evaluated automagically, as the following:
  108. | Endpoint type | kube-master | non-master | external |
  109. |------------------------------|----------------|---------------------|---------------------|
  110. | Local LB (default) | https://bip:sp | https://lc:nsp | https://m[0].aip:sp |
  111. | Local LB + Unmanaged here LB | https://bip:sp | https://lc:nsp | https://ext |
  112. | External LB, no internal | https://bip:sp | https://lb:lp | https://lb:lp |
  113. | No ext/int LB | https://bip:sp | https://m[0].aip:sp | https://m[0].aip:sp |
  114. Where:
  115. * `m[0]` - the first node in the `kube-master` group;
  116. * `lb` - LB FQDN, `apiserver_loadbalancer_domain_name`;
  117. * `ext` - Externally load balanced VIP:port and FQDN, not managed by Kubespray;
  118. * `lc` - localhost;
  119. * `bip` - a custom bind IP or localhost for the default bind IP '0.0.0.0';
  120. * `nsp` - nginx secure port, `nginx_kube_apiserver_port`, defers to `sp`;
  121. * `sp` - secure port, `kube_apiserver_port`;
  122. * `lp` - LB port, `loadbalancer_apiserver.port`, defers to the secure port;
  123. * `ip` - the node IP, defers to the ansible IP;
  124. * `aip` - `access_ip`, defers to the ip.
  125. A second and a third column represent internal cluster access modes. The last
  126. column illustrates an example URI to access the cluster APIs externally.
  127. Kubespray has nothing to do with it, this is informational only.
  128. As you can see, the masters' internal API endpoints are always
  129. contacted via the local bind IP, which is `https://bip:sp`.
  130. **Note** that for some cases, like healthchecks of applications deployed by
  131. Kubespray, the masters' APIs are accessed via the insecure endpoint, which
  132. consists of the local `kube_apiserver_insecure_bind_address` and
  133. `kube_apiserver_insecure_port`.