实时同步数据库变更,这个框架真是神器 - 今日头条

本文由 简悦 SimpRead 转码, 原文地址 www.toutiao.com

提起 Debezium 这个框架,相信大多数普通开发者都比较陌生,但是提及它所属的公司大家一定不会陌生。

我们数据库中的数据一直在变化,有时候我们希望能监听数据库数据的变化并根据变化做出一些反应,比如更新对应变化数据的缓存、增量同步到其它数据源、对数据进行检测和审计等等。而这种技术就叫变更数据捕获(Change Data Capture) 。对于这种技术我们可能知道一个国内比较知名的框架 Canal,非常好用!但是 Canal 有一个局限性就是只能用于 Mysql 的变更数据捕获。今天来介绍另一种更加强大的分布式 CDC 框架 Debezium

提起 Debezium 这个框架,相信大多数普通开发者都比较陌生,但是提及它所属的公司大家一定不会陌生。

https://p6.toutiaoimg.com/origin/pgc-image/f71a7e8eaaf640039908a0ae63c6b993.png?from=pc

红帽公司

没错就是开源界最成功的红帽公司Debezium 是为捕获数据更改的流式处理框架,开源免费。Debezium 近乎实时地监控数据库行级别 (row-level) 的数据变更,并针对变更可以做出反应。而且只有已提交的变更才是可见的,所以不用担心事务问题或者更改被回滚的问题。Debezium 为所有的数据库更改事件提供了一个统一的模型,所以不用担心每种数据库系统的复杂性。Debezium 提供了对 MongoDBMySQLPostgreSQLSQL ServerOracleDB2 等数据库的支持。

另外借助于 Kafka Connector 可以开发出一个基于事件流的变更捕获平台,具有高容错率和极强的扩展性。

https://p6.toutiaoimg.com/origin/pgc-image/16bbead1e1df4237ab9519ee92373fda.png?from=pc

Debezium Kafka 架构

如图所示,部署了用于 MySQL 和 PostgresSQL 的 Debezium Kafka 连接器以捕获对这两种类型数据库的更改事件,然后将这些更改通过下游的 Kafka Connector 将记录传输到其他系统或者数据库(例如 Elasticsearch、数据仓库、分析系统)或缓存。

另一种玩法就是将 Debezium 内置到应用程序中,来做一个类似消息总线的设施,将数据变更事件传递给订阅的下游系统中。

https://p6.toutiaoimg.com/origin/pgc-image/b3987c4636934665afb215bc8d040c18.png?from=pc

Debezium 内置服务器架构

Debezium 对数据的完整性和可用性也是做了不少的工作。Debezium 用持久化的、有副本备份的日志来记录数据库数据变化的历史,因此,你的应用可以随时停止再重启,而不会错过它停止运行时发生的事件,保证了所有的事件都能被正确地、完全地处理掉。

稍后我会演示一个 Spring Boot 集成 Debezium 的数据捕获系统。

理论介绍并不能让你直观感受到 Debezium 的能力,所以接下来我将使用嵌入式 Debezium 引擎来演示一下。

https://p6.toutiaoimg.com/origin/pgc-image/116dfa84ca284847b6202f0a520695e7.png?from=pc

流程图

如上图所示,当我们变更 MySQL 数据库中的某行数据时,通过 Debezium 实时监听到 binlog 日志的变化触发捕获变更事件,然后获取到变更事件模型,并做出响应(消费)。接下来我们来搭建环境。

为了方便这里使用 MySQL 的 Docker 容器,对应的脚本为:

1
2
3
4
5
6
# 运行mysql容器 
docker run --name mysql-service -v d:/mysql/data:/var/lib/mysql -p 3306:3306 -e TZ=Asia/Shanghai -e MYSQL_ROOT_PASSWORD=123456 -d mysql:5.7 --character-set-server=utf8mb4 --collation-server=utf8mb4_unicode_ci --default-time_zone="+8:00"
# 设置binlog位置
docker exec mysql-service bash -c "echo 'log-bin=/var/lib/mysql/mysql-bin' >> /etc/mysql/mysql.conf.d/mysqld.cnf"
# 配置 mysql的server-id
docker exec mysql-service bash -c "echo 'server-id=123454' >> /etc/mysql/mysql.conf.d/mysqld.cnf"

上面的脚本运行了一个用户名为 root、密码为 123456 并且将数据挂载到本地路径 d:/mysql/data 的 MySQL 容器,同时开启了 binlog 日志,并设置 server-id 为 123454,这些信息后面配置会用。

请注意如果不使用 root 用户的话,需要保证用户具有 SELECT, RELOAD, SHOW DATABASES, REPLICATION SLAVE, REPLICATION CLIENT 五种权限。

Spring Boot 的应用中加入下列依赖:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
<dependency>
        <groupId>io.debezium</groupId>
        <artifactId>debezium-api</artifactId>
        <version>${debezium.version}</version>
    </dependency>
    <dependency>
        <groupId>io.debezium</groupId>
        <artifactId>debezium-embedded</artifactId>
        <version>${debezium.version}</version>
    </dependency>
    <dependency>
        <groupId>io.debezium</groupId>
        <artifactId>debezium-connector-mysql</artifactId>
        <version>${debezium.version}</version>
    </dependency>

目前最新的版本号为 1.5.2.Final

