// Copyright (c) .NET Foundation and contributors. All rights reserved. // Licensed under the MIT license. See LICENSE file in the project root for full license information. using System; using System.Diagnostics; using System.IO; namespace Microsoft.DotNet.Tools.Test.Utilities { // Contains path parsing utilities. // We need our own because System.IO.Path is insufficient for our purposes // For example we need to be able to work with invalid paths or paths containing wildcards internal static class PathUtilities { // We consider '/' a directory separator on Unix like systems. // On Windows both / and \ are equally accepted. internal static readonly char DirectorySeparatorChar = IsUnixLikePlatform ? '/' : '\\'; internal static readonly char AltDirectorySeparatorChar = '/'; internal static readonly string DirectorySeparatorStr = new string(DirectorySeparatorChar, 1); internal const char VolumeSeparatorChar = ':'; private static bool IsUnixLikePlatform => Path.DirectorySeparatorChar == '/'; internal static bool IsDirectorySeparator(char c) => c == DirectorySeparatorChar || c == AltDirectorySeparatorChar; internal static string TrimTrailingSeparators(string s) { int lastSeparator = s.Length; while (lastSeparator > 0 && IsDirectorySeparator(s[lastSeparator - 1])) { lastSeparator = lastSeparator - 1; } if (lastSeparator != s.Length) { s = s.Substring(0, lastSeparator); } return s; } internal static string GetExtension(string path) => FileNameUtilities.GetExtension(path); internal static string ChangeExtension(string path, string extension) => FileNameUtilities.ChangeExtension(path, extension); internal static string RemoveExtension(string path) => FileNameUtilities.ChangeExtension(path, extension: null); internal static string GetFileName(string path) => FileNameUtilities.GetFileName(path); /// /// Get directory name from path. /// /// /// Unlike it /// doesn't check for invalid path characters, /// doesn't strip any trailing directory separators (TODO: tomat), /// doesn't recognize UNC structure \\computer-name\share\directory-name\file-name (TODO: tomat). /// /// Prefix of path that represents a directory. internal static string GetDirectoryName(string path) { int fileNameStart = FileNameUtilities.IndexOfFileName(path); if (fileNameStart < 0) { return null; } return path.Substring(0, fileNameStart); } internal static PathKind GetPathKind(string path) { if (string.IsNullOrWhiteSpace(path)) { return PathKind.Empty; } // "C:\" // "\\machine" (UNC) // "/etc" (Unix) if (IsAbsolute(path)) { return PathKind.Absolute; } // "." // ".." // ".\" // "..\" if (path.Length > 0 && path[0] == '.') { if (path.Length == 1 || IsDirectorySeparator(path[1])) { return PathKind.RelativeToCurrentDirectory; } if (path[1] == '.') { if (path.Length == 2 || IsDirectorySeparator(path[2])) { return PathKind.RelativeToCurrentParent; } } } if (!IsUnixLikePlatform) { // "\" // "\foo" if (path.Length >= 1 && IsDirectorySeparator(path[0])) { return PathKind.RelativeToCurrentRoot; } // "C:foo" if (path.Length >= 2 && path[1] == VolumeSeparatorChar && (path.Length <= 2 || !IsDirectorySeparator(path[2]))) { return PathKind.RelativeToDriveDirectory; } } // "foo.dll" return PathKind.Relative; } internal static bool IsAbsolute(string path) { if (string.IsNullOrEmpty(path)) { return false; } if (IsUnixLikePlatform) { return path[0] == DirectorySeparatorChar; } // "C:\" if (IsDriveRootedAbsolutePath(path)) { // Including invalid paths (e.g. "*:\") return true; } // "\\machine\share" // Including invalid/incomplete UNC paths (e.g. "\\foo") return path.Length >= 2 && IsDirectorySeparator(path[0]) && IsDirectorySeparator(path[1]); } /// /// Returns true if given path is absolute and starts with a drive specification ("C:\"). /// private static bool IsDriveRootedAbsolutePath(string path) { Debug.Assert(!IsUnixLikePlatform); return path.Length >= 3 && path[1] == VolumeSeparatorChar && IsDirectorySeparator(path[2]); } /// /// Get a prefix of given path which is the root of the path. /// /// /// Root of an absolute path or null if the path isn't absolute or has invalid format (e.g. "\\"). /// It may or may not end with a directory separator (e.g. "C:\", "C:\foo", "\\machine\share", etc.) . /// internal static string GetPathRoot(string path) { if (string.IsNullOrWhiteSpace(path)) { return null; } int length = GetPathRootLength(path); return (length != -1) ? path.Substring(0, length) : null; } private static int GetPathRootLength(string path) { Debug.Assert(!string.IsNullOrEmpty(path)); if (IsUnixLikePlatform) { if (IsDirectorySeparator(path[0])) { // "/*" return 1; } } else { // "C:\" if (IsDriveRootedAbsolutePath(path)) { return 3; } if (IsDirectorySeparator(path[0])) { // "\\machine\share" return GetUncPathRootLength(path); } } return -1; } /// /// Calculates the length of root of an UNC path. /// /// /// "\\server\share" is root of UNC path "\\server\share\dir1\dir2\file". /// private static int GetUncPathRootLength(string path) { Debug.Assert(IsDirectorySeparator(path[0])); // root: // [directory-separator]{2,}[^directory-separator]+[directory-separator]+[^directory-separator]+ int serverIndex = IndexOfNonDirectorySeparator(path, 1); if (serverIndex < 2) { return -1; } int separator = IndexOfDirectorySeparator(path, serverIndex); if (separator == -1) { return -1; } int shareIndex = IndexOfNonDirectorySeparator(path, separator); if (shareIndex == -1) { return -1; } int rootEnd = IndexOfDirectorySeparator(path, shareIndex); return rootEnd == -1 ? path.Length : rootEnd; } private static int IndexOfDirectorySeparator(string path, int start) { for (int i = start; i < path.Length; i++) { if (IsDirectorySeparator(path[i])) { return i; } } return -1; } private static int IndexOfNonDirectorySeparator(string path, int start) { for (int i = start; i < path.Length; i++) { if (!IsDirectorySeparator(path[i])) { return i; } } return -1; } /// /// Combines an absolute path with a relative. /// /// Absolute root path. /// Relative path. /// /// An absolute combined path, or null if is /// absolute (e.g. "C:\abc", "\\machine\share\abc"), /// relative to the current root (e.g. "\abc"), /// or relative to a drive directory (e.g. "C:abc\def"). /// /// internal static string CombineAbsoluteAndRelativePaths(string root, string relativePath) { Debug.Assert(IsAbsolute(root)); return CombinePossiblyRelativeAndRelativePaths(root, relativePath); } /// /// Combine two paths, the first of which may be absolute. /// /// First path: absolute, relative, or null. /// Second path: relative and non-null. /// null, if is null; a combined path, otherwise. /// internal static string CombinePossiblyRelativeAndRelativePaths(string rootOpt, string relativePath) { if (string.IsNullOrEmpty(rootOpt)) { return null; } switch (GetPathKind(relativePath)) { case PathKind.Empty: return rootOpt; case PathKind.Absolute: case PathKind.RelativeToCurrentRoot: case PathKind.RelativeToDriveDirectory: return null; } return CombinePathsUnchecked(rootOpt, relativePath); } internal static string CombinePathsUnchecked(string root, string relativePath) { Debug.Assert(!string.IsNullOrEmpty(root)); char c = root[root.Length - 1]; if (!IsDirectorySeparator(c) && c != VolumeSeparatorChar) { return root + DirectorySeparatorStr + relativePath; } return root + relativePath; } internal static string RemoveTrailingDirectorySeparator(string path) { if (path.Length > 0 && IsDirectorySeparator(path[path.Length - 1])) { return path.Substring(0, path.Length - 1); } else { return path; } } /// /// Determines whether an assembly reference is considered an assembly file path or an assembly name. /// used, for example, on values of /r and #r. /// internal static bool IsFilePath(string assemblyDisplayNameOrPath) { Debug.Assert(assemblyDisplayNameOrPath != null); string extension = FileNameUtilities.GetExtension(assemblyDisplayNameOrPath); return string.Equals(extension, ".dll", StringComparison.OrdinalIgnoreCase) || string.Equals(extension, ".exe", StringComparison.OrdinalIgnoreCase) || assemblyDisplayNameOrPath.IndexOf(DirectorySeparatorChar) != -1 || assemblyDisplayNameOrPath.IndexOf(AltDirectorySeparatorChar) != -1; } } }