ForwardedPortDynamic.NET.cs 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. using System;
  2. using System.Linq;
  3. using System.Text;
  4. using System.Net;
  5. using System.Net.Sockets;
  6. using System.Threading;
  7. using Renci.SshNet.Channels;
  8. using Renci.SshNet.Common;
  9. namespace Renci.SshNet
  10. {
  11. public partial class ForwardedPortDynamic
  12. {
  13. private TcpListener _listener;
  14. private readonly object _listenerLocker = new object();
  15. partial void InternalStart()
  16. {
  17. // If port already started don't start it again
  18. if (this.IsStarted)
  19. return;
  20. var ip = IPAddress.Any;
  21. if (!string.IsNullOrEmpty(this.BoundHost))
  22. {
  23. ip = this.BoundHost.GetIPAddress();
  24. }
  25. var ep = new IPEndPoint(ip, (int)this.BoundPort);
  26. this._listener = new TcpListener(ep);
  27. this._listener.Start();
  28. this._listenerTaskCompleted = new ManualResetEvent(false);
  29. this.ExecuteThread(() =>
  30. {
  31. try
  32. {
  33. while (true)
  34. {
  35. lock (this._listenerLocker)
  36. {
  37. if (this._listener == null)
  38. break;
  39. }
  40. var socket = this._listener.AcceptSocket();
  41. this.ExecuteThread(() =>
  42. {
  43. try
  44. {
  45. using (var channel = this.Session.CreateChannelDirectTcpip())
  46. {
  47. var version = new byte[1];
  48. socket.Receive(version);
  49. if (version[0] == 4)
  50. {
  51. this.HandleSocks4(socket, channel);
  52. }
  53. else if (version[0] == 5)
  54. {
  55. this.HandleSocks5(socket, channel);
  56. }
  57. else
  58. {
  59. throw new NotSupportedException(string.Format("SOCKS version {0} is not supported.", version));
  60. }
  61. channel.Bind();
  62. channel.Close();
  63. }
  64. }
  65. catch (Exception exp)
  66. {
  67. this.RaiseExceptionEvent(exp);
  68. }
  69. });
  70. }
  71. }
  72. catch (SocketException exp)
  73. {
  74. if (exp.SocketErrorCode != SocketError.Interrupted)
  75. {
  76. this.RaiseExceptionEvent(exp);
  77. }
  78. }
  79. catch (Exception exp)
  80. {
  81. this.RaiseExceptionEvent(exp);
  82. }
  83. finally
  84. {
  85. this._listenerTaskCompleted.Set();
  86. }
  87. });
  88. this.IsStarted = true;
  89. }
  90. partial void InternalStop()
  91. {
  92. // If port not started you cant stop it
  93. if (!this.IsStarted)
  94. return;
  95. lock (this._listenerLocker)
  96. {
  97. this._listener.Stop();
  98. this._listener = null;
  99. }
  100. this._listenerTaskCompleted.WaitOne(this.Session.ConnectionInfo.Timeout);
  101. this._listenerTaskCompleted.Dispose();
  102. this._listenerTaskCompleted = null;
  103. this.IsStarted = false;
  104. }
  105. private void HandleSocks4(Socket socket, IChannelDirectTcpip channel)
  106. {
  107. using (var stream = new NetworkStream(socket))
  108. {
  109. var commandCode = stream.ReadByte();
  110. // TODO: See what need to be done depends on the code
  111. var portBuffer = new byte[2];
  112. stream.Read(portBuffer, 0, portBuffer.Length);
  113. var port = (uint)(portBuffer[0] * 256 + portBuffer[1]);
  114. var ipBuffer = new byte[4];
  115. stream.Read(ipBuffer, 0, ipBuffer.Length);
  116. var ipAddress = new IPAddress(ipBuffer);
  117. var username = ReadString(stream);
  118. var host = ipAddress.ToString();
  119. this.RaiseRequestReceived(host, port);
  120. channel.Open(host, port, this, socket);
  121. stream.WriteByte(0x00);
  122. if (channel.IsOpen)
  123. {
  124. stream.WriteByte(0x5a);
  125. }
  126. else
  127. {
  128. stream.WriteByte(0x5b);
  129. }
  130. stream.Write(portBuffer, 0, portBuffer.Length);
  131. stream.Write(ipBuffer, 0, ipBuffer.Length);
  132. }
  133. }
  134. private void HandleSocks5(Socket socket, IChannelDirectTcpip channel)
  135. {
  136. using (var stream = new NetworkStream(socket))
  137. {
  138. var authenticationMethodsCount = stream.ReadByte();
  139. var authenticationMethods = new byte[authenticationMethodsCount];
  140. stream.Read(authenticationMethods, 0, authenticationMethods.Length);
  141. stream.WriteByte(0x05);
  142. if (authenticationMethods.Min() == 0)
  143. {
  144. stream.WriteByte(0x00);
  145. }
  146. else
  147. {
  148. stream.WriteByte(0xFF);
  149. }
  150. var version = stream.ReadByte();
  151. if (version != 5)
  152. throw new ProxyException("SOCKS5: Version 5 is expected.");
  153. var commandCode = stream.ReadByte();
  154. if (stream.ReadByte() != 0)
  155. {
  156. throw new ProxyException("SOCKS5: 0 is expected.");
  157. }
  158. var addressType = stream.ReadByte();
  159. IPAddress ipAddress;
  160. byte[] addressBuffer;
  161. switch (addressType)
  162. {
  163. case 0x01:
  164. {
  165. addressBuffer = new byte[4];
  166. stream.Read(addressBuffer, 0, 4);
  167. ipAddress = new IPAddress(addressBuffer);
  168. }
  169. break;
  170. case 0x03:
  171. {
  172. var length = stream.ReadByte();
  173. addressBuffer = new byte[length];
  174. stream.Read(addressBuffer, 0, addressBuffer.Length);
  175. ipAddress = IPAddress.Parse(new Common.ASCIIEncoding().GetString(addressBuffer));
  176. }
  177. break;
  178. case 0x04:
  179. {
  180. addressBuffer = new byte[16];
  181. stream.Read(addressBuffer, 0, 16);
  182. ipAddress = new IPAddress(addressBuffer);
  183. }
  184. break;
  185. default:
  186. throw new ProxyException(string.Format("SOCKS5: Address type '{0}' is not supported.", addressType));
  187. }
  188. var portBuffer = new byte[2];
  189. stream.Read(portBuffer, 0, portBuffer.Length);
  190. var port = (uint)(portBuffer[0] * 256 + portBuffer[1]);
  191. var host = ipAddress.ToString();
  192. this.RaiseRequestReceived(host, port);
  193. channel.Open(host, port, this, socket);
  194. stream.WriteByte(0x05);
  195. if (channel.IsOpen)
  196. {
  197. stream.WriteByte(0x00);
  198. }
  199. else
  200. {
  201. stream.WriteByte(0x01);
  202. }
  203. stream.WriteByte(0x00);
  204. var buffer = ipAddress.GetAddressBytes();
  205. if (ipAddress.AddressFamily == AddressFamily.InterNetwork)
  206. {
  207. stream.WriteByte(0x01);
  208. }
  209. else if (ipAddress.AddressFamily == AddressFamily.InterNetwork)
  210. {
  211. stream.WriteByte(0x04);
  212. }
  213. else
  214. {
  215. throw new NotSupportedException("Not supported address family.");
  216. }
  217. stream.Write(buffer, 0, buffer.Length);
  218. stream.Write(portBuffer, 0, portBuffer.Length);
  219. }
  220. }
  221. private static string ReadString(NetworkStream stream)
  222. {
  223. StringBuilder text = new StringBuilder();
  224. var aa = (char)stream.ReadByte();
  225. while (aa != 0)
  226. {
  227. text.Append(aa);
  228. aa = (char)stream.ReadByte();
  229. }
  230. return text.ToString();
  231. }
  232. }
  233. }