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
(Swift) 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.
func chilkatTest() { // This example requires the Chilkat API to have been previously unlocked. // See Global Unlock Sample for sample code. let crypt = CkoCrypt2()! var success: Bool // Indicate that the CAdES-T timestamp tokens must also pass validation for the signature to be validated. let cmsOptions = CkoJsonObject()! cmsOptions.updateBool("ValidateTimestampTokens", value: true) crypt.cmsOptions = cmsOptions.emit() // 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 = crypt.verifyP7M("qa_data/cades/CAdES-T/Signature-C-T-1.p7m", destPath: "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.) var json: CkoJsonObject? = crypt.lastJsonData() json.emitCompact = false print("\(json!.emit()!)") // 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" // } // } // } // } // ] // } // } // } // var i: Int var count_i: Int var strVal: String? var certSerialNumber: String? var certIssuerCN: String? var certIssuerDN: String? var certDigestAlgOid: String? var certDigestAlgName: String? var contentType: String? let signingTime = CkoDtObj()! var messageDigest: String? var signingAlgOid: String? var signingAlgName: String? var authAttrContentTypeName: String? var authAttrContentTypeOid: String? var authAttrSigningTimeName: String? let authAttrSigningTimeUtctime = CkoDtObj()! var authAttrMessageDigestName: String? var authAttrMessageDigestDigest: String? var authAttrSigningCertificateV2Name: String? var authAttrSigningCertificateV2Der: String? var unauthAttrTimestampTokenName: String? var unauthAttrTimestampTokenDer: String? var unauthAttrTimestampTokenTimestampSignatureVerified: Bool var unauthAttrTimestampTokenTstInfoTsaPolicyId: String? var unauthAttrTimestampTokenTstInfoMessageImprintHashAlg: String? var unauthAttrTimestampTokenTstInfoMessageImprintDigest: String? var unauthAttrTimestampTokenTstInfoMessageImprintDigestMatches: Bool var unauthAttrTimestampTokenTstInfoSerialNumber: String? let unauthAttrTimestampTokenTstInfoGenTime = CkoDtObj()! var j: Int var count_j: Int // Iterate over the hash algorithms used in the signature. i = 0 count_i = json!.size(ofArray: "pkcs7.verify.digestAlgorithms").intValue while i < count_i { json.i = i strVal = json!.string(of: "pkcs7.verify.digestAlgorithms[i]") i = i + 1 } // For each signer... i = 0 count_i = json!.size(ofArray: "pkcs7.verify.signerInfo").intValue while i < count_i { json.i = i // Get information about the certificate used by this signer. certSerialNumber = json!.string(of: "pkcs7.verify.signerInfo[i].cert.serialNumber") certIssuerCN = json!.string(of: "pkcs7.verify.signerInfo[i].cert.issuerCN") certIssuerDN = json!.string(of: "pkcs7.verify.signerInfo[i].cert.issuerDN") certDigestAlgOid = json!.string(of: "pkcs7.verify.signerInfo[i].cert.digestAlgOid") certDigestAlgName = json!.string(of: "pkcs7.verify.signerInfo[i].cert.digestAlgName") // Get additional information for this signer, such as the signingTime, signature algorithm, etc. contentType = json!.string(of: "pkcs7.verify.signerInfo[i].contentType") json!.dt(of: "pkcs7.verify.signerInfo[i].signingTime", bLocal: false, dt: signingTime) messageDigest = json!.string(of: "pkcs7.verify.signerInfo[i].messageDigest") signingAlgOid = json!.string(of: "pkcs7.verify.signerInfo[i].signingAlgOid") signingAlgName = json!.string(of: "pkcs7.verify.signerInfo[i].signingAlgName") // -------------------------------- // Examine authenticated attributes. // -------------------------------- // contentType if json!.hasMember("pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.3\"") == true { authAttrContentTypeName = json!.string(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.3\".name") authAttrContentTypeOid = json!.string(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.3\".oid") } // signingTime if json!.hasMember("pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.5\"") == true { authAttrSigningTimeName = json!.string(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.5\".name") json!.dt(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.5\".utctime", bLocal: false, dt: authAttrSigningTimeUtctime) } // messageDigest if json!.hasMember("pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.4\"") == true { authAttrMessageDigestName = json!.string(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.4\".name") authAttrMessageDigestDigest = json!.string(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.4\".digest") } // signingCertificateV2 if json!.hasMember("pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.16.2.47\"") == true { authAttrSigningCertificateV2Name = json!.string(of: "pkcs7.verify.signerInfo[i].authAttr.\"1.2.840.113549.1.9.16.2.47\".name") authAttrSigningCertificateV2Der = json!.string(of: "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 json!.hasMember("pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\"") == true { unauthAttrTimestampTokenName = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".name") unauthAttrTimestampTokenDer = json!.string(of: "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 = json!.bool(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".timestampSignatureVerified") unauthAttrTimestampTokenTstInfoTsaPolicyId = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.tsaPolicyId") unauthAttrTimestampTokenTstInfoMessageImprintHashAlg = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.messageImprint.hashAlg") unauthAttrTimestampTokenTstInfoMessageImprintDigest = json!.string(of: "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 = json!.bool(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.messageImprint.digestMatches") unauthAttrTimestampTokenTstInfoSerialNumber = json!.string(of: "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) json!.dt(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".tstInfo.genTime", bLocal: false, dt: unauthAttrTimestampTokenTstInfoGenTime) // The following code gets details about the validity of the timestampToken's signature... j = 0 count_j = json!.size(ofArray: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.digestAlgorithms").intValue while j < count_j { json.j = j strVal = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.digestAlgorithms[j]") j = j + 1 } j = 0 count_j = json!.size(ofArray: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo").intValue while j < count_j { json.j = j certSerialNumber = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.serialNumber") certIssuerCN = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.issuerCN") certIssuerDN = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.issuerDN") certDigestAlgOid = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.digestAlgOid") certDigestAlgName = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].cert.digestAlgName") contentType = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].contentType") messageDigest = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].messageDigest") signingAlgOid = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].signingAlgOid") signingAlgName = json!.string(of: "pkcs7.verify.signerInfo[i].unauthAttr.\"1.2.840.113549.1.9.16.2.14\".verify.signerInfo[j].signingAlgName") authAttrContentTypeName = json!.string(of: "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 = json!.string(of: "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 = json!.string(of: "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 = json!.string(of: "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 = json!.string(of: "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 = json!.string(of: "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 { print("\(crypt.lastErrorText!)") print("CAdES-T verification failed.") } else { print("CAdES-T signature is valid.") } json = nil } |
© 2000-2025 Chilkat Software, Inc. All Rights Reserved.