我們下面來實(shí)現(xiàn)一個架構(gòu),heartbeat+drbd+nfs實(shí)現(xiàn)mysql和網(wǎng)站數(shù)據(jù)的同步,keepalived實(shí)現(xiàn)nginx的高可用,而用nginx和dns輪詢實(shí)現(xiàn)負(fù)載均衡。
架構(gòu)說明
目錄規(guī)劃
/usr/local/src/lnmp:用來存放源碼工具等等
/data:用來存放所有數(shù)據(jù)和NFS以及DRBD的掛載
/data/shell:用來存放所有管理腳本
/data/mysql:用來掛載DRBD的mysql資源,以供mysql存放數(shù)據(jù)庫
/data/wwwnfs:用來掛載DRBD生成的www資源,以供兩個節(jié)點(diǎn)掛載到各個節(jié)點(diǎn)的/data/www目錄,以供論壇等程序數(shù)據(jù)使用
/data/www:用來掛載NFS資源,用來存放論壇(網(wǎng)站)等程序數(shù)據(jù)
拓?fù)涔ぷ髟?/h4>
內(nèi)網(wǎng):
1,DRBD網(wǎng)絡(luò)存儲創(chuàng)建出兩個資源,一個mysql給mysql數(shù)據(jù)庫同步用,一個www給web(論壇)數(shù)據(jù)NFS共享掛載用,虛擬出兩個虛擬IP,一個是 192.168.1.100,用來連接數(shù)據(jù)庫,一個是192.168.1.200,用來給節(jié)點(diǎn)掛載NFS
注意:NFS底下掛載了三次:DRBD掛載一次,文件系統(tǒng)掛載一次,客戶端掛載一次
2,Heartbeat來實(shí)現(xiàn)DRBD的HA,同時虛擬出兩個內(nèi)網(wǎng)IP,并管理NFS,MySQL的啟動和關(guān)閉
外網(wǎng):
1,兩個節(jié)點(diǎn)都用Nginx做均衡器,通過內(nèi)網(wǎng)調(diào)度負(fù)載兩個節(jié)點(diǎn),實(shí)現(xiàn)內(nèi)部均衡
2,DNS配置雙IP對應(yīng)一個域名的方式來實(shí)現(xiàn)DNS輪詢,實(shí)現(xiàn)外網(wǎng)均衡
3,Keepalived使用雙主(master)配置虛擬出兩個虛擬IP:節(jié)點(diǎn)一 12.12.12.100和節(jié)點(diǎn)二 12.12.12.200,同時共外網(wǎng)訪問,兩個節(jié)點(diǎn)互為主從關(guān)系,當(dāng)某個節(jié)點(diǎn)掛掉的時候,另外一個節(jié)點(diǎn)將同時是兩個資源的master,同時擁有兩個虛 擬IP,實(shí)現(xiàn)資源轉(zhuǎn)移。
我們知道DNS的缺點(diǎn)就是生效慢,分配資源不合理,理論上有可能把所有的請求都發(fā)送給同一節(jié)點(diǎn),導(dǎo)致均衡不合理導(dǎo)致所有資源不可用,這里我們由于有 了NGINX內(nèi)部負(fù)載,就不怕DNS輪詢不均衡了,因?yàn)镹GINX內(nèi)部有嚴(yán)謹(jǐn)?shù)恼{(diào)度方式,不管那臺請求有多少,在內(nèi)部都能實(shí)現(xiàn)理想的調(diào)度,這樣就能把 DNS負(fù)載均衡和NGINX完美結(jié)合,是硬件資源得到合理的利用,然后利用keepalive保證了每個節(jié)點(diǎn)的可靠性,幾乎完美!
拓?fù)鋱D如下:
架構(gòu)實(shí)現(xiàn)
LNMP架構(gòu)配置
配置LNMp架構(gòu)需要注意兩點(diǎn):
注意一:這里MYSQL都不要初始化,不要啟動!后面有專門的配置的
注意二:nginx所有端口都改成 8080,因?yàn)橐粫€要安裝nginx來做均衡器并對外提供服務(wù),所以不要用默認(rèn)的80
注意三、nginx和php-fpm運(yùn)行的用戶都是www。
安裝配置NFS
1、安裝NFS
- yum install nfs-utils nfs4-acl-tools portmap
2、配置/etc/exports
- /data/wwwnfs 192.168.1.0/24(rw,,no_root_squash,sync,anonuid=502,anongid=502)
注意:
/data/wwwnfs:就是給兩個節(jié)點(diǎn)掛載的目錄,所有網(wǎng)站程序都放在這里,實(shí)現(xiàn)論壇程序等數(shù)據(jù)的共享(同步)
anonuid=502,anongid=502:這個表示客戶端上任何用戶進(jìn)入到掛載目錄都以uid=502和gid=502身份,我這里這個代表的是www用戶
3、啟動
- service portmap start
- service nfs start
切忌,必須先啟動portmap
- chkconfig nfs off
- chkconfig portmap on
注意:portmap服務(wù)器必須常駐,且不收heartbeat管理;而nfs這必須要用heartbeat來管理他的啟動和關(guān)閉,所以這里要關(guān)閉nfs開機(jī)自動啟動
同時要啟動鎖機(jī)制,因?yàn)橥瑫r有兩個節(jié)點(diǎn)要使用同一份數(shù)據(jù),所以需要有總裁,這個尤其是在NFS給mysql用的時候是必須要用的,對于論壇或網(wǎng)站, 要看情況,如果存在對同一文件同時修改的時候必須要啟動NFS鎖機(jī)制,如果沒有這種情況,那么建議不要啟動,啟動了會降低NFS的性能:
- /sbin/rpc.lockd
- echo “/sbin/rpc.lockd” >>/etc/rc.local
4、開機(jī)自動掛載
- echo “sleep 20″ >>/etc/rc.local
- echo “/bin/mount -t nfs 192.168.1.200:/data/wwwnfs /data/www” >>/etc/rc.local
為什么為延遲20秒再掛載nfs?因?yàn)槿绻坏却⒓磼燧d,會發(fā)現(xiàn)掛載不上,這是由于heartbeat啟動用的vip還沒設(shè)置好的原因。
立即掛載:
- mount -a
安裝配置DRBD
安裝方法見:http://www./2012/02/drbd-compile-install-deploy/
配置文件
DRBD有三種配置文件:
/usr/local/drbd/etc/drbd.conf
/usr/local/drbd/etc/drbd.d/global_common.conf
/usr/local/drbd/etc/drbd.d/*.res
1、drbd.conf
- include “drbd.d/global_common.conf”;
- include “drbd.d/*.res”;
2、global_common.conf
- global {
- usage-count yes;
- }
- common {
- net {
- protocol C;
- }
- }
3、mysql.res和www.res
mysql.res:
- vi /usr/local/drbd/etc/drbd.d/mysql.res
- #資源組的名稱
- resource mysql{
- #定義主服務(wù)器資源
- on node1{
- #建立塊設(shè)備文件
- device /dev/drbd1;
- #要用于復(fù)制的分區(qū)
- disk /dev/sdb1;
- #定義偵聽IP和端口
- address 192.168.1.10:7788;
- #meta data信息存放的方式,這里為內(nèi)部存儲,即和真實(shí)數(shù)據(jù)放在一起存儲
- meta-disk internal;
- }
- #定義備服務(wù)器資源
- on node2{
- device /dev/drbd1;
- disk /dev/sdb1;
- address 192.168.1.20:7788;
- meta-disk internal;
- }
- }
www.res:
- vi /usr/local/drbd/etc/drbd.d/www.res
- #資源組的名稱
- resource www{
- #定義主服務(wù)器資源
- on node2{
- #建立塊設(shè)備文件
- device /dev/drbd2;
- #要用于復(fù)制的分區(qū)
- disk /dev/sdb2;
- #定義偵聽IP和端口
- address 192.168.1.20:7789;
- #meta data信息存放的方式,這里為內(nèi)部存儲,即和真實(shí)數(shù)據(jù)放在一起存儲
- meta-disk internal;
- }
- #定義備服務(wù)器資源
- on node1{
- device /dev/drbd2;
- disk /dev/sdb2;
- address 192.168.1.10:7789;
- meta-disk internal;
- }
- }
最后復(fù)制這些文件到node2。
初始化DRBD資源
1)在各個節(jié)點(diǎn)啟用資源mysql和www
- modprobe drbd
- dd if=/dev/zero of=/dev/sdb1 bs=1M count=10
- dd if=/dev/zero of=/dev/sdb2 bs=1M count=10
- drbdadm create-md mysql
- drbdadm create-md www
- drbdadm up mysql
- drbdadm up www
2),提升各個節(jié)點(diǎn)上的主
在node1上:
- drbdadm primary –force mysql
在node2上:
- drbdadm primary –force www
3)格式化drbd塊設(shè)備
在node1上
- mkfs.ext3 /dev/drbd1
在node2上
- mkfs.ext3 /dev/drbd2
4)掛載分區(qū)
在node1上
- mount /dev/drbd1 /data/mysql
在node2上
- mount /dev/drbd2 /data/wwwfs
安裝配置heartbeat
1、安裝heartbeat
- yum install heartbeat
安裝完后會自動建立用戶hacluster和組haclient
確保兩個節(jié)點(diǎn)上hacluster用戶的的UID和GID相同
2、同步兩臺節(jié)點(diǎn)的時間
- rm -rf /etc/localtime
- \cp -f /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
- yum install -y ntp
- ntpdate -d cn.pool.ntp.org
3、配置/etc/ha.d/ha.cf
- debugfile /var/log/ha-debug #打開錯誤日志報(bào)告
- keepalive 2 #兩秒檢測一次心跳線連接
- deadtime 10 #10 秒測試不到主服務(wù)器心跳線為有問題出現(xiàn)
- warntime 6 #警告時間(最好在 2 ~ 10 之間)
- initdead 120 #初始化啟動時 120 秒無連接視為正常,或指定heartbeat
- #在啟動時,需要等待120秒才去啟動任何資源。
- udpport 694 #用 udp 的 694 端口連接
- ucast eth0 192.168.1.20 #單播方式連接(主從都寫對方的 ip 進(jìn)行連接)
- node node1 #聲明主服(注意是主機(jī)名uname -n不是域名)
- node node2 #聲明備服(注意是主機(jī)名uname -n不是域名)
- auto_failback on #自動切換(主服恢復(fù)后可自動切換回來)這個不要開啟
- respawn hacluster /usr/lib/heartbeat/ipfail #監(jiān)控ipfail進(jìn)程是否掛掉,如果掛掉就重啟它
4、/etc/ha.d/authkeys
- auth 1
- 1 crc
5、/etc/ha.d/haresources
- node1 IPaddr::192.168.1.100/24/eth0 drbddisk::mysql Filesystem::/dev/drbd1::/data/mysql::ext3 mysqld portmap
- node2 IPaddr::192.168.1.200/24/eth0 drbddisk::www Filesystem::/dev/drbd2::/data/wwwnfs::ext3 portmap nfs
6、創(chuàng)建nfs管理腳本
- vi /etc/ha.d/resource.d/nfs
寫入:
- #!/bin/bash
- NFSD=/etc/rc.d/init.d/nfs
- NFSDPID=`/sbin/pidof nfsd`
- case $1 in
- start)
- $NFSD start;
- ;;
- stop)
- $NFSD stop;
- if [ "$NFSDPID" != " " ];then
- for NFSPID in $NFSDPID
- do /bin/kill -9 $NFSPID;
- done
- fi
- ;;
- *)
- echo “Syntax incorrect. You need one of {start|stop }”
- ;;
- esac
先啟動node1的heartbeat,再啟動node2的heartbeat
啟動成功后,這里有幾項(xiàng)需要檢查
node1:
1、執(zhí)行ip a,檢查是否已經(jīng)設(shè)置有虛擬ip 192.168.1.100
2、執(zhí)行cat /proc/drbd檢查狀態(tài)是否正常
3、執(zhí)行df -h查看/dev/drbd1是否已經(jīng)掛載到/data/mysql
4、執(zhí)行service mysqld status查看mysql是否已經(jīng)啟動
node2:
1、執(zhí)行ip a查看是否已經(jīng)設(shè)置虛擬ip 192.168.1.200
2、執(zhí)行cat /proc/drbd檢查狀態(tài)是否正常
3、執(zhí)行df -h查看/dev/drbd2是否已經(jīng)掛載到/data/wwwnfs和192.168.1.200:/data/wwwnfs是否已經(jīng)掛載到/data/www
nginx均衡器配置
- user www;
- worker_processes 1;
- error_log /var/log/nginx/error.log warn;
- pid /var/run/nginx.pid;
- events {
- worker_connections 1024;
- }
- http {
- include /etc/nginx/mime.types;
- default_type application/octet-stream;
- log_format main ‘$remote_addr – $remote_user [$time_local] “$request” ‘
- ’$status $body_bytes_sent “$http_referer” ‘
- ’”$http_user_agent” “$http_x_forwarded_for”‘;
- access_log /var/log/nginx/access.log main;
- sendfile on;
- #tcp_nopush on;
- keepalive_timeout 65;
- #gzip on;
- upstream www._server
- {
- server 192.168.1.10:8080 weight=3 max_fails=2 fail_timeout=30s;
- server 192.168.1.20:8080 weight=9 max_fails=2 fail_timeout=30s;
- }
- server
- {
- listen 80;
- server_name www.;
- location / {
- root /data/www/www.;
- index index.php index.htm index.html;
- proxy_redirect off;
- proxy_set_header Host $host;
- proxy_set_header X-Real-IP $remote_addr;
- proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
- proxy_pass http://www._server;
- }
- access_log off;
- }
- server
- {
- listen 8080;
- server_name www.;
- index index.html index.htm index.php;
- root /data/www/www.;
- #limit_conn crawler 20;
- location ~ \.php$ {
- root /data/www/www.;
- fastcgi_pass 127.0.0.1:9000;
- fastcgi_index index.php;
- fastcgi_param SCRIPT_FILENAME /data/www/www./$fastcgi_script_name;
- include fastcgi_params;
- }
- location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$
- {
- expires 30d;
- }
- location ~ .*\.(js|css)?$
- {
- expires 1h;
- }
- access_log off;
- }
- }
這里定義了兩臺用于負(fù)載均衡的機(jī)子,分別是192.168.1.10:8080和192.168.1.20:8080,通過proxy_pass http://www._server代理循詢轉(zhuǎn)發(fā)到這兩臺機(jī),達(dá)到負(fù)載均衡的作用。
你可以建立index.php,里面寫入:
- <?php
- echo $_SERVER['SERVER_ADDR'];
- ?>
如果連續(xù)刷新幾次,得到不同的IP,證明已經(jīng)均衡負(fù)載到不同的服務(wù)器。
Keepalived實(shí)現(xiàn)nginx和php的HA
1、keepalived安裝
安裝方法見:http://www./2012/02/nginx-keepalived-high-availability/
2、配置
節(jié)點(diǎn)一node1配置如下:
- global_defs {
- notification_email {
- admin@
- }
- notification_email_from keepalived@domain.com
- smtp_server 127.0.0.1
- smtp_connect_timeout 30
- router_id LVS_DEVEL
- }
- vrrp_instance VI_1 {
- state MASTER ############ 輔機(jī)為 BACKUP
- interface eth0
- virtual_router_id 100
- mcast_src_ip 192.168.1.10 ########### 本機(jī)IP
- priority 102 ########### 權(quán)值要比 back 高
- advert_int 1
- authentication {
- auth_type PASS
- auth_pass 1111
- }
- virtual_ipaddress {
- 12.12.12.100
- }
- }
- vrrp_instance VI_1 {
- state BACKUP
- interface eth0
- virtual_router_id 200
- mcast_src_ip 192.168.1.101 ########### 本機(jī)IP
- priority 101 ##########權(quán)值 要比 master 低。。
- advert_int 1
- authentication {
- auth_type PASS
- auth_pass 1111
- }
- virtual_ipaddress {
- 12.12.12.200
- }
- }
節(jié)點(diǎn)二配置:
- global_defs {
- notification_email {
- admin@
- }
- notification_email_from keepalived@domain.com
- smtp_server 127.0.0.1
- smtp_connect_timeout 30
- router_id LVS_DEVEL
- }
- vrrp_instance VI_1 {
- state BACKUP
- interface eth0
- virtual_router_id 100
- mcast_src_ip 192.168.1.20 ########### 本機(jī)IP
- priority 101 ##########權(quán)值 要比 master 低。。
- advert_int 1
- authentication {
- auth_type PASS
- auth_pass 1111
- }
- virtual_ipaddress {
- 12.12.12.100
- }
- }
- vrrp_instance VI_1 {
- state MASTER ############ 輔機(jī)為 BACKUP
- interface eth0
- virtual_router_id 200
- mcast_src_ip 192.168.1.103 ########### 本機(jī)IP
- priority 102 ########### 權(quán)值要比 back 高
- advert_int 1
- authentication {
- auth_type PASS
- auth_pass 1111
- }
- virtual_ipaddress {
- 12.12.12.200
- }
- }
3、創(chuàng)建監(jiān)控腳本
node1監(jiān)控腳本:
- vi /opt/check.sh
- #!/bin/bash
- while :
- do
- mysqlcheck=`/usr/bin/mysqladmin -uroot ping 2>&1`
- mysqlcode=`echo $?`
- heartbeat=`ps -C heartbeat –no-header | wc -l`
- if [ $mysqlcode -ne 0 ] ;then
- if [ $heartbeat-ne 0 ];then
- service heartbeat stop
- fi
- fi
- phpcheck=`ps -C php-fpm –no-header | wc -l`
- nginxcheck=`ps -C nginx –no-header | wc -l`
- keepalivedcheck=`ps -C keepalived –no-header | wc -l`
- if [ $nginxcheck -eq 0 ]|| [ $phpcheck -eq 0 ];then
- if [ $keepalivedcheck -ne 0 ];then
- killall -TERM keepalived
- else
- echo “keepalived is stoped”
- fi
- else
- if [ $keepalivedcheck -eq 0 ];then
- /etc/init.d/keepalived start
- else
- echo “keepalived is running”
- fi
- fi
- sleep 5
- done
node2監(jiān)控腳本:
- #!/bin/bash
- while :
- do
- phpcheck=`ps -C php-cgi –no-header | wc -l`
- nginxcheck=`ps -C nginx –no-header | wc -l`
- keepalivedcheck=`ps -C keepalived –no-header | wc -l`
- if [ $nginxcheck -eq 0 ]|| [ $phpcheck -eq 0 ];then
- if [ $keepalivedcheck -ne 0 ];then
- killall -TERM keepalived
- else
- echo “keepalived is stoped”
- fi
- else
- if [ $keepalivedcheck -eq 0 ];then
- /etc/init.d/keepalived start
- else
- echo “keepalived is running”
- fi
- fi
- sleep 5
- done
這個監(jiān)控代碼實(shí)現(xiàn)了mysql,nginx,php-fpm的HA。
加上權(quán)限,并執(zhí)行
- chmod +x /opt/check.sh
- nohup sh /opt/check.sh &
設(shè)置開機(jī)啟動:
echo “nohup sh /opt/check.sh &” >> /etc/rc.loal
4、測試keepalived
分別啟動keepalived
- service keepalived start
1)執(zhí)行ip a檢查node1和node2是否已經(jīng)存在vip:12.12.12.100和12.12.12.200
2)測試nginx和php-fpm的HA。在node1執(zhí)行service nginx stop或者service php-fpm stop停止nginx或php-fpm,過幾秒鐘后你會發(fā)現(xiàn)node2已經(jīng)接管了vip 12.12.12.100,并且使用vip 12.12.12.100或12.12.12.200瀏覽nginx網(wǎng)頁你會發(fā)現(xiàn)網(wǎng)頁顯示的IP一直是192.168.1.20,表明 keepalived已經(jīng)成功接管node1的vip和nginx或php-fpm服務(wù)。
3)測試mysql HA。在node1執(zhí)行service mysqld stop停止mysql服務(wù),幾秒后在node2查看,發(fā)現(xiàn)node2已經(jīng)接管vip 192.168.1.100,并且已經(jīng)啟動mysql服務(wù)。
注意:在恢復(fù)mysql或nginx,php-fpm時,先停止監(jiān)控腳本,要不heartbeat或keepalived還沒實(shí)現(xiàn)接管又被停止。
參考:http://bbs./thread-965-1-1.html
本文由“www.126cm.com”收集編輯,如涉及版權(quán)問題請郵件通知。