Unicode C
Unicode C
Validate CAdES-T Signature (.p7m)
See more CAdES Examples
Validates a CAdES-T CMS signature and extracts the time-stamp token and gets information about it. Also validates the time-stamp token.Chilkat Unicode C Downloads
#include <C_CkCrypt2W.h>
#include <C_CkJsonObjectW.h>
#include <C_CkDtObjW.h>
void ChilkatSample(void)
{
BOOL success;
HCkCrypt2W crypt;
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;
success = FALSE;
// 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 retrieves and parses the last JSON data.
// 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 last JSON data.
// The detailed results of the signature validation are available in the last JSON data.
// (If the non-success return status was caused by an error such as "file not found", then the
// last JSON data would be empty.)
json = CkJsonObjectW_Create();
CkCrypt2W_GetLastJsonData(crypt,json);
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");
}
CkCrypt2W_Dispose(crypt);
CkJsonObjectW_Dispose(cmsOptions);
CkJsonObjectW_Dispose(json);
CkDtObjW_Dispose(signingTime);
CkDtObjW_Dispose(authAttrSigningTimeUtctime);
CkDtObjW_Dispose(unauthAttrTimestampTokenTstInfoGenTime);
}