I'm trying to create a stream that receive an underlying stream in the constructor, and automatically encrypt data when we read it. The goal is that when we read from the EncryptingStream : - it read the underlying stream - it encrypt the bytes read from the underlying stream - it returns the encrypted bytes
My current code looks like this:
public class EncryptingStream : BaseCryptographicStream
{
protected CryptoStream _current;
protected ICryptoTransform _transform;
protected Stream _underlyingStream;
#region CTor
public EncryptingStream(Stream source, string encryptionKey)
: base(target)
{
this._underlyingStream = source;
this.Initialize(encryptionKey);
}
#endregion
protected void Initialize(string encryptionKey)
{
var input = CryptoTools.GenerateAlgorithmInputs(encryptionKey);
RijndaelManaged rijndael = new RijndaelManaged();
rijndael.Mode = CipherMode.CBC;
rijndael.Key = input[0];
rijndael.IV = input[1];
this._transform = rijndael.CreateEncryptor();
this._current = new CryptoStream(this._underlyingStream, this._transform, CryptoStreamMode.Write);
}
public override int Read(byte[] buffer, int offset, int count)
{
??
}
}
Notice: BaseCryptographicStream
inherit Stream and implement Stream abstract members not implemented by the class above.
My problem is that I'm stuck implementing the Read method.
I can't use the CryptoStream as this stream write to an output stream.
I think I've to use the ICryptoTransform
object but I've no idea on how to read correctly. Problem here is the mismatch between the length of encrypted and unencrypted data. so the buffer read from the underlying stream will have a different size than the buffer given to EncryptingStream.
Does someone know it this kind of stream is possible, and any idea on how to do it ?
Update 1: The problem I try to solve here is for WCF streaming. I stream (big) files to WCF and don't want to put the whole file in memory. Using CryptoStream means that I must use a intermediate MemoryStream to encrypt the content, which mean the whole file will be in memory
Instead of rolling your own, just use the standard CryptoStream
class. To enable reading from it, use CryptoStreamMode.Read
in the constructor call.