#region License /* * WebHeaderCollection.cs * * This code is derived from System.Net.WebHeaderCollection.cs of Mono * (http://www.mono-project.com). * * The MIT License * * Copyright (c) 2003 Ximian, Inc. (http://www.ximian.com) * Copyright (c) 2007 Novell, Inc. (http://www.novell.com) * Copyright (c) 2012-2014 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: * - Lawrence Pit * - Gonzalo Paniagua Javier * - Miguel de Icaza */ #endregion using System; using System.Collections; using System.Collections.Generic; using System.Collections.Specialized; using System.Net; using System.Runtime.InteropServices; using System.Runtime.Serialization; using System.Security.Permissions; using System.Text; namespace WebSocketSharp.Net { /// /// Provides a collection of the HTTP headers associated with a request or response. /// [Serializable] [ComVisible (true)] public class WebHeaderCollection : NameValueCollection, ISerializable { #region Private Fields private static readonly Dictionary _headers; private bool _internallyCreated; private HttpHeaderType _state; #endregion #region Static Constructor static WebHeaderCollection () { _headers = new Dictionary (StringComparer.InvariantCultureIgnoreCase) { { "Accept", new HttpHeaderInfo () { Name = "Accept", Type = HttpHeaderType.Request | HttpHeaderType.Restricted | HttpHeaderType.MultiValue } }, { "AcceptCharset", new HttpHeaderInfo () { Name = "Accept-Charset", Type = HttpHeaderType.Request | HttpHeaderType.MultiValue } }, { "AcceptEncoding", new HttpHeaderInfo () { Name = "Accept-Encoding", Type = HttpHeaderType.Request | HttpHeaderType.MultiValue } }, { "AcceptLanguage", new HttpHeaderInfo () { Name = "Accept-language", Type = HttpHeaderType.Request | HttpHeaderType.MultiValue } }, { "AcceptRanges", new HttpHeaderInfo () { Name = "Accept-Ranges", Type = HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Age", new HttpHeaderInfo () { Name = "Age", Type = HttpHeaderType.Response } }, { "Allow", new HttpHeaderInfo () { Name = "Allow", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Authorization", new HttpHeaderInfo () { Name = "Authorization", Type = HttpHeaderType.Request | HttpHeaderType.MultiValue } }, { "CacheControl", new HttpHeaderInfo () { Name = "Cache-Control", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Connection", new HttpHeaderInfo () { Name = "Connection", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted | HttpHeaderType.MultiValue } }, { "ContentEncoding", new HttpHeaderInfo () { Name = "Content-Encoding", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "ContentLanguage", new HttpHeaderInfo () { Name = "Content-Language", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "ContentLength", new HttpHeaderInfo () { Name = "Content-Length", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted } }, { "ContentLocation", new HttpHeaderInfo () { Name = "Content-Location", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "ContentMd5", new HttpHeaderInfo () { Name = "Content-MD5", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "ContentRange", new HttpHeaderInfo () { Name = "Content-Range", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "ContentType", new HttpHeaderInfo () { Name = "Content-Type", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted } }, { "Cookie", new HttpHeaderInfo () { Name = "Cookie", Type = HttpHeaderType.Request } }, { "Cookie2", new HttpHeaderInfo () { Name = "Cookie2", Type = HttpHeaderType.Request } }, { "Date", new HttpHeaderInfo () { Name = "Date", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted } }, { "Expect", new HttpHeaderInfo () { Name = "Expect", Type = HttpHeaderType.Request | HttpHeaderType.Restricted | HttpHeaderType.MultiValue } }, { "Expires", new HttpHeaderInfo () { Name = "Expires", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "ETag", new HttpHeaderInfo () { Name = "ETag", Type = HttpHeaderType.Response } }, { "From", new HttpHeaderInfo () { Name = "From", Type = HttpHeaderType.Request } }, { "Host", new HttpHeaderInfo () { Name = "Host", Type = HttpHeaderType.Request | HttpHeaderType.Restricted } }, { "IfMatch", new HttpHeaderInfo () { Name = "If-Match", Type = HttpHeaderType.Request | HttpHeaderType.MultiValue } }, { "IfModifiedSince", new HttpHeaderInfo () { Name = "If-Modified-Since", Type = HttpHeaderType.Request | HttpHeaderType.Restricted } }, { "IfNoneMatch", new HttpHeaderInfo () { Name = "If-None-Match", Type = HttpHeaderType.Request | HttpHeaderType.MultiValue } }, { "IfRange", new HttpHeaderInfo () { Name = "If-Range", Type = HttpHeaderType.Request } }, { "IfUnmodifiedSince", new HttpHeaderInfo () { Name = "If-Unmodified-Since", Type = HttpHeaderType.Request } }, { "KeepAlive", new HttpHeaderInfo () { Name = "Keep-Alive", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "LastModified", new HttpHeaderInfo () { Name = "Last-Modified", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "Location", new HttpHeaderInfo () { Name = "Location", Type = HttpHeaderType.Response } }, { "MaxForwards", new HttpHeaderInfo () { Name = "Max-Forwards", Type = HttpHeaderType.Request } }, { "Pragma", new HttpHeaderInfo () { Name = "Pragma", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "ProxyConnection", new HttpHeaderInfo () { Name = "Proxy-Connection", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted } }, { "ProxyAuthenticate", new HttpHeaderInfo () { Name = "Proxy-Authenticate", Type = HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "ProxyAuthorization", new HttpHeaderInfo () { Name = "Proxy-Authorization", Type = HttpHeaderType.Request } }, { "Public", new HttpHeaderInfo () { Name = "Public", Type = HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Range", new HttpHeaderInfo () { Name = "Range", Type = HttpHeaderType.Request | HttpHeaderType.Restricted | HttpHeaderType.MultiValue } }, { "Referer", new HttpHeaderInfo () { Name = "Referer", Type = HttpHeaderType.Request | HttpHeaderType.Restricted } }, { "RetryAfter", new HttpHeaderInfo () { Name = "Retry-After", Type = HttpHeaderType.Response } }, { "SecWebSocketAccept", new HttpHeaderInfo () { Name = "Sec-WebSocket-Accept", Type = HttpHeaderType.Response | HttpHeaderType.Restricted } }, { "SecWebSocketExtensions", new HttpHeaderInfo () { Name = "Sec-WebSocket-Extensions", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted | HttpHeaderType.MultiValueInRequest } }, { "SecWebSocketKey", new HttpHeaderInfo () { Name = "Sec-WebSocket-Key", Type = HttpHeaderType.Request | HttpHeaderType.Restricted } }, { "SecWebSocketProtocol", new HttpHeaderInfo () { Name = "Sec-WebSocket-Protocol", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValueInRequest } }, { "SecWebSocketVersion", new HttpHeaderInfo () { Name = "Sec-WebSocket-Version", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted | HttpHeaderType.MultiValueInResponse } }, { "Server", new HttpHeaderInfo () { Name = "Server", Type = HttpHeaderType.Response } }, { "SetCookie", new HttpHeaderInfo () { Name = "Set-Cookie", Type = HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "SetCookie2", new HttpHeaderInfo () { Name = "Set-Cookie2", Type = HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Te", new HttpHeaderInfo () { Name = "TE", Type = HttpHeaderType.Request } }, { "Trailer", new HttpHeaderInfo () { Name = "Trailer", Type = HttpHeaderType.Request | HttpHeaderType.Response } }, { "TransferEncoding", new HttpHeaderInfo () { Name = "Transfer-Encoding", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.Restricted | HttpHeaderType.MultiValue } }, { "Translate", new HttpHeaderInfo () { Name = "Translate", Type = HttpHeaderType.Request } }, { "Upgrade", new HttpHeaderInfo () { Name = "Upgrade", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "UserAgent", new HttpHeaderInfo () { Name = "User-Agent", Type = HttpHeaderType.Request | HttpHeaderType.Restricted } }, { "Vary", new HttpHeaderInfo () { Name = "Vary", Type = HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Via", new HttpHeaderInfo () { Name = "Via", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "Warning", new HttpHeaderInfo () { Name = "Warning", Type = HttpHeaderType.Request | HttpHeaderType.Response | HttpHeaderType.MultiValue } }, { "WwwAuthenticate", new HttpHeaderInfo () { Name = "WWW-Authenticate", Type = HttpHeaderType.Response | HttpHeaderType.Restricted | HttpHeaderType.MultiValue } } }; } #endregion #region Internal Constructors internal WebHeaderCollection (bool internallyCreated) { _internallyCreated = internallyCreated; _state = HttpHeaderType.Unspecified; } #endregion #region Protected Constructors /// /// Initializes a new instance of the class from /// the specified and . /// /// /// A that contains the serialized object data. /// /// /// A that specifies the source for the deserialization. /// /// /// is . /// /// /// An element with the specified name isn't found in . /// protected WebHeaderCollection ( SerializationInfo serializationInfo, StreamingContext streamingContext) { if (serializationInfo == null) throw new ArgumentNullException ("serializationInfo"); try { _internallyCreated = serializationInfo.GetBoolean ("InternallyCreated"); _state = (HttpHeaderType) serializationInfo.GetInt32 ("State"); var cnt = serializationInfo.GetInt32 ("Count"); for (var i = 0; i < cnt; i++) { base.Add ( serializationInfo.GetString (i.ToString ()), serializationInfo.GetString ((cnt + i).ToString ())); } } catch (SerializationException ex) { throw new ArgumentException (ex.Message, "serializationInfo", ex); } } #endregion #region Public Constructors /// /// Initializes a new instance of the class. /// public WebHeaderCollection () { _state = HttpHeaderType.Unspecified; } #endregion #region Public Properties /// /// Gets all header names in the collection. /// /// /// An array of that contains all header names in the collection. /// public override string[] AllKeys { get { return base.AllKeys; } } /// /// Gets the number of headers in the collection. /// /// /// An that represents the number of headers in the collection. /// public override int Count { get { return base.Count; } } /// /// Gets or sets the specified request in the collection. /// /// /// A that represents the value of the request . /// /// /// One of the enum values, represents the request header /// to get or set. /// /// /// /// is a restricted header. /// /// /// -or- /// /// /// contains invalid characters. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the request /// . /// public string this[HttpRequestHeader header] { get { return Get (Convert (header)); } set { Add (header, value); } } /// /// Gets or sets the specified response in the collection. /// /// /// A that represents the value of the response . /// /// /// One of the enum values, represents the response header /// to get or set. /// /// /// /// is a restricted header. /// /// /// -or- /// /// /// contains invalid characters. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the response /// . /// public string this[HttpResponseHeader header] { get { return Get (Convert (header)); } set { Add (header, value); } } /// /// Gets a collection of header names in the collection. /// /// /// A that contains all header names /// in the collection. /// public override NameObjectCollectionBase.KeysCollection Keys { get { return base.Keys; } } #endregion #region Private Methods private void add (string name, string value, bool ignoreRestricted) { var act = ignoreRestricted ? (Action ) addWithoutCheckingNameAndRestricted : addWithoutCheckingName; doWithCheckingState (act, checkName (name), value, true); } private void addWithoutCheckingName (string name, string value) { doWithoutCheckingName (base.Add, name, value); } private void addWithoutCheckingNameAndRestricted (string name, string value) { base.Add (name, checkValue (value)); } private static int checkColonSeparated (string header) { var i = header.IndexOf (':'); if (i == -1) throw new ArgumentException ("No colon could be found.", "header"); return i; } private static HttpHeaderType checkHeaderType (string name) { var info = getHeaderInfo (name); return info == null ? HttpHeaderType.Unspecified : info.IsRequest && !info.IsResponse ? HttpHeaderType.Request : !info.IsRequest && info.IsResponse ? HttpHeaderType.Response : HttpHeaderType.Unspecified; } private static string checkName (string name) { if (name == null || name.Length == 0) throw new ArgumentNullException ("name"); name = name.Trim (); if (!IsHeaderName (name)) throw new ArgumentException ("Contains invalid characters.", "name"); return name; } private void checkRestricted (string name) { if (!_internallyCreated && isRestricted (name, true)) throw new ArgumentException ("This header must be modified with the appropiate property."); } private void checkState (bool response) { if (_state == HttpHeaderType.Unspecified) return; if (response && _state == HttpHeaderType.Request) throw new InvalidOperationException ( "This collection has already been used to store the request headers."); if (!response && _state == HttpHeaderType.Response) throw new InvalidOperationException ( "This collection has already been used to store the response headers."); } private static string checkValue (string value) { if (value == null || value.Length == 0) return String.Empty; value = value.Trim (); if (value.Length > 65535) throw new ArgumentOutOfRangeException ("value", "Greater than 65,535 characters."); if (!IsHeaderValue (value)) throw new ArgumentException ("Contains invalid characters.", "value"); return value; } private static string convert (string key) { HttpHeaderInfo info; return _headers.TryGetValue (key, out info) ? info.Name : String.Empty; } private void doWithCheckingState ( Action action, string name, string value, bool setState) { var type = checkHeaderType (name); if (type == HttpHeaderType.Request) doWithCheckingState (action, name, value, false, setState); else if (type == HttpHeaderType.Response) doWithCheckingState (action, name, value, true, setState); else action (name, value); } private void doWithCheckingState ( Action action, string name, string value, bool response, bool setState) { checkState (response); action (name, value); if (setState && _state == HttpHeaderType.Unspecified) _state = response ? HttpHeaderType.Response : HttpHeaderType.Request; } private void doWithoutCheckingName (Action action, string name, string value) { checkRestricted (name); action (name, checkValue (value)); } private static HttpHeaderInfo getHeaderInfo (string name) { foreach (var info in _headers.Values) if (info.Name.Equals (name, StringComparison.InvariantCultureIgnoreCase)) return info; return null; } private static bool isRestricted (string name, bool response) { var info = getHeaderInfo (name); return info != null && info.IsRestricted (response); } private void removeWithoutCheckingName (string name, string unuse) { checkRestricted (name); base.Remove (name); } private void setWithoutCheckingName (string name, string value) { doWithoutCheckingName (base.Set, name, value); } #endregion #region Internal Methods internal static string Convert (HttpRequestHeader header) { return convert (header.ToString ()); } internal static string Convert (HttpResponseHeader header) { return convert (header.ToString ()); } internal void InternalRemove (string name) { base.Remove (name); } internal void InternalSet (string header, bool response) { var pos = checkColonSeparated (header); InternalSet (header.Substring (0, pos), header.Substring (pos + 1), response); } internal void InternalSet (string name, string value, bool response) { value = checkValue (value); if (IsMultiValue (name, response)) base.Add (name, value); else base.Set (name, value); } internal static bool IsHeaderName (string name) { return name != null && name.Length > 0 && name.IsToken (); } internal static bool IsHeaderValue (string value) { return value.IsText (); } internal static bool IsMultiValue (string headerName, bool response) { if (headerName == null || headerName.Length == 0) return false; var info = getHeaderInfo (headerName); return info != null && info.IsMultiValue (response); } internal string ToStringMultiValue (bool response) { var buff = new StringBuilder (); Count.Times ( i => { var key = GetKey (i); if (IsMultiValue (key, response)) foreach (var val in GetValues (i)) buff.AppendFormat ("{0}: {1}\r\n", key, val); else buff.AppendFormat ("{0}: {1}\r\n", key, Get (i)); }); return buff.Append ("\r\n").ToString (); } #endregion #region Protected Methods /// /// Adds a header to the collection without checking whether the header is on the restricted /// header list. /// /// /// A that represents the name of the header to add. /// /// /// A that represents the value of the header to add. /// /// /// is or empty. /// /// /// or contains invalid characters. /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow /// the . /// protected void AddWithoutValidate (string headerName, string headerValue) { add (headerName, headerValue, true); } #endregion #region Public Methods /// /// Adds the specified to the collection. /// /// /// A that represents the header with the name and value separated by /// a colon (':'). /// /// /// is , empty, or the name part of /// is empty. /// /// /// /// doesn't contain a colon. /// /// /// -or- /// /// /// is a restricted header. /// /// /// -or- /// /// /// The name or value part of contains invalid characters. /// /// /// /// The length of the value part of is greater than 65,535 characters. /// /// /// The current instance doesn't allow /// the . /// public void Add (string header) { if (header.IsNullOrEmpty ()) throw new ArgumentNullException ("header"); var pos = checkColonSeparated (header); add (header.Substring (0, pos), header.Substring (pos + 1), false); } /// /// Adds the specified request with the specified /// to the collection. /// /// /// One of the enum values, represents the request header /// to add. /// /// /// A that represents the value of the header to add. /// /// /// /// is a restricted header. /// /// /// -or- /// /// /// contains invalid characters. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the request /// . /// public void Add (HttpRequestHeader header, string value) { doWithCheckingState (addWithoutCheckingName, Convert (header), value, false, true); } /// /// Adds the specified response with the specified /// to the collection. /// /// /// One of the enum values, represents the response header /// to add. /// /// /// A that represents the value of the header to add. /// /// /// /// is a restricted header. /// /// /// -or- /// /// /// contains invalid characters. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the response /// . /// public void Add (HttpResponseHeader header, string value) { doWithCheckingState (addWithoutCheckingName, Convert (header), value, true, true); } /// /// Adds a header with the specified and /// to the collection. /// /// /// A that represents the name of the header to add. /// /// /// A that represents the value of the header to add. /// /// /// is or empty. /// /// /// /// or contains invalid characters. /// /// /// -or- /// /// /// is a restricted header name. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the header /// . /// public override void Add (string name, string value) { add (name, value, false); } /// /// Removes all headers from the collection. /// public override void Clear () { base.Clear (); _state = HttpHeaderType.Unspecified; } /// /// Get the value of the header at the specified in the collection. /// /// /// A that receives the value of the header. /// /// /// An that represents the zero-based index of the header to find. /// /// /// is out of allowable range of indexes for the collection. /// public override string Get (int index) { return base.Get (index); } /// /// Get the value of the header with the specified in the collection. /// /// /// A that receives the value of the header if found; otherwise, /// . /// /// /// A that represents the name of the header to find. /// public override string Get (string name) { return base.Get (name); } /// /// Gets the enumerator used to iterate through the collection. /// /// /// An instance used to iterate through the collection. /// public override IEnumerator GetEnumerator () { return base.GetEnumerator (); } /// /// Get the name of the header at the specified in the collection. /// /// /// A that receives the header name. /// /// /// An that represents the zero-based index of the header to find. /// /// /// is out of allowable range of indexes for the collection. /// public override string GetKey (int index) { return base.GetKey (index); } /// /// Gets an array of header values stored in the specified position /// of the collection. /// /// /// An array of that receives the header values if found; otherwise, /// . /// /// /// An that represents the zero-based index of the header to find. /// /// /// is out of allowable range of indexes for the collection. /// public override string[] GetValues (int index) { var vals = base.GetValues (index); return vals != null && vals.Length > 0 ? vals : null; } /// /// Gets an array of header values stored in the specified . /// /// /// An array of that receives the header values if found; otherwise, /// . /// /// /// A that represents the name of the header to find. /// public override string[] GetValues (string header) { var vals = base.GetValues (header); return vals != null && vals.Length > 0 ? vals : null; } /// /// Populates the specified with the data needed to serialize /// the . /// /// /// A that holds the serialized object data. /// /// /// A that specifies the destination for the serialization. /// /// /// is . /// [SecurityPermission ( SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.SerializationFormatter)] public override void GetObjectData ( SerializationInfo serializationInfo, StreamingContext streamingContext) { if (serializationInfo == null) throw new ArgumentNullException ("serializationInfo"); serializationInfo.AddValue ("InternallyCreated", _internallyCreated); serializationInfo.AddValue ("State", (int) _state); var cnt = Count; serializationInfo.AddValue ("Count", cnt); cnt.Times ( i => { serializationInfo.AddValue (i.ToString (), GetKey (i)); serializationInfo.AddValue ((cnt + i).ToString (), Get (i)); }); } /// /// Determines whether the specified header can be set for the request. /// /// /// true if the header is restricted; otherwise, false. /// /// /// A that represents the name of the header to test. /// /// /// is or empty. /// /// /// contains invalid characters. /// public static bool IsRestricted (string headerName) { return isRestricted (checkName (headerName), false); } /// /// Determines whether the specified header can be set for the request or the response. /// /// /// true if the header is restricted; otherwise, false. /// /// /// A that represents the name of the header to test. /// /// /// true if does the test for the response; for the request, false. /// /// /// is or empty. /// /// /// contains invalid characters. /// public static bool IsRestricted (string headerName, bool response) { return isRestricted (checkName (headerName), response); } /// /// Implements the interface and raises the deserialization event /// when the deserialization is complete. /// /// /// An that represents the source of the deserialization event. /// public override void OnDeserialization (object sender) { } /// /// Removes the specified request from the collection. /// /// /// One of the enum values, represents the request header /// to remove. /// /// /// is a restricted header. /// /// /// The current instance doesn't allow the request /// . /// public void Remove (HttpRequestHeader header) { doWithCheckingState (removeWithoutCheckingName, Convert (header), null, false, false); } /// /// Removes the specified response from the collection. /// /// /// One of the enum values, represents the response header /// to remove. /// /// /// is a restricted header. /// /// /// The current instance doesn't allow the response /// . /// public void Remove (HttpResponseHeader header) { doWithCheckingState (removeWithoutCheckingName, Convert (header), null, true, false); } /// /// Removes the specified header from the collection. /// /// /// A that represents the name of the header to remove. /// /// /// is or empty. /// /// /// /// contains invalid characters. /// /// /// -or- /// /// /// is a restricted header name. /// /// /// /// The current instance doesn't allow the header /// . /// public override void Remove (string name) { doWithCheckingState (removeWithoutCheckingName, checkName (name), null, false); } /// /// Sets the specified request to the specified value. /// /// /// One of the enum values, represents the request header /// to set. /// /// /// A that represents the value of the request header to set. /// /// /// /// is a restricted header. /// /// /// -or- /// /// /// contains invalid characters. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the request /// . /// public void Set (HttpRequestHeader header, string value) { doWithCheckingState (setWithoutCheckingName, Convert (header), value, false, true); } /// /// Sets the specified response to the specified value. /// /// /// One of the enum values, represents the response header /// to set. /// /// /// A that represents the value of the response header to set. /// /// /// /// is a restricted header. /// /// /// -or- /// /// /// contains invalid characters. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the response /// . /// public void Set (HttpResponseHeader header, string value) { doWithCheckingState (setWithoutCheckingName, Convert (header), value, true, true); } /// /// Sets the specified header to the specified value. /// /// /// A that represents the name of the header to set. /// /// /// A that represents the value of the header to set. /// /// /// is or empty. /// /// /// /// or contains invalid characters. /// /// /// -or- /// /// /// is a restricted header name. /// /// /// /// The length of is greater than 65,535 characters. /// /// /// The current instance doesn't allow the header /// . /// public override void Set (string name, string value) { doWithCheckingState (setWithoutCheckingName, checkName (name), value, true); } /// /// Converts the current to an array of . /// /// /// An array of that receives the converted current /// . /// public byte[] ToByteArray () { return Encoding.UTF8.GetBytes (ToString ()); } /// /// Returns a that represents the current /// . /// /// /// A that represents the current . /// public override string ToString () { var buff = new StringBuilder (); Count.Times (i => buff.AppendFormat ("{0}: {1}\r\n", GetKey (i), Get (i))); return buff.Append ("\r\n").ToString (); } #endregion #region Explicit Interface Implementations /// /// Populates the specified with the data needed to serialize /// the current . /// /// /// A that holds the serialized object data. /// /// /// A that specifies the destination for the serialization. /// /// /// is . /// [SecurityPermission ( SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.SerializationFormatter, SerializationFormatter = true)] void ISerializable.GetObjectData ( SerializationInfo serializationInfo, StreamingContext streamingContext) { GetObjectData (serializationInfo, streamingContext); } #endregion } }