1.  背景

例如,存在一套redis主从(主从节点在不同的主机上),应用程序通过主库的ip进行读写操作。 但是,主库一旦出现故障,虽然有从库,且从库提升为主库,但是应用程序如果想使用从库则必须修改配置,重启应用方可生效。如用此情况,则涉及的人员比较多,且应用程序恢复使用的时间比较长。对于此情况,可以采取以下2种解决方式解决:

a)  配置VIP

在Redis主库服务器上配置vip,当主库出现问题时,配置脚本将vip自动切换至从节点,并将从节点提升为读写状态。应用程序中配置的是vip,主库异常时,从库自动提升为主库对外提供服务,应用程序无需做任何操作。

b)  使用DNS

应用程序通过配置内网域名连接redis,DNS服务器对应域名映射到redis主库服务器IP。 当redis主库异常时,将redis从节点提升为读写主库,修改DNS域名映射关系至redis从节点ip,此时应用程序也无需进行操作。

注:

以上2种方式均存在缺陷,例如:

  • vip方式存在问题:  当主从节点不在同一个机房或同一网段时,将无法使用相同的vip。
  • DNS方式: 使用DNS方式将有DNS缓存问题,即修改域名映射后域名仍可能解析到原主库机器的ip。

vip配合哨兵的高可用方式将在后续介绍,本次先介绍DNS服务器配置及dns方式解决方案。

2.  DNS服务配置

2.1  安装DNS服务

# 安装bind相关工具
yum install bind bind-utils bind-devel bind-libs bind-chroot -y

2.2  修改配置文件

vim /etc/named.conf
## 编译对应内容
// named.conf
//
// Provided by Red Hat bind package to configure the ISC BIND named(8) DNS
// server as a caching only nameserver (as a localhost DNS resolver only).
//
// See /usr/share/doc/bind*/sample/ for example named configuration files.
// options {
// listen-on port 53 { 127.0.0.1; }; // 此行注释
listen-on port { any; }; // 添加此行
listen-on-v6 port { ::; };
directory "/var/named";
dump-file "/var/named/data/cache_dump.db";
statistics-file "/var/named/data/named_stats.txt";
memstatistics-file "/var/named/data/named_mem_stats.txt";
// allow-query { localhost; }; // 注释此行
allow-query { any; }; // 添加此行
forwarders {114.114.114.114; }; // 添加此行,这是在DNS服务器不知道域名解析的时候询问这个IP的主机,这个IP的主机必须联网
recursion yes; dnssec-enable yes;
dnssec-validation yes; /* Path to ISC DLV key */
bindkeys-file "/etc/named.iscdlv.key"; managed-keys-directory "/var/named/dynamic";
}; logging {
channel default_debug {
file "data/named.run";
severity dynamic;
};
}; zone "." IN {
type hint;
file "named.ca";
}; include "/etc/named.rfc1912.zones";
include "/etc/named.root.key";

2.3   检查语法

named-checkconf

2.4   启动DNS服务

/etc/init.d/named  start

2.5  配置DNS正向解析文件

a)  在/etc/named.rfc1912.zones添加正向解析配置

vim /etc/named.rfc1912.zones  # 此文件名在上面步骤的/etc/named.conf文件末尾有指定

##  此文件末尾追加如下内容

zone "redis.com" IN {                         //  redis.com 名自定义,即需要解析的域名
type master; // dns域类型为master
file "redis.com.zone"; // redis.com.zone 文件名自定义,后续文件名需与此一致
allow-update { none; };
};

b)  根据上一步的情况,配置解析文件

# 拷贝文件
cp -p named.localhost redis.com.zone # 拷贝文件,注意要连同权限一起拷贝,因权限不一致,启动会报错 vim redis.com.zone $TTL 1D
@ IN SOA www.redis.com. rname.invalid. (
; serial
1D ; refresh
1H ; retry
1W ; expire
3H ) ; minimum
NS www.redis.com. dns IN A 192.168.56.208
www IN A 192.168.56.208

2.6  配置DNS反向解析文件

a)  在/etc/named.rfc1912.zones添加反向解析配置

可以将正向解析与反向解析配置在一个文件里,即file配置为相同的文件名。本次分开配置来演示

vim /etc/named.rfc1912.zones  # 此文件名在上面步骤的/etc/named.conf文件末尾有指定

##  此文件末尾追加如下内容

zone "56.168.192.in-addr.arpa" IN {
type master;
file "redis.com.local";
allow-update { none; };
};

b)   根据上一步的情况,配置解析文件

