Внедрение шифрования RC4 и кодирования URL в цели c

Я должен внедрить алгоритм шифрования RC4 в свое приложение для кодирования URL. Образец приведен ниже для справки.

Фактическое значение до шифрования: 28.10.2013

Ожидаемое значение после шифрования: ˆ!˜·hÇÔÞò

После кодирования URL это должно быть: %0C%88%21%98%B7h%C7%D4%DE%F2

ИСПОЛЬЗУЕМЫЙ КЛЮЧ: @"psw"

Я пробовал, но значение после шифрования и кодирования URL возвращается следующим образом:

После шифрования : !·hÇÔÞò

После кодирования URL: %0C%C2%88%21%C2%98%C2%B7h%C3%87%C3%94%C3%9E%C3%B2

Я попытался преобразовать java-код в объективный код c. Java-код работает нормально, если цель c не работает, что вы можете видеть в приведенном выше результате.

Вот код Java

import java.net.URLEncoder;

public class RC4 {

private char[] key;
private int[] sbox;
private static final int SBOX_LENGTH = 256;
private static final int KEY_MIN_LENGTH = 1;

public static void main(String[] args) {
    try {
        RC4 rc4 = new RC4("psw");
        char[] result = rc4.encrypt("10/28/2013".toCharArray());
        System.out.println("encrypted string:\n"
                + new String(toByteArray(result)));
        System.out.println("decrypted string:\n"
                + new String(rc4.decrypt(result)));
        System.out.println("decrypted string:\n"
                + URLEncoder.encode(new String(toByteArray(result))));

    } catch (InvalidKeyException e) {
        System.err.println(e.getMessage());
    }

}

static byte[] toByteArray(char[] chars) {
    byte[] bytes = new byte[chars.length];
    for (int i = 0; i < chars.length; i++) {
        // bytes[i*2] = (byte) (chars[i] >> 8);
        bytes[i] = (byte) chars[i];
    }
    return bytes;
}

public RC4(String key) throws InvalidKeyException {
    setKey(key);
}

public RC4() {
}

public char[] decrypt(final char[] msg) {
    return encrypt(msg);
}

public char[] encrypt(final char[] msg) {
    sbox = initSBox(key);
    char[] code = new char[msg.length];
    int i = 0;
    int j = 0;
    for (int n = 0; n < msg.length; n++) {
        i = (i + 1) % SBOX_LENGTH;
        j = (j + sbox[i]) % SBOX_LENGTH;
        swap(i, j, sbox);
        int rand = sbox[(sbox[i] + sbox[j]) % SBOX_LENGTH];
        code[n] = (char) (rand ^ (int) msg[n]);
    }
    return code;
}

private int[] initSBox(char[] key) {
    int[] sbox = new int[SBOX_LENGTH];
    int j = 0;

    for (int i = 0; i < SBOX_LENGTH; i++) {
        sbox[i] = i;
    }

    for (int i = 0; i < SBOX_LENGTH; i++) {
        j = (j + sbox[i] + key[i % key.length]) % SBOX_LENGTH;
        swap(i, j, sbox);
    }
    return sbox;
}

private void swap(int i, int j, int[] sbox) {
    int temp = sbox[i];
    sbox[i] = sbox[j];
    sbox[j] = temp;
}

public void setKey(String key) throws InvalidKeyException {
    if (!(key.length() >= KEY_MIN_LENGTH && key.length() < SBOX_LENGTH)) {
        throw new InvalidKeyException("Key length has to be between "
                + KEY_MIN_LENGTH + " and " + (SBOX_LENGTH - 1));
    }

    this.key = key.toCharArray();
}

public class InvalidKeyException extends Exception {

    private static final long serialVersionUID = 1L;

    public InvalidKeyException(String message) {
        super(message);
    }

}

}

Его соответствующий код цели c

-(NSString *)encrypt:(NSString *)string{

[self frameSBox:@"psw"];
unichar code[string.length];
const unichar* buffer = code;

int i = 0;
int j = 0;
for (int n = 0; n < string.length; n++) {
    i = (i + 1) % self.SBOX_LENGTH;
    j = (j + [[self.sBox objectAtIndex:i]integerValue]) % self.SBOX_LENGTH;
    [self swap:i with:j];

    int index=([[self.sBox objectAtIndex:i] integerValue]+[[self.sBox objectAtIndex:j] integerValue]);

    int rand=([[self.sBox objectAtIndex:(index%self.SBOX_LENGTH)] integerValue]);

    code[n]=(rand  ^  (int)[string characterAtIndex:n]);
}
buffer = code;
return  [NSString stringWithCharacters:buffer length:string.length];
}


-(NSArray *)frameSBox:(NSString *)keyValue{


   if (self.sBox == nil) {
    self.sBox=[[NSMutableArray alloc]init];
}

self.SBOX_LENGTH=256;    
int j = 0;

for (int i = 0; i < self.SBOX_LENGTH; i++) {
    [self.sBox addObject:[NSNumber numberWithInteger:i]];
}

for (int i = 0; i < self.SBOX_LENGTH; i++) {
    j = (j + [[self.sBox objectAtIndex:i] integerValue] + [keyValue characterAtIndex:(i % keyValue.length)]) % self.SBOX_LENGTH;
    [self swap:i with:j];
}

return self.sBox;
}


