【问题标题】:iOS AES Encryption - Fail to EncryptiOS AES 加密 - 加密失败
【发布时间】:2014-07-01 12:02:33
【问题描述】:

在我的项目中,我必须实施 AES 128 CBC 加密。我正在使用 Category 并且基于 NSData。这是我的加密代码:

- (NSData*)AES128Decrypt
{
    char ivPtr[kCCKeySizeAES128 + 1];
    bzero(ivPtr, sizeof(ivPtr));

    // fetch iv data
    [iv getCString:ivPtr maxLength:sizeof(ivPtr) encoding:NSUTF8StringEncoding];


    // 'key' should be 32 bytes for AES256, will be null-padded otherwise
    char keyPtr[kCCKeySizeAES128 + 1]; // room for terminator (unused)
    bzero(keyPtr, sizeof(keyPtr)); // fill with zeroes (for padding)

    // fetch key data
    [key getCString:keyPtr maxLength:sizeof(keyPtr) encoding:NSUTF8StringEncoding];


    NSUInteger dataLength = [self length];   // dataLength = 19

    //See the doc: For block ciphers, the output size will always be less than or
    //equal to the input size plus the size of one block.
    //That's why we need to add the size of one block here
    size_t bufferSize           = dataLength + kCCBlockSizeAES128;
    void* buffer                = malloc(bufferSize);

    size_t numBytesDecrypted    = 0;
    CCCryptorStatus cryptStatus = CCCrypt(kCCDecrypt, kCCAlgorithmAES128, 0,
                                          keyPtr, kCCKeySizeAES128,
                                          ivPtr,
                                          [self bytes], dataLength, 
                                          buffer, bufferSize, 
                                          &numBytesDecrypted);  // buffer = 0 & numBytesDecrypted = 0

    if (cryptStatus == kCCSuccess)
    {
        //the returned NSData takes ownership of the buffer and will free it on deallocation
        return [NSData dataWithBytes:buffer length:numBytesDecrypted] ;  // returns 0
    }

    free(buffer); //free the buffer;
    return nil;
}

这就是我从视图类中调用它的方式:

- (void) testActuallyEncrypting :(NSString*) hexString {
    NSLog(@"String to Encrypt : %@", hexString); // prints test12

    @try {
    //Convert NSString to NSData
    NSData *data = [self dataFromHexString:hexString];  // [hexString dataUsingEncoding:NSUTF8StringEncoding];  //
    // // Prepare the NSDAta obj to store the encrypted pswd
    NSData *encryptedData = [NSData dataWithBytes:[data bytes] length:[data length]];  // 6bytes
    NSData *decryptedData = [encryptedData AES128Decrypt];  // 0bytes
    NSString *decryptedString = [NSString stringWithUTF8String:[decryptedData bytes]];  // NULL Exception
    NSLog(@"Decrypted String : %@", decryptedString);

    decryptedString = [self addPaddingToString:decryptedString];
    decryptedData = [NSData dataWithBytes:[decryptedString UTF8String] length:[[decryptedString dataUsingEncoding:NSUTF8StringEncoding] length]];
    encryptedData = [decryptedData AES128Encrypt];
    if (encryptedData!=nil)
    {
        NSString *encryptedHexString = [self hexStringFromData:encryptedData];
        NSLog(@"Encrypted HexString : %@",encryptedHexString);

     }
    }@catch (NSException *ex) {
        NSLog(@"Exception : %@", ex);
    }
}

我正在传递要加密的"test12" 字符串。在调用AES128Decrypt 时,decryptedData 为 0,因此下一行 decryptedString 抛出 Null 异常 - Exception : *** +[NSString stringWithUTF8String:]: NULL cString

谁能帮我知道为什么decryptedData 为空。我在 AES128Decrypt 方法中哪里出错了?

请帮助我。从过去的两天开始,我一直坚持这一点。在互联网上搜索了很多,但找不到任何解决方案来解决它。 非常感谢任何帮助。谢谢。

更新:- 在我的班级中添加了@Zaph 的方法并正在调用它。

NSLog(@"String to Encrypt : %@", hexString);
NSString *iv = @"fedcba9876543210";
NSString *key = @"0123456789abcdef";

// Convert str to encrypt, iv & key from NSString to NSData
NSData *dataIn = [hexString dataUsingEncoding:NSUTF8StringEncoding];
NSData *ivData = [iv dataUsingEncoding:NSUTF8StringEncoding];
NSData *symKey = [key dataUsingEncoding:NSUTF8StringEncoding];
NSError *error;

