Pacemaker+corosync搭建双节点HA集群的可靠性验证

简介: 前一篇>中为确保共享资源的不被破坏,配置了3节点集群,本文想验证一下双节点时有什么风险。 Pacemaker的手册上也有描述,Pacemaker支持法定投票和资源抢占2种方式防止脑裂。
前一篇>中为确保共享资源的不被破坏,配置了3节点集群,本文想验证一下双节点时有什么风险。

Pacemaker的手册上也有描述,Pacemaker支持法定投票和资源抢占2种方式防止脑裂。法定投票的方式确实很可靠,但是至少需要3票。如果在主备双机以外再专门搞一台机器以满足法定投票要求似乎太浪费。因此一些商业集群软件(比如MSCS,RHCS)除了节点以外还引入的仲裁盘。这个仲裁盘也算上一票,加上2个节点,共3票,只要获得其中2票即可。

参考
http://www.adirectory.blog.com/2013/01/cluster-quorum-disk/


使用仲裁盘需要有共享存储,对许多企业用户来说共享存储是标配,所以也不需要额外的投资。
但是开源的Pacemaker就不依赖于共享存储,也就没有仲裁盘的说法。那么在Pacemaker上配置抢占资源如何呢?我们没找到配置的案例,但设想可以这样做:
提供一个文件服务器(比如NFS),在上面建一个文件作为锁文件。然后自己写一个RA(不知道有没有现成的这样的RA),它做的事情就是start的时候以写的方式打开这个文件,每次的monitor操作就是往文件里写一个值再读出来。把这个RA资源作为其他资源的依赖资源,万一发生脑裂谁抢到这个资源谁就是主节点。
这个方法有个问题,就是这个资源成了单点故障点,万一文件服务器挂了,HA集群也起不来了。
(当然,如果HA的业务本身天生有一个只可能被一个节点获得的资源,就皆大欢喜了。)


下面验证一下Pacemaker在没有法定投票没有抢占资源的情况下会怎么样?发生脑裂时会不会导致一个共享资源在2个节点上同时被加载?

1. 环境构成

