下面的扩展代码基于redis 5.0.2进行扩展, 对于其他的redis版本, 我没有进行相关的测试。考虑到redis集群的修改频率,这段代码应该同时适用于其他的redis版本。

下面为修改的代码:

static clusterManagerNode *clusterManagerByIpPort(const char* ip, int port);
static int clusterManagerCommandMoveSlot(int argc, char **argv);
{"move-slot", clusterManagerCommandMoveSlot, ,
"from-host:from-port to-host:to-port slotNo", NULL},
// 根据ip和端口号来获取对应的redis集群节点
clusterManagerNode *clusterManagerByIpPort(const char* ip, int port)
{
if (cluster_manager.nodes == NULL) return NULL;
clusterManagerNode *found = NULL;
sds lcip = sdsempty();
lcip = sdscpy(lcip, ip);
sdstolower(lcip);
listIter li;
listNode *ln;
listRewind(cluster_manager.nodes, &li);
while ((ln = listNext(&li)) != NULL) {
clusterManagerNode *n = ln->value;
if (n->ip && !sdscmp(n->ip, lcip) && (n->port == port)) {
found = n;
break;
}
}
sdsfree(lcip);
return found;
}
// 在redis集群中实现移动槽位功能
static int clusterManagerCommandMoveSlot(int argc, char **argv) {
assert(argc == ); char *to_ip = NULL, *from_ip = NULL;
int to_port = , from_port = ;
if (!getClusterHostFromCmdArgs(, argv + , &to_ip, &to_port))
goto invalid_args;
if (!getClusterHostFromCmdArgs(, argv, &from_ip, &from_port))
goto invalid_args;
char *end = NULL;
char *slotStr = argv[];
int slotNo = strtol(slotStr, &end, );
if ((size_t)(end - slotStr) != strlen(slotStr)) {
clusterManagerLogErr("*** the slotNo should be number.\n");
return ;
}
clusterManagerLogInfo("Moving slot number: %d from node %s:%d to node %s:%d\n",
slotNo, from_ip, from_port, to_ip, to_port); clusterManagerNode *tonode = clusterManagerNewNode(to_ip, to_port);
if (!clusterManagerLoadInfoFromNode(tonode, )) return ; clusterManagerCheckCluster();
if (cluster_manager.errors && listLength(cluster_manager.errors) > ) {
fflush(stdout);
fprintf(stderr,
"*** Please fix your cluster problems before moving slot\n");
return ;
} clusterManagerNode *fromnode = clusterManagerByIpPort(from_ip, from_port);
// check fromnode and tonode is valid
const char *invalid_node_msg = "*** The specified node (%s:%d) is not known "
"or not a master, please retry.\n"; if (!fromnode || (fromnode->flags & CLUSTER_MANAGER_FLAG_SLAVE)) {
clusterManagerLogErr(invalid_node_msg, from_ip, from_port);
return ;
} else if (!tonode || (tonode->flags & CLUSTER_MANAGER_FLAG_SLAVE)) {
clusterManagerLogErr(invalid_node_msg, to_ip, to_port);
return ;
} else if (!strcmp(fromnode->name, tonode->name) && (fromnode->port == tonode->port)){
clusterManagerLogErr( "*** It is not possible to use "
"the target node as "
"source node.\n");
return ;
} // check fromnode has the slot
if (slotNo < || slotNo >= CLUSTER_MANAGER_SLOTS) {
// Don't give a range hint for using it when someone is familiar with redis
clusterManagerLogErr("*** The slotNo is invalid, slotNo: %d\n", slotNo);
return ;
} else if (fromnode->slots[slotNo] == (uint8_t)()) {
clusterManagerLogErr("*** From node don't have the slot: %d\n", slotNo);
return ;
} int opts = CLUSTER_MANAGER_OPT_VERBOSE | CLUSTER_MANAGER_OPT_UPDATE;
char *err = NULL;
int result = clusterManagerMoveSlot(fromnode, tonode, slotNo,
opts, &err);
if (!result) {
if (err != NULL) {
//clusterManagerLogErr("\n%s\n", err);
zfree(err);
}
} else {
clusterManagerLogInfo("Moving finished with success.\n");
} return result; invalid_args:
fprintf(stderr, CLUSTER_MANAGER_INVALID_HOST_ARG);
return ;
}

上述代码, 就是新增的在集群中移动槽位的功能, 使用方法如下:

(1)在redis-cli.c文件中添加上述代码

(2)在redis根目录调用make命令, 编译代码

(3)对编译生成的redis-cli执行如下的命令:

  $ src/redis-cli --cluster help

其中的move-slot from-host:from-port to-host:to-port slotNo就是新增的子命令。假设192.168.5.1:6379分配了槽位0-5460,192.168.5.2:6379分配了槽位5461-10922,192.168.5.3:6379分配了槽位10923-16383, 通过调用src/redis-cli --cluster move-slot 192.168.5.1:6379 192.168.5.2:6379 10,这样192.168.5.1:6379当前拥有槽位0-9,11-5460,而192.168.5.2:6379拥有槽位10,5461-10922。

对于redis5.0.2的redis-cli.c文件的修改后的文件可以参考:

https://github.com/ss-torres/redis-cli-addition.git

如果有什么好的建议或者提议,欢迎提出。

