// Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. using System; using System.ComponentModel; using Microsoft.AspNetCore.Testing; using Xunit; namespace Microsoft.AspNetCore.Http { public class PathStringTests { [Fact] public void CtorThrows_IfPathDoesNotHaveLeadingSlash() { // Act and Assert ExceptionAssert.ThrowsArgument(() => new PathString("hello"), "value", "The path in 'value' must start with '/'."); } [Fact] public void Equals_EmptyPathStringAndDefaultPathString() { // Act and Assert Assert.Equal(default(PathString), PathString.Empty); Assert.Equal(default(PathString), PathString.Empty); Assert.True(PathString.Empty == default(PathString)); Assert.True(default(PathString) == PathString.Empty); Assert.True(PathString.Empty.Equals(default(PathString))); Assert.True(default(PathString).Equals(PathString.Empty)); } [Fact] public void NotEquals_DefaultPathStringAndNonNullPathString() { // Arrange var pathString = new PathString("/hello"); // Act and Assert Assert.NotEqual(default(PathString), pathString); } [Fact] public void NotEquals_EmptyPathStringAndNonNullPathString() { // Arrange var pathString = new PathString("/hello"); // Act and Assert Assert.NotEqual(pathString, PathString.Empty); } [Fact] public void HashCode_CheckNullAndEmptyHaveSameHashcodes() { Assert.Equal(PathString.Empty.GetHashCode(), default(PathString).GetHashCode()); } [Theory] [InlineData(null, null)] [InlineData("", null)] public void AddPathString_HandlesNullAndEmptyStrings(string appString, string concatString) { // Arrange var appPath = new PathString(appString); var concatPath = new PathString(concatString); // Act var result = appPath.Add(concatPath); // Assert Assert.False(result.HasValue); } [Theory] [InlineData("", "/", "/")] [InlineData("/", null, "/")] [InlineData("/", "", "/")] [InlineData("/", "/test", "/test")] [InlineData("/myapp/", "/test/bar", "/myapp/test/bar")] [InlineData("/myapp/", "/test/bar/", "/myapp/test/bar/")] public void AddPathString_HandlesLeadingAndTrailingSlashes(string appString, string concatString, string expected) { // Arrange var appPath = new PathString(appString); var concatPath = new PathString(concatString); // Act var result = appPath.Add(concatPath); // Assert Assert.Equal(expected, result.Value); } [Fact] public void ImplicitStringConverters_WorksWithAdd() { var scheme = "http"; var host = new HostString("localhost:80"); var pathBase = new PathString("/base"); var path = new PathString("/path"); var query = new QueryString("?query"); var fragment = new FragmentString("#frag"); var result = scheme + "://" + host + pathBase + path + query + fragment; Assert.Equal("http://localhost:80/base/path?query#frag", result); result = pathBase + path + query + fragment; Assert.Equal("/base/path?query#frag", result); result = path + "text"; Assert.Equal("/pathtext", result); } [Theory] [InlineData("/test/path", "/TEST", true)] [InlineData("/test/path", "/TEST/pa", false)] [InlineData("/TEST/PATH", "/test", true)] [InlineData("/TEST/path", "/test/pa", false)] [InlineData("/test/PATH/path/TEST", "/TEST/path/PATH", true)] public void StartsWithSegments_DoesACaseInsensitiveMatch(string sourcePath, string testPath, bool expectedResult) { var source = new PathString(sourcePath); var test = new PathString(testPath); var result = source.StartsWithSegments(test); Assert.Equal(expectedResult, result); } [Theory] [InlineData("/test/path", "/TEST", true)] [InlineData("/test/path", "/TEST/pa", false)] [InlineData("/TEST/PATH", "/test", true)] [InlineData("/TEST/path", "/test/pa", false)] [InlineData("/test/PATH/path/TEST", "/TEST/path/PATH", true)] public void StartsWithSegmentsWithRemainder_DoesACaseInsensitiveMatch(string sourcePath, string testPath, bool expectedResult) { var source = new PathString(sourcePath); var test = new PathString(testPath); var result = source.StartsWithSegments(test, out var remaining); Assert.Equal(expectedResult, result); } [Theory] [InlineData("/test/path", "/TEST", StringComparison.OrdinalIgnoreCase, true)] [InlineData("/test/path", "/TEST", StringComparison.Ordinal, false)] [InlineData("/test/path", "/TEST/pa", StringComparison.OrdinalIgnoreCase, false)] [InlineData("/test/path", "/TEST/pa", StringComparison.Ordinal, false)] [InlineData("/TEST/PATH", "/test", StringComparison.OrdinalIgnoreCase, true)] [InlineData("/TEST/PATH", "/test", StringComparison.Ordinal, false)] [InlineData("/TEST/path", "/test/pa", StringComparison.OrdinalIgnoreCase, false)] [InlineData("/TEST/path", "/test/pa", StringComparison.Ordinal, false)] [InlineData("/test/PATH/path/TEST", "/TEST/path/PATH", StringComparison.OrdinalIgnoreCase, true)] [InlineData("/test/PATH/path/TEST", "/TEST/path/PATH", StringComparison.Ordinal, false)] public void StartsWithSegments_DoesMatchUsingSpecifiedComparison(string sourcePath, string testPath, StringComparison comparison, bool expectedResult) { var source = new PathString(sourcePath); var test = new PathString(testPath); var result = source.StartsWithSegments(test, comparison); Assert.Equal(expectedResult, result); } [Theory] [InlineData("/test/path", "/TEST", StringComparison.OrdinalIgnoreCase, true)] [InlineData("/test/path", "/TEST", StringComparison.Ordinal, false)] [InlineData("/test/path", "/TEST/pa", StringComparison.OrdinalIgnoreCase, false)] [InlineData("/test/path", "/TEST/pa", StringComparison.Ordinal, false)] [InlineData("/TEST/PATH", "/test", StringComparison.OrdinalIgnoreCase, true)] [InlineData("/TEST/PATH", "/test", StringComparison.Ordinal, false)] [InlineData("/TEST/path", "/test/pa", StringComparison.OrdinalIgnoreCase, false)] [InlineData("/TEST/path", "/test/pa", StringComparison.Ordinal, false)] [InlineData("/test/PATH/path/TEST", "/TEST/path/PATH", StringComparison.OrdinalIgnoreCase, true)] [InlineData("/test/PATH/path/TEST", "/TEST/path/PATH", StringComparison.Ordinal, false)] public void StartsWithSegmentsWithRemainder_DoesMatchUsingSpecifiedComparison(string sourcePath, string testPath, StringComparison comparison, bool expectedResult) { var source = new PathString(sourcePath); var test = new PathString(testPath); var result = source.StartsWithSegments(test, comparison, out var remaining); Assert.Equal(expectedResult, result); } [Theory] // unreserved [InlineData("/abc123.-_~", "/abc123.-_~")] // colon [InlineData("/:", "/:")] // at [InlineData("/@", "/@")] // sub-delims [InlineData("/!$&'()*+,;=", "/!$&'()*+,;=")] // reserved [InlineData("/?#[]", "/%3F%23%5B%5D")] // pct-encoding [InlineData("/单行道", "/%E5%8D%95%E8%A1%8C%E9%81%93")] // mixed [InlineData("/index/单行道=(x*y)[abc]", "/index/%E5%8D%95%E8%A1%8C%E9%81%93=(x*y)%5Babc%5D")] [InlineData("/index/单行道=(x*y)[abc]_", "/index/%E5%8D%95%E8%A1%8C%E9%81%93=(x*y)%5Babc%5D_")] // encoded [InlineData("/http%3a%2f%2f[foo]%3A5000/", "/http%3a%2f%2f%5Bfoo%5D%3A5000/")] [InlineData("/http%3a%2f%2f[foo]%3A5000/%", "/http%3a%2f%2f%5Bfoo%5D%3A5000/%25")] [InlineData("/http%3a%2f%2f[foo]%3A5000/%2", "/http%3a%2f%2f%5Bfoo%5D%3A5000/%252")] [InlineData("/http%3a%2f%2f[foo]%3A5000/%2F", "/http%3a%2f%2f%5Bfoo%5D%3A5000/%2F")] public void ToUriComponentEscapeCorrectly(string input, string expected) { var path = new PathString(input); Assert.Equal(expected, path.ToUriComponent()); } [Fact] public void PathStringConvertsOnlyToAndFromString() { var converter = TypeDescriptor.GetConverter(typeof(PathString)); PathString result = (PathString)converter.ConvertFromInvariantString("/foo"); Assert.Equal("/foo", result.ToString()); Assert.Equal("/foo", converter.ConvertTo(result, typeof(string))); Assert.True(converter.CanConvertFrom(typeof(string))); Assert.False(converter.CanConvertFrom(typeof(int))); Assert.False(converter.CanConvertFrom(typeof(bool))); Assert.True(converter.CanConvertTo(typeof(string))); Assert.False(converter.CanConvertTo(typeof(int))); Assert.False(converter.CanConvertTo(typeof(bool))); } [Fact] public void PathStringStaysEqualAfterAssignments() { PathString p1 = "/?"; string s1 = p1; PathString p2 = s1; Assert.Equal(p1, p2); } } }