website/content/zh/docs/tasks/run-application/run-replicated-stateful-app...

657 lines
30 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

---
title: 运行一个有状态的应用程序
content_template: templates/tutorial
weight: 30
---
<!-- ---
reviewers:
- enisoc
- erictune
- foxish
- janetkuo
- kow3ns
- smarterclayton
title: Run a Replicated Stateful Application
content_template: templates/tutorial
weight: 30
--- -->
{{% capture overview %}}
<!-- This page shows how to run a replicated stateful application using a
[StatefulSet](/docs/concepts/workloads/controllers/statefulset/) controller.
The example is a MySQL single-master topology with multiple slaves running
asynchronous replication. -->
该页面显示如何使用[StatefulSet](/docs/concepts/workloads/controllers/statefulset/) 控制器去运行一个有状态的应用程序。此例是一主多从的 MySQL 集群。
<!-- Note that **this is not a production configuration**.
In particular, MySQL settings remain on insecure defaults to keep the focus
on general patterns for running stateful applications in Kubernetes. -->
请注意 **这不是生产配置**
重点是, MySQL 设置保留在不安全的默认值上,使重点放在 Kubernetes 中运行有状态应用程序的常规模式。
{{% /capture %}}
{{% capture prerequisites %}}
<!-- * {{< include "task-tutorial-prereqs.md" >}} {{< version-check >}}
* {{< include "default-storage-class-prereqs.md" >}}
* This tutorial assumes you are familiar with
[PersistentVolumes](/docs/concepts/storage/persistent-volumes/)
and [StatefulSets](/docs/concepts/workloads/controllers/statefulset/),
as well as other core concepts like [Pods](/docs/concepts/workloads/pods/pod/),
[Services](/docs/concepts/services-networking/service/), and
[ConfigMaps](/docs/tasks/configure-pod-container/configure-pod-configmap/).
* Some familiarity with MySQL helps, but this tutorial aims to present
general patterns that should be useful for other systems. -->
* {{< include "task-tutorial-prereqs.md" >}} {{< version-check >}}
* {{< include "default-storage-class-prereqs.md" >}}
* 本教程假定您熟悉
[PersistentVolumes](/docs/concepts/storage/persistent-volumes/)
与 [StatefulSets](/docs/concepts/workloads/controllers/statefulset/),
以及其他核心概念,例如[Pods](/docs/concepts/workloads/pods/pod/),
[Services](/docs/concepts/services-networking/service/), 与
[ConfigMaps](/docs/tasks/configure-pod-container/configure-pod-configmap/).
* 熟悉 MySQL 会有所帮助,但是本教程旨在介绍对其他系统应该有用的常规模式。
{{% /capture %}}
{{% capture objectives %}}
<!-- * Deploy a replicated MySQL topology with a StatefulSet controller.
* Send MySQL client traffic.
* Observe resistance to downtime.
* Scale the StatefulSet up and down. -->
* 使用 StatefulSet 控制器部署复制的 MySQL 拓扑。
* 发送 MySQL 客户端流量。
* 观察对宕机的抵抗力。
* 缩放 StatefulSet 的大小。
{{% /capture %}}
{{% capture lessoncontent %}}
<!-- ## Deploy MySQL -->
## 部署 MySQL
<!-- The example MySQL deployment consists of a ConfigMap, two Services,
and a StatefulSet. -->
部署 MySQL 示例,包含一个 ConfigMap两个 Services与一个 StatefulSet。
### ConfigMap
<!-- Create the ConfigMap from the following YAML configuration file: -->
从以下的 YAML 配置文件创建 ConfigMap
{{< codenew file="application/mysql/mysql-configmap.yaml" >}}
```shell
kubectl apply -f https://k8s.io/examples/application/mysql/mysql-configmap.yaml
```
<!-- This ConfigMap provides `my.cnf` overrides that let you independently control
configuration on the MySQL master and slaves.
In this case, you want the master to be able to serve replication logs to slaves
and you want slaves to reject any writes that don't come via replication. -->
这个 ConfigMap 提供 `my.cnf` 覆盖,使您可以独立控制 MySQL 主服务器和从服务器的配置。
在这种情况下,您希望主服务器能够将复制日志提供给从服务器,并且希望从服务器拒绝任何不是通过复制进行的写操作。
<!-- There's nothing special about the ConfigMap itself that causes different
portions to apply to different Pods.
Each Pod decides which portion to look at as it's initializing,
based on information provided by the StatefulSet controller. -->
ConfigMap 本身没有什么特别之处,它可以使不同部分应用于不同的 Pod。
每个 Pod 都会决定在初始化时要看基于 StatefulSet 控制器提供的信息。
<!-- ### Services -->
### Services
<!-- Create the Services from the following YAML configuration file: -->
从以下 YAML 配置文件创建服务:
{{< codenew file="application/mysql/mysql-services.yaml" >}}
```shell
kubectl apply -f https://k8s.io/examples/application/mysql/mysql-services.yaml
```
<!-- The Headless Service provides a home for the DNS entries that the StatefulSet
controller creates for each Pod that's part of the set.
Because the Headless Service is named `mysql`, the Pods are accessible by
resolving `<pod-name>.mysql` from within any other Pod in the same Kubernetes
cluster and namespace. -->
Headless Service 给 StatefulSet 控制器为集合中每个 Pod 创建的 DNS 条目提供了一个宿主。因为 Headless Service 名为 `mysql`,所以可以通过在同一 Kubernetes 集群和 namespace 中的任何其他 Pod 内解析 `<pod-name>.mysql` 来访问 Pod。
<!-- The Client Service, called `mysql-read`, is a normal Service with its own
cluster IP that distributes connections across all MySQL Pods that report
being Ready. The set of potential endpoints includes the MySQL master and all
slaves. -->
客户端 Service 称为 `mysql-read`,是一种常规 Service具有其自己的群集 IP该群集 IP 在报告为就绪的所有MySQL Pod 中分配连接。可能端点的集合包括 MySQL 主节点和所有从节点。
<!-- Note that only read queries can use the load-balanced Client Service.
Because there is only one MySQL master, clients should connect directly to the
MySQL master Pod (through its DNS entry within the Headless Service) to execute
writes. -->
请注意,只有读取查询才能使用负载平衡的客户端 Service。因为只有一个 MySQL 主服务器,所以客户端应直接连接到 MySQL 主服务器 Pod (通过其在 Headless Service 中的 DNS 条目)以执行写入操作。
### StatefulSet
<!-- Finally, create the StatefulSet from the following YAML configuration file: -->
最后,从以下 YAML 配置文件创建 StatefulSet
{{< codenew file="application/mysql/mysql-statefulset.yaml" >}}
```shell
kubectl apply -f https://k8s.io/examples/application/mysql/mysql-statefulset.yaml
```
<!-- You can watch the startup progress by running: -->
您可以通过运行以下命令查看启动进度:
```shell
kubectl get pods -l app=mysql --watch
```
<!-- After a while, you should see all 3 Pods become Running: -->
一段时间后您应该看到所有3个 Pod 都开始运行:
```
NAME READY STATUS RESTARTS AGE
mysql-0 2/2 Running 0 2m
mysql-1 2/2 Running 0 1m
mysql-2 2/2 Running 0 1m
```
<!-- Press **Ctrl+C** to cancel the watch.
If you don't see any progress, make sure you have a dynamic PersistentVolume
provisioner enabled as mentioned in the [prerequisites](#before-you-begin). -->
输入**Ctrl+C**取消观察。
如果您看不到任何进度,确保已启用[前提条件](#before-you-begin)中提到的动态 PersistentVolume 预配器。
<!-- This manifest uses a variety of techniques for managing stateful Pods as part of
a StatefulSet. The next section highlights some of these techniques to explain
what happens as the StatefulSet creates Pods. -->
该清单使用多种技术来管理作为 StatefulSet 一部分的有状态 Pod。下一节重点介绍其中一些技巧以解释 StatefulSet 创建 Pod 时发生的状况。
<!-- ## Understanding stateful Pod initialization -->
## 了解有状态的 Pod 初始化
<!-- The StatefulSet controller starts Pods one at a time, in order by their
ordinal index.
It waits until each Pod reports being Ready before starting the next one. -->
StatefulSet 控制器一次按顺序启动 Pod 序数索引。它一直等到每个 Pod 报告就绪为止,然后再开始下一个 Pod。
<!-- In addition, the controller assigns each Pod a unique, stable name of the form
`<statefulset-name>-<ordinal-index>`, which results in Pods named `mysql-0`,
`mysql-1`, and `mysql-2`. -->
此外,控制器为每个 Pod 分配一个唯一,稳定的表单名称 `<statefulset-name>-<ordinal-index>` 其结果是 Pods 名为 mysql-0mysql-1 和 mysql-2。
<!-- The Pod template in the above StatefulSet manifest takes advantage of these
properties to perform orderly startup of MySQL replication. -->
上述 StatefulSet 清单中的 Pod 模板利用这些属性来执行 MySQL 复制的有序启动。
<!-- ### Generating configuration -->
### 生成配置
<!-- Before starting any of the containers in the Pod spec, the Pod first runs any
[Init Containers](/docs/concepts/workloads/pods/init-containers/)
in the order defined. -->
在启动 Pod 规范中的任何容器之前, Pod 首先运行任何[初始容器](/docs/concepts/workloads/pods/init-containers/)按照定义的顺序。
<!-- The first Init Container, named `init-mysql`, generates special MySQL config
files based on the ordinal index. -->
第一个名为 `init-mysql` 的初始化容器,根据序号索引生成特殊的 MySQL 配置文件。
<!-- The script determines its own ordinal index by extracting it from the end of
the Pod name, which is returned by the `hostname` command.
Then it saves the ordinal (with a numeric offset to avoid reserved values)
into a file called `server-id.cnf` in the MySQL `conf.d` directory.
This translates the unique, stable identity provided by the StatefulSet
controller into the domain of MySQL server IDs, which require the same
properties. -->
该脚本通过从 Pod 名称的末尾提取索引来确定自己的序号索引,该名称由 `hostname` 命令返回。
然后将序数(带有数字偏移量以避免保留值)保存到 MySQL conf.d 目录中的文件 server-id.cnf 中。
这将转换 StatefulSet 提供的唯一,稳定的身份控制器进入需要相同属性的 MySQL 服务器 ID 的范围。
<!-- The script in the `init-mysql` container also applies either `master.cnf` or
`slave.cnf` from the ConfigMap by copying the contents into `conf.d`.
Because the example topology consists of a single MySQL master and any number of
slaves, the script simply assigns ordinal `0` to be the master, and everyone
else to be slaves. -->
通过将内容复制到 conf.d 中init-mysql 容器中的脚本也可以应用 ConfigMap 中的 master.cnf 或 slave.cnf。由于示例拓扑由单个 MySQL 主节点和任意数量的从节点组成,因此脚本仅将序数 `0` 指定为主节点,而将其他所有人指定为从节点。与 StatefulSet 控制器的[部署顺序保证](/docs/concepts/workloads/controllers/statefulset/#deployment-and-scaling-guarantees/),这样可以确保 MySQL 主服务器在创建从服务器之前已准备就绪,以便它们可以开始复制。
<!-- ### Cloning existing data -->
### 克隆现有数据
<!-- In general, when a new Pod joins the set as a slave, it must assume the MySQL
master might already have data on it. It also must assume that the replication
logs might not go all the way back to the beginning of time. -->
通常,当新的 Pod 作为从节点加入集合时,必须假定 MySQL 主节点可能已经有数据。还必须假设复制日志可能不会一直追溯到时间的开始。
<!-- These conservative assumptions are the key to allow a running StatefulSet
to scale up and down over time, rather than being fixed at its initial size. -->
这些保守假设的关键是允许正在运行的 StatefulSet 随时间扩大和缩小而不是固定在其初始大小。
<!-- The second Init Container, named `clone-mysql`, performs a clone operation on
a slave Pod the first time it starts up on an empty PersistentVolume.
That means it copies all existing data from another running Pod,
so its local state is consistent enough to begin replicating from the master. -->
第二个名为 `clone-mysql` 的初始化容器,第一次在从属 Pod 上以空 PersistentVolume 启动时,会对从属 Pod 执行克隆操作。这意味着它将从另一个运行的 Pod 复制所有现有数据,因此其本地状态足够一致,可以开始主从服务器复制。
<!-- MySQL itself does not provide a mechanism to do this, so the example uses a
popular open-source tool called Percona XtraBackup.
During the clone, the source MySQL server might suffer reduced performance.
To minimize impact on the MySQL master, the script instructs each Pod to clone
from the Pod whose ordinal index is one lower.
This works because the StatefulSet controller always ensures Pod `N` is
Ready before starting Pod `N+1`. -->
MySQL 本身不提供执行此操作的机制,因此该示例使用了一种流行的开源工具 Percona XtraBackup。
在克隆期间,源 MySQL 服务器可能会降低性能。
为了最大程度地减少对 MySQL 主机的影响,该脚本指示每个 Pod 从序号较低的 Pod 中克隆。
可以这样做的原因是 StatefulSet 控制器始终确保在启动 Pod N + 1 之前 Pod N 已准备就绪。
<!-- ### Starting replication -->
### 开始复制
<!-- After the Init Containers complete successfully, the regular containers run.
The MySQL Pods consist of a `mysql` container that runs the actual `mysqld`
server, and an `xtrabackup` container that acts as a
[sidecar](https://kubernetes.io/blog/2015/06/the-distributed-system-toolkit-patterns). -->
初始化容器成功完成后,常规容器将运行。
MySQL Pods 由运行实际 `mysqld` 服务器的 `mysql` 容器和充当[辅助工具](https://kubernetes.io/blog/2015/06/the-distributed-system-toolkit-patterns)的 xtrabackup 容器组成。
<!-- The `xtrabackup` sidecar looks at the cloned data files and determines if
it's necessary to initialize MySQL replication on the slave.
If so, it waits for `mysqld` to be ready and then executes the
`CHANGE MASTER TO` and `START SLAVE` commands with replication parameters
extracted from the XtraBackup clone files. -->
`xtrabackup` 辅助工具查看克隆的数据文件,并确定是否有必要在从属服务器上初始化 MySQL 复制。
如果是这样,它将等待 `mysqld` 准备就绪,然后执行带有从 XtraBackup 克隆文件中提取的复制参数 `CHANGE MASTER TO``START SLAVE` 命令。
<!-- Once a slave begins replication, it remembers its MySQL master and
reconnects automatically if the server restarts or the connection dies.
Also, because slaves look for the master at its stable DNS name
(`mysql-0.mysql`), they automatically find the master even if it gets a new
Pod IP due to being rescheduled. -->
一旦从服务器开始复制后,它会记住其 MySQL 主服务器。并且如果服务器重新启动或连接中断,则会自动重新连接。
另外,因为从服务器会以其稳定的 DNS 名称查找主服务器(`mysql-0.mysql`),即使由于重新安排而获得新的 Pod IP他们也会自动找到主服务器。
<!-- Lastly, after starting replication, the `xtrabackup` container listens for
connections from other Pods requesting a data clone.
This server remains up indefinitely in case the StatefulSet scales up, or in
case the next Pod loses its PersistentVolumeClaim and needs to redo the clone. -->
最后,开始复制后,`xtrabackup` 容器监听来自其他 Pod 的连接数据克隆请求。
如果 StatefulSet 扩大规模,或者下一个 Pod 失去其 PersistentVolumeClaim 并需要重新克隆,则此服务器将无限期保持运行。
<!-- ## Sending client traffic -->
## 发送客户端流量
<!-- You can send test queries to the MySQL master (hostname `mysql-0.mysql`)
by running a temporary container with the `mysql:5.7` image and running the
`mysql` client binary. -->
您可以通过运行带有 `mysql:5.7` 镜像的临时容器并运行 `mysql` 客户端二进制文件,将测试查询发送到 MySQL 主服务器(主机名 `mysql-0.mysql` )。
```shell
kubectl run mysql-client --image=mysql:5.7 -i --rm --restart=Never --\
mysql -h mysql-0.mysql <<EOF
CREATE DATABASE test;
CREATE TABLE test.messages (message VARCHAR(250));
INSERT INTO test.messages VALUES ('hello');
EOF
```
<!-- Use the hostname `mysql-read` to send test queries to any server that reports
being Ready: -->
使用主机名 `mysql-read` 将测试查询发送到任何报告为就绪的服务器:
```shell
kubectl run mysql-client --image=mysql:5.7 -i -t --rm --restart=Never --\
mysql -h mysql-read -e "SELECT * FROM test.messages"
```
<!-- You should get output like this: -->
您应该获得如下输出:
```
Waiting for pod default/mysql-client to be running, status is Pending, pod ready: false
+---------+
| message |
+---------+
| hello |
+---------+
pod "mysql-client" deleted
```
<!-- To demonstrate that the `mysql-read` Service distributes connections across
servers, you can run `SELECT @@server_id` in a loop: -->
为了演示 `mysql-read` 服务在服务器之间分配连接,您可以在循环中运行 `SELECT @@server_id`
```shell
kubectl run mysql-client-loop --image=mysql:5.7 -i -t --rm --restart=Never --\
bash -ic "while sleep 1; do mysql -h mysql-read -e 'SELECT @@server_id,NOW()'; done"
```
<!-- You should see the reported `@@server_id` change randomly, because a different
endpoint might be selected upon each connection attempt: -->
您应该看到报告的 `@@server_id` 发生随机变化,因为每次尝试连接时都可能选择了不同的端点:
```
+-------------+---------------------+
| @@server_id | NOW() |
+-------------+---------------------+
| 100 | 2006-01-02 15:04:05 |
+-------------+---------------------+
+-------------+---------------------+
| @@server_id | NOW() |
+-------------+---------------------+
| 102 | 2006-01-02 15:04:06 |
+-------------+---------------------+
+-------------+---------------------+
| @@server_id | NOW() |
+-------------+---------------------+
| 101 | 2006-01-02 15:04:07 |
+-------------+---------------------+
```
<!-- You can press **Ctrl+C** when you want to stop the loop, but it's useful to keep
it running in another window so you can see the effects of the following steps. -->
要停止循环时可以按 **Ctrl+C** ,但是让它在另一个窗口中运行非常有用,这样您就可以看到以下步骤的效果。
<!-- ## Simulating Pod and Node downtime -->
## 模拟 Pod 和 Node 的宕机时间
<!-- To demonstrate the increased availability of reading from the pool of slaves
instead of a single server, keep the `SELECT @@server_id` loop from above
running while you force a Pod out of the Ready state. -->
为了证明从从节点缓存而不是单个服务器读取数据的可用性提高,请在使 Pod 退出 Ready 状态时,保持 `SELECT @@server_id` 循环从上面运行。
<!-- ### Break the Readiness Probe -->
### 退出 Readiness Probe
<!-- The [readiness probe](/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#define-readiness-probes)
for the `mysql` container runs the command `mysql -h 127.0.0.1 -e 'SELECT 1'`
to make sure the server is up and able to execute queries. -->
`mysql` 容器的[readiness probe](/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#define-readiness-probes)运行命令 `mysql -h 127.0.0.1 -e 'SELECT 1'`,以确保服务器已启动并能够执行查询。
<!-- One way to force this readiness probe to fail is to break that command: -->
迫使 readiness probe 失败的一种方法就是执行该命令:
```shell
kubectl exec mysql-2 -c mysql -- mv /usr/bin/mysql /usr/bin/mysql.off
```
<!-- This reaches into the actual container's filesystem for Pod `mysql-2` and
renames the `mysql` command so the readiness probe can't find it.
After a few seconds, the Pod should report one of its containers as not Ready,
which you can check by running: -->
这进入 Pod `mysql-2` 的实际容器文件系统,并重命名 `mysql` 命令,以便 readiness probe 无法找到它。
几秒钟后, Pod 会将其中一个容器报告为未就绪,您可以通过运行以下命令进行检查:
```shell
kubectl get pod mysql-2
```
<!-- Look for `1/2` in the `READY` column: -->
`就绪` 列中查找 ` 1/2`
```
NAME READY STATUS RESTARTS AGE
mysql-2 1/2 Running 0 3m
```
<!-- At this point, you should see your `SELECT @@server_id` loop continue to run,
although it never reports `102` anymore.
Recall that the `init-mysql` script defined `server-id` as `100 + $ordinal`,
so server ID `102` corresponds to Pod `mysql-2`. -->
此时,您应该会看到 `SELECT @@server_id` 循环继续运行,尽管它不再报告 `102`
回想一下,`init-mysql` 脚本将 `server-id` 定义为 `100 + $ordinal` ,因此服务器 ID `102` 对应于 Pod `mysql-2`
<!-- Now repair the Pod and it should reappear in the loop output
after a few seconds: -->
现在修复 Pod几秒钟后它应该重新出现在循环输出中
```shell
kubectl exec mysql-2 -c mysql -- mv /usr/bin/mysql.off /usr/bin/mysql
```
<!-- ### Delete Pods -->
### 删除 Pods
<!-- The StatefulSet also recreates Pods if they're deleted, similar to what a
ReplicaSet does for stateless Pods. -->
如果删除了 Pod则 StatefulSet 还会重新创建 Pod类似于 ReplicaSet 对无状态 Pod 所做的操作。
```shell
kubectl delete pod mysql-2
```
<!-- The StatefulSet controller notices that no `mysql-2` Pod exists anymore,
and creates a new one with the same name and linked to the same
PersistentVolumeClaim.
You should see server ID `102` disappear from the loop output for a while
and then return on its own. -->
StatefulSet 控制器注意到不再存在 `mysql-2` Pod并创建了一个具有相同名称并链接到相同 PersistentVolumeClaim 的新 Pod。
您应该看到服务器 ID `102` 从循环输出中消失了一段时间,然后自行返回。
<!-- ### Drain a Node -->
### 排除 Node
<!-- If your Kubernetes cluster has multiple Nodes, you can simulate Node downtime
(such as when Nodes are upgraded) by issuing a
[drain](/docs/reference/generated/kubectl/kubectl-commands/#drain). -->
如果您的 Kubernetes 集群具有多个节点,则可以通过发出以下命令[drain](/docs/reference/generated/kubectl/kubectl-commands/#drain)来模拟节点停机时间(例如升级节点时)。
<!-- First determine which Node one of the MySQL Pods is on: -->
首先确定 MySQL Pods 之一在哪个节点上:
```shell
kubectl get pod mysql-2 -o wide
```
<!-- The Node name should show up in the last column: -->
节点名称应显示在最后一列中:
```
NAME READY STATUS RESTARTS AGE IP NODE
mysql-2 2/2 Running 0 15m 10.244.5.27 kubernetes-node-9l2t
```
<!-- Then drain the Node by running the following command, which cordons it so
no new Pods may schedule there, and then evicts any existing Pods.
Replace `<node-name>` with the name of the Node you found in the last step. -->
然后通过运行以下命令耗尽节点,该命令将其封锁,以使新的 Pod 不能在那里调度,然后驱逐任何现有的 Pod。
`<node-name>` 替换为在上一步中找到的 Node 的名称。
<!-- This might impact other applications on the Node, so it's best to
**only do this in a test cluster**. -->
这可能会影响节点上的其他应用程序,因此最好
**仅在测试集群中执行此操作**
```shell
kubectl drain <node-name> --force --delete-local-data --ignore-daemonsets
```
<!-- Now you can watch as the Pod reschedules on a different Node: -->
现在,您可以观察 Pod 在其他节点上的重新安排:
```shell
kubectl get pod mysql-2 -o wide --watch
```
<!-- It should look something like this: -->
它看起来应该像这样:
```
NAME READY STATUS RESTARTS AGE IP NODE
mysql-2 2/2 Terminating 0 15m 10.244.1.56 kubernetes-node-9l2t
[...]
mysql-2 0/2 Pending 0 0s <none> kubernetes-node-fjlm
mysql-2 0/2 Init:0/2 0 0s <none> kubernetes-node-fjlm
mysql-2 0/2 Init:1/2 0 20s 10.244.5.32 kubernetes-node-fjlm
mysql-2 0/2 PodInitializing 0 21s 10.244.5.32 kubernetes-node-fjlm
mysql-2 1/2 Running 0 22s 10.244.5.32 kubernetes-node-fjlm
mysql-2 2/2 Running 0 30s 10.244.5.32 kubernetes-node-fjlm
```
<!-- And again, you should see server ID `102` disappear from the
`SELECT @@server_id` loop output for a while and then return. -->
再次,您应该看到服务器 ID `102``SELECT @@server_id` 循环输出一段时间,然后返回。
<!-- Now uncordon the Node to return it to a normal state: -->
现在 uncordon 节点,使其恢复为正常模式:
```shell
kubectl uncordon <node-name>
```
<!-- ## Scaling the number of slaves -->
## 扩展从节点数量
<!-- With MySQL replication, you can scale your read query capacity by adding slaves.
With StatefulSet, you can do this with a single command: -->
使用 MySQL 复制,您可以通过添加从节点来扩展读取查询的能力。
使用 StatefulSet您可以使用单个命令执行此操作
```shell
kubectl scale statefulset mysql --replicas=5
```
<!-- Watch the new Pods come up by running: -->
查看新的 Pod 的运行情况:
```shell
kubectl get pods -l app=mysql --watch
```
<!-- Once they're up, you should see server IDs `103` and `104` start appearing in
the `SELECT @@server_id` loop output.
You can also verify that these new servers have the data you added before they
existed: -->
一旦 Pod 启动,您应该看到服务器 IDs `103``104` 开始出现在 `SELECT @@server_id` 循环输出中。
您还可以验证这些新服务器在存在之前已添加了数据:
```shell
kubectl run mysql-client --image=mysql:5.7 -i -t --rm --restart=Never --\
mysql -h mysql-3.mysql -e "SELECT * FROM test.messages"
```
```
Waiting for pod default/mysql-client to be running, status is Pending, pod ready: false
+---------+
| message |
+---------+
| hello |
+---------+
pod "mysql-client" deleted
```
<!-- Scaling back down is also seamless: -->
向下缩放也是不停顿的:
```shell
kubectl scale statefulset mysql --replicas=3
```
<!-- Note, however, that while scaling up creates new PersistentVolumeClaims
automatically, scaling down does not automatically delete these PVCs.
This gives you the choice to keep those initialized PVCs around to make
scaling back up quicker, or to extract data before deleting them. -->
但是请注意,按比例放大会自动创建新的 PersistentVolumeClaims而按比例缩小不会自动删除这些 PVC。
这使您可以选择保留那些初始化的 PVC以更快地进行缩放或者在删除它们之前提取数据。
<!-- You can see this by running: -->
您可以通过运行以下命令查看此信息:
```shell
kubectl get pvc -l app=mysql
```
<!-- Which shows that all 5 PVCs still exist, despite having scaled the
StatefulSet down to 3: -->
这表明,尽管将 StatefulSet 缩小为3所有5个 PVC 仍然存在:
```
NAME STATUS VOLUME CAPACITY ACCESSMODES AGE
data-mysql-0 Bound pvc-8acbf5dc-b103-11e6-93fa-42010a800002 10Gi RWO 20m
data-mysql-1 Bound pvc-8ad39820-b103-11e6-93fa-42010a800002 10Gi RWO 20m
data-mysql-2 Bound pvc-8ad69a6d-b103-11e6-93fa-42010a800002 10Gi RWO 20m
data-mysql-3 Bound pvc-50043c45-b1c5-11e6-93fa-42010a800002 10Gi RWO 2m
data-mysql-4 Bound pvc-500a9957-b1c5-11e6-93fa-42010a800002 10Gi RWO 2m
```
<!-- If you don't intend to reuse the extra PVCs, you can delete them: -->
如果您不打算重复使用多余的 PVC则可以删除它们
```shell
kubectl delete pvc data-mysql-3
kubectl delete pvc data-mysql-4
```
{{% /capture %}}
{{% capture cleanup %}}
<!-- 1. Cancel the `SELECT @@server_id` loop by pressing **Ctrl+C** in its terminal,
or running the following from another terminal: -->
1. 通过在终端上按 **Ctrl+C** 取消 `SELECT @@server_id` 循环,或从另一个终端运行以下命令:
```shell
kubectl delete pod mysql-client-loop --now
```
<!-- 1. Delete the StatefulSet. This also begins terminating the Pods. -->
1. 删除 StatefulSet。这也开始终止 Pod。
```shell
kubectl delete statefulset mysql
```
<!-- 1. Verify that the Pods disappear.
They might take some time to finish terminating. -->
1. 验证 Pod 消失。
他们可能需要一些时间才能完成终止。
```shell
kubectl get pods -l app=mysql
```
<!-- You'll know the Pods have terminated when the above returns: -->
当以上内容返回时,您将知道 Pod 已终止:
```
No resources found.
```
<!-- 1. Delete the ConfigMap, Services, and PersistentVolumeClaims. -->
1. 删除 ConfigMapServices 和 PersistentVolumeClaims。
```shell
kubectl delete configmap,service,pvc -l app=mysql
```
<!-- 1. If you manually provisioned PersistentVolumes, you also need to manually
delete them, as well as release the underlying resources.
If you used a dynamic provisioner, it automatically deletes the
PersistentVolumes when it sees that you deleted the PersistentVolumeClaims.
Some dynamic provisioners (such as those for EBS and PD) also release the
underlying resources upon deleting the PersistentVolumes. -->
1. 如果您手动设置 PersistentVolume则还需要手动删除它们并释放基础资源。
如果您使用了动态预配器,当得知您删除 PersistentVolumeClaims 时,它将自动删除 PersistentVolumes。
一些动态预配器(例如用于 EBS 和 PD 的预配器)也会在删除 PersistentVolumes 时释放基础资源。
{{% /capture %}}
{{% capture whatsnext %}}
<!-- * Look in the [Helm Charts repository](https://github.com/kubernetes/charts)
for other stateful application examples. -->
* 在[Helm Charts 存储库](https://github.com/kubernetes/charts)中查找其他有状态的应用程序示例。
{{% /capture %}}