I have used this guide to encrypt the value of txtCode.Text
into a text file, which is then hidden - The value of the encrypted data is going to be used as the password for the database, so I need to read this value, decrypt it, then pass it into the connection string.
However, in this guide, the decryption is done by typing in a value, creating a bytKey
and bytIV
and then comparing the result of encryption of the newl typed string with the value in the text file. Obviously I can't ask the user to enter the password every time the database needs to be opened, so how can I achieve decryption using this code?
The code which decrypts and encrypts the data is
Public Sub EncryptOrDecryptFile(ByVal strInputFile As String, ByVal strOutputFile As String, _
ByVal bytKey() As Byte, ByVal bytIV() As Byte, ByVal Direction As CryptoAction)
Try
fsInput = New System.IO.FileStream(strInputFile, FileMode.Open, FileAccess.Read)
fsOutput = New System.IO.FileStream(strOutputFile, FileMode.OpenOrCreate, FileAccess.Write)
fsOutput.SetLength(0)
Dim bytBuffer(4096) As Byte
Dim lngBytesProcessed As Long = 0
Dim lngFileLength As Long = fsInput.Length
Dim intBytesInCurrentBlock As Integer
Dim csCryptoStream As CryptoStream
Dim cspRijndael As New System.Security.Cryptography.RijndaelManaged
Select Case Direction
Case CryptoAction.ActionEncrypt
csCryptoStream = New CryptoStream(fsOutput, _
cspRijndael.CreateEncryptor(bytKey, bytIV), _
CryptoStreamMode.Write)
Case CryptoAction.ActionDecrypt
csCryptoStream = New CryptoStream(fsOutput, _
cspRijndael.CreateDecryptor(bytKey, bytIV), _
CryptoStreamMode.Write)
End Select
While lngBytesProcessed < lngFileLength
intBytesInCurrentBlock = fsInput.Read(bytBuffer, 0, 4096)
csCryptoStream.Write(bytBuffer, 0, intBytesInCurrentBlock)
lngBytesProcessed = lngBytesProcessed + _
CLng(intBytesInCurrentBlock)
End While
csCryptoStream.Close()
fsInput.Close()
fsOutput.Close()
Catch ex As Exception
errorLog(ex)
End Try
End Sub
I need to use this, or another, subroutine to read the file, decrypt the data and then store it in a variable to pass into the connection string - Can this be done? If so, how?
After numerous attempts with that code of yours I decided to ditch it - it is simply old, ineffective and unsafe (as noted by Plutonix).
Here's a much more efficient piece of code where I've modified and improved an old class of mine - called
Aes256Stream
. It is derived from theCryptoStream
and will handle the IV automatically for you.The two main things to remember are:
CreateEncryptionStream()
creates a write only stream for performing encryption.CreateDecryptionStream()
creates a read only stream for performing decryption.Performing encryption
The simplest example of performing encryption only requires you to create a new
Aes256Stream
and write to it:Notes:
<output stream>
is the stream to encrypt the data to (for instance aFileStream
).<encryption key>
is the key (as a byte array) to use when encrypting the data.<buffer>
,<offset>
and<length>
behave just like when you write to a regular stream.Performing decryption
Likewise the most simple example of performing decryption only requires you to create a new
Aes256Stream
and read from it:Notes:
<input stream>
is the stream to decrypt data from (for instance aFileStream
).<decryption key>
(same as<encryption key>
above).AesStream.Length
is equal to calling<input stream>.Length
(same goes forAesStream.Position
and<input stream>.Position
).For your use
Here's how you can adapt your code to work with it...
Methods:
Usage example:
Online test
An online testable version of this code can be found here: https://dotnetfiddle.net/PXaJF8