使用PHP和nodejs进行通讯时候遇到双方加解密结果不一致的问题。
注意到crypto.createCipher(algorithm, password[, options])
方法有如下的提示。
The implementation of crypto.createCipher() derives keys using the OpenSSL
function EVP_BytesToKey with the digest algorithm set to MD5, one iteration, and no salt.
createCipher
方法只接受key
,不接受iv
参数,iv
参数是使用openssl
的EVP_BytesToKey
方法对key
进行扩展得来的,salt
为空,hash
算法为MD5
。
可以使用新的createCipheriv
方法代替,但原来的NodeJs代码不能修改,只能从php方面入手修改。
简易实现-使用php对key进行扩展,再进行加解密。
function EVP_BytesToKey($salt, $password) {$bytes = '';$last = '';while(strlen($bytes) < 48) {$last = hash('md5', $last . $password . $salt, true);$bytes.= $last;}return $bytes;}
//从扩展结果中分离真正用于通讯的key和iv
$key = '12345678901234561234567890123456';
$devd = EVP_BytesToKey('', $key);
$key = substr($devd, 0, 32);
$iv = substr($devd, 32, 16);@openssl_encrypt('test-data', "aes-256-cbc", $key, 1, $iv);
nodejs直接使用key进行加解密
const key = '12345678901234561234567890123456';
crypto.createCipher("aes-256-cbc", key)
EVP_BytesToKey的完整实现
/*** @param string $password* @param int $nkey 需要生成的密钥长度* @param int $niv 需要生成的iv长度* @param string $hash HASH算法,默认MD5* @param string $salt salt。默认为空* @param int $round HASH运行轮数* @return array*/
function EVP_BytesToKey(string $password, int $nkey = 32, int $niv = 16, string $hash = 'md5', string $salt = '', int $round = 1): array
{$bytes = '';$last = '';$total = $nkey + $niv;while (strlen($bytes) < $total) {$last = hash($hash, $last . $password . $salt, true);for ($i = 1; $i < $round; $i++) {$last = hash($hash, $last, true);}$bytes .= $last;}return ['key' => substr($bytes, 0, $nkey),'iv' => substr($bytes, $nkey, $niv),];}
更多语言实现
https://github.com/sometiny/evp_bytestokey