PHP7中openssl与mcrypt的兼容问题

概况

项目中为了安全,对一些数据进行了加密,加密库是mcrypt,分别是
mcrypt_encrypt(MCRYPT_RIJNDAEL_128,$cipher_key,$plain_text, MCRYPT_MODE_CBC, $iv);
                                     
mcrypt_decrypt(MCRYPT_RIJNDAEL_128, $cipher_key,$cipher_text, MCRYPT_MODE_CBC $iv);

其中$cipher_key是使用hash_pbkdf2加密后的长度是32的数据
$iv长度是16

升级7.1之后使用openssl_encrypt和openssl_decrypt替代

  • 解密方法参数

      openssl_decrypt($cipher_text, 'aes-256-cbc', $cipher_key,  OPENSSL_RAW_DATA | OPENSSL_ZERO_PADDING , $iv);
      其中$cipher_key和$iv与之前一致
    

    通过这个可以完美的解密mcrypt_encrypt的数据;

那现在的问题就是要用openssl_encrypt的加密数据能用以上解密方法完美解密出来。

经过几次方式实验之后,发现只有用OPENSSL_RAW_DATA选项时能用以上解密方法解密出明文:

    openssl_encrypt($plain_text, 'aes-256-cbc', $cipher_key, OPENSSL_RAW_DATA, $iv);

但是解密后的明文的结束会多出几个字符(分别是ASCII的\1~\16),而且是有规则的长度,如16个\16,15个\15....1个\1。

以往用mcrypt解密之后都会用使用rtrim($plain_text, "\0"),处理最后填充的\0,但是这次结尾不是\0;

试错过程是:

  • 填充的问题

    处理方式是,计算长度后自己在明文后填充\0,使得明文长度是16的倍数,但是发现加了之后结尾多了16个\16,所以这个不是导致问原因;
    
  • openssl_encrypt有bug

      翻看PHP-src代码,找到openssl_encrypt源码,从代码逻辑上看只找到添加\0的逻辑,所以问题并不是PHP-src中的openssl_encrypt的问题
    
  • openssl在ubuntu 16.04有bug

     这一步没去验证排除 
    

还有一种方法就是,特殊处理这些字符串,因为这些字符串看起来也是有规律的,只需要在尾部截取掉就可以了,但是我现在只有有限数据,而且不知道真实原因,这个方法只能最后不得已才能使用,只能继续查找问题;

翻开以前打开的网页发现了一段话openssl_encrypt() adds PKCS7 padding to the plaintext before encrypting with a block cipher in CBC or ECB mode.
也就是CBC或ECB模式下,会使用PCKS7填充明文后再加密

PCKS7是什么Wiki

Padding is in whole bytes. The value of each added byte is the number of bytes that are added, i.e. N bytes, each of value N are added. The number of bytes added will depend on the block boundary to which the message needs to be extended.

The padding will be one of:

01
02 02
03 03 03
04 04 04 04
05 05 05 05 05
06 06 06 06 06 06
etc.

那么问题找到了,openssl_encrypt在OPENSSL_RAW_DATA选项下会按照这个规则添加结尾的字符串。

解决办法

        $pad = ord($text[strlen($text) - 1]);
        if ($pad>0 && $pad <= 16)
        {
            $text = substr($text, 0, -$pad);
        }

你可能感兴趣的:(PHP7中openssl与mcrypt的兼容问题)