16
20

More than 5 years have passed since last update.

Ansibleで本番レベルのMySQLレプリケーションを構築して見よう。

Last updated at Posted at 2017-11-01

はじまる前に

この記事では主にAnsibleの機能を紹介したいと思います。
playbookに記述したMySQLの詳細を知りたい方は次の記事を参照してください。

やりたいこと

実際のサービス環境でMySQLを運用する場合、このくらいは必要だと思ったことを書いてみた。

  • master1台とslave2台のレプリケーション構成
  • スロークエリを残す。
  • MySQLをstrictモードで使う。

これらをAnsibleのplaybookで作成する。

ファイル構成

# mysqlのrole
roles/mysql
├── handlers
│   └── main.yml
├── tasks
│   └── main.yml
└── vars
    └── main.yml

# dbグループの変数設定
group_vars
└── db.yml

# 各dbの変数設定
# master: db1
# slave:  db2, db3
host_vars
├── db1.yml
├── db2.yml
└── db3.yml

# hosts.ini
[db]
db[1:3]

# /etc/hosts
192.168.33.32 db1
192.168.33.33 db2
192.168.33.34 db3

playbook

MySQLを初期化する方法には二つがある。

  • mysqld --initialize
  • mysqld --initialize-insecure

詳細は「はじまる前に」にて紹介した記事を参照しよう。
最初はインタラクティブで実行されるmysql_secure_installationを使わないために「mysqld --instialize-insecure」で初期化し、mysql_secure_installationと同等の処理を行うようにplaybookを作成した。
その後、mysql_secure_installationをインタラクティブではない自動で実行する方法を知り、それに合わせてplaybookを修正した。
二つのplaybookを全部みて見よう。

mysqld --initialize-insecureの場合

insecureでmysqlを初期化すると、昔のように「mysql -u root」だけで接続ができる。
後で、mysql_secure_installationと同等な処理を行えばいい。

roles/mysql/tasks/main.yml
- name: install the mysql rpm
  yum:
    name: "{{ mysql_yum_repository }}"
    state: present

- name: install the mysql
  yum: name={{ item }} state=present
  with_items: "{{ mysql_yum_packages }}"

- name: check if mysql log file size is 0 or not
  stat: path={{ mysql_log_file }}
  register: log_file
  changed_when: false

- name: initialize insecure mysql
  command: mysqld --initialize-insecure --user=mysql
  changed_when: false
  when: log_file.stat.size == 0

- name: start the mysqld
  service: name=mysqld state=started enabled=yes

- name: change the mysql from insecure to secure
  command: mysql -u root -e "{{ mysql_secure_installation_query }}"
  changed_when: false
  when: log_file.stat.size == 0

- name: create a log file to write the slow query
  file:
    path: "{{ mysql_slow_log_file }}"
    state: touch
    owner: mysql
    group: mysql
    mode: 0644
  when: log_file.stat.size == 0

- name: setting up to my.cnf
  blockinfile:
    path: /etc/my.cnf
    block: |
      {{ item.content }}
    marker: "# {mark} {{ item.name }}"
    validate: mysqld --defaults-file=%s --verbose --help
  with_items: "{{ mysql_my_cnf }}"
  register: config_file

- name: restart the mysqld
  service: name=mysqld state=restarted
  when: config_file.changed

- name: create replication user to master
  mysql_user:
    login_user: "{{ mysql_user_name }}"
    login_password: "{{ mysql_user_password }}"
    name: "{{ mysql_replication_user.name }}"
    host: "{{ mysql_replication_user.host | default('%') }}"
    password: "{{ mysql_replication_user.password }}"
    priv: "{{ mysql_replication_user_priv | default('*.*:REPLICATION SLAVE,REPLICATION CLIENT') }}"
    state: present
  when: >
    mysql_replication_role == 'master'
    and mysql_replication_master is defined
    and mysql_replication_user is defined

- name: get replication slave status to slave
  mysql_replication:
    mode: getslave
    login_user: "{{ mysql_user_name }}"
    login_password: "{{ mysql_user_password }}"
  register: slave
  when: >
    mysql_replication_role == 'slave'
    and mysql_replication_master is defined
    and mysql_replication_user is defined

- debug: msg={{ slave }}

- name: change replication master to slave
  mysql_replication:
    mode: changemaster
    login_user: "{{ mysql_user_name }}"
    login_password: "{{ mysql_user_password }}"
    master_host: "{{ mysql_replication_master.host }}"
    master_port: "{{ mysql_replication_master.port | default(3306) }}"
    master_user: "{{ mysql_replication_user.name }}"
    master_password: "{{ mysql_replication_user.password }}"
    master_auto_position: 1
  when: >
    mysql_replication_role == 'slave'
    and slave.Is_Slave is defined and not slave.Is_Slave
    and mysql_replication_master is defined
    and mysql_replication_user is defined

- name: start replication to slave
  mysql_replication:
    mode: startslave
    login_user: "{{ mysql_user_name }}"
    login_password: "{{ mysql_user_password }}"
  when: >
    mysql_replication_role == 'slave'
    and slave.Is_Slave is defined and not slave.Is_Slave
    and mysql_replication_master is defined
    and mysql_replication_user is defined
  • install the mysql rpm

yum repositoryをインストールする。

  • install the mysql

