123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643 |
- /*
- * Copyright (c) Contributors, http://opensimulator.org/
- * See CONTRIBUTORS.TXT for a full list of copyright holders.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are met:
- * * Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * * Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * * Neither the name of the OpenSimulator Project nor the
- * names of its contributors may be used to endorse or promote products
- * derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
- * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
- * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
- * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
- * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
- * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- using System;
- using System.Threading;
- using System.Collections.Generic;
- using System.Collections;
- using System.Reflection;
- using log4net;
- using Nini.Config;
- using OpenMetaverse;
- using OpenSim.Framework;
- using OpenSim.Framework.Servers;
- using OpenSim.Framework.Servers.HttpServer;
- using OpenSim.Region.Framework.Interfaces;
- using OpenSim.Region.Framework.Scenes;
- namespace OpenSim.Region.CoreModules.Scripting.LSLHttp
- {
- /// <summary>
- /// Data describing an external URL set up by a script.
- /// </summary>
- public class UrlData
- {
- /// <summary>
- /// Scene object part hosting the script
- /// </summary>
- public UUID hostID;
- /// <summary>
- /// The item ID of the script that requested the URL.
- /// </summary>
- public UUID itemID;
- /// <summary>
- /// The script engine that runs the script.
- /// </summary>
- public IScriptModule engine;
- /// <summary>
- /// The generated URL.
- /// </summary>
- public string url;
- /// <summary>
- /// The random UUID component of the generated URL.
- /// </summary>
- public UUID urlcode;
- /// <summary>
- /// The external requests currently being processed or awaiting retrieval for this URL.
- /// </summary>
- public Dictionary<UUID, RequestData> requests;
- }
- public class RequestData
- {
- public UUID requestID;
- public Dictionary<string, string> headers;
- public string body;
- public int responseCode;
- public string responseBody;
- public string responseType = "text/plain";
- //public ManualResetEvent ev;
- public bool requestDone;
- public int startTime;
- public string uri;
- }
- /// <summary>
- /// This module provides external URLs for in-world scripts.
- /// </summary>
- public class UrlModule : ISharedRegionModule, IUrlModule
- {
- private static readonly ILog m_log =
- LogManager.GetLogger(
- MethodBase.GetCurrentMethod().DeclaringType);
- /// <summary>
- /// Indexs the URL request metadata (which script requested it, outstanding requests, etc.) by the request ID
- /// randomly generated when a request is received for this URL.
- /// </summary>
- /// <remarks>
- /// Manipulation or retrieval from this dictionary must be locked on m_UrlMap to preserve consistency with
- /// m_UrlMap
- /// </remarks>
- private Dictionary<UUID, UrlData> m_RequestMap = new Dictionary<UUID, UrlData>();
- /// <summary>
- /// Indexs the URL request metadata (which script requested it, outstanding requests, etc.) by the full URL
- /// </summary>
- private Dictionary<string, UrlData> m_UrlMap = new Dictionary<string, UrlData>();
- /// <summary>
- /// Maximum number of external urls that can be set up by this module.
- /// </summary>
- private int m_TotalUrls = 100;
- private uint https_port = 0;
- private IHttpServer m_HttpServer = null;
- private IHttpServer m_HttpsServer = null;
- private string m_ExternalHostNameForLSL = "";
- public string ExternalHostNameForLSL
- {
- get { return m_ExternalHostNameForLSL; }
- }
- public Type ReplaceableInterface
- {
- get { return null; }
- }
- public string Name
- {
- get { return "UrlModule"; }
- }
- public void Initialise(IConfigSource config)
- {
- m_ExternalHostNameForLSL = config.Configs["Network"].GetString("ExternalHostNameForLSL", System.Environment.MachineName);
- bool ssl_enabled = config.Configs["Network"].GetBoolean("https_listener",false);
- if (ssl_enabled)
- https_port = (uint) config.Configs["Network"].GetInt("https_port",0);
- IConfig llFunctionsConfig = config.Configs["LL-Functions"];
- if (llFunctionsConfig != null)
- m_TotalUrls = llFunctionsConfig.GetInt("max_external_urls_per_simulator", m_TotalUrls);
- }
- public void PostInitialise()
- {
- }
- public void AddRegion(Scene scene)
- {
- if (m_HttpServer == null)
- {
- // There can only be one
- //
- m_HttpServer = MainServer.Instance;
- //
- // We can use the https if it is enabled
- if (https_port > 0)
- {
- m_HttpsServer = MainServer.GetHttpServer(https_port);
- }
- }
- scene.RegisterModuleInterface<IUrlModule>(this);
- scene.EventManager.OnScriptReset += OnScriptReset;
- }
- public void RegionLoaded(Scene scene)
- {
- IScriptModule[] scriptModules = scene.RequestModuleInterfaces<IScriptModule>();
- foreach (IScriptModule scriptModule in scriptModules)
- {
- scriptModule.OnScriptRemoved += ScriptRemoved;
- scriptModule.OnObjectRemoved += ObjectRemoved;
- }
- }
- public void RemoveRegion(Scene scene)
- {
- }
- public void Close()
- {
- }
- public UUID RequestURL(IScriptModule engine, SceneObjectPart host, UUID itemID)
- {
- UUID urlcode = UUID.Random();
- lock (m_UrlMap)
- {
- if (m_UrlMap.Count >= m_TotalUrls)
- {
- engine.PostScriptEvent(itemID, "http_request", new Object[] { urlcode.ToString(), "URL_REQUEST_DENIED", "" });
- return urlcode;
- }
- string url = "http://" + m_ExternalHostNameForLSL + ":" + m_HttpServer.Port.ToString() + "/lslhttp/" + urlcode.ToString() + "/";
- UrlData urlData = new UrlData();
- urlData.hostID = host.UUID;
- urlData.itemID = itemID;
- urlData.engine = engine;
- urlData.url = url;
- urlData.urlcode = urlcode;
- urlData.requests = new Dictionary<UUID, RequestData>();
-
- m_UrlMap[url] = urlData;
-
- string uri = "/lslhttp/" + urlcode.ToString() + "/";
-
- m_HttpServer.AddPollServiceHTTPHandler(
- uri,
- new PollServiceEventArgs(HttpRequestHandler, HasEvents, GetEvents, NoEvents, urlcode));
- m_log.DebugFormat(
- "[URL MODULE]: Set up incoming request url {0} for {1} in {2} {3}",
- uri, itemID, host.Name, host.LocalId);
- engine.PostScriptEvent(itemID, "http_request", new Object[] { urlcode.ToString(), "URL_REQUEST_GRANTED", url });
- }
- return urlcode;
- }
- public UUID RequestSecureURL(IScriptModule engine, SceneObjectPart host, UUID itemID)
- {
- UUID urlcode = UUID.Random();
- if (m_HttpsServer == null)
- {
- engine.PostScriptEvent(itemID, "http_request", new Object[] { urlcode.ToString(), "URL_REQUEST_DENIED", "" });
- return urlcode;
- }
- lock (m_UrlMap)
- {
- if (m_UrlMap.Count >= m_TotalUrls)
- {
- engine.PostScriptEvent(itemID, "http_request", new Object[] { urlcode.ToString(), "URL_REQUEST_DENIED", "" });
- return urlcode;
- }
- string url = "https://" + m_ExternalHostNameForLSL + ":" + m_HttpsServer.Port.ToString() + "/lslhttps/" + urlcode.ToString() + "/";
- UrlData urlData = new UrlData();
- urlData.hostID = host.UUID;
- urlData.itemID = itemID;
- urlData.engine = engine;
- urlData.url = url;
- urlData.urlcode = urlcode;
- urlData.requests = new Dictionary<UUID, RequestData>();
- m_UrlMap[url] = urlData;
-
- string uri = "/lslhttps/" + urlcode.ToString() + "/";
-
- m_HttpsServer.AddPollServiceHTTPHandler(
- uri,
- new PollServiceEventArgs(HttpRequestHandler, HasEvents, GetEvents, NoEvents, urlcode));
- m_log.DebugFormat(
- "[URL MODULE]: Set up incoming secure request url {0} for {1} in {2} {3}",
- uri, itemID, host.Name, host.LocalId);
- engine.PostScriptEvent(itemID, "http_request", new Object[] { urlcode.ToString(), "URL_REQUEST_GRANTED", url });
- }
- return urlcode;
- }
- public void ReleaseURL(string url)
- {
- lock (m_UrlMap)
- {
- UrlData data;
- if (!m_UrlMap.TryGetValue(url, out data))
- {
- return;
- }
- foreach (UUID req in data.requests.Keys)
- m_RequestMap.Remove(req);
- m_log.DebugFormat(
- "[URL MODULE]: Releasing url {0} for {1} in {2}",
- url, data.itemID, data.hostID);
- RemoveUrl(data);
- m_UrlMap.Remove(url);
- }
- }
-
- public void HttpContentType(UUID request, string type)
- {
- lock (m_UrlMap)
- {
- if (m_RequestMap.ContainsKey(request))
- {
- UrlData urlData = m_RequestMap[request];
- urlData.requests[request].responseType = type;
- }
- else
- {
- m_log.Info("[HttpRequestHandler] There is no http-in request with id " + request.ToString());
- }
- }
- }
-
- public void HttpResponse(UUID request, int status, string body)
- {
- lock (m_UrlMap)
- {
- if (m_RequestMap.ContainsKey(request))
- {
- UrlData urlData = m_RequestMap[request];
- urlData.requests[request].responseCode = status;
- urlData.requests[request].responseBody = body;
- //urlData.requests[request].ev.Set();
- urlData.requests[request].requestDone =true;
- }
- else
- {
- m_log.Info("[HttpRequestHandler] There is no http-in request with id " + request.ToString());
- }
- }
- }
- public string GetHttpHeader(UUID requestId, string header)
- {
- lock (m_UrlMap)
- {
- if (m_RequestMap.ContainsKey(requestId))
- {
- UrlData urlData = m_RequestMap[requestId];
- string value;
- if (urlData.requests[requestId].headers.TryGetValue(header, out value))
- return value;
- }
- else
- {
- m_log.Warn("[HttpRequestHandler] There was no http-in request with id " + requestId);
- }
- }
- return String.Empty;
- }
- public int GetFreeUrls()
- {
- lock (m_UrlMap)
- return m_TotalUrls - m_UrlMap.Count;
- }
- public void ScriptRemoved(UUID itemID)
- {
- // m_log.DebugFormat("[URL MODULE]: Removing script {0}", itemID);
-
- lock (m_UrlMap)
- {
- List<string> removeURLs = new List<string>();
- foreach (KeyValuePair<string, UrlData> url in m_UrlMap)
- {
- if (url.Value.itemID == itemID)
- {
- RemoveUrl(url.Value);
- removeURLs.Add(url.Key);
- foreach (UUID req in url.Value.requests.Keys)
- m_RequestMap.Remove(req);
- }
- }
- foreach (string urlname in removeURLs)
- m_UrlMap.Remove(urlname);
- }
- }
- public void ObjectRemoved(UUID objectID)
- {
- lock (m_UrlMap)
- {
- List<string> removeURLs = new List<string>();
- foreach (KeyValuePair<string, UrlData> url in m_UrlMap)
- {
- if (url.Value.hostID == objectID)
- {
- RemoveUrl(url.Value);
- removeURLs.Add(url.Key);
- foreach (UUID req in url.Value.requests.Keys)
- m_RequestMap.Remove(req);
- }
- }
- foreach (string urlname in removeURLs)
- m_UrlMap.Remove(urlname);
- }
- }
- private void RemoveUrl(UrlData data)
- {
- m_HttpServer.RemoveHTTPHandler("", "/lslhttp/" + data.urlcode.ToString() + "/");
- }
- private Hashtable NoEvents(UUID requestID, UUID sessionID)
- {
- Hashtable response = new Hashtable();
- UrlData urlData;
- lock (m_UrlMap)
- {
- // We need to return a 404 here in case the request URL was removed at exactly the same time that a
- // request was made. In this case, the request thread can outrace llRemoveURL() and still be polling
- // for the request ID.
- if (!m_RequestMap.ContainsKey(requestID))
- {
- response["int_response_code"] = 404;
- response["str_response_string"] = "";
- response["keepalive"] = false;
- response["reusecontext"] = false;
- return response;
- }
- urlData = m_RequestMap[requestID];
- if (System.Environment.TickCount - urlData.requests[requestID].startTime > 25000)
- {
- response["int_response_code"] = 500;
- response["str_response_string"] = "Script timeout";
- response["content_type"] = "text/plain";
- response["keepalive"] = false;
- response["reusecontext"] = false;
-
- //remove from map
- urlData.requests.Remove(requestID);
- m_RequestMap.Remove(requestID);
- return response;
- }
- }
- return response;
- }
- private bool HasEvents(UUID requestID, UUID sessionID)
- {
- lock (m_UrlMap)
- {
- // We return true here because an external URL request that happened at the same time as an llRemoveURL()
- // can still make it through to HttpRequestHandler(). That will return without setting up a request
- // when it detects that the URL has been removed. The poller, however, will continue to ask for
- // events for that request, so here we will signal that there are events and in GetEvents we will
- // return a 404.
- if (!m_RequestMap.ContainsKey(requestID))
- {
- return true;
- }
- UrlData urlData = m_RequestMap[requestID];
- if (!urlData.requests.ContainsKey(requestID))
- {
- return true;
- }
- // Trigger return of timeout response.
- if (System.Environment.TickCount - urlData.requests[requestID].startTime > 25000)
- {
- return true;
- }
- return urlData.requests[requestID].requestDone;
- }
- }
- private Hashtable GetEvents(UUID requestID, UUID sessionID, string request)
- {
- Hashtable response;
- lock (m_UrlMap)
- {
- UrlData url = null;
- RequestData requestData = null;
- if (!m_RequestMap.ContainsKey(requestID))
- return NoEvents(requestID, sessionID);
- url = m_RequestMap[requestID];
- requestData = url.requests[requestID];
- if (!requestData.requestDone)
- return NoEvents(requestID, sessionID);
- response = new Hashtable();
- if (System.Environment.TickCount - requestData.startTime > 25000)
- {
- response["int_response_code"] = 500;
- response["str_response_string"] = "Script timeout";
- response["content_type"] = "text/plain";
- response["keepalive"] = false;
- response["reusecontext"] = false;
- return response;
- }
- //put response
- response["int_response_code"] = requestData.responseCode;
- response["str_response_string"] = requestData.responseBody;
- response["content_type"] = requestData.responseType;
- // response["content_type"] = "text/plain";
- response["keepalive"] = false;
- response["reusecontext"] = false;
- //remove from map
- url.requests.Remove(requestID);
- m_RequestMap.Remove(requestID);
- }
- return response;
- }
- public void HttpRequestHandler(UUID requestID, Hashtable request)
- {
- string uri = request["uri"].ToString();
- bool is_ssl = uri.Contains("lslhttps");
- try
- {
- Hashtable headers = (Hashtable)request["headers"];
- // string uri_full = "http://" + m_ExternalHostNameForLSL + ":" + m_HttpServer.Port.ToString() + uri;// "/lslhttp/" + urlcode.ToString() + "/";
- int pos1 = uri.IndexOf("/");// /lslhttp
- int pos2 = uri.IndexOf("/", pos1 + 1);// /lslhttp/
- int pos3 = uri.IndexOf("/", pos2 + 1);// /lslhttp/<UUID>/
- string uri_tmp = uri.Substring(0, pos3 + 1);
- //HTTP server code doesn't provide us with QueryStrings
- string pathInfo;
- string queryString;
- queryString = "";
- pathInfo = uri.Substring(pos3);
- UrlData urlData = null;
- lock (m_UrlMap)
- {
- string url;
- if (is_ssl)
- url = "https://" + m_ExternalHostNameForLSL + ":" + m_HttpsServer.Port.ToString() + uri_tmp;
- else
- url = "http://" + m_ExternalHostNameForLSL + ":" + m_HttpServer.Port.ToString() + uri_tmp;
- // Avoid a race - the request URL may have been released via llRequestUrl() whilst this
- // request was being processed.
- if (!m_UrlMap.TryGetValue(url, out urlData))
- return;
- //for llGetHttpHeader support we need to store original URI here
- //to make x-path-info / x-query-string / x-script-url / x-remote-ip headers
- //as per http://wiki.secondlife.com/wiki/LlGetHTTPHeader
- RequestData requestData = new RequestData();
- requestData.requestID = requestID;
- requestData.requestDone = false;
- requestData.startTime = System.Environment.TickCount;
- requestData.uri = uri;
- if (requestData.headers == null)
- requestData.headers = new Dictionary<string, string>();
- foreach (DictionaryEntry header in headers)
- {
- string key = (string)header.Key;
- string value = (string)header.Value;
- requestData.headers.Add(key, value);
- }
- foreach (DictionaryEntry de in request)
- {
- if (de.Key.ToString() == "querystringkeys")
- {
- System.String[] keys = (System.String[])de.Value;
- foreach (String key in keys)
- {
- if (request.ContainsKey(key))
- {
- string val = (String)request[key];
- queryString = queryString + key + "=" + val + "&";
- }
- }
- if (queryString.Length > 1)
- queryString = queryString.Substring(0, queryString.Length - 1);
- }
- }
- //if this machine is behind DNAT/port forwarding, currently this is being
- //set to address of port forwarding router
- requestData.headers["x-remote-ip"] = requestData.headers["remote_addr"];
- requestData.headers["x-path-info"] = pathInfo;
- requestData.headers["x-query-string"] = queryString;
- requestData.headers["x-script-url"] = urlData.url;
- urlData.requests.Add(requestID, requestData);
- m_RequestMap.Add(requestID, urlData);
- }
- urlData.engine.PostScriptEvent(
- urlData.itemID,
- "http_request",
- new Object[] { requestID.ToString(), request["http-method"].ToString(), request["body"].ToString() });
- }
- catch (Exception we)
- {
- //Hashtable response = new Hashtable();
- m_log.Warn("[HttpRequestHandler]: http-in request failed");
- m_log.Warn(we.Message);
- m_log.Warn(we.StackTrace);
- }
- }
- private void OnScriptReset(uint localID, UUID itemID)
- {
- ScriptRemoved(itemID);
- }
- }
- }
|