1. 程式人生 > >redis的分散式解決方案--codis

redis的分散式解決方案--codis

codis是豌豆莢開源的分散式伺服器,目前處於穩定階段。

原文地址:https://github.com/wandoulabs/codis/blob/master/doc/tutorial_zh.md

Codis 是一個分散式 Redis 解決方案, 對於上層的應用來說, 連線到 Codis Proxy 和連線原生的 Redis Server 沒有明顯的區別 (不支援的命令列表), 上層應用可以像使用單機的 Redis 一樣使用, Codis 底層會處理請求的轉發, 不停機的資料遷移等工作, 所有後邊的一切事情, 對於前面的客戶端來說是透明的, 可以簡單的認為後邊連線的是一個記憶體無限大的 Redis 服務.

基本框架如下:

Codis 由四部分組成:

Codis Proxy (codis-proxy)
Codis Manager (codis-config)
Codis Redis (codis-server)
ZooKeeper

codis-proxy 是客戶端連線的 Redis 代理服務, codis-proxy 本身實現了 Redis 協議, 表現得和一個原生的 Redis 沒什麼區別 (就像 Twemproxy), 對於一個業務來說, 可以部署多個 codis-proxy, codis-proxy 本身是無狀態的.

codis-config 是 Codis 的管理工具, 支援包括, 新增/刪除 Redis 節點, 新增/刪除 Proxy 節點, 發起資料遷移等操作. codis-config 本身還自帶了一個 http server, 會啟動一個 dashboard, 使用者可以直接在瀏覽器上觀察 Codis 叢集的執行狀態.

codis-server 是 Codis 專案維護的一個 Redis 分支, 基於 2.8.13 開發, 加入了 slot 的支援和原子的資料遷移指令. Codis 上層的 codis-proxy 和 codis-config 只能和這個版本的 Redis 互動才能正常執行.

ZooKeeper(以下簡稱ZK)是一個分散式協調服務框架,可以做到各節點之間的資料強一致性。簡單的理解就是在一個節點修改某個變數的值後,在其他節點可以最新的變化,這種變化是事務性的。通過在ZK節點上註冊監聽器,就可以獲得資料的變化。

Codis 依賴 ZooKeeper 來存放資料路由表和 codis-proxy 節點的元資訊, codis-config 發起的命令都會通過 ZooKeeper 同步到各個存活的 codis-proxy.

注:1.codis新版本支援redis到2.8.21

2.codis-group實現redis的水平擴充套件

下面我們來部署環境:

10.80.80.124       zookeeper_1codis-configcodis-server-master,slavecodis_proxy_1

10.80.80.126       zookeeper_2 codis-server-master,slavecodis _proxy_2

10.80.80.123       zookeeper_3 codis-serve-master,slavecodis _proxy_3

說明:

1.為了確保zookeeper的穩定性與可靠性,我們在124、126、123上搭建zookeeper叢集來對外提供服務;

2.codis-cofig作為分散式redis的管理工具,在整個分散式伺服器中只需要一個就可以完成管理任務;

3.codis-server和codis-proxy在3臺伺服器提供redis和代理服務。

一.部署zookeeper叢集

1.配置hosts(在3臺伺服器上)

10.80.80.124 codis1
10.80.80.126 codis2
10.80.80.123 codis3

2.配置java環境(在3臺伺服器上)

vim /etc/profile
##JAVA###
export JAVA_HOME=/usr/local/jdk1.7.0_71
export JRE_HOME=/usr/local/jdk1.7.0_71/jre
export PATH=$JAVA_HOME/bin:$PATH
export CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar

source /etc/profile
3.安裝zookeeper(在3臺伺服器上)
cd /usr/local/src
wget http://mirror.bit.edu.cn/apache/zookeeper/zookeeper-3.4.6/zookeeper-3.4.6.tar.gz
tar -zxvf zookeeper-3.4.6.tar.gz -C /usr/local
4.配置環境變數(在3臺伺服器上)
vim /etc/profile
#zookeeper
ZOOKEEPER_HOME=/usr/local/zookeeper-3.4.6
export PATH=$PATH:$ZOOKEEPER_HOME/bin

