// ReSharper disable InconsistentNaming using System; using LibHac.Common; using LibHac.Fs; using LibHac.Util; using Xunit; namespace LibHac.Tests.Fs { public class PathFormatterTests { public static TheoryData<string, string, string, Result> TestData_Normalize_EmptyPath => new() { { @"", "", @"", ResultFs.InvalidPathFormat.Value }, { @"", "E", @"", Result.Success }, { @"/aa/bb/../cc", "E", @"/aa/cc", Result.Success } }; [Theory, MemberData(nameof(TestData_Normalize_EmptyPath))] public static void Normalize_EmptyPath(string path, string pathFlags, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, 0x301, expectedNormalized, expectedResult); } public static TheoryData<string, string, string, Result> TestData_Normalize_MountName => new() { { @"mount:/aa/bb", "", @"", ResultFs.InvalidPathFormat.Value }, { @"mount:/aa/bb", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"mount:/aa/bb", "M", @"mount:/aa/bb", Result.Success }, { @"mount:/aa/./bb", "M", @"mount:/aa/bb", Result.Success }, { @"mount:\aa\bb", "M", @"mount:", ResultFs.InvalidPathFormat.Value }, { @"m:/aa/bb", "M", @"", ResultFs.InvalidPathFormat.Value }, { @"mo>unt:/aa/bb", "M", @"", ResultFs.InvalidCharacter.Value }, { @"moun?t:/aa/bb", "M", @"", ResultFs.InvalidCharacter.Value }, { @"mo&unt:/aa/bb", "M", @"mo&unt:/aa/bb", Result.Success }, { @"/aa/./bb", "M", @"/aa/bb", Result.Success }, { @"mount/aa/./bb", "M", @"", ResultFs.InvalidPathFormat.Value } }; [Theory, MemberData(nameof(TestData_Normalize_MountName))] public static void Normalize_MountName(string path, string pathFlags, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, 0x301, expectedNormalized, expectedResult); } public static TheoryData<string, string, string, Result> TestData_Normalize_WindowsPath => new() { { @"c:/aa/bb", "", @"", ResultFs.InvalidPathFormat.Value }, { @"c:\aa\bb", "", @"", ResultFs.InvalidCharacter.Value }, { @"\\host\share", "", @"", ResultFs.InvalidCharacter.Value }, { @"\\.\c:\", "", @"", ResultFs.InvalidCharacter.Value }, { @"\\.\c:/aa/bb/.", "", @"", ResultFs.InvalidCharacter.Value }, { @"\\?\c:\", "", @"", ResultFs.InvalidCharacter.Value }, { @"mount:\\host\share\aa\bb", "M", @"mount:", ResultFs.InvalidCharacter.Value }, { @"mount:\\host/share\aa\bb", "M", @"mount:", ResultFs.InvalidCharacter.Value }, { @"mount:/\\aa\..\bb", "MW", @"mount:", ResultFs.InvalidPathFormat.Value }, { @"mount:/c:\aa\..\bb", "MW", @"mount:c:/bb", Result.Success }, { @"mount:/aa/bb", "MW", @"mount:/aa/bb", Result.Success }, { @"/mount:/aa/bb", "MW", @"/mount:/aa/bb", ResultFs.InvalidCharacter.Value }, { @"/mount:/aa/bb", "W", @"/mount:/aa/bb", ResultFs.InvalidCharacter.Value }, { @"a:aa/../bb", "MW", @"a:aa/bb", Result.Success }, { @"a:aa\..\bb", "MW", @"a:aa/bb", Result.Success }, { @"/a:aa\..\bb", "W", @"/bb", Result.Success }, { @"\\?\c:\.\aa", "W", @"\\?\c:/aa", Result.Success }, { @"\\.\c:\.\aa", "W", @"\\.\c:/aa", Result.Success }, { @"\\.\mount:\.\aa", "W", @"\\./mount:/aa", ResultFs.InvalidCharacter.Value }, { @"\\./.\aa", "W", @"\\./aa", Result.Success }, { @"\\/aa", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\\aa", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\", "W", @"/", Result.Success }, { @"\\host\share", "W", @"\\host\share/", Result.Success }, { @"\\host\share\path", "W", @"\\host\share/path", Result.Success }, { @"\\host\share\path\aa\bb\..\cc\.", "W", @"\\host\share/path/aa/cc", Result.Success }, { @"\\host\", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\ho$st\share\path", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\host:\share\path", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\..\share\path", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\host\s:hare\path", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\host\.\path", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\host\..\path", "W", @"", ResultFs.InvalidPathFormat.Value }, { @"\\host\sha:re", "W", @"", ResultFs.InvalidPathFormat.Value }, { @".\\host\share", "RW", @"..\\host\share/", Result.Success } }; [Theory, MemberData(nameof(TestData_Normalize_WindowsPath))] public static void Normalize_WindowsPath(string path, string pathFlags, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, 0x301, expectedNormalized, expectedResult); } public static TheoryData<string, string, string, Result> TestData_Normalize_RelativePath => new() { { @"./aa/bb", "", @"", ResultFs.InvalidPathFormat.Value }, { @"./aa/bb/../cc", "R", @"./aa/cc", Result.Success }, { @".\aa/bb/../cc", "R", @"..", ResultFs.InvalidCharacter.Value }, { @".", "R", @".", Result.Success }, { @"../aa/bb", "R", @"", ResultFs.DirectoryUnobtainable.Value }, { @"/aa/./bb", "R", @"/aa/bb", Result.Success }, { @"mount:./aa/bb", "MR", @"mount:./aa/bb", Result.Success }, { @"mount:./aa/./bb", "MR", @"mount:./aa/bb", Result.Success }, { @"mount:./aa/bb", "M", @"mount:", ResultFs.InvalidPathFormat.Value } }; [Theory, MemberData(nameof(TestData_Normalize_RelativePath))] public static void Normalize_RelativePath(string path, string pathFlags, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, 0x301, expectedNormalized, expectedResult); } public static TheoryData<string, string, string, Result> TestData_Normalize_Backslash => new() { { @"\aa\bb\..\cc", "", @"", ResultFs.InvalidPathFormat.Value }, { @"\aa\bb\..\cc", "B", @"", ResultFs.InvalidPathFormat.Value }, { @"/aa\bb\..\cc", "", @"", ResultFs.InvalidCharacter.Value }, { @"/aa\bb\..\cc", "B", @"/cc", Result.Success }, { @"/aa\bb\cc", "", @"", ResultFs.InvalidCharacter.Value }, { @"/aa\bb\cc", "B", @"/aa\bb\cc", Result.Success }, { @"\\host\share\path\aa\bb\cc", "W", @"\\host\share/path/aa/bb/cc", Result.Success }, { @"\\host\share\path\aa\bb\cc", "WB", @"\\host\share/path/aa/bb/cc", Result.Success }, { @"/aa/bb\../cc/..\dd\..\ee/..", "", @"", ResultFs.InvalidCharacter.Value }, { @"/aa/bb\../cc/..\dd\..\ee/..", "B", @"/aa", Result.Success } }; [Theory, MemberData(nameof(TestData_Normalize_Backslash))] public static void Normalize_Backslash(string path, string pathFlags, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, 0x301, expectedNormalized, expectedResult); } public static TheoryData<string, string, string, Result> TestData_Normalize_All => new() { { @"mount:./aa/bb", "WRM", @"mount:./aa/bb", Result.Success }, { @"mount:./aa/bb\cc/dd", "WRM", @"mount:./aa/bb/cc/dd", Result.Success }, { @"mount:./aa/bb\cc/dd", "WRMB", @"mount:./aa/bb/cc/dd", Result.Success }, { @"mount:./.c:/aa/bb", "RM", @"mount:./.c:/aa/bb", ResultFs.InvalidCharacter.Value }, { @"mount:.c:/aa/bb", "WRM", @"mount:./.c:/aa/bb", ResultFs.InvalidCharacter.Value }, { @"mount:./cc:/aa/bb", "WRM", @"mount:./cc:/aa/bb", ResultFs.InvalidCharacter.Value }, { @"mount:./\\host\share/aa/bb", "MW", @"mount:", ResultFs.InvalidPathFormat.Value }, { @"mount:./\\host\share/aa/bb", "WRM", @"mount:.\\host\share/aa/bb", Result.Success }, { @"mount:.\\host\share/aa/bb", "WRM", @"mount:..\\host\share/aa/bb", Result.Success }, { @"mount:..\\host\share/aa/bb", "WRM", @"mount:.", ResultFs.DirectoryUnobtainable.Value }, { @".\\host\share/aa/bb", "WRM", @"..\\host\share/aa/bb", Result.Success }, { @"..\\host\share/aa/bb", "WRM", @".", ResultFs.DirectoryUnobtainable.Value }, { @"mount:\\host\share/aa/bb", "MW", @"mount:\\host\share/aa/bb", Result.Success }, { @"mount:\aa\bb", "BM", @"mount:", ResultFs.InvalidPathFormat.Value }, { @"mount:/aa\bb", "BM", @"mount:/aa\bb", Result.Success }, { @".//aa/bb", "RW", @"./aa/bb", Result.Success }, { @"./aa/bb", "R", @"./aa/bb", Result.Success }, { @"./c:/aa/bb", "RW", @"./c:/aa/bb", ResultFs.InvalidCharacter.Value } }; [Theory, MemberData(nameof(TestData_Normalize_All))] public static void Normalize_All(string path, string pathFlags, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, 0x301, expectedNormalized, expectedResult); } public static TheoryData<string, string, int, string, Result> TestData_Normalize_SmallBuffer => new() { { @"/aa/bb", "M", 1, @"", ResultFs.TooLongPath.Value }, { @"mount:/aa/bb", "MR", 6, @"", ResultFs.TooLongPath.Value }, { @"mount:/aa/bb", "MR", 7, @"mount:", ResultFs.TooLongPath.Value }, { @"aa/bb", "MR", 3, @"./", ResultFs.TooLongPath.Value }, { @"\\host\share", "W", 13, @"\\host\share", ResultFs.TooLongPath.Value } }; [Theory, MemberData(nameof(TestData_Normalize_SmallBuffer))] public static void Normalize_SmallBuffer(string path, string pathFlags, int bufferSize, string expectedNormalized, Result expectedResult) { NormalizeImpl(path, pathFlags, bufferSize, expectedNormalized, expectedResult); } private static void NormalizeImpl(string path, string pathFlags, int bufferSize, string expectedNormalized, Result expectedResult) { byte[] buffer = new byte[bufferSize]; Result result = PathFormatter.Normalize(buffer, path.ToU8Span(), GetPathFlags(pathFlags)); Assert.Equal(expectedResult, result); Assert.Equal(expectedNormalized, StringUtils.Utf8ZToString(buffer)); } public static TheoryData<string, string, bool, int, Result> TestData_IsNormalized_EmptyPath => new() { { @"", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"", "E", true, 0, Result.Success }, { @"/aa/bb/../cc", "E", false, 0, Result.Success } }; [Theory, MemberData(nameof(TestData_IsNormalized_EmptyPath))] public static void IsNormalized_EmptyPath(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { IsNormalizedImpl(path, pathFlags, expectedIsNormalized, expectedLength, expectedResult); } public static TheoryData<string, string, bool, int, Result> TestData_IsNormalized_MountName => new() { { @"mount:/aa/bb", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:/aa/bb", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:/aa/bb", "M", true, 12, Result.Success }, { @"mount:/aa/./bb", "M", false, 6, Result.Success }, { @"mount:\aa\bb", "M", false, 0, ResultFs.InvalidPathFormat.Value }, { @"m:/aa/bb", "M", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mo>unt:/aa/bb", "M", false, 0, ResultFs.InvalidCharacter.Value }, { @"moun?t:/aa/bb", "M", false, 0, ResultFs.InvalidCharacter.Value }, { @"mo&unt:/aa/bb", "M", true, 13, Result.Success }, { @"/aa/./bb", "M", false, 0, Result.Success }, { @"mount/aa/./bb", "M", false, 0, ResultFs.InvalidPathFormat.Value } }; [Theory, MemberData(nameof(TestData_IsNormalized_MountName))] public static void IsNormalized_MountName(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { IsNormalizedImpl(path, pathFlags, expectedIsNormalized, expectedLength, expectedResult); } public static TheoryData<string, string, bool, int, Result> TestData_IsNormalized_WindowsPath => new() { { @"c:/aa/bb", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"c:/aa/bb", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"c:\aa\bb", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\host\share", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\.\c:\", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\.\c:/aa/bb/.", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\?\c:\", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:\\host\share\aa\bb", "M", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:\\host/share\aa\bb", "M", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:/\\aa\..\bb", "MW", false, 0, Result.Success }, { @"mount:/c:\aa\..\bb", "MW", false, 0, Result.Success }, { @"mount:/aa/bb", "MW", true, 12, Result.Success }, { @"/mount:/aa/bb", "MW", false, 0, ResultFs.InvalidCharacter.Value }, { @"/mount:/aa/bb", "W", false, 0, ResultFs.InvalidCharacter.Value }, { @"a:aa/../bb", "MW", false, 8, Result.Success }, { @"a:aa\..\bb", "MW", false, 0, Result.Success }, { @"/a:aa\..\bb", "W", false, 0, ResultFs.DirectoryUnobtainable.Value }, { @"\\?\c:\.\aa", "W", false, 0, Result.Success }, { @"\\.\c:\.\aa", "W", false, 0, Result.Success }, { @"\\.\mount:\.\aa", "W", false, 0, Result.Success }, { @"\\./.\aa", "W", false, 0, Result.Success }, { @"\\/aa", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\\aa", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\", "W", false, 0, Result.Success }, { @"\\host\share", "W", false, 0, Result.Success }, { @"\\host\share\path", "W", false, 0, Result.Success }, { @"\\host\share\path\aa\bb\..\cc\.", "W", false, 0, Result.Success }, { @"\\host\", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\ho$st\share\path", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\host:\share\path", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\..\share\path", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\host\s:hare\path", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\host\.\path", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\host\..\path", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\\host\sha:re", "W", false, 0, ResultFs.InvalidPathFormat.Value }, { @".\\host\share", "RW", false, 0, Result.Success } }; [Theory, MemberData(nameof(TestData_IsNormalized_WindowsPath))] public static void IsNormalized_WindowsPath(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { IsNormalizedImpl(path, pathFlags, expectedIsNormalized, expectedLength, expectedResult); } public static TheoryData<string, string, bool, int, Result> TestData_IsNormalized_RelativePath => new() { { @"./aa/bb", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"./aa/bb/../cc", "R", false, 1, Result.Success }, { @".\aa/bb/../cc", "R", false, 0, Result.Success }, { @".", "R", true, 1, Result.Success }, { @"../aa/bb", "R", false, 0, ResultFs.DirectoryUnobtainable.Value }, { @"/aa/./bb", "R", false, 0, Result.Success }, { @"mount:./aa/bb", "MR", true, 13, Result.Success }, { @"mount:./aa/./bb", "MR", false, 7, Result.Success }, { @"mount:./aa/bb", "M", false, 0, ResultFs.InvalidPathFormat.Value } }; [Theory, MemberData(nameof(TestData_IsNormalized_RelativePath))] public static void IsNormalized_RelativePath(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { IsNormalizedImpl(path, pathFlags, expectedIsNormalized, expectedLength, expectedResult); } public static TheoryData<string, string, bool, int, Result> TestData_IsNormalized_Backslash => new() { { @"\aa\bb\..\cc", "", false, 0, ResultFs.InvalidPathFormat.Value }, { @"\aa\bb\..\cc", "B", false, 0, ResultFs.InvalidPathFormat.Value }, { @"/aa\bb\..\cc", "", false, 0, ResultFs.DirectoryUnobtainable.Value }, { @"/aa\bb\..\cc", "B", false, 0, ResultFs.DirectoryUnobtainable.Value }, { @"/aa\bb\cc", "", false, 0, ResultFs.InvalidCharacter.Value }, { @"/aa\bb\cc", "B", true, 9, Result.Success }, { @"\\host\share\path\aa\bb\cc", "W", false, 0, Result.Success }, { @"\\host\share\path\aa\bb\cc", "WB", false, 0, Result.Success }, { @"/aa/bb\../cc/..\dd\..\ee/..", "", false, 0, ResultFs.DirectoryUnobtainable.Value }, { @"/aa/bb\../cc/..\dd\..\ee/..", "B", false, 0, ResultFs.DirectoryUnobtainable.Value } }; [Theory, MemberData(nameof(TestData_IsNormalized_Backslash))] public static void IsNormalized_Backslash(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { IsNormalizedImpl(path, pathFlags, expectedIsNormalized, expectedLength, expectedResult); } public static TheoryData<string, string, bool, int, Result> TestData_IsNormalized_All => new() { { @"mount:./aa/bb", "WRM", true, 13, Result.Success }, { @"mount:./aa/bb\cc/dd", "WRM", false, 0, Result.Success }, { @"mount:./aa/bb\cc/dd", "WRMB", true, 19, Result.Success }, { @"mount:./.c:/aa/bb", "RM", false, 0, ResultFs.InvalidCharacter.Value }, { @"mount:.c:/aa/bb", "WRM", false, 0, Result.Success }, { @"mount:./cc:/aa/bb", "WRM", false, 0, ResultFs.InvalidCharacter.Value }, { @"mount:./\\host\share/aa/bb", "MW", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:./\\host\share/aa/bb", "WRM", false, 0, Result.Success }, { @"mount:.\\host\share/aa/bb", "WRM", false, 0, Result.Success }, { @"mount:..\\host\share/aa/bb", "WRM", false, 0, Result.Success }, { @".\\host\share/aa/bb", "WRM", false, 0, Result.Success }, { @"..\\host\share/aa/bb", "WRM", false, 0, Result.Success }, { @"mount:\\host\share/aa/bb", "MW", true, 24, Result.Success }, { @"mount:\aa\bb", "BM", false, 0, ResultFs.InvalidPathFormat.Value }, { @"mount:/aa\bb", "BM", true, 12, Result.Success }, { @".//aa/bb", "RW", false, 1, Result.Success }, { @"./aa/bb", "R", true, 7, Result.Success }, { @"./c:/aa/bb", "RW", false, 0, ResultFs.InvalidCharacter.Value } }; [Theory, MemberData(nameof(TestData_IsNormalized_All))] public static void IsNormalized_All(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { IsNormalizedImpl(path, pathFlags, expectedIsNormalized, expectedLength, expectedResult); } private static void IsNormalizedImpl(string path, string pathFlags, bool expectedIsNormalized, long expectedLength, Result expectedResult) { Result result = PathFormatter.IsNormalized(out bool isNormalized, out int length, path.ToU8Span(), GetPathFlags(pathFlags)); Assert.Equal(expectedResult, result); if (result.IsSuccess()) { Assert.Equal(expectedIsNormalized, isNormalized); if (isNormalized) { Assert.Equal(expectedLength, length); } } } [Fact] public static void IsNormalized_InvalidUtf8() { ReadOnlySpan<byte> invalidUtf8 = new byte[] { 0x44, 0xE3, 0xAA, 0x55, 0x50 }; Result result = PathFormatter.IsNormalized(out _, out _, invalidUtf8, new PathFlags()); Assert.Result(ResultFs.InvalidPathFormat, result); } private static PathFlags GetPathFlags(string pathFlags) { var flags = new PathFlags(); foreach (char c in pathFlags) { switch (c) { case 'B': flags.AllowBackslash(); break; case 'E': flags.AllowEmptyPath(); break; case 'M': flags.AllowMountName(); break; case 'R': flags.AllowRelativePath(); break; case 'W': flags.AllowWindowsPath(); break; } } return flags; } } }