#region License /* * HttpListenerPrefix.cs * * This code is derived from ListenerPrefix.cs (System.Net) of Mono * (http://www.mono-project.com). * * The MIT License * * Copyright (c) 2005 Novell, Inc. (http://www.novell.com) * Copyright (c) 2012-2020 sta.blockhead * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #endregion #region Authors /* * Authors: * - Gonzalo Paniagua Javier * - Oleg Mihailik */ #endregion using System; namespace WebSocketSharp.Net { internal sealed class HttpListenerPrefix { #region Private Fields private string _host; private HttpListener _listener; private string _original; private string _path; private string _port; private string _prefix; private bool _secure; #endregion #region Internal Constructors /// /// Initializes a new instance of the class /// with the specified URI prefix and HTTP listener. /// /// /// This constructor must be called after calling the CheckPrefix method. /// /// /// A that specifies the URI prefix. /// /// /// A that specifies the HTTP listener. /// internal HttpListenerPrefix (string uriPrefix, HttpListener listener) { _original = uriPrefix; _listener = listener; parse (uriPrefix); } #endregion #region Public Properties public string Host { get { return _host; } } public bool IsSecure { get { return _secure; } } public HttpListener Listener { get { return _listener; } } public string Original { get { return _original; } } public string Path { get { return _path; } } public string Port { get { return _port; } } #endregion #region Private Methods private void parse (string uriPrefix) { if (uriPrefix.StartsWith ("https")) _secure = true; var len = uriPrefix.Length; var host = uriPrefix.IndexOf (':') + 3; var root = uriPrefix.IndexOf ('/', host + 1, len - host - 1); var colon = uriPrefix.LastIndexOf (':', root - 1, root - host - 1); if (uriPrefix[root - 1] != ']' && colon > host) { _host = uriPrefix.Substring (host, colon - host); _port = uriPrefix.Substring (colon + 1, root - colon - 1); } else { _host = uriPrefix.Substring (host, root - host); _port = _secure ? "443" : "80"; } _path = uriPrefix.Substring (root); _prefix = String.Format ( "{0}://{1}:{2}{3}", _secure ? "https" : "http", _host, _port, _path ); } #endregion #region Public Methods public static void CheckPrefix (string uriPrefix) { if (uriPrefix == null) throw new ArgumentNullException ("uriPrefix"); var len = uriPrefix.Length; if (len == 0) { var msg = "An empty string."; throw new ArgumentException (msg, "uriPrefix"); } var schm = uriPrefix.StartsWith ("http://") || uriPrefix.StartsWith ("https://"); if (!schm) { var msg = "The scheme is not 'http' or 'https'."; throw new ArgumentException (msg, "uriPrefix"); } var end = len - 1; if (uriPrefix[end] != '/') { var msg = "It ends without '/'."; throw new ArgumentException (msg, "uriPrefix"); } var host = uriPrefix.IndexOf (':') + 3; if (host >= end) { var msg = "No host is specified."; throw new ArgumentException (msg, "uriPrefix"); } if (uriPrefix[host] == ':') { var msg = "No host is specified."; throw new ArgumentException (msg, "uriPrefix"); } var root = uriPrefix.IndexOf ('/', host, len - host); if (root == host) { var msg = "No host is specified."; throw new ArgumentException (msg, "uriPrefix"); } if (uriPrefix[root - 1] == ':') { var msg = "No port is specified."; throw new ArgumentException (msg, "uriPrefix"); } if (root == end - 1) { var msg = "No path is specified."; throw new ArgumentException (msg, "uriPrefix"); } } /// /// Determines whether the current instance is equal to the specified /// instance. /// /// /// This method will be required to detect duplicates in any collection. /// /// /// /// An instance to compare to the current instance. /// /// /// An reference to a instance. /// /// /// /// true if the current instance and have /// the same URI prefix; otherwise, false. /// public override bool Equals (object obj) { var pref = obj as HttpListenerPrefix; return pref != null && _prefix.Equals (pref._prefix); } /// /// Gets the hash code for the current instance. /// /// /// This method will be required to detect duplicates in any collection. /// /// /// An that represents the hash code. /// public override int GetHashCode () { return _prefix.GetHashCode (); } public override string ToString () { return _prefix; } #endregion } }