新闻中心
Go语言openpgp库中GPG用户ID签名无效问题的深度解析与解决方案

本文深入探讨了使用go语言`go.crypto/openpgp`库进行gpg用户id签名时,生成签名被gpg工具判定为“bad signature”的问题。核心原因在于该库早期版本中`signidentity`函数底层实现存在缺陷,错误地使用了密钥签名的算法而非用户id签名的算法。文章将指导读者理解问题根源,并强调使用最新`golang.org/x/crypto/openpgp`库的重要性,同时提供健壮的签名实现指南。
GPG用户ID签名问题背景与现象
在使用Go语言处理OpenPGP密钥时,一个常见需求是使用私钥为某个公钥的用户ID(User ID)创建签名,以证明该用户ID与公钥的关联性。开发者通常会选择go.crypto/openpgp(或其更现代的路径golang.org/x/crypto/openpgp)库来完成这项任务。然而,一些用户在使用该库的早期版本进行签名后,发现生成的签名无法通过标准的GPG工具(如gpg --check-sigs)验证,并报告为“bad Signature”。
例如,以下代码片段展示了尝试使用私钥priEnt对公钥pubEnt的某个用户ID进行签名的典型操作:
// 假设 priEnt 和 pubEnt 已经正确加载并解密
usrIdstring := "some_user_id@example.com" // 假设这是要签名的用户ID
errSign := pubEnt.SignIdentity(usrIdstring, &priEnt, nil)
if errSign != nil {
fmt.Println("签名用户ID失败:", errSign.Error())
return
}
// 将签名的公钥实体序列化为ASCII Armor格式尽管代码逻辑看起来合理,但如果使用的openpgp库版本过旧,上述操作产生的签名很可能被外部GPG工具拒绝。
核心问题分析:openpgp库的内部缺陷
导致“bad Signature”问题的根本原因在于code.google.com/p/go.crypto/openpgp库的早期版本中存在一个关键的实现缺陷。具体来说,Signature.SignUserId()函数(这是openpgp.Entity.SignIdentity方法底层调用的一个函数)在生成用户ID签名时,错误地使用了用于“密钥认证”(Key Certification)的算法,而非用于“用户ID认证”(User ID Certification)的正确算法。
在OpenPGP协议中,对密钥的签名和对用户ID的签名是两种不同的操作,它们使用不同的签名子包类型(Signature Subpacket Types)和认证机制:
- 密钥认证:通常用于证明一个子密钥(Subkey)属于一个主密钥(Primary Key),或一个密钥是另一个密钥的撤销签名。
- 用户ID认证:用于证明一个用户ID(例如“John Doe john.doe@example.com”)确实与某个公钥相关联。
由于库的缺陷,当开发者尝试对用户ID进行签名时,底层实际上生成了一个格式不正确的签名,导致GPG工具无法正确解析和验证。此外,该库的PublicKey.VerifyUserIdSignature()函数也存在类似问题,它在验证用户ID签名时,未能正确使用哈希中的公钥,导致其仅对自签名用户ID有效。
CA.LA
第一款时尚产品在线设计平台,服装设计系统
94
查看详情
这些问题在Go语言社区中被发现并报告,并已通过补丁在后续版本中得到修复。因此,问题的核心并非开发者的代码逻辑错误,而是所使用的openpgp库版本存在内部缺陷。
解决方案与最佳实践
解决此问题的最直接且推荐的方法是:升级并使用最新版本的golang.org/x/crypto/openpgp模块。 code.google.com/p/go.crypto/openpgp是Go语言模块系统建立之前的旧路径,已经不再维护。golang.org/x/crypto是官方维护的扩展加密库,会持续接收更新和错误修复。
1. 迁移到现代openpgp库
首先,确保您的项目使用Go模块,并将openpgp库的导入路径更新为:
import (
"golang.org/x/crypto/openpgp"
"golang.org/x/crypto/openpgp/armor"
"golang.org/x/crypto/openpgp/packet"
// ... 其他标准库导入
)然后运行 go mod tidy 或 go get golang.org/x/crypto/openpgp 来下载并更新依赖。
2. 健壮的Go语言签名实现
以下是一个经过优化和错误处理的示例,展示了如何使用更新后的openpgp库来加载密钥、解密私钥并为公钥的用户ID创建签名。
package main
import (
"bytes"
"fmt"
"io"
"log"
"golan
g.org/x/crypto/openpgp"
"golang.org/x/crypto/openpgp/armor"
"golang.org/x/crypto/openpgp/packet"
)
// SignPubKeyWithUserID 为给定的公钥实体签名指定的用户ID
// asciiPub: ASCII Armor编码的公钥字符串
// asciiPri: ASCII Armor编码的私钥字符串
// priPwd: 私钥的密码
// userIDToSign: 要签名的用户ID字符串
// 返回 ASCII Armor 编码的已签名公钥,或错误
func SignPubKeyWithUserID(asciiPub, asciiPri, priPwd, userIDToSign string) (string, error) {
// 1. 加载私钥实体
priEntity, err := readArmoredKeyRing(asciiPri)
if err != nil {
return "", fmt.Errorf("加载私钥失败: %w", err)
}
if len(priEntity) == 0 || priEntity[0].PrivateKey == nil {
return "", fmt.Errorf("未找到有效的私钥实体")
}
signer := priEntity[0] // 假设我们使用密钥环中的第一个私钥作为签名者
// 2. 解密私钥
if signer.PrivateKey.Encrypted {
err = signer.PrivateKey.Decrypt([]byte(priPwd))
if err != nil {
return "", fmt.Errorf("解密私钥失败: %w", err)
}
}
// 3. 加载公钥实体
pubEntityList, err := readArmoredKeyRing(asciiPub)
if err != nil {
return "", fmt.Errorf("加载公钥失败: %w", err)
}
if len(pubEntityList) == 0 {
return "", fmt.Errorf("未找到有效的公钥实体")
}
targetPubKey := pubEntityList[0] // 假设我们签名密钥环中的第一个公钥
// 4. 查找要签名的用户ID
foundUserID := false
for _, identity := range targetPubKey.Identities {
if identity.UserId.Id == userIDToSign {
// 5. 使用签名者的私钥对公钥的用户ID进行签名
// 注意:这里调用的是 openpgp.Entity 的 SignIdentity 方法
// 如果 openpgp 库版本正确,此方法会调用正确的底层签名逻辑
err = targetPubKey.SignIdentity(userIDToSign, signer, nil)
if err != nil {
return "", fmt.Errorf("签名用户ID '%s' 失败: %w", userIDToSign, err)
}
foundUserID = true
break
}
}
if !foundUserID {
return "", fmt.Errorf("在公钥中未找到要签名的用户ID: '%s'", userIDToSign)
}
// 6. 将签名的公钥实体序列化回ASCII Armor格式
signedKeyArmor, err := writeArmoredKey(targetPubKey)
if err != nil {
return "", fmt.Errorf("序列化已签名公钥失败: %w", err)
}
return signedKeyArmor, nil
}
// readArmoredKeyRing 从ASCII Armor字符串中读取密钥环
func readArmoredKeyRing(armoredKey string) (openpgp.EntityList, error) {
reader := bytes.NewReader([]byte(armoredKey))
entityList, err := openpgp.ReadArmoredKeyRing(reader)
if err != nil {
return nil, fmt.Errorf("读取ASCII Armor密钥环失败: %w", err)
}
return entityList, nil
}
// writeArmoredKey 将openpgp.Entity序列化为ASCII Armor字符串
func writeArmoredKey(entity *openpgp.Entity) (string, error) {
buf := new(bytes.Buffer)
writer, err := armor.Encode(buf, openpgp.PublicKeyType, nil)
if err != nil {
return "", fmt.Errorf("创建ASCII Armor编码器失败: %w", err)
}
err = entity.Serialize(writer)
if err != nil {
return "", fmt.Errorf("序列化实体失败: %w", err)
}
err = writer.Close()
if err != nil {
return "", fmt.Errorf("关闭编码器失败: %w", err)
}
return buf.String(), nil
}
func main() {
// 替换为你的实际ASCII Armor编码的公钥、私钥和密码
// 警告:在生产环境中,密钥不应硬编码或直接暴露
// 这里的示例密钥是虚拟的,请勿直接使用
dummyPublicKeyArmor := `-----BEGIN PGP PUBLIC KEY BLOCK-----
Version: GnuPG v2
mQENBF+fL3YBCADr51k/g770/p/iK6f4W8V2q6b6v9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j以上就是Go语言openpgp库中GPG用户ID签名无效问题的深度解析与解决方案的详细内容,更多请关注其它相关文章!
# 第一个
# 项目营销推广方式
# SEO云排名 拓宽科技
# 天津seo外包行者seo06
# 酒店官方网站优化内容
# 昭通抖音seo搜索服务
# 安徽网页优化seo靠谱
# 江苏seo平台技巧
# 图书营销短视频推广模式
# 滨海seo优化项目
# 大众车机优化网站
# 未找到
# 新和
# 而非
# 序列化
# 库中
# go
# 环中
# 这是
# 加载
# 公钥
# igs
# crypto
# red
# 标准库
# google
# ai
# 工具
# 编码
# cad
# go语言
# golang
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
俄罗斯搜索引擎Yandex指南 附2025年免登录官网入口
steam官方入口大全 steam账号注册及操作指南
C++如何操作注册表_Windows平台下C++读写注册表的API函数详解
J*a如何使用AtomicInteger控制计数_J*a无锁计数器性能分析
sublime如何优雅地处理行尾空格_sublime自动清理多余空白字符配置
J*aScript 字符串标签转换:使用正则表达式高效替换
ArrayList与LinkedList核心操作的Big-O复杂度分析
台积电1.4nm工艺A14瞄准2028:10年来性能提升80%
J*a应用程序首次运行自动创建文件与目录的最佳实践
动漫共和国防屏蔽稳定域名-动漫共和国官方正版直达通道
Python中如何避免重复条件判断:利用数据结构实现动态逻辑
实现全屏滚动与导航点:专业教程
C++如何比较两个字符串_C++ string compare函数与操作符对比
Win11怎么设置鼠标指针速度_Win11提高鼠标指针精确度选项
uc浏览器网页版入口 uc浏览器网页版最新网址
Go调试环境为何无法启动_Go调试器启动失败原因与解决策略
qq音乐在线播放入口_qq音乐电脑版登录链接
新手怎么开始学化妆 零基础化妆入门教程
J*aScript类型检查_j*ascript代码规范
如何修改开机登录密码_Windows账户安全设置超详细教程【必学】
响应式图片在网页设计中的正确实现方法
如何在Promise链中优雅地中断后续then执行
Win11输入法不见了怎么办_Windows11恢复语言栏显示方法
Mac怎么锁定备忘录_Mac备忘录加密设置教程
Golang如何使用new_Go new分配内存机制讲解
Golang如何使用const iota_Go iota常量计数器讲解
css链接悬停下划线样式如何自定义_使用::after结合content和transition
海棠账号登录入口_登录海棠账户同步阅读记录
J*aScript数据结构转换:将对象数组按类别分组
快手赚钱渠道_快手收益来源
MongoDB Aggregation:在嵌套对象数组中精确匹配ObjectId
如何使用Rector自动化升级旧代码_通过Composer安装和配置Rector进行代码重构
Django模型中自动计算可用余额的实现方法
Safari自带网页翻译功能怎么用 无需插件轻松看懂外文网站【方法】
Win10自动更新怎么关闭 Win10永久关闭系统更新的两种方法【终极版】
MAC怎么在地图App里使用“四处看看”_MAC体验部分城市的3D实景街景
抖音网页版怎么|直播|_抖音网页版开播操作指南
Win11截图该按哪些键 Win11截屏完整流程解析【教程】
QQ邮箱官方邮箱登录入口 QQ邮箱网页版快速访问
限制HTML日期输入框的日期选择范围
Win10快速启动功能利弊分析 Win10开启或关闭快速启动教程【技巧】
Win10磁盘清理工具在哪 Win10打开并使用磁盘清理【教程】
HTML空白字符处理机制:渲染、DOM与编码实践
uc浏览器网页版极速入口 uc网页浏览器网页版流畅体验
Angular中父组件异步更新子组件复选框状态的实践指南
解决Tabulator日期时间排序问题的专业指南
html两个JS只运行一个怎么办_让双JS在html中都运行方法【技巧】
Python自定义类排序:解决lambda键值访问TypeError的实践指南
Win11蓝牙耳机断连怎么解决 Win11蓝牙设置重新配对与驱动更新【技巧】
12306选座怎么选到商务座_12306商务座选择与配置说明


2025-11-21
浏览次数:次
返回列表
g.org/x/crypto/openpgp"
"golang.org/x/crypto/openpgp/armor"
"golang.org/x/crypto/openpgp/packet"
)
// SignPubKeyWithUserID 为给定的公钥实体签名指定的用户ID
// asciiPub: ASCII Armor编码的公钥字符串
// asciiPri: ASCII Armor编码的私钥字符串
// priPwd: 私钥的密码
// userIDToSign: 要签名的用户ID字符串
// 返回 ASCII Armor 编码的已签名公钥,或错误
func SignPubKeyWithUserID(asciiPub, asciiPri, priPwd, userIDToSign string) (string, error) {
// 1. 加载私钥实体
priEntity, err := readArmoredKeyRing(asciiPri)
if err != nil {
return "", fmt.Errorf("加载私钥失败: %w", err)
}
if len(priEntity) == 0 || priEntity[0].PrivateKey == nil {
return "", fmt.Errorf("未找到有效的私钥实体")
}
signer := priEntity[0] // 假设我们使用密钥环中的第一个私钥作为签名者
// 2. 解密私钥
if signer.PrivateKey.Encrypted {
err = signer.PrivateKey.Decrypt([]byte(priPwd))
if err != nil {
return "", fmt.Errorf("解密私钥失败: %w", err)
}
}
// 3. 加载公钥实体
pubEntityList, err := readArmoredKeyRing(asciiPub)
if err != nil {
return "", fmt.Errorf("加载公钥失败: %w", err)
}
if len(pubEntityList) == 0 {
return "", fmt.Errorf("未找到有效的公钥实体")
}
targetPubKey := pubEntityList[0] // 假设我们签名密钥环中的第一个公钥
// 4. 查找要签名的用户ID
foundUserID := false
for _, identity := range targetPubKey.Identities {
if identity.UserId.Id == userIDToSign {
// 5. 使用签名者的私钥对公钥的用户ID进行签名
// 注意:这里调用的是 openpgp.Entity 的 SignIdentity 方法
// 如果 openpgp 库版本正确,此方法会调用正确的底层签名逻辑
err = targetPubKey.SignIdentity(userIDToSign, signer, nil)
if err != nil {
return "", fmt.Errorf("签名用户ID '%s' 失败: %w", userIDToSign, err)
}
foundUserID = true
break
}
}
if !foundUserID {
return "", fmt.Errorf("在公钥中未找到要签名的用户ID: '%s'", userIDToSign)
}
// 6. 将签名的公钥实体序列化回ASCII Armor格式
signedKeyArmor, err := writeArmoredKey(targetPubKey)
if err != nil {
return "", fmt.Errorf("序列化已签名公钥失败: %w", err)
}
return signedKeyArmor, nil
}
// readArmoredKeyRing 从ASCII Armor字符串中读取密钥环
func readArmoredKeyRing(armoredKey string) (openpgp.EntityList, error) {
reader := bytes.NewReader([]byte(armoredKey))
entityList, err := openpgp.ReadArmoredKeyRing(reader)
if err != nil {
return nil, fmt.Errorf("读取ASCII Armor密钥环失败: %w", err)
}
return entityList, nil
}
// writeArmoredKey 将openpgp.Entity序列化为ASCII Armor字符串
func writeArmoredKey(entity *openpgp.Entity) (string, error) {
buf := new(bytes.Buffer)
writer, err := armor.Encode(buf, openpgp.PublicKeyType, nil)
if err != nil {
return "", fmt.Errorf("创建ASCII Armor编码器失败: %w", err)
}
err = entity.Serialize(writer)
if err != nil {
return "", fmt.Errorf("序列化实体失败: %w", err)
}
err = writer.Close()
if err != nil {
return "", fmt.Errorf("关闭编码器失败: %w", err)
}
return buf.String(), nil
}
func main() {
// 替换为你的实际ASCII Armor编码的公钥、私钥和密码
// 警告:在生产环境中,密钥不应硬编码或直接暴露
// 这里的示例密钥是虚拟的,请勿直接使用
dummyPublicKeyArmor := `-----BEGIN PGP PUBLIC KEY BLOCK-----
Version: GnuPG v2
mQENBF+fL3YBCADr51k/g770/p/iK6f4W8V2q6b6v9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j9z0j