iOS RSA加密与SHA签名|与Java完全同步

一篇搞定RSA加密与SHA签名|与Java完全同步  

转载:https://www.cnblogs.com/On1Key/p/5459144.html

基础知识

什么是RSA?

答:RSA是一种非对称加密算法,常用来对传输数据进行加密,配合上数字摘要算法,也可以进行文字签名。

RSA加密中padding?

答:padding即填充方式,由于RSA加密算法中要加密的明文是要比模数小的,padding就是通过一些填充方式来限制明文的长度。后面会详细介绍padding的几种模式以及分段加密。

加密和加签有什么区别?

答:加密:公钥放在客户端,并使用公钥对数据进行加密,服务端拿到数据后用私钥进行解密;

加签:私钥放在客户端,并使用私钥对数据进行加签,服务端拿到数据后用公钥进行验签。

前者完全为了加密;后者主要是为了防恶意攻击,防止别人模拟我们的客户端对我们的服务器进行攻击,导致服务器瘫痪。

基本原理


RSA使用“密钥对”对数据进行加密解密,在加密解密前需要先生存公钥(Public Key)和私钥(Private Key)。

公钥(Public key): 用于加密数据. 用于公开, 一般存放在数据提供方, 例如iOS客户端。

私钥(Private key): 用于解密数据. 必须保密, 私钥泄露会造成安全问题。

iOS中的Security.framework提供了对RSA算法的支持,这种方式需要对密匙对进行处理, 根据public key生成证书, 通过private key生成p12格式的密匙。想想jave直接用字符串进行加密解密简单多了。(⊙o⊙)…

实战

证书生成

RSA加密这块公钥、私钥必不可少的。Apple是不支持直接使用字符串进行加密解密的,推荐使用p12文件。这边教大家去生成在加密中使用到的所有文件,并提供给Java使用,想当年这个公钥私钥搞了半天了。 %>_<% 

这里将会生成iOS和java使用的一套公私钥,记住 只是生成了一套 两端使用的不同文件,需要多套的话可以再操作一波生成第二套。

生成模长为1024bit的私钥

openssl genrsa -out private_key.pem 1024

生成certification require file

openssl req -new -key private_key.pem -out rsaCertReq.csr

生成certification 并指定过期时间

openssl x509 -req -days3650-in rsaCertReq.csr -signkey private_key.pem -out rsaCert.crt

生成公钥供iOS使用

openssl x509 -outform der -in rsaCert.crt -out public_key.der

生成私钥供iOS使用 这边会让你输入密码,后期用到在生成secKeyRef的时候会用到这个密码

openssl pkcs12 -export -out private_key.p12 -inkey private_key.pem -in rsaCert.crt

生成pem结尾的公钥供Java使用

openssl rsa -in private_key.pem -out rsa_public_key.pem -pubout

生成pem结尾的私钥供Java使用

openssl pkcs8 -topk8 -in private_key.pem -out pkcs8_private_key.pem -nocrypt

以上所有的步骤都是在终端下完成的哦 (^__^)

生成公钥和私钥的secKeyRef

//根据你的p12文件生成私钥对应的SecKeyRef 这边返回若是nil 请检查你p12文件的生成步骤- 

-(SecKeyRef)getPrivateKeyRefrenceFromData:(NSData*)p12Data password:(NSString*)password {

SecKeyRef privateKeyRef = NULL;

NSMutableDictionary * options = [[NSMutableDictionary alloc] init];

[options setObject: password forKey:(__bridge id)kSecImportExportPassphrase];

CFArrayRef items = CFArrayCreate(NULL,0,0, NULL);

OSStatus securityError = SecPKCS12Import((__bridge CFDataRef) p12Data, (__bridge CFDictionaryRef)options, &items);if(securityError == noErr && CFArrayGetCount(items) >0) {

    CFDictionaryRef identityDict = CFArrayGetValueAtIndex(items,0);

    SecIdentityRef identityApp = (SecIdentityRef)CFDictionaryGetValue(identityDict, kSecImportItemIdentity);

    securityError = SecIdentityCopyPrivateKey(identityApp, &privateKeyRef);

    if(securityError != noErr) {

        privateKeyRef = NULL;

    }

}

CFRelease(items);return privateKeyRef;

}


-//根据你的der文件公钥对应的SecKeyRef

- (SecKeyRef)getPublicKeyRefrenceFromeData:    (NSData*)derData {

SecCertificateRef myCertificate = SecCertificateCreateWithData(kCFAllocatorDefault, (__bridge CFDataRef)derData);

SecPolicyRef myPolicy = SecPolicyCreateBasicX509();

SecTrustRef myTrust;

OSStatus status = SecTrustCreateWithCertificates(myCertificate,myPolicy,&myTrust);

SecTrustResultType trustResult;if(status == noErr) {

    status = SecTrustEvaluate(myTrust, &trustResult);

}

SecKeyRef securityKey = SecTrustCopyPublicKey(myTrust);

CFRelease(myCertificate);

CFRelease(myPolicy);

CFRelease(myTrust);return securityKey;

}


加密与解密


