Estoy tratando de determinar si una cadena de entrada de un usuario es válida para representar una ruta a una carpeta. Por válido, me refiero a formateado correctamente.
En mi aplicación, la carpeta representa un destino de instalación. Siempre que la ruta de la carpeta sea válida, quiero determinar si la carpeta existe y crearla si no es así.
Actualmente estoy usando IO.Directory.Exists( String path )
. Encuentro que esto funciona bien, excepto cuando el usuario no formatea la cadena correctamente. Cuando eso sucede, este método devolverá falso, lo que indica que la carpeta no existe. Pero esto es un problema porque no podré crear la carpeta después.
En mi búsqueda en Google encontré una sugerencia para usar una expresión regular para verificar si el formato es el adecuado. No tengo experiencia con las expresiones regulares y me pregunto si ese es un enfoque viable. Esto es lo que encontré:
Regex r = new Regex( @"^(([a-zA-Z]\:)|(\\))(\\{1}|((\\{1})[^\\]([^/:*?<>""|]*))+)$" );
return r.IsMatch( path );
¿Una prueba de expresión regular en combinación con Directory.Exists()
, me proporcionaría un método suficientemente bueno para verificar si la ruta es válida y si existe? Sé que esto variará según el sistema operativo y otros factores, pero el programa está dirigido solo a usuarios de Windows .
fuente
Respuestas:
Llamada
Path.GetFullPath
; arrojará excepciones si la ruta no es válida.Para no permitir rutas relativas (como
Word
), llame aPath.IsPathRooted
.fuente
cd C:\\\\\\\Windows\\\\\\\System32
. Para Windows, no puedo encontrar una fuente autorizada que documente este comportamiento, pero, por supuesto, agradecería un puntero a uno.De hecho, no estoy de acuerdo con SLaks. Esa solución no funcionó para mí. La excepción no sucedió como se esperaba. Pero este código funcionó para mí:
if(System.IO.Directory.Exists(path)) { ... }
fuente
Path.GetFullPath da solo las siguientes excepciones
La forma alternativa es utilizar lo siguiente:
/// <summary> /// Validate the Path. If path is relative append the path to the project directory by default. /// </summary> /// <param name="path">Path to validate</param> /// <param name="RelativePath">Relative path</param> /// <param name="Extension">If want to check for File Path</param> /// <returns></returns> private static bool ValidateDllPath(ref string path, string RelativePath = "", string Extension = "") { // Check if it contains any Invalid Characters. if (path.IndexOfAny(Path.GetInvalidPathChars()) == -1) { try { // If path is relative take %IGXLROOT% as the base directory if (!Path.IsPathRooted(path)) { if (string.IsNullOrEmpty(RelativePath)) { // Exceptions handled by Path.GetFullPath // ArgumentException path is a zero-length string, contains only white space, or contains one or more of the invalid characters defined in GetInvalidPathChars. -or- The system could not retrieve the absolute path. // // SecurityException The caller does not have the required permissions. // // ArgumentNullException path is null. // // NotSupportedException path contains a colon (":") that is not part of a volume identifier (for example, "c:\"). // PathTooLongException The specified path, file name, or both exceed the system-defined maximum length. For example, on Windows-based platforms, paths must be less than 248 characters, and file names must be less than 260 characters. // RelativePath is not passed so we would take the project path path = Path.GetFullPath(RelativePath); } else { // Make sure the path is relative to the RelativePath and not our project directory path = Path.Combine(RelativePath, path); } } // Exceptions from FileInfo Constructor: // System.ArgumentNullException: // fileName is null. // // System.Security.SecurityException: // The caller does not have the required permission. // // System.ArgumentException: // The file name is empty, contains only white spaces, or contains invalid characters. // // System.IO.PathTooLongException: // The specified path, file name, or both exceed the system-defined maximum // length. For example, on Windows-based platforms, paths must be less than // 248 characters, and file names must be less than 260 characters. // // System.NotSupportedException: // fileName contains a colon (:) in the middle of the string. FileInfo fileInfo = new FileInfo(path); // Exceptions using FileInfo.Length: // System.IO.IOException: // System.IO.FileSystemInfo.Refresh() cannot update the state of the file or // directory. // // System.IO.FileNotFoundException: // The file does not exist.-or- The Length property is called for a directory. bool throwEx = fileInfo.Length == -1; // Exceptions using FileInfo.IsReadOnly: // System.UnauthorizedAccessException: // Access to fileName is denied. // The file described by the current System.IO.FileInfo object is read-only.-or- // This operation is not supported on the current platform.-or- The caller does // not have the required permission. throwEx = fileInfo.IsReadOnly; if (!string.IsNullOrEmpty(Extension)) { // Validate the Extension of the file. if (Path.GetExtension(path).Equals(Extension, StringComparison.InvariantCultureIgnoreCase)) { // Trim the Library Path path = path.Trim(); return true; } else { return false; } } else { return true; } } catch (ArgumentNullException) { // System.ArgumentNullException: // fileName is null. } catch (System.Security.SecurityException) { // System.Security.SecurityException: // The caller does not have the required permission. } catch (ArgumentException) { // System.ArgumentException: // The file name is empty, contains only white spaces, or contains invalid characters. } catch (UnauthorizedAccessException) { // System.UnauthorizedAccessException: // Access to fileName is denied. } catch (PathTooLongException) { // System.IO.PathTooLongException: // The specified path, file name, or both exceed the system-defined maximum // length. For example, on Windows-based platforms, paths must be less than // 248 characters, and file names must be less than 260 characters. } catch (NotSupportedException) { // System.NotSupportedException: // fileName contains a colon (:) in the middle of the string. } catch (FileNotFoundException) { // System.FileNotFoundException // The exception that is thrown when an attempt to access a file that does not // exist on disk fails. } catch (IOException) { // System.IO.IOException: // An I/O error occurred while opening the file. } catch (Exception) { // Unknown Exception. Might be due to wrong case or nulll checks. } } else { // Path contains invalid characters } return false; }
fuente
Aquí hay una solución que aprovecha el uso de Path.GetFullPath como se recomienda en la respuesta de @SLaks .
En el código que incluyo aquí, tenga en cuenta que
IsValidPath(string path)
está diseñado de tal manera que la persona que llama no tiene que preocuparse por el manejo de excepciones .También puede encontrar que el método al que llama,
TryGetFullPath(...)
también tiene mérito por sí solo cuando desea intentar con seguridad obtener una ruta absoluta ./// <summary> /// Gets a value that indicates whether <paramref name="path"/> /// is a valid path. /// </summary> /// <returns>Returns <c>true</c> if <paramref name="path"/> is a /// valid path; <c>false</c> otherwise. Also returns <c>false</c> if /// the caller does not have the required permissions to access /// <paramref name="path"/>. /// </returns> /// <seealso cref="Path.GetFullPath"/> /// <seealso cref="TryGetFullPath"/> public static bool IsValidPath(string path) { string result; return TryGetFullPath(path, out result); } /// <summary> /// Returns the absolute path for the specified path string. A return /// value indicates whether the conversion succeeded. /// </summary> /// <param name="path">The file or directory for which to obtain absolute /// path information. /// </param> /// <param name="result">When this method returns, contains the absolute /// path representation of <paramref name="path"/>, if the conversion /// succeeded, or <see cref="String.Empty"/> if the conversion failed. /// The conversion fails if <paramref name="path"/> is null or /// <see cref="String.Empty"/>, or is not of the correct format. This /// parameter is passed uninitialized; any value originally supplied /// in <paramref name="result"/> will be overwritten. /// </param> /// <returns><c>true</c> if <paramref name="path"/> was converted /// to an absolute path successfully; otherwise, false. /// </returns> /// <seealso cref="Path.GetFullPath"/> /// <seealso cref="IsValidPath"/> public static bool TryGetFullPath(string path, out string result) { result = String.Empty; if (String.IsNullOrWhiteSpace(path)) { return false; } bool status = false; try { result = Path.GetFullPath(path); status = true; } catch (ArgumentException) { } catch (SecurityException) { } catch (NotSupportedException) { } catch (PathTooLongException) { } return status; }
fuente
Use este código
string DirectoryName = "Sample Name For Directory Or File"; Path.GetInvalidFileNameChars() .Where(x => DirectoryName.Contains(x)) .Count() > 0 || DirectoryName == "con"
fuente
Path.GetInvalidFileNameChars().Any(DirectoryName.Contains) || DirectoryName == "con"
private bool IsValidPath(string path) { Regex driveCheck = new Regex(@"^[a-zA-Z]:\\$"); if (!driveCheck.IsMatch(path.Substring(0, 3))) return false; string strTheseAreInvalidFileNameChars = new string(Path.GetInvalidPathChars()); strTheseAreInvalidFileNameChars += @":/?*" + "\""; Regex containsABadCharacter = new Regex("[" + Regex.Escape(strTheseAreInvalidFileNameChars) + "]"); if (containsABadCharacter.IsMatch(path.Substring(3, path.Length - 3))) return false; DirectoryInfo dir = new DirectoryInfo(Path.GetFullPath(path)); if (!dir.Exists) dir.Create(); return true; }
fuente
No he tenido ningún problema con este código:
private bool IsValidPath(string path, bool exactPath = true) { bool isValid = true; try { string fullPath = Path.GetFullPath(path); if (exactPath) { string root = Path.GetPathRoot(path); isValid = string.IsNullOrEmpty(root.Trim(new char[] { '\\', '/' })) == false; } else { isValid = Path.IsPathRooted(path); } } catch(Exception ex) { isValid = false; } return isValid; }
Por ejemplo, estos devolverían falso:
IsValidPath("C:/abc*d"); IsValidPath("C:/abc?d"); IsValidPath("C:/abc\"d"); IsValidPath("C:/abc<d"); IsValidPath("C:/abc>d"); IsValidPath("C:/abc|d"); IsValidPath("C:/abc:d"); IsValidPath(""); IsValidPath("./abc"); IsValidPath("/abc"); IsValidPath("abc"); IsValidPath("abc", false);
Y estos volverían verdaderos:
IsValidPath(@"C:\\abc"); IsValidPath(@"F:\FILES\"); IsValidPath(@"C:\\abc.docx\\defg.docx"); IsValidPath(@"C:/abc/defg"); IsValidPath(@"C:\\\//\/\\/\\\/abc/\/\/\/\///\\\//\defg"); IsValidPath(@"C:/abc/def~`!@#$%^&()_-+={[}];',.g"); IsValidPath(@"C:\\\\\abc////////defg"); IsValidPath(@"/abc", false);
fuente
Una solución independiente del sistema operativo más sencilla.
public static class PathHelper { public static void ValidatePath(string path) { if (!Directory.Exists(path)) Directory.CreateDirectory(path).Delete(); } }
Uso:
try { PathHelper.ValidatePath(path); } catch(Exception e) { // handle exception }
Directory.CreateDirectory()
arrojará automáticamente en todas las situaciones siguientes:fuente