通过 python 脚本迁移 Redis 数据
背景
- 需求:需要将的 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 sysmylogger = None
fdata = Nonedef 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 loggerdef 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):continuekey_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))continuefd.write(gen_redis_proto(*arr) + "\n")# 设置 ttlttl = 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_nameif 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/shwhile read -r line
donohup 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
相关文章:
通过 python 脚本迁移 Redis 数据
背景 需求:需要将的 Redis 数据迁移由云厂商 A 迁移至云厂商 B问题:云版本的 Redis 版本不支持 SYNC、MIGRATE、BGSAVE 等命令,使得许多工具用不了(如 redis-port) 思路 (1)从 Redis A 获取所…...
nodejs之express学习(1)
安装 npm i express使用 // 导入 const express require(express) // 创建应用 const app express() // 创建路由 app.get(/home,(req,res)>{res.end("hello express") }) app.listen(3000,()>{console.log("服务已启动~") })路由的介绍 什么是…...
【LeetCode】121. 买卖股票的最佳时机
121. 买卖股票的最佳时机 难度:简单 题目 给定一个数组 prices ,它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。 你只能选择 某一天 买入这只股票,并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获…...
Vue3-VueRouter4路由语法解析
1.创建路由实例由createRouter实现 2.路由模式 1)history模式使用createWebHistory():地址栏不带# 2)hash模式使用createWebHashHistory():地址栏带# 3)参数是基础路径,默认/ 括号里的就是设置路径的前…...
ChromeDriver最新版本下载与安装方法
关于ChromeDriver最新下载地址:https://googlechromelabs.github.io/chrome-for-testing/ 下载与安装 setp1:查看Chrome浏览器版本 首先,需要检查Chrome浏览器的版本。请按照以下步骤进行: 打开Chrome浏览器。 点击浏览器右上角…...
illuminate/database 使用 四
文档:Hyperf Database: Getting Started - Laravel 10.x - The PHP Framework For Web Artisans 因为hyperf使用illuminate/database,所以按照文章,看illuminate/database代码实现。 一、读写分离 根据文档读写的host可以分开。设置读写分…...
Spring面向切面编程(AOP);Spring控制反转(IOC);解释一下Spring AOP里面的几个名词;Spring 的 IoC支持哪些功能
文章目录 Spring面向切面编程(AOP)什么是AOPSpring AOP and AspectJ AOP 的区别?Spring AOP中的动态代理如何理解 Spring 中的代理?解释一下Spring AOP里面的几个名词Spring在运行时通知对象Spring切面可以应用5种类型的通知:什么是切面 Aspe…...
vatee万腾的科技征途:Vatee独特探索的数字化力量
在数字化时代的浪潮中,Vatee万腾以其独特的科技征途成为引领者。公司在数字化领域的探索之路不仅是技术的创新,更是一种对未知的勇敢涉足,是对新时代的深刻洞察和积极实践。 Vatee万腾通过独特的探索,展示了在数字化征途上的创新力…...
MySQL学习day03
一、SQL图形化界面工具 常用比较常用的图形化界面有sqlyog、mavicat、datagrip datagrip工具使用相当方便,功能比前面两种都要强大。 DataGrip工具的安装和使用请查看这篇文档:DataGrip 安装教程 DML-介绍 DML全称是Data Manipulation Language(数据…...
《QT从基础到进阶·三十七》QWidget实现左侧导航栏效果
NavigationBarPlugin插件类实现了对左侧导航栏的管理,我们可以在导航栏插件中添加界面,并用鼠标点击导航栏能够切换对应的界面。 源码在文章末尾 实现效果如下: NavigationBarPlugin实现的接口如下: class NAVIGATIONBAR_EXP…...
sftp学习
什么是sftp? sftp的简单操作 远程连接 int RobostSftp::Init(QString ip,QString port,QString user_name, QString user_password) { int rc;session ssh_new();if (!session) {fprintf(stderr, "ssh initialization failed\n");// return 1…...
C++之STL库:string类(用法列举和总结)
前言 大家在学习STL库的时候一定要学会看英文文档,俗话说熟能生巧,所以还得多练!在使用string类之前,要包含头文件#include <string>和using namespace std; 文档链接:string - C Reference 一、string——构造…...
小程序中的大道理--综述
前言 以下将用一个小程序来探讨一些大道理, 这些大道理包括可扩展性, 抽象与封装, 可维护性, 健壮性, 团队合作, 工具的利用, 可测试性, 自顶向下, 分而治之, 分层, 可读性, 模块化, 松耦合, MVC, 领域模型, 甚至对称性, 香农的信息论等等. 为什么不用大程序来说大道理呢? …...
tlais智能学习辅助系统-修改部门功能实现
学习黑马程序员的JavaWeb课程,自己写的部门信息修改部分程序 控制层: //DeptController.java /** * 根据ID查询部门信息 * param id * return */ GetMapping("/{id}") public Result select(PathVariable Integer id){log.info("查询id…...
GLM: 自回归空白填充的多任务预训练语言模型
当前,ChatGLM-6B 在自然语言处理领域日益流行。其卓越的技术特点和强大的语言建模能力使其成为对话语言模型中的佼佼者。让我们深入了解 ChatGLM-6B 的技术特点,探索它在对话模型中的创新之处。 GLM: 自回归空白填充的多任务预训练语言模型 ChatGLM-6B 技…...
函数递归所应满足的条件
1.递归的概念 递归是学习C语⾔函数绕不开的⼀个话题,那什么是递归呢? 递归其实是⼀种解决问题的⽅法,在C语⾔中,递归就是函数⾃⼰调⽤⾃⼰。 递归的思想: 把⼀个⼤型复杂问题层层转化为⼀个与原问题相似,但…...
Python入职某新员工大量使用Lambda表达式,却被老员工喷是屎山
Python中Lambda表达式是一种简洁而强大的特性,其在开发中的使用优缺点明显,需要根据具体场景权衡取舍。 Lambda表达式的优点之一是它的紧凑语法,适用于一些短小而简单的函数。这种形式使得代码更为精炼,特别在一些函数式编程场景中,Lambda表达式可以提高代码的表达力。此外…...
Android Bitmap保存成至手机图片文件,Kotlin
Android Bitmap保存成至手机图片文件,Kotlin fun saveBitmap(name: String?, bm: Bitmap) {val savePath Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).toString()if (!Files.exists(Paths.get(savePath))) {Log.d("保存文…...
frp V0.52.3 搭建
下载 https://github.com/fatedier/frp/releases/ 此版本暂时没有windows的,想在windows使用请下载v0.52.2 简易搭建 frps.toml的配置文件,以下12000、8500需要在云服务器中的防火墙中开放tcp # bindPort为frps和frpc通信的端口,需要在防…...
最近数据分析面试的一点感悟...
我是阿粥,也是小z 最近面了不少应届的同学(数据分析岗位),颇有感触,与各位分享。 简历可以润色,但要适度 运用一些原则,如STAR法则,让简历逻辑更清晰,条块分明࿰…...
龙虾白嫖指南,请查收~胃
1. 什么是 Apache SeaTunnel? Apache SeaTunnel 是一个非常易于使用、高性能、支持实时流式和离线批处理的海量数据集成平台。它的目标是解决常见的数据集成问题,如数据源多样性、同步场景复杂性以及资源消耗高的问题。 核心特性 丰富的数据源支持&#…...
告别无效流量!亚马逊关键词挖掘:新手 7 天精准获客不浪费
亚马逊日常运营,关键词选不对,广告全白费:花大价钱投热门大词,点击多、转化少,ACoS 居高不下;自己想的关键词没人搜,广告预算花不出去,零曝光零订单;只盯着 10 几个…...
C# 面试高频题:装箱和拆箱是如何影响性能的?负
OCP原则 ocp指开闭原则,对扩展开放,对修改关闭。是七大原则中最基本的一个原则。 依赖倒置原则(DIP) 什么是依赖倒置原则 核心是面向接口编程、面向抽象编程, 不是面向具体编程。 依赖倒置原则的目的 降低耦合度&#…...
PX4无人机实战调试:从光流集成到安全返航的完整流程解析
1. 光流传感器集成与配置 光流传感器是无人机在室内或近地飞行时的关键部件,它通过分析连续图像帧之间的像素位移来估算飞行器的水平速度。对于PX4飞控来说,微空MTF-01这类光流模块的集成需要特别注意硬件接口和参数配置的匹配性。 实际调试时最容易忽略…...
vivado hls设计通用评价指标
一、评价vivado hls设计好坏的指标 资料利用、功耗、流水线、循环展开、数据流、数组分割、浮点转定点。 1.延迟 2.吞吐量 3.启动间隔II Vivado HLS设计的评价指标主要围绕资源、时序、吞吐量和延迟这四大核心维度展开。二、Area面积资源 LUT,FF,BRAM,DSP 硬件电路所…...
OpenRecall与AI助手集成:打造终极个人记忆增强系统
OpenRecall与AI助手集成:打造终极个人记忆增强系统 【免费下载链接】openrecall OpenRecall is a fully open-source, privacy-first alternative to proprietary solutions like Microsofts Windows Recall. With OpenRecall, you can easily access your digital …...
Maxwell电机多目标尺寸优化:基于Ansys Maxwell与OptiSlang的内嵌式永...
Maxwell电机多目标尺寸优化 Ansys Maxwell 和OptiSlang 有案例电机,永磁同步电机内嵌式 满足电机多尺寸参数入手,满足多尺寸联动优化,最终达到多参数优化效果 提供源文件,提供操作视频概述 本文档旨在详细阐述一套基于 Ansys Maxw…...
保姆级教程:基于ROS Melodic和MoveIt!,手把手搭建双RM65机械臂协同控制系统
基于ROS Melodic和MoveIt!的双RM65机械臂协同控制系统实战指南 在工业自动化与智能制造领域,双机械臂协同作业正成为提升生产效率的关键技术。想象一下,两台机械臂如同默契的舞者,在装配线上精准配合,完成单台设备难以企及的复杂任…...
Windows注册表深度解析:核心结构与关键应用场景
1. Windows注册表的核心结构解析 第一次打开注册表编辑器时,那个树状结构的界面可能会让人望而生畏。但当你理解它的设计逻辑后,会发现这个"系统大管家"其实很有条理。注册表的核心是五大根键(Root Key),每个…...
LLM服务SLA跌破99.2%?(GPU资源利用率不足31%真相曝光)——弹性伸缩动态水位算法实战手册
第一章:LLM服务SLA跌破99.2%?——GPU资源利用率不足31%的系统性归因 2026奇点智能技术大会(https://ml-summit.org) 当大模型推理服务的端到端SLA连续三小时低于99.2%,而监控平台却显示A100集群平均GPU利用率长期徘徊在28%–31%之间时&#…...
