如何使用 Javascript WebCrypto API 加载 PKCS#12 数字证书

IT技术 javascript digital-signature digital-certificate webcrypto-api
2021-03-13 00:20:16

我正在尝试使用 WebCrypto API 对数据进行签名,但不是创建私钥/公钥并将其导出到 pkcs#1 或 8,我真的很想使用用户的 PKCS#12 来对数据进行签名。我已经阅读了 W3C 规范,但不能充分了解它,也找不到任何关于如何做到这一点的好材料。现在我想把 ActiveX 和 Java Applet 放在一边。有没有办法调整以下内容:

var buffer = encode(prompt("Please enter your password"));
    //TODO:
    //implement a prompt for a pfx or cert

  return crypto.subtle.importKey("raw", buffer, "PBKDF2", false, usages);
    //TODO:
    //instead of importing it, ask for the certificate's pass to sign data
    //with crypto.subtle.sign

任何指针?

更新 这是我一直在工作的代码

<script src="forge.min.js"></script>

<script>
    var errorsReportedByVerifier;
    errorsReportedByVerifier = checkStorage() && checkBrowserAPIs();
    if (!errorsReportedByVerifier){
        console.log("adding click event");
        document.getElementById('btnPfx').addEventListener('click', handlePFXFile, false);
        storeVariables();
        getVariables();
    }


    function handlePFXFile(evnt) {
        console.log("handling pfx")
        //alert(document.getElementById('pfx').value);

        //error happens in 1st line
        //error object does not accept property replace
        //forge.min.js Line 1, Column: 17823
        var p12Der = forge.util.decode64(document.getElementById('pfx').valueOf());
        //var pkcs12Asn1 = forge.asn1.fromDer(p12Der);
        //var pkcs12 = forge.pkcs12.pkcs12FromAsn1(pkcs12Asn1, false, 'pss');
        console.log("pkcs12");
    }
</script>
1个回答

Web 加密 api 不支持 PKCS #12。您可以使用第三方库将 p12 解码为伪造https://github.com/digitalbazaar/forge#pkcs12并在 webcrypto 中加载 privateKey

读取 PKCS#12 证书

PKCS#12 存储在 DER 中,因此首先从 File 中读取它或使用预先存储的 base64

//Reading certificate from a 'file' form field
var reader = new FileReader();
reader.onload = function(e) {               
    var contents = e.target.result;
    var pkcs12Der = arrayBufferToString(contents)
    var pkcs12B64 = forge.util.encode64(pkcs12Der);     
    //do something else...

}   
reader.readAsArrayBuffer(file);

function arrayBufferToString( buffer ) {
    var binary = '';
    var bytes = new Uint8Array( buffer );
    var len = bytes.byteLength;
    for (var i = 0; i < len; i++) {
        binary += String.fromCharCode( bytes[ i ] );
    }
    return binary;
}

//p12 certificate stored in Base64 format
var pkcs12Der= forge.util.decode64(pkcs12B64);

使用伪造解码 PKCS#12 并提取私钥

然后将DER格式解码为ASN1,让forge读取内容

var pkcs12Asn1 = forge.asn1.fromDer(pkcs12Der);
var pkcs12 = forge.pkcs12.pkcs12FromAsn1(pkcs12Asn1, false, password);

然后从pkcs12所需证书中获取私钥(参见伪造文档)并转换为 PKCS #8 以使用 webcrypto 导入

// load keypair and cert chain from safe content(s) 
for(var sci = 0; sci < pkcs12.safeContents.length; ++sci) {
    var safeContents = pkcs12.safeContents[sci];

    for(var sbi = 0; sbi < safeContents.safeBags.length; ++sbi) {
        var safeBag = safeContents.safeBags[sbi];

        // this bag has a private key
        if(safeBag.type === forge.pki.oids.keyBag) {
            //Found plain private key
            privateKey = safeBag.key;
        } else if(safeBag.type === forge.pki.oids.pkcs8ShroudedKeyBag) {
            // found encrypted private key
            privateKey = safeBag.key;
        } else if(safeBag.type === forge.pki.oids.certBag) {
            // this bag has a certificate...        
        }   
    }
}

转换为 PKCS#8

function _privateKeyToPkcs8(privateKey) {
     var rsaPrivateKey = forge.pki.privateKeyToAsn1(privateKey);
     var privateKeyInfo = forge.pki.wrapRsaPrivateKey(rsaPrivateKey);
     var privateKeyInfoDer = forge.asn1.toDer(privateKeyInfo).getBytes();
     var privateKeyInfoDerBuff = stringToArrayBuffer(privateKeyInfoDer);
     return privateKeyInfoDerBuff;
 }
 function stringToArrayBuffer(data){
     var arrBuff = new ArrayBuffer(data.length);
     var writer = new Uint8Array(arrBuff);
     for (var i = 0, len = data.length; i < len; i++) {
         writer[i] = data.charCodeAt(i);
     }
     return arrBuff;
  }

在 Webcrypto 中导入密钥

最后在 webcrypto 中导入密钥

function _importCryptoKeyPkcs8(privateKey,extractable) {
    var privateKeyInfoDerBuff = _privateKeyToPkcs8(privateKey);

    //Import the webcrypto key
    return crypto.subtle.importKey(
            'pkcs8', 
            privateKeyInfoDerBuff, 
            { name: "RSASSA-PKCS1-v1_5", hash:{name:"SHA-256"}},
            extractable, 
            ["sign"]);        

}
_importCryptoKeyPkcs8(entry.privateKey,extractable).    
        then(function(cryptoKey) {
            //your cryptokey is here!!!
        });

电子签名

使用从上述方法返回的导入的加密密钥,您可以使用 webcrypto 进行签名。

var digestToSign = forge.util.decode64(digestToSignB64);
var digestToSignBuf = stringToArrayBuffer(digestToSign);

crypto.subtle.sign(
            {name: "RSASSA-PKCS1-v1_5"},
            cryptoKey,
            digestToSignBuf)
.then(function(signature){
    signatureB64 = forge.util.encode64(arrayBufferToString(signature))
});

我包括来自 base64 的编码,因为数据转换不是微不足道的

在 pkc12 中,如果您需要构建 AdES 等高级格式,您还拥有认证链

Forge 是纯 javascript。您不需要 html 或 css。¿你构建了缩小的 .js 吗?不要一一导入源文件,因为是依赖关系。请同时提供您正在执行的代码的详细信息
2021-04-26 00:20:16
是否有一个预定义的 html+css 模板,我必须使用伪造库。我收到如下错误:“未捕获的类型错误:无法读取未定义的 pkcs12.js:109 的属性‘类’”
2021-05-10 00:20:16
还有:util.js:1569 Uncaught TypeError: input.replace is not a function util.js.1569 和其他几个错误。我从github下载了一个最新版本
2021-05-17 00:20:16
我已经更新了我一直在处理的代码,我现在有了 forge 的缩小版本并得到了一个不同的错误,我进入了代码并找到了它发生的这一行 t.decode64=function(e){e= e.replace(/[^A-Za-z0-9\+\/\=]/g,"");
2021-05-17 00:20:16
您可以safeBag.type === forge.pki.oids.certBag && safeBag.attributes.localKeyId以 PEM 格式(Base64 编码)存储公共证书 ( )。使用以下伪造功能:forge.pki.certificateToPem(safeBag.cert);解码证书使用forge.pki.certificateFromPem(certificatePem);
2021-05-21 00:20:16