通过扩展redis-cli来实现在redis集群中移动槽位的更多相关文章

  1. 通过memcached来实现对tomcat集群中Session的共享策略

    近期在做一套集群的实现,实现的方案是在Linux下完成对Apache + Tomcat 负载均衡的功能. 上述功能已经实现,有需要了解的朋友可以看我另外一篇博文. Linux下Apache与Tomca ...

  2. Redis系列5:深入分析Cluster 集群模式

    Redis系列1:深刻理解高性能Redis的本质 Redis系列2:数据持久化提高可用性 Redis系列3:高可用之主从架构 Redis系列4:高可用之Sentinel(哨兵模式) 1 背景 前面我们 ...

  3. redis之(十八)redis的支持水平扩容的集群特性,以及插槽的相关操作

    [一]主从集群的缺点,客户端分片的缺点 (1)主从+哨兵的redis集群,只是做主从备份,数据冗余的一种处理.但在存储空间的扩展上还是有限制.因为集群中的节点都是存储同样的数据.单一节点的容量,就可以 ...

  4. redis 与java的连接 和集群环境下Session管理

    redis 的安装与设置开机自启(https://www.cnblogs.com/zhulina-917/p/11746993.html)  第一步: a) 搭建环境 引入 jedis jar包 co ...

  5. 使用redis进行基于shiro的session集群共享

    之前写过一篇nginx多tomcat负载均衡,主要记录了使用nginx对多个tomcat 进行负载均衡,其实进行负载均衡之前还有一个问题没有解决,那就是集群间的session共享,不然用户在登录网站之 ...

  6. Redis数据库 02事务| 持久化| 主从复制| 集群

    1. Redis事务 Redis不支持事务,此事务不是关系型数据库中的事务: Redis事务是一个单独的隔离操作:事务中的所有命令都会序列化.按顺序地执行.事务在执行的过程中,不会被其他客户端发送来的 ...

  7. Linux下redis 的部署、主从与集群

    老男孩Python全栈6期——redis--------------------------Linux 操作系统 默认的内存管理机制RSS:page cache:anno page:Linux操作系统 ...

  8. Linux--6 redis订阅发布、持久化、集群cluster、nginx入门

    一.redis发布订阅 Redis 通过 PUBLISH .SUBSCRIBE 等命令实现了订阅与发布模式. 其实从Pub/Sub的机制来看,它更像是一个广播系统,多个Subscriber可以订阅多个 ...

  9. Redis安装(单机及各类集群,阿里云)

    Redis安装(单机及各类集群,阿里云) 前言 上周,我朋友突然悄悄咪咪地指着手机上的一篇博客说,这是你的博客吧.我看了一眼,是之前发布的<Rabbit安装(单机及集群,阿里云>.我朋友很 ...

随机推荐

  1. h5开发安卓机型点击输入框调起输入法,输入框被键盘遮挡的解决方法

    前言: 从以前的项目中找一个问题的解决方案,顺带找到了这个安卓机型调起输入法,页面没有自动上滑导致输入框被弹起的键盘遮挡的解决方案.这个问题只有安卓机型页面中的输入框处于底部(也就是底部键盘区域)的时 ...

  2. sass、less和stylus 相同与不同

    sass.less和stylus的安装使用和入门实践 https://www.jianshu.com/p/1eaf366814e2 stylus 基础教程 https://blog.csdn.net/ ...

  3. 002-maven开发Java脚手架archrtype【如无定制开发,请直接看3.3使用】

    一.概述 项目基础构建需要:项目结构,spring框架,orm,连接池,数据库,单元测试等等. 上述即使复用:001-脚手架发展,基础代码结构+mybatis代码生成,基础代码结构,也需要修改成自己单 ...

  4. 【Linux】walle 部署上线单报错:mv: cannot overwrite directory ‘/www’ with non-directory

    错误截图 问题分析:项目设置中. 目标集群部署路径错误, 举例: 假设你 项目名称:laofan 在目标服务器的路径: /www/wwwdata/laofan 那么你在标集群部署路径 就可以写: /w ...

  5. wamp64显示黄色图标不能忍

    哎,昨天硬盘合区了下,重新安装了wamp64,删库的时候忘记备份数据库,灾难啊,只能自己重新建库建表了,深刻的教训啊. 然后还启动后是黄色图标,不能忍啊,发现wamp64需要启动三个服务,mysql, ...

  6. Delphi ADOConnection连接

    unit Unit_DM; interface usesSysUtils, Classes, DB, ADODB,inifiles,windows,forms,controls; typeTDM = ...

  7. 【ARM-Linux开发】ubuntu查看文件大小

    使用Linux命令df 和du,df 但是df只能查看一级文件夹大小.使用比例.档案系统及其挂入点,但对文件却无能为力.du可以查看文件及文件夹的大小.所以基本上是两者配合使用. 一 df h参数, ...

  8. Appium元素定位难点:tap坐标定位不准确

    tap用法 1.tap是模拟手指点击页面上元素语法有两个参数,第一个是positions,是list类型最多五个点,duration是持续时间,单位毫秒 tap(self, positions, du ...

  9. Python之让 字符串内的转义字符 不做任何处理

    一.在字符串前面加上 'r' 就可以了 print("\ntext_1") print(r"\ntest_2") 二.在转义字符的 '\' 前面再加一个 '\' ...

  10. Linux删除含有特殊符号文件名的文件

    1. 文件名含有特殊字符,直接使用 rm 可能删除不了,可以使用如下方法: 1) 使用 ls -i 查处该文件的 inode 号,假设为123    2) 使用find命令删除: rm `find . ...