基于Java和DataX工具实现数据同步的后台管理,包括数据同步任务的生成,任务的管理,查看任务的执行日志,解析任务的执行结果等功能。
内含一些技术实现方案、心得体会和填坑经验等干货。
阅读本文之前,需要提前了解一下DataX的含义、使用场景和基本使用方法。
公司要做一个数据中心的项目,包括数据标准平台,数据集成平台,数据监控平台,数据共享平台。主要从两大技术方向去实现:
Kafka
。负责微服务之间实时的消息共享DataX
。数据同步负责大数据量的全量/增量同步,同时也是对消息队列的一种补救措施,即使消息共享失败,在下一次数据同步时,也能够确保全部数据的一致性。DataX的介绍此文不做说明,如需了解,请查阅相关资料。本文仅记录相关使用心得。Datax开源地址
Datax实现数据同步
任务的方式是执行一个python命令
操作一个配置了同步信息的json文件:
python [datax.py的路径] [同步任务的json文件路径]
python datax.py D:\Software\install\Environment\DataX\datax\job\mysql2mysql.json
Datax的数据同步,落到实处就是执行上述的python命令。但是对于一个数据中心项目来说,我们需要的不仅仅只是完成这个数据同步的操作。我们需要实现的是全流程的可配置,可视化,可监控,可定时
。
全流程描述:
cron表达式
。json文件
,一个包含上述python命令
和输出日志路径
的shell脚本
,一个linux定时任务
。通过定时任务执行shell脚本,即可完成定时执行数据同步任务的操作。(所有文件的命名
都与本次的任务ID
相关)。数据同步任务信息
,包括任务ID,所有文件的存放路径,cron表达式等信息执行结果和执行日志
。每次查看的时候,后端根据数据库中记录的当前任务的日志存放路径,读取该路径下的所有文件,解析并记录其中的内容,输出执行成功与否和全部的日志信息,并将解析结果存入数据库中,方便下次查阅。暂停、开始和删除
操作。通过删除和新增linux的定时任务实现任务的暂停和开始。同步任务只有在暂停状态下可进行删除操作,包括删除数据库记录,删除linux存放的日志文件等。在 linux服务器上的准备:
chmod –R 755 {你的目录}
创建实体类接收前端参数并生成datax需要的标准json数据。
分享一个在线JSON转实体类的网址
{
"job":
{
"content":
[
{
"reader":
{
"parameter":
{
"password":"YourPassword",
"column":
[
"id",
"school_code",
"link_id"
],
"connection":
[
{
"jdbcUrl":
[
"YourUrl"
],
"table":
[
"c_pass_dormitory_1"
]
}
],
"where":"",
"splitPk":"",
"username":"YourUserName"
},
"name":"mysqlreader"
},
"writer":
{
"parameter":
{
"postSql":
[
],
"password":"YourPassWord",
"column":
[
"id",
"school_code",
"link_id"
],
"connection":
[
{
"jdbcUrl":"YourUrl",
"table":
[
"c_pass_dormitory"
]
}
],
"writeMode":"insert",
"batchSize":"1024",
"preSql":
[
],
"username":"YourUserName"
},
"name":"mysqlwriter"
}
}
],
"setting":
{
"errorLimit":
{
"percentage":0.15,
"record":0
},
"speed":
{
"record":100,
"channel":1,
"byte":0
}
}
}
注意:在根据json生成实体类的时候,speed设置中的byte参数转为实体类会有点小问题
private int byte;
这是不合法的,这里可以重命名一下:
@Data
public class Speed {
private int channel;
private int record=100;
private int sByte;
}
但是要在生成json文件的时候,将sByte替换为byte:
java代码:
//生成json文件
private String json(String jsonString) throws Exception {
// 拼接文件完整路径
String fullPath = dataxJsonAddress + Tools.getId() + ".json";
// 生成json格式文件
// 保证创建一个新文件
File file = createFile(fullPath);
// 格式化json字符串
jsonString = JsonFormatTool.formatJson(jsonString);
// 将格式化后的字符串写入文件
java.io.Writer write = new OutputStreamWriter(new FileOutputStream(file), "UTF-8");
write.write(jsonString);
write.flush();
write.close();
return fullPath;
}
Datax是通过python命令执行Datax任务,因此所生成的shell脚本中 要包括 datax.py、json文件和log的路径信息。
生成shell脚本的代码:
//创建.sh文件
private String createShell(String jsonString, String logAddress) throws Exception {
//拼接shell脚本描述
String sellString = "#!/bin/bash\n" +
"source /etc/profile\n" +
"# 截至时间设置为当前时间戳\n" +
"end_time=$(date +%s)\n" +
"# 开始时间设置为60s前时间戳\n" +
"create_time=$(($end_time - 60))\n" +
"[dataxPy] [dataxJson] >>[dataxLog].`date +%Y%m%d%H%M` 2>&1 &";
sellString = sellString.replace("[dataxPy]", dataxPyAddress)
.replace("[dataxJson]", jsonString)
.replace("[dataxLog]", logAddress);
// 拼接文件完整路径
String fullPath = dataxShellAddress + Tools.getId() + ".sh";
File file = createFile(fullPath);
// 将格式化后的字符串写入文件
java.io.Writer write = new OutputStreamWriter(new FileOutputStream(file), "UTF-8");
write.write(sellString);
write.flush();
write.close();
return fullPath;
}
服务器上生成的shell脚本:
有了shell脚本后,就可以根据前台传递的cron表达式
,创建linux的定时任务
,定时执行shell脚本,实现数据同步。因此必须要给新生成的shell脚本添加 可执行权限
,Java代码如下:
//给shell文件添加 X 权限,shellAddress:shell脚本的全路径
new ProcessBuilder("/bin/chmod", "755", shellAddress).start();
最终的shell脚本内容:
#!/bin/bash
source /etc/profile
# 截至时间设置为当前时间戳
end_time=$(date +%s)
# 开始时间设置为60s前时间戳
create_time=$(($end_time - 60))
/root/datax/bin/datax.py /root/datax/datax_json/fce1411e47ef45649c5ff8f8930db5a0.json >>/root/datax/datax_log/d32f460e1339480cb84c286af8223370.`date +%Y%m%d%H%M` 2>&1 &
内容说明:
# datax的python可执行文件路径
/root/datax/bin/datax.py
# json文件的路径
/root/datax/datax_json/fce1411e47ef45649c5ff8f8930db5a0.json
# 本次任务的输出日志路径
>>/root/datax/datax_log/d32f460e1339480cb84c286af8223370.`date +%Y%m%d%H%M` 2>&1 &
>>/root/datax/datax_log/d32f460e1339480cb84c286af8223370.`date +%Y%m%d%H%M` 2>&1 &
日志文件的命名采用 任务ID.时间 方式,这样就可以根据任务ID获取该任务下的所有执行日志,用于数据监控。
Java代码:
//配置 linux上的定时任务
private void createCron(String cron,String shellAddress) throws Exception{
//执行操作指令 (echo "*/1 * * * * /root/datax/datax_sh/text.sh >>/root/crontab 2>&1 &";crontab -l )| crontab
String cmd = "(echo \"" + cron + " " + shellAddress + " >>/root/crontab 2>&1 &\";crontab -l )| crontab";
System.out.println("cmd命令" + cmd);
Runtime.getRuntime().exec(new String[]{"/bin/sh", "-c", cmd});
}
定时任务的执行脚本里,已经配置了执行日志的输出路径
,每次执行后都会在指定目录下生成指定命名规则
的日志文件。
部分日志文件如图:
上文也描述了,日志命名的规范,这样的命名方式是为后期数据监控
做准备。可以解析出每次数据同步任务的执行状态
和执行日志。
任务表的字段信息:
CREATE TABLE `c_info_datax_task` (
`id` varchar(36) NOT NULL DEFAULT '' COMMENT '主键',
`school_code` varchar(20) DEFAULT NULL COMMENT '学校编码',
`task_name` varchar(50) DEFAULT '' COMMENT '任务名',
`task_log_address` varchar(300) DEFAULT '' COMMENT '日志地址',
`task_json_address` varchar(300) DEFAULT '' COMMENT 'json地址',
`task_json_data` varchar(2000) DEFAULT NULL COMMENT 'json字符串信息',
`task_shell_address` varchar(300) DEFAULT '' COMMENT 'shell地址',
`task_cron` varchar(300) DEFAULT '' COMMENT 'cron表达式',
`status` varchar(10) DEFAULT '1' COMMENT '状态',
`deleted` varchar(1) DEFAULT '0' COMMENT '1:已删除,0:正常',
`remarks` varchar(255) DEFAULT '' COMMENT '备注',
`create_user` varchar(36) DEFAULT NULL COMMENT '创建者',
`create_time` datetime DEFAULT NULL COMMENT '创建时间',
`update_user` varchar(36) DEFAULT NULL COMMENT '更新人',
`update_time` datetime DEFAULT NULL COMMENT '更新时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=COMPACT COMMENT='数据同步-任务表';
至此,一个基于DataX的数据同步任务就配置好了,大体步骤: