Как расшифровать зашифрованную строку AES-256 из CryptoJS с помощью Java?

У меня есть зашифрованная строка AES-256 от CryptoJS с парольной фразой. Мне нужно расшифровать его на Java, но я не могу понять, как это сделать. Похоже, что для расшифровки вам нужны IV, ключ и соль, и как в главной странице CryptoJS, зашифрованные данные уже содержат все из них, и CryptoJS может каким-то образом извлечь их из зашифрованного ввода.

Кто-нибудь знает, как это сделать? Я видел так много примеров о CryptoJS - Java encrypt / decrypt, но большинство из них используют жестко запрограммированный IV / ключ или просто отправляют IV / ключ со стороны cryptoJS на сторону Java. Все, что у меня есть, - это кодовая фраза, точно такая же, как у этого сайта!


person kientux    schedule 19.03.2015    source источник
comment
Как вы создали зашифрованный текст в CryptoJS (лучше всего показать код) и можете ли вы изменить реализацию JS?   -  person Artjom B.    schedule 19.03.2015
comment
@ArtjomB. Нет, я не могу изменить реализацию JS. Я считаю, что зашифрованный текст создается просто с помощью этого: AES - CryptoJS, потому что я можно использовать его для расшифровки строки.   -  person kientux    schedule 19.03.2015
comment
Но как он создается? Это например: CryptoJS.AES.encrypt("message", "passphrase").toString()?   -  person Artjom B.    schedule 19.03.2015
comment
По этой ссылке вы можете увидеть var encrypted = CryptoJS.AES.encrypt("Message", "Secret Passphrase");. Это исходный код.   -  person kientux    schedule 19.03.2015


Ответы (1)


Когда сообщение зашифровано таким образом:

CryptoJS.AES.encrypt("message", "passphrase")

используется подход на основе пароля. Для этого CryptoJS генерирует новую соль и использует эту соль вместе с парольной фразой для получения ключа и IV (я воссоздал функцию деривации для этот вопрос).
После создания зашифрованного текста специальный форматер OpenSSL используется для кодирования зашифрованного текста. в том числе использованная соль:

var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);

Вам нужно декодировать строку в кодировке Base64, чтобы получить массив байтов, если он. Затем вы можете проверить, совпадает ли заголовок:

byte[] ctBytes = Base64.getDecoder().decode(ciphertext.getBytes("UTF-8"));
System.out.println("Is salted: " + new String(Arrays.copyOf(ctBytes, 8)).equals("Salted__"));

После этого вы можете восстановить из него соль и зашифрованный текст:

byte[] saltBytes = Arrays.copyOfRange(ctBytes, 8, 16);
byte[] ciphertextBytes = Arrays.copyOfRange(ctBytes, 16, ctBytes.length);

Получите ключ + IV:

byte[] key = new byte[keySize/8];
byte[] iv = new byte[ivSize/8];
EvpKDF(password.getBytes("UTF-8"), keySize, ivSize, saltBytes, key, iv);

И расшифруем зашифрованный текст:

Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
cipher.init(Cipher.DECRYPT_MODE, new SecretKeySpec(key, "AES"),
        new IvParameterSpec(iv));
byte[] recoveredPlaintextBytes = cipher.doFinal(ciphertextBytes);
String recoveredPlaintext = new String(recoveredPlaintextBytes);

Полный код:

public static void main(String[] args) throws UnsupportedEncodingException, GeneralSecurityException {
    String ciphertext = "U2FsdGVkX1+0m/gle/XQX1shjnpveUrl1fO3oOlurPMlTks6+oQlEPfOrucihzEz";
    String plaintext = "This is some example plaintext";
    String password = "This is a very strong password";
    int keySize = 256;
    int ivSize = 128;

    // var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);
    byte[] ctBytes = Base64.getDecoder().decode(ciphertext.getBytes("UTF-8"));
    System.out.println("Is salted: " + Arrays.equals(Arrays.copyOf(ctBytes, 8), new byte[]{0x53, 0x61, 0x6c, 0x74, 0x65, 0x64, 0x5f, 0x5f}));
    System.out.println("Is salted: " + new String(Arrays.copyOf(ctBytes, 8)).equals("Salted__"));

    byte[] saltBytes = Arrays.copyOfRange(ctBytes, 8, 16);
    System.out.println("Salt matches: " + Arrays.equals(saltBytes, hexStringToByteArray("b49bf8257bf5d05f")));

    byte[] ciphertextBytes = Arrays.copyOfRange(ctBytes, 16, ctBytes.length);
    System.out.println("CT matches: " + Arrays.equals(ciphertextBytes, hexStringToByteArray("5b218e7a6f794ae5d5f3b7a0e96eacf3254e4b3afa842510f7ceaee722873133")));

    byte[] key = new byte[keySize/8];
    byte[] iv = new byte[ivSize/8];
    EvpKDF(password.getBytes("UTF-8"), keySize, ivSize, saltBytes, key, iv);

    Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
    cipher.init(Cipher.DECRYPT_MODE, new SecretKeySpec(key, "AES"), new IvParameterSpec(iv));
    byte[] recoveredPlaintextBytes = cipher.doFinal(ciphertextBytes);
    String recoveredPlaintext = new String(recoveredPlaintextBytes);

    System.out.println("Recovered Plaintext: " + recoveredPlaintext);
    System.out.println("Expected Plaintext: " + plaintext);
}

