Mysql replicas need to set binlog_format

For GTID replication we set the replica source (i.e. master)
to have a binlog_format of MIXED. We aren't setting this on
the replicas so they default to STATEMENT. This causes a problem
with certain "non-deterministic" functions (e.g. RAND()). This
changes replica config templates for Mysql and Percona to use
MIXED mode.

A new scenario test was introduced to validate that the
replicas have this set.

MariaDB doesn't appear to have this issue so it wasn't changed.

Scenario tests for Mysql, Percona and MariaDB were run to
validate this change.

Change-Id: I936cd9bc53a812af19653e9b5b472103fab2b6c1
Closes-bug: 1563541
This commit is contained in:
Doug Shelley 2016-03-31 23:18:38 +00:00
parent 9f877e5ba7
commit 9b03fec1e1
4 changed files with 57 additions and 2 deletions

View File

@ -0,0 +1,4 @@
---
fixes:
- Fixed default configuration template for MySQL to ensure that
replication uses binlog_format. Bug 1563541.

View File

@ -1,5 +1,6 @@
[mysqld] [mysqld]
log_bin = /var/lib/mysql/data/mysql-bin.log log_bin = /var/lib/mysql/data/mysql-bin.log
binlog_format = MIXED
relay_log = /var/lib/mysql/data/mysql-relay-bin.log relay_log = /var/lib/mysql/data/mysql-relay-bin.log
relay_log_info_repository = TABLE relay_log_info_repository = TABLE
relay_log_recovery = 1 relay_log_recovery = 1

View File

@ -1,5 +1,6 @@
[mysqld] [mysqld]
log_bin = /var/lib/mysql/data/mysql-bin.log log_bin = /var/lib/mysql/data/mysql-bin.log
binlog_format = MIXED
relay_log = /var/lib/mysql/data/mysql-relay-bin.log relay_log = /var/lib/mysql/data/mysql-relay-bin.log
relay_log_info_repository = TABLE relay_log_info_repository = TABLE
relay_log_recovery = 1 relay_log_recovery = 1

View File

@ -86,6 +86,7 @@ class ReplicationRunner(TestRunner):
CheckInstance(instance._info).slaves() CheckInstance(instance._info).slaves()
self.assert_true( self.assert_true(
set(replica_ids).issubset(self._get_replica_set(instance_id))) set(replica_ids).issubset(self._get_replica_set(instance_id)))
self._validate_master(instance_id)
def _get_replica_set(self, master_id): def _get_replica_set(self, master_id):
instance = self.get_instance(master_id) instance = self.get_instance(master_id)
@ -97,6 +98,7 @@ class ReplicationRunner(TestRunner):
CheckInstance(instance._info).replica_of() CheckInstance(instance._info).replica_of()
self.assert_equal(master_id, instance._info['replica_of']['id'], self.assert_equal(master_id, instance._info['replica_of']['id'],
'Unexpected replication master ID') 'Unexpected replication master ID')
self._validate_replica(instance_id)
def run_create_multiple_replicas(self, expected_states=['BUILD', 'ACTIVE'], def run_create_multiple_replicas(self, expected_states=['BUILD', 'ACTIVE'],
expected_http_code=200): expected_http_code=200):
@ -282,6 +284,20 @@ class ReplicationRunner(TestRunner):
def run_cleanup_master_instance(self): def run_cleanup_master_instance(self):
pass pass
def _validate_master(self, instance_id):
"""This method is intended to be overridden by each
datastore as needed. It is to be used for any database
specific master instance validation.
"""
pass
def _validate_replica(self, instance_id):
"""This method is intended to be overridden by each
datastore as needed. It is to be used for any database
specific replica instance validation.
"""
pass
class MysqlReplicationRunner(ReplicationRunner): class MysqlReplicationRunner(ReplicationRunner):
@ -291,10 +307,43 @@ class MysqlReplicationRunner(ReplicationRunner):
self.auth_client.users.delete(self.master_id, user.name, self.auth_client.users.delete(self.master_id, user.name,
user.host) user.host)
def _validate_master(self, instance_id):
"""For Mysql validate that the master has its
binlog_format set to MIXED.
"""
client = self.test_helper.get_client(
self.get_instance_host(instance_id))
self._validate_binlog_fmt(instance_id, client)
class MariadbReplicationRunner(MysqlReplicationRunner): def _validate_replica(self, instance_id):
pass """For Mysql validate that any replica has its
binlog_format set to MIXED and it is in read_only
mode.
"""
client = self.test_helper.get_client(
self.get_instance_host(instance_id))
self._validate_binlog_fmt(instance_id, client)
self._validate_read_only(instance_id, client)
def _validate_binlog_fmt(self, instance_id, client):
binlog_fmt = self._get_mysql_variable(client, 'binlog_format')
self.assert_equal('MIXED', binlog_fmt,
'Wrong binlog format detected for %s' % instance_id)
def _validate_read_only(self, instance_id, client):
read_only = self._get_mysql_variable(client, 'read_only')
self.assert_equal('ON', read_only, 'Wrong read only mode detected '
'for %s' % instance_id)
def _get_mysql_variable(self, client, variable):
cmd = "SHOW GLOBAL VARIABLES LIKE '%s'" % variable
row = client.execute(cmd).fetchone()
return row['Value']
class PerconaReplicationRunner(MysqlReplicationRunner): class PerconaReplicationRunner(MysqlReplicationRunner):
pass pass
class MariadbReplicationRunner(MysqlReplicationRunner):
pass