mysql_yum_packagesという変数にListで記述したパッケージをインストールする。

  • check if mysql log file size is 0 or not

最初1回だけ実行したいタスクがあって、書いた。
mysqldが起動される前にログファイルのstat情報を取得して保存しておく必要がある。
他のタスクの実行可否を決めるための情報なので、ターゲットノードには影響を与えていないタスクだ。
changedになる必要がないので、「changed_when: false」で上書きした。
この件についてもっと知りたい人は次の記事が役に立つと思う。

Ansibleのcommand系(command、shell)を使う時の冪等性を保つ方法について

  • initialize insecure mysql

insecureでmysqlを初期化する。
最初1回だけ実行すればいい。

  • start the mysqld

mysqld起動する。
mysqlクライアントを使って何かの処理をしなければならないから。

  • change the mysql from insecure to secure

insecureのまま運用するのはセキュリティ上問題があるので、mysql_secure_installationと同等のクエリを実行してsecureな状態を作る。
最初1回だけ実行すればいい。

  • create a log file to write the slow query

遅いクエリが実行された場合は、それをログとして残さないと障害が発生して原因を探るとき面倒になる。
「state: touch」はLinuxコマンドのtouchと同様。
最初1回だけ実行すればいい。

  • setting up to my.cnf

最初からある程度設定されたmy.cnfファイルを用意した方がいいかも。
ここではMySQLの設定を一つずつ適用してみたいと思って、わざと変数にListとして保存した。

  • restart the mysqld

my.cnfファイルを変更したので、mysqldを再起動する必要がある。
handlerにタスクを用意してなぜnotifyを使って通知していないのかについて説明する。
Ansibleのhandlerは全てのタスクが終了した後、1回だけ実行されるようになっている。
もう一度言う。playbookの中にある全てのタスクが終了した後だ。
roleやinclude_tasksで読み込んでいる全てのタスクが終了しないとhandlerは実行されない。
my.cnfは変更されて次のタスクが残されている場合は、そのタスクを実行する前に再起動しないといけない。

  • create replication user to master

レプリケーション構成でslaveからmasterへ接続するユーザーを生成する。
以前のタスクでrootパスワードを変更したので、login_userとlogin_passwordを指定する必要がある。
「host: "{{ mysql_replication_user.host | default('%') }}"」にてdefault filterが使われている。
mysql_replication_user.hostが定義されていない場合はdefault filterで定義された'%'が割り当てられる。
全てのサーバーから接近できるようになるということだ。
privは特権を設定する部分だ。
詳細はAnsibleドキュメントのmysql_userモジュールを参照にしよう。
条件式の先頭にある>だが、|との違いは改行コード(\n)の有無だ。
>は改行して書いても改行コードが入らない。|は改行コードが入る。
mysql_replication_roleという変数はサーバー別に定義されている。
masterサーバーの場合は'master'で、slaveサーバーの場合は'slave'になる。
mysql_replication_masterとmysql_replication_userという変数は「group_vars/db.yml」に定義されている。
このタスクはmasterサーバーでレプリケーション関連変数が定義されている場合に実行される。

  • get replication slave status to slave

slaveサーバーからslaveの状態を取得してregisterでslaveという名前で保存する。
debugでslave変数を確認してみるとどんな情報が取得されるのかすぐ分かる。
最初実行した時と2度目以降の値が異なるので、是非確認して見よう。
このタスクはslaveサーバーでレプリケーション関連変数が定義されている場合に実行される。

  • change replication master to slave

slaveサーバーからchange masterを実行する。
masterサーバーの情報をslaveに記録する。
詳細は「はじまる前に」のMySQL関連記事を参照にしよう。
このタスクはslaveサーバーでレプリケーション関連変数が定義されていて先ほど取得したslave状態からIs_Slaveが定義されていてその値がfalseの場合に実行される。
このタスクでは処理結果をslaveという変数に保存することがポイントだ。

  • start replication to slave

slaveサーバーからレプリケーションのslaveモードを実行する。
このタスクの実行条件は「change replication master to slave」タスクと同じだ。

mysqld --initializeの場合

こっちがMySQLサーバー設置の正式ルートであろう。
既存playbookとの差分を用意した。

roles/mysql/tasks/main.yml
diff --git a/roles/mysql/tasks/main.yml b/roles/mysql/tasks/main.yml
index b680ff4..b3eb029 100644
--- a/roles/mysql/tasks/main.yml
+++ b/roles/mysql/tasks/main.yml
@@ -12,16 +12,23 @@
   register: log_file
   changed_when: false

-- name: initialize insecure mysql
-  command: mysqld --initialize-insecure --user=mysql
+- name: start the mysqld
+  service: name=mysqld state=started enabled=yes
+
+- name: get the temporary root password
+  shell: cat {{ mysql_log_file }} | grep 'temporary password'
+  register: temporary_password
   changed_when: false
   when: log_file.stat.size == 0

-- name: start the mysqld
-  service: name=mysqld state=started enabled=yes
+- name: change the root password
+  command: >
+    mysqladmin password '{{ mysql_user_password }}' -u root -p'{{ temporary_password.stdout.split(" ")[-1] }}'
+  changed_when: false
+  when: log_file.stat.size == 0

