Skip to content

Commit

Permalink
fix: InvalidInputFile error occurs if file contains URI escaped chara…
Browse files Browse the repository at this point in the history
…cters
  • Loading branch information
filzrev committed Feb 15, 2024
1 parent 6dde3b7 commit 9deb63b
Show file tree
Hide file tree
Showing 2 changed files with 60 additions and 5 deletions.
10 changes: 5 additions & 5 deletions src/Docfx.Common/Path/PathUtility.cs
Original file line number Diff line number Diff line change
Expand Up @@ -69,14 +69,14 @@ public static string MakeRelativePath(string basePath, string absolutePath)
return absolutePath;
}

Uri relativeUri = fromUri.MakeRelativeUri(toUri);
string relativePath = Uri.UnescapeDataString(relativeUri.ToString());

if (string.Equals(toUri.Scheme, "FILE", StringComparison.InvariantCultureIgnoreCase))
if (toUri.IsFile && !toUri.OriginalString.StartsWith("file://", StringComparison.InvariantCultureIgnoreCase))
{
relativePath = relativePath.Replace(Path.AltDirectorySeparatorChar, Path.DirectorySeparatorChar);
return Path.GetRelativePath(basePath, absolutePath).BackSlashToForwardSlash();
}

Uri relativeUri = fromUri.MakeRelativeUri(toUri);
string relativePath = Uri.UnescapeDataString(relativeUri.ToString());

return relativePath.BackSlashToForwardSlash();
}

Expand Down
55 changes: 55 additions & 0 deletions test/Docfx.Common.Tests/PathUtilityTest.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using FluentAssertions;
using Xunit;

namespace Docfx.Common.Tests;

public class PathUtilityTest
{
[Theory]
[MemberData(nameof(TestData.AdditionalTests), MemberType = typeof(TestData), DisableDiscoveryEnumeration = false)]
public void TestMakeRelativePath(string basePath, string targetPath, string expected)
{
// Act
var result = PathUtility.MakeRelativePath(basePath, targetPath);

// Assert
result.Should().Be(expected);
}

[Theory]
[MemberData(nameof(TestData.EscapedPaths), MemberType = typeof(TestData), DisableDiscoveryEnumeration = true)]
public void TestMakeRelativePathWithEncodedPath(string inputPath)
{
// Arrange
string basePath = "./";
var expected = inputPath;

// Act
var result = PathUtility.MakeRelativePath(basePath, inputPath);

// Assert
result.Should().Be(expected);
}

private static class TestData
{
public static TheoryData<string, string, string> AdditionalTests = new()
{
{ @"/a/b/d", @"/a/b/file.md", @"../file.md"}, // root relative path
{ @"~/a/b/d", @"~/a/b/file.md", @"../file.md"}, // user home directory relative path
{ @"./", @"\\UNCPath\file.md", @"//UNCPath/file.md"}, // UNC path
{ @"./", @"file:///C:/temp/test.md", @"file:/C:/temp/test.md"}, // `file:` Uri path
{ @"file:///C:/temp", @"file:///C:/temp/test.md", @"test.md"}, // `file:` Uri relative path
{ @"/temp/dir", @"/temp/dir/subdir/", @"subdir/"}, // If target path endsWith directory separator char. resolved path should contain directory separator.
};

public static TheoryData<string> EscapedPaths = new()
{
"EscapedHypen(%2D).md", // Contains escaped hypen char
"EscapedSpace(%20)_with_NonAsciiChar(α).md", // Contains escaped space char and non-unicode char
};
}
}

0 comments on commit 9deb63b

Please sign in to comment.