如何获得openssl_decrypt二进制输出?

时间:2016-06-30 13:18:58

标签: php encryption pyopenssl

我正在尝试使用openssl_encrypt / openssl_decrypt加密/解密某些数据。目的如下:用户在GUI中输入一些数据,这些数据被加密并存储在数据库中。稍后,它将仅针对特定用户配置文件进行检索和解密。

加密部分工作正常,因为我已按照同事的建议,即将输出包装到bin2hex函数中。
重点是,即使我尝试使用hex2bin转换输出,我也无法在解密数据时获得二进制输出。我总是得到类似"��Ps�1�_G�5�OUT"的输出。

我的想法已经不多了,实际上我现在还不知道该怎么做。

这是我为测试此功能而编写的示例代码:

PHP

function encrypt($string) {

$cypher = 'aes-256-cbc';
$key = 's7aBkf4Ypn59bWviQziPDXyPasdaYlhQ';

$ivsize = openssl_cipher_iv_length($cypher);
$iv = openssl_random_pseudo_bytes($ivsize); 

$encripted = openssl_encrypt(
        $string, $cypher, $key, OPENSSL_RAW_DATA | OPENSSL_ZERO_PADDING, $iv
);

return $iv . $encripted;
}

function decrypt($string) {

$cypher = 'aes-256-cbc';
$key = 's7aBkf4Ypn59bWviQziPDXyPasdaYlhQ'; 

$ivsize = openssl_cipher_iv_length($cypher);
$iv = mb_substr($string, 0, $ivsize, '8bit');
$decrypted = mb_substr($string, $ivsize, null, '8bit');

$output = openssl_decrypt(
                $decrypted, $cifrado, $key, OPENSSL_RAW_DATA | OPENSSL_ZERO_PADDING, $iv
);

return $output;
}

HTML:

<!doctype html>
 <html>
    <head><title>TEST</title></head>
    <body>
        <div style='margin-left:250px;'>
            <form action="test.php" method="POST">
            Encrypt
            <input type="text" name="encrypt" value=''/>
            <input type="submit" name="send" value='Send'/>
            <br/><br/>
            <?php
            if (isset($_POST['encrypt']) && !empty($_POST['encrypt'])) {
                echo 'encrypted string: ' . bin2hex(encrypt($_POST['encrypt']));
            }
            ?>
            <br/>
            <br/>
            Decrypt
            <input type="text" name="decrypt" value=''/>
            <input type="submit" name="send" value='Send'/>
            <br/><br/>
            <?php
            if (isset($_POST['decrypt']) && !empty($_POST['decrypt'])) {
                echo 'decrypted string: ' . decrypt($_POST['decrypt']);
                var_dump(decrypt($_POST['decrypt']));
            }
            ?>
        </form>
</body>

任何想法或帮助都将不胜感激。

我的PHP版本是5.4.45-0 + deb7u2。

提前致谢。

1 个答案:

答案 0 :(得分:2)

根据@Zaph的评论更新。

有一个小的语法错误。但是,代码的问题在于,在加密例程中,您只能使用以下其中一个:OPENSSL_RAW_DATA或OPENSSL_ZERO_PADDING。它在文档中提到。

来自@Zaph的注释:应该使用PKCS#7填充,它是一般的填充。事实证明,PHP OpenSSL的默认值是PKCS#7。所以,不要添加任何填充选项,你会得到正确的东西。

Openssl_encrypt()在使用CBC或ECB模式的分组密码加密之前,将PKCS7填充添加到明文。解密后Openssl_decrypt()剥离填充。

我使用的是PHP 5.4.4。

因此,唯一需要的选项是:OPENSSL_RAW_DATA。

使加密字符串能够安全地存储在任何地方。我有base64_encoded

演示于:ideone.com

加密:

/**
 * Encrypt a string
 * 
 *     
 * @Param string $data 
 * @Param string $key
 * 
 * @return string  - base64_encoded   
 */ 
function encrypt($data, $key) {

  $cypher = 'aes-256-cbc';  
  $ivSize  = openssl_cipher_iv_length($cypher);
  $ivData  = openssl_random_pseudo_bytes($ivSize);

  $encripted = openssl_encrypt($data, 
                              $cypher, 
                              $key, 
                              OPENSSL_RAW_DATA, 
                              $ivData);


  return base64_encode($ivData  . $encripted);
}

解密:

/**
 * Decrypt a string
 * 
 * @Param string $data 
 * @Param string $key
 * 
 * @return string  - original text   
 */ 
function decrypt($data, $key) {

  $cypher = 'aes-256-cbc';  
  $ivSize  = openssl_cipher_iv_length($cypher);

  $data = base64_decode($data);
  $ivData   = substr($data, 0, $ivSize);

  $encData = substr($data, $ivSize);

  $output = openssl_decrypt($encData, 
                            $cypher, 
                            $key, 
                            OPENSSL_RAW_DATA, 
                            $ivData);
  return $output;
}

运行它:

$srcText = "Hello World! - " . uniqid();
$key    = 's7aBkf4Ypn59bWviQziPDXyPasdaYlhQ';

$srcEncrypted  = '';
$srcDecrypted  = '';

$srcEncrypted = encrypt($srcText, $key);

$srcDecrypted = decrypt($srcEncrypted, $key);

var_dump($srcText, 
         $srcEncrypted, 
         $srcDecrypted, 
         $srcDecrypted == $srcText);

示例输出:

string 'Hello World! - 5776adf944c52' (length=28)

string 'NOjIIMM0visDbJPmBsAMgH+OQbYiReLBSvzg5JVZTMUOCAtk3CO7FBNs/Dn/vE9s' (length=64)

string 'Hello World! - 5776adf944c52' (length=28)

boolean true