source /etc/profile
5.修改zookeeper配置檔案(在3臺伺服器上)
#建立zookeeper的資料目錄和日誌目錄
mkdir -p /data/zookeeper/zk1/{data,log}
cd /usr/local/zookeeper-3.4.6/conf
cp zoo_sample.cfg zoo.cfg
vim /etc/zoo.cfg
tickTime=2000
initLimit=10
syncLimit=5
dataDir=/data/zookeeper/zk1/data
dataLogDir=/data/zookeeper/zk1/log
clientPort=2181
server.1=codis1:2888:3888
server.2=codis2:2888:3888
server.3=codis3:2888:3888
6.在dataDir下建立myid檔案,對應節點id(在3臺伺服器上)
#在124上 
cd /data/zookeeper/zk1/data
echo 1 > myid
#在126上 
cd /data/zookeeper/zk1/data
echo 2 > myid
#在123上 
cd /data/zookeeper/zk1/data
echo 3 > myid
7.啟動zookeeper服務(在3臺伺服器上)
/usr/local/zookeeper-3.4.6/bin/zkServer.sh start
注:在你所在的當前目錄下會生成一個zookeeper.out的日誌檔案,裡面記錄了啟動過程中的詳細資訊;由於叢集沒有全部資訊,會報“myid 2或myid 3 未啟動”的資訊,當叢集全部啟動後就會正常,我們可以忽略。

8.檢視zookeeper所有節點的狀態(在3臺伺服器上)

#124
/usr/local/zookeeper-3.4.6/bin/zkServer.sh status
JMX enabled by default
Using config: /usr/local/zookeeper-3.4.6/bin/../conf/zoo.cfg
Mode: leader
#126
/usr/local/zookeeper-3.4.6/bin/zkServer.sh status
JMX enabled by default
Using config: /usr/local/zookeeper-3.4.6/bin/../conf/zoo.cfg
Mode: follower
#123
/usr/local/zookeeper-3.4.6/bin/zkServer.sh status
JMX enabled by default
Using config: /usr/local/zookeeper-3.4.6/bin/../conf/zoo.cfg
Mode: follower
二.部署codis叢集

1.安裝go語言(在3臺伺服器上)

tar -zxvf go1.4.2.linux-amd64.tar.gz -C /usr/local/
2.新增go環境變數(在3臺伺服器上)
vim /etc/profile
#go
export PATH=$PATH:/usr/local/go/bin
export GOPATH=/usr/local/codis

source /etc/profile
3.安裝codis(在3臺伺服器上)
go get github.com/wandoulabs/codis
cd $GOPATH/src/github.com/wandoulabs/codis
#執行編譯測試指令碼,編譯go和reids。 
./bootstrap.sh
make gotest
#將編譯好後,把bin目錄和一些指令碼複製過去/usr/local/codis目錄下
mkdir -p /usr/local/codis/{conf,redis_conf,scripts}
cp -rf bin /usr/local/codis/  
cp sample/config.ini /usr/local/codis/conf/
cp -rf sample/redis_conf /usr/local/codis
cp -rf sample/* /usr/local/codis/scripts
4.配置codis-proxy(在3臺伺服器上,在此以124為例)
#124
cd /usr/local/codis/conf
vim config.ini
zk=codis1:2181,codis2:2181,codis3:2181
product=codis
#此處配置圖形化介面的dashboard,注意codis叢集只要一個即可,因此全部指向10.80.80.124:18087
dashboard_addr=192.168.3.124:18087
coordinator=zookeeper
backend_ping_period=5
session_max_timeout=1800
session_max_bufsize=131072
session_max_pipeline=128
proxy_id=codis_proxy_1
#126
cd /usr/local/codis/conf
vim config.ini
zk=codis1:2181,codis2:2181,codis3:2181
product=codis
#此處配置圖形化介面的dashboard,注意codis叢集只要一個即可,因此全部指向10.80.80.124:18087
dashboard_addr=192.168.3.124:18087
coordinator=zookeeper
backend_ping_period=5
session_max_timeout=1800
session_max_bufsize=131072
session_max_pipeline=128
proxy_id=codis_proxy_2
#123
cd /usr/local/codis/conf
vim config.ini
zk=codis1:2181,codis2:2181,codis3:2181
product=codis
#此處配置圖形化介面的dashboard,注意codis叢集只要一個即可,因此全部指向10.80.80.124:18087
dashboard_addr=192.168.3.124:18087
coordinator=zookeeper
backend_ping_period=5
session_max_timeout=1800
session_max_bufsize=131072
session_max_pipeline=128
proxy_id=codis_proxy_3
5.修改codis-server的配置檔案(在3臺伺服器上)
#建立codis-server的資料目錄和日誌目錄
mkdir -p /data/codis/codis-server/{data,logs}
cd /usr/local/codis/redis_conf
#主庫
vim 6380.conf
daemonize yes
pidfile /var/run/redis_6380.pid
port 6379
logfile "/data/codis_server/logs/codis_6380.log"
save 900 1
save 300 10
save 60 10000
dbfilename 6380.rdb
dir /data/codis_server/data
#從庫
cp 6380.conf 6381.conf
sed -i 's/6380/6381/g' 6381.conf
6.新增核心引數
echo "vm.overcommit_memory = 1" >>  /etc/sysctl.conf
sysctl -p
7.按照啟動流程啟動
cat /usr/loca/codis/scripts/usage.md
0. start zookeeper 
1. change config items in config.ini 
2. ./start_dashboard.sh 
3. ./start_redis.sh 
4. ./add_group.sh 
5. ./initslot.sh 
6. ./start_proxy.sh 
7. ./set_proxy_online.sh 
8. open browser to http://localhost:18087/admin
雖然scripts目錄下面有相應啟動指令碼,也可以用startall.sh全部啟動,但剛開始建議手動啟動,以熟悉codis啟動過程。

由於之前zookeeper已經啟動,下面我們來啟動其他專案。

注:1.在啟動過程中需要指定相關日誌目錄或配置檔案目錄,為便於統一管理,我們都放在/data/codis下;

2.dashboard在codis叢集中只需要在一臺伺服器上啟動即可,此處在124上啟動;凡是用codis_config的命令都是在124上操作,其他啟動項需要在3臺伺服器上操作。

相關命令如下:

/usr/local/codis/bin/codis-config -h
usage: codis-config  [-c <config_file>] [-L <log_file>] [--log-level=<loglevel>]
		<command> [<args>...]
options:
   -c	set config file
   -L	set output log file, default is stdout
   --log-level=<loglevel>	set log level: info, warn, error, debug [default: info]

commands:
	server
	slot
	dashboard
	action
	proxy


(1)啟動dashboard(在124上啟動)

#dashboard的日誌目錄和訪問目錄
mkdir -p /data/codis/codis_dashboard/logs
codis_home=/usr/local/codis
log_path=/data/codis/codis_dashboard/logs
nohup $codis_home/bin/codis-config -c $codis_home/conf/config.ini -L $log_path/dashboard.log dashboard --addr=:18087 --http-log=$log_path/requests.log &>/dev/null &

通過10.80.80.124:18087即可訪問圖形管理介面


(2)啟動codis-server(在3臺伺服器上)
/usr/local/codis/bin/codis-server /data/codis_server/conf/6380.conf
/usr/local/codis/bin/codis-server /data/codis_server/conf/6381.conf
(3)新增 Redis Server Group(124上)

注意:每一個 Server Group 作為一個 Redis 伺服器組存在, 只允許有一個 master, 可以有多個 slave, group id 僅支援大於等於1的整數

目前我們在3臺伺服器上分了3組,因此我們需要新增3組,每組由一主一從構成

#相關命令	
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server
usage:
	codis-config server list
	codis-config server add <group_id> <redis_addr> <role>
	codis-config server remove <group_id> <redis_addr>
	codis-config server promote <group_id> <redis_addr>
	codis-config server add-group <group_id>
	codis-config server remove-group <group_id>
#group 1
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 1 10.80.80.124:6380 master
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 1 10.80.80.124:6381 slave
#group 2
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 2 10.80.80.126:6380 master
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 2 10.80.80.126:6381 slave
#group 3
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 3 10.80.80.123:6380 master
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 3 10.80.80.123:6381 slave

注意:1.點選“Promote to Master”就會將slave的redis提升為master,而原來的master會自動下線。

    2./usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini server add 可以新增機器到相應組中,也可以更新redis的主/從角色。

    3.若為新機器,此處的keys應該為空

(4) 設定 server group 服務的 slot 範圍(124上)

#相關命令
/usr/local/codis/bin/codis-config -c /usr/local/codis/conf/config.ini slot
usage:
	codis-config slot init [-f]
	codis-config slot info <slot_id>
	codis-config slot set <slot_id> <group_id> <status>
	codis-config slot range-set <slot_from> <slot_to> <group_id> <status>
	codis-config slot migrate <slot_from> <slot_to> <group_id> [--delay=<delay_time_in_ms>]
	codis-config slot rebalance [--delay=<delay_time_in_ms>]
	
#Codis 採用 Pre-sharding 的技術來實現資料的分片, 預設分成 1024 個 slots (0-1023), 對於每個key來說, 通過以下公式確定所屬的 Slot Id : SlotId = crc32(key) % 1024 每一個 slot 都會有一個特定的 server group id 來表示這個 slot 的資料由哪個 server group 來提供。

我們在此將1024個slot分為三段,分配如下:
/usr/local/codis/bin/codis-config -c conf/config.ini slot range-set 0 340 1 online
/usr/local/codis/bin/codis-config -c conf/config.ini slot range-set 341 681 2 online
/usr/local/codis/bin/codis-config -c conf/config.ini slot range-set 682 1023 3 online
(5)啟動codis-proxy(在3臺伺服器上)
#codis_proxy的日誌目錄
mkdir -p /data/codis/codis_proxy/logs
codis_home=/usr/local/codis
log_path=/data/codis/codis_proxy/logs
nohup $codis_home/bin/codis-proxy --log-level warn -c $codis_home/conf/config.ini -L $log_path/codis_proxy_1.log  --cpu=8 --addr=0.0.0.0:19000 --http-addr=0.0.0.0:11000 > $log_path/nohup.out 2>&1 &

黑線處:codis讀取伺服器的主機名。

注意:若客戶端相關訪問proxy,需要在客戶端新增hosts

(6)上線codis-proxy

codis_home=/usr/local/codis
log_path=/data/codis/codis_proxy/logs
nohup $codis_home/bin/codis-proxy --log-level warn -c $codis_home/conf/config.ini -L $log_path/codis_proxy_1.log  --cpu=8 --addr=0.0.0.0:19000 --http-addr=0.0.0.0:11000 > $log_path/nohup.out 2>&1 &

注:啟動codis_proxy後,proxy此時處於offline狀態,無法對外提供服務,必須將其上線後才能對外提供服務。

ok,至此codis已經可以對外提供服務了。

三.HA

codis的ha分為前端proxy的ha以及後端codis-server的ha,在此簡單說下proxy的ha。

對於上層proxy來說,尤其是java客戶端來,codis提供jodis(修改過的jedis)來實現proxy的ha。它會通過監控zk上的註冊資訊來實時獲得當前可用的proxy列表,既可以保證高可用性,也可以通過輪流請求所有的proxy實現負載均衡;支援proxy的自動上線和下線。