From 0bdc8f158699c7d284fdbe60ed9003ef68b9c1f8 Mon Sep 17 00:00:00 2001 From: Christopher Jeffrey Date: Tue, 26 Aug 2014 10:24:27 -0700 Subject: [PATCH] paypro: major refactor. create rfc5280 extension parsing function. --- lib/PayPro.js | 410 ++++++++++++++++++-------------------------------- 1 file changed, 146 insertions(+), 264 deletions(-) diff --git a/lib/PayPro.js b/lib/PayPro.js index 493ad4b..d454710 100644 --- a/lib/PayPro.js +++ b/lib/PayPro.js @@ -145,270 +145,7 @@ PayPro.prototype.x509Verify = function() { // Handle Cert Extensions // http://tools.ietf.org/html/rfc5280#section-4.2 // - var ext; - var eid; - var extensions = { - authorityKeyIdentifier: null, - subjectKeyIdentifier: null, - keyUsage: null, - certificatePolicies: null, - policyMappings: null, - subjectAlternativeName: null, - issuerAlternativeName: null, - subjectDirectoryAttributes: null, - basicConstraints: null, - nameConstraints: null, - policyConstraints: null, - extendedKeyUsage: null, - CRLDistributionPoints: null, - inhibitAnyPolicy: null, - freshestCRL: null, - authorityInformationAccess: null, - subjectInformationAccess: null, - standardUnknown: [], - unknown: [], - }; - - for (var i = 0; i < c.tbsCertificate.extensions.length; i++) { - ext = c.tbsCertificate.extensions[i]; - eid = ext.extnID; - - // id-ce extensions - Standard Extensions - if (eid.length === 4 && eid[0] === 2 && eid[1] === 5 && eid[2] === 29) { - switch (eid[3]) { - // Authority Key Identifier - case 35: - print('Authority Key Identifier:'); - print(ext.extnValue); - extensions.authorityKeyIdentifier = ext.extnValue; - // parse - extensions.authorityKeyIdentifier = rfc5280.AuthorityKeyIdentifier.decode( - extensions.authorityKeyIdentifier, - 'der', { partial: false }); - print(extensions.authorityKeyIdentifier); - break; - // Subject Key Identifier - case 14: // VERY IMPORTANT, especially is cA (basic constraints) is true (it is) - print('Subject Key Identifier:'); - print(ext.extnValue); - extensions.subjectKeyIdentifier = ext.extnValue; - // parse - extensions.subjectKeyIdentifier = rfc5280.SubjectKeyIdentifier.decode( - extensions.subjectKeyIdentifier, - 'der', { partial: false }); - print(extensions.subjectKeyIdentifier); - break; - // Key Usage - case 15: - print('Key Usage:'); - print(ext.extnValue); - extensions.keyUsage = ext.extnValue; - // parse - data = rfc5280.KeyUsage.decode( - extensions.keyUsage, - 'der').data[0]; - extensions.keyUsage = { - digitalSignature: !!((data >> 0) & 1), - nonRepudiation: !!((data >> 1) & 1), - // nonRepudiation renamed to contentCommitment: - contentCommitment: !!((data >> 1) & 1), - keyEncipherment: !!((data >> 2) & 1), - dataEncipherment: !!((data >> 3) & 1), - keyAgreement: !!((data >> 4) & 1), - keyCertSign: !!((data >> 5) & 1), - cRLSign: !!((data >> 6) & 1), - encipherOnly: !!((data >> 7) & 1), - decipherOnly: !!((data >> 8) & 1) - }; - print(extensions.keyUsage); - break; - // Certificate Policies - case 32: - print('Certificate Policies:'); - print(ext.extnValue); - extensions.certificatePolicies = ext.extnValue; - // parse - extensions.certificatePolicies = rfc5280.CertificatePolicies.decode( - extensions.certificatePolicies, - 'der', { partial: false }); - print(extensions.certificatePolicies); - break; - // Policy Mappings - case 33: - print('Policy Mappings:'); - print(ext.extnValue); - extensions.policyMappings = ext.extnValue; - // parse - extensions.policyMappings = rfc5280.PolicyMappings.decode( - extensions.policyMappings, - 'der', { partial: false }); - print(extensions.policyMappings); - break; - // Subject Alternative Name - case 17: - print('Subject Alternative Name:'); - print(ext.extnValue); - extensions.subjectAlternativeName = ext.extnValue; - // parse - extensions.subjectAlternativeName = rfc5280.SubjectAlternativeName.decode( - extensions.subjectAlternativeName, - 'der', { partial: false }); - print(extensions.subjectAlternativeName); - break; - // Issuer Alternative Name - case 18: - print('Issuer Alternative Name:'); - print(ext.extnValue); - extensions.issuerAlternativeName = ext.extnValue; - // parse - extensions.issuerAlternativeName = rfc5280.IssuerAlternativeName.decode( - extensions.issuerAlternativeName, - 'der', { partial: false }); - print(extensions.issuerAlternativeName); - break; - // Subject Directory Attributes - case 9: - print('Subject Directory Attributes:'); - print(ext.extnValue); - extensions.subjectDirectoryAttributes = ext.extnValue; - // parse - extensions.subjectDirectoryAttributes = rfc5280.SubjectDirectoryAttributes.decode( - extensions.subjectDirectoryAttributes, - 'der', { partial: false }); - print(extensions.subjectDirectoryAttributes); - break; - // Basic Constraints - case 19: - print('Basic Constraints:'); - print(ext.extnValue); - extensions.basicConstraints = ext.extnValue; - // parse - extensions.basicConstraints = rfc5280.BasicConstraints.decode( - extensions.basicConstraints, - 'der', { partial: false }); - print(extensions.basicConstraints); - break; - // Name Constraints - case 30: - print('Name Constraints:'); - print(ext.extnValue); - extensions.nameConstraints = ext.extnValue; - // parse - extensions.nameConstraints = rfc5280.NameConstraints.decode( - extensions.nameConstraints, - 'der', { partial: false }); - print(extensions.nameConstraints); - break; - // Policy Constraints - case 36: - print('Policy Constraints:'); - print(ext.extnValue); - extensions.policyConstraints = ext.extnValue; - // parse - extensions.policyConstraints = rfc5280.PolicyConstraints.decode( - extensions.policyConstraints, - 'der', { partial: false }); - print(extensions.policyConstraints); - break; - // Extended Key Usage - case 37: - print('Extended Key Usage'); - print(ext.extnValue); - extensions.extendedKeyUsage = ext.extnValue; - // parse - extensions.extendedKeyUsage = rfc5280.ExtendedKeyUsage.decode( - extensions.extendedKeyUsage, - 'der', { partial: false }); - print(extensions.extendedKeyUsage); - break; - // CRL Distribution Points - case 31: - print('CRL Distribution Points:'); - print(ext.extnValue); - extensions.CRLDistributionPoints = ext.extnValue; - // parse - extensions.CRLDistributionPoints = rfc5280.CRLDistributionPoints.decode( - extensions.CRLDistributionPoints, - 'der', { partial: false }); - print(extensions.CRLDistributionPoints); - break; - // Inhibit anyPolicy - case 54: - print('Inhibit Any Policy:'); - print(ext.extnValue); - extensions.inhibitAnyPolicy = ext.extnValue; - // parse - extensions.inhibitAnyPolicy = rfc5280.InhibitAnyPolicy.decode( - extensions.inhibitAnyPolicy, - 'der', { partial: false }); - print(extensions.inhibitAnyPolicy); - break; - // Freshest CRL - case 46: - print('Freshest CRL:'); - print(ext.extnValue); - extensions.freshestCRL = ext.extnValue; - // parse - extensions.freshestCRL = rfc5280.FreshestCRL.decode( - extensions.freshestCRL, - 'der', { partial: false }); - print(extensions.freshestCRL); - break; - // Unknown Extension (not documented anywhere, probably non-standard) - default: - extensions.unknown.push(ext); - extensions.standardUnknown.push(ext); - break; - } - continue; - } - - // id-pe extensions - Private Internet Extensions - if (eid.length === 8 - && eid[0] === 1 - && eid[1] === 3 - && eid[2] === 6 - && eid[3] === 1 - && eid[4] === 5 - && eid[5] === 5 - && eid[6] === 7) { - switch (eid[3]) { - // Authority Information Access - // id-pe: - case 1: - print('Authority Information Access:'); - print(ext.extnValue); - extensions.authorityInformationAccess = ext.extnValue; - // parse - extensions.authorityInformationAccess = rfc5280.AuthorityInformationAccess.decode( - extensions.authorityInformationAccess, - 'der'); - print(extensions.freshestCRL); - break; - // Subject Information Access - // id-pe: - case 11: - print('Subject Information Access:'); - print(ext.extnValue); - extensions.subjectInformationAccess = ext.extnValue; - // parse - extensions.subjectInformationAccess = rfc5280.SubjectInformationAccess.decode( - extensions.subjectInformationAccess, - 'der'); - print(extensions.subjectInformationAccess); - break; - // Unknown Extension (not documented anywhere, probably non-standard) - default: - extensions.unknown.push(ext); - extensions.standardUnknown.push(ext); - break; - } - continue; - } - - extensions.unknown.push(ext); - } - + var extensions = rfc5280.parseExtensions(c.tbsCertificate, { partial: false }); var extensionsVerified = !extensions.unknown.filter(function(ext) { return ext.critical; }).length; @@ -1115,6 +852,151 @@ rfc5280.SubjectInformationAccess = asn1.define('SubjectInformationAccess', funct this.seqof(AccessDescription); }); +rfc5280.extensions = { + standard: { + // id-ce extensions - Standard Extensions + prefix: [2, 5, 29], + 35: 'Authority Key Identifier', + 14: 'Subject Key Identifier', + // VERY IMPORTANT, especially is cA (basic constraints) is true (it is) + 15: { + name: 'Key Usage', + parse: function(data, decoded, ext) { + data = data[0]; + return { + digitalSignature: !!((data >> 0) & 1), + nonRepudiation: !!((data >> 1) & 1), + // nonRepudiation renamed to contentCommitment: + contentCommitment: !!((data >> 1) & 1), + keyEncipherment: !!((data >> 2) & 1), + dataEncipherment: !!((data >> 3) & 1), + keyAgreement: !!((data >> 4) & 1), + keyCertSign: !!((data >> 5) & 1), + cRLSign: !!((data >> 6) & 1), + encipherOnly: !!((data >> 7) & 1), + decipherOnly: !!((data >> 8) & 1) + }; + } + }, + 32: 'Certificate Policies', + 33: 'Policy Mappings', + 17: 'Subject Alternative Name', + 18: 'Issuer Alternative Name', + 9: 'Subject Directory Attributes', + 19: 'Basic Constraints', + 30: 'Name Constraints', + 36: 'Policy Constraints', + 37: 'Extended Key Usage', + 31: 'CRL Distribution Points', + 54: 'Inhibit anyPolicy', + 46: 'Freshest CRL', + // Unknown Extension (not documented anywhere, probably non-standard) + _: 'Unknown Extension' + }, + // id-pe extensions - Private Internet Extensions + priv: { + prefix: [1, 3, 6, 1, 5, 5, 7], + 1: 'Authority Information Access', + 11: 'Subject Information Access', + // Unknown Extension (not documented anywhere, probably non-standard) + _: 'Unknown Extension' + } +}; + +Object.keys(rfc5280.extensions).forEach(function(typeName) { + var type = rfc5280.extensions[typeName]; + Object.keys(type).forEach(function(suffix) { + var id, prop, schemaName, schema, parse; + + if (suffix === 'prefix') + return; + + var prefix = type.prefix; + var name = type[suffix]; + + if (typeof name === 'object') { + var obj = name; + name = obj.name; + parse = obj.parse; + } + + id = prefix.concat(suffix).join('.'); + + if (/^[A-Z]+ /.test(name)) { + // CRL Distribution Points - > CRLDistributionPoints + prop = name.replace(/ /g, ''); + } else { + prop = (name[0].toLowerCase()) + name.substring(1).replace(/ /g, ''); + } + + schemaName = name.replace(/ /g, ''); + schema = rfc5280[schemaName]; + + rfc5280.extensions[id] = { + typeName: typeName, + prefix: prefix, + suffix: suffix, + id: id, + name: name, + prop: prop, + schemaName: schemaName, + schema: schema, + parse: parse + }; + }); +}); + +rfc5280.parseExtensions = function(tbsCertificate, options) { + var edata, eid, ext, decoded, data; + + var output = {}; + var unknown = 0; + + for (var i = 0; i < tbsCertificate.extensions.length; i++) { + edata = tbsCertificate.extensions[i]; + eid = edata.extnID.join('.'); + + if (ext = rfc5280.extensions[eid]) { + // Parse Extension + decoded = ext.schema.decode(edata.extnValue, 'der', options); + + // If the Extension needs extra parsing (i.e. bitstrs) + data = ext.parse + ? ext.parse(decoded.data, decoded, ext) + : decoded; + + // Tack on some useful info + + // Comment for debugging: + // data._edata = edata; + // data._ext = ext; + + if (ext.parse) { + data._decoded = decoded; + } + + // Add our decoded extension to the output + output[ext.prop] = data; + + // XXX Debug + print('------------'); + print('%s (%s):', ext.name, ext.id); + print('Buffer:'); + print(edata.extnValue); + print('Extension:'); + print(data); + } else { + // Add unknown extension: + output['Unkown_' + unknown] = edata; + + // XXX Debug + print('Unknown extension: %s', eid); + } + } + + return output; +}; + /** * Debug */