Upvote:4
(NSData *)signCSR:(NSData *)derCSR forDays:(double)days NIDstToKeep:(NSSet *)nidsToAllow
{
const unsigned char * ptr;
NSUInteger len;
// Gather the details for the CA cert (my cert) from
// OSX.
//
X509 * x509_mycert = NULL;
SecIdentityRef identityRef = [self secIdentityRef];
if (!identityRef)
return nil;
SecCertificateRef certificateRef = [self secCertificateRef];
if (!certificateRef)
return nil;
CFDataRef certAsDer = SecCertificateCopyData(certificateRef);
if (!certAsDer)
return nil;
// Jump over the fence to OpenSSL - and create
// an X509 version.
//
ptr = CFDataGetBytePtr(certAsDer);
len = CFDataGetLength(certAsDer);
if (!(d2i_X509(&x509_mycert, &ptr, len)))
return nil;
// And likewise for the CSR.
//
ptr = (const unsigned char *)[derCSR bytes];
len = [derCSR length];
X509_REQ *req = NULL;
if (!(d2i_X509_REQ(&req, &ptr, len)))
return nil;
// Copy the CSR into a an actual x509 tenative
// structure; i.e. the cert we'll issue signed.
//
X509 * x509_to_sign = X509_new();
assert(X509_set_subject_name(x509_to_sign,req->req_info->subject));
assert(X509_set_issuer_name(x509_to_sign, X509_get_subject_name(x509_mycert)));
EVP_PKEY * pubkey_csr = X509_REQ_get_pubkey(req);
X509_set_pubkey(x509_to_sign,pubkey_csr);
EVP_PKEY_free(pubkey_csr);
X509_gmtime_adj(X509_get_notBefore(x509_to_sign),0L);
X509_gmtime_adj(X509_get_notAfter(x509_to_sign),(long)floor(60*60*24*days));
if (nidsToAllow && [nidsToAllow count]) {
// Faily blindly copy all known extensions.
//
for(int i = X509_get_ext_count(x509_to_sign); i > 0; i--) {
X509_EXTENSION * ext = X509_get_ext(x509_to_sign,i-i);
int nid = OBJ_obj2nid(ext->object);
if ([nidsToAllow containsObject:[NSNumber numberWithInt:nid]]) {
// NSLog(@"Keeping %s at %d", OBJ_nid2sn(nid),i-i);
continue;
}
// NSLog(@"Killing %s at %d", OBJ_nid2sn(nid),i-1);
X509_delete_ext(x509_to_sign, i-i);
}
} else {
// wipe them all.
//
while (X509_get_ext_count(x509_to_sign) > 0) {
X509_delete_ext(x509_to_sign, 0);
}
};
// Set a random serial.
//
ASN1_INTEGER *bs = ASN1_INTEGER_new();
long rnd = 0;
if (!(RAND_bytes((unsigned char *)&rnd, sizeof(rnd))))
return nil;
rnd = labs(rnd);
ASN1_INTEGER_set(bs, rnd);
if (!((X509_set_serialNumber(x509_to_sign,bs))))
return nil;
ASN1_INTEGER_free(bs);
// Force v3
//
X509V3_CTX ctx2;
X509_set_version(x509_to_sign,2); /* version 3 certificate */
X509V3_set_ctx(&ctx2, x509_mycert, x509_to_sign, NULL, NULL, 0);
// Pull in additional x509v3 sections.
//
if (DBA) {
assert(X509V3_set_nconf(&ctx2, conf));
assert(X509V3_EXT_add_nconf(conf, &ctx2, section, x509_to_sign));
}
// Specify signature type.
//
x509_to_sign->cert_info->signature->algorithm = OBJ_nid2obj(NID_sha1WithRSAEncryption);
// Construct the ASN.1 blob which contains all the information
// we are going to sign.
//
const ASN1_ITEM * it = ASN1_ITEM_rptr(X509_CINF);
unsigned char *buf_in=NULL;
ASN1_VALUE * asn = (ASN1_VALUE *)(x509_to_sign->cert_info);
int inl = ASN1_item_i2d(asn,&buf_in, it);
// Small area to hold the signature on the SHA1 hash.
//
size_t sigLen = SecKeyGetBlockSize(privateKey);
uint8_t * sig = (uint8_t *)malloc(sigLen * sizeof(uint8_t));
memset((void*)sig, 0, sigLen);
#if TARGET_OS_IPHONE
// IPhone Way of doing it - where we need to construct the
// SHA1 of the 'to sign' area ourselves. As SecKeyRawSign
// does not seem to handly anything beyond a SHA1.
//
NSData * buffToSign = [NSData dataWithBytes:buf_in length:inl];
NSData * buffSha1ToSign = [buffToSign sha1];
size_t sigLenUsed = sigLen;
OSStatus status = SecKeyRawSign(privateKey, kSecPaddingPKCS1SHA1, [buffSha1ToSign bytes], [buffSha1ToSign length], sig, &sigLenUsed);
assert(status == noErr);
assert(sigLenUsed == sigLen);
#else
// MacOSX offical way of doing this - which seems to do the SHA1 fun,
// padding and games deeper down; and where we simply pass the blob.
//
CFErrorRef error = NULL;
SecTransformRef signer = SecSignTransformCreate(privateKey, &error);
if (error) { CFShow(error); assert(0); };
NSData * blockToSign = [NSData dataWithBytes:buf_in length:inl];
SecTransformSetAttribute(signer, kSecTransformInputAttributeName,
(__bridge CFTypeRef) blockToSign, &error);
if (error) { CFShow(error); assert(0); };
CFDataRef signature = SecTransformExecute(signer, &error);
if (error) { CFShow(error); assert(0); };
assert(sigLen == CFDataGetLength(signature));
bcopy(CFDataGetBytePtr(signature), sig, sigLen);
#endif
// Wrap up the rest of the block with the bits and bobs needed to
// create a valid ASN1 block to share as a PEM or DER. Which most
// crucially is about copying the just created signature into it.
//
x509_to_sign->sig_alg->algorithm = OBJ_nid2obj(NID_sha1WithRSAEncryption);
x509_to_sign->signature->data = sig;
x509_to_sign->signature->length = sigLen;
x509_to_sign->signature->flags&= ~(ASN1_STRING_FLAG_BITS_LEFT|0x07);
x509_to_sign->signature->flags|=ASN1_STRING_FLAG_BITS_LEFT;
unsigned char * derbuff = NULL;
int derlen = i2d_X509(x509_to_sign, &derbuff);
NSData * signedDer = [NSData dataWithBytes:derbuff length:derlen];
free(sig);
OPENSSL_free(x509_to_sign);
OPENSSL_free(x509_mycert);
return signedDer;
}
Credit Goes to: stackoverflow.com
Related question with same questions but different answers
- GENERATING A SELF-SIGNED CERTIFICATE WITH IOS SECURITY.FRAMEWORK? IN IOS
- HOW TO IMPLEMENT THIS CELL ANIMATION WITH UICOLLECTIONVIEW IN IOS
- IOS: SHOW MODAL VIEW CONTROLLER WITH FULL SCREEN IMAGE IN IOS
- STRANGE BEHAVIOR OF NAVIGATIONBAR AFTER MULTIPLE PUSH AND POP WITH ANIMATION NO. IOS7 IN IOS
- TAKE SQUARE IMAGE WITH CORDOVA (PHONEGAP) CAMERA API? IN IOS
- SWIZZLING A METHOD WITH VARIABLE ARGUMENTS AND FORWARD THE MESSAGE - BAD ACCESS IN IOS
- THREAD-SAFE DESIGN WITH ARC IN IOS
- GETTING ALL PHOTOS INCLUDED IN A SINGLE FACEBOOK FEED POST WITH EITHER GRAPH API OR FQL? IN IOS
- MAGICALRECORDS PERSISTING IN BACKGROUND THREAD WITH IMAGE FETCHING? IN IOS
- AFNETWORKING - REQUESTS WITH ETAG IN IOS
- COMBINING NAVIGATION CONTROLLER WITH TAB BAR CONTROLLER IN IOS
- TROUBLE WITH CAANIMATION IN IOS
- CALCULATE ACCURATE BPM FROM MIDI CLOCK IN OBJC WITH COREMIDI IN IOS
- HOW TO MENTION USERS WITH "@" IN APP LIKE INSTAGRAM IN IOS
- MEMORY WARNING ON THE DEVICE, NOT WITH INSTRUMENTS IN IOS
- FAILED TO LOAD WEBPAGE WITH ERROR: THE REQUESTED URL WAS NOT FOUND ON THIS SERVER IN IOS
- ISSUE WITH GPS ICON WHILE USING "LOCATION SIGNIFICANT CHANGE" AND USER PERCEPTION IN IOS
- MORE THAN ONE CUSTOM CELL WITH IN THE SAME XIB IN IOS
- BLOCKS EXAMPLE IN IOS5 IN IOS
- REUSING UITABLEVIEWCELL WITH GCD IN IOS
- IOS, REMOTE SERVER SEARCH WITH RESTKIT IN IOS
- PARSE JSON WITH NSJSONSERIALIZATION CLASS USING OBJECTFORKEY IN IOS IN IOS
- USE DATABASE (SUCH AS SQLITE) WITH COCOS2D-X IN IOS
- SET UP A RED5 SERVER WITH FFMPEG FOR LIVE STREAMING TO IOS IN IOS
- HOW TO POST MORE PARAMETERS WITH IMAGE UPLOAD CODE IN IOS? IN IOS
- CSS MEDIA QUERIES ISSUE WITH IOS DEVICES IN IOS
- IOS SAFARI CRASHING WITH LARGE POST DATA FORM SUBMIT IN IOS
- UITABLEVIEW LOADING THUMBNAIL IMAGES ASYNCHRONOUSLY WITH CACHE IN IOS
- ACCESSING APPLE'S ROOT CERTIFICATE ON IOS IN IOS
- DISTRIBUTE UNSIGNED APP IPHONE FOR JAILBROKEN IPHONE WITH NO CYDIA IN IOS
- WINDOW.OPEN('', '_BLANK') DOESN'T OPEN NEW TAB ON IOS ONLY IN IOS
- UIBEZIERPATH AND APPLYTRANSFORM IN IOS
- BEST WAY TO CREATE CUSTOM BUILDING'S MAP FOR IOS IN IOS
- UI AUTOMATION TOOLS FOR IOS APPS THAT CAN IDENTIFY CALAYER OBJECTS IN IOS
- IS IT POSSIBLE TO PROGRAMMATICALLY ACCESS THE ERROR CODES LOGGED BY COREBLUETOOTH? IN IOS
- ARE THERE ANY LIBRARIES IN IOS FOR IDENTIFYING PHONETICALLY SAME SOUND IN IOS
- SETTING UISEARCHBAR BACKGROUND COLOR TO WHITE IN IOS IN IOS
- IOS CREATE THE INVERSE OF A CASHAPELAYER MASK IN IOS
- HOW TO ACCESS MEMBERS ON A C++ REFERENCE OR POINTER IN A METAL KERNEL? IN IOS
- UINAVIGATIONBAR RIGHT BUTTON NOT SHOWING UP IN IOS
- UITABLEVIEW DELETE AND RELOAD CELLS IN IOS
- HOW DO I CREATE A CUSTOM UITABLEVIEWCELL WITH NIB IN XCODE 4? IN IOS
- WHAT IS THE BETTER WAY TO CALL MULTIPLE SERVICES IN IOS? IN IOS
- 'FOLLY/PORTABILITY.H' FILE NOT FOUND REACT LIBRARY IN IOS
- SIRI SHORTCUTS: HOW TO REQUIRE UNLOCKING THE PHONE IN IOS
- APP STORE - WHERE TO DOWNLOAD THE REPORT OF APP DOWNLOADS? IN IOS
- A PROBLEM OCCURRED CONFIGURING PROJECT ':APP'. IN REACT-NATIVE AND GIVE SOME BYTE CODE AS AN ERROR IN IOS
- MATCH THE ZOOM/BOUNDS OF A MAPKIT MAP WITH THE ZOOM/BOUNDS OF A ROUTEME MAP IN IOS
- WHERE DOES SWIFT ABI COME INTO PICTURE? IN IOS
- USING ASIHTTPREQUEST TO POST NESTED PARAMS USING A NSDICTIONARY IN IOS
- HOW CAN I EXCLUDE A SECRET KEY FROM AN OPEN-SOURCED XCODE PROJECT? IN IOS
- SIGN DATA USING RSA PRIVATE KEY IN IOS
- IOS: CAN I TELL IF A USER HAS OPTED-OUT OF APPLE'S "SHARE WITH APP DEVELOPERS" SETTING? IN IOS
- CACHE API DOESN'T WORK ON SAFARI IOS SIMULATOR AND ON MAC IN IOS
- HOW TO CREATE A GLOWING EFFECT WITH A UILABEL? IN IOS
- SET TEXT TO BE UNDERLINE WITHOUT SELECTION IN IOS
- IOS DEVELOPER ENTERPRISE PROGRAM - CAN DEVICE USERS INSTALL THE APP DIRECTLY? IN IOS