實(shí)現(xiàn)在線?+?離線模式進(jìn)行遷移?Redis?數(shù)據(jù)實(shí)戰(zhàn)指南
redis-full-check的使用背景
在經(jīng)歷了之前的文章內(nèi)容章節(jié)內(nèi)容,已完成Redis遷移后,可能會(huì)存在以下問題需要進(jìn)行數(shù)據(jù)遷移之后的對(duì)比。例如,如果Redis遷移的過程出現(xiàn)異常,源端與目的端Redis的數(shù)據(jù)將會(huì)不一致。
在Redis遷移完成后進(jìn)行數(shù)據(jù)校驗(yàn)可以檢查數(shù)據(jù)的一致性,該如何校驗(yàn)就是我們本文的內(nèi)容。我們?cè)谶@里采用的是阿里開源的數(shù)據(jù)對(duì)比工具與Redis-Shake形成伴侶模式的開源工具redis-full-check,使用redis-full-check進(jìn)行校驗(yàn)?zāi)軌蛘页霎惓?shù)據(jù),為數(shù)據(jù)對(duì)齊提供可靠依據(jù),本文主要介紹如何使用RedisFullCheck。
redis-full-check的基本介紹
redis-full-check是阿里云自研的Redis數(shù)據(jù)校驗(yàn)工具,能夠提取源端和目的端的數(shù)據(jù)進(jìn)行多輪差異化比較,并將比較結(jié)果記錄在一個(gè)SQLite3數(shù)據(jù)庫中,從而達(dá)到全量數(shù)據(jù)校驗(yàn)的目的。
遷移源端和目的端Redis實(shí)例需為主從版、單節(jié)點(diǎn)版、開源集群版以及部分云上帶proxy的集群版(阿里云、騰訊云)。
開源地址redis-full-check源碼地址: ??https://github.com/aliyun/redis-full-check??
redis-full-check下載地址: ??https://github.com/alibaba/RedisFullCheck/releases??
編譯源碼
運(yùn)行 ./bin/redis-full-check.darwin64 or redis-full-check.linux64,它分別在OSX和Linux中構(gòu)建,然而,二進(jìn)制文件并不總是最新版本。 或者您可以根據(jù)以下步驟構(gòu)建red- full-check自己:
git clone https://github.com/alibaba/RedisFullCheck.git cd RedisFullCheck/src/vendor GOPATH=`pwd`/../..; govendor sync
注意:必須先安裝govendor,然后拉出所有依賴
cd ../../ && ./build.sh
執(zhí)行build.sh進(jìn)行編譯即可。
基本原理redis-full-check通過全量對(duì)比源端和目的端的redis中的數(shù)據(jù)的方式來進(jìn)行數(shù)據(jù)校驗(yàn),其比較方式通過多輪次比較:每次都會(huì)抓取源和目的端的數(shù)據(jù)進(jìn)行差異化比較,記錄不一致的數(shù)據(jù)進(jìn)入下輪對(duì)比(記錄在sqlite3 db中)。通過多倫比較不斷收斂,減少因數(shù)據(jù)增量同步導(dǎo)致的源庫和目的庫的數(shù)據(jù)不一致。最后sqlite中存在的數(shù)據(jù)就是最終的差異結(jié)果。數(shù)據(jù)對(duì)比介紹
redis-full-check對(duì)比的方向是單向,如果希望對(duì)比雙向,則需要對(duì)比2次,第一次以A為源庫,B為目的庫,第二次以B為源庫,A為目的庫。
首次對(duì)別:會(huì)抓取源庫A的數(shù)據(jù),然后檢測是否位于B中,反向不會(huì)檢測,它檢測的是源庫是否是目的庫的子集。每次比較,會(huì)先抓取比較的key。
第一輪是從源庫中進(jìn)行抓取,后面輪次是從sqlite3 db中進(jìn)行抓取;抓取key之后是分別抓取key對(duì)應(yīng)的field和value進(jìn)行對(duì)比,然后將存在差異的部分存入sqlite3 db中,用于下次比較。
數(shù)據(jù)對(duì)比核心流程
下圖是基本的數(shù)據(jù)流圖。
對(duì)比模式(comparemode)有三種可選:
KeyOutline:只對(duì)比key值是否相等。ValueOutline:只對(duì)比value值的長度是否相等。FullValue:對(duì)比key值、value長度、value值是否相等。
對(duì)比會(huì)進(jìn)行comparetimes輪(默認(rèn)comparetimes=3)比較:
第一輪,首先找出在源庫上所有的key,然后分別從源庫和目的庫抓取進(jìn)行比較。第二輪,開始迭代比較,只比較上一輪結(jié)束后仍然不一致的key和field。不一致場景下分析
對(duì)于key不一致的情況,包括lack_source ,lack_target 和type,從源庫和目的庫重新取key、value進(jìn)行比較。
value不一致的string,重新比較key:從源和目的取key、value比較。value不一致的hash、set和zset,只重新比較不一致的field,之前已經(jīng)比較且相同的filed不再比較。這是為了防止對(duì)于大key情況下,如果更新頻繁,將會(huì)導(dǎo)致校驗(yàn)永遠(yuǎn)不通過的情況。value不一致的list,重新比較key:從源和目的取key、value比較。
每輪之間會(huì)停止一定的時(shí)間(Interval)。
對(duì)于hash,set,zset,list大key處理采用以下方式:len <= 5192,直接取全量field、value進(jìn)行比較,使用如下命令:hgetall,smembers,zrange 0 -1 withscores,lrange 0 -1。len > 5192,使用hscan,sscan,zscan,lrange分批取field和value。使用redis-full-check解壓redis-full-check.tar.gz:tar -xvf redis-full-check.tar.gz
執(zhí)行如下命令進(jìn)行數(shù)據(jù)校驗(yàn):單機(jī)實(shí)例之間進(jìn)行數(shù)據(jù)對(duì)比檢測
./redis-full-check -s $(source_redis_ip_port) -p $(source_password) -t $(target_redis_ip_port) -a $(target_password)
集群實(shí)例之間進(jìn)行數(shù)據(jù)對(duì)比檢測
./redis-full-check -s "<Redis集群地址1連接地址:Redis集群地址1端口號(hào);Redis集群地址2連接地址:Redis集群地址2端口號(hào);Redis集群地址3連接地址:Redis集群地址3端口號(hào)>" -p <Redis集群密碼> -t <Redis連接地址:Redis端口號(hào)> -a <Redis密碼> --comparemode=1 --comparetimes=1 --qps=10 --batchcount=100 --sourcedbtype=1 --targetdbfilterlist=0
參數(shù)信息介紹
redis-full-check中主要參數(shù)如下:
-s, --source=SOURCE 源redis庫地址(ip:port),如果是集群版,那么需要以分號(hào)(;)分割不同的db,只需要配置主或者從的其中之一。例如:10.1.1.1:1000;10.2.2.2:2000;10.3.3.3:3000。 -p, --sourcepassword=Password 源redis庫密碼 --sourceauthtype=AUTH-TYPE 源庫管理權(quán)限,開源reids下此參數(shù)無用。 --sourcedbtype= 源庫的類別,0:db(standalone單節(jié)點(diǎn)、主從),1: cluster(集群版),2: 阿里云 --sourcedbfilterlist= 源庫需要抓取的邏輯db白名單,以分號(hào)(;)分割,例如:0;5;15表示db0,db5和db15都會(huì)被抓取 -t, --target=TARGET 目的redis庫地址(ip:port) -a, --targetpassword=Password 目的redis庫密碼 --targetauthtype=AUTH-TYPE 目的庫管理權(quán)限,開源reids下此參數(shù)無用。 --targetdbtype= 參考sourcedbtype --targetdbfilterlist= 參考sourcedbfilterlist -d, --db=Sqlite3-DB-FILE 對(duì)于差異的key存儲(chǔ)的sqlite3 db的位置,默認(rèn)result.db --comparetimes=COUNT 比較輪數(shù) -m, --comparemode= 比較模式,1表示全量比較,2表示只對(duì)比value的長度,3只對(duì)比key是否存在,4全量比較的情況下,忽略大key的比較 --id= 用于打metric --jobid= 用于打metric --taskid= 用于打metric -q, --qps= qps限速閾值 --interval=Second 每輪之間的時(shí)間間隔 --batchcount=COUNT 批量聚合的數(shù)量 --parallel=COUNT 比較的并發(fā)協(xié)程數(shù),默認(rèn)5 --log=FILE log文件 --result=FILE 不一致結(jié)果記錄到result文件中,格式:'db diff-type key field' --metric=FILE metric文件 --bigkeythreshold=COUNT 大key拆分的閾值,用于comparemode=4 -f, --filterlist=FILTER 需要比較的key列表,以豎線(|)分割。例如:"abc*|efg|m*"表示對(duì)比'abc', 'abc1', 'efg', 'm', 'mxyz',不對(duì)比'efgh', 'p'。 -v, --version
對(duì)比實(shí)現(xiàn)案例對(duì)比2個(gè)主從版/單節(jié)點(diǎn):
./redis-full-check -t 10.101.72.137:30661 -s 10.101.72.137:30551
對(duì)比主從和集群:
./redis-full-check -s "100.81.164.177:21331;100.81.164.177:21332;100.81.164.177:21333" -t 10.101.72.137:30551 --comparemode=1 --comparetimes=1 --qps=10 --batchcount=100 --sourcedbtype=1 --targetdbfilterlist=0
由于集群版只有db0,所以如果一端是集群版,另一端是非集群版(多個(gè)邏輯db),則需要添加sourcedbfilterlist或者targetdbfilterlist(非集群版本的一端)
查詢對(duì)比結(jié)果
sqlite> select * from key; id key type conflict_type db source_len target_len ---------- --------------- ---------- ------------- ---------- ---------- ---------- 1 keydiff1_string string value 1 6 6 2 keydiff_hash hash value 0 2 1 3 keydiff_string string value 0 6 6 4 key_string_diff string value 0 6 6 5 keylack_string string lack_target 0 6 0 sqlite> sqlite> select * from field; id field conflict_type key_id ---------- ---------- ------------- ---------- 1 k1 lack_source 2 2 k2 value 2 3 k3 lack_target 2
對(duì)比結(jié)果差異分析不一致類型
redis-full-check判斷不一致的方式主要分為2類:key不一致和value不一致。
key不一致
key不一致主要分為以下幾種情況:
lack_target : key存在于源庫,但不存在于目的庫。type: key存在于源庫和目的庫,但是類型不一致。value: key存在于源庫和目的庫,且類型一致,但是value不一致。value不一致不同數(shù)據(jù)類型有不同的對(duì)比標(biāo)準(zhǔn):string: value不同。hash: 存在field,滿足下面3個(gè)條件之一: field存在于源端,但不存在與目的端。field存在于目的端,但不存在與源端。field同時(shí)存在于源和目的端,但是value不同。set/zset:與hash類似。list: 與hash類似。
field沖突類型有以下幾種情況(只存在于hash,set,zset,list類型key中)
lack_source: field存在于源端key,field不存在與目的端key。lack_target: field不存在與源端key,field存在于目的端key。value: field存在于源端key和目的端key,但是field對(duì)應(yīng)的value不同。
參考資料
到此這篇關(guān)于實(shí)現(xiàn)在線 + 離線模式進(jìn)行遷移 Redis 數(shù)據(jù)實(shí)戰(zhàn)指南的文章就介紹到這了,更多相關(guān)遷移 Redis 數(shù)據(jù)實(shí)戰(zhàn)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Redis有序集合類型的操作_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
今天通過本文給大家說一下Redis中最后一個(gè)數(shù)據(jù)類型 “有序集合類型”,需要的的朋友參考下吧2017-08-08redis中如何使用lua腳本讓你的靈活性提高5個(gè)逼格詳解
這篇文章主要給大家介紹了關(guān)于redis中如何使用lua腳本讓你的靈活性提高5個(gè)逼格的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-10-10Redis數(shù)據(jù)庫分布式設(shè)計(jì)方案介紹
大家好,本篇文章主要講的是Redis數(shù)據(jù)庫分布式設(shè)計(jì)方案介紹,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下2022-01-01dubbo服務(wù)使用redis注冊(cè)中心的系列異常解決
這篇文章主要為大家介紹了dubbo服務(wù)在使用redis注冊(cè)中心遇到的一系列異常的解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2022-03-03