File: System\UriTemplateLiteralPathSegment.cs
Project: ndp\cdf\src\WCF\ServiceModel\System.ServiceModel.csproj (System.ServiceModel)
//----------------------------------------------------------------
// Copyright (c) Microsoft Corporation.  All rights reserved.
//----------------------------------------------------------------
 
namespace System
{
    using System.Collections.Specialized;
    using System.Runtime;
    using System.ServiceModel;
    using System.Text;
 
    // thin wrapper around string; use type system to help ensure we
    // are doing canonicalization right/consistently
    class UriTemplateLiteralPathSegment : UriTemplatePathSegment, IComparable<UriTemplateLiteralPathSegment>
    {
 
        // segment doesn't store trailing slash
        readonly string segment;
        static Uri dummyUri = new Uri("http://localhost");
 
        UriTemplateLiteralPathSegment(string segment)
            : base(segment, UriTemplatePartType.Literal, segment.EndsWith("/", StringComparison.Ordinal))
        {
            Fx.Assert(segment != null, "bad literal segment");
            if (this.EndsWithSlash)
            {
                this.segment = segment.Remove(segment.Length - 1);
            }
            else
            {
                this.segment = segment;
            }
        }
        public static new UriTemplateLiteralPathSegment CreateFromUriTemplate(string segment, UriTemplate template)
        {
            // run it through UriBuilder to escape-if-necessary it
            if (string.Compare(segment, "/", StringComparison.Ordinal) == 0)
            {
                // running an empty segment through UriBuilder has unexpected/wrong results
                return new UriTemplateLiteralPathSegment("/");
            }
            if (segment.IndexOf(UriTemplate.WildcardPath, StringComparison.Ordinal) != -1)
            {
                throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new FormatException(
                    SR.GetString(SR.UTInvalidWildcardInVariableOrLiteral, template.originalTemplate, UriTemplate.WildcardPath)));
            }
            // '*' is not usually escaped by the Uri\UriBuilder to %2a, since we forbid passing a
            // clear character and the workaroud is to pass the escaped form, we should replace the
            // escaped form with the regular one.
            segment = segment.Replace("%2a", "*").Replace("%2A", "*");
            UriBuilder ub = new UriBuilder(dummyUri);
            ub.Path = segment;
            string escapedIfNecessarySegment = ub.Uri.AbsolutePath.Substring(1);
            if (escapedIfNecessarySegment == string.Empty)
            {
                // This path through UriBuilder will sometimes '----' various segments
                // such as '../' and './'.  When this happens and the result is an empty
                // string, we should just throw and tell the user we don't handle that.
                throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("segment",
                    SR.GetString(SR.UTInvalidFormatSegmentOrQueryPart, segment));
            }
            return new UriTemplateLiteralPathSegment(escapedIfNecessarySegment);
        }
        public static UriTemplateLiteralPathSegment CreateFromWireData(string segment)
        {
            return new UriTemplateLiteralPathSegment(segment);
        }
 
        public string AsUnescapedString()
        {
            Fx.Assert(this.segment != null, "this should only be called by Bind\\Lookup");
            return Uri.UnescapeDataString(this.segment);
        }
        public override void Bind(string[] values, ref int valueIndex, StringBuilder path)
        {
            if (this.EndsWithSlash)
            {
                path.AppendFormat("{0}/", AsUnescapedString());
            }
            else
            {
                path.Append(AsUnescapedString());
            }
        }
 
        public int CompareTo(UriTemplateLiteralPathSegment other)
        {
            return StringComparer.OrdinalIgnoreCase.Compare(this.segment, other.segment);
        }
 
        public override bool Equals(object obj)
        {
            UriTemplateLiteralPathSegment lps = obj as UriTemplateLiteralPathSegment;
            if (lps == null)
            {
                Fx.Assert("why would we ever call this?");
                return false;
            }
            else
            {
                return ((this.EndsWithSlash == lps.EndsWithSlash) &&
                    StringComparer.OrdinalIgnoreCase.Equals(this.segment, lps.segment));
            }
        }
        public override int GetHashCode()
        {
            return StringComparer.OrdinalIgnoreCase.GetHashCode(this.segment);
        }
 
        public override bool IsEquivalentTo(UriTemplatePathSegment other, bool ignoreTrailingSlash)
        {
            if (other == null)
            {
                Fx.Assert("why would we ever call this?");
                return false;
            }
            if (other.Nature != UriTemplatePartType.Literal)
            {
                return false;
            }
            UriTemplateLiteralPathSegment otherAsLiteral = other as UriTemplateLiteralPathSegment;
            Fx.Assert(otherAsLiteral != null, "The nature requires that this will be OK");
            return IsMatch(otherAsLiteral, ignoreTrailingSlash);
        }
        public override bool IsMatch(UriTemplateLiteralPathSegment segment, bool ignoreTrailingSlash)
        {
            if (!ignoreTrailingSlash && (segment.EndsWithSlash != this.EndsWithSlash))
            {
                return false;
            }
            return (CompareTo(segment) == 0);
        }
        public bool IsNullOrEmpty()
        {
            return string.IsNullOrEmpty(this.segment);
        }
        public override void Lookup(string segment, NameValueCollection boundParameters)
        {
            Fx.Assert(StringComparer.OrdinalIgnoreCase.Compare(AsUnescapedString(), segment) == 0,
                "How can that be? Lookup is expected to be called after IsMatch");
        }
    }
}