# 拷贝文件
cp -p named.localhost redis.com.local # 拷贝文件,注意要连同权限一起拷贝,因权限不一致,启动会报错 vim redis.com.local $TTL 1D
@ IN SOA www.redis.com. rname.invalid. (
; serial
1D ; refresh
1H ; retry
1W ; expire
3H ) ; minimum
NS @
A 192.168.56.208
AAAA ::
IN PTR www.redis.com.

2.7  重启DNS服务

/etc/init.d/named restart

2.8  测试DNS服务器

在另一台主机上测试DNS是否可用,操作步骤如下。

# 修改域名解析文件
vim /etc/resolv.conf # 将创建的DNS服务器地址添加至此文件
nameserver 192.168.56.209

 正向解析测试

 nslookup www.redis.com
# 结果如下
Server: 192.168.56.209
Address: 192.168.56.209# Name: www.redis.com
Address: 192.168.56.208

反向解析测试

nslookup  192.168.56.208
# 结果如下:
Server: 192.168.56.209
Address: 192.168.56.209# 208.56.168.192.in-addr.arpa name = www.redis.com.

 ping 域名测试

 ping  www.redis.com
# 结果如下
PING www.redis.com (192.168.56.208) () bytes of data.
bytes from www.redis.com (192.168.56.208): icmp_seq= ttl= time=0.229 ms
bytes from www.redis.com (192.168.56.208): icmp_seq= ttl= time=0.287 ms
bytes from www.redis.com (192.168.56.208): icmp_seq= ttl= time=0.276 ms
bytes from www.redis.com (192.168.56.208): icmp_seq= ttl= time=0.224 ms

至此,DNS服务器搭建并测试完毕,下面进入正题。

3.  搭建Redis主从

关于redis搭建之前的文章已经介绍过,详细信息可参考历史文章https://www.cnblogs.com/gjc592/p/11098047.html。

3.1  搭建主、从节点redis实例,部署过程完全一致

a)  依赖包安装

yum -y install cpp binutils glibc glibc-kernheaders glibc-common glibc-devel gcc make gcc-c++ libstdc++-devel tcl

b)  安装包准备

可以从官网  https://redis.io下载最新版redis

wget  http://download.redis.io/releases/redis-4.0.14.tar.gz
tar -zxvf redis-4.0.14.tar.gz

c)   编译、安装

cd   redis-4.0.14
make
make install

d) 创建目录、修改配置文件、启动redis

建议创建单独的redis目录

#  创建目录
mkdir -p /data/redis/redis6379 # 拷贝配置文件
cp redis.conf /data/redis/redis6379/ # 修改配置文件
vim redis.conf
修改如下部分
bind 0.0.0.0 可以指定所有地址均可访问,若指定对应网段或IP 修改此处即可
daemonize yes 放在后台执行,建议修改为yes
pidfile /data/redis/redis6379/redis_6379.pid 指定pid文件目录及文件名
logfile "/data/redis/redis6379/redis6379.log" 指定log文件目录及文件名 # 其他参数在生产环境中可适当调整 # 启动redis
redis-server redis.conf

 3.2  配置主从

在从服务器执行如下命令配置主从

127.0.0.1:> slaveof   192.168.56.208  6379    ##  即输入对应的redis主库的ip 即端口

查看主从状态

127.0.0.1:> info Replication
## 结果如下
# Replication
role:slave
master_host:192.168.56.208
master_port:
master_link_status:up // up代表已正常同步
master_last_io_seconds_ago:
master_sync_in_progress:
slave_repl_offset:
slave_priority:
slave_read_only:
connected_slaves:
master_replid:22830eb406e63f0a85d3d912a44e1b80dba6c860
master_replid2:
master_repl_offset:
second_repl_offset:-
repl_backlog_active:
repl_backlog_size:
repl_backlog_first_byte_offset:
repl_backlog_histlen:

至此,redis主从同步配置完成

4.  测试域名方式操作redis

编写程序,测试使用域名方式连接redis

注意:程序运行机器需添加对应的内网DNS服务器,即2.8中的操作。本次使用python进行测试

4.1  安装Python所需的包-- redis

python连接redis需安装redis包,关于Python升级,pip安装等历史文章里均有,如有需要可以参考操作

pip  install redis

4.2  编写简单的Python操作redis的测试程序

 vim test_redis.py 

#  内容如下

#!/usr/bin/python
# coding=utf-8 import redis
v_ip ='www.redis.com'
v_port = 6379
v_passwd=''
r = redis.Redis(host=v_ip,port=v_port,password=v_passwd,db=0)
r.set('test_key1','test1')
result = r.get('test_key1')
print result ,"设置键成功并获取到values"
r.delete('test_key1')
print "删除键完毕"
result1 = r.get('test_key1')
print result1,"验证删除成功"

