Canal binlog 日志 Dump 流程分析

本文涉及的产品
RDS MySQL DuckDB 分析主实例,集群系列 4核8GB
RDS AI 助手,专业版
简介: Canal binlog 日志 Dump 流程分析

Canal 的 dump 支持串行和并行模式两种模式,本篇重点梳理 dump 的核心流程,以便对 dump 过程有一个充分的了解,更好的理解 Canal 的实现原理与细节,下一篇中将重点关注Canal是如何引入并行模式来提高dump的性能,即并行编程相关的技巧。


从前面的文章我们得知 Canal binlog 日志解析的基本流程如下图所示:

e3d5123f04bbafed33d874e9bfae69eb.png


解析来重点梳理一下 dump 命令的发送逻辑,特别是日志的处理流程,一些基本的日志格式。


1、 dump 流程分析


在 Canal 中 dump 方法声明如下:


891f91b608189c8db519ca0b65542ddc.png

带有参数 MultiStageCoprocessor 为并行处理模式,底层使用了disruptor 高性能并发框架,下文将重点关注学习。我们今天重点来看一下串行dump的实现,其方法声明如下:

cdca713df60e1928d96e7e85413e6c2b.png

其方法参数说明如下:


  • String binlogfilename
     binlog 文件名称,例如  mysql-bin.000038。
  • Long binlogPosition
    在文件中的偏移量。
  • SinkFunction func
    每解析出一条binlog日志的处理函数。


接下来我们直奔主题,一起来看一下 MysqlConnection 关于 dump 的实现流程。

595a39d15e8d0ca62d008deda3dd0037.png

Step1:在发送dump之前先设置相关的参数。


  • set wait_timeout=9999999
    连接空闲超时时间,默认为8消息,用于 Canal Slave 的等待超时时间远大于默认值。
  • set net_write_timeout=1800
    网络写请求超时时间,针对正在进行数据读写的连接,该值默认为 60s。
  • set net_read_timeout=1800
    网络读请求超时时间,针对正在进行数据读写的连接,该值默认为 30s。
  • set names 'binary'
    设置服务端返回结果时不做编码转化,直接按照数据库的二进制编码进行发送,由客户端自己根据需求进行编码转化。
  • set @master_binlog_checksum= @@global.binlog_checksum
    设置master_binlog_checksum,因为在mysql5.6之后为binlog引入了checksum机制,例如crc32,canal作为mysql slave,需要与服务端相关参数保持一致。
  • set @slave_uuid=uuid()
    canal相对与mysql数据库服务而言就是一个从服务器,这个指令用于设置server_id,使用uuid,避免server_id重复。
  • SET @master_heartbeat_period=15
    设置客户端与服务端心跳发送间隔,默认为15s。

b52d1634bd5be6efdbc8bb687f927ce5.png

Step2:从主库查询binlog checksum,具体向主库发送 select @@global.binlog_checksum 语句。

92ef87ac282b0705e57cdbc30b1918ad.png

Step3:向MySQL Master 注册从节点,告知客户端的host、port、用户名与密码、serverId,具体实现是发送命令CODE为 0x15。

3087ccd752893371f6b50d858ec4fcad.png

Step4:向 MySQL Master 发送 dump 请求,MySQL是基于请求与应答模式,发送请求命令后,就会向网络通道中写入响应请求。(在这里大家不妨先大概思考一下如何读取 dump 命令的返回值,这部分虽然涉及到网络相关的知识,我在这边会稍微简单提一下)。

852a6b87df2388c6dcb5a9f7cb128254.png

Step5:构建 DirectLogFetcher对象,实现基于 socket 的日志拉取服务,并构建 LogDecoder 对象,用于解析 binlog 日志。

53c27cd52ec91cb08eb976501f977129.png

  • Step6:使用 while 循环反复拉取消息,通过通过 LogDecoder 对二进制流进行解析,提取一条完整的binlog事件,交给 SinkFunction 去处理,并且如果开启了半同步机制,则需要向master发送ACK。既然是while循环,该方法的退出条件还是值得我们关注的:
  • fetch.fetch()方法返回 false
  • SinkFunction 的 sink 方法 false,SinkFunction的详细处理流程将在下文介绍,这里先告知返回false的情况是 binlog 日志解析线程已停止运行。


上面粗略的介绍了 dump 命令的几个核心关键步骤,要想详细掌握其实现细节,我们必须继续深入探讨如下几个问题:


  • DirectLogFetcher 内部工作机制
  • LogDecoder binlog 日志解析
  • 发送Dump底层网实现思路


2、DirectLogFetcher 内部工作机制


