IT评测·应用市场-qidao123.com技术社区

标题: 【建议收藏】超详细的Canal入门,看这篇就够了!!! [打印本页]

作者: 愛在花開的季節    时间: 2023-3-6 15:22
标题: 【建议收藏】超详细的Canal入门,看这篇就够了!!!
概述

canal是阿里巴巴旗下的一款开源项目,纯Java开发。基于数据库增量日志解析,提供增量数据订阅&消费,目前主要支持了MySQL(也支持mariaDB)。
背景

早期,阿里巴巴B2B公司因为存在杭州和美国双机房部署,存在跨机房同步的业务需求。不过早期的数据库同步业务,主要是基于trigger的方式获取增量变更,不过从2010年开始,阿里系公司开始逐步的尝试基于数据库的日志解析,获取增量变更进行同步,由此衍生出了增量订阅&消费的业务,从此开启了一段新纪元。ps. 目前内部使用的同步,已经支持mysql5.x和oracle部分版本的日志解析
基于日志增量订阅&消费支持的业务:
当前的 canal 支持源端 MySQL 版本包括 5.1.x , 5.5.x , 5.6.x , 5.7.x , 8.0.x
工作原理

Mysql的BinLog

它记录了所有的DDL和DML(除了数据查询语句)语句,以事件形式记录,还包含语句所执行的消耗的时间。主要用来备份和数据同步。
binlog 有三种模式:STATEMENT、ROW、MIXED
举例说明

举例来说,下面的sql
  1. COPYupdate user set age=20
复制代码
对应STATEMENT模式只有一条记录,对应ROW模式则有可能有成千上万条记录(取决数据库中的记录数)。
MySQL主备复制原理


mysql的binlog文件长这个样子。
  1. COPYmysql-bin.003831
  2. mysql-bin.003840  
  3. mysql-bin.003849  
  4. mysql-bin.003858
复制代码
启用Binlog注意以下几点:
canal能够同步数据的原理

理解了mysql的主从同步的机制再来看canal就比较清晰了,canal主要是听过伪装成mysql从server来向主server拉取数据。

Canal架构

canal的设计理念

canal的组件化设计非常好,有点类似于tomcat的设计。使用组合设计,依赖倒置,面向接口的设计。

canal的组件

每个canal instance 有多个组件构成。在conf/spring/default-instance.xml中配置了这些组件。他其实是使用了spring的容器来进行这些组件管理的。
instance 包含的组件

这里是一个cannalInstance工作所包含的大组件。截取自 conf/spring/default-instance.xml
  1. COPY<bean id="instance" >
  2.     <property name="destination" value="${canal.instance.destination}" />
  3.     <property name="eventParser">
  4.         <ref local="eventParser" />
  5.     </property>
  6.     <property name="eventSink">
  7.         <ref local="eventSink" />
  8.     </property>
  9.     <property name="eventStore">
  10.         <ref local="eventStore" />
  11.     </property>
  12.     <property name="metaManager">
  13.         <ref local="metaManager" />
  14.     </property>
  15.     <property name="alarmHandler">
  16.         <ref local="alarmHandler" />
  17.     </property>
  18. </bean>  
复制代码
EventParser设计

eventParser 最基本的组件,类似于mysql从库的dump线程,负责从master中获取bin_log

整个parser过程大致可分为几步:
EventSink设计

eventSink 数据的归集,使用设置的filter对bin log进行过滤,工作的过程如下。

说明:
数据过滤:支持通配符的过滤模式,表名,字段内容等
数据路由/分发:解决1:n (1个parser对应多个store的模式)
数据归并:解决n:1 (多个parser对应1个store)
数据加工:在进入store之前进行额外的处理,比如join
数据1:n业务

为了合理的利用数据库资源, 一般常见的业务都是按照schema进行隔离,然后在mysql上层或者dao这一层面上,进行一个数据源路由,屏蔽数据库物理位置对开发的影响,阿里系主要是通过cobar/tddl来解决数据源路由问题。
所以,一般一个数据库实例上,会部署多个schema,每个schema会有由1个或者多个业务方关注
数据n:1业务

