我正在编写一个允许用户上传文件和下载文件的Web应用程序。允许使用doc或jpeg等文件。文件在上传过程中加密,在下载过程中解密
我使用AES作为算法,所有文件都有一个密钥,但每个文件的盐不同。我还要确保加密和解密方法都使用相同的密钥和填充方法
但是,当网站尝试解密文件时,它会创建一个损坏的文件,其大小比原始文件大
因此,我不确定加密或解密过程是否会混淆文件
这是源代码:
//Encryption method
private bool SaveEnryptFile(FileUpload fileUp)
{
try
{
string fName = fileUp.PostedFile.FileName;
string outputDir = Path.Combine(ConfigurationManager.AppSettings["clientDocFolder"] + CurrentUser.Name);
DirectoryInfo di = new DirectoryInfo(outputDir);
di.Create();
string outputFile = Path.Combine(outputDir, fName);
if (File.Exists(outputFile))
{
throw new Exception("File already exists");
}
else
{
byte[] file = new byte[fileUp.PostedFile.ContentLength];
fileUp.PostedFile.InputStream.Read(file, 0, fileUp.PostedFile.ContentLength);
//get the key string from web.config
var key = ConfigurationManager.AppSettings["keyFile"];
//randomly create a salt
byte[] salt = new byte[8];
var rng = new RNGCryptoServiceProvider();
rng.GetBytes(salt);
var derivedBytes = new Rfc2898DeriveBytes(key, salt);
using (AesCryptoServiceProvider alg = new AesCryptoServiceProvider())
{
alg.Key = derivedBytes.GetBytes(alg.KeySize / 8);
alg.IV = derivedBytes.GetBytes(alg.BlockSize / 8);
alg.Padding = PaddingMode.Zeros;
// Create a decrytor to perform the stream transform.
using (ICryptoTransform encryptor = alg.CreateEncryptor())
{
using (var fs = File.Create(outputFile))
{
//store the salt in the encrypted file
fs.Write(salt, 0, 8);
//write encrypted bytes to encrypted file
using (CryptoStream cs = new CryptoStream(fs, encryptor, CryptoStreamMode.Write))
{
cs.Write(file, 0, file.Length);
cs.FlushFinalBlock();
}
}
}
}
return true;
}
}
catch (Exception ex)
{
throw new Exception(ex.Message);
}
}
//解密方法:
var path = Path.Combine(ConfigurationManager.AppSettings["clientDocFolder"] + CurrentUser.Name + "\\" + ((WebControl)sender).Attributes["DocumentID"]);
if (File.Exists(path))
{
using (FileStream fs = new FileStream(path, FileMode.Open, FileAccess.Read))
{
//Get the salt from the encrypted file
byte[] salt = new byte[8];
fs.Read(salt, 0, salt.Length);
//get key string from web.config file
string key = ConfigurationManager.AppSettings["keyFile"];
var derivedBytes = new Rfc2898DeriveBytes(key, salt);
AesCryptoServiceProvider alg = new AesCryptoServiceProvider();
alg.Key = derivedBytes.GetBytes(alg.KeySize / 8);
alg.IV = derivedBytes.GetBytes(alg.BlockSize / 8);
alg.Padding = PaddingMode.Zeros;
using (ICryptoTransform decryptor = alg.CreateDecryptor())
{
//byte array to store encrypted bytes.
//I use fs.Length-8 because first 8 bytes were used to store salt
byte[] encryptedBytes = new byte[fs.Length - 8];
int encryptedByteCnt = fs.Read(encryptedBytes, 0, encryptedBytes.Length);
using (MemoryStream ms = new MemoryStream(encryptedBytes))
{
byte[] plainBytes = new byte[encryptedByteCnt];
using (CryptoStream cs = new CryptoStream(ms, decryptor, CryptoStreamMode.Read))
{
//decrypt encrypted bytes into a byte array
int decryptedByteCnt = cs.Read(plainBytes, 0, plainBytes.Length);
}
//Write decrypted bytes in response stream
Response.ContentType = "application/octet-stream";
Response.AddHeader("content-disposition", "attachment; filename=" + Path.GetFileName(path));
Response.OutputStream.Write(plainBytes, 0, plainBytes.Length);
Response.Flush();
}
}
}
}
else
{
ScriptManager.RegisterStartupScript(this, this.GetType(), "DownloadDocument", "alert('" + String.Format("There is no file such as {0} exists. Please contact us if you face this problem.);", ((WebControl)sender).Attributes["DocumentID"]) + "'", true);
return;
}
答案 0 :(得分:0)
原来问题是http发布文件。这导致输入字节不被读取到数组。
最后我需要在读取到0之前设置流的位置
所以我在加密方法中稍微更改了代码,如下所示:
从
byte[] file = new byte[fileUp.PostedFile.ContentLength];
fileUp.PostedFile.InputStream.Read(file, 0, fileUp.PostedFile.ContentLength);
要:
byte[] file = new byte[fileUp.PostedFile.ContentLength];
Stream st = fileUp.FileContent;
st.Position = 0;
st.Read(file, 0, file.Length);
st.Close();
然而,我遇到了解密问题。它将在解密的字节数组中添加额外的字节。他们中的大多数都是0。
更新:我更改了解密的写入内容,因此不会写入尾随零。但是现在从服务器发送的下载文件将包含页面本身的html。