LoginSignup
5

More than 5 years have passed since last update.

docker-composeで手軽にgroup replicationの試験環境構築をしてみる

Last updated at Posted at 2017-12-21

この記事はMySQL Casual Advent Calendar 2017の22日目です。

はじめに

huatoです。
MySQ 5.6のサポート期限が見えてきたので最近はMySQL5.7の動作試験を進めてます。
Group Replicationの検証目的で複数インスタンス立ち上げるのが面倒だったのでdocker-composeで手軽に
試験環境を構築してみました。
なおGroup Replication自体の説明はいろいろな方が記事を書かれているので除外してます。

docker周りの設定

CentOS7.3をインストールしてdocker動作に必要な設定をぽちぽちとすすめます。
ubuntuの環境インストール方法などもマニュアルが用意されてます。

yum install -y yum-utils device-mapper-persistent-data lvm2

yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo

yum-config-manager --disable docker-ce-edge
yum makecache fast

yum install docker-ce
cd /etc/
mkdir -p docker
vi daemon.json

{
  "storage-driver": "devicemapper"
}

{
        "dns": ["8.8.8.8", "8.8.4.4"]
}

dockerサービス起動
service docker restart

docker-compose

dockerコマンドでインスタンスごとに立ち上げていくこともできますが
複数環境を構築・管理するのであればdocker-composeが便利です。

1ライナーでインストール自体は終わります。

sudo curl -L https://github.com/docker/compose/releases/download/1.17.0/docker-compose-`uname -s`-`uname -m` -o /usr/local/bin/docker-compose
chmod 755 /usr/local/bin/docker-compose

各設定ファイルを配置する場所を作成します。
cd /root/
mkdir compose

構成管理に必要となるdocker-compose.ymlを用意します。
公式イメージを使うとパスワード設定方法などの考慮をしなくてよいので楽です。

設定はreference manualを元にしています。
https://dev.mysql.com/doc/refman/5.7/en/group-replication-configuring-instances.html
性能試験に関しては考慮していないためGroup Replication動作に必要となる最低限の設定のみです。

docker-compose.yml

version: '2'
services:
  node1:
     restart: always
     image: mysql:5.7
     networks:
          db:
           ipv4_address: 192.168.2.1
     expose:
      - "6606"
     volumes:
       - /data/mysql1:/var/lib/mysql
       - $PWD/node1.cnf:/etc/my.cnf
     privileged: true
     extra_hosts:
       - "node1:192.168.2.1"
       - "node2:192.168.2.2"
       - "node3:192.168.2.3"
     env_file: pass.env

  node2:
     restart: always
     image: mysql:5.7
     networks:
          db:
           ipv4_address: 192.168.2.2
     expose:
      - "6606"
     volumes:
       - /data/mysql2:/var/lib/mysql
       - $PWD/node2.cnf:/etc/my.cnf
     privileged: true
     extra_hosts:
       - "node1:192.168.2.1"
       - "node2:192.168.2.2"
       - "node3:192.168.2.3"
     env_file: pass.env

  node3:
     restart: always
     image: mysql:5.7
     networks:
          db:
           ipv4_address: 192.168.2.3
     expose:
      - "6606"
     volumes:
       - /data/mysql3:/var/lib/mysql
       - $PWD/node3.cnf:/etc/my.cnf
     privileged: true
     extra_hosts:
       - "node1:192.168.2.1"
       - "node2:192.168.2.2"
       - "node3:192.168.2.3"
     env_file: pass.env

  node5:
     restart: always
     image: mysql:5.7
     networks:
          db:
           ipv4_address: 192.168.2.5
     volumes:
       - /data/mysql5:/var/lib/mysql
       - $PWD/node5.cnf:/etc/my.cnf
     privileged: true
     extra_hosts:
       - "node1:192.168.2.1"
       - "node2:192.168.2.2"
       - "node3:192.168.2.3"
     env_file: pass.env

networks:
  db:
   ipam:
    config:
      - subnet: 192.168.0.0/22

pass.env

MYSQL_RANDOM_ROOT_PASSWORD=yes

各ノードごとにmy.cnfを用意します。

node1.cnf

[mysqld]
server_id=1
gtid_mode=ON
enforce_gtid_consistency=ON
binlog_checksum=NONE
log_slave_updates=ON
log-bin
binlog_format=ROW
binlog_error_action=IGNORE_ERROR
binlog_rows_query_log_events
binlog_row_image=minimal
character-set-server=utf8mb4
master_info_repository=TABLE
relay_log_info_repository=TABLE
innodb_buffer_pool_dump_at_shutdown=ON
innodb_buffer_pool_load_at_startup=OFF
innodb_strict_mode=0
innodb_checksum_algorithm=innodb
innodb_flush_method=O_DIRECT
log_timestamps=SYSTEM
log_error=/var/lib/mysql/mysql.err
transaction_write_set_extraction=XXHASH64
loose-group_replication_group_name="aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee"
loose-group_replication_start_on_boot=off
loose-group_replication_auto_increment_increment=1
loose-group_replication_local_address="192.168.2.1:6606"
loose-group_replication_group_seeds= "192.168.2.1:6606,192.168.2.2:6606,192.168.2.3:6606"
loose-group_replication_ip_whitelist="192.168.2.1,192.168.2.2,192.168.2.3,127.0.0.1"
loose-group_replication_bootstrap_group=off

