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:
parent
9f877e5ba7
commit
9b03fec1e1
@ -0,0 +1,4 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- Fixed default configuration template for MySQL to ensure that
|
||||||
|
replication uses binlog_format. Bug 1563541.
|
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user