📰 来源: 博客园
在数据库校验和修复工具中,连接串几乎是最敏感的配置之一。过去为了使用方便,很多配置文件会直接写入数据库明文密码。
gt-checksum v4.0.0 起,srcDSN / dstDSN 中的 password 必须使用 ENC[...] 密文,并新增独立工具 gt-dsn-crypt 生成 32 字节 base64 key 与 AES-256-GCM 密文,让连接串密码保护从“建议项”变成“强约束”。
[yejr@db160 gt-checksum]$ cat docs/gt-checksum-v4.0.0-dsn-encrypt-article.md
在数据库校验和修复工具中,连接串几乎是最敏感的配置之一。过去为了使用方便,很多配置文件会直接写入数据库明文密码。
gt-checksum v4.0.0 起,srcDSN / dstDSN 中的 password 必须使用 ENC[...] 密文,并新增独立工具 gt-dsn-crypt 生成 32 字节 base64 key 与 AES-256-GCM 密文,让连接串密码保护从“建议项”变成“强约束”。
v4.0.0 新增连接串密码加密能力,核心变化可以概括为三点:
新增工具:gt-dsn-crypt
gt-dsn-crypt 提供三个子命令:
# 1. 生成 32 字节 base64 key
KEY=$(gt-dsn-crypt gen-key)
# 2. 推荐从文件读取 password,避免明文进入 shell history
printf '%s' '数据库密码' > ./password.txt
# 3. 生成 ENC[...] 密文
GT_CHECKSUM_DSN_KEY="$KEY" gt-dsn-crypt encrypt --password-file ./password.txt
# 4. 启动 gt-checksum 时提供同一个 key
GT_CHECKSUM_DSN_KEY="$KEY" gt-checksum -c ./gc.conf
配置文件中的 DSN 变为:
srcDSN=mysql|user:ENC[v1:aes256gcm:default:...:...]@tcp(src-host:3306)/information_schema?charset=utf8mb4
dstDSN=mysql|user:ENC[v1:aes256gcm:default:...:...]@tcp(dst-host:3306)/information_schema?charset=utf8mb4
二、功能作用及使用场景深入解读
2.1 为什么要强制加密 DSN password?
在生产环境中,明文数据库密码的风险远比想象中更常见:
gc.conf 通常会被放在运维目录、部署目录或自动化平台中。一旦目录权限配置不当,明文数据库密码就可能被非授权用户读取。
场景二:日志与报错泄露
工具启动失败、参数校验失败、连接池初始化失败时,如果直接打印 DSN,明文 password 可能进入终端日志、CI 日志、工单截图或监控采集系统。
场景三:多人协作与审计
迁移项目通常涉及 DBA、研发、运维、测试多角色协作。配置文件被多人传递时,明文密码会在聊天工具、邮件、文档中扩散,后续很难追踪和回收。
场景四:临时文件与历史命令残留
即使最终配置文件被删除,明文密码仍可能残留在 shell history、备份文件、编辑器 swap 文件中。加密后,即使密文泄露,没有 key 也无法直接还原 password。
因此,v4.0.0 不再只是“建议不要写明文密码”,而是在启动阶段直接校验:DSN password 不是 ENC[...],程序 fail-fast 退出。
2.2 ENC[...] 密文格式是什么?
v4.0.0 使用统一的密文格式:
ENC[v1:aes256gcm:<kid>:<nonce_b64url>:<ciphertext_b64url>]
ENC[v1:aes256gcm:default:REPLACE_NONCE:REPLACE_CIPHERTEXT]
需要注意的是,只加密 password 片段,不会加密 host、port、库名、charset、SSL 参数等连接配置信息。例如:
# 加密前(v4.0.0 起不再支持)
srcDSN=mysql|user:plain_password@tcp(src-host:3306)/information_schema?charset=utf8mb4
# 加密后
srcDSN=mysql|user:ENC[v1:aes256gcm:default:...:...]@tcp(src-host:3306)/information_schema?charset=utf8mb4
这种设计兼顾了安全性和可运维性:密码被保护起来,但连接目标、库名和参数仍然可见,便于排查配置问题。
2.3 AES-256-GCM:同时保证保密性和完整性
gt-dsn-crypt 使用 Go 标准库中的 AES-GCM 实现:
block, _ := aes.NewCipher(key)
gcm, _ := cipher.NewGCM(block)
nonce := make([]byte, gcm.NonceSize())
rand.Read(nonce)
ciphertext := gcm.Seal(nil, nonce, []byte(password), nil)
ENC[...] 密文。这意味着 ENC[...] 不只是“混淆字符串”,而是具备认证加密能力的密文格式。
2.4 key 从哪里来?
v4.0.0 的 key 管理遵循两个原则:不内置默认 key,不从配置文件读取 key。
key 来源只有两种:
# 使用环境变量
GT_CHECKSUM_DSN_KEY="$KEY" gt-checksum -c ./gc.conf
# 使用 --key 覆盖环境变量
gt-checksum -c ./gc.conf --key "$KEY"
repairDB 和 gt-dsn-crypt 也使用同样的 key 机制:
# repairDB 使用环境变量
GT_CHECKSUM_DSN_KEY="$KEY" repairDB -conf ./gc.conf ./fixsql
#
🔗 原文链接: 点击阅读原文
文章评论