JuiceFS PB 级数据同步优化:断点续传、安全与带宽控制

2026-06-18
张旭辉

在数据迁移、跨云同步与对象存储备份等场景中,juicefs sync 常用于执行大规模数据同步任务。当数据规模达到 TB 到 PB 级、对象数量达到数百万甚至数十亿级时,单次任务执行周期通常会延长到数小时甚至数天。

在这个过程中,系统运行过程中通常会逐步暴露出以下几类问题:

  • 任务在网络抖动、进程异常退出或节点重启后,难以从一致状态继续执行,往往需要重新扫描或重复处理;
  • 数据备份场景可能存在明文暴露风险并可能面临合规与安全要求;
  • 多个同步任务并发运行时,带宽资源竞争明显,整体传输过程缺乏有效的全局控制手段。

围绕这些场景,JuiceFS 1.4 在 sync 中提供了三项能力增强:断点续传、数据加解密,以及全局流量控制。本文将围绕这三项能力,介绍其适用场景、实现机制和使用方式。

01 断点续传

在早期版本中,用户同步过程中遇到错误或者中断,重新执行任务时,juicefs sync 需要重新扫描源端和目标端,再判断哪些对象已经完成、哪些对象仍需复制。对于上亿级对象或大量大文件场景,仅重新扫描本身就可能带来显著的时间成本和对象存储请求开销。

为了解决这一问题,我们在 JuiceFS 1.4 中为 sync 引入了 断点续传机制。启用后,sync 会将任务进度保存到目标端。当任务中断后,用户只需重新执行相同命令,sync 就会自动查找并加载与当前源端、目标端及关键参数匹配的 checkpoint,从上次未完成的位置继续执行,避免从头重新处理。

工作原理

启用 断点续传 后,sync 会在目标端保存一个 JSON 格式的状态文件,文件名格式如下:

.juicefs-sync-checkpoint.<hash>.json

其中,<hash> 由源端、目标端和关键同步参数计算得到,用于确保当前任务只加载与自身匹配的 checkpoint,避免不同同步任务之间误用状态文件。

断点续传的运行流程如下图所示:

juicefs sync checkpoint 的保存、恢复与清理流程
juicefs sync checkpoint 的保存、恢复与清理流程
  • sync 启动后,首先在目标端查找与当前任务匹配的 checkpoint。
  • 如果找到匹配项,则从上次保存的状态恢复执行;如果未找到,则按正常流程扫描并开始同步。sync 会并发遍历多个前缀,每个前缀都有独立状态,包括是否已完成遍历、上次遍历到的位置、待同步对象以及同步失败的对象。
  • 从 checkpoint 恢复时,sync 会先从每个前缀中记录的待同步对象和失败对象重新加入任务队列,对于上次尚未遍历完成的前缀,则从记录的位置继续扫描并同步;已经完成遍历的前缀只会继续处理 checkpoint 中尚未完成的对象。
  • 任务运行过程中,sync 会按设定间隔异步保存当前进度,默认每 10 秒保存一次。
  • 任务正常完成后,checkpoint 文件会被自动删除;如果任务中断或失败,则会保留下来,供下次执行相同命令时继续恢复。
集群模式架构示意图
集群模式架构示意图

在集群模式下,checkpoint 只有一份,由 Manager 统一维护。Worker 不直接读写目标端的 checkpoint 文件,而是负责从 Manager 拉取任务、执行同步并回传结果。Manager 会将 Worker 回传的完成对象、失败对象、统计信息和 multipart 状态合并到全局 checkpoint 中。

使用方式

# 启用断点续传
juicefs sync --enable-checkpoint SRC DST

# 自定义 checkpoint 保存间隔(默认 10s)
juicefs sync --enable-checkpoint --checkpoint-interval 30s SRC DST

# 忽略已有 checkpoint,强制从头同步
juicefs sync --enable-checkpoint --checkpoint-force-reset SRC DST

02 数据加解密

在跨云备份和归档场景中,客户端加密是常见的合规要求,例如数据主权、静态数据保护、敏感数据迁移等。此前,juicefs sync 没有原生加解密能力,用户如果希望将数据加密后写在目的端,通常需要借助外部工具额外处理。

在 JuiceFS 1.4 中,我们将流式加解密能力集成到 sync 流程中,使用户可以在数据同步的同时完成加密、解密或重新加密,主要支持以下三类场景:

  • 加密写入:将明文数据加密后写入目标端,适用于加密备份和归档场景。
  • 解密恢复:从源端读取加密数据,解密后写入目标端,适用于数据恢复或明文迁移。
  • 重新加密:使用旧密钥解密源端数据,再使用新密钥加密后写入目标端,适用于密钥轮换或加密算法迁移。

工作原理:分块流式加密

为了支持对象存储的 Range GET,并避免一次性加载大文件带来的高内存占用,sync 采用固定 1 MiB 分块的流式加密方案。每个文件会先被拆分为多个明文块,再分别加密写入目标端。

原始文件结构可以理解为:

[chunk 1: 1 MiB][chunk 2: 1 MiB] ... [chunk N: ≤1 MiB]

加密后,每个明文块会对应生成一个加密块。每个加密块由 4 字节头部和密文数据组成,其中 4 字节头部用于记录该块的实际密文长度,即 ct_len

每个加密块: [4B ct_len][ciphertext + padding]

加密后的文件: [encrypted chunk 1][encrypted chunk 2] ... [encrypted chunk N]

加密块的大小由明文块大小和加密开销共同决定,可以理解为 plainChunkSize + overhead。其中,plainChunkSize 固定为 1 MiB,overhead 取决于所使用的加密算法和密钥类型。

