通过 python 脚本迁移 Redis 数据

28 篇文章 1 订阅
6 篇文章 0 订阅

背景

  • 需求:需要将的 Redis 数据迁移由云厂商 A 迁移至云厂商 B
  • 问题:云版本的 Redis 版本不支持 SYNC、MIGRATE、BGSAVE 等命令,使得许多工具用不了(如 redis-port)

思路

  • (1)从 Redis A 获取所有 key,依次导出数据
  • (2)将导出的数据加载到 Redis B

实现

示例:将 127.0.0.1:6379 数据迁移至 127.0.0.1:6380

导出脚本 export.py

from optparse import OptionParser
import os
import redis
import logging
import sys

mylogger = None
fdata = None


def new_logger(
        logger_name='AppName',
        level=logging.INFO,
        to_file=True,
        log_file_name="app.log",
        format='[%(asctime)s] %(filename)s:%(lineno)d %(message)s'):
    if to_file:
        handler = logging.FileHandler(log_file_name)
    else:
        handler = logging.StreamHandler(sys.stdout)
    handler.setFormatter(logging.Formatter(format))

    logger = logging.getLogger(logger_name)
    logger.addHandler(handler)
    logger.setLevel(level)

    return logger


def gen_redis_proto(*args):
    '''Write out the string in redis protocol so it can be replayed back later'''
    proto = '*{0}\\r\\n'.format(len(args))
    for arg in args:
        proto += '${0}\\r\\n'.format(len(arg))
        proto += '{0}\\r\\n'.format(arg)

    return proto


# 只取出指定前缀的 key
def match_key(key: str):
    return key.startswith('normal')

def extract(options, fd, logg: logging.Logger):
    src_r = redis.StrictRedis(host=options.redis_source_url, port=options.redis_source_port, db=options.redis_source_db)
    all_keys = src_r.keys('*')
    for key in all_keys:
        key_type = ''
        arr = []
        try:
            key = key.decode('utf8')
            if not match_key(key):
                continue

            key_type = src_r.type(key).decode('utf8')
            if key_type == 'hash':
                arr.append('HMSET')
                arr.append(key)
                for k, v in src_r.hgetall(key).items():
                    arr.append(k.decode('utf8'))
                    arr.append(v.decode('utf8'))
            elif key_type == 'string':
                arr.append('SET')
                arr.append(key)
                arr.append(src_r.get(key).decode('utf8'))
            elif key_type == 'set':
                arr.append('SADD')
                arr.append(key)
                arr.extend([v.decode('utf8') for v in src_r.smembers(key)])
            elif key_type == 'list':
                arr.append('LPUSH')
                arr.append(key)
                arr.extend([v.decode('utf8') for v in src_r.lrange(key, 0, -1)])
            elif key_type == 'zset':
                arr.append('ZADD')
                arr.append(key)
                for member, score in src_r.zrange(key, 0, -1, withscores=True):
                    arr.append(str(score))
                    arr.append(member.decode('utf8'))
            else:
                # TODO 其它的数据类型
                logg.error('Unsupported key type detected: {}, key: {}'.format(key_type, key))
                continue

            fd.write(gen_redis_proto(*arr) + "\n")

			# 设置 ttl
            ttl = src_r.ttl(key)
            if ttl != -1:
                fd.write(gen_redis_proto(*['EXPIRE', key, str(ttl)]) + "\n")
        except Exception as e:
            logg.error('Unsupported key type detected: {}, key: {}, error: {}'.format(key_type, key, e.__str__()))


if __name__ == '__main__':
    parser = OptionParser()

    parser.add_option('-s', '--redis-source-url',
                      action='store',
                      dest='redis_source_url',
                      help='The url of the source redis which is to be cloned [required]')
    parser.add_option('-p', '--redis-source-port',
                      action='store',
                      dest='redis_source_port',
                      default=6379,
                      type=int,
                      help='The port of the source redis which is to be cloned [required, \
                            default: 6379]')
    parser.add_option('-n', '--redis-source-db',
                      action='store',
                      dest='redis_source_db',
                      default=0,
                      type=int,
                      help='The db num of the source redis[required, default: 0]')

    parser.add_option('-o', '--redis-data-output-file-name',
                      action='store',
                      dest='redis_data_output_file_name',
                      default='redis.data',
                      type=str,
                      help='The output file name of the source redis data[required, default: redis.data]')

    parser.add_option('-l', '--log-file-name',
                      action='store',
                      dest='log_file_name',
                      default='app.log',
                      type=str,
                      help='The log file name[required, default: app.log]')

    (options, args) = parser.parse_args()

    if not (options.redis_source_url and options.redis_source_port):
        parser.error('redis-source-url, redis-source-port are required arguments. Please see help')

    data_path = options.redis_data_output_file_name
    if os.path.exists(data_path):
        os.remove(data_path)

    mylogger = new_logger(to_file=True, level=logging.ERROR, log_file_name=options.log_file_name)
    with open(data_path, 'a+') as fd:
        extract(options, fd, mylogger)

导出数据

$ python export.py -s 127.0.0.1 -p 6379

$ head redis.data
*5\r\n$5\r\nLPUSH\r\n$11\r\nnormalQueue\r\n$3\r\nccc\r\n$2\r\nbb\r\n$3\r\naaa\r\n
*8\r\n$4\r\nSADD\r\n$9\r\nnormalSet\r\n$2\r\ndd\r\n$2\r\nbb\r\n$2\r\ncc\r\n$2\r\nee\r\n$2\r\naa\r\n$2\r\nff\r\n
*3\r\n$6\r\nEXPIRE\r\n$9\r\nnormalSet\r\n$4\r\n1728\r\n
*6\r\n$5\r\nHMSET\r\n$10\r\nnormalHash\r\n$2\r\nk1\r\n$2\r\nv1\r\n$2\r\nk2\r\n$2\r\nv2\r\n
*3\r\n$3\r\nSET\r\n$9\r\nnormalStr\r\n$3\r\nvvv\r\n

导入脚本 load.sh

#!/bin/sh

while read -r line
do
    nohup printf "%b" "$line"| redis-cli -p 6380 --pipe >> load-std.log 2>> load-err.log &
done < $1

导入数据

sh load.sh redis.data

参考

  • https://stackoverflow.com/questions/44288974/sync-with-master-failed-err-unknown-command-sync
  • https://gist.github.com/jimmyislive/0efd7a6a1c7f7afd73e8#file-clone_redis-py
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陈挨踢

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值