node2.cnf,node3.cnfはnode1.cnfをコピー後
server_idの値の変更とloose-group_replication_local_addressを自身のIPに書き換えます。
my.cnfに明示的に記載していませんがauto_increment_offsetの値はgroup replication開始時にserver-idで上書きされていました。
回避策は現時点で無さそうですのでアプリケーションでauto_incrementを使っている場合は気にしたほうが良いかと。

参考:https://mysqlhighavailability.com/mysql-group-replication-auto-increment-configuration-handling/

起動
/usr/local/bin/docker-compose up -d
-dはバックグラウンド起動に必要となるオプションです。

各環境にログインします。
docker ps -a コマンドでインスタンスのNAMESを確認しましょう

CONTAINER ID        IMAGE               COMMAND                  CREATED             STATUS              PORTS                NAMES
ba96f05116fa        mysql:5.7           "docker-entrypoint..."   8 hours ago         Up 8 hours          3306/tcp             compose_node5_1
4eeef5032e12        mysql:5.7           "docker-entrypoint..."   8 hours ago         Up 8 hours          3306/tcp, 6606/tcp   compose_node3_1
c97ea26cc3a2        mysql:5.7           "docker-entrypoint..."   8 hours ago         Up 8 hours          3306/tcp, 6606/tcp   compose_node2_1
dd8c636252cb        mysql:5.7           "docker-entrypoint..."   8 hours ago         Up 8 hours          3306/tcp, 6606/tcp   compose_node1_1

docker exec -it NAMES /bin/bash で各環境にログインできます。
デタッチはCtrl+p+qです。
Ctrl+Cすると環境が停止されますので気をつけてください。

各環境ごとに実行するクエリが異なります。
mysqlのログイン時のrootパスワードはランダム生成のためログから確認します。

docker-compose logs | grep PASS

node2_1  | GENERATED ROOT PASSWORD: caid0aochifaeshoo5zaizooYai0kaet
node3_1  | GENERATED ROOT PASSWORD: Phahbahzoz6gee2ied0Gii9eihei9xif
node1_1  | GENERATED ROOT PASSWORD: shoa2ohChoh6eicheiBohyooxe2ooJei
node5_1  | GENERATED ROOT PASSWORD: zi9neeng1mue8ni3Oongiu2Pee4yaech

ログイン
mysql -uroot -p

【node1】

reset master;
SET SQL_LOG_BIN=0;
CREATE USER 'rpl_user'@'%';
GRANT REPLICATION SLAVE ON *.* TO 'rpl_user'@'%' IDENTIFIED BY 'dummy2';
FLUSH PRIVILEGES;
CHANGE MASTER TO MASTER_USER='rpl_user', MASTER_PASSWORD='dummy2' FOR CHANNEL 'group_replication_recovery';
INSTALL PLUGIN group_replication SONAME 'group_replication.so';
SET SQL_LOG_BIN=1;
SET GLOBAL group_replication_bootstrap_group=ON;
START GROUP_REPLICATION;
SET GLOBAL group_replication_bootstrap_group=OFF;

【node2,node3】

reset master;
SET SQL_LOG_BIN=0;
CREATE USER 'rpl_user'@'%';
GRANT REPLICATION SLAVE ON *.* TO 'rpl_user'@'%' IDENTIFIED BY 'dummy2';
FLUSH PRIVILEGES;
CHANGE MASTER TO MASTER_USER='rpl_user', MASTER_PASSWORD='dummy2' FOR CHANNEL 'group_replication_recovery';
INSTALL PLUGIN group_replication SONAME 'group_replication.so';
SET SQL_LOG_BIN=1;
START GROUP_REPLICATION;

5.7.19以前のMySQLバージョンから起動した場合、Group Replication起動時にエラーで失敗することがあります。
起動に失敗した場合はmysql_upgradeを実施することで解消されました。

mysql_upgrade -uroot -p

3インスタンスでの作業が終わったらマスターがどの環境になっているか確認してみます。
SHOW SLAVE STATUSでは確認できないためperformance_schemaの値を参照します。

select * from performance_schema.replication_group_members;

+---------------------------+--------------------------------------+--------------+-------------+--------------+
| CHANNEL_NAME              | MEMBER_ID                            | MEMBER_HOST  | MEMBER_PORT | MEMBER_STATE |
+---------------------------+--------------------------------------+--------------+-------------+--------------+
| group_replication_applier | d33b4161-d016-11e7-a59f-0242c0a80203 | 4eeef5032e12 |        3306 | ONLINE       |
| group_replication_applier | d3dabd9c-d016-11e7-a6db-0242c0a80201 | dd8c636252cb |        3306 | ONLINE       |
| group_replication_applier | d4c00f25-d016-11e7-a50d-0242c0a80202 | c97ea26cc3a2 |        3306 | ONLINE       |
+---------------------------+--------------------------------------+--------------+-------------+--------------+
3 rows in set (0.00 sec)