juicefs sync 分块流式加密的基本结构与读写方式
juicefs sync 分块流式加密的基本结构与读写方式

这种设计的好处是,随机读取时只需要根据偏移定位到对应的加密块,并下载相关块数据,不必读取整个文件。相应地,加密后的对象会包含额外头部、填充和加密元数据,因此目标端对象通常会比原始明文文件更大。

支持的算法

参数值 对称算法 密钥封装 适用场景
aes256gcm-rsa(默认) AES-256-GCM RSA 通用场景
chacha20-rsa ChaCha20-Poly1305 RSA 对 AES 硬件加速支持有限的环境
sm4gcm SM4-GCM SM2 需要国密算法的场景

使用方式

下面以 RSA 密钥为例说明加密、解密和重新加密的使用方式。

生成密钥对:

# 生成 RSA 私钥(公钥内嵌其中,JuiceFS 自动提取)
openssl genrsa -out private.pem 2048

# 带密码保护的私钥
openssl genrsa -aes256 -out private.pem 2048

场景一:加密写入目标端

juicefs sync /local/data s3://mybucket/backup 
    --encrypt-rsa-key /path/to/private.pem

场景二:解密读取源端,用于数据恢复或明文迁移。

juicefs sync s3://mybucket/backup /local/data 
    --decrypt-rsa-key /path/to/private.pem

场景三:重新加密,用于密钥轮换或算法迁移。

# 解密旧密钥加密的数据,用新密钥重新加密写入新存储
juicefs sync s3://old-bucket/encrypted s3://new-bucket/re-encrypted 
    --decrypt-rsa-key /path/to/old-private.pem 
    --encrypt-rsa-key /path/to/new-private.pem

如果私钥设置了密码,可以通过环境变量传入。

# 加密场景使用 JFS_ENCRYPT_RSA_PASSPHRASE
export JFS_ENCRYPT_RSA_PASSPHRASE="your-passphrase"
juicefs sync /local/data s3://mybucket/backup --encrypt-rsa-key private.pem

# 解密场景使用 JFS_DECRYPT_RSA_PASSPHRASE
export JFS_DECRYPT_RSA_PASSPHRASE="your-passphrase"
juicefs sync s3://mybucket/backup /local/data --decrypt-rsa-key private.pem

注意

  • 加密后的数据采用 JuiceFS 私有格式存储,需通过 juicefs sync 并提供对应密钥进行解密读取。
  • 请妥善备份加解密所用私钥;私钥一旦丢失,已加密数据将无法解密访问。

03 全局流量控制

在早期版本中,juicefs sync 已经支持通过 --bwlimit 对单个 sync 进程限速。但在多个 sync 进程同时运行时,例如分布式同步中的多个 Worker,或多个独立同步任务共享同一条出口链路,单进程限速并不能约束整体带宽使用量,仍然可能导致出口带宽被打满,影响其他业务流量。

我们在 JuiceFS 1.4 中新增了 --traffic-control-url 参数。用户可以将多个 sync 进程接入同一个外部流量控制服务,由该服务统一分配带宽配额,从而实现跨进程、跨任务的全局限速。

工作原理

全局流量控制采用令牌桶模型。多个 sync 进程在传输数据前,都会向同一个流量控制服务申请字节配额:

juicefs sync 全局流量控制实现流程
juicefs sync 全局流量控制实现流程

每个 sync 进程在传输数据前,会向控制服务申请一定数量的字节配额(credit)。控制服务根据当前总带宽使用情况,决定本次授予多少配额,以及配额的有效时间。配额用完后,sync 会继续申请新的配额;如果配额即将过期但尚未用完,未使用的部分会提前归还给控制服务。

控制服务通过 HTTP 接口提供配额申请和归还能力,接口需由用户自行实现或接入现有服务:

POST /traffic-control
Content-Type: application/json

请求:
{"bytes": 1048576}
  bytes > 0: 申请 bytes 字节的额度
  bytes < 0: 归还 |bytes| 字节的未使用额度

响应:
{"granted": 524288, "expired": 1000}
  granted: 本次授予的字节数
  expired: 额度有效期(毫秒)

在同步过程中,sync 会在传输数据前向流量控制服务申请配额。如果当前没有可用配额,传输会阻塞等待,直到获得新的配额。通过这种方式,多个同步任务可以共享同一个全局带宽上限,避免并发任务各自限速但总流量失控的问题。

使用方式

# 先部署流量控制服务(示例:监听 8080 端口,限制总带宽 100 Mbps)
# (服务实现由用户自行决定,juicefs 只负责调用接口)

# 多个 sync 进程接入同一个控制服务
juicefs sync SRC1 DST1 --traffic-control-url http://127.0.0.1:8080/traffic-control &
juicefs sync SRC2 DST2 --traffic-control-url http://127.0.0.1:8080/traffic-control &

--traffic-control-url 可与 --bwlimit 同时使用,两个限制独立生效:--bwlimit 用于限制单个 sync 进程的最大带宽,--traffic-control-url 用于控制多个 sync 进程的全局带宽。

# 单进程不超过 50 Mbps,同时所有进程合计不超过服务端配置的上限
juicefs sync SRC DST 
    --bwlimit 50 
    --traffic-control-url http://controller:8080/traffic-control

04 小结

JuiceFS 1.4 对 sync 的增强包括:断点续传降低了任务中断后的恢复成本,数据加解密提高了数据备份的安全性,全局流量控制则帮助多个同步任务更有序地共享带宽。对于数据迁移、跨云同步、对象存储备份和加密归档等场景,用户可以根据任务规模、网络环境和安全要求,灵活组合使用这些能力。

Author

张旭辉
Juicedata 系统工程师