沿用前一篇>(http://blog.chinaunix.net/uid-20726500-id-4453488.html)的环境,但改成双节点集群。


共享存储服务器
OS:CentOS release 6.5 (Final)
主机名:disknode
网卡1:
  保留
  网卡类型:NAT
  IP:192.168.152.120
网卡2:
  用于共享盘的iscsi通信
  网卡类型:Host-Only
  IP:192.168.146.120
网卡3:
  用于external/ssh fence设备通信
  网卡类型:桥接
  IP:10.167.217.107


HA节点1
OS:CentOS release 6.5 (Final)
主机名:hanode1
网卡1:
  用于集群公开IP(192.168.152.200)和集群内部消息的通信
  网卡类型:NAT
  IP:192.168.152.130
网卡2:
  用于共享盘的iscsi通信
  网卡类型:Host-Only
  IP:192.168.146.130
网卡3:
  用于external/ssh fence设备通信
  网卡类型:桥接
  IP:10.167.217.169


HA节点2
OS:CentOS release 6.5 (Final)
主机名:hanode2
网卡1:
  用于集群公开IP(192.168.152.200)和集群内部消息的通信
  网卡类型:NAT
  IP:192.168.152.140
网卡2:
  用于共享盘的iscsi通信
  网卡类型:Host-Only
  IP:192.168.146.140
网卡3:
  用于external/ssh fence设备通信
  网卡类型:桥接
  IP:10.167.217.171


集群公开IP
  192.168.152.200


2. 环境配置

在原来已配好的3节点的环境上,把disknode从集群管理里移除。

修改配置使达不到法定票数时的动作为忽略
no-quorum-policy=ignore

并修改法定票数
expected-quorum-votes=2

将disknode相关的配置删掉
location no_iscsid rs_iscsid -inf: disknode
location votenode ClusterIP -inf: disknode

[root@hanode1 ~]# crm configure edit
node disknode
node hanode1
node hanode2
primitive ClusterIP IPaddr2 \
        params ip=192.168.152.200 cidr_netmask=32 \
        op monitor interval=30s
primitive DataFS Filesystem \
        params device="/dev/sdc" directory="/mnt/pg" fstype=ext4 \
        op monitor interval=15s
primitive pg93 pgsql \
        meta target-role=Started is-managed=true migration-threshold=INFINITY failure-timeout=60s \
        op monitor interval=15s
primitive rs_iscsid lsb:iscsid \
        op monitor interval=30s \
        meta target-role=Started
primitive st-ssh stonith:external/ssh \
        params hostlist="hanode1 hanode2"
group PgGroup ClusterIP rs_iscsid DataFS pg93
clone st-sshclone st-ssh
property cib-bootstrap-options: \
        dc-version=1.1.9-2a917dd \
        cluster-infrastructure="classic openais (with plugin)" \
        expected-quorum-votes=2 \
        stonith-enabled=true \
        no-quorum-policy=ignore \
        last-lrm-refresh=1409756808
#vim:set syntax=pcmk


在disknode上停掉corosync服务
[root@disknode ~]# /etc/init.d/corosync stop 
Signaling Corosync Cluster Engine (corosync) to terminate: [  OK  ]
Waiting for corosync services to unload:.                  [  OK  ]
[root@disknode ~]# chkconfig corosync off


进入剩下的其中一个节点hanode1的终端,从CIB中移除disknode
[root@hanode1 ~]# crm_node -R disknode --force

结果很奇怪,它居然把hanode2删掉了,多试几次还遇到把hanode2重启的情况。
[root@hanode1 ~]# crm status
Last updated: Fri Sep  5 23:48:50 2014
Last change: Fri Sep  5 23:47:50 2014 via crm_node on hanode1
Stack: classic openais (with plugin)
Current DC: hanode1 - partition with quorum
Version: 1.1.9-2a917dd
2 Nodes configured, 2 expected votes
6 Resources configured.




Node disknode: UNCLEAN (offline)
Online: [ hanode1 ]


 Resource Group: PgGroup
     ClusterIP (ocf::heartbeat:IPaddr2): Started hanode1 
     rs_iscsid (lsb:iscsid): Started hanode1 
     DataFS (ocf::heartbeat:Filesystem): Started hanode1 
     pg93 (ocf::heartbeat:pgsql): Started hanode1 
 Clone Set: st-sshclone [st-ssh]
     Started: [ hanode1 ]
     Stopped: [ st-ssh:1 ]


/var/log/messages日志里有类似这样的错误消息
Sep  5 21:27:53 hanode1 corosync[3827]:   [pcmk  ] info: pcmk_remove_member: Sent: remove-peer:disknode
Sep  5 21:27:53 hanode1 corosync[3827]:   [pcmk  ] ERROR: ais_get_int: Characters left over after parsing 'disknode': 'disknode'


再执行一次,这回把disknode删掉了
[root@hanode1 ~]# crm_node -R disknode --force
[root@hanode1 ~]# crm status
Last updated: Fri Sep  5 23:50:16 2014
Last change: Fri Sep  5 23:50:14 2014 via crm_node on hanode1
Stack: classic openais (with plugin)
Current DC: hanode1 - partition with quorum
Version: 1.1.9-2a917dd
1 Nodes configured, 2 expected votes
5 Resources configured.




Online: [ hanode1 ]


 Resource Group: PgGroup
     ClusterIP (ocf::heartbeat:IPaddr2): Started hanode1 
     rs_iscsid (lsb:iscsid): Started hanode1 
     DataFS (ocf::heartbeat:Filesystem): Started hanode1 
     pg93 (ocf::heartbeat:pgsql): Started hanode1 
 Clone Set: st-sshclone [st-ssh]
     Started: [ hanode1 ]


再重启hanode2服务器,状态终于正常了。
[root@hanode1 ~]# crm status
Last updated: Sat Sep  6 00:01:16 2014
Last change: Fri Sep  5 23:57:54 2014 via crmd on hanode1
Stack: classic openais (with plugin)
Current DC: hanode1 - partition with quorum
Version: 1.1.9-2a917dd
2 Nodes configured, 2 expected votes
6 Resources configured.




Online: [ hanode1 hanode2 ]


 Resource Group: PgGroup
     ClusterIP (ocf::heartbeat:IPaddr2): Started hanode1 
     rs_iscsid (lsb:iscsid): Started hanode1 
     DataFS (ocf::heartbeat:Filesystem): Started hanode1 
     pg93 (ocf::heartbeat:pgsql): Started hanode1 
 Clone Set: st-sshclone [st-ssh]
     Started: [ hanode1 hanode2 ]

*)本来只是想重启corosync服务,但是停不掉corosync服务,于是杀掉corosync进程,正准备启动corosync服务,发现hanode2被fencing掉了。


