Wrote myself a small password management utility that is supposed to support encryption of a master password file. Everything seems to work fine, and no exceptions are thrown during the entire encrypt/decrypt operation. However upon trying to parse data from the resulting decryption, a Unexpected end of file has occurred. The following elements are not closed: Account. Line 6, position 10, exception is thrown.
When looking at the decrypted XML file in web browser, it only shows the first </Account> node, all other nodes did not make it out of the decryption process. I've attached the code segment below....where did I screw the pooch?
public partial class Form1 : Form
{
public string fileName = "passfile.xml";
public DataSet ds = new DataSet("Account List");
public DataTable accounts = new DataTable("Accounts");
public Form1()
{
InitializeComponent();
accountGroupsBox.Enabled = false;
menuStrip1.Enabled = false;
button2.Enabled = false;
Types type = new Types();
this.accountGroupsBox.Items.AddRange(type.accountTypes);
accounts.Columns.AddRange(new DataColumn[] {
new DataColumn("Username"),
new DataColumn("Password"),
new DataColumn("Description")});
dataGridView1.DataSource = accounts;
dataGridView1.Columns["Username"].FillWeight = 25;
dataGridView1.Columns["Password"].FillWeight = 25;
dataGridView1.Columns["Description"].FillWeight = 50;
}
private void addNewPasswordToolStripMenuItem_Click(object sender, EventArgs e)
{
Form2 addAccount = new Form2(this);
addAccount.Show();
}
private void S_Click(object sender, EventArgs e)
{
accountGroupsBox.Enabled = true;
menuStrip1.Enabled = true;
button2.Enabled = true;
label2.Text = "Interface Unlocked";
}
private void accountGroupsBox_SelectedIndexChanged(object sender, EventArgs e)
{
accounts.Clear();
try
{
XmlDocument doc = new XmlDocument();
doc.Load(fileName);
foreach (XmlNode node in doc.GetElementsByTagName("Account"))
{
if (node["AccountType"].InnerText == accountGroupsBox.SelectedItem.ToString())
{
DataRow row = accounts.Rows.Add(
node["Username"].InnerText,
node["Password"].InnerText,
node["Description"].InnerText);
}
}
}
catch (FileLoadException ex)
{
MessageBox.Show(ex.ToString(), "Error!", MessageBoxButtons.OK);
}
}
public void Encrypt()
{
string temp = Path.GetTempFileName();
string path = System.IO.Path.GetDirectoryName(Application.ExecutablePath);
byte[] pword = Encoding.UTF8.GetBytes(textBox1.Text);
byte[] hash = SHA256.Create().ComputeHash(pword);
byte[] iv = MD5.Create().ComputeHash(pword);
byte[] key = hash;
using(FileStream fsInput = new FileStream(fileName, FileMode.Open, FileAccess.Read))
using(SymmetricAlgorithm alg = Aes.Create())
using(ICryptoTransform enc = alg.CreateEncryptor(key, iv))
using (FileStream fsOutput = new FileStream(temp, FileMode.Create, FileAccess.Write))
using (CryptoStream cs = new CryptoStream(fsOutput, enc, CryptoStreamMode.Write))
{
byte[] byteInput = new byte[fsInput.Length - 1];
fsInput.Read(byteInput, 0, byteInput.Length);
cs.Write(byteInput, 0, byteInput.Length);
}
FileInfo old = new FileInfo(fileName);
FileInfo newer = new FileInfo(temp);
old.Delete();
newer.MoveTo(Path.Combine(path, fileName));
}
public void Decrypt()
{
string temp = Path.GetTempFileName();
string path = System.IO.Path.GetDirectoryName(Application.ExecutablePath);
byte[] pword = Encoding.UTF8.GetBytes(textBox1.Text);
byte[] hash = SHA256.Create().ComputeHash(pword);
byte[] iv = MD5.Create().ComputeHash(pword);
byte[] key = hash;
using(FileStream fsInput = new FileStream(fileName, FileMode.Open, FileAccess.Read))
using(SymmetricAlgorithm alg = Aes.Create())
using(ICryptoTransform enc = alg.CreateDecryptor(key, iv))
using (CryptoStream cs = new CryptoStream(fsInput, enc, CryptoStreamMode.Read))
{
StreamWriter sw = new StreamWriter(temp);
sw.Write(new StreamReader(cs).ReadToEnd());
sw.Flush();
sw.Close();
}
FileInfo encrypted = new FileInfo(fileName);
FileInfo decrypted = new FileInfo(temp);
encrypted.Delete();
decrypted.MoveTo(Path.Combine(path, fileName));
}

1 answers
I'm an idiot. What I had originally posted was all wrong. It was wrong because I didn't encrypt correctly which caused all the problems. The key was to allow the CryptoStream to close (and therefore flush its buffers) before interpreting the data. I didn't and I imagine your code does the same. Try flushing your CryptoStream at the end of the block to see if that fixes it.
I'll leave a now 100% working code I had put earlier for your reference. I wrote up an Encryption class to handle the encryption and decryption of data based on a password. This program also tests it. You're free to use it if you wish.
[edit]
Fixed++.
answered one year ago by:
538
45
Wasnt sure which site to catch you on. Like I noted earlier, the flushing thing is working nicely. Based on the example you've put here, I understand what the encryption class is doing, but its not clear to me how I would modify this for what I'm trying to do. I figure the entire encryption part shouldnt be changed, I would just change the portions that output the decrypted data to the Console to instead direct the output into the Temp file and do the move/rename like I've been doing?