-- name: change the mysql from insecure to secure
-  command: mysql -u root -e "{{ mysql_secure_installation_query }}"
+- name: run mysql_secure_installation
+  command: mysql_secure_installation -u root -p'{{ mysql_user_password }}' -D
   changed_when: false
   when: log_file.stat.size == 0

MySQLを初期化しない状態でmysqldを起動すると、自動的にsecureで初期化してしまう。
それでmysqldを起動する前にinsecureで初期化が必要だった。
今度はmysql_secure_installationを使うので、こんな処理は要らない。

代わりに初期化のとき割り当てられたrootパスワードを取得する必要がある。
初期rootパスワードを改めて決めたrootパスワードで変更後、変更したrootパスワードを用いてmysql_secure_installationを実行する。
このタスクは最初1回だけ走ればいい。
それで「log_file.stat.size == 0」の条件を各タスクに入れて置いた。

注意を払わなければならないのは、rootパスワードをsingle quotationで囲む必要があるということだ。
その理由については「はじまる前に」の記事内コメントに書かれている。

variables

このplaybookを作成することでどこにどんな変数を書けばいいのかしっかりと掴んだ。
group_vars、host_vars、vars順で説明する。

group_vars

MySQLサーバー群で共通的に使われる変数を設定する。

group_vars/db.yml
mysql_user_name: root
mysql_user_password: <PASSWORD>
mysql_replication_user:
  name: repl
  host: 192.168.33.%
  password: <PASSWORD>
mysql_replication_master:
  host: 192.168.33.32
  port: 3306

上記の変数らはmasterとslaveのようなサーバーの役割に関係なく共通的に使われる。
group_varsとhost_varsはパスワードのような重要情報を書く時に有用だ。
バージョン管理の対象から外せて運用しやすい。
勿論暗号化ができれば暗号化した内容を書いた方がいいだろう。

host_vars

各サーバー別情報を保存する変数を設定する。

host_vars/db1.yml
mysql_server_id: 1
mysql_replication_role: 'master'
host_vars/db2.yml
mysql_server_id: 2
mysql_replication_role: 'slave'
host_vars/db3.yml
mysql_server_id: 3
mysql_replication_role: 'slave'

mysql_server_idは各サーバーの識別番号になる。
MySQLはいろんなレプリケーション構成ができる。
その構成に合わせて各サーバーの変数を設定すればいい。

vars

このplaybookで使われる変数を設定する。

roles/mysql/vars/main.yml
mysql_yum_repository: https://dev.mysql.com/get/mysql57-community-release-el7-11.noarch.rpm
mysql_yum_packages:
  - MySQL-python
  - mysql-community-server
mysql_secure_installation_query: >
  ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY '{{ mysql_user_password }}';
  DELETE FROM mysql.user WHERE User='';
  DELETE FROM mysql.user WHERE User='root' AND Host NOT IN ('localhost', '127.0.0.1', '::1');
  DROP DATABASE IF EXISTS test;
  DELETE FROM mysql.db WHERE Db='test' OR Db='test\\_%';
  FLUSH PRIVILEGES;
mysql_log_file: /var/log/mysqld.log
mysql_slow_log_file: /var/log/mysqld-slow.log
mysql_my_cnf:
  - name: replication
    content: |
      server-id={{ mysql_server_id }}
      gtid_mode=on
      enforce-gtid-consistency
      log-bin=mysql-bin
      log_slave_updates
  - name: slow query log
    content: |
      slow_query_log=1
      slow_query_log_file={{ mysql_slow_log_file }}
      long_query_time=1
      log_queries_not_using_indexes
      log_slow_admin_statements
  - name: enable strict sql mode
    content: |
      sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY

playbookの中から変更される可能性があるものについては変数にした方がいい。
mysql_yum_repositoryとmysql_yum_packagesはバージョンによって十分変更される可能性が高いため、変数にして置いた。
mysql_secure_installation_queryはmysql_secure_installationと同等な処理を行うためのクエリ文である。
>で定義されているので、playbookでは1行で処理される。
mysql_secure_installationを使う場合は削除してもいい。
mysql_log_fileはデフォルトのログファイルで、mysql_slow_log_fileはスロークエリを残すためのログファイルである。
mysql_my_cnfでは各global設定をListで書いた。
my.cnfファイルに書く時はblockinfileモジュールを使っている。
Ansibleで設定ファイルを書く方法については次の記事を参照しよう。

Ansible ファイルを書き換える方法

参考

結果

おまけで処理結果をコピペして置いた。
最初実行する時と2度目以降実行する時の違いに目を通しておこう。
変更事項がない場合は2度目実行した時にchanaged=0でなければならない。

 ansible (master) ✗ vagrant destroy -f db1 db2 db3; vagrant up db1 db2 db3; ansible-playbook db.yml
==> db3: Forcing shutdown of VM...
==> db3: Destroying VM and associated drives...
==> db2: Forcing shutdown of VM...
==> db2: Destroying VM and associated drives...
==> db1: Forcing shutdown of VM...
==> db1: Destroying VM and associated drives...
Bringing machine 'db1' up with 'virtualbox' provider...
Bringing machine 'db2' up with 'virtualbox' provider...
Bringing machine 'db3' up with 'virtualbox' provider...
==> db1: Importing base box 'bento/centos-7.3'...
==> db1: Matching MAC address for NAT networking...
==> db1: Checking if box 'bento/centos-7.3' is up to date...
==> db1: Setting the name of the VM: db1
==> db1: Clearing any previously set network interfaces...
==> db1: Preparing network interfaces based on configuration...
    db1: Adapter 1: nat
    db1: Adapter 2: hostonly
