【问题标题】:Go and Node.js base64 encoding, decoding problemsGo 和 Node.js base64 编码、解码问题
【发布时间】:2019-04-17 21:32:28
【问题描述】:

我在 Go 中对字符串进行 base64 编码

然后用javascript解码(我尝试了3种不同的方法)

我无法获得与原始 Go 字符串匹配的 javascript 结果

Go(编码)

a := []byte {138,143,163,224,178,73,161,15,240,121,53,192,198,182,52,245}
fmt.Println("a", string(a), a, len(a))
b := base64.StdEncoding.EncodeToString(a)
fmt.Println("b", b, []byte(b), len([]byte(b)))

js(解码)

 const b = [105,111,43,106,52,76,74,74,111,81,47,119,101,84,88,65,120,114,89,48,57,81,61,61];
 let bString = aesjs.utils.utf8.fromBytes(b);
 console.log("b", bString, b, b.length);

 let a1String = atob(bString);
 let a2String = Base64.decode(bString);
 let a3String = Buffer.from(bString, 'base64').toString('utf8');
 let a1 = aesjs.utils.utf8.toBytes(a1String);
 let a2 = aesjs.utils.utf8.toBytes(a2String);
 let a3 = aesjs.utils.utf8.toBytes(a3String);
 console.log("a", a1, a1.length, a2, a2.length, a3, a3.length);

所有 3 种方法都失败,即 a1 != a, a2 != a, a3 != a

我认为 base64 编码/解码会很简单

我错过了什么?谢谢

编辑:js 代码有一个“错字” - 但问题仍然存在,即 ai != a,甚至 a1 != a2 == a3

编辑:如果原始 a 很简单,那么一切正常。 但是当 a 包含不可打印的字符时,我无法在 js 中解码 它的工作示例: a := []byte {65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65} 在这里它不起作用: a := []byte {138, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65, 65}

【问题讨论】:

  • 你的 go 代码是正确的。所以问题出在JS方面。

标签: javascript go base64


【解决方案1】:

发件人:https://github.com/ricmoo/aes-js/blob/master/README.md

UTF8 should NOT be used to store arbitrary binary data as it is a string encoding format, not a binary encoding format

所以不要使用utf8相关函数来转换原始(非utf8)二进制数据。

【讨论】:

  • 那么解决办法是什么?
【解决方案2】:

在 Go 中,字符串是 UTF8:

in := "Good afternoon, or こんにちは"
buf := []byte(in)
str := base64.StdEncoding.EncodeToString(buf)
fmt.Println(str)

打印 base64 编码(ASCII 安全)表示: R29vZCBhZnRlcm5vb24sIG9yIOOBk+OCk+OBq+OBoeOBrw==

JS 字符串是 UTF-16。因此,您需要从 base64 解码并从 utf8 转换为 utf-16。您可以使用库或辅助函数,例如:

function b64DecodeUnicode(str) {
    return decodeURIComponent(atob(str).split('').map(function(c) {
        return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
    }).join(''));
}

var str = "R29vZCBhZnRlcm5vb24sIG9yIOOBk+OCk+OBq+OBoeOBrw==";
var result = b64DecodeUnicode(str);
"Good afternoon, or こんにちは"

The b64DecodeUnicode() function is copied from this answer, see also the link to MDN for detailed info.

【讨论】:

  • 这适用于正确的 UTF8 字符串,因此我将其标记为正确。不幸的是,它并没有解决我的问题,虽然这不在此处的范围内:我的编码数据是二进制的(加密的结果),我认为这里@colminator 的评论可能适用
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2012-01-12
  • 2011-08-03
  • 2015-07-21
  • 1970-01-01
  • 2020-12-15
  • 2011-09-22
  • 2016-07-13
相关资源
最近更新 更多