一、Amoeba概述

1、amoeba简介

     Amoeba是一个类似MySQL Proxy的分布式数据库中间代理层软件,是由陈思儒开发的一个开源的java项目。其主要功能包括读写分离,垂直分库,水平分库等,经过测试,发现其功能和稳定性都非常的不错,如果需要构架分布式数据库环境,采用Amoeba是一个不错的方案。目前Amoeba一共包括For aladdin,For MySQL和For Oracle三个版本,本文主要关注For MySQL版本的一个读写分离实现。 

    Amoeba处于在应用和数据库之间,扮演一个中介的角色,将应用传递过来的SQL语句经过分析后,将写的语句交给Master库执行,将读的语句路由到Slave库执行(当然也可以到Master读,这个完全看配置)。Amoeba实现了简单的负载均衡(采用轮询算法)和Failover。如果配置了多个读的库,则任何一个读的库出现宕机,不会导致整个系统故障,Amoeba能自动将读请求路由到其他可用的库上,当然,写还是单点的依赖于Master数据库的,这个需要通过数据库的切换,或者水平分割等技术来提升Master库的可用性。


2、amoeba与mysql-proxy的区别

        在MySQL proxy 6.0版本上面如果想要读写分离并且读集群、写集群机器比较多情况下,用mysql proxy 需要相当大的工作量,目前mysql proxy没有现成的 lua脚本。mysql proxy根本没有配置文件, lua脚本就是它的全部,当然lua是相当方便的。那么同样这种东西需要编写大量的脚本才能完成一 个复杂的配置。而Amoeba只需要进行相关的配置就可以满足需求。

wKiom1N56-_ThsOOAACHTdXnoRY687.jpg


二、实验环境

服务器角色IP地址安装软件功用操作系统
主服务器172.16.8.1mysql可读写
centos6.5
从服务器172.16.8.2mysql只可读,与主服务器做读的负载均衡centos6.5
读写分离服务器172.16.8.3amoeba实现读写分离,将写操作发往主服务器,将读操作进行负载均衡!centos6.5


三、amoeba的详细配置

1、主从服务器的配置

在主服务器上安装mysql

#yum -y install mysql mysql-server


  • 修改配置文件:

# vim /etc/my.cnf
//添加如下:
server-id       = 1


  • 启动mysql,并 创建授权账户

#service mysqld start
#mysql
mysql> grant replication slave,replication client on *.* to "gulong"@'172.16.8.2' identified by '123';
mysql> flush privileges;


从服务器上的也安装mysql

#yum -y install mysql mysql-server


  • 修改配置文件:

# vim /etc/my.cnf
//添加如下:
server-id       = 2
relay_log = relay_bin

 

  • 启动mysql,连接主服务器

#service mysqld start
#mysql
mysql> change master to master_host='172.16.8.1',master_user='gulong',master_password='123'


  • 查看主从状态:

mysql> start slave;
mysql> show slave status\G;

wKioL1N5_0-ikOXMAAL4ujA0L78776.jpg

到此,主从服务器配置好了!


2、amoeba的配置

首先下载jdk与amoeba安装包:

jdk

amoeba


  • 安装jdk:

#chmod +x jdk-6u31-linux-x64-rpm.bin 
#./jdk-6u31-linux-x64-rpm.bin 
#vim /etc/profile.d/java.sh
//添加如下内容:
export JAVA_HOME=/usr/java/latest
export PATH=$JAVA_HOME/bin:$PATH
#. /etc/profile.d/java.sh


查看java状态:

wKiom1N6AjOTCK1CAACLxY7v5ic182.jpg


安装amoeba:

#mkdir /usr/local/amoeba
#tar xf amoeba-mysql-binary-2.2.0.tar.gz -C /usr/local/amoeba/
#cd /usr/local/amoeba/

Amoeba for MySQL的使用是很简单的,主要是通过xml文件来实现的。
1. 配置文件介绍:
(1.) dbServers.xml           想象Amoeba作为数据库代理层,它一定会和很多数据库保持通信,因此它必须知道由它代理的数据库如何连接,比如最基础的:主机IP、端口、Amoeba使用的用户名和密码等等。这些信息存储在$AMOEBA_HOME/conf/dbServers.xml中。


(2.) rule.xml         Amoeba为了完成数据切分提供了完善的切分规则配置,为了了解如何分片数据、如何将数据库返回的数据整合,它必须知道切分规则。与切分规则相关的信息存储在$AMOEBA_HOME/conf/rule.xml中。


(3.) functionMap.xml          当我们书写SQL来操作数据库的时候,常常会用到很多不同的数据库函数,比如:UNIX_TIMESTAMP()、SYSDATE()等等。这些函数如何被Amoeba解析呢?$AMOEBA_HOME/conf/functionMap.xml描述了函数名和函数处理的关系。


(4.) ruleFunctionMap.xml          对$AMOEBA_HOME/conf/rule.xml进行配置时,会用到一些我们自己定义的函数,比如我们需要对用户ID求HASH值来切分数据,这些函数在$AMOEBA_HOME/conf/ruleFunctionMap.xml中定义。


(5.) access_list.conf                 Amoeba可以制定一些可访问以及拒绝访问的主机IP地址,这部分配置在$AMOEBA_HOME/conf/access_list.conf中。