2.1 DirectLogFetcher 类图


e1090d11ccc756c0d72a63b175ff0dbb.png

DirectLogFetcher的类继承体系如上图所示,我们来看一下其关键点:


  • LogBuffer
    日志buffer,主要定义如下属性:
  • byte[] buffer
    缓存区中数据容器。
  • int origin
    当前buffer中的读指针
  • int limit
    当前buffer的最大可读可写指针
  • int position
    当前buffer的写指针。
  • int semival


是否需要发送ACK(用于半同步)。
LogBuffer封装了字节相关的操作,不仅定义了上面的属性,也定义了字节读取相关众多API,其截图如下:

0edc977d39ed38c6fb4eaf54fa5f88b5.png

  • LogFetcher binlog日志抓取抽象类,定义了如下关键属性与抽象方法。
  • int DEFAULT_INITIAL_CAPACITY
    LogBuffer中的初始容量,默认为8K。
  • float DEFAULT_GROWTH_FACTOR
    容量增长因子,默认为 2.0。
  • int   BIN_LOG_HEADER_SIZE
    binlog日志条目 header 的长度,固定为4字节。
  • float  factor
    增长因子。
  • public abstract boolean fetch()
    抓取binlog日志。
  • public abstract void close()
        关闭 Fetch。


  • DirectLogFetcher Canal LogFetcher模式实现类,其核心属性如下:
  • SocketChannel channel
    网络通道,用于发送dump请求的网络通道。
  • boolean issemi = false
    是否开启半同步。


2.2 fetch流程详解


接下来我们重点剖析 DirectLogFetcher 的 fetch 方法,来探究其实现原理。


在研究DirectLogFetcher的fetch方法之前,我们先重点跟踪一下其内部网络读写方法fetch0方法,该方法是具体与网络读写相关的实现。

30c66e0fa15501b6c878102619d1eca5.png

在详细介绍该方法之前先来介绍一下其参数的含义:


  • int off
    从通道中读取到的内容放入到buffer中的起始位置
  • int len
    期望从通道中读取的字节长度。


该方法的实现关键点如下:


  • 首先先确保接收缓存区有足够的剩余空间,如果空间不足,则进行扩容。
  • 然后从通道中读取指定长度的字节。


接下来我们来重点看一下DirectLogFetcher的fetch的实现流程。

88b3041f7763a077a3449b3803b92f13.png

Step1:尝试从网络通道中读取4个字节(即读取协议的头部),如果通道中还没有可读取内容,返回false,造成的效果是一次 dump 请求结束。

01dceb224714c644dbf50effa7257690.png

Step2:从上文读到的4个字节分别读出该网络包的总长度以及当前包的序号,从这里可以看成MySQL协议头为4字节,前3个字节为网络包的总长度,第4个字节为包的序列号。再取出数据包的长度后,继续向通道中读取netlen个字节,即读取一个完整的数据包到buffer中。

31ac0f7a875ae9e9dce97f8740cf712f.png

Step3:继续从数据包中读取一个字节,判断该包的状态码,是否是一个成功的响应,如果是错误的响应,会向外抛出一次,Canal 会记录dump命令执行错误的次数。

4646eea6a112e278f7c74dd820aa55a1.png

Step4:如果一个包的长度为允许的最大包长度,则继续读取,这个主要是根据MySQL协议做的处理,即读取到一个数据包,然后返回true,表示拉取到一条日志,然后通过LogDecoder解码,然后传入到sink方法中,进行日志的后续处理。

13b9220d29fe9179d5cb86469eba150b.png

Step5:这一步的目的,就是将buffer中的当前指针指向数据的开始位置。这样一次 fetch就结束了。


从上面的流程来看,DirectLogFetcher#fetch 方法结束后,就将进入到LogDecoder中。经过一次DirectLogFetcher#fetch方法后,即取回一条binlog日志,即二进制流,接下来就根据binlog协议对其解析。本文暂不深入该方法,如果大家想深入数据库中间件方面,可以作为一个很好的示例,面向MySQL通信协议进行编程。


3、SinkFunction


通过 LogDecoder从中解析一个事件后,会调用SinkFunction的sink方法,如果该方法返回 false,一次dump请求将介绍,接下来我们看一下其sink方法。

8160a1a9b8a073c1de96d0a181adae27.png

该方法的实现比较简单,这里不打算继续深入,我们重点来看一下 Canal.Entry 的结构:

6c2135824a5326835195f4603001df4d.png

这个结构是基于Canal做架构设计,解决顺序消费、数据不丢失一个重要参考依据,没解析一条事务,最终放入到环形缓存区,环形缓存区尽量以一个事务提交到Sink组件,其代码如下:

