ASN.1与数字证书

2020-4 基于BouncyCastle的ASN.1分析工具设计与实现 Week3

  • X.509是一种通用的证书格式,所有有效证书都符合ITU-T X.509国际标准;
  • 因此(理论上)为一种应用创建的证书可以用于任何其他符合X.509标准的应用。在一份证书中,必须证明公钥及其所有者的姓名是一致 的。对X.509证书来说,认证者总是 CA或由CA指定的人,一份X.509证书是一些标准字段的集合,这些字段包含有关用户或设备及其相应公钥的信息。X.509标准定义了证书中应该包含哪 些信息,并描述了这些信息是如何编码的(即数据格式),所有的X.509证书包含以下数据: 
    • X.509版本号:指出该证书使用了哪种版本的 X.509标准,版本号会影响证书中的一些特定信息。目前的版本是3。
    • 证书持有人的公钥:包括证书持有人的公钥、算法(指明密钥属于哪种密码系统)的标识符和其他相关的密钥参数。
    • 证书的序列号:由CA给予每一个证书分配的唯一的数字型编号,当证书被取消时,实际上是将此证书序列号放入由CA签发的CRL(Certificate Revocation List证书作废表,或证书黑名单表)中。这也是序列号唯一的原因。
    • 主题信息:证书持有人唯一的标识符(或称DN-distinguished name)这个名字在 Internet上应该是唯一的。DN由许多部分组成,
      CN=Bob Allen, OU=Total Network Security Division O=Network Associates, Inc. C=US
    • 这些信息指出该科目的通用名、组织单位、组织和国家或者证书持有人的姓名、服务处所等信息。
    • 证书的有效期:证书起始日期和时间以及终止日期和时间;指明证书在这两个时间内有效。
    • 认证机构:证书发布者,是签发该证书的实体唯一的CA的X.500名字。使用该证书意味着信任签发证书的实体。(注意:在某些情况下,比如根或顶级CA证书,发布者自己签发证书)
    • 发布者的数字签名:这是使用发布者私钥生成的签名,以确保这个证书在发放之后没有被撰改过。
    • 签名算法标识符:用来指定CA签署证书时所使用的签名算法。算法标识符用来指定CA签发证书时所使用的公开密钥算法和HASH算法。
      X.509证书格式  为了利用公共密钥这种密码系统,必须将公共密钥分发出去。最通用的一种签名证书格式 被称为X.509格式。X.509格式的证书被VeriSign、微软、网景和其他许多公司广泛应用于对电子邮件消息进行签名,对程序代码进行认证,以及 对许多其他类型的数据进行认证等等。X.509标准是由国际电话标准机构,即国际电报电话咨询委员会(CCITT)提出的用于目录服务的X.500系列建 议的组成部分。
  • X.509证书的具体结构是用一种形式化表示来描述的,称为"抽象语法表示法#1"(abstract syntax notation)即ASN.1。图9-13显示了第三版X.509格式的ASN.1定义。虽然具体的语法对我们并不重要,但是你可以看到,ASN.1为 证书文件的结构给出了精确的定义。"基本编码规则"(basic encoding rules),即BER,精确地描述了如何将该结构保存为二进制文件。也就是说,BER描述了如何对整数、字符串、位串以及诸如SEQUENCE、 CHOICE和OPTIONAL的结构进行编码的方法。
    [Certificate ::= SEQUENCE {
    tbsCertificate TBSCertificate,
    signatureAlgorithm AlgorithmIdentifier,
    signature BIT STRING
    }
    TBSCertificate ::= SEQUENCE {
    version [0] EXPLICIT Version DEFAULT v1,
    serialNumber CertificateSerialNumber,
    signature AlgorithmIdentifier,
    issuer Name,
    validity Validity,
    subject Name,
    subjectPublicKeyInfo SubjectPublicKeyInfo,
    issuerUniqueID [1] IMPLICIT UniqueIdentifier OPTIONAL,
    -- If present, version must be v2or v3
    subjectUniqueID [2] IMPLICIT UniqueIdentifier OPTIONAL,
    -- If present, version must be v2or v3
    extensions [3] EXPLICIT Extensions OPTIONAL
    -- If present, version must be v3
    }
    Version ::= INTEGER {
    v1(0), v2(1), v3(2)
    }
    CertificateSerialNumber ::= INTEGER
    Validity ::= SEQUENCE {
    notBefore CertificateValidityDate,
    notAfter CertificateValidityDate
    }
    CertificateValidityDate ::= CHOICE {
    utcTime UTCTime,
    generalTime GeneralizedTime
    }
    UniqueIdentifier ::= BIT STRING
    SubjectPublicKeyInfo ::= SEQUENCE {
    algorithm AlgorithmIdentifier,
    subjectPublicKey BIT STRING
    }
    Extensions ::= SEQUENCE OF Extension
    Extension ::= SEQUENCE {
    extnID OBJECT IDENTIFIER,
    critical BOOLEAN DEFAULT FALSE,
    extnValue OCTET STRING
    }
    

解析X509证书

  • 从磁盘上的证书文件中读取证书数据

    unsigned char* pbX509Data; // 证书数据
    unsigned long ulX509DataLen; // 证书数据长度
    
  • 获取CertContext

    PCCERT_CONTEXT pCertContext = CertCreateCertificateContext(X509_ASN_ENCODING, pbX509Data, ulX509DataLen);
    
  • 获取证书信息

    pCertContext->pCertInfo->dwVersion; // 证书版本号
    CRYPT_INTEGER_BLOB snBlob = pCertContext->pCertInfo->SerialNumber; // 证书SN
    CERT_NAME_BLOB issuerBlob = pCertContext->pCertInfo->Issuer; // 证书颁发者
    CERT_NAME_BLOB subjectBlob = pCertContext->pCertInfo->Subject; // 证书主题
    // 证书有效起始日期
    SYSTEMTIME sysTime;
    memset(&sysTime, 0, sizeof(sysTime));
    FileTimeToSystemTime(&pCertContext->pCertInfo->NotBefore, &sysTime);
    char szTime[128] = {0};
    sprintf_s(szTime, 128, "%d年%d月%d日 %d:%d:%d", sysTime.wYear, sysTime.wMonth, sysTime.wDay, sysTime.wHour, sysTime.wMinute, sysTime.wSecond);
    // 证书有效终止日期
    memset(&sysTime, 0, sizeof(sysTime));
    FileTimeToSystemTime(&pCertContext->pCertInfo->NotAfter, &sysTime);
    memset(szTime, 0, sizeof(szTime));
    sprintf_s(szTime, 128, "%d年%d月%d日 %d:%d:%d", sysTime.wYear, sysTime.wMonth, sysTime.wDay, sysTime.wHour, sysTime.wMinute, sysTime.wSecond);
    
  • 创建临时密钥容器

    HCRYPTPROV hTmpProv = NULL;
    CryptAcquireContext(&hTmpProv, "My_Temporary_Container", NULL, PROV_RSA_AES, 0); // NULL表示使用系统默认CSP
    
  • 向容器中导入公钥,获取公钥句柄

    HCRYPTKEY hKey = NULL;
    CERT_PUBLIC_KEY_INFO certPubKeyInfo = pCertContext->pCertInfo->SubjectPublicKeyInfo;
    CryptImportPublic KeyInfo(hTmpProv, X509_ASN_ENCODING|PKCS_7_ASN_ENCODING, &certPubKeyInfo, &hKey);
    
  • 导出公钥(最好采用二次调用方式)

    unsigned char* pBuf = NULL;
    unsigned long ulBufLen = 0;
    CryptExportKey(hKey, 0, PUBLICKEYBLOB, 0, pBuf, &ulBufLen);
    pBuf = new unsigned char[ulBufLen];
    memset(pBuf, 0, ulBufLen);
    CryptExportKey(hKey, 0, PUBLICKEYBLOB, 0, pBuf, &ulBufLen);
    
  • 获取公钥信息

    unsigned char* p = pBuf + sizeof(PUBLICKEYSTRUC);
    (*(RSAPUBKEY*)p).bitlen; // 公钥模长(以bit为单位)
    (*(RSAPUBKEY*)p).pubexp; // 公钥的e(注意字节顺序)
    p += sizeof(RSAPUBKEY); // 公钥的n(注意字节顺序)
    
  • 清理

    delete[] pBuf;
    pBuf = NULL;
    CryptDestroyKey(hKey);
    CryptReleaseContext(hTmpProv, 0);
    CertFreeCertificateContext(pCertContext);
    

猜你喜欢

转载自www.cnblogs.com/SANFENs/p/12747529.html