(6.) log4j.xml                          Amoeba允许用户配置输出日志级别以及方式,配置方法使用log4j的文件格式,文件是$AMOEBA_HOME/conf/log4j.xml。
其中,我们主要用到dbServer.xml 和 amoeba.xml 。


  • 修改ameoba.xml配置文件

[root@node1 ~]# cd /usr/local/amoeba/conf/
[root@node1 conf]# vim amoeba.xml

wKiom1N6BGSheZzKAAOu5Zluq3k515.jpg

wKioL1N6CE6wO8zYAASyhlALdtg922.jpg


  • 修改dbServers.xml 配置文件

[root@node1 conf]# vim dbServers.xml 
//修改如下:
<?xml version="1.0" encoding="gbk"?>

<!DOCTYPE amoeba:dbServers SYSTEM "dbserver.dtd">
<amoeba:dbServers xmlns:amoeba="http://amoeba.meidusa.com/">

                <!-- 
                        Each dbServer needs to be configured into a Pool,
                        If you need to configure multiple dbServer with load balancing that can be simplified by the following configuration:
                         add attribute with name virtual = "true" in dbServer, but the configuration does not allow the element with name factoryConfig
                         such as 'multiPool' dbServer   
                -->

        <dbServer name="abstractServer" abstractive="true">
                <factoryConfig class="com.meidusa.amoeba.mysql.net.MysqlServerConnectionFactory">
                        <property name="manager">${defaultManager}</property>
                        <property name="sendBufferSize">64</property>
                        <property name="receiveBufferSize">128</property>

                        <!-- mysql port -->
                        <property name="port">3306</property>

                        <!-- mysql schema -->
                        <property name="schema">test</property>

                        <!-- mysql user -->
                        //设置连接后端数据库的账户密码,需要在主mysql上授权
                        <property name="user">root</property> 
                        <property name="password">admin</property>
                        <!--  mysql password
                        <property name="password">password</property>
                        -->
                </factoryConfig>

                <poolConfig class="com.meidusa.amoeba.net.poolable.PoolableObjectPool">
                        <property name="maxActive">500</property>
                        <property name="maxIdle">500</property>
                        <property name="minIdle">10</property>
                        <property name="minEvictableIdleTimeMillis">600000</property>
                        <property name="timeBetweenEvictionRunsMillis">600000</property>
                        <property name="testOnBorrow">true</property>
                        <property name="testOnReturn">true</property>
                        <property name="testWhileIdle">true</property>
                </poolConfig>
        </dbServer>
         
         //将172.16.8.1添加为master主服务器
        <dbServer name="master"  parent="abstractServer">
                <factoryConfig>
                        <!-- mysql ip -->
                        <property name="ipAddress">172.16.8.1</property>
                </factoryConfig>
        </dbServer>

        //172.16.8.2添加为从的slave服务器
        <dbServer name="slave"  parent="abstractServer">
                <factoryConfig>
                        <!-- mysql ip -->
                        <property name="ipAddress">172.16.8.2</property>
                </factoryConfig>
        </dbServer>

         //将master与slave添加为一个mysql集群,命名为readservers,采用轮询的负载均衡机制
        <dbServer name="readservers" virtual="true">
                <poolConfig class="com.meidusa.amoeba.server.MultipleServerPool">
                        <!-- Load balancing strategy: 1=ROUNDROBIN , 2=WEIGHTBASED , 3=HA-->
                        <property name="loadbalance">1</property>
//1代表轮询
                        <!-- Separated by commas,such as: server1,server2,server1 -->
                        <property name="poolNames">master,slave</property>
                </poolConfig>
        </dbServer>

</amoeba:dbServers>


  • 在主mysql服务器上授权远程连接账户:

mysql> grant all on *.* to "root"@'172.16.%.%' identified by 'admin';
mysql> flush privileges;



  • 启动amoeba,并测试:

[root@node1 bin]# cd /usr/local/amoeba/
[root@node1 bin]# ./amoeba start
[root@node1 ~]# mysql -uroot -pmypass -h172.16.8.3

wKiom1N6C1SS6caAAAF2uxH1thY499.jpgwKioL1N6C2-RYrSWAAIHTG648mw897.jpg


3、测试:

连接amoeba

创建数据库,

MySQL [mysql]> create database wb;//创建数据库
MySQL [mysql]> select user,password from mysql.user;//查询数据库
MySQL [mysql]> show databases;

可以通过使用tcpdump在master、slave服务器上抓包,创建一个数据库,并多使用几次查询语句,验证写操作是否被调度至master(172.16.8.1),而读操作会以轮叫的方式调度至master和slave,可以分摊mysql的读请求!

[root@localhost ~]# tcpdump -i eth0 -nn -XX ip dst host 172.16.8.1 and tcp dst port 3306
[root@localhost ~]# tcpdump -i eth0 -nn -XX ip dst host 172.16.8.2 and tcp dst port 3306

wKiom1N6ELahVl17AAXEoqXrFJg612.jpgwKioL1N6EL7Rw8MbAAeuR2GzYXs528.jpg


至此,基于amoeba的读写分离已经介绍完毕,不足之处,请多加指点!