2012-01-14 68 views
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(); 
+1

你可以粘贴你是怎么做到的编码/解码? – greut 2012-01-14 13:20:46

+0

解密文件与原文件大小相同吗? – 2012-01-14 13:24:53

+0

在Windows上我想? – 2012-01-14 13:25:50

回答

5

我认为这个问题是您将二进制PDF数据视为PHP和C#端的文本。

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无效。

当你把解密的回报string,使用

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

当您返回它作为byte[](推荐),使用

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

此外,虽然我不确定如果没有阅读方法的所有文档,它似乎数据填充在PHP中,但从来没有在C#中“无衬垫”。 – 2012-01-14 14:32:56

+1

@Joachim:OP声明该过程适用于文本数据,这让我认为默认的填充设置是可以的。 – 2012-01-14 14:38:41

+0

是的,我绝对不知道这是一个问题,但我最终会注意到这些文件的大小相同:) – 2012-01-14 14:51:41