144 lines
5.8 KiB
C#
144 lines
5.8 KiB
C#
|
//----------------------------------------------------------------
|
||
|
// 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");
|
||
|
}
|
||
|
}
|
||
|
}
|