词条 | x509 |
释义 | X.509是一种非常通用的证书格式。所有的证书都符合ITU-T X.509国际标准;因此(理论上)为一种应用创建的证书可以用于任何其他符合X.509标准的应用。在一份证书中,必须证明公钥及其所有者的姓名是一致的。对X.509证书来说,认证者总是 CA或由CA指定的人,一份X.509证书是一些标准字段的集合,这些字段包含有关用户或设备及其相应公钥的信息。X.509标准定义了证书中应该包含哪些信息,并描述了这些信息是如何编码的(即数据格式),所有的X.509证书包含以下数据: 1、X.509版本号:指出该证书使用了哪种版本的X.509标准,版本号会影响证书中的一些特定信息。目前的版本是3。 2、证书持有人的公钥:包括证书持有人的公钥、算法(指明密钥属于哪种密码系统)的标识符和其他相关的密钥参数。 3、证书的序列号:由CA给予每一个证书分配的唯一的数字型编号,当证书被取消时,实际上是将此证书序列号放入由CA签发的CRL(Certificate Revocation List证书作废表,或证书黑名单表)中。这也是序列号唯一的原因。 4、主题信息:证书持有人唯一的标识符(或称DN-distinguished name)这个名字在 Internet上应该是唯一的。DN由许多部分组成,看起来象这样: CN=Bob Allen, OU=Total Network Security Division O=Network Associates, Inc. C=US 这些信息指出该科目的通用名、组织单位、组织和国家或者证书持有人的姓名、服务处所等信息。 5、证书的有效期:证书起始日期和时间以及终止日期和时间;指明证书在这两个时间内有效。 6、认证机构:证书发布者,是签发该证书的实体唯一的CA的X.500名字。使用该证书意味着信任签发证书的实体。(注意:在某些情况下,比如根或顶级CA证书,发布者自己签发证书) 7、发布者的数字签名:这是使用发布者私钥生成的签名,以确保这个证书在发放之后没有被撰改过。 8、签名算法标识符:用来指定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 { 解析X509证书 1.从磁盘上的证书文件中读取证书数据 unsigned char* pbX509Data; // 证书数据 unsigned long ulX509DataLen; // 证书数据长度 2.获取CertContext PCCERT_CONTEXT pCertContext = CertCreateCertificateContext(X509_ASN_ENCODING, pbX509Data, ulX509DataLen); 3.获取证书信息 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); 4.创建临时密钥容器 HCRYPTPROV hTmpProv = NULL; CryptAcquireContext(&hTmpProv, "My_Temporary_Container", NULL, PROV_RSA_AES, 0); // NULL表示使用系统默认CSP 5.向容器中导入公钥,获取公钥句柄 HCRYPTKEY hKey = NULL; CERT_PUBLIC_KEY_INFO certPubKeyInfo = pCertContext->pCertInfo->SubjectPublicKeyInfo; CryptImportPublicKeyInfo(hTmpProv, X509_ASN_ENCODING|PKCS_7_ASN_ENCODING, &certPubKeyInfo, &hKey); 6.导出公钥(最好采用二次调用方式) 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); 7.获取公钥信息 unsigned char* p = pBuf + sizeof(PUBLICKEYSTRUC); (*(RSAPUBKEY*)p).bitlen; // 公钥模长(以bit为单位) (*(RSAPUBKEY*)p).pubexp; // 公钥的e(注意字节顺序) p += sizeof(RSAPUBKEY); // 公钥的n(注意字节顺序) 8.清理工作 delete[] pBuf; pBuf = NULL; CryptDestroyKey(hKey); CryptReleaseContext(hTmpProv, 0); CertFreeCertificateContext(pCertContext); |
随便看 |
百科全书收录4421916条中文百科知识,基本涵盖了大多数领域的百科知识,是一部内容开放、自由的电子版百科全书。