-(void)swap:(int)i with:(int)j{

id tempObj = [self.sBox objectAtIndex:i];
[self.sBox replaceObjectAtIndex:i withObject:[self.sBox objectAtIndex:j]];
[self.sBox replaceObjectAtIndex:j withObject:tempObj];

}

   //I'm calling in a different class by creating the RC4encryptor object 
NSString *unescaped =  [[RC4StringEncryptor encryptor] encrypt:@"10/28/2013"];    
logDebug(@"the value is:%@",unescaped);

NSString *escapedString = (NSString *)CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes(
        NULL,
        (__bridge CFStringRef) unescaped,
        NULL,
        CFSTR("!*'();:@&=+$,/?%#[]\" "),
        kCFStringEncodingUTF8));

    NSLog(@"escapedString: %@",escapedString);

я тоже пробовал

#import <CommonCrypto/CommonCryptor.h>
@implementation NSData (RC4)

- (NSData *)RC4EncryptWithKey:(NSString *)key {
char keyPtr[kCCKeySizeMinRC4+1];
bzero(keyPtr, sizeof(keyPtr)); 

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

NSUInteger dataLength = [self length];

size_t bufferSize = dataLength + kCCKeySizeMinRC4;

void *buffer = malloc(bufferSize);

size_t numBytesEncrypted = 0;
CCCryptorStatus cryptStatus = CCCrypt(kCCEncrypt, kCCAlgorithmRC4, kCCModeRC4,
                                      keyPtr, kCCKeySizeMinRC4,
                                      NULL ,
                                      [self bytes], dataLength,
                                      buffer, bufferSize,
                                      &numBytesEncrypted);
if (cryptStatus == kCCSuccess) {
    return [NSData dataWithBytesNoCopy:buffer length:numBytesEncrypted];
}

free(buffer);
return nil;
}

- (NSData*) encryptString:(NSString*)plaintext withKey:(NSString*)key {
    return [[plaintext dataUsingEncoding:NSUTF8StringEncoding] RC4EncryptWithKey:key];
}

Но этот приведенный выше код не работает для меня. Пожалуйста, помогите мне в этом....


person Raghunathan    schedule 15.11.2013    source источник
comment
Вы нашли решение? Я хотел бы увидеть полный класс категории RC4   -  person Paulius Vindzigelskis    schedule 13.11.2014
comment
1. RC4 не предназначен для нового использования, он небезопасен. 2. Гораздо чаще данные кодируются в Base64, чем в кодировке URL.   -  person zaph    schedule 13.11.2014


Ответы (1)


Каким-то образом я исправил ваш код, и он отлично шифрует-дешифрует. Вот:

@interface RC4Crypt()

@property (nonatomic, strong) NSArray * key;
@property (nonatomic, strong) NSMutableArray * sBox;

@end

@implementation RC4Crypt

static const int SBOX_LENGTH = 256;
static const int KEY_MIN_LENGTH = 1;

-(NSString *)encrypt:(NSString *)string withKey:(NSString *)key{

    self.sBox = [[self frameSBox:key] mutableCopy];
    unichar code[string.length];

    int i = 0;
    int j = 0;
    for (int n = 0; n < string.length; n++) {
        i = (i + 1) % SBOX_LENGTH;
        j = (j + [[self.sBox objectAtIndex:i]integerValue]) % SBOX_LENGTH;
        [self.sBox exchangeObjectAtIndex:i withObjectAtIndex:j];

        int index=([self.sBox[i] integerValue]+[self.sBox[j] integerValue]);

        int rand=([self.sBox[(index%SBOX_LENGTH)] integerValue]);

        code[n]=(rand  ^  (int)[string characterAtIndex:n]);
    }
    const unichar* buffer;
    buffer = code;

    return  [NSString stringWithCharacters:buffer length:string.length];
}

- (NSString*) decrypt:(NSString*)string withKey:(NSString*)key
{
    return [self encrypt:string withKey:key];
}


-(NSArray *)frameSBox:(NSString *)keyValue{

    NSMutableArray *sBox = [[NSMutableArray alloc] initWithCapacity:SBOX_LENGTH];

    int j = 0;

    for (int i = 0; i < SBOX_LENGTH; i++) {
    [sBox addObject:[NSNumber numberWithInteger:i]];
    }

    for (int i = 0; i < SBOX_LENGTH; i++) {
        j = (j + [sBox[i] integerValue] + [keyValue characterAtIndex:(i % keyValue.length)]) % SBOX_LENGTH;
        [sBox exchangeObjectAtIndex:i withObjectAtIndex:j];
    }

    return [NSArray arrayWithArray:sBox];
}

@end
person Paulius Vindzigelskis    schedule 13.11.2014
comment
Действительно, действительно, для шифрования лучше использовать Common Crypto. Кроме того, предоставленный код не был проверен на наличие ошибок и уязвимостей, поэтому ему нельзя доверять для коммерческих проектов. - person zaph; 13.11.2014
comment
Спасибо ребята. Я починил это, - person Raghunathan; 02.03.2015