4.3   运行测试程序,看是否能通过域名操作成功

 python test_redis.py
# 结果如下
test1 设置键成功并获取到values
删除键完毕
None 验证删除成功

有图有真相

说明,使用域名操作redis正常。

5.  模拟redis主库异常

5.1  关闭主库

127.0.0.1:> shutdown
not connected> exit

5.2  测试程序使用redis情况

python test_redis.py
## 报错
Traceback (most recent call last):
File "test_redis.py", line , in <module>
r.set('test_key1','test1')
File "/usr/local/python2.7/lib/python2.7/site-packages/redis/client.py", line , in set
return self.execute_command('SET', *pieces)
File "/usr/local/python2.7/lib/python2.7/site-packages/redis/client.py", line , in execute_command
connection = pool.get_connection(command_name, **options)
File "/usr/local/python2.7/lib/python2.7/site-packages/redis/connection.py", line , in get_connection
connection.connect()
File "/usr/local/python2.7/lib/python2.7/site-packages/redis/connection.py", line , in connect
raise ConnectionError(self._error_message(e))
redis.exceptions.ConnectionError: Error connecting to www.redis.com:. Connection refused.

即,此时redis已无法使用

5.3   提升从库为读写库

从库默认为只读,断开主从复制后将会变为读写库

a)  查看此时从库复制状态

127.0.0.1:> info Replication
# Replication
role:slave
master_host:192.168.56.208
master_port:
master_link_status:down // 主从同步已断开
master_last_io_seconds_ago:-
master_sync_in_progress:
slave_repl_offset:
master_link_down_since_seconds:
slave_priority:
slave_read_only:
connected_slaves:
master_replid:22830eb406e63f0a85d3d912a44e1b80dba6c860
master_replid2:
master_repl_offset:
second_repl_offset:-
repl_backlog_active:
repl_backlog_size:
repl_backlog_first_byte_offset:
repl_backlog_histlen:

b)  断开主从同步

127.0.0.1:> slaveof  no one
OK
127.0.0.1:> info Replication
# Replication
role:master // 断开后,已变成主库
connected_slaves:
master_replid:180df5fbdc8cf8999b27ad42e6c57eb3be31b6b2
master_replid2:22830eb406e63f0a85d3d912a44e1b80dba6c860
master_repl_offset:
second_repl_offset:
repl_backlog_active:
repl_backlog_size:
repl_backlog_first_byte_offset:
repl_backlog_histlen:

因此时域名仍指向原主库,所以程序依旧异常。

6. 切换域名指向

6.1 修改配置文件

将DNS服务中对应域名的IP地址改为从库地址

 vim redis.com.zone
## 修改
$TTL 1D
@ IN SOA www.redis.com. rname.invalid. (
; serial
1D ; refresh
1H ; retry
1W ; expire
3H ) ; minimum
NS www.redis.com. dns IN A 192.168.56.207
www IN A 192.168.56.207 vim redis.com.local
# 修改后
$TTL 1D
@       IN SOA  www.redis.com. rname.invalid. (
                                        0       ; serial
                                        1D      ; refresh
                                        1H      ; retry
                                        1W      ; expire
                                        3H )    ; minimum
        NS      @
        A       192.168.56.207
        AAAA    ::1
207     IN PTR www.redis.com.

6.2  重启DNS服务或刷新缓存

本次测试直接重启DNS服务

/etc/init.d/named  restart

6.3  简单测试域名解析情况

nslookup www.redis.com   #正向测试DNS
# 结果如下
nslookup www.redis.com
Server:        192.168.56.209
Address:    192.168.56.209#53 Name:    www.redis.com
Address: 192.168.56.207

说明已修改成功

7.  最终测试

再次使用python 程序测试操作redis情况

python test_redis.py
# 运行结果
test1 设置键成功并获取到values
删除键完毕
None 验证删除成功

此时应用程序未做任何修改,可以正常使用。

ps:

以上测试步骤中部分有省略,如果错误,欢迎指正。

耿小厨已开通个人微信公众号,想进一步沟通或想了解其他文章的同学可以关注我

