Chilkat HOME Android™ AutoIt C C# C++ Chilkat2-Python CkPython Classic ASP DataFlex Delphi DLL Go Java Node.js Objective-C PHP Extension Perl PowerBuilder PowerShell PureBasic Ruby SQL Server Swift Tcl Unicode C Unicode C++ VB.NET VBScript Visual Basic 6.0 Visual FoxPro Xojo Plugin
(Unicode C) Validate CAdES-T Signature (.p7m)Validates a CAdES-T CMS signature and extracts the time-stamp token and gets information about it. Also validates the time-stamp token. Note: This example requires Chilkat v9.5.0.78 or greater.
#include <C_CkCrypt2W.h> #include <C_CkJsonObjectW.h> #include <C_CkDtObjW.h> void ChilkatSample(void) { HCkCrypt2W crypt; BOOL success; HCkJsonObjectW cmsOptions; HCkJsonObjectW json; int i; int count_i; const wchar_t *strVal; const wchar_t *certSerialNumber; const wchar_t *certIssuerCN; const wchar_t *certIssuerDN; const wchar_t *certDigestAlgOid; const wchar_t *certDigestAlgName; const wchar_t *contentType; HCkDtObjW signingTime; const wchar_t *messageDigest; const wchar_t *signingAlgOid; const wchar_t *signingAlgName; const wchar_t *authAttrContentTypeName; const wchar_t *authAttrContentTypeOid; const wchar_t *authAttrSigningTimeName; HCkDtObjW authAttrSigningTimeUtctime; const wchar_t *authAttrMessageDigestName; const wchar_t *authAttrMessageDigestDigest; const wchar_t *authAttrSigningCertificateV2Name; const wchar_t *authAttrSigningCertificateV2Der; const wchar_t *unauthAttrTimestampTokenName; const wchar_t *unauthAttrTimestampTokenDer; BOOL unauthAttrTimestampTokenTimestampSignatureVerified; const wchar_t *unauthAttrTimestampTokenTstInfoTsaPolicyId; const wchar_t *unauthAttrTimestampTokenTstInfoMessageImprintHashAlg; const wchar_t *unauthAttrTimestampTokenTstInfoMessageImprintDigest; BOOL unauthAttrTimestampTokenTstInfoMessageImprintDigestMatches; const wchar_t *unauthAttrTimestampTokenTstInfoSerialNumber; HCkDtObjW unauthAttrTimestampTokenTstInfoGenTime; int j; int count_j; // This example requires the Chilkat API to have been previously unlocked. // See Global Unlock Sample for sample code. crypt = CkCrypt2W_Create(); // Indicate that the CAdES-T timestamp tokens must also pass validation for the signature to be validated. cmsOptions = CkJsonObjectW_Create(); CkJsonObjectW_UpdateBool(cmsOptions,L"ValidateTimestampTokens",TRUE); CkCrypt2W_putCmsOptions(crypt,CkJsonObjectW_emit(cmsOptions)); // Validate the .p7m and extract the original signed data to an output file. // Note: The timestampToken is an unauthenticated attribute. See the code below that parses the LastJsonData // for details about examining timestampToken. success = CkCrypt2W_VerifyP7M(crypt,L"qa_data/cades/CAdES-T/Signature-C-T-1.p7m",L"qa_output/out.dat"); // Get information about the CMS signature in the LastJsonData. // The detailed results of the signature validation are available in LastJsonData. // (If the non-success return status was caused by an error such as "file not found", then the // LastJsonData would be empty.) json = CkCrypt2W_LastJsonData(crypt); CkJsonObjectW_putEmitCompact(json,FALSE); wprintf(L"%s\n",CkJsonObjectW_emit(json)); // Here is a sample result: // See the parsing code below.. // Use this online tool to generate parsing code from sample JSON: // Generate Parsing Code from JSON // { // "pkcs7": { // "verify": { // "digestAlgorithms": [ // "sha256" // ], // "signerInfo": [ // { // "cert": { // "serialNumber": "00DCB814678CDB", // "issuerCN": "LevelBCAOK", // "issuerDN": "", // "digestAlgOid": "2.16.840.1.101.3.4.2.1", // "digestAlgName": "SHA256" // }, // "contentType": "1.2.840.113549.1.7.1", // "signingTime": "131203065741Z", // "messageDigest": "JJZt41Nt8VsYahP+Xti4rR3vBDkUfRd6gquItl6R5Os=", // "signingAlgOid": "1.2.840.113549.1.1.1", // "signingAlgName": "RSA-PKCSV-1_5", // "authAttr": { // "1.2.840.113549.1.9.3": { // "name": "contentType", // "oid": "1.2.840.113549.1.7.1" // }, // "1.2.840.113549.1.9.5": { // "name": "signingTime", // "utctime": "131203065741Z" // }, // "1.2.840.113549.1.9.4": { // "name": "messageDigest", // "digest": "JJZt41Nt8VsYahP+Xti4rR3vBDkUfRd6gquItl6R5Os=" // }, // "1.2.840.113549.1.9.16.2.47": { // "name": "signingCertificateV2", // "der": "MIGIMIGFMIGCBCBJrxOU0w0dWGsVovjLv9QDH3syB5mLVv3grSYA40x9IDBeMFOkUTBPMQswCQYDVQQGEwJGUjENMAsGA1UEChMERVRTSTEcMBoGA1UECwwTUGx1Z3Rlc3RzXzIwMTMtMjAxNDETMBEGA1UEAxMKTGV2ZWxCQ0FPSwIHANy4FGeM2w==" // } // }, // "unauthAttr": { // "1.2.840.113549.1.9.16.2.14": { // "name": "timestampToken", // "der": "MIIL+AYJKoZI...u7CfcjURNTY=", // "verify": { // "digestAlgorithms": [ // "sha256" // ], // "signerInfo": [ // { // "cert": { // "serialNumber": "01AA4592D36C61", // "issuerCN": "RootCAOK", // "issuerDN": "", // "digestAlgOid": "2.16.840.1.101.3.4.2.1", // "digestAlgName": "SHA256" // }, // "contentType": "1.2.840.113549.1.9.16.1.4", // "messageDigest": "NSsMUrfoyCQ0OszPE1YLx1j3EyyCiBmnE5Sua6ghu/Q=", // "signingAlgOid": "1.2.840.113549.1.1.1", // "signingAlgName": "RSA-PKCSV-1_5", // "authAttr": { // "1.2.840.113549.1.9.3": { // "name": "contentType", // "oid": "1.2.840.113549.1.9.16.1.4" // }, // "1.2.840.113549.1.9.4": { // "name": "messageDigest", // "digest": "NSsMUrfoyCQ0OszPE1YLx1j3EyyCiBmnE5Sua6ghu/Q=" // }, // "1.2.840.113549.1.9.16.2.47": { // "name": "signingCertificateV2", // "der": "MIGGMIGDMIGABCDB/np5UxvhcPnSxD2Kme+C88uXGCMWLAvFPHNvTApTWDBcMFGkTzBNMQswCQYDVQQGEwJGUjENMAsGA1UEChMERVRTSTEcMBoGA1UECwwTUGx1Z3Rlc3RzXzIwMTMtMjAxNDERMA8GA1UEAxMIUm9vdENBT0sCBwGqRZLTbGE=" // } // } // } // ] // }, // "timestampSignatureVerified": true, // "tstInfo": { // "tsaPolicyId": "1.3.6.1.4.1.2706.2.2.5.2.1.1.1", // "messageImprint": { // "hashAlg": "sha256", // "digest": "C8xEe9NA4X1cUyHGX9zG89ipmQ2byFs3aa+Xe4Fz2P0=", // "digestMatches": true // }, // "serialNumber": "313E162121D922", // "genTime": "20131203065742Z" // } // } // } // } // ] // } // } // } // signingTime = CkDtObjW_Create(); authAttrSigningTimeUtctime = CkDtObjW_Create(); unauthAttrTimestampTokenTstInfoGenTime = CkDtObjW_Create(); // Iterate over the hash algorithms used in the signature. i = 0; count_i = CkJsonObjectW_SizeOfArray(json,L"pkcs7.verify.digestAlgorithms"); while (i < count_i) { CkJsonObjectW_putI(json,i); strVal = CkJsonObjectW_stringOf(json,L"pkcs7.verify.digestAlgorithms[i]"); i = i + 1; } // For each signer... i = 0; count_i = CkJsonObjectW_SizeOfArray(json,L"pkcs7.verify.signerInfo"); while (i < count_i) { CkJsonObjectW_putI(json,i); // Get information about the certificate used by this signer. certSerialNumber = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].cert.serialNumber"); certIssuerCN = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].cert.issuerCN"); certIssuerDN = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].cert.issuerDN"); certDigestAlgOid = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].cert.digestAlgOid"); certDigestAlgName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].cert.digestAlgName"); // Get additional information for this signer, such as the signingTime, signature algorithm, etc. contentType = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].contentType"); CkJsonObjectW_DtOf(json,L"pkcs7.verify.signerInfo[i].signingTime",FALSE,signingTime); messageDigest = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].messageDigest"); signingAlgOid = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].signingAlgOid"); signingAlgName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].signingAlgName"); // -------------------------------- // Examine authenticated attributes. // -------------------------------- // contentType if (CkJsonObjectW_HasMember(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.3\"") == TRUE) { authAttrContentTypeName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.3\".name"); authAttrContentTypeOid = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.3\".oid"); } // signingTime if (CkJsonObjectW_HasMember(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.5\"") == TRUE) { authAttrSigningTimeName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.5\".name"); CkJsonObjectW_DtOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.5\".utctime",FALSE,authAttrSigningTimeUtctime); } // messageDigest if (CkJsonObjectW_HasMember(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.4\"") == TRUE) { authAttrMessageDigestName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.4\".name"); authAttrMessageDigestDigest = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.4\".digest"); } // signingCertificateV2 if (CkJsonObjectW_HasMember(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.16.2.47\"") == TRUE) { authAttrSigningCertificateV2Name = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.16.2.47\".name"); authAttrSigningCertificateV2Der = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.16.2.47\".der"); } // -------------------------------- // Examine unauthenticated attributes. // -------------------------------- // timestampToken (the timestampToken is what makes this signature a CAdES-T) if (CkJsonObjectW_HasMember(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\"") == TRUE) { unauthAttrTimestampTokenName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".name"); unauthAttrTimestampTokenDer = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".der"); // This is where we find out if the timestampToken's signature is valid. unauthAttrTimestampTokenTimestampSignatureVerified = CkJsonObjectW_BoolOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".timestampSignatureVerified"); unauthAttrTimestampTokenTstInfoTsaPolicyId = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.tsaPolicyId"); unauthAttrTimestampTokenTstInfoMessageImprintHashAlg = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.messageImprint.hashAlg"); unauthAttrTimestampTokenTstInfoMessageImprintDigest = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.messageImprint.digest"); // Here is where we check to see if the digest in the timestampToken's messageImprint matches the digest of the signature of this signerInfo unauthAttrTimestampTokenTstInfoMessageImprintDigestMatches = CkJsonObjectW_BoolOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.messageImprint.digestMatches"); unauthAttrTimestampTokenTstInfoSerialNumber = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.serialNumber"); // Here is where we get the date/time of the timestampToken (i.e. when it was timestamped) CkJsonObjectW_DtOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.genTime",FALSE,unauthAttrTimestampTokenTstInfoGenTime); // The following code gets details about the validity of the timestampToken's signature... j = 0; count_j = CkJsonObjectW_SizeOfArray(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.digestAlgorithms"); while (j < count_j) { CkJsonObjectW_putJ(json,j); strVal = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.digestAlgorithms[j]"); j = j + 1; } j = 0; count_j = CkJsonObjectW_SizeOfArray(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo"); while (j < count_j) { CkJsonObjectW_putJ(json,j); certSerialNumber = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.serialNumber"); certIssuerCN = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.issuerCN"); certIssuerDN = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.issuerDN"); certDigestAlgOid = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.digestAlgOid"); certDigestAlgName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.digestAlgName"); contentType = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].contentType"); messageDigest = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].messageDigest"); signingAlgOid = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].signingAlgOid"); signingAlgName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].signingAlgName"); authAttrContentTypeName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].authAttr.\"1.2.840.113549.1.9.3\".name"); authAttrContentTypeOid = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].authAttr.\"1.2.840.113549.1.9.3\".oid"); authAttrMessageDigestName = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].authAttr.\"1.2.840.113549.1.9.4\".name"); authAttrMessageDigestDigest = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].authAttr.\"1.2.840.113549.1.9.4\".digest"); authAttrSigningCertificateV2Name = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].authAttr.\"1.2.840.113549.1.9.16.2.47\".name"); authAttrSigningCertificateV2Der = CkJsonObjectW_stringOf(json,L"pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].authAttr.\"1.2.840.113549.1.9.16.2.47\".der"); j = j + 1; } } i = i + 1; } if (success != TRUE) { wprintf(L"%s\n",CkCrypt2W_lastErrorText(crypt)); wprintf(L"CAdES-T verification failed.\n"); } else { wprintf(L"CAdES-T signature is valid.\n"); } CkJsonObjectW_Dispose(json); CkCrypt2W_Dispose(crypt); CkJsonObjectW_Dispose(cmsOptions); CkDtObjW_Dispose(signingTime); CkDtObjW_Dispose(authAttrSigningTimeUtctime); CkDtObjW_Dispose(unauthAttrTimestampTokenTstInfoGenTime); } |
© 2000-2025 Chilkat Software, Inc. All Rights Reserved.