MySQL GTID 有一些運算函數可以幫助我們在運維工作中提高運維效率。
1 GTID內置函數
MySQL 包含GTID_SUBSET、GTID_SUBTRACT、WAIT_FOR_EXECUTED_GTID_SET、WAIT_UNTIL_SQL_THREAD_AFTER_GTIDS
?4個內置函數,用于GTID集合的基本運算。
1.1 GTID_SUBSET(set1,set2)
給定兩個GTID集set1和set2,set1是set2的子集返回true,否則返回false。
root@(none) 10:29:28>select
gtid_subset('53442434-8bfa-11e9-bc15-005056a50f77:1-2','8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-2') c8,
gtid_subset('53442434-8bfa-11e9-bc15-005056a50f77:1-2','8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-2,53442434-8bfa-11e9-bc15-005056a50f77:1-2') c9;
+----+----+
| c8 | c9 |
+----+----+
| 0 | 1 |
+----+----+
1 row in set (0.00 sec)
1.2? GTID_SUBTRACT(set1,set2)
GTID_SUBTRACT(gtid_set1, gtid_set2) 返回一個新的 GTID 集,表示 gtid_set1 中存在但 gtid_set2 中不存在的 GTID(即 gtid_set1 減去 gtid_set2 的結果)。
select gtid_subtract('53442434-8bfa-11e9-bc15-005056a50f77:1-2,8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-2','8eed0f5b-6f9b-11e9-94a9-005056a57a4e:2-3') c9;
root@(none) 10:37:49>select gtid_subtract('53442434-8bfa-11e9-bc15-005056a50f77:1-2,8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-2','8eed0f5b-6f9b-11e9-94a9-005056a57a4e:2-3') c9\G;
*************************** 1. row ***************************
c9: 53442434-8bfa-11e9-bc15-005056a50f77:1-2,
8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1
1 row in set (0.00 sec)
select gtid_subtract('53442434-8bfa-11e9-bc15-005056a50f77:1-5','53442434-8bfa-11e9-bc15-005056a50f77:3-10') c8
root@(none) 10:45:32>select gtid_subtract('53442434-8bfa-11e9-bc15-005056a50f77:1-5','53442434-8bfa-11e9-bc15-005056a50f77:3-10') c8;
+------------------------------------------+
| c8 |
+------------------------------------------+
| 53442434-8bfa-11e9-bc15-005056a50f77:1-2 |
+------------------------------------------+
1 row in set (0.01 sec)
1.3 WAIT_FOR_EXECUTED_GTID_SET(gtid_set[, timeout])
等到服務器應用了包含在 gtid_set 中的所有事務。如果指定可選的 timeout 值 (秒數),超時會使函數停止等待而退出。
select wait_for_executed_gtid_set('53442434-8bfa-11e9-bc15-005056a50f77:1-7');
?
select wait_for_executed_gtid_set('53442434-8bfa-11e9-bc15-005056a50f77:8',5);
1.4 WAIT_UNTIL_SQL_THREAD_AFTER_GTIDS(gtid_set[, timeout][,channel])
與WAIT_FOR_EXECUTED_GTID_SET
類似,但針對單個啟動的復制通道.
1.5 可以自定義函數
# 如果兩個GTID集相同,函數返回非零值
create function gtid_is_equal(gtid_set_1 longtext, gtid_set_2 longtext)
returns int deterministic
? return gtid_subset(gtid_set_1, gtid_set_2) and gtid_subset(gtid_set_2, gtid_set_1);
# 如果兩個GTID集不相交,函數返回非零值
create function gtid_is_disjoint(gtid_set_1 longtext, gtid_set_2 longtext)
returns int deterministic
? return gtid_subset(gtid_set_1, gtid_subtract(gtid_set_1, gtid_set_2));
# 如果兩個GTID集不相交,則函數返回非零,sum是兩個集的并集
create function gtid_is_disjoint_union(gtid_set_1 longtext, gtid_set_2 longtext, sum longtext)
returns int deterministic
? return gtid_is_equal(gtid_subtract(sum, gtid_set_1), gtid_set_2) and
? ? ? ? ?gtid_is_equal(gtid_subtract(sum, gtid_set_2), gtid_set_1);
# 函數返回格式化的GTID集。沒有空格且沒有重復,UUID按字母順序排列,間隔按數字順序排列
create function gtid_normalize(g longtext)
returns longtext deterministic
return gtid_subtract(g, '');
# 函數返回兩個GTID集的并集
create function gtid_union(gtid_set_1 longtext, gtid_set_2 longtext)
returns longtext deterministic
? return gtid_normalize(concat(gtid_set_1, ',', gtid_set_2));
# 函數返回兩個GTID集的交集
create function gtid_intersection(gtid_set_1 longtext, gtid_set_2 longtext)
returns longtext deterministic
? return gtid_subtract(gtid_set_1, gtid_subtract(gtid_set_1, gtid_set_2));
# 函數返回兩個GTID集的對稱差集
create function gtid_symmetric_difference(gtid_set_1 longtext, gtid_set_2 longtext)
returns longtext deterministic
? return gtid_subtract(concat(gtid_set_1, ',', gtid_set_2), gtid_intersection(gtid_set_1, gtid_set_2));
# 函數返回除去指定UUID的GTID集
create function gtid_subtract_uuid(gtid_set longtext, uuid text)
returns longtext deterministic
? return gtid_subtract(gtid_set, concat(uuid, ':1-', (1 << 63) - 2));
# 函數返回指定UUID的GTID集
create function gtid_intersection_with_uuid(gtid_set longtext, uuid text)
returns longtext deterministic
? return gtid_subtract(gtid_set, gtid_subtract_uuid(gtid_set, uuid));
?
2 GTID使用示例
2.1、監控從庫是否有人執行了寫操作,生產GTID:
使用?gtid_subset 來驗證,從庫的gtid一定是主庫的gtid的子集。
GTID_SUBSET(set1,set2)
執行 GTID_SUBSET(從庫gtid,主庫gtid),如果是true就沒問題,如果是false就是有人誤操作了從庫。
從庫('53442434-8bfa-11e9-bc15-005056a50f77:1-2,8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-6')
主庫('53442434-8bfa-11e9-bc15-005056a50f77:1-3,8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-5')
主庫('53442434-8bfa-11e9-bc15-005056a50f77:1-3')
主庫('53442434-8bfa-11e9-bc15-005056a50f77:1-3',8eed0f5b-6f9b-11e9-94a9-005056a57a4e:1-7')
結果:
2.2、驗證從庫的復制是否最新:
內置函數GTID_SUBSET
和GTID_SUBTRACT
可用于檢查從庫是應用了主庫的每個事務。使用GTID_SUBSET
執行此檢查,在從庫上執行以下命令:
master_gtid_executed=`mysql -uroot -proot -h192.168.2.80 -N -e "select replace(@@global.gtid_executed,char(10),'')"`
slave_gtid_executed=`mysql -uroot -proot -N -e "select replace(@@global.gtid_executed,char(10),'')"` sql="select gtid_subset('$master_gtid_executed', '$slave_gtid_executed')"
dls
mysql -uroot -proot -e "$sql"?
gtid_subset('dd746660-528a-11ed-9c86-000c293b9f86:1-14,e189b1a5-529d-11ed-992e-000c29c1da06:1', 'dd746660-528a-11ed-9c86-000c293b9f86:1-14,e189b1a5-529d-11ed-992e-000c29c1da06:1') ;
2.3?驗證復制拓撲中的服務器是否執行過本地事務
自定義函數 GTID_INTERSECTION_WITH_UUID 可用于驗證服務器是否執行過本地事務。可以在服務器上發出以下語句來檢查:
SELECT GTID_INTERSECTION_WITH_UUID(@@GLOBAL.gtid_executed, my_server_uuid);
2.4 檢查從庫上的異常事務
自定義函數 GTID_SUBTRACT_UUID 可用于檢查從庫是否只接收到源自其指定主庫的事務。對于單主復制,執行以下語句,server_uuid_of_master 是主庫的 server_uuid:
SELECT GTID_SUBTRACT_UUID(@@GLOBAL.gtid_executed, server_uuid_of_master);
?
如果結果不為空,則返回的事務是不是源自指定主庫的異常事務。對于多主復制拓撲中的從庫,重復該功能,例如:
SELECT GTID_SUBTRACT_UUID(GTID_SUBTRACT_UUID(@@GLOBAL.gtid_executed,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?server_uuid_of_master_1),
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?server_uuid_of_master_2);
?