node.js — получение разных значений при шифровании в NodeJS и переполнение стека OpenSSL

Я работаю над связью клиент-сервер и застрял в том, чтобы обе стороны придумали одинаковое значение зашифрованного токена. Не могу понять, почему они разные. Ключи и векторы инициализации вместе с самим сообщением одинаковы.

Вот функция, которая выполняет шифрование в коде клиента:

int main()
{

try
{
std::string message = "HelloWorld";

while ((message.size() & 0xf) != 0xf)
message += " ";

size_t inputslength = message.length();
unsigned char aes_input[inputslength];

memset(aes_input, 0, inputslength/8);

strcpy((char*) aes_input, message.c_str());
unsigned char iv[] = {'0','f','9','3','8','7','b','3','f','9','4','b','f','0','6','f'};

unsigned char aes_key[] = {'Z','T','k','0','Y','T','U','5','Y','j','N','h','M','j','k','4','N','G','I','3','N','m','I','x','N','W','E','x','N','z','d','i'};

// buffers for encryption and decryption
const size_t encslength = ((inputslength + AES_BLOCK_SIZE) / AES_BLOCK_SIZE) * AES_BLOCK_SIZE;
unsigned char enc_out[encslength];
unsigned char dec_out[inputslength];
memset(enc_out, 0, sizeof(enc_out));
memset(dec_out, 0, sizeof(dec_out));

AES_KEY enc_key, dec_key;
AES_set_encrypt_key(aes_key, AES_KEYLENGTH, &enc_key);
AES_cbc_encrypt(aes_input, enc_out, inputslength, &enc_key, iv, AES_ENCRYPT);

AES_set_decrypt_key(aes_key, AES_KEYLENGTH, &dec_key);AES_cbc_encrypt(enc_out, dec_out, encslength, &dec_key, iv, AES_DECRYPT);

printf("original:\t");
hex_print(aes_input, sizeof(aes_input));
printf("encrypt:\t");
hex_print(enc_out, sizeof(enc_out));

printf("decrypt:\t");
hex_print(dec_out, sizeof(dec_out));

std::stringstream ss;
for(int i = 0; i < encslength; i++)
{
ss << enc_out[i];
}
return 0;
}
}

Выход

original: 48 65 6C 6C 6F 57 6F 72 6C 64 20 20 20 20 20
encrypt: 72 70 A2 0D FB A1 65 15 17 97 6E 5D 36 23 E2 FA
decrypt: 0A 73 F7 52 AC C1 68 54 1D CA 7A 1F 70 33 F4

Между тем .. на сервере:

function encryptToken(token)
{
const iv = '0f9387b3f94bf06f';
const key = 'ZTk0YTU5YjNhMjk4NGI3NmIxNWExNzdi';

console.log("key len: " + key.length);

const encrypt = (value) => {
const cipher = crypto.createCipheriv('AES-256-CBC', key, iv);
let encrypted = cipher.update(value, 'utf8', 'hex');
encrypted += cipher.final('hex');
return encrypted;
};

console.log('Encrypteddd value: ', encrypt('HelloWorld'));
}

Выход

Encrypteddd value:  0c491f8c5256b9744550688fc54926e8

Прежде чем попробовать CBC-256 для шифрования, я попробовал более простой режим шифрования, ECB-128, и все сводится к одной и той же проблеме. Различные токены шифрования создаются на стороне клиента и сервера, что приводит к невозможности расшифровки того, что происходит на стороне сервера. Любые мозговые штурмы помогут, пожалуйста. У меня заканчиваются идеи, спасибо.

Обновление 12.26 —

После получения совета относительно вектора инициализации и длины массива на стороне клиента … вот мой обновленный код с выводом:

int main()
{

try
{
std::string message = "HelloWorld";
while ((message.size() & 0xf) != 0xf)
message += " ";

size_t inputslength = message.length();
unsigned char aes_input[inputslength+1];

memset(aes_input, 0, inputslength/8);

strcpy((char*) aes_input, message.c_str());
unsigned char iv[] = {0x0f, 0x93, 0x87, 0xb3, 0xf9, 0x4b, 0xf0, 0x6f};
unsigned char aes_key[] = {'Z','T','k','0','Y','T','U','5','Y','j','N','h','M','j','k','4','N','G','I','3','N','m','I','x','N','W','E','x','N','z','d','i'};

// buffers for encryption and decryption
const size_t encslength = ((inputslength + AES_BLOCK_SIZE) / AES_BLOCK_SIZE) * AES_BLOCK_SIZE;
unsigned char enc_out[encslength];
unsigned char dec_out[inputslength];
memset(enc_out, 0, sizeof(enc_out));
memset(dec_out, 0, sizeof(dec_out));

AES_KEY enc_key, dec_key;
AES_set_encrypt_key(aes_key, AES_KEYLENGTH, &enc_key);
AES_cbc_encrypt(aes_input, enc_out, inputslength, &enc_key, iv, AES_ENCRYPT);

AES_set_decrypt_key(aes_key, AES_KEYLENGTH, &dec_key);AES_cbc_encrypt(enc_out, dec_out, encslength, &dec_key, iv, AES_DECRYPT);

printf("original:\t");
hex_print(aes_input, sizeof(aes_input));
printf("encrypt:\t");
hex_print(enc_out, sizeof(enc_out));

printf("decrypt:\t");
hex_print(dec_out, sizeof(dec_out));

std::stringstream ss;
for(int i = 0; i < encslength; i++)
{
ss << enc_out[i];
}
return 0;
}

}

//Output:
original:   48 65 6C 6C 6F 57 6F 72 6C 64 00
encrypt:    54 CD 98 20 59 D9 7B 2D D4 23 ED EC D0 13 97 59

Код Nodejs не изменился, и это остается выводом:

Encrypteddd value:  0c491f8c5256b9744550688fc54926e8

1

Решение

Итак, вот сделка. Каждый звонок AES_cbc_encrypt изменит значение вектора инициализации. Они делают это так, чтобы вы могли звонить AES_*_encrypt и обрабатывать сообщения размером более одного блока. Но потому что вызов шифрования меняет значение ivвызов дешифрования получает разные вектор инициализации.

Одним (ужасным) решением было бы сделать два вектора:

unsigned char iv_encrypt[] = { /* stuff */ };
unsigned char iv_decrypt[] = { /* same stuff */ };

Таким образом, вы будете передавать одни и те же данные каждому AES_cbc_encrypt вызов. Это, по крайней мере, покажет, что вы можете расшифровать исходные данные. Лучшим и гибким способом достижения вашей цели будет использование клона вашего вектора инициализации для каждого вызова. Что-то вроде:

unsigned char iv[] = { /* stuff */ };
unsigned char *tmp_iv = static_cast<unsigned char*>( malloc( sizeof( iv ) ) );
...
memcpy( tmp_iv, iv, sizeof(iv) );
AES_cbc_encrypt(aes_input, enc_out, inputslength, &enc_key, tmp_iv, AES_ENCRYPT);
...
memcpy( tmp_iv, iv, sizeof(iv) );
AES_cbc_encrypt(enc_out, dec_out, inputslength, &dec_key, tmp_iv, AES_DECRYPT);
1

Другие решения

Других решений пока нет …