==> db1: Forwarding ports...
    db1: 22 (guest) => 2232 (host) (adapter 1)
==> db1: Booting VM...
==> db1: Waiting for machine to boot. This may take a few minutes...
    db1: SSH address: 127.0.0.1:2232
    db1: SSH username: vagrant
    db1: SSH auth method: private key
    db1: Warning: Remote connection disconnect. Retrying...
    db1: Warning: Connection reset. Retrying...
    db1: Warning: Remote connection disconnect. Retrying...
    db1: Warning: Connection reset. Retrying...
    db1: Warning: Remote connection disconnect. Retrying...
    db1: Warning: Connection reset. Retrying...
    db1: Warning: Remote connection disconnect. Retrying...
    db1: Warning: Connection reset. Retrying...
    db1: Warning: Remote connection disconnect. Retrying...
    db1: Warning: Connection reset. Retrying...
    db1: Warning: Remote connection disconnect. Retrying...
==> db1: Machine booted and ready!
==> db1: Checking for guest additions in VM...
==> db1: Setting hostname...
==> db1: Configuring and enabling network interfaces...
    db1: SSH address: 127.0.0.1:2232
    db1: SSH username: vagrant
    db1: SSH auth method: private key
==> db1: Mounting shared folders...
    db1: /vagrant => /Users/devtopia
==> db2: Importing base box 'bento/centos-7.3'...
==> db2: Matching MAC address for NAT networking...
==> db2: Checking if box 'bento/centos-7.3' is up to date...
==> db2: Setting the name of the VM: db2
==> db2: Clearing any previously set network interfaces...
==> db2: Preparing network interfaces based on configuration...
    db2: Adapter 1: nat
    db2: Adapter 2: hostonly
==> db2: Forwarding ports...
    db2: 22 (guest) => 2233 (host) (adapter 1)
==> db2: Booting VM...
==> db2: Waiting for machine to boot. This may take a few minutes...
    db2: SSH address: 127.0.0.1:2233
    db2: SSH username: vagrant
    db2: SSH auth method: private key
    db2: Warning: Connection reset. Retrying...
    db2: Warning: Remote connection disconnect. Retrying...
    db2: Warning: Connection reset. Retrying...
    db2: Warning: Remote connection disconnect. Retrying...
    db2: Warning: Connection reset. Retrying...
    db2: Warning: Remote connection disconnect. Retrying...
    db2: Warning: Connection reset. Retrying...
    db2: Warning: Remote connection disconnect. Retrying...
==> db2: Machine booted and ready!
==> db2: Checking for guest additions in VM...
==> db2: Setting hostname...
==> db2: Configuring and enabling network interfaces...
    db2: SSH address: 127.0.0.1:2233
    db2: SSH username: vagrant
    db2: SSH auth method: private key
==> db2: Mounting shared folders...
    db2: /vagrant => /Users/devtopia
==> db3: Importing base box 'bento/centos-7.3'...
==> db3: Matching MAC address for NAT networking...
==> db3: Checking if box 'bento/centos-7.3' is up to date...
==> db3: Setting the name of the VM: db3
==> db3: Clearing any previously set network interfaces...
==> db3: Preparing network interfaces based on configuration...
    db3: Adapter 1: nat
    db3: Adapter 2: hostonly
==> db3: Forwarding ports...
    db3: 22 (guest) => 2234 (host) (adapter 1)
==> db3: Booting VM...
==> db3: Waiting for machine to boot. This may take a few minutes...
    db3: SSH address: 127.0.0.1:2234
    db3: SSH username: vagrant
    db3: SSH auth method: private key
    db3: Warning: Remote connection disconnect. Retrying...
    db3: Warning: Connection reset. Retrying...
    db3: Warning: Remote connection disconnect. Retrying...
    db3: Warning: Connection reset. Retrying...
    db3: Warning: Remote connection disconnect. Retrying...
    db3: Warning: Connection reset. Retrying...
    db3: Warning: Remote connection disconnect. Retrying...
    db3: Warning: Connection reset. Retrying...
==> db3: Machine booted and ready!
==> db3: Checking for guest additions in VM...
==> db3: Setting hostname...
==> db3: Configuring and enabling network interfaces...
    db3: SSH address: 127.0.0.1:2234
    db3: SSH username: vagrant
    db3: SSH auth method: private key
==> db3: Mounting shared folders...
    db3: /vagrant => /Users/devtopia

PLAY [db] ****************************************************************************************************************************************************************************************************************************************

TASK [Gathering Facts] ***************************************************************************************************************************************************************************************************************************
ok: [db2]
ok: [db3]
ok: [db1]

TASK [base : include_tasks] **********************************************************************************************************************************************************************************************************************
included: /Users/devtopia/Apps/ansible/roles/base/tasks/common.yml for db1, db2, db3

TASK [base : install common yum packages] ********************************************************************************************************************************************************************************************************
changed: [db1] => (item=[u'libselinux-python', u'epel-release', u'bind-utils', u'telnet'])
changed: [db3] => (item=[u'libselinux-python', u'epel-release', u'bind-utils', u'telnet'])
changed: [db2] => (item=[u'libselinux-python', u'epel-release', u'bind-utils', u'telnet'])