public static byte[] EvpKDF(byte[] password, int keySize, int ivSize, byte[] salt, byte[] resultKey, byte[] resultIv) throws NoSuchAlgorithmException {
    return EvpKDF(password, keySize, ivSize, salt, 1, "MD5", resultKey, resultIv);
}

public static byte[] EvpKDF(byte[] password, int keySize, int ivSize, byte[] salt, int iterations, String hashAlgorithm, byte[] resultKey, byte[] resultIv) throws NoSuchAlgorithmException {
    keySize = keySize / 32;
    ivSize = ivSize / 32;
    int targetKeySize = keySize + ivSize;
    byte[] derivedBytes = new byte[targetKeySize * 4];
    int numberOfDerivedWords = 0;
    byte[] block = null;
    MessageDigest hasher = MessageDigest.getInstance(hashAlgorithm);
    while (numberOfDerivedWords < targetKeySize) {
        if (block != null) {
            hasher.update(block);
        }
        hasher.update(password);
        block = hasher.digest(salt);
        hasher.reset();

        // Iterations
        for (int i = 1; i < iterations; i++) {
            block = hasher.digest(block);
            hasher.reset();
        }

        System.arraycopy(block, 0, derivedBytes, numberOfDerivedWords * 4,
                Math.min(block.length, (targetKeySize - numberOfDerivedWords) * 4));

        numberOfDerivedWords += block.length/4;
    }

    System.arraycopy(derivedBytes, 0, resultKey, 0, keySize * 4);
    System.arraycopy(derivedBytes, keySize * 4, resultIv, 0, ivSize * 4);

    return derivedBytes; // key + iv
}

/**
 * Copied from https://stackoverflow.com/a/140861
 * */
public static byte[] hexStringToByteArray(String s) {
    int len = s.length();
    byte[] data = new byte[len / 2];
    for (int i = 0; i < len; i += 2) {
        data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4)
                + Character.digit(s.charAt(i+1), 16));
    }
    return data;
}

Код JavaScript:

var pw = "This is a very strong password";
var pt = "This is some example plaintext";
var encrypted = CryptoJS.AES.encrypt(pt, pw);
encrypted.toString(); // U2FsdGVkX1+0m/gle/XQX1shjnpveUrl1fO3oOlurPMlTks6+oQlEPfOrucihzEz
encrypted.salt.toString(); // b49bf8257bf5d05f
encrypted.ciphertext.toString(); // 5b218e7a6f794ae5d5f3b7a0e96eacf3254e4b3afa842510f7ceaee722873133
person Artjom B.    schedule 19.03.2015
comment
Приносим извинения за поздний ответ и большое вам спасибо! Работает очень хорошо. Из-за моих слабых знаний о криптографии я, возможно, никогда этого не пойму. - person kientux; 20.03.2015
comment
@kientux Это помогает читать исходный код, так как это открытый исходный код. Я также ничего об этом не знал, пока вопросы об этом не возникли здесь, на SO. - person Artjom B.; 20.03.2015
comment
Извините за вопрос, но у меня проблема с шифрованием в Java. Я пытаюсь реализовать метод шифрования CryptoJS, и я думаю, что прогресс будет следующим: Генерация соли - ›Используйте свой метод EvpKDF для получения ключа и IV -› Зашифруйте с помощью Cipher - ›Верните base64 как результат. Я пробовал использовать этот фрагмент кода: AES256Cryptor, но что-то не так, и я не могу расшифровать успешно, с BadPaddingException. Извините за беспокойство снова. - person kientux; 13.11.2015
comment
Не беда, я разобрался! И еще раз спасибо! Внимательно прочитав эту строку: After the ciphertext is produced a special OpenSSL formatter is used to encode the ciphertext including the used salt:, я понял, что мне нужно создать 8 байтов из строки "Salted__", затем объединить 8 байтов saltBytes, а затем cipherBytes! Именно то, что вы делаете, чтобы расшифровать из base64. Спасибо, я так много узнал сегодня. - person kientux; 13.11.2015
comment
@ArtjomB., Что означает EvpKDF? - person James Selvakumar; 18.01.2017
comment
@JamesSelvakumar Это имя парольной функции k ey d erivation f, используемой в CryptoJS и производной от OpenSSL EVP_BytesToKey - person Artjom B.; 18.01.2017