同样,当一个业务的数据规模达到一定的量级后,必然会涉及到水平拆分和垂直拆分的问题,针对这些拆分的数据需要处理时,就需要链接多个store进行处理,消费的位点就会变成多份,而且数据消费的进度无法得到尽可能有序的保证。
所以,在一定业务场景下,需要将拆分后的增量数据进行归并处理,比如按照时间戳/全局id进行排序归并.
EventStore设计

eventStore 用来存储filter过滤后的数据,canal目前的数据只在这里存储,工作流程如下


定义了3个cursor

借鉴Disruptor的RingBuffer的实现,将RingBuffer拉直来看:

实现说明:

metaManager

metaManager 用来存储一些原数据,比如消费到的游标,当前活动的server等信息
alarmHandler

alarmHandler 报警,这个一般情况下就是错误日志,理论上应该是可以定制成邮件等形式,但是目前不支持
各个组件目前支持的类型

canal采用了spring bean container的方式来组装一个canal instance ,目的是为了能够更加灵活。
canal通过这些组件的选取可以达到不同使用场景的效果,比如单机的话,一般使用file来存储metadata就行了,HA的话一般使用zookeeper来存储metadata。
eventParser

eventParser 目前只有三种

eventSink

eventSink 目前只有EntryEventSink 就是基于mysql的binlog数据对象的处理操作
eventStore

eventStore 目前只有一种 MemoryEventStoreWithBuffer,内部使用了一个ringbuffer 也就是说canal解析的数据都是存在内存中的,并没有到zookeeper当中。
metaManager

metaManager 这个比较多,其实根据元数据存放的位置可以分为三大类,memory,file,zookeeper
Canal-HA机制

canal是支持HA的,其实现机制也是依赖zookeeper来实现的,用到的特性有watcher和EPHEMERAL节点(和session生命周期绑定),与HDFS的HA类似。
canal的ha分为两部分,canal server和canal client分别有对应的ha实现
server ha的架构图如下

大致步骤:
Canal Client的方式和canal server方式类似,也是利用zookeeper的抢占EPHEMERAL节点的方式进行控制.
canal的工作过程

dump日志

