跳到主要内容

证书CMS解封装

从API 22开始,支持证书CMS解封装。

PKCS#7是用于存储签名或加密数据的标准语法。CMS作为PKCS#7的扩展,支持的数据类型包括数据、签名数据、封装数据、签名和封装数据、摘要数据以及加密数据。该标准常用于保护数据的完整性和机密性。

目前仅支持CMS签名数据和封装数据。

开发步骤

  1. 导入证书算法库框架模块

    import { cert } from '@kit.DeviceCertificateKit';
  2. 调用证书CMS封装进行CMS数据封装。

  3. 调用cert.createCmsParser创建CmsParser对象。

  4. 调用cmsParser.setRawData设置CMS数据。

  5. 调用cmsParser.decryptEnvelopedData解密封装数据。

解封装示例:

import { cert } from '@kit.DeviceCertificateKit';

let ECC_256_PUBKEY: string =
'-----BEGIN CERTIFICATE-----\n' +
'MIICGDCCAb6gAwIBAgIGAXKnJjrAMAoGCCqGSM49BAMCMFcxCzAJBgNVBAYTAkNO\n' +
'MQ8wDQYDVQQIDAbpmZXopb8xDzANBgNVBAcMBuilv+WuiTEPMA0GA1UECgwG5rWL\n' +
'6K+VMRUwEwYDVQQDDAzkuK3mlofmtYvor5UwHhcNMjUwOTE2MDY0MTMwWhcNMzUw\n' +
'OTE0MDY0MTMwWjBXMQswCQYDVQQGEwJDTjEPMA0GA1UECAwG6ZmV6KW/MQ8wDQYD\n' +
'VQQHDAbopb/lrokxDzANBgNVBAoMBua1i+ivlTEVMBMGA1UEAwwM5Lit5paH5rWL\n' +
'6K+VMFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEB06h4SzOryi3d7PW9yN2wACC\n' +
'VxlduBQjVLWZlDKhFKkdZjve8mUyytSSbBj/rrzR2XmzUzofuNkUbAtje3DDJqN2\n' +
'MHQwHQYDVR0OBBYEFNtUldgBESf31bwTnYtApIctaSdtMB8GA1UdIwQYMBaAFNtU\n' +
'ldgBESf31bwTnYtApIctaSdtMAsGA1UdDwQEAwIBBjAJBgNVHREEAjAAMAkGA1Ud\n' +
'EgQCMAAwDwYDVR0TAQH/BAUwAwEB/zAKBggqhkjOPQQDAgNIADBFAiEAzxzaG2vR\n' +
'zUnFFL3X3lRQ0IOJrb6cvkSZuaFd4bW2lgUCIHW6QGGnECDFMbDNz7Og9kjkt+3k\n' +
'FmEJWqEMYudBH3Ul\n' +
'-----END CERTIFICATE-----';
let ECC_256_PRIVATE: string =
'-----BEGIN PRIVATE KEY-----\n' +
'MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQgOYwEyIw3ZNIAL4xO\n' +
'pP6eVcQYcrL2sfnt6vB0z9tKmMmhRANCAAQHTqHhLM6vKLd3s9b3I3bAAIJXGV24\n' +
'FCNUtZmUMqEUqR1mO97yZTLK1JJsGP+uvNHZebNTOh+42RRsC2N7cMMm\n' +
'-----END PRIVATE KEY-----';

// string转Uint8Array。
function stringToUint8Array(str: string): Uint8Array {
let arr: number[] = [];
for (let i = 0, j = str.length; i < j; i++) {
arr.push(str.charCodeAt(i));
};
return new Uint8Array(arr);
}

async function createX509Cert(inStream: string): Promise<cert.X509Cert> {
let encodingBlob: cert.EncodingBlob = {
data: stringToUint8Array(inStream),
encodingFormat: cert.EncodingFormat.FORMAT_PEM
};
let x509Cert: cert.X509Cert = await cert.createX509Cert(encodingBlob);
return x509Cert;
}

async function testCmsDecryptTest() {
try {
let plainText: Uint8Array = new Uint8Array([0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07]);
let x509CertEc: cert.X509Cert = await createX509Cert(ECC_256_PUBKEY);
let cms: cert.CmsGenerator = cert.createCmsGenerator(cert.CmsContentType.ENVELOPED_DATA);
let option: cert.CmsGeneratorOptions = {
outFormat: cert.CmsFormat.PEM
};
cms.setRecipientEncryptionAlgorithm(cert.CmsRecipientEncryptionAlgorithm.AES_128_GCM);
let recipientInfo: cert.CmsRecipientInfo = {
keyAgreeInfo: {
cert: x509CertEc,
digestAlgorithm: cert.CmsKeyAgreeRecipientDigestAlgorithm.SHA256
}
};
await cms.addRecipientInfo(recipientInfo);
console.info('add recipient result: success, digestAlgorithm = ' + recipientInfo.keyAgreeInfo?.digestAlgorithm);
let envelopeData = await cms.doFinal(plainText, option);
console.info('doFinal result: success, envelopeData = ' + envelopeData);
let cipherText = await cms.getEncryptedContentData();
console.info('getEncryptedContentData result: success, cipherText = ' + cipherText);
let config: cert.CmsEnvelopedDecryptionConfig = {
keyInfo: {
key: ECC_256_PRIVATE
},
};
let cmsDecrypt: cert.CmsParser = cert.createCmsParser();
await cmsDecrypt.setRawData(envelopeData, cert.CmsFormat.PEM);
let decPlainText: Uint8Array = await cmsDecrypt.decryptEnvelopedData(config);
console.info('[XTS] decryptEnvelopedData result: success, decPlainText = ' + decPlainText);
console.info('decryptEnvelopedData result: success.');
} catch (error) {
console.error(`verifySignedData failed: errCode: ${error.code}, message: ${error.message}`);
}
}