TASK [base : add mappings to /etc/hosts] *********************************************************************************************************************************************************************************************************
changed: [db3] => (item={u'ip': u'192.168.33.10', u'name': u'jumpbox'})
changed: [db2] => (item={u'ip': u'192.168.33.10', u'name': u'jumpbox'})
changed: [db1] => (item={u'ip': u'192.168.33.10', u'name': u'jumpbox'})
changed: [db3] => (item={u'ip': u'192.168.33.21', u'name': u'dev'})
changed: [db2] => (item={u'ip': u'192.168.33.21', u'name': u'dev'})
changed: [db1] => (item={u'ip': u'192.168.33.21', u'name': u'dev'})
changed: [db3] => (item={u'ip': u'192.168.33.22', u'name': u'test'})
changed: [db2] => (item={u'ip': u'192.168.33.22', u'name': u'test'})
changed: [db1] => (item={u'ip': u'192.168.33.22', u'name': u'test'})
changed: [db3] => (item={u'ip': u'192.168.33.23', u'name': u'live'})
changed: [db2] => (item={u'ip': u'192.168.33.23', u'name': u'live'})
changed: [db1] => (item={u'ip': u'192.168.33.23', u'name': u'live'})
changed: [db3] => (item={u'ip': u'192.168.33.24', u'name': u'gembox'})
changed: [db1] => (item={u'ip': u'192.168.33.24', u'name': u'gembox'})
changed: [db2] => (item={u'ip': u'192.168.33.24', u'name': u'gembox'})
changed: [db1] => (item={u'ip': u'192.168.33.31', u'name': u'proxy'})
changed: [db3] => (item={u'ip': u'192.168.33.31', u'name': u'proxy'})
changed: [db2] => (item={u'ip': u'192.168.33.31', u'name': u'proxy'})
changed: [db1] => (item={u'ip': u'192.168.33.32', u'name': u'db1'})
changed: [db3] => (item={u'ip': u'192.168.33.32', u'name': u'db1'})
changed: [db2] => (item={u'ip': u'192.168.33.32', u'name': u'db1'})
changed: [db1] => (item={u'ip': u'192.168.33.33', u'name': u'db2'})
changed: [db2] => (item={u'ip': u'192.168.33.33', u'name': u'db2'})
changed: [db3] => (item={u'ip': u'192.168.33.33', u'name': u'db2'})
changed: [db1] => (item={u'ip': u'192.168.33.34', u'name': u'db3'})
changed: [db3] => (item={u'ip': u'192.168.33.34', u'name': u'db3'})
changed: [db2] => (item={u'ip': u'192.168.33.34', u'name': u'db3'})
changed: [db1] => (item={u'ip': u'192.168.33.41', u'name': u'redmine'})
changed: [db3] => (item={u'ip': u'192.168.33.41', u'name': u'redmine'})
changed: [db2] => (item={u'ip': u'192.168.33.41', u'name': u'redmine'})
changed: [db1] => (item={u'ip': u'192.168.33.42', u'name': u'chat'})
changed: [db3] => (item={u'ip': u'192.168.33.42', u'name': u'chat'})
changed: [db2] => (item={u'ip': u'192.168.33.42', u'name': u'chat'})
changed: [db3] => (item={u'ip': u'192.168.33.51', u'name': u'gitlab'})
changed: [db2] => (item={u'ip': u'192.168.33.51', u'name': u'gitlab'})
changed: [db1] => (item={u'ip': u'192.168.33.51', u'name': u'gitlab'})
changed: [db3] => (item={u'ip': u'192.168.33.52', u'name': u'runner'})
changed: [db2] => (item={u'ip': u'192.168.33.52', u'name': u'runner'})
changed: [db1] => (item={u'ip': u'192.168.33.52', u'name': u'runner'})

TASK [base : include_tasks] **********************************************************************************************************************************************************************************************************************
included: /Users/devtopia/Apps/ansible/roles/base/tasks/locale.yml for db1, db2, db3

TASK [base : set up the locale to LANG="ja_JP.UTF-8" if os is CentOS 6.x] ************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [base : set up the locale to LANG="ja_JP.UTF-8" if os is CentOS 7.x] ************************************************************************************************************************************************************************
ok: [db3]
ok: [db1]
ok: [db2]

TASK [base : include_tasks] **********************************************************************************************************************************************************************************************************************
included: /Users/devtopia/Apps/ansible/roles/base/tasks/timezone.yml for db1, db2, db3

TASK [base : set up the timezone to Asia/Tokyo if os is CentOS 6.x] ******************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [base : modify the symbolick link to the Asia/Tokyo] ****************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [base : set up the timezone to Asia/Tokyo if os is CentOS 7.x] ******************************************************************************************************************************************************************************
ok: [db1]
ok: [db2]
ok: [db3]

TASK [base : include_tasks] **********************************************************************************************************************************************************************************************************************
included: /Users/devtopia/Apps/ansible/roles/base/tasks/ntp.yml for db1, db2, db3

TASK [base : install the ntp] ********************************************************************************************************************************************************************************************************************
changed: [db1]
changed: [db3]
changed: [db2]

