// DeflateStream.cs
// ------------------------------------------------------------------
//
// Copyright (c) 2009 Dino Chiesa and Microsoft Corporation.
// All rights reserved.
//
// This code module is part of DotNetZip, a zipfile class library.
//
// ------------------------------------------------------------------
//
// This code is licensed under the Microsoft Public License.
// See the file License.txt for the license details.
// More info on: http://dotnetzip.codeplex.com
//
// ------------------------------------------------------------------
//
// last saved (in emacs):
// Time-stamp: <2009-November-03 18:48:49>
//
// ------------------------------------------------------------------
//
// This module defines the DeflateStream class, which can be used as a replacement for
// the System.IO.Compression.DeflateStream class in the .NET BCL.
//
// ------------------------------------------------------------------
using System;
namespace Ionic.Zlib
{
///
/// A class for compressing and decompressing streams using the Deflate algorithm.
///
///
///
///
///
/// The DeflateStream is a Decorator on a . It adds DEFLATE compression or decompression to any stream.
///
///
/// Using this stream, applications can compress or decompress data via
/// stream Read and Write operations. Either compresssion or
/// decompression can occur through either reading or writing. The compression
/// format used is DEFLATE, which is documented in IETF RFC 1951, "DEFLATE
/// Compressed Data Format Specification version 1.3.".
///
///
/// This class is similar to , except that ZlibStream
/// adds the RFC 1950 - ZLIB
/// framing bytes to a compressed stream when compressing, or expects the RFC1950
/// framing bytes when decompressing. The DeflateStream does not.
///
///
///
///
///
///
internal class DeflateStream : System.IO.Stream
{
internal ZlibBaseStream _baseStream;
internal System.IO.Stream _innerStream;
bool _disposed;
///
/// Create a DeflateStream using the specified CompressionMode.
///
///
/// When mode is CompressionMode.Compress, the DeflateStream
/// will use the default compression level. The "captive" stream will be closed
/// when the DeflateStream is closed.
///
///
/// This example uses a DeflateStream to compress data from a file, and writes
/// the compressed data to another file.
///
/// using (System.IO.Stream input = System.IO.File.OpenRead(fileToCompress))
/// {
/// using (var raw = System.IO.File.Create(fileToCompress + ".deflated"))
/// {
/// using (Stream compressor = new DeflateStream(raw, CompressionMode.Compress))
/// {
/// byte[] buffer = new byte[WORKING_BUFFER_SIZE];
/// int n;
/// while ((n= input.Read(buffer, 0, buffer.Length)) != 0)
/// {
/// compressor.Write(buffer, 0, n);
/// }
/// }
/// }
/// }
///
///
/// Using input As Stream = File.OpenRead(fileToCompress)
/// Using raw As FileStream = File.Create(fileToCompress & ".deflated")
/// Using compressor As Stream = New DeflateStream(raw, CompressionMode.Compress)
/// Dim buffer As Byte() = New Byte(4096) {}
/// Dim n As Integer = -1
/// Do While (n <> 0)
/// If (n > 0) Then
/// compressor.Write(buffer, 0, n)
/// End If
/// n = input.Read(buffer, 0, buffer.Length)
/// Loop
/// End Using
/// End Using
/// End Using
///
///
/// The stream which will be read or written.
/// Indicates whether the DeflateStream will compress or decompress.
public DeflateStream(System.IO.Stream stream, CompressionMode mode)
: this(stream, mode, CompressionLevel.Default, false)
{
}
///
/// Create a DeflateStream using the specified CompressionMode and the specified CompressionLevel.
///
///
///
///
/// When mode is CompressionMode.Decompress, the level parameter
/// is ignored. The "captive" stream will be closed when the DeflateStream is
/// closed.
///
///
///
///
///
/// This example uses a DeflateStream to compress data from a file, and writes
/// the compressed data to another file.
///
/// using (System.IO.Stream input = System.IO.File.OpenRead(fileToCompress))
/// {
/// using (var raw = System.IO.File.Create(fileToCompress + ".deflated"))
/// {
/// using (Stream compressor = new DeflateStream(raw,
/// CompressionMode.Compress,
/// CompressionLevel.BestCompression))
/// {
/// byte[] buffer = new byte[WORKING_BUFFER_SIZE];
/// int n= -1;
/// while (n != 0)
/// {
/// if (n > 0)
/// compressor.Write(buffer, 0, n);
/// n= input.Read(buffer, 0, buffer.Length);
/// }
/// }
/// }
/// }
///
///
///
/// Using input As Stream = File.OpenRead(fileToCompress)
/// Using raw As FileStream = File.Create(fileToCompress & ".deflated")
/// Using compressor As Stream = New DeflateStream(raw, CompressionMode.Compress, CompressionLevel.BestCompression)
/// Dim buffer As Byte() = New Byte(4096) {}
/// Dim n As Integer = -1
/// Do While (n <> 0)
/// If (n > 0) Then
/// compressor.Write(buffer, 0, n)
/// End If
/// n = input.Read(buffer, 0, buffer.Length)
/// Loop
/// End Using
/// End Using
/// End Using
///
///
/// The stream to be read or written while deflating or inflating.
/// Indicates whether the DeflateStream will compress or decompress.
/// A tuning knob to trade speed for effectiveness.
public DeflateStream(System.IO.Stream stream, CompressionMode mode, CompressionLevel level)
: this(stream, mode, level, false)
{
}
///
/// Create a DeflateStream using the specified
/// CompressionMode, and explicitly specify whether the
/// stream should be left open after Deflation or Inflation.
///
///
///
///
///
/// This constructor allows the application to request that the captive stream
/// remain open after the deflation or inflation occurs. By default, after
/// Close() is called on the stream, the captive stream is also
/// closed. In some cases this is not desired, for example if the stream is a
/// memory stream that will be re-read after compression. Specify true for
/// the parameter to leave the stream open.
///
///
///
/// The DeflateStream will use the default compression level.
///
///
///
/// See the other overloads of this constructor for example code.
///
///
///
///
/// The stream which will be read or written. This is called the
/// "captive" stream in other places in this documentation.
///
///
///
/// Indicates whether the DeflateStream will compress or decompress.
///
///
/// true if the application would like the stream to
/// remain open after inflation/deflation.
public DeflateStream(System.IO.Stream stream, CompressionMode mode, bool leaveOpen)
: this(stream, mode, CompressionLevel.Default, leaveOpen)
{
}
///
/// Create a DeflateStream using the specified CompressionMode
/// and the specified CompressionLevel, and explicitly specify whether
/// the stream should be left open after Deflation or Inflation.
///
///
///
///
///
/// When mode is CompressionMode.Decompress, the level parameter is ignored.
///
///
///
/// This constructor allows the application to request that the captive stream
/// remain open after the deflation or inflation occurs. By default, after
/// Close() is called on the stream, the captive stream is also
/// closed. In some cases this is not desired, for example if the stream is a
/// that will be re-read after
/// compression. Specify true for the parameter
/// to leave the stream open.
///
///
///
///
///
///
/// This example shows how to use a DeflateStream to compress data from
/// a file, and store the compressed data into another file.
///
///
/// using (var output = System.IO.File.Create(fileToCompress + ".deflated"))
/// {
/// using (System.IO.Stream input = System.IO.File.OpenRead(fileToCompress))
/// {
/// using (Stream compressor = new DeflateStream(output, CompressionMode.Compress, CompressionLevel.BestCompression, true))
/// {
/// byte[] buffer = new byte[WORKING_BUFFER_SIZE];
/// int n= -1;
/// while (n != 0)
/// {
/// if (n > 0)
/// compressor.Write(buffer, 0, n);
/// n= input.Read(buffer, 0, buffer.Length);
/// }
/// }
/// }
/// // can write additional data to the output stream here
/// }
///
///
///
/// Using output As FileStream = File.Create(fileToCompress & ".deflated")
/// Using input As Stream = File.OpenRead(fileToCompress)
/// Using compressor As Stream = New DeflateStream(output, CompressionMode.Compress, CompressionLevel.BestCompression, True)
/// Dim buffer As Byte() = New Byte(4096) {}
/// Dim n As Integer = -1
/// Do While (n <> 0)
/// If (n > 0) Then
/// compressor.Write(buffer, 0, n)
/// End If
/// n = input.Read(buffer, 0, buffer.Length)
/// Loop
/// End Using
/// End Using
/// ' can write additional data to the output stream here.
/// End Using
///
///
/// The stream which will be read or written.
/// Indicates whether the DeflateStream will compress or decompress.
/// true if the application would like the stream to remain open after inflation/deflation.
/// A tuning knob to trade speed for effectiveness.
public DeflateStream(System.IO.Stream stream, CompressionMode mode, CompressionLevel level, bool leaveOpen)
{
_innerStream = stream;
_baseStream = new ZlibBaseStream(stream, mode, level, ZlibStreamFlavor.DEFLATE, leaveOpen);
}
#region Zlib properties
///
/// This property sets the flush behavior on the stream.
///
/// See the ZLIB documentation for the meaning of the flush behavior.
///
virtual public FlushType FlushMode
{
get { return (this._baseStream._flushMode); }
set
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
this._baseStream._flushMode = value;
}
}
///
/// The size of the working buffer for the compression codec.
///
///
///
///
/// The working buffer is used for all stream operations. The default size is
/// 1024 bytes. The minimum size is 128 bytes. You may get better performance
/// with a larger buffer. Then again, you might not. You would have to test
/// it.
///
///
///
/// Set this before the first call to Read() or Write() on the
/// stream. If you try to set it afterwards, it will throw.
///
///
public int BufferSize
{
get
{
return this._baseStream._bufferSize;
}
set
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
if (this._baseStream._workingBuffer != null)
throw new ZlibException("The working buffer is already set.");
if (value < ZlibConstants.WorkingBufferSizeMin)
throw new ZlibException(String.Format("Don't be silly. {0} bytes?? Use a bigger buffer, at least {1}.", value, ZlibConstants.WorkingBufferSizeMin));
this._baseStream._bufferSize = value;
}
}
///
/// The ZLIB strategy to be used during compression.
///
///
///
/// By tweaking this parameter, you may be able to optimize the compression for
/// data with particular characteristics.
///
public CompressionStrategy Strategy
{
get
{
return this._baseStream.Strategy;
}
set
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
this._baseStream.Strategy = value;
}
}
/// Returns the total number of bytes input so far.
virtual public long TotalIn
{
get
{
return this._baseStream._z.TotalBytesIn;
}
}
/// Returns the total number of bytes output so far.
virtual public long TotalOut
{
get
{
return this._baseStream._z.TotalBytesOut;
}
}
#endregion
#region System.IO.Stream methods
///
/// Dispose the stream.
///
///
/// This may or may not result in a Close() call on the captive stream.
/// See the constructors that have a leaveOpen parameter for more information.
///
protected override void Dispose(bool disposing)
{
try
{
if (!_disposed)
{
if (disposing && (this._baseStream != null))
this._baseStream.Close();
_disposed = true;
}
}
finally
{
base.Dispose(disposing);
}
}
///
/// Indicates whether the stream can be read.
///
///
/// The return value depends on whether the captive stream supports reading.
///
public override bool CanRead
{
get
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
return _baseStream._stream.CanRead;
}
}
///
/// Indicates whether the stream supports Seek operations.
///
///
/// Always returns false.
///
public override bool CanSeek
{
get { return false; }
}
///
/// Indicates whether the stream can be written.
///
///
/// The return value depends on whether the captive stream supports writing.
///
public override bool CanWrite
{
get
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
return _baseStream._stream.CanWrite;
}
}
///
/// Flush the stream.
///
public override void Flush()
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
_baseStream.Flush();
}
///
/// Reading this property always throws a .
///
public override long Length
{
get { throw new NotImplementedException(); }
}
///
/// The position of the stream pointer.
///
///
///
/// Setting this property always throws a . Reading will return the total bytes
/// written out, if used in writing, or the total bytes read in, if used in
/// reading. The count may refer to compressed bytes or uncompressed bytes,
/// depending on how you've used the stream.
///
public override long Position
{
get
{
if (this._baseStream._streamMode == Ionic.Zlib.ZlibBaseStream.StreamMode.Writer)
return this._baseStream._z.TotalBytesOut;
if (this._baseStream._streamMode == Ionic.Zlib.ZlibBaseStream.StreamMode.Reader)
return this._baseStream._z.TotalBytesIn;
return 0;
}
set { throw new NotImplementedException(); }
}
///
/// Read data from the stream.
///
///
///
///
/// If you wish to use the DeflateStream to compress data while
/// reading, you can create a DeflateStream with
/// CompressionMode.Compress, providing an uncompressed data stream.
/// Then call Read() on that DeflateStream, and the data read will be
/// compressed as you read. If you wish to use the DeflateStream to
/// decompress data while reading, you can create a DeflateStream with
/// CompressionMode.Decompress, providing a readable compressed data
/// stream. Then call Read() on that DeflateStream, and the data read
/// will be decompressed as you read.
///
///
///
/// A DeflateStream can be used for Read() or Write(), but not both.
///
///
///
/// The buffer into which the read data should be placed.
/// the offset within that data array to put the first byte read.
/// the number of bytes to read.
/// the number of bytes actually read
public override int Read(byte[] buffer, int offset, int count)
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
return _baseStream.Read(buffer, offset, count);
}
///
/// Calling this method always throws a .
///
/// this is irrelevant, since it will always throw!
/// this is irrelevant, since it will always throw!
/// irrelevant!
public override long Seek(long offset, System.IO.SeekOrigin origin)
{
throw new NotImplementedException();
}
///
/// Calling this method always throws a .
///
/// this is irrelevant, since it will always throw!
public override void SetLength(long value)
{
throw new NotImplementedException();
}
///
/// Write data to the stream.
///
///
///
///
/// If you wish to use the DeflateStream to compress data while
/// writing, you can create a DeflateStream with
/// CompressionMode.Compress, and a writable output stream. Then call
/// Write() on that DeflateStream, providing uncompressed data
/// as input. The data sent to the output stream will be the compressed form
/// of the data written. If you wish to use the DeflateStream to
/// decompress data while writing, you can create a DeflateStream with
/// CompressionMode.Decompress, and a writable output stream. Then
/// call Write() on that stream, providing previously compressed
/// data. The data sent to the output stream will be the decompressed form of
/// the data written.
///
///
///
/// A DeflateStream can be used for Read() or Write(),
/// but not both.
///
///
///
///
/// The buffer holding data to write to the stream.
/// the offset within that data array to find the first byte to write.
/// the number of bytes to write.
public override void Write(byte[] buffer, int offset, int count)
{
if (_disposed) throw new ObjectDisposedException("DeflateStream");
_baseStream.Write(buffer, offset, count);
}
#endregion
///
/// Compress a string into a byte array using DEFLATE.
///
///
///
/// Uncompress it with .
///
///
///
///
///
///
/// A string to compress. The string will first be encoded
/// using UTF8, then compressed.
///
///
/// The string in compressed form
public static byte[] CompressString(String s)
{
using (var ms = new System.IO.MemoryStream())
{
System.IO.Stream compressor =
new DeflateStream(ms, CompressionMode.Compress, CompressionLevel.BestCompression);
ZlibBaseStream.CompressString(s, compressor);
return ms.ToArray();
}
}
///
/// Compress a byte array into a new byte array using DEFLATE.
///
///
///
/// Uncompress it with .
///
///
///
///
///
///
/// A buffer to compress.
///
///
/// The data in compressed form
public static byte[] CompressBuffer(byte[] b)
{
using (var ms = new System.IO.MemoryStream())
{
System.IO.Stream compressor =
new DeflateStream( ms, CompressionMode.Compress, CompressionLevel.BestCompression );
ZlibBaseStream.CompressBuffer(b, compressor);
return ms.ToArray();
}
}
///
/// Uncompress a DEFLATE'd byte array into a single string.
///
///
///
///
///
///
/// A buffer containing GZIP-compressed data.
///
///
/// The uncompressed string
public static String UncompressString(byte[] compressed)
{
using (var input = new System.IO.MemoryStream(compressed))
{
System.IO.Stream decompressor =
new DeflateStream(input, CompressionMode.Decompress);
return ZlibBaseStream.UncompressString(compressed, decompressor);
}
}
///
/// Uncompress a DEFLATE'd byte array into a byte array.
///
///
///
///
///
///
/// A buffer containing data that has been compressed with DEFLATE.
///
///
/// The data in uncompressed form
public static byte[] UncompressBuffer(byte[] compressed)
{
using (var input = new System.IO.MemoryStream(compressed))
{
System.IO.Stream decompressor =
new DeflateStream( input, CompressionMode.Decompress );
return ZlibBaseStream.UncompressBuffer(compressed, decompressor);
}
}
}
}