reflection - 从 Golang 类型中提取未导出字段的正确方法是什么?

标签 reflection ssh go type-conversion rsa

我正在尝试使用 SSH 公钥(x.crypto.ssh.PublicKey,通过解析 authorized_keys 生成)并提取一个普通的旧 crypto.rsa.PublicKey 来自它(我知道 key 是 ssh-rsa 类型)。这意味着,在 ssh-land 中, key 是(未导出的)类型 rsaPublicKey

通过使用 reflect.Interface() 或普通的 fmt.Sprintf("%v") 我可以获得如下所示的值:

&{133141753607255377833524803712241410600224583899447743985716492314976605735038858392516407436607315136967439536840885454950028631410155683051419836325912444338003188332463816050354540934271243064035037856360864190161298769948076508355604915775510460445701536855931828420791030023079646791573156484306628882221 35}

我知道,通过阅读 ssh 包源代码,ssh.rsaPublickey 实际上被定义为我想要的类型:

type rsaPublicKey rsa.PublicKey

因此这个值已经是我想要的东西了。 (实际上,使用 "%+v",我什至可以看到它的字段(当然)被命名为 NE。 )

我的问题是,除了使用 fmt 将这些数字中的每一个提取到字符串中,我不知道有什么更好的方法可以让 Go 相信这实际上是一个 rsa.PublicKey。 sscan,然后分别转换成big.Intint,然后用我刚才得到的这两个值作为模数和指数来创建一个新的rsa.PublicKey.

这行得通,但看起来令人难以置信。我明白了,因为它没有被导出,可能没有直接的路径,但肯定有比重新转换它们的可打印表示更好的方法来获取这些字段。

ssh.PublicKey 确实有一个 Marshal() 函数,但这给了我有线格式表示,如果我通过那里,我猜避免滥用类型系统,但我最终会在那里编写自己的字节流到大 Int 到 RSA 转换器,这似乎并没有真正的帮助。到那时,我还不如从 authorized_keys 文件开始,然后自己解析它,这看起来也很愚蠢。

正确的做法是什么?

最佳答案

您可以使用reflect将该接口(interface)值转换回原始rsa.PublicKey:

rsaKey := reflect.ValueOf(sshKey).
    Convert(reflect.TypeOf(&rsa.PublicKey{})).Interface().(*rsa.PublicKey)

带有类似示例的 Playground :http://play.golang.org/p/bcYhYHrIl_ .

关于reflection - 从 Golang 类型中提取未导出字段的正确方法是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31593329/

相关文章:

c# - 解析 C# 代码(作为字符串)并插入其他方法

javascript - 给定对象在对象中的位置,如何修改对象属性

html - 带有附加参数的 HTML 中的 ssh://链接

git - 如何配置系统通过 ssh 反向隧道/代理使用 git?

c# - 如何从未引用的程序集中获取类型?

mysql - 用于 Mysql 的 Ole 数据库连接器

go - 如何在 golang 网络服务器上设置 HTTPS?

go - 接口(interface)的复制或新实例

go - 使用 Kite 和 Kontrol 的分布式微服务

java - Java 模块指令如何影响对模块的反射访问?