89fec72644ac017e69a7836ea9abb69f.png

这里主要有如下几个关键点:


  • 首先需要调用EventSink组件将解析出来的数据传入EventSink。
  • EventSink组件处理成功后,会提交解析位点。


相关实践学习
通过日志服务实现云资源OSS的安全审计
本实验介绍如何通过日志服务实现云资源OSS的安全审计。
相关文章
|
9月前
|
存储 运维 监控
SelectDB 实现日志高效存储与实时分析,完成任务可领取积分、餐具套装/水杯/帆布包!
SelectDB 实现日志高效存储与实时分析,完成任务可领取积分、餐具套装/水杯/帆布包!
|
4月前
|
SQL 运维 关系型数据库
深入探讨MySQL的二进制日志(binlog)选项
总结而言,对MySQL binlogs深度理解并妥善配置对数据库运维管理至关重要;它不仅关系到系统性能优化也是实现高可靠性架构设计必须考虑因素之一。通过精心规划与周密部署可以使得该机能充分发挥作用而避免潜在风险带来影响。
170 6
|
4月前
|
监控 安全 搜索推荐
使用EventLog Analyzer进行日志取证分析
EventLog Analyzer助力企业通过集中采集、归档与分析系统日志及syslog,快速构建“数字犯罪现场”,精准追溯安全事件根源。其强大搜索功能可秒级定位入侵时间、人员与路径,生成合规与取证报表,确保日志安全防篡改,大幅提升调查效率,为执法提供有力证据支持。
195 0
|
6月前
|
监控 安全 NoSQL
【DevOps】Logstash详解:高效日志管理与分析工具
Logstash是ELK Stack核心组件之一,具备强大的日志收集、处理与转发能力。它支持多种数据来源,提供灵活的过滤、转换机制,并可通过插件扩展功能,广泛应用于系统日志分析、性能优化及安全合规等领域,是现代日志管理的关键工具。
1006 0
|
8月前
|
SQL 监控 关系型数据库
MySQL日志分析:binlog、redolog、undolog三大日志的深度探讨。
数据库管理其实和写小说一样,需要规划,需要修订,也需要有能力回滚。理解这些日志的作用与优化,就像把握写作工具的使用与运用,为我们的数据库保驾护航。
359 23
|
8月前
|
自然语言处理 监控 安全
阿里云发布可观测MCP!支持自然语言查询和分析多模态日志
阿里云可观测官方发布了Observable MCP Server,提供了一系列访问阿里云可观测各产品的工具能力,包含阿里云日志服务SLS、阿里云应用实时监控服务ARMS等,支持用户通过自然语言形式查询
1159 0
阿里云发布可观测MCP!支持自然语言查询和分析多模态日志
|
7月前
|
人工智能 运维 监控
Aipy实战:分析apache2日志中的网站攻击痕迹
Apache2日志系统灵活且信息全面,但安全分析、实时分析和合规性审计存在较高技术门槛。为降低难度,可借助AI工具如aipy高效分析日志,快速发现攻击痕迹并提供反制措施。通过结合AI与学习技术知识,新手运维人员能更轻松掌握复杂日志分析任务,提升工作效率与技能水平。
|
9月前
|
SQL 运维 关系型数据库
MySQL Binlog 日志查看方法及查看内容解析
本文介绍了 MySQL 的 Binlog(二进制日志)功能及其使用方法。Binlog 记录了数据库的所有数据变更操作,如 INSERT、UPDATE 和 DELETE,对数据恢复、主从复制和审计至关重要。文章详细说明了如何开启 Binlog 功能、查看当前日志文件及内容,并解析了常见的事件类型,包括 Format_desc、Query、Table_map、Write_rows、Update_rows 和 Delete_rows 等,帮助用户掌握数据库变化历史,提升维护和排障能力。
|
5月前
|
存储 SQL 关系型数据库
MySQL中binlog、redolog与undolog的不同之处解析
每个都扮演回答回溯与错误修正机构角色: BinLog像历史记载员详细记载每件大大小小事件; RedoLog则像紧急救援队伍遇见突發情況追踪最后活动轨迹尽力补救; UndoLog就类似时间机器可倒带历史让一切归位原始样貌同时兼具平行宇宙观察能让多人同时看见各自期望看见历程而互不干扰.
297 9
|
6月前
|
存储 SQL 关系型数据库
MySQL的Redo Log与Binlog机制对照分析
通过合理的配置和细致的管理,这两种日志机制相互配合,能够有效地提升MySQL数据库的可靠性和稳定性。
235 10