select member_host from performance_schema.global_status inner join performance_schema.replication_group_members where variable_name='group_replication_primary_member' AND member_id=variable_value;

+--------------+
| member_host  |
+--------------+
| dd8c636252cb |
+--------------+
1 row in set (0.00 sec)

最後にgroup replicationの構成をくみ上げたら各環境のmy.cnfを書き換えてください。
この設定を修正しないとサーバ再起動時にgroup replicationの設定が反映されずレプリケーショングループに自動的に戻りません。

loose-group_replication_start_on_boot=off
=>
group_replication_start_on_boot=on

Group Replicationが最大9インスタンスまでという制限があるため
Group Replicationの管理対象外のスレーブをGroup Replicationのマスターにぶら下げる場合についても書いておきます。

node5.cnf

[mysqld]
server_id=1
gtid_mode=ON
enforce_gtid_consistency=ON
binlog_checksum=NONE
log_slave_updates=ON
log-bin
binlog_format=ROW
binlog_error_action=IGNORE_ERROR
log_error=/var/lib/mysql/mysql.err
binlog_rows_query_log_events
binlog_row_image=minimal
character-set-server=utf8mb4
master_info_repository=TABLE
relay_log_info_repository=TABLE
innodb_buffer_pool_dump_at_shutdown=ON
innodb_buffer_pool_load_at_startup=OFF
innodb_strict_mode=0
innodb_checksum_algorithm=innodb
innodb_flush_method=O_DIRECT
log_timestamps=SYSTEM

【node5】

reset master;
CHANGE MASTER TO MASTER_USER='rpl_user', MASTER_PASSWORD='dummy2',MASTER_AUTO_POSITION = 1,MASTER_HOST='192.168.2.1';
START SLAVE;

通常のGTIDレプリケーションとして動作します。

mysql> show slave status\G;
*************************** 1. row ***************************
               Slave_IO_State: Waiting for master to send event
                  Master_Host: 192.168.2.1
                  Master_User: rpl_user
                  Master_Port: 3306
                Connect_Retry: 60
              Master_Log_File: dd8c636252cb-bin.000001
          Read_Master_Log_Pos: 1082
               Relay_Log_File: ba96f05116fa-relay-bin.000002
                Relay_Log_Pos: 1301
        Relay_Master_Log_File: dd8c636252cb-bin.000001
             Slave_IO_Running: Yes
            Slave_SQL_Running: Yes
              Replicate_Do_DB:
          Replicate_Ignore_DB:
           Replicate_Do_Table:
       Replicate_Ignore_Table:
      Replicate_Wild_Do_Table:
  Replicate_Wild_Ignore_Table:
                   Last_Errno: 0
                   Last_Error:
                 Skip_Counter: 0
          Exec_Master_Log_Pos: 1082
              Relay_Log_Space: 1507
              Until_Condition: None
               Until_Log_File:
                Until_Log_Pos: 0
           Master_SSL_Allowed: No
           Master_SSL_CA_File:
           Master_SSL_CA_Path:
              Master_SSL_Cert:
            Master_SSL_Cipher:
               Master_SSL_Key:
        Seconds_Behind_Master: 0
Master_SSL_Verify_Server_Cert: No
                Last_IO_Errno: 0
                Last_IO_Error:
               Last_SQL_Errno: 0
               Last_SQL_Error:
  Replicate_Ignore_Server_Ids:
             Master_Server_Id: 1
                  Master_UUID: d3dabd9c-d016-11e7-a6db-0242c0a80201
             Master_Info_File: mysql.slave_master_info
                    SQL_Delay: 0
          SQL_Remaining_Delay: NULL
      Slave_SQL_Running_State: Slave has read all relay log; waiting for more updates
           Master_Retry_Count: 86400
                  Master_Bind:
      Last_IO_Error_Timestamp:
     Last_SQL_Error_Timestamp:
               Master_SSL_Crl:
           Master_SSL_Crlpath:
           Retrieved_Gtid_Set: aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee:1-3
            Executed_Gtid_Set: aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee:1-3
                Auto_Position: 1
         Replicate_Rewrite_DB:
                 Channel_Name:
           Master_TLS_Version:
1 row in set (0.00 sec)

ERROR:
No query specified

docker-composeでのGroup Replication試験環境についての説明はこれで以上ですが
Group ReplicationだけではDB運用には不十分でアプリケーションからの接続をmysql routerやProxySQL等で管理しないといけません。
機会があればそちらも記事にしたいと思います。

参考にさせていただいた記事

http://datacharmer.blogspot.jp/2017/01/mysql-group-replication-installation.html
https://dev.mysql.com/doc/refman/5.7/en/group-replication-launching.html

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
What you can do with signing up
5