然后声明需要的配置:

 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
/**
     * Debezium 配置.
     *
     * @return configuration
     */
    @Bean
    io.debezium.config.Configuration debeziumConfig() {
        return io.debezium.config.Configuration.create()
//            连接器的Java类名称
                .with("connector.class", MySqlConnector.class.getName())
//            偏移量持久化,用来容错 默认值
                .with("offset.storage", "org.apache.kafka.connect.storage.FileOffsetBackingStore")
//                偏移量持久化文件路径 默认/tmp/offsets.dat  如果路径配置不正确可能导致无法存储偏移量 可能会导致重复消费变更
//                如果连接器重新启动,它将使用最后记录的偏移量来知道它应该恢复读取源信息中的哪个位置。
                .with("offset.storage.file.filename", "C:/Users/n1/IdeaProjects/spring-boot-debezium/tmp/offsets.dat")
//                捕获偏移量的周期
                .with("offset.flush.interval.ms", "6000")
//               连接器的唯一名称
                .with("name", "mysql-connector")
//                数据库的hostname
                .with("database.hostname", "localhost")
//                端口
                .with("database.port", "3306")
//                用户名
                .with("database.user", "root")
//                密码
                .with("database.password", "123456")
//                 包含的数据库列表
                .with("database.include.list", "etl")
//                是否包含数据库表结构层面的变更,建议使用默认值true
                .with("include.schema.changes", "false")
//                mysql.cnf 配置的 server-id
                .with("database.server.id", "123454")
//                 MySQL 服务器或集群的逻辑名称
                .with("database.server.name", "customer-mysql-db-server")
//                历史变更记录
                .with("database.history", "io.debezium.relational.history.FileDatabaseHistory")
//                历史变更记录存储位置 
                .with("database.history.file.filename", "C:/Users/n1/IdeaProjects/spring-boot-debezium/tmp/dbhistory.dat")
                .build();
    }

配置分为两部分:

  • 一部分是 Debezium Engine 的配置属性,参见 Debezium Engine 配置 [1]
  • 一部分是 Mysql Connector 的配置属性,参见 Mysql Connector 配置 [2]

应用程序需要为运行的 Mysql Connector 启动一个 Debezium 引擎,这个引擎会以异步线程的形式运行,它包装了整个 Mysql Connector 连接器的生命周期。声明一个引擎需要以下几步:

  • 声明收到数据变更捕获信息的格式,提供了 JSON、Avro、Protobuf、Connect、CloudEvents 等格式。
  • 加载上面定义的配置。
  • 声明消费数据更改事件的函数方法。

声明的伪代码:

1
2
3
4
DebeziumEngine<RecordChangeEvent<SourceRecord>> debeziumEngine = DebeziumEngine.create(ChangeEventFormat.of(Connect.class))
        .using(configuration.asProperties())
        .notifying(this::handlePayload)
        .build();

handlePayload 方法为:

 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
private void handlePayload(List<RecordChangeEvent<SourceRecord>> recordChangeEvents, DebeziumEngine.RecordCommitter<RecordChangeEvent<SourceRecord>> recordCommitter) {
    recordChangeEvents.forEach(r -> {
        SourceRecord sourceRecord = r.record();
        Struct sourceRecordChangeValue = (Struct) sourceRecord.value();

        if (sourceRecordChangeValue != null) {
            // 判断操作的类型 过滤掉读 只处理增删改   这个其实可以在配置中设置
            Envelope.Operation operation = Envelope.Operation.forCode((String) sourceRecordChangeValue.get(OPERATION));

            if (operation != Envelope.Operation.READ) {
                String record = operation == Envelope.Operation.DELETE ? BEFORE : AFTER;
                // 获取增删改对应的结构体数据
                Struct struct = (Struct) sourceRecordChangeValue.get(record);
                // 将变更的行封装为Map
                Map<String, Object> payload = struct.schema().fields().stream()
                        .map(Field::name)
                        .filter(fieldName -> struct.get(fieldName) != null)
                        .map(fieldName -> Pair.of(fieldName, struct.get(fieldName)))
                        .collect(toMap(Pair::getKey, Pair::getValue));
                // 这里简单打印一下
                System.out.println("payload = " + payload);
            }
        }
    });
}

引擎的启动和关闭正好契合 Spring Bean 的生命周期:

 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
@Data
public class DebeziumServerBootstrap implements InitializingBean, SmartLifecycle {

    private final Executor executor = Executors.newSingleThreadExecutor();
    private DebeziumEngine<?> debeziumEngine;

    @Override
    public void start() {
        executor.execute(debeziumEngine);
    }

    @SneakyThrows
    @Override
    public void stop() {
        debeziumEngine.close();
    }

    @Override
    public boolean isRunning() {
        return false;
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        Assert.notNull(debeziumEngine, "debeziumEngine must not be null");
    }
}

启动该 Spring Boot 项目,你可以采用各种手段往数据库增删改数据,观察会有类似下面的打印:

1
payload = {user_id=1123213, username=felord.cn, age=11 , gender=0, enabled=1}

说明 Debezium 监听到了数据库的变更。你可以想想这种技术在哪些场景有用武之地。好了今天的分享就到这里,感谢大家的支持!原创不易,请多多关注、点赞、转发、再看。

项目源码地址: https://gitee.com/felord/spring-boot-debezium