TASK [base : copy the ntp's config file] *********************************************************************************************************************************************************************************************************
changed: [db3]
changed: [db2]
changed: [db1]

TASK [base : ntp is running and enabled] *********************************************************************************************************************************************************************************************************
changed: [db3]
changed: [db2]
changed: [db1]

TASK [base : include_tasks] **********************************************************************************************************************************************************************************************************************
included: /Users/devtopia/Apps/ansible/roles/base/tasks/sudoers.yml for db1, db2, db3

TASK [base : create administrators] **************************************************************************************************************************************************************************************************************
changed: [db3]
changed: [db1]
changed: [db2]

TASK [base : grant sudo permission to the admin user] ********************************************************************************************************************************************************************************************
changed: [db2]
changed: [db1]
changed: [db3]

TASK [base : copy a public key to the target node] ***********************************************************************************************************************************************************************************************
changed: [db1]
changed: [db2]
changed: [db3]

TASK [base : copy a private key to the target node] **********************************************************************************************************************************************************************************************
changed: [db1]
changed: [db2]
changed: [db3]

TASK [mysql : install the mysql rpm] *************************************************************************************************************************************************************************************************************
changed: [db1]
changed: [db2]
changed: [db3]

TASK [mysql : install the mysql] *****************************************************************************************************************************************************************************************************************
changed: [db3] => (item=[u'MySQL-python', u'mysql-community-server'])
changed: [db2] => (item=[u'MySQL-python', u'mysql-community-server'])
changed: [db1] => (item=[u'MySQL-python', u'mysql-community-server'])

TASK [mysql : check if mysql log file size is 0 or not] ******************************************************************************************************************************************************************************************
ok: [db3]
ok: [db1]
ok: [db2]

TASK [mysql : start the mysqld] ******************************************************************************************************************************************************************************************************************
changed: [db2]
changed: [db3]
changed: [db1]

TASK [mysql : get the temporary root password] ***************************************************************************************************************************************************************************************************
ok: [db1]
ok: [db2]
ok: [db3]

TASK [mysql : change the root password] **********************************************************************************************************************************************************************************************************
ok: [db2]
ok: [db1]
ok: [db3]

TASK [mysql : run mysql_secure_installation] *****************************************************************************************************************************************************************************************************
ok: [db1]
ok: [db2]
ok: [db3]

TASK [mysql : create a log file to write the slow query] *****************************************************************************************************************************************************************************************
changed: [db2]
changed: [db3]
changed: [db1]

TASK [mysql : setting up to my.cnf] **************************************************************************************************************************************************************************************************************
changed: [db1] => (item={u'content': u'server-id=1\ngtid_mode=on\nenforce-gtid-consistency\nlog-bin=mysql-bin\nlog_slave_updates\n', u'name': u'replication'})
changed: [db2] => (item={u'content': u'server-id=2\ngtid_mode=on\nenforce-gtid-consistency\nlog-bin=mysql-bin\nlog_slave_updates\n', u'name': u'replication'})
changed: [db3] => (item={u'content': u'server-id=3\ngtid_mode=on\nenforce-gtid-consistency\nlog-bin=mysql-bin\nlog_slave_updates\n', u'name': u'replication'})
changed: [db1] => (item={u'content': u'slow_query_log=1\nslow_query_log_file=/var/log/mysqld-slow.log\nlong_query_time=1\nlog_queries_not_using_indexes\nlog_slow_admin_statements\n', u'name': u'slow query log'})
changed: [db2] => (item={u'content': u'slow_query_log=1\nslow_query_log_file=/var/log/mysqld-slow.log\nlong_query_time=1\nlog_queries_not_using_indexes\nlog_slow_admin_statements\n', u'name': u'slow query log'})
changed: [db3] => (item={u'content': u'slow_query_log=1\nslow_query_log_file=/var/log/mysqld-slow.log\nlong_query_time=1\nlog_queries_not_using_indexes\nlog_slow_admin_statements\n', u'name': u'slow query log'})
changed: [db1] => (item={u'content': u'sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY\n', u'name': u'enable strict sql mode'})
changed: [db2] => (item={u'content': u'sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY\n', u'name': u'enable strict sql mode'})
changed: [db3] => (item={u'content': u'sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY\n', u'name': u'enable strict sql mode'})

TASK [mysql : restart the mysqld] ****************************************************************************************************************************************************************************************************************
changed: [db2]
changed: [db1]
changed: [db3]

TASK [mysql : create replication user to master] *************************************************************************************************************************************************************************************************
skipping: [db2]
skipping: [db3]
changed: [db1]

TASK [mysql : get replication slave status to slave] *********************************************************************************************************************************************************************************************
skipping: [db1]
ok: [db2]
ok: [db3]

TASK [mysql : debug] *****************************************************************************************************************************************************************************************************************************
ok: [db1] => {
    "msg": {
        "changed": false,
        "skip_reason": "Conditional result was False",
        "skipped": true
    }
}
ok: [db2] => {
    "msg": {
        "Is_Slave": false,
        "changed": false,
        "failed": false,
        "msg": "Server is not configured as mysql slave"
    }
}
ok: [db3] => {
    "msg": {
        "Is_Slave": false,
        "changed": false,
        "failed": false,
        "msg": "Server is not configured as mysql slave"
    }
}

TASK [mysql : change replication master to slave] ************************************************************************************************************************************************************************************************
skipping: [db1]
changed: [db2]
changed: [db3]

TASK [mysql : start replication to slave] ********************************************************************************************************************************************************************************************************
skipping: [db1]
changed: [db2]
changed: [db3]

RUNNING HANDLER [base : restart ntpd] ************************************************************************************************************************************************************************************************************
changed: [db3]
changed: [db2]
changed: [db1]

PLAY RECAP ***************************************************************************************************************************************************************************************************************************************
db1                        : ok=30   changed=17   unreachable=0    failed=0
db2                        : ok=32   changed=18   unreachable=0    failed=0
db3                        : ok=32   changed=18   unreachable=0    failed=0

 ansible (master) ✗ ansible-playbook db.yml -t mysql

PLAY [db] ****************************************************************************************************************************************************************************************************************************************

TASK [Gathering Facts] ***************************************************************************************************************************************************************************************************************************
ok: [db2]
ok: [db3]
ok: [db1]

TASK [mysql : install the mysql rpm] *************************************************************************************************************************************************************************************************************
ok: [db2]
ok: [db1]
ok: [db3]

TASK [mysql : install the mysql] *****************************************************************************************************************************************************************************************************************
ok: [db1] => (item=[u'MySQL-python', u'mysql-community-server'])
ok: [db2] => (item=[u'MySQL-python', u'mysql-community-server'])
ok: [db3] => (item=[u'MySQL-python', u'mysql-community-server'])

TASK [mysql : check if mysql log file size is 0 or not] ******************************************************************************************************************************************************************************************
ok: [db1]
ok: [db2]
ok: [db3]

TASK [mysql : start the mysqld] ******************************************************************************************************************************************************************************************************************
ok: [db2]
ok: [db3]
ok: [db1]

TASK [mysql : get the temporary root password] ***************************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [mysql : change the root password] **********************************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [mysql : run mysql_secure_installation] *****************************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [mysql : create a log file to write the slow query] *****************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [mysql : setting up to my.cnf] **************************************************************************************************************************************************************************************************************
ok: [db3] => (item={u'content': u'server-id=3\ngtid_mode=on\nenforce-gtid-consistency\nlog-bin=mysql-bin\nlog_slave_updates\n', u'name': u'replication'})
ok: [db1] => (item={u'content': u'server-id=1\ngtid_mode=on\nenforce-gtid-consistency\nlog-bin=mysql-bin\nlog_slave_updates\n', u'name': u'replication'})
ok: [db2] => (item={u'content': u'server-id=2\ngtid_mode=on\nenforce-gtid-consistency\nlog-bin=mysql-bin\nlog_slave_updates\n', u'name': u'replication'})
ok: [db1] => (item={u'content': u'slow_query_log=1\nslow_query_log_file=/var/log/mysqld-slow.log\nlong_query_time=1\nlog_queries_not_using_indexes\nlog_slow_admin_statements\n', u'name': u'slow query log'})
ok: [db3] => (item={u'content': u'slow_query_log=1\nslow_query_log_file=/var/log/mysqld-slow.log\nlong_query_time=1\nlog_queries_not_using_indexes\nlog_slow_admin_statements\n', u'name': u'slow query log'})
ok: [db2] => (item={u'content': u'slow_query_log=1\nslow_query_log_file=/var/log/mysqld-slow.log\nlong_query_time=1\nlog_queries_not_using_indexes\nlog_slow_admin_statements\n', u'name': u'slow query log'})
ok: [db2] => (item={u'content': u'sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY\n', u'name': u'enable strict sql mode'})
ok: [db1] => (item={u'content': u'sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY\n', u'name': u'enable strict sql mode'})
ok: [db3] => (item={u'content': u'sql_mode=TRADITIONAL,NO_AUTO_VALUE_ON_ZERO,ONLY_FULL_GROUP_BY\n', u'name': u'enable strict sql mode'})

TASK [mysql : restart the mysqld] ****************************************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [mysql : create replication user to master] *************************************************************************************************************************************************************************************************
skipping: [db2]
skipping: [db3]
ok: [db1]

TASK [mysql : get replication slave status to slave] *********************************************************************************************************************************************************************************************
skipping: [db1]
ok: [db3]
ok: [db2]

TASK [mysql : debug] *****************************************************************************************************************************************************************************************************************************
ok: [db1] => {
    "msg": {
        "changed": false,
        "skip_reason": "Conditional result was False",
        "skipped": true
    }
}
ok: [db2] => {
    "msg": {
        "Auto_Position": 1,
        "Channel_Name": "",
        "Connect_Retry": 60,
        "Exec_Master_Log_Pos": 647,
        "Executed_Gtid_Set": "6a579e67-beab-11e7-b99e-08002737f846:1-2",
        "Is_Slave": true,
        "Last_Errno": 0,
        "Last_Error": "",
        "Last_IO_Errno": 0,
        "Last_IO_Error": "",
        "Last_IO_Error_Timestamp": "",
        "Last_SQL_Errno": 0,
        "Last_SQL_Error": "",
        "Last_SQL_Error_Timestamp": "",
        "Master_Bind": "",
        "Master_Host": "192.168.33.32",
        "Master_Info_File": "/var/lib/mysql/master.info",
        "Master_Log_File": "mysql-bin.000001",
        "Master_Port": 3306,
        "Master_Retry_Count": 86400,
        "Master_SSL_Allowed": "No",
        "Master_SSL_CA_File": "",
        "Master_SSL_CA_Path": "",
        "Master_SSL_Cert": "",
        "Master_SSL_Cipher": "",
        "Master_SSL_Crl": "",
        "Master_SSL_Crlpath": "",
        "Master_SSL_Key": "",
        "Master_SSL_Verify_Server_Cert": "No",
        "Master_Server_Id": 1,
        "Master_TLS_Version": "",
        "Master_UUID": "6a579e67-beab-11e7-b99e-08002737f846",
        "Master_User": "repl",
        "Read_Master_Log_Pos": 647,
        "Relay_Log_File": "db2-relay-bin.000002",
        "Relay_Log_Pos": 860,
        "Relay_Log_Space": 1065,
        "Relay_Master_Log_File": "mysql-bin.000001",
        "Replicate_Do_DB": "",
        "Replicate_Do_Table": "",
        "Replicate_Ignore_DB": "",
        "Replicate_Ignore_Server_Ids": "",
        "Replicate_Ignore_Table": "",
        "Replicate_Rewrite_DB": "",
        "Replicate_Wild_Do_Table": "",
        "Replicate_Wild_Ignore_Table": "",
        "Retrieved_Gtid_Set": "6a579e67-beab-11e7-b99e-08002737f846:1-2",
        "SQL_Delay": 0,
        "SQL_Remaining_Delay": null,
        "Seconds_Behind_Master": 0,
        "Skip_Counter": 0,
        "Slave_IO_Running": "Yes",
        "Slave_IO_State": "Waiting for master to send event",
        "Slave_SQL_Running": "Yes",
        "Slave_SQL_Running_State": "Slave has read all relay log; waiting for more updates",
        "Until_Condition": "None",
        "Until_Log_File": "",
        "Until_Log_Pos": 0,
        "changed": false,
        "failed": false
    }
}
ok: [db3] => {
    "msg": {
        "Auto_Position": 1,
        "Channel_Name": "",
        "Connect_Retry": 60,
        "Exec_Master_Log_Pos": 647,
        "Executed_Gtid_Set": "6a579e67-beab-11e7-b99e-08002737f846:1-2",
        "Is_Slave": true,
        "Last_Errno": 0,
        "Last_Error": "",
        "Last_IO_Errno": 0,
        "Last_IO_Error": "",
        "Last_IO_Error_Timestamp": "",
        "Last_SQL_Errno": 0,
        "Last_SQL_Error": "",
        "Last_SQL_Error_Timestamp": "",
        "Master_Bind": "",
        "Master_Host": "192.168.33.32",
        "Master_Info_File": "/var/lib/mysql/master.info",
        "Master_Log_File": "mysql-bin.000001",
        "Master_Port": 3306,
        "Master_Retry_Count": 86400,
        "Master_SSL_Allowed": "No",
        "Master_SSL_CA_File": "",
        "Master_SSL_CA_Path": "",
        "Master_SSL_Cert": "",
        "Master_SSL_Cipher": "",
        "Master_SSL_Crl": "",
        "Master_SSL_Crlpath": "",
        "Master_SSL_Key": "",
        "Master_SSL_Verify_Server_Cert": "No",
        "Master_Server_Id": 1,
        "Master_TLS_Version": "",
        "Master_UUID": "6a579e67-beab-11e7-b99e-08002737f846",
        "Master_User": "repl",
        "Read_Master_Log_Pos": 647,
        "Relay_Log_File": "db3-relay-bin.000002",
        "Relay_Log_Pos": 860,
        "Relay_Log_Space": 1065,
        "Relay_Master_Log_File": "mysql-bin.000001",
        "Replicate_Do_DB": "",
        "Replicate_Do_Table": "",
        "Replicate_Ignore_DB": "",
        "Replicate_Ignore_Server_Ids": "",
        "Replicate_Ignore_Table": "",
        "Replicate_Rewrite_DB": "",
        "Replicate_Wild_Do_Table": "",
        "Replicate_Wild_Ignore_Table": "",
        "Retrieved_Gtid_Set": "6a579e67-beab-11e7-b99e-08002737f846:1-2",
        "SQL_Delay": 0,
        "SQL_Remaining_Delay": null,
        "Seconds_Behind_Master": 0,
        "Skip_Counter": 0,
        "Slave_IO_Running": "Yes",
        "Slave_IO_State": "Waiting for master to send event",
        "Slave_SQL_Running": "Yes",
        "Slave_SQL_Running_State": "Slave has read all relay log; waiting for more updates",
        "Until_Condition": "None",
        "Until_Log_File": "",
        "Until_Log_Pos": 0,
        "changed": false,
        "failed": false
    }
}

TASK [mysql : change replication master to slave] ************************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

TASK [mysql : start replication to slave] ********************************************************************************************************************************************************************************************************
skipping: [db1]
skipping: [db2]
skipping: [db3]

PLAY RECAP ***************************************************************************************************************************************************************************************************************************************
db1                        : ok=8    changed=0    unreachable=0    failed=0
db2                        : ok=8    changed=0    unreachable=0    failed=0
db3                        : ok=8    changed=0    unreachable=0    failed=0

16
20
0

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
  3. You can use dark theme
What you can do with signing up
16
20