Message.cs 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. using System.IO;
  2. using System.Linq;
  3. using Renci.SshNet.Common;
  4. using System.Globalization;
  5. using Renci.SshNet.Abstractions;
  6. using Renci.SshNet.Compression;
  7. using Renci.SshNet.Security.Cryptography;
  8. namespace Renci.SshNet.Messages
  9. {
  10. /// <summary>
  11. /// Base class for all SSH protocol messages
  12. /// </summary>
  13. public abstract class Message : SshData
  14. {
  15. /// <summary>
  16. /// Gets the index that represents zero in current data type.
  17. /// </summary>
  18. /// <value>
  19. /// The index of the zero reader.
  20. /// </value>
  21. protected override int ZeroReaderIndex
  22. {
  23. get
  24. {
  25. return 1;
  26. }
  27. }
  28. #if TUNING
  29. /// <summary>
  30. /// Gets the size of the message in bytes.
  31. /// </summary>
  32. /// <value>
  33. /// The size of the messages in bytes.
  34. /// </value>
  35. protected override int BufferCapacity
  36. {
  37. get
  38. {
  39. return 1; // Message type
  40. }
  41. }
  42. /// <summary>
  43. /// Writes the message to the specified <see cref="SshDataStream"/>.
  44. /// </summary>
  45. protected override void WriteBytes(SshDataStream stream)
  46. {
  47. var messageAttribute = GetType().GetCustomAttributes<MessageAttribute>(true).FirstOrDefault();
  48. if (messageAttribute == null)
  49. throw new SshException(string.Format(CultureInfo.CurrentCulture, "Type '{0}' is not a valid message type.", GetType().AssemblyQualifiedName));
  50. stream.WriteByte(messageAttribute.Number);
  51. base.WriteBytes(stream);
  52. }
  53. internal byte[] GetPacket(byte paddingMultiplier, Compressor compressor)
  54. {
  55. const int outboundPacketSequenceSize = 4;
  56. var messageLength = BufferCapacity;
  57. SshDataStream sshDataStream;
  58. if (messageLength == -1 || compressor != null)
  59. {
  60. sshDataStream = new SshDataStream(DefaultCapacity);
  61. // skip:
  62. // * 4 bytes for the outbound packet sequence
  63. // * 4 bytes for the packet data length
  64. // * one byte for the packet padding length
  65. sshDataStream.Seek(outboundPacketSequenceSize + 4 + 1, SeekOrigin.Begin);
  66. if (compressor != null)
  67. {
  68. // obtain uncompressed message payload
  69. var uncompressedDataStream = new SshDataStream(messageLength != -1 ? messageLength : DefaultCapacity);
  70. WriteBytes(uncompressedDataStream);
  71. // compress message payload
  72. var compressedMessageData = compressor.Compress(uncompressedDataStream.ToArray());
  73. // add compressed message payload
  74. sshDataStream.Write(compressedMessageData, 0, compressedMessageData.Length);
  75. }
  76. else
  77. {
  78. // add message payload
  79. WriteBytes(sshDataStream);
  80. }
  81. messageLength = (int) sshDataStream.Length - (outboundPacketSequenceSize + 4 + 1);
  82. var packetLength = messageLength + 4 + 1;
  83. // determine the padding length
  84. var paddingLength = GetPaddingLength(paddingMultiplier, packetLength);
  85. // add padding bytes
  86. var paddingBytes = new byte[paddingLength];
  87. HashAlgorithmFactory.GenerateRandom(paddingBytes);
  88. sshDataStream.Write(paddingBytes, 0, paddingLength);
  89. var packetDataLength = GetPacketDataLength(messageLength, paddingLength);
  90. // skip bytes for outbound packet sequence
  91. sshDataStream.Seek(outboundPacketSequenceSize, SeekOrigin.Begin);
  92. // add packet data length
  93. sshDataStream.Write(packetDataLength.GetBytes(), 0, 4);
  94. // add packet padding length
  95. sshDataStream.WriteByte(paddingLength);
  96. }
  97. else
  98. {
  99. var packetLength = messageLength + 4 + 1;
  100. // determine the padding length
  101. var paddingLength = GetPaddingLength(paddingMultiplier, packetLength);
  102. var packetDataLength = GetPacketDataLength(messageLength, paddingLength);
  103. // lets construct an SSH data stream of the exact size required
  104. sshDataStream = new SshDataStream(packetLength + paddingLength + outboundPacketSequenceSize);
  105. // skip bytes for outbound packet sequenceSize
  106. sshDataStream.Seek(outboundPacketSequenceSize, SeekOrigin.Begin);
  107. // add packet data length
  108. sshDataStream.Write(packetDataLength.GetBytes(), 0, 4);
  109. // add packet padding length
  110. sshDataStream.WriteByte(paddingLength);
  111. // add message payload
  112. WriteBytes(sshDataStream);
  113. // add padding bytes
  114. var paddingBytes = new byte[paddingLength];
  115. HashAlgorithmFactory.GenerateRandom(paddingBytes);
  116. sshDataStream.Write(paddingBytes, 0, paddingLength);
  117. }
  118. return sshDataStream.ToArray();
  119. }
  120. private static uint GetPacketDataLength(int messageLength, byte paddingLength)
  121. {
  122. return (uint) (messageLength + paddingLength + 1);
  123. }
  124. private static byte GetPaddingLength(byte paddingMultiplier, long packetLength)
  125. {
  126. var paddingLength = (byte)((-packetLength) & (paddingMultiplier - 1));
  127. if (paddingLength < paddingMultiplier)
  128. {
  129. paddingLength += paddingMultiplier;
  130. }
  131. return paddingLength;
  132. }
  133. #else
  134. /// <summary>
  135. /// Gets data bytes array
  136. /// </summary>
  137. /// <returns>Byte array representation of the message</returns>
  138. public override byte[] GetBytes()
  139. {
  140. var messageAttribute = GetType().GetTypeInfo().GetCustomAttributes(typeof(MessageAttribute), true).SingleOrDefault() as MessageAttribute;
  141. if (messageAttribute == null)
  142. throw new SshException(string.Format(CultureInfo.CurrentCulture, "Type '{0}' is not a valid message type.", GetType().AssemblyQualifiedName));
  143. var data = new List<byte>(base.GetBytes());
  144. data.Insert(0, messageAttribute.Number);
  145. return data.ToArray();
  146. }
  147. #endif
  148. /// <summary>
  149. /// Returns a <see cref="System.String"/> that represents this instance.
  150. /// </summary>
  151. /// <returns>
  152. /// A <see cref="System.String"/> that represents this instance.
  153. /// </returns>
  154. public override string ToString()
  155. {
  156. var messageAttribute = GetType().GetCustomAttributes<MessageAttribute>(true).SingleOrDefault();
  157. if (messageAttribute == null)
  158. return string.Format(CultureInfo.CurrentCulture, "'{0}' without Message attribute.", GetType().FullName);
  159. return messageAttribute.Name;
  160. }
  161. }
  162. }