云数据库Polardb如何获取事务提交时间戳

polardb 需开启binlog

  • 修改当前值loose_polar_log_bin

说明

  • 数据库引擎为MySQL 5.6的PolarDB集群,当前值修改为ON_WITH_GTID;

  • 数据库引擎为MySQL 8.0的PolarDB集群,当前值修改为ON。

canal拉取binlog

简介:(阿里巴巴 MySQL binlog 增量订阅&消费组件)

参考:https://github.com/alibaba/canal

下载

  • 下载 canal, 访问 release 页面 , 选择需要的包下载, 如以 1.1.4 版本为例

    1
    wget https://github.com/alibaba/canal/releases/download/canal-1.1.4/canal.deployer-1.1.4.tar.gz

canal-python

一.canal-python 简介

canal-python 是阿里巴巴开源项目 Canal是阿里巴巴mysql数据库binlog的增量订阅&消费组件 的 python 客户端。为 python 开发者提供一个更友好的使用 Canal 的方式。Canal 是mysql数据库binlog的增量订阅&消费组件。

基于日志增量订阅&消费支持的业务:

  1. 数据库镜像
  2. 数据库实时备份
  3. 多级索引 (卖家和买家各自分库索引)
  4. search build
  5. 业务cache刷新
  6. 价格变化等重要业务消息

关于 Canal 的更多信息请访问 https://github.com/alibaba/canal/wiki

二.应用场景

canal-python 作为Canal的客户端,其应用场景就是Canal的应用场景。关于应用场景在Canal介绍一节已有概述。举一些实际的使用例子:

1.代替使用轮询数据库方式来监控数据库变更,有效改善轮询耗费数据库资源。

2.根据数据库的变更实时更新搜索引擎,比如电商场景下商品信息发生变更,实时同步到商品搜索引擎 Elasticsearch、solr等

3.根据数据库的变更实时更新缓存,比如电商场景下商品价格、库存发生变更实时同步到redis

4.数据库异地备份、数据同步

5.根据数据库变更触发某种业务,比如电商场景下,创建订单超过xx时间未支付被自动取消,我们获取到这条订单数据的状态变更即可向用户推送消息。

6.将数据库变更整理成自己的数据格式发送到kafka等消息队列,供消息队列的消费者进行消费。

三.工作原理

canal-python 是 Canal 的 python 客户端,它与 Canal 是采用的Socket来进行通信的,传输协议是TCP,交互协议采用的是 Google Protocol Buffer 3.0。

四.工作流程

1.Canal连接到mysql数据库,模拟slave

2.canal-python 与 Canal 建立连接

2.数据库发生变更写入到binlog

5.Canal向数据库发送dump请求,获取binlog并解析

4.canal-python 向 Canal 请求数据库变更

4.Canal 发送解析后的数据给canal-python

5.canal-python收到数据,消费成功,发送回执。(可选)

6.Canal记录消费位置。

五.快速启动

安装Canal

Canal 的安装以及配置使用请查看 https://github.com/alibaba/canal/wiki/QuickStart

安装java请查看 https://www.cnblogs.com/lumama520/p/11058927.html

环境要求

python >= 3

构建canal python客户端

1
2
pip install canal-python
pip install protobuf

建立与Canal的连接

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
vim canalpython.py
import time

from canal.client import Client
from canal.protocol import EntryProtocol_pb2
from canal.protocol import CanalProtocol_pb2

client = Client()
client.connect(host='127.0.0.1', port=11111)
client.check_valid(username=b'', password=b'')
client.subscribe(client_id=b'1001', destination=b'example', filter=b'.*\\..*')

while True:
message = client.get(100)
entries = message['entries']
for entry in entries:
entry_type = entry.entryType
if entry_type in [EntryProtocol_pb2.EntryType.TRANSACTIONBEGIN, EntryProtocol_pb2.EntryType.TRANSACTIONEND]:
continue
row_change = EntryProtocol_pb2.RowChange()
row_change.MergeFromString(entry.storeValue)
event_type = row_change.eventType
header = entry.header
database = header.schemaName
table = header.tableName
event_type = header.eventType
for row in row_change.rowDatas:
format_data = dict()
if event_type == EntryProtocol_pb2.EventType.DELETE:
for column in row.beforeColumns:
format_data = {
column.name: column.value
}
elif event_type == EntryProtocol_pb2.EventType.INSERT:
for column in row.afterColumns:
format_data = {
column.name: column.value
}
else:
format_data['before'] = format_data['after'] = dict()
for column in row.beforeColumns:
format_data['before'][column.name] = column.value
for column in row.afterColumns:
format_data['after'][column.name] = column.value
data = dict(
db=database,
table=table,
event_type=event_type,
data=format_data,
)
print(data)
time.sleep(1)

client.disconnect()

事务时间戳获取

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
[root@Oracle01 ~]# mysql -uxxx -pxxx -hlocalhost
mysql> use qinxidb
Database changed
mysql> insert into tb (userId) values (9161);
Query OK, 1 row affected (0.10 sec)

[root@Oracle01 canal]# python3 canalpython.py
connected to 127.0.0.1:11111
Auth succed
Subscribe succed
header {
version: 1
logfileName: "mybinlog.000011"
logfileOffset: 38812146
serverId: 4
serverenCode: "UTF-8"
executeTime: 1586929590000
sourceType: MYSQL
eventLength: 75
}
entryType: TRANSACTIONBEGIN
storeValue: " \204\254\001"

header {
version: 1
logfileName: "mybinlog.000011"
logfileOffset: 38812269
serverId: 4
serverenCode: "UTF-8"
executeTime: 1586929590000
sourceType: MYSQL
schemaName: "qinxidb"
tableName: "tb"
eventLength: 40
eventType: INSERT
props {
key: "rowsCount"
value: "1"
}
}
entryType: ROWDATA
storeValue: "\010\177\020\001P\000b#\022!\010\000\020\004\032\006userId \000(\0010\000B\0049161R\007int(11)"

{'db': 'qinxidb', 'table': 'tb', 'event_type': 1, 'data': {'userId': '9161'}}
header {
version: 1
logfileName: "mybinlog.000011"
logfileOffset: 38812309
serverId: 4
serverenCode: "UTF-8"
executeTime: 1586929590000
sourceType: MYSQL
eventLength: 31
}
entryType: TRANSACTIONEND
storeValue: "\022\00551434"