redis高可用之DNS篇的更多相关文章

  1. Redis 高可用篇:你管这叫主从架构数据同步原理?

    在<Redis 核心篇:唯快不破的秘密>中,「码哥」揭秘了 Redis 五大数据类型底层的数据结构.IO 模型.线程模型.渐进式 rehash 掌握了 Redis 快的本质原因. 接着,在 ...

  2. Windows版本redis高可用方案探究

    目录 Windows版本redis高可用方案探究 前言 搭建redis主从 配置主redis-28380 配置从redis-23381 配置从redis-23382 将redis部署为服务 启动red ...

  3. Redis高可用详解:持久化技术及方案选择

    文章摘自:https://www.cnblogs.com/kismetv/p/9137897.html 前言 在上一篇文章中,介绍了Redis的内存模型,从这篇文章开始,将依次介绍Redis高可用相关 ...

  4. 如何构建 Redis 高可用架构?

    温国兵 民工哥技术之路 今天 1 .题记 Redis 是一个开源的使用 ANSI C 语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value 数据库,并提供多种语言的 API. 如今,互 ...

  5. Redis高可用集群-哨兵模式(Redis-Sentinel)搭建配置教程【Windows环境】

    No cross,no crown . 不经历风雨,怎么见彩虹. Redis哨兵模式,用现在流行的话可以说就是一个"哨兵机器人",给"哨兵机器人"进行相应的配置 ...

  6. Redis高可用之集群配置(六)

    0.Redis目录结构 1)Redis介绍及部署在CentOS7上(一) 2)Redis指令与数据结构(二) 3)Redis客户端连接以及持久化数据(三) 4)Redis高可用之主从复制实践(四) 5 ...

  7. Redis高可用之哨兵模式Sentinel配置与启动(五)

    0.Redis目录结构 1)Redis介绍及部署在CentOS7上(一) 2)Redis指令与数据结构(二) 3)Redis客户端连接以及持久化数据(三) 4)Redis高可用之主从复制实践(四) 5 ...

  8. Redis高可用之主从复制实践(四)

    0.Redis目录结构 1)Redis介绍及部署在CentOS7上(一) 2)Redis指令与数据结构(二) 3)Redis客户端连接以及持久化数据(三) 4)Redis高可用之主从复制实践(四) 5 ...

  9. Redis高可用架构—Keepalive+VIP

    最近整理一下Redis高可用架构的文档,也准备分享出来,虽然这些架构也不是很复杂.Redis的高可用方案目前主要尝试过5种方式,其中2种方式已经在线上使用. 1)Redis Master-Slave ...

随机推荐

  1. MyBatis 核心配置综述之Executor

    目录 MyBatis四大组件之 Executor执行器 Executor的继承结构 Executor创建过程以及源码分析 Executor接口的主要方法 Executor 的现实抽象 上一篇我们对Sq ...

  2. 提高JavaScript 技能的12个概念

    JavaScript 是一种复杂的语言.如果是你是高级或者初级 JavaScript 开发人员,了解它的基本概念非常重要.本文介绍 JavaScript 至关重要的12个概念,但绝对不是说 JavaS ...

  3. Bzoj 2563: 阿狸和桃子的游戏 题解

    2563: 阿狸和桃子的游戏 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 970  Solved: 695[Submit][Status][Discu ...

  4. C++小游戏——井字棋

    #include<cstdio> #include<windows.h> #include<ctime> int main() { srand(time(NULL) ...

  5. [小米OJ] 4. 最长连续数列

    思路: 时间限制为O(n),即不能使用先排序后寻找的方法. 这里利用哈希表查询插入复杂度都为O(1)的特性来解,利用一个哈希表来保存每一个数字以及其所在数列的长度. 遍历每一个数字n:查询表中是否存在 ...

  6. 小白开学Asp.Net Core 《九》

    小白开学Asp.Net Core <九> — — 前端篇(不务正业) 在<小白开学Asp.Net Core 三>中使用了X-admin 2.x 和 Layui将管理后端的界面重 ...

  7. 一个简单的JS倒计时

    看到很多商城都是抢购倒计时的功能,今天闲来无事做了个倒计时.全当学习JS. 主要思路:主要用到Date对象,声明一个变量获取当前时间,在声明一个变量获取结束时间,结束时间-当前时间=剩余时间(倒计时) ...

  8. python元类深入理解

    1.python 中的类 在python中,类也是一个对象,只不过这个对象拥有生成实例的能力,我们一般使用class XXX来定义一个类,在python解释器执行到这个地方的时候会自动创建出这个对象, ...

  9. JSP数据交互(一)

    1.JSP内置对象:JSP内置对象是 Web 容器创建的一组对象,不用通过手动new就可以使用 2.JSP9大内置对象:      对象名称 类型 request (请求对象) javax.servl ...

  10. Jquery第一次考核

    1. 什么是JS JavaScript 缩写.一种计算机脚本语言 JavaScript是一种动态.弱类型.基于原型的语言,通过浏览器可以直接执行 2. JS三大组成部件 ECMAScript DOM ...