यह इतना आसान है, बस ईसीबी या किसी भी अन्य एन्क्रिप्शन विधियों का उपयोग करके धारा (स्ट्रीम। स्थिति) की स्थिति के आधार पर डेटा के समान आकार के साथ एक लंबी कुंजी उत्पन्न करें और फिर XOR लागू करें। यह तलाशने योग्य, बहुत तेज़ और 1 से 1 एन्क्रिप्शन है, जो आउटपुट लंबाई इनपुट लंबाई के बराबर है। यह स्मृति कुशल है और आप इसे विशाल फ़ाइलों पर उपयोग कर सकते हैं। मुझे लगता है कि इस विधि का उपयोग आधुनिक विनज़िप एईएस एन्क्रिप्शन में भी किया जाता है। साल
प्रत्येक धारा के लिए एक अद्वितीय नमक का उपयोग करना है अन्यथा कोई एन्क्रिप्शन: केवल एक चीज है कि आप सावधान किया जाना चाहिए कि है। मैंने इसका परीक्षण नहीं किया है, लेकिन अगर आपको लगता है कि इसमें कोई समस्या है तो कृपया मुझे बताएं।
public class SeekableAesStream : Stream
{
private Stream baseStream;
private AesManaged aes;
private ICryptoTransform encryptor;
public bool autoDisposeBaseStream { get; set; } = true;
/// <param name="salt">//** WARNING **: MUST be unique for each stream otherwise there is NO security</param>
public SeekableAesStream(Stream baseStream, string password, byte[] salt)
{
this.baseStream = baseStream;
using (var key = new PasswordDeriveBytes(password, salt))
{
aes = new AesManaged();
aes.KeySize = 128;
aes.Mode = CipherMode.ECB;
aes.Padding = PaddingMode.None;
aes.Key = key.GetBytes(aes.KeySize/8);
aes.IV = new byte[16]; //zero buffer is adequate since we have to use new salt for each stream
encryptor = aes.CreateEncryptor(aes.Key, aes.IV);
}
}
private void cipher(byte[] buffer, int offset, int count, long streamPos)
{
//find block number
var blockSizeInByte = aes.BlockSize/8;
var blockNumber = (streamPos/blockSizeInByte) + 1;
var keyPos = streamPos % blockSizeInByte;
//buffer
var outBuffer = new byte[blockSizeInByte];
var nonce = new byte[blockSizeInByte];
var init = false;
for (int i = offset; i < count; i++)
{
//encrypt the nonce to form next xor buffer (unique key)
if (!init || (keyPos % blockSizeInByte) == 0)
{
BitConverter.GetBytes(blockNumber).CopyTo(nonce, 0);
encryptor.TransformBlock(nonce, 0, nonce.Length, outBuffer, 0);
if (init) keyPos = 0;
init = true;
blockNumber++;
}
buffer[i] ^= outBuffer[keyPos]; //simple XOR with generated unique key
keyPos++;
}
}
public override bool CanRead { get { return baseStream.CanRead; } }
public override bool CanSeek { get { return baseStream.CanSeek; } }
public override bool CanWrite { get { return baseStream.CanWrite; } }
public override long Length { get { return baseStream.Length; } }
public override long Position { get { return baseStream.Position; } set { baseStream.Position = value; } }
public override void Flush() { baseStream.Flush(); }
public override void SetLength(long value) { baseStream.SetLength(value); }
public override long Seek(long offset, SeekOrigin origin) { return baseStream.Seek(offset, origin); }
public override int Read(byte[] buffer, int offset, int count)
{
var streamPos = Position;
var ret = baseStream.Read(buffer, offset, count);
cipher(buffer, offset, count, streamPos);
return ret;
}
public override void Write(byte[] buffer, int offset, int count)
{
cipher(buffer, offset, count, Position);
baseStream.Write(buffer, offset, count);
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
encryptor?.Dispose();
aes?.Dispose();
if (autoDisposeBaseStream)
baseStream?.Dispose();
}
base.Dispose(disposing);
}
}
उपयोग:
static void test()
{
var buf = new byte[255];
for (byte i = 0; i < buf.Length; i++)
buf[i] = i;
//encrypting
var uniqueSalt = new byte[16]; //** WARNING **: MUST be unique for each stream otherwise there is NO security
var baseStream = new MemoryStream();
var cryptor = new SeekableAesStream(baseStream, "password", uniqueSalt);
cryptor.Write(buf, 0, buf.Length);
//decrypting at position 200
cryptor.Position = 200;
var decryptedBuffer = new byte[50];
cryptor.Read(decryptedBuffer, 0, 50);
}
हाय, आप हमें जान सकते हैं कि आप अब तक कर रहे हैं? मेरा मानना है कि यदि आपके क्रिप्टोस्ट्रीम से जुड़े स्ट्रीम रीडर हैं तो आप स्थिति को स्थानांतरित कर सकते हैं और खोज सकते हैं ... –