- (NSData*)rsaEncryptData:(NSData*)data {

    SecKeyRef key = [self getPublicKey];

    size_t cipherBufferSize = SecKeyGetBlockSize(key);

    uint8_t *cipherBuffer = malloc(cipherBufferSize *sizeof(uint8_t));

    size_t blockSize = cipherBufferSize -11;

      size_t blockCount = (size_t)ceil([data length] / (double)blockSize);

      NSMutableData *encryptedData = [[NSMutableData alloc] init];

    for(inti=0; i

    unsigned longbufferSize = MIN(blockSize , [data length] - i * blockSize);

    NSData *buffer = [data subdataWithRange:NSMakeRange(i * blockSize, bufferSize)];

    OSStatus status = SecKeyEncrypt(key, kSecPaddingPKCS1, (constuint8_t *)[buffer bytes], [buffer length], cipherBuffer, &cipherBufferSize);

    if(status != noErr) {

        return nil;

    }

    NSData *encryptedBytes = [[NSData alloc] initWithBytes:(constvoid*)cipherBuffer length:cipherBufferSize];

    [encryptedData appendData:encryptedBytes];

    }

  if (cipherBuffer){

    free(cipherBuffer);

  }

  return encryptedData;

  }

- (NSData*)rsaDecryptData:(NSData*)data {

SecKeyRef key = [self getPrivatKey];

size_t cipherBufferSize = SecKeyGetBlockSize(key);

size_t blockSize = cipherBufferSize;

size_t blockCount = (size_t)ceil([data length] / (double)blockSize);

NSMutableData *decryptedData = [[NSMutableData alloc] init];for(inti =0; i < blockCount; i++) {

    unsigned longbufferSize = MIN(blockSize , [data length] - i * blockSize);

    NSData *buffer = [data subdataWithRange:NSMakeRange(i * blockSize, bufferSize)];

    size_t cipherLen = [buffer length];

    void*cipher = malloc(cipherLen);

    [buffer getBytes:cipher length:cipherLen];

    size_t plainLen = SecKeyGetBlockSize(key);

    void*plain = malloc(plainLen);

    OSStatus status = SecKeyDecrypt(key, kSecPaddingPKCS1, cipher, cipherLen, plain, &plainLen);

    if(status != noErr) {

        return nil;

    }

    NSData *decryptedBytes = [[NSData alloc] initWithBytes:(constvoid*)plain length:plainLen];

    [decryptedData appendData:decryptedBytes];

}return decryptedData;

}


RSA加密中的Padding


RSA_PKCS1_PADDING 填充模式,最常用的模式

要求: 输入:必须 比 RSA 钥模长(modulus) 短至少11个字节, 也就是 RSA_size(rsa) – 11 如果输入的明文过长,必须切割,然后填充。

输出:和modulus一样长

根据这个要求,对于1024bit的密钥,block length = 1024/8 – 11 = 117 字节

RSA_PKCS1_OAEP_PADDING

输入:RSA_size(rsa) – 41

输出:和modulus一样长

RSA_NO_PADDING  不填充

输入:可以和RSA钥模长一样长,如果输入的明文过长,必须切割, 然后填充

输出:和modulus一样长

签名与验证

//对数据进行sha256签名

- (NSData *)rsaSHA256SignData:(NSData *)plainData {

  SecKeyRef key = [self getPrivatKey];

  size_t signedHashBytesSize = SecKeyGetBlockSize(key);

  uint8_t* signedHashBytes = malloc(signedHashBytesSize);

  memset(signedHashBytes, 0x0, signedHashBytesSize);

  size_t hashBytesSize = CC_SHA256_DIGEST_LENGTH;

  uint8_t* hashBytes = malloc(hashBytesSize);

  if(!CC_SHA256([plainData bytes], (CC_LONG)[plainData length], hashBytes)) {

    return nil;

}

      SecKeyRawSign(key,

              kSecPaddingPKCS1SHA256,

              hashBytes,

              hashBytesSize,

              signedHashBytes,

              &signedHashBytesSize);

    NSData* signedHash = [NSData dataWithBytes:signedHashBytes

                                    length:(NSUInteger)signedHashBytesSize];

    if (hashBytes)

    free(hashBytes);

    if (signedHashBytes)

    free(signedHashBytes);

    return signedHash;

    }


//这边对签名的数据进行验证 验签成功,则返回YES

- (BOOL)rsaSHA256VerifyData:(NSData *)plainData    withSignature:(NSData *)signature {

    SecKeyRef key = [self getPublicKey];

    size_t signedHashBytesSize = SecKeyGetBlockSize(key);

    constvoid* signedHashBytes = [signature bytes];

    size_t hashBytesSize = CC_SHA256_DIGEST_LENGTH;

    uint8_t* hashBytes = malloc(hashBytesSize);

    if(!CC_SHA256([plainData bytes], (CC_LONG)[plainData length], hashBytes)) {

      return NO;

    }

      OSStatus status = SecKeyRawVerify(key,

                                  kSecPaddingPKCS1SHA256,

                                  hashBytes,

                                  hashBytesSize,

                                  signedHashBytes,

                                  signedHashBytesSize);

    returnstatus == errSecSuccess;

    }


文章到此就结束了,希望这篇文章对大家有所帮助。想看demo的请点击:XYRSACryptor

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容