PollServiceHttpRequest.cs 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. /*
  2. * Copyright (c) Contributors, http://opensimulator.org/
  3. * See CONTRIBUTORS.TXT for a full list of copyright holders.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are met:
  7. * * Redistributions of source code must retain the above copyright
  8. * notice, this list of conditions and the following disclaimer.
  9. * * Redistributions in binary form must reproduce the above copyright
  10. * notice, this list of conditions and the following disclaimer in the
  11. * documentation and/or other materials provided with the distribution.
  12. * * Neither the name of the OpenSimulator Project nor the
  13. * names of its contributors may be used to endorse or promote products
  14. * derived from this software without specific prior written permission.
  15. *
  16. * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
  17. * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  18. * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  19. * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
  20. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  21. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  22. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  23. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  24. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  25. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. */
  27. using System;
  28. using System.Collections;
  29. using System.Collections.Generic;
  30. using System.Reflection;
  31. using System.Text;
  32. using HttpServer;
  33. using log4net;
  34. using OpenMetaverse;
  35. namespace OpenSim.Framework.Servers.HttpServer
  36. {
  37. public class PollServiceHttpRequest
  38. {
  39. private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
  40. public readonly PollServiceEventArgs PollServiceArgs;
  41. public readonly IHttpClientContext HttpContext;
  42. public readonly IHttpRequest Request;
  43. public readonly int RequestTime;
  44. public readonly UUID RequestID;
  45. public int contextHash;
  46. /*
  47. private void GenContextHash()
  48. {
  49. Random rnd = new Random();
  50. contextHash = 0;
  51. if (Request.Headers["remote_addr"] != null)
  52. contextHash = (Request.Headers["remote_addr"]).GetHashCode() << 16;
  53. else
  54. contextHash = rnd.Next() << 16;
  55. if (Request.Headers["remote_port"] != null)
  56. {
  57. string[] strPorts = Request.Headers["remote_port"].Split(new char[] { ',' });
  58. contextHash += Int32.Parse(strPorts[0]);
  59. }
  60. else
  61. contextHash += rnd.Next() & 0xffff;
  62. }
  63. */
  64. public PollServiceHttpRequest(
  65. PollServiceEventArgs pPollServiceArgs, IHttpClientContext pHttpContext, IHttpRequest pRequest)
  66. {
  67. PollServiceArgs = pPollServiceArgs;
  68. HttpContext = pHttpContext;
  69. Request = pRequest;
  70. RequestTime = System.Environment.TickCount;
  71. RequestID = UUID.Random();
  72. // GenContextHash();
  73. contextHash = HttpContext.contextID;
  74. }
  75. internal void DoHTTPGruntWork(Hashtable responsedata)
  76. {
  77. OSHttpResponse response
  78. = new OSHttpResponse(new HttpResponse(HttpContext, Request), HttpContext);
  79. byte[] buffer = srvDoHTTPGruntWork(responsedata, response);
  80. if(Request.Body.CanRead)
  81. Request.Body.Dispose();
  82. response.SendChunked = false;
  83. response.ContentLength64 = buffer.Length;
  84. response.ContentEncoding = Encoding.UTF8;
  85. try
  86. {
  87. response.OutputStream.Write(buffer, 0, buffer.Length);
  88. response.Send();
  89. buffer = null;
  90. }
  91. catch (Exception ex)
  92. {
  93. if(ex is System.Net.Sockets.SocketException)
  94. {
  95. // only mute connection reset by peer so we are not totally blind for now
  96. if(((System.Net.Sockets.SocketException)ex).SocketErrorCode != System.Net.Sockets.SocketError.ConnectionReset)
  97. m_log.Warn("[POLL SERVICE WORKER THREAD]: Error ", ex);
  98. }
  99. else
  100. m_log.Warn("[POLL SERVICE WORKER THREAD]: Error ", ex);
  101. }
  102. PollServiceArgs.RequestsHandled++;
  103. }
  104. internal byte[] srvDoHTTPGruntWork(Hashtable responsedata, OSHttpResponse response)
  105. {
  106. int responsecode;
  107. string responseString = String.Empty;
  108. byte[] responseData = null;
  109. string contentType;
  110. if (responsedata == null)
  111. {
  112. responsecode = 500;
  113. responseString = "No response could be obtained";
  114. contentType = "text/plain";
  115. responsedata = new Hashtable();
  116. }
  117. else
  118. {
  119. try
  120. {
  121. //m_log.Info("[BASE HTTP SERVER]: Doing HTTP Grunt work with response");
  122. responsecode = (int)responsedata["int_response_code"];
  123. if (responsedata["bin_response_data"] != null)
  124. responseData = (byte[])responsedata["bin_response_data"];
  125. else
  126. responseString = (string)responsedata["str_response_string"];
  127. contentType = (string)responsedata["content_type"];
  128. if (responseString == null)
  129. responseString = String.Empty;
  130. }
  131. catch
  132. {
  133. responsecode = 500;
  134. responseString = "No response could be obtained";
  135. contentType = "text/plain";
  136. responsedata = new Hashtable();
  137. }
  138. }
  139. if (responsedata.ContainsKey("error_status_text"))
  140. {
  141. response.StatusDescription = (string)responsedata["error_status_text"];
  142. }
  143. if (responsedata.ContainsKey("http_protocol_version"))
  144. {
  145. response.ProtocolVersion = (string)responsedata["http_protocol_version"];
  146. }
  147. if (responsedata.ContainsKey("keepalive"))
  148. {
  149. bool keepalive = (bool)responsedata["keepalive"];
  150. response.KeepAlive = keepalive;
  151. }
  152. // Cross-Origin Resource Sharing with simple requests
  153. if (responsedata.ContainsKey("access_control_allow_origin"))
  154. response.AddHeader("Access-Control-Allow-Origin", (string)responsedata["access_control_allow_origin"]);
  155. //Even though only one other part of the entire code uses HTTPHandlers, we shouldn't expect this
  156. //and should check for NullReferenceExceptions
  157. if (string.IsNullOrEmpty(contentType))
  158. {
  159. contentType = "text/html";
  160. }
  161. // The client ignores anything but 200 here for web login, so ensure that this is 200 for that
  162. response.StatusCode = responsecode;
  163. if (responsecode == (int)OSHttpStatusCode.RedirectMovedPermanently)
  164. {
  165. response.RedirectLocation = (string)responsedata["str_redirect_location"];
  166. response.StatusCode = responsecode;
  167. }
  168. response.AddHeader("Content-Type", contentType);
  169. if (responsedata.ContainsKey("headers"))
  170. {
  171. Hashtable headerdata = (Hashtable)responsedata["headers"];
  172. foreach (string header in headerdata.Keys)
  173. response.AddHeader(header, headerdata[header].ToString());
  174. }
  175. byte[] buffer;
  176. if (responseData != null)
  177. {
  178. buffer = responseData;
  179. }
  180. else
  181. {
  182. if (!(contentType.Contains("image")
  183. || contentType.Contains("x-shockwave-flash")
  184. || contentType.Contains("application/x-oar")
  185. || contentType.Contains("application/vnd.ll.mesh")))
  186. {
  187. // Text
  188. buffer = Encoding.UTF8.GetBytes(responseString);
  189. }
  190. else
  191. {
  192. // Binary!
  193. buffer = Convert.FromBase64String(responseString);
  194. }
  195. response.SendChunked = false;
  196. response.ContentLength64 = buffer.Length;
  197. response.ContentEncoding = Encoding.UTF8;
  198. }
  199. return buffer;
  200. }
  201. internal void DoHTTPstop()
  202. {
  203. OSHttpResponse response
  204. = new OSHttpResponse(new HttpResponse(HttpContext, Request), HttpContext);
  205. if(Request.Body.CanRead)
  206. Request.Body.Dispose();
  207. response.ContentLength64 = 0;
  208. response.ContentEncoding = Encoding.UTF8;
  209. response.KeepAlive = false;
  210. response.SendChunked = false;
  211. response.StatusCode = 503;
  212. try
  213. {
  214. response.Send();
  215. }
  216. catch
  217. {
  218. }
  219. }
  220. }
  221. }