Compressor.cs 4.8 KB

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