2

我有一个 PHP 程序,它将 PDF 文件加密为 .xxx 文件,此输出正在由一个 C# 程序读取,该程序将这个 .xxx 文件解密回 PDF 文件。

我的问题是,当我打开由 C# 解密的文件时,PDF 阅读器告诉我文件已损坏.. 当我在 PHP 中加密纯文本并在 C# 上解密时,我得到了我加密的文件.. 所以问题只出现在 PDF 文件中,或者换句话说,它出现在BINARY文件中

有什么建议么 ?!

笔记:

  1. 在 PHP 中,我使用 mcrypt 扩展 Rijndael 算法 CBC PKCS7 填充(填充是手动完成的)
  2. 在 C# 中,我使用 RijndaelManaged 类来加密和解密数据

编辑

这是我在 PHP 中使用的加密方法:

    function encrypt($key, $iv, $text) {
        ini_set ( 'memory_limit', '-1' );
        $mcrypt_cipher = MCRYPT_RIJNDAEL_256;
        $mcrypt_mode = MCRYPT_MODE_CBC;
        $text=addpadding($text,mcrypt_get_block_size($mcrypt_cipher,'cbc'));
        $encrypted = rtrim ( mcrypt_encrypt ( $mcrypt_cipher, $key, $text, $mcrypt_mode, $iv ), "\0" );
        $encrypted = base64_encode ( $encrypted );
        return $encrypted;
    }

这是C#中的解密方法:

    public static string DecryptString(string message, string KeyString, string IVString)
    {
        byte[] Key = Encoding.UTF8.GetBytes(KeyString);
        byte[] IV = Encoding.UTF8.GetBytes(IVString);

        string decrypted = null;
        RijndaelManaged rj = new RijndaelManaged();
        rj.BlockSize = 256;
        rj.Key = Key;
        rj.IV = IV;
        rj.Mode = CipherMode.CBC;
        rj.Padding = PaddingMode.PKCS7;
        try
        {
            MemoryStream ms = new MemoryStream();
            //Encoding enc = new UTF8Encoding();
            byte[] messageBytes = Convert.FromBase64String(message);
            using (CryptoStream cs = new CryptoStream(ms, rj.CreateDecryptor(Key, IV), CryptoStreamMode.Write))
            {
                //byte[] messageBytes = enc.GetBytes(message);

                cs.Write(messageBytes, 0, messageBytes.Length);
                cs.Close();
            }
            byte[] encoded = ms.ToArray();
            decrypted = Encoding.UTF8.GetString(encoded);

            ms.Close();
        }
        catch (Exception e)
        {
            MessageBox.Show("An error occurred:"+ e.Message);
        }
        finally
        {
            rj.Clear();
        }

        return decrypted;
    }

以下是我如何在 C# 中调用解密以及如何编写输出:

                string Key = cryptography.MD5("X-Ware" + cryptography.MD5("123"));
                string IV = cryptography.MD5("XWare");
                string decrypted = cryptography.DecryptString(contents, Key, IV);
                string outputFilename = cryptography.MD5(OFD.FileName) + ".tmp";

                StreamWriter sw = new StreamWriter("C:\\Windows\\Temp\\" + outputFilename, false, Encoding.UTF8);
                BinaryWriter bw = new BinaryWriter(sw.BaseStream, Encoding.UTF8);
                //sw.Write(decrypted);
                bw.Write(decrypted);
                sw.Close();
                bw.Close();
4

1 回答 1

5

我认为问题在于您在 PHP 和 C# 端都将二进制 PDF 数据视为文本。

 decrypted = Encoding.UTF8.GetString(encoded);

encoded如果表示二进制数据,则毫无意义。您可能应该跳过这一步并将您的定义DecryptString()为返回byte[]。然后也重命名它。

如果您确实希望将其作为字符串,则使用 ASCII 或 ANSI 编码可能会更好:

 decrypted = Encoding.ASCII.GetString(encoded);

但错误可能已经发生在 PHP 端,我不知道。

另外,我刚刚注意到:

    StreamWriter sw = new StreamWriter("C:\\Windows\\Temp\\" + outputFilename,  
           false, Encoding.UTF8);
    BinaryWriter bw = new BinaryWriter(sw.BaseStream, Encoding.UTF8);

这是创建 BinaryWriter 的一种非常复杂的方法。将不使用编码。和

 bw.Write(decrypted);

这将使用长度前缀写入字符串,这肯定会使您的 PDF 无效。

当您保留 Decrypt as 的返回时string,请使用

  File.WriteAllText("C:\\Windows\\Temp\\" + outputFilename, decrypted);

当您将其作为byte[](推荐)返回时,请使用

 File.WriteAllBytes("C:\\Windows\\Temp\\" + outputFilename, decrypted);
于 2012-01-14T13:59:43.690 回答