Compressor.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. using Renci.SshNet.Security;
  2. using System.IO;
  3. using System;
  4. namespace Renci.SshNet.Compression
  5. {
  6. /// <summary>
  7. /// Represents base class for compression algorithm implementation
  8. /// </summary>
  9. public abstract class Compressor : Algorithm, IDisposable
  10. {
  11. private readonly ZlibStream _compressor;
  12. private readonly ZlibStream _decompressor;
  13. private MemoryStream _compressorStream;
  14. private MemoryStream _decompressorStream;
  15. /// <summary>
  16. /// Gets or sets a value indicating whether compression is active.
  17. /// </summary>
  18. /// <value>
  19. /// <c>true</c> if compression is active; otherwise, <c>false</c>.
  20. /// </value>
  21. protected bool IsActive { get; set; }
  22. /// <summary>
  23. /// Gets the session.
  24. /// </summary>
  25. protected Session Session { get; private set; }
  26. /// <summary>
  27. /// Initializes a new instance of the <see cref="Compressor"/> class.
  28. /// </summary>
  29. public Compressor()
  30. {
  31. this._compressorStream = new MemoryStream();
  32. this._decompressorStream = new MemoryStream();
  33. this._compressor = new ZlibStream(this._compressorStream, CompressionMode.Compress);
  34. this._decompressor = new ZlibStream(this._decompressorStream, CompressionMode.Decompress);
  35. }
  36. /// <summary>
  37. /// Initializes the algorithm
  38. /// </summary>
  39. /// <param name="session">The session.</param>
  40. public virtual void Init(Session session)
  41. {
  42. this.Session = session;
  43. }
  44. /// <summary>
  45. /// Compresses the specified data.
  46. /// </summary>
  47. /// <param name="data">Data to compress.</param>
  48. /// <returns>Compressed data</returns>
  49. public virtual byte[] Compress(byte[] data)
  50. {
  51. if (!this.IsActive)
  52. {
  53. return data;
  54. }
  55. this._compressorStream.SetLength(0);
  56. this._compressor.Write(data, 0, data.Length);
  57. return this._compressorStream.ToArray();
  58. }
  59. /// <summary>
  60. /// Decompresses the specified data.
  61. /// </summary>
  62. /// <param name="data">Compressed data.</param>
  63. /// <returns>Decompressed data.</returns>
  64. public virtual byte[] Decompress(byte[] data)
  65. {
  66. if (!this.IsActive)
  67. {
  68. return data;
  69. }
  70. this._decompressorStream.SetLength(0);
  71. this._decompressor.Write(data, 0, data.Length);
  72. return this._decompressorStream.ToArray();
  73. }
  74. #region IDisposable Members
  75. private bool _isDisposed;
  76. /// <summary>
  77. /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged ResourceMessages.
  78. /// </summary>
  79. public void Dispose()
  80. {
  81. Dispose(true);
  82. GC.SuppressFinalize(this);
  83. }
  84. /// <summary>
  85. /// Releases unmanaged and - optionally - managed resources
  86. /// </summary>
  87. /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged ResourceMessages.</param>
  88. protected virtual void Dispose(bool disposing)
  89. {
  90. // Check to see if Dispose has already been called.
  91. if (!this._isDisposed)
  92. {
  93. // If disposing equals true, dispose all managed
  94. // and unmanaged ResourceMessages.
  95. if (disposing)
  96. {
  97. // Dispose managed ResourceMessages.
  98. if (this._compressorStream != null)
  99. {
  100. this._compressorStream.Dispose();
  101. this._compressorStream = null;
  102. }
  103. if (this._decompressorStream != null)
  104. {
  105. this._decompressorStream.Dispose();
  106. this._decompressorStream = null;
  107. }
  108. }
  109. // Note disposing has been done.
  110. this._isDisposed = true;
  111. }
  112. }
  113. /// <summary>
  114. /// Releases unmanaged resources and performs other cleanup operations before the
  115. /// <see cref="SshCommand"/> is reclaimed by garbage collection.
  116. /// </summary>
  117. ~Compressor()
  118. {
  119. // Do not re-create Dispose clean-up code here.
  120. // Calling Dispose(false) is optimal in terms of
  121. // readability and maintainability.
  122. Dispose(false);
  123. }
  124. #endregion
  125. }
  126. }