3. 切换测试

3.1 场景1:主服务器的corosync进程down

杀掉主服务器的corosync进程
[root@hanode1 ~]# ps -ef|grep corosync
root      1355     1  0 00:00 ?        00:00:02 corosync
root      4606  2103  0 00:18 pts/0    00:00:00 grep corosync
[root@hanode1 ~]# kill -9 1355


马上发现hanode1被重启,而后hanode2接管服务。
[root@hanode2 ~]# crm status
Last updated: Sat Sep  6 00:22:45 2014
Last change: Fri Sep  5 23:57:54 2014 via crmd on hanode1
Stack: classic openais (with plugin)
Current DC: hanode2 - partition with quorum
Version: 1.1.9-2a917dd
2 Nodes configured, 2 expected votes
6 Resources configured.




Online: [ hanode1 hanode2 ]


 Resource Group: PgGroup
     ClusterIP (ocf::heartbeat:IPaddr2): Started hanode2 
     rs_iscsid (lsb:iscsid): Started hanode2 
     DataFS (ocf::heartbeat:Filesystem): Started hanode2 
     pg93 (ocf::heartbeat:pgsql): Started hanode2 
 Clone Set: st-sshclone [st-ssh]
     Started: [ hanode1 hanode2 ]


查看hanode2上的日志,也发现hanode2在fencing成功后再接管资源,保证了资源不会被2个节点同时拥有。
[root@hanode2 ~]# vi /var/log/messages
Sep  6 00:15:08 hanode2 stonith-ng[1362]:   notice: initiate_remote_stonith_op: Initiating remote operation reboot for hanode1: 3daae4d8-f3f5-47bb-ac32-1a7106099eca (0)
Sep  6 00:15:13 hanode2 stonith-ng[1362]:   notice: log_operation: Operation 'reboot' [2149] (call 0 from crmd.1366) for host 'hanode1' with device 'st-ssh' returned: 0 (OK)
Sep  6 00:15:13 hanode2 stonith-ng[1362]:   notice: remote_op_done: Operation reboot of hanode1 by hanode2 for crmd.1366@hanode2.3daae4d8: OK
...
Sep  6 00:15:14 hanode2 pengine[1365]:   notice: LogActions: Start   rs_iscsid#011(hanode2)
Sep  6 00:15:14 hanode2 pengine[1365]:   notice: LogActions: Start   DataFS#011(hanode2)
Sep  6 00:15:14 hanode2 pengine[1365]:   notice: LogActions: Start   pg93#011(hanode2)


3.1 场景2:主备服务间的心跳网络断开

在VMWare上禁用主服务器hanode2的心跳网卡。好戏来了,hanode1和hanode2同时干掉了对方。
hanode1和hanode2启动后,hanode1手快了一点,hanode2被杀掉。hanode2起来后又把hanode1干掉...
看来这种场景下,就是2个节点互砍,谁也成不了赢家。