NSData *result = [LoginViewController doCipher:dataIn iv:ivData key:symKey context:kCCEncrypt error:&error]; // result = 16bytes

if (result != nil) {
    // Convert result to satring
    NSString *resultStr = [[NSString alloc] initWithData:result encoding:NSUTF8StringEncoding];
    NSLog(@"Encrypted Str = %@", resultStr );  // Encrypted Str = (null)  ????

为什么转换后的字符串是 null ?请提供任何帮助。谢谢

【问题讨论】:

  • 请勿使用@try@catch,除非捕获编程错误。在 Objective-C 中,它们不是控制结构。
  • 我发现将 NSData 转换为 NSString - 使用 NSUTF8StringEncoding 编码返回 null。我尝试使用 NSASCIIStringEncoding,它返回值。但是字符串值和数据值是不同的。他们都应该是一样的吗?
  • 并非所有数据都是有效的 UTR-8。对于数据的字符串表示,一个常见的约定是使用 Base64 编码。而不是NSString *resultStr = [[NSString alloc] initWithData:result encoding:NSUTF8StringEncoding]; 使用base64 编码:NSString *resultBase64Str = [result base64EncodedStringWithOptions:0]; 如果你使用NSASCIIStringEncoding,你会得到不可显示的字符,可能不是你想要的。

标签: ios encryption cryptography


【解决方案1】:

加密没必要这么复杂,这里有一个基本的加密/解密方法。 iv 和 key 的长度必须正确。值上下文是kCCEncryptkCCDecrypt

+ (NSData *)doCipher:(NSData *)dataIn
                  iv:(NSData *)iv
                 key:(NSData *)symmetricKey
             context:(CCOperation)encryptOrDecrypt
               error:(NSError **)error
{
    CCCryptorStatus ccStatus   = kCCSuccess;
    size_t          cryptBytes = 0;
    NSMutableData  *dataOut    = [NSMutableData dataWithLength:dataIn.length + kCCBlockSizeAES128];

    ccStatus = CCCrypt( encryptOrDecrypt,
                       kCCAlgorithmAES128,
                       kCCOptionPKCS7Padding,
                       symmetricKey.bytes, 
                       kCCKeySizeAES128,
                       iv.bytes,
                       dataIn.bytes,
                       dataIn.length,
                       dataOut.mutableBytes,
                       dataOut.length,
                       &cryptBytes);

    if (ccStatus == kCCSuccess) {
        dataOut.length = cryptBytes;
    }
    else {
        if (error) {
            *error = [NSError errorWithDomain:@"kEncryptionError"
                                         code:ccStatus
                                     userInfo:nil];
        }
        dataOut = nil;
    }

    return dataOut;
}

【讨论】:

  • 感谢上面的代码。我将它添加到我的 viewController 类中。将 iv, key, str 以 NSData 形式传递(iv & key 为 16 字节)并调用方法 - NSData *result = [LoginViewController doCipher:dataIn iv:ivData key:symKey context:kCCEncrypt error:&error];但是当我将其转换为 NSString 时,我得到了 16 字节的结果。你能检查一下上面的代码吗 - 已经在更新下添加了这个。 error 也是 nil - 所以没有错误。为什么字符串是 null ?你能帮忙找到那个吗?
  • 在获得结果时,我将其转换为字符串(它仅通过 NSASCIIStringEncoding 工作。然后我尝试将字符串转换回 nsdata。我得到其中 3 个不同的值。他们 3 不应该是相同或至少两个 nsdata。毫无疑问,在解密结果并转换为字符串时,我通过了原始字符串并且它正在工作。但是想知道这一点以获取知识,并且很想知道相同的信息。谢谢。
  • 在您的代码中,您使用了 kCCOptionPKCS7Padding,我认为这将用于 PKCS 实现。我必须使用CBC。 CBC的任何特殊设置/参数值??
  • kCCOptionPKCS7Padding 选项用于 AES。如果输入清除数据的长度不是块大小的精确倍数,则需要它。
  • 并非所有数据都是有效的 UTF-8 字符串。通常的解决方案是使用 base64 编码,即将加密数据转换为 base64 字符串。上面的代码提供了数据的加密/解密,具体是NSData,也就是加密所做的,数据加密。如果需要某种其他格式,需要在加密/解密前后完成,那就是格式转换。
猜你喜欢
  • 2015-11-01
  • 2016-06-25
  • 1970-01-01
  • 2011-08-27
  • 1970-01-01
  • 1970-01-01
  • 2013-12-06
  • 1970-01-01
  • 2015-10-31
相关资源
最近更新 更多