启动时去MySQL 进行dump操作的binlog 位置确定
工作的过程。在启动一个canal instance 的时候,首先启动一个eventParser 线程来进行数据的dump 当他去master拉取binlog的时候需要binlog的位置,这个位置的确定是按照如下的顺序来确定的(这个地方讲述的是HA模式哈)。
后面的eventParser的操作就会以内存中(MemoryLogPositionManager)存储的binlog位置去master进行dump操作了。
mysql的show master status 操作
  1. COPYmysql> show master status\G
  2. *************************** 1. row ***************************
  3.              File: mysql-bin.000028
  4.          Position: 635762367
  5.      Binlog_Do_DB:
  6. Binlog_Ignore_DB:
  7. Executed_Gtid_Set: 18db0532-6a08-11e8-a13e-52540042a113:1-2784514,
  8. 318556ef-4e47-11e6-81b6-52540097a9a8:1-30002,
  9. ac5a3780-63ad-11e8-a9ac-52540042a113:1-5,
  10. be44d87c-4f25-11e6-a0a8-525400de9ffd:1-156349782
  11. 1 row in set (0.00 sec
复制代码
归集(sink)和存储(store)

数据在dump回来之后进行的归集(sink)和存储(store)
sink操作是可以支撑将多个eventParser的数据进行过滤filter
filter使用的是instance.properties中配置的filter,当然这个filter也可以由canal的client端在进行subscribe的时候进行设置。如果在client端进行了设置,那么服务端配置文件instance.properties的配置都会失效
sink 之后将过滤后的数据存储到eventStore当中去。
目前eventStore的实现只有一个MemoryEventStoreWithBuffer,也就是基于内存的ringbuffer,使用这个store有一个特点,这个ringbuffer是基于内存的,大小是有限制的(bufferSize = 16 * 1024 也就是16M),所以,当canal的客户端消费比较慢的时候,ringbuffer中存满了就会阻塞sink操作,那么正读取mysql binlog的eventParser线程也会受阻。
  这种设计其实也是有道理的。 因为canal的操作是pull 模型,不是producer push的模型,所以他没必要存储太多数据,这样就可以避免了数据存储和持久化管理的一些问题。使数据管理的复杂度大大降低。
上面这些整个是canal的parser 线程的工作流程,主要对应的就是将数据从mysql搞下来,做一些基本的归集和过滤,然后存储到内存中。
binlog的消费者

canal从mysql订阅了binlog以后主要还是想要给消费者使用。那么binlog是在什么时候被消费呢。这就是另一条主线了。就像咱们做一个toC的系统,管理系统是必须的,用户使用的app或者web又是一套,eventParser 线程就像是管理系统,往里面录入基础数据。canal的client就像是app端一样,是这些数据的消费方。
  binlog的主要消费者就是canal的client端。使用的协议是基于tcp的google.protobuf,当然tcp的模式是io多路复用,也就是nio。当我们的client发起请求之后,canal的server端就会从eventStore中将数据传输给客户端。根据客户端的ack机制,将binlog的元数据信息定期同步到zookeeper当中。
canal的目录结构

配置父目录:
在下面可以看到
  1. COPYcanal
  2. ├── bin
  3. │   ├── canal.pid
  4. │   ├── startup.bat
  5. │   ├── startup.sh
  6. │   └── stop.sh
  7. └── conf
  8.     ├── canal.properties
  9.     ├── gamer ---目录
  10.     ├── ww_social ---目录
  11.     ├── wother ---目录
  12.     ├── nihao ---目录
  13.     ├── liveim ---目录
  14.     ├── logback.xml
  15.     ├── spring ---目录
  16.     ├── ym ---目录
  17.     └── xrm_ppp ---目录
复制代码
这里是全部展开的目录
  1. COPYcanal
  2. ├── bin
  3. │   ├── canal.pid
  4. │   ├── startup.bat
  5. │   ├── startup.sh
  6. │   └── stop.sh
  7. └── conf
  8.     ├── canal.properties
  9.     ├── game_center
  10.     │   └── instance.properties
  11.     ├── ww_social
  12.     │   ├── h2.mv.db
  13.     │   ├── h2.trace.db
  14.     │   └── instance.properties
  15.     ├── wwother
  16.     │   ├── h2.mv.db
  17.     │   └── instance.properties
  18.     ├── nihao
  19.     │   ├── h2.mv.db
  20.     │   ├── h2.trace.db
  21.     │   └── instance.properties
  22.     ├── movie
  23.     │   ├── h2.mv.db
  24.     │   └── instance.properties
  25.     ├── logback.xml
  26.     ├── spring
  27.     │   ├── default-instance.xml
  28.     │   ├── file-instance.xml
  29.     │   ├── group-instance.xml
  30.     │   ├── local-instance.xml
  31.     │   ├── memory-instance.xml
  32.     │   └── tsdb
  33.     │       ├── h2-tsdb.xml
  34.     │       ├── mysql-tsdb.xml
  35.     │       ├── sql
  36.     │       └── sql-map
  37.     └── ym
  38.         └── instance.properties
复制代码
Canal应用场景

同步缓存redis/全文搜索ES

canal一个常见应用场景是同步缓存/全文搜索,当数据库变更后通过binlog进行缓存/ES的增量更新。当缓存/ES更新出现问题时,应该回退binlog到过去某个位置进行重新同步,并提供全量刷新缓存/ES的方法,如下图所示。

下发任务

  另一种常见应用场景是下发任务,当数据变更时需要通知其他依赖系统。其原理是任务系统监听数据库变更,然后将变更的数据写入MQ/kafka进行任务下发,比如商品数据变更后需要通知商品详情页、列表页、搜索页等先关系统。这种方式可以保证数据下发的精确性,通过MQ发送消息通知变更缓存是无法做到这一点的,而且业务系统中不会散落着各种下发MQ的代码,从而实现了下发归集,如下图所示。

数据异构

在大型网站架构中,DB都会采用分库分表来解决容量和性能问题,但分库分表之后带来的新问题。比如不同维度的查询或者聚合查询,此时就会非常棘手。一般我们会通过数据异构机制来解决此问题。
所谓的数据异构,那就是将需要join查询的多表按照某一个维度又聚合在一个DB中。让你去查询。canal就是实现数据异构的手段之一。

本文由传智教育博学谷狂野架构师教研团队发布。
如果本文对您有帮助,欢迎关注和点赞;如果您有任何建议也可留言评论或私信,您的支持是我坚持创作的动力。
转载请注明出处!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!




欢迎光临 IT评测·应用市场-qidao123.com技术社区 (https://dis.qidao123.com/) Powered by Discuz! X3.4