4. 脑裂的改进

前面的场景2实际是心跳的网络成了单点故障,可以在心跳网络上引入网络设备的冗余,提高心跳网络的稳定性。除此以外还没有别的方法呢。设想有2种方法:


4.1 方法1

把hanode1和hanode2的心跳线接到同一个路由器上,并用这个路由器的ip地址作为pingnode。用这个pingnode做仲裁。
下面看看这个方法有没效。路由器的ip为192.168.152.2


[root@hanode1 ~]# crm configure edit


node hanode1
node hanode2
primitive ClusterIP IPaddr2 \
        params ip=192.168.152.200 cidr_netmask=32 \
        op monitor interval=30s
primitive DataFS Filesystem \
        params device="/dev/sdc" directory="/mnt/pg" fstype=ext4 \
        op monitor interval=15s
primitive pg93 pgsql \
        meta target-role=Started is-managed=true migration-threshold=INFINITY failure-timeout=60s \
        op monitor interval=15s
primitive pingCheck ocf:pacemaker:ping \
        params name=default_ping_set host_list=192.168.152.2 multiplier=100 \
        op start timeout=60s interval=0s on-fail=restart \
        op monitor timeout=60s interval=10s on-fail=restart \
        op stop timeout=60s interval=0s on-fail=ignore
primitive rs_iscsid lsb:iscsid \
        op monitor interval=30s \
        meta target-role=Started
primitive st-ssh stonith:external/ssh \
        params hostlist="hanode1 hanode2"
group PgGroup ClusterIP rs_iscsid DataFS pg93
clone clnPingCheck pingCheck
clone st-sshclone st-ssh
location rsc_location PgGroup \
        rule $id="rsc_location-rule" -inf: not_defined default_ping_set or default_ping_set lt 100
order rsc_orderi 0: clnPingCheck PgGroup
property cib-bootstrap-options: \
        dc-version=1.1.9-2a917dd \
        cluster-infrastructure="classic openais (with plugin)" \
        expected-quorum-votes=2 \
        stonith-enabled=true \
        no-quorum-policy=ignore \
        last-lrm-refresh=1409756808
#vim:set syntax=pcmk


修改后把2个节点的corosync服务重启一下,过一会状态更新。
[root@hanode1 ~]# crm_mon -Afr1
Last updated: Sat Sep  6 01:36:57 2014
Last change: Sat Sep  6 01:36:08 2014 via cibadmin on hanode1
Stack: classic openais (with plugin)
Current DC: hanode1 - partition with quorum
Version: 1.1.9-2a917dd
2 Nodes configured, 2 expected votes
8 Resources configured.




Online: [ hanode1 hanode2 ]


Full list of resources:


 Resource Group: PgGroup
     ClusterIP (ocf::heartbeat:IPaddr2): Started hanode1 
     rs_iscsid (lsb:iscsid): Started hanode1 
     DataFS (ocf::heartbeat:Filesystem): Started hanode1 
     pg93 (ocf::heartbeat:pgsql): Started hanode1 
 Clone Set: st-sshclone [st-ssh]
     Started: [ hanode1 hanode2 ]
 Clone Set: clnPingCheck [pingCheck]
     Started: [ hanode1 hanode2 ]


Node Attributes:
* Node hanode1:
    + default_ping_set                 : 100       
* Node hanode2:
    + default_ping_set                 : 100       


Migration summary:
* Node hanode2: 
* Node hanode1: 


现在把主服务的心跳网卡禁掉。结果和以前一样,2个机器还是互杀。原因在于fencing机制在pingCheck之前动作,pingCheck仅仅可以影响资源的位置。看来这个方法不行。

4.1 方法2

把stonith-action 的动作从reboot改为off,这样手快的那一方有可能成为赢家。
[root@hanode1 ~]# crm_attribute --attr-name stonith-action  --attr-value off


这里用的测试用的stonith设备external/ssh是不支持poweroff的,为了测试把external/ssh脚本改了一下,最终修改过的external/ssh脚本见附录。
再把前面加的pingCheck去掉,再试一次,把主服务的心跳网卡禁掉。很不幸2个机器都关机了。
看了下external/ssh的脚本,关机前sleep了2秒,把这个sleep去掉再试。这次终于有1个幸存了。


POWEROFF_COMMAND="echo 'sleep 2; /sbin/poweroff -nf' | SHELL=/bin/sh at now >/dev/null 2>&1"
==》
POWEROFF_COMMAND="echo '/sbin/poweroff -nf' | SHELL=/bin/sh at now >/dev/null 2>&1"


5. 结论

在双节点的情况下,只要有fencing设备就可以确保共享资源不被破坏了。如果没有fencing设备,就必须要配置抢占资源。当心跳网络出现故障时没法保障双机集群依然可用,但通过将stonith-action 的设置为off,可在很大概率上使得脑裂时有1台机器还活着。


6. 附录

修改过的 external/ssh脚本
[root@hanode1 ~]# cat /usr/lib64/stonith/plugins/external/ssh

点击(此处)折叠或打开

  1. #!/bin/sh
  2. #
  3. # External STONITH module for ssh.
  4. #
  5. # Copyright (c) 2004 SUSE LINUX AG - Lars Marowsky-Bree
  6. #
  7. # This program is free software; you can redistribute it and/or modify
  8. # it under the terms of version 2 of the GNU General Public License as
  9. # published by the Free Software Foundation.
  10. #
  11. # This program is distributed in the hope that it would be useful, but
  12. # WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  14. #
  15. # Further, this software is distributed without any warranty that it is
  16. # free of the rightful claim of any third person regarding infringement
  17. # or the like. Any license provided herein, whether implied or
  18. # otherwise, applies only to this software file. Patent licenses, if
  19. # any, provided herein do not apply to combinations of this program with
  20. # other software, or any other product whatsoever.
  21. #
  22. # You should have received a copy of the GNU General Public License
  23. # along with this program; if not, write the Free Software Foundation,
  24. # Inc., 59 Temple Place - Suite 330, Boston MA 02111-1307, USA.
  25. #
  26. SSH_COMMAND="/usr/bin/ssh -q -x -o PasswordAuthentication=no -o StrictHostKeyChecking=no -n -l root"
  27. #SSH_COMMAND="/usr/bin/ssh -q -x -n -l root"
  28. REBOOT_COMMAND="echo 'sleep 2; /sbin/reboot -nf' | SHELL=/bin/sh at now >/dev/null 2>&1"
  29. # Warning: If you select this poweroff command, it'll physically
  30. # power-off the machine, and quite a number of systems won't be remotely
  31. # revivable.
  32. # TODO: Probably should touch a file on the server instead to just
  33. # prevent heartbeat et al from being started after the reboot.
  34. POWEROFF_COMMAND="echo '/sbin/poweroff -nf' | SHELL=/bin/sh at now >/dev/null 2>&1"
  35. #POWEROFF_COMMAND="echo 'sleep 2; /sbin/reboot -nf' | SHELL=/bin/sh at now >/dev/null 2>&1"
  36. # Rewrite the hostlist to accept "," as a delimeter for hostnames too.
  37. hostlist=`echo $hostlist | tr ',' ' '`
  38. is_host_up() {
  39. for j in 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
  40. do
  41. if
  42. ping -w1 -c1 "$1" >/dev/null 2>&1
  43. then
  44. sleep 1
  45. else
  46. return 1
  47. fi
  48. done
  49. return 0
  50. }
  51. echo hostlist="$hostlist" para="$*" >>/var/stonith_ssh.log
  52. case $1 in
  53. gethosts)
  54. for h in $hostlist ; do
  55. echo $h
  56. done
  57. exit 0
  58. ;;
  59. on)
  60. # Can't really be implemented because ssh cannot power on a system
  61. # when it is powered off.
  62. exit 1
  63. ;;
  64. off)
  65. # Shouldn't really be implemented because if ssh cannot power on a
  66. # system, it shouldn't be allowed to power it off.
  67. # exit 1
  68. # ;;
  69. h_target=`echo $2 | tr A-Z a-z`
  70. for h in $hostlist
  71. do
  72. h=`echo $h | tr A-Z a-z`
  73. [ "$h" != "$h_target" ] &&
  74. continue
  75. if
  76. case ${livedangerously} in
  77. [Yy]*) is_host_up $h;;
  78. *) true;;
  79. esac
  80. then
  81. $SSH_COMMAND "$2" "$POWEROFF_COMMAND"
  82. # Good thing this is only for testing...
  83. if
  84. is_host_up $h
  85. then
  86. exit 1
  87. else
  88. exit 0
  89. fi
  90. else
  91. # well... Let's call it successful, after all this is only for testing...
  92. exit 0
  93. fi
  94. done
  95. exit 1
  96. ;;
  97. reset)
  98. h_target=`echo $2 | tr A-Z a-z`
  99. for h in $hostlist
  100. do
  101. h=`echo $h | tr A-Z a-z`
  102. [ "$h" != "$h_target" ] &&
  103. continue
  104. if
  105. case ${livedangerously} in
  106. [Yy]*) is_host_up $h;;
  107. *) true;;
  108. esac
  109. then
  110. $SSH_COMMAND "$2" "$REBOOT_COMMAND"
  111. # Good thing this is only for testing...
  112. if
  113. is_host_up $h
  114. then
  115. exit 1
  116. else
  117. exit 0
  118. fi
  119. else
  120. # well... Let's call it successful, after all this is only for testing...
  121. exit 0
  122. fi
  123. done
  124. exit 1
  125. ;;
  126. status)
  127. if
  128. [ -z "$hostlist" ]
  129. then
  130. exit 1
  131. fi
  132. for h in $hostlist
  133. do
  134. if
  135. ping -w1 -c1 "$h" 2>&1 | grep "unknown host"
  136. then
  137. exit 1
  138. fi
  139. done
  140. exit 0
  141. ;;
  142. getconfignames)
  143. echo "hostlist"
  144. exit 0
  145. ;;
  146. getinfo-devid)
  147. echo "ssh STONITH device"
  148. exit 0
  149. ;;
  150. getinfo-devname)
  151. echo "ssh STONITH external device"
  152. exit 0
  153. ;;
  154. getinfo-devdescr)
  155. echo "ssh-based host reset"
  156. echo "Fine for testing, but not suitable for production!"
  157. echo "Only reboot action supported, no poweroff, and, surprisingly enough, no poweron."
  158. exit 0
  159. ;;
  160. getinfo-devurl)
  161. echo "http://openssh.org"
  162. exit 0
  163. ;;
  164. getinfo-xml)
  165. cat
  166. Hostlist
  167. The list of hosts that the STONITH device controls
  168. Live Dangerously!!
  169. Set to "yes" if you want to risk your system's integrity.
  170. Of course, since this plugin isn't for production, using it
  171. in production at all is a bad idea. On the other hand,
  172. setting this parameter to yes makes it an even worse idea.
  173. Viva la Vida Loca!
  174. SSHXML
  175. exit 0
  176. ;;
  177. *)
  178. exit 1
  179. ;;
  180. esac


相关文章
|
Kubernetes 网络协议 网络安全
安装k8s Master高可用集群
安装k8s Master高可用集群 主机 角色 组件 172.18.6.101 K8S Master Kubelet,kubectl,cni,etcd 172.18.6.102 K8S Master Kubelet,kubectl,cni,etcd 172.
3590 0