mirror of
https://github.com/Thealexbarney/LibHac.git
synced 2024-11-14 10:49:41 +01:00
749 lines
No EOL
26 KiB
C#
749 lines
No EOL
26 KiB
C#
//
|
|
// Copyright (c) 2008-2011, Kenneth Bell
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Text.RegularExpressions;
|
|
using DiscUtils.Internal;
|
|
using DiscUtils.Streams;
|
|
|
|
namespace DiscUtils.Vfs
|
|
{
|
|
/// <summary>
|
|
/// Base class for VFS file systems.
|
|
/// </summary>
|
|
/// <typeparam name="TDirEntry">The concrete type representing directory entries.</typeparam>
|
|
/// <typeparam name="TFile">The concrete type representing files.</typeparam>
|
|
/// <typeparam name="TDirectory">The concrete type representing directories.</typeparam>
|
|
/// <typeparam name="TContext">The concrete type holding global state.</typeparam>
|
|
public abstract class VfsFileSystem<TDirEntry, TFile, TDirectory, TContext> : DiscFileSystem
|
|
where TDirEntry : VfsDirEntry
|
|
where TFile : IVfsFile
|
|
where TDirectory : class, IVfsDirectory<TDirEntry, TFile>, TFile
|
|
where TContext : VfsContext
|
|
{
|
|
private readonly ObjectCache<long, TFile> _fileCache;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the VfsFileSystem class.
|
|
/// </summary>
|
|
/// <param name="defaultOptions">The default file system options.</param>
|
|
protected VfsFileSystem(DiscFileSystemOptions defaultOptions)
|
|
: base(defaultOptions)
|
|
{
|
|
_fileCache = new ObjectCache<long, TFile>();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the global shared state.
|
|
/// </summary>
|
|
protected TContext Context { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets or sets the object representing the root directory.
|
|
/// </summary>
|
|
protected TDirectory RootDirectory { get; set; }
|
|
|
|
/// <summary>
|
|
/// Gets the volume label.
|
|
/// </summary>
|
|
public abstract override string VolumeLabel { get; }
|
|
|
|
/// <summary>
|
|
/// Copies a file - not supported on read-only file systems.
|
|
/// </summary>
|
|
/// <param name="sourceFile">The source file.</param>
|
|
/// <param name="destinationFile">The destination file.</param>
|
|
/// <param name="overwrite">Whether to permit over-writing of an existing file.</param>
|
|
public override void CopyFile(string sourceFile, string destinationFile, bool overwrite)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a directory - not supported on read-only file systems.
|
|
/// </summary>
|
|
/// <param name="path">The path of the new directory.</param>
|
|
public override void CreateDirectory(string path)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes a directory - not supported on read-only file systems.
|
|
/// </summary>
|
|
/// <param name="path">The path of the directory to delete.</param>
|
|
public override void DeleteDirectory(string path)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes a file - not supported on read-only file systems.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file to delete.</param>
|
|
public override void DeleteFile(string path)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Indicates if a directory exists.
|
|
/// </summary>
|
|
/// <param name="path">The path to test.</param>
|
|
/// <returns>true if the directory exists.</returns>
|
|
public override bool DirectoryExists(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
|
|
if (dirEntry != null)
|
|
{
|
|
return dirEntry.IsDirectory;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Indicates if a file exists.
|
|
/// </summary>
|
|
/// <param name="path">The path to test.</param>
|
|
/// <returns>true if the file exists.</returns>
|
|
public override bool FileExists(string path)
|
|
{
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
|
|
if (dirEntry != null)
|
|
{
|
|
return !dirEntry.IsDirectory;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the names of subdirectories in a specified directory matching a specified
|
|
/// search pattern, using a value to determine whether to search subdirectories.
|
|
/// </summary>
|
|
/// <param name="path">The path to search.</param>
|
|
/// <param name="searchPattern">The search string to match against.</param>
|
|
/// <param name="searchOption">Indicates whether to search subdirectories.</param>
|
|
/// <returns>Array of directories matching the search pattern.</returns>
|
|
public override string[] GetDirectories(string path, string searchPattern, SearchOption searchOption)
|
|
{
|
|
Regex re = Utilities.ConvertWildcardsToRegEx(searchPattern);
|
|
|
|
List<string> dirs = new List<string>();
|
|
DoSearch(dirs, path, re, searchOption == SearchOption.AllDirectories, true, false);
|
|
return dirs.ToArray();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the names of files in a specified directory matching a specified
|
|
/// search pattern, using a value to determine whether to search subdirectories.
|
|
/// </summary>
|
|
/// <param name="path">The path to search.</param>
|
|
/// <param name="searchPattern">The search string to match against.</param>
|
|
/// <param name="searchOption">Indicates whether to search subdirectories.</param>
|
|
/// <returns>Array of files matching the search pattern.</returns>
|
|
public override string[] GetFiles(string path, string searchPattern, SearchOption searchOption)
|
|
{
|
|
Regex re = Utilities.ConvertWildcardsToRegEx(searchPattern);
|
|
|
|
List<string> results = new List<string>();
|
|
DoSearch(results, path, re, searchOption == SearchOption.AllDirectories, false, true);
|
|
return results.ToArray();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the names of all files and subdirectories in a specified directory.
|
|
/// </summary>
|
|
/// <param name="path">The path to search.</param>
|
|
/// <returns>Array of files and subdirectories matching the search pattern.</returns>
|
|
public override string[] GetFileSystemEntries(string path)
|
|
{
|
|
string fullPath = path;
|
|
if (!fullPath.StartsWith(@"\", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
fullPath = @"\" + fullPath;
|
|
}
|
|
|
|
TDirectory parentDir = GetDirectory(fullPath);
|
|
return Utilities.Map(parentDir.AllEntries,
|
|
m => Utilities.CombinePaths(fullPath, FormatFileName(m.FileName)));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the names of files and subdirectories in a specified directory matching a specified
|
|
/// search pattern.
|
|
/// </summary>
|
|
/// <param name="path">The path to search.</param>
|
|
/// <param name="searchPattern">The search string to match against.</param>
|
|
/// <returns>Array of files and subdirectories matching the search pattern.</returns>
|
|
public override string[] GetFileSystemEntries(string path, string searchPattern)
|
|
{
|
|
Regex re = Utilities.ConvertWildcardsToRegEx(searchPattern);
|
|
|
|
TDirectory parentDir = GetDirectory(path);
|
|
|
|
List<string> result = new List<string>();
|
|
foreach (TDirEntry dirEntry in parentDir.AllEntries)
|
|
{
|
|
if (re.IsMatch(dirEntry.SearchName))
|
|
{
|
|
result.Add(Utilities.CombinePaths(path, dirEntry.FileName));
|
|
}
|
|
}
|
|
|
|
return result.ToArray();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Moves a directory.
|
|
/// </summary>
|
|
/// <param name="sourceDirectoryName">The directory to move.</param>
|
|
/// <param name="destinationDirectoryName">The target directory name.</param>
|
|
public override void MoveDirectory(string sourceDirectoryName, string destinationDirectoryName)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Moves a file.
|
|
/// </summary>
|
|
/// <param name="sourceName">The file to move.</param>
|
|
/// <param name="destinationName">The target file name.</param>
|
|
/// <param name="overwrite">Overwrite any existing file.</param>
|
|
public override void MoveFile(string sourceName, string destinationName, bool overwrite)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Opens the specified file.
|
|
/// </summary>
|
|
/// <param name="path">The full path of the file to open.</param>
|
|
/// <param name="mode">The file mode for the created stream.</param>
|
|
/// <param name="access">The access permissions for the created stream.</param>
|
|
/// <returns>The new stream.</returns>
|
|
public override SparseStream OpenFile(string path, FileMode mode, FileAccess access)
|
|
{
|
|
if (!CanWrite)
|
|
{
|
|
if (mode != FileMode.Open)
|
|
{
|
|
throw new NotSupportedException("Only existing files can be opened");
|
|
}
|
|
|
|
if (access != FileAccess.Read)
|
|
{
|
|
throw new NotSupportedException("Files cannot be opened for write");
|
|
}
|
|
}
|
|
|
|
string fileName = Utilities.GetFileFromPath(path);
|
|
string attributeName = null;
|
|
|
|
int streamSepPos = fileName.IndexOf(':');
|
|
if (streamSepPos >= 0)
|
|
{
|
|
attributeName = fileName.Substring(streamSepPos + 1);
|
|
}
|
|
|
|
string dirName;
|
|
try
|
|
{
|
|
dirName = Utilities.GetDirectoryFromPath(path);
|
|
}
|
|
catch (ArgumentException)
|
|
{
|
|
throw new IOException("Invalid path: " + path);
|
|
}
|
|
|
|
string entryPath = Utilities.CombinePaths(dirName, fileName);
|
|
TDirEntry entry = GetDirectoryEntry(entryPath);
|
|
if (entry == null)
|
|
{
|
|
if (mode == FileMode.Open)
|
|
{
|
|
throw new FileNotFoundException("No such file", path);
|
|
}
|
|
TDirectory parentDir = GetDirectory(Utilities.GetDirectoryFromPath(path));
|
|
entry = parentDir.CreateNewFile(Utilities.GetFileFromPath(path));
|
|
}
|
|
else if (mode == FileMode.CreateNew)
|
|
{
|
|
throw new IOException("File already exists");
|
|
}
|
|
|
|
if (entry.IsSymlink)
|
|
{
|
|
entry = ResolveSymlink(entry, entryPath);
|
|
}
|
|
|
|
if (entry.IsDirectory)
|
|
{
|
|
throw new IOException("Attempt to open directory as a file");
|
|
}
|
|
TFile file = GetFile(entry);
|
|
|
|
SparseStream stream = null;
|
|
if (string.IsNullOrEmpty(attributeName))
|
|
{
|
|
stream = new BufferStream(file.FileContent, access);
|
|
}
|
|
else
|
|
{
|
|
IVfsFileWithStreams fileStreams = file as IVfsFileWithStreams;
|
|
if (fileStreams != null)
|
|
{
|
|
stream = fileStreams.OpenExistingStream(attributeName);
|
|
if (stream == null)
|
|
{
|
|
if (mode == FileMode.Create || mode == FileMode.OpenOrCreate)
|
|
{
|
|
stream = fileStreams.CreateStream(attributeName);
|
|
}
|
|
else
|
|
{
|
|
throw new FileNotFoundException("No such attribute on file", path);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
throw new NotSupportedException(
|
|
"Attempt to open a file stream on a file system that doesn't support them");
|
|
}
|
|
}
|
|
|
|
if (mode == FileMode.Create || mode == FileMode.Truncate)
|
|
{
|
|
stream.SetLength(0);
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the attributes of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The file or directory to inspect.</param>
|
|
/// <returns>The attributes of the file or directory.</returns>
|
|
public override FileAttributes GetAttributes(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return RootDirectory.FileAttributes;
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
if (dirEntry == null)
|
|
{
|
|
throw new FileNotFoundException("File not found", path);
|
|
}
|
|
|
|
if (dirEntry.HasVfsFileAttributes)
|
|
{
|
|
return dirEntry.FileAttributes;
|
|
}
|
|
return GetFile(dirEntry).FileAttributes;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the attributes of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The file or directory to change.</param>
|
|
/// <param name="newValue">The new attributes of the file or directory.</param>
|
|
public override void SetAttributes(string path, FileAttributes newValue)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the creation time (in UTC) of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file or directory.</param>
|
|
/// <returns>The creation time.</returns>
|
|
public override DateTime GetCreationTimeUtc(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return RootDirectory.CreationTimeUtc;
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
if (dirEntry == null)
|
|
{
|
|
throw new FileNotFoundException("No such file or directory", path);
|
|
}
|
|
|
|
if (dirEntry.HasVfsTimeInfo)
|
|
{
|
|
return dirEntry.CreationTimeUtc;
|
|
}
|
|
return GetFile(dirEntry).CreationTimeUtc;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the creation time (in UTC) of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file or directory.</param>
|
|
/// <param name="newTime">The new time to set.</param>
|
|
public override void SetCreationTimeUtc(string path, DateTime newTime)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the last access time (in UTC) of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file or directory.</param>
|
|
/// <returns>The last access time.</returns>
|
|
public override DateTime GetLastAccessTimeUtc(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return RootDirectory.LastAccessTimeUtc;
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
if (dirEntry == null)
|
|
{
|
|
throw new FileNotFoundException("No such file or directory", path);
|
|
}
|
|
|
|
if (dirEntry.HasVfsTimeInfo)
|
|
{
|
|
return dirEntry.LastAccessTimeUtc;
|
|
}
|
|
return GetFile(dirEntry).LastAccessTimeUtc;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the last access time (in UTC) of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file or directory.</param>
|
|
/// <param name="newTime">The new time to set.</param>
|
|
public override void SetLastAccessTimeUtc(string path, DateTime newTime)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the last modification time (in UTC) of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file or directory.</param>
|
|
/// <returns>The last write time.</returns>
|
|
public override DateTime GetLastWriteTimeUtc(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return RootDirectory.LastWriteTimeUtc;
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
if (dirEntry == null)
|
|
{
|
|
throw new FileNotFoundException("No such file or directory", path);
|
|
}
|
|
|
|
if (dirEntry.HasVfsTimeInfo)
|
|
{
|
|
return dirEntry.LastWriteTimeUtc;
|
|
}
|
|
return GetFile(dirEntry).LastWriteTimeUtc;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the last modification time (in UTC) of a file or directory.
|
|
/// </summary>
|
|
/// <param name="path">The path of the file or directory.</param>
|
|
/// <param name="newTime">The new time to set.</param>
|
|
public override void SetLastWriteTimeUtc(string path, DateTime newTime)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the length of a file.
|
|
/// </summary>
|
|
/// <param name="path">The path to the file.</param>
|
|
/// <returns>The length in bytes.</returns>
|
|
public override long GetFileLength(string path)
|
|
{
|
|
TFile file = GetFile(path);
|
|
if (file == null || (file.FileAttributes & FileAttributes.Directory) != 0)
|
|
{
|
|
throw new FileNotFoundException("No such file", path);
|
|
}
|
|
|
|
return file.FileLength;
|
|
}
|
|
|
|
internal TFile GetFile(TDirEntry dirEntry)
|
|
{
|
|
long cacheKey = dirEntry.UniqueCacheId;
|
|
|
|
TFile file = _fileCache[cacheKey];
|
|
if (file == null)
|
|
{
|
|
file = ConvertDirEntryToFile(dirEntry);
|
|
_fileCache[cacheKey] = file;
|
|
}
|
|
|
|
return file;
|
|
}
|
|
|
|
internal TDirectory GetDirectory(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return RootDirectory;
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
|
|
if (dirEntry != null && dirEntry.IsSymlink)
|
|
{
|
|
dirEntry = ResolveSymlink(dirEntry, path);
|
|
}
|
|
|
|
if (dirEntry == null || !dirEntry.IsDirectory)
|
|
{
|
|
throw new DirectoryNotFoundException("No such directory: " + path);
|
|
}
|
|
|
|
return (TDirectory)GetFile(dirEntry);
|
|
}
|
|
|
|
internal TDirEntry GetDirectoryEntry(string path)
|
|
{
|
|
return GetDirectoryEntry(RootDirectory, path);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets all directory entries in the specified directory and sub-directories.
|
|
/// </summary>
|
|
/// <param name="path">The path to inspect.</param>
|
|
/// <param name="handler">Delegate invoked for each directory entry.</param>
|
|
protected void ForAllDirEntries(string path, DirEntryHandler handler)
|
|
{
|
|
TDirectory dir = null;
|
|
TDirEntry self = GetDirectoryEntry(path);
|
|
|
|
if (self != null)
|
|
{
|
|
handler(path, self);
|
|
if (self.IsDirectory)
|
|
{
|
|
dir = GetFile(self) as TDirectory;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dir = GetFile(path) as TDirectory;
|
|
}
|
|
|
|
if (dir != null)
|
|
{
|
|
foreach (TDirEntry subentry in dir.AllEntries)
|
|
{
|
|
ForAllDirEntries(Utilities.CombinePaths(path, subentry.FileName), handler);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the file object for a given path.
|
|
/// </summary>
|
|
/// <param name="path">The path to query.</param>
|
|
/// <returns>The file object corresponding to the path.</returns>
|
|
protected TFile GetFile(string path)
|
|
{
|
|
if (IsRoot(path))
|
|
{
|
|
return RootDirectory;
|
|
}
|
|
if (path == null)
|
|
{
|
|
return default(TFile);
|
|
}
|
|
|
|
TDirEntry dirEntry = GetDirectoryEntry(path);
|
|
if (dirEntry == null)
|
|
{
|
|
throw new FileNotFoundException("No such file or directory", path);
|
|
}
|
|
|
|
return GetFile(dirEntry);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts a directory entry to an object representing a file.
|
|
/// </summary>
|
|
/// <param name="dirEntry">The directory entry to convert.</param>
|
|
/// <returns>The corresponding file object.</returns>
|
|
protected abstract TFile ConvertDirEntryToFile(TDirEntry dirEntry);
|
|
|
|
/// <summary>
|
|
/// Converts an internal directory entry name into an external one.
|
|
/// </summary>
|
|
/// <param name="name">The name to convert.</param>
|
|
/// <returns>The external name.</returns>
|
|
/// <remarks>
|
|
/// This method is called on a single path element (i.e. name contains no path
|
|
/// separators).
|
|
/// </remarks>
|
|
protected virtual string FormatFileName(string name)
|
|
{
|
|
return name;
|
|
}
|
|
|
|
private static bool IsRoot(string path)
|
|
{
|
|
return string.IsNullOrEmpty(path) || path == @"\";
|
|
}
|
|
|
|
private TDirEntry GetDirectoryEntry(TDirectory dir, string path)
|
|
{
|
|
string[] pathElements = path.Split(new[] { '\\' }, StringSplitOptions.RemoveEmptyEntries);
|
|
return GetDirectoryEntry(dir, pathElements, 0);
|
|
}
|
|
|
|
private TDirEntry GetDirectoryEntry(TDirectory dir, string[] pathEntries, int pathOffset)
|
|
{
|
|
TDirEntry entry;
|
|
|
|
if (pathEntries.Length == 0)
|
|
{
|
|
return dir.Self;
|
|
}
|
|
entry = dir.GetEntryByName(pathEntries[pathOffset]);
|
|
if (entry != null)
|
|
{
|
|
if (pathOffset == pathEntries.Length - 1)
|
|
{
|
|
return entry;
|
|
}
|
|
if (entry.IsDirectory)
|
|
{
|
|
return GetDirectoryEntry((TDirectory)ConvertDirEntryToFile(entry), pathEntries, pathOffset + 1);
|
|
}
|
|
throw new IOException(string.Format(CultureInfo.InvariantCulture,
|
|
"{0} is a file, not a directory", pathEntries[pathOffset]));
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private void DoSearch(List<string> results, string path, Regex regex, bool subFolders, bool dirs, bool files)
|
|
{
|
|
TDirectory parentDir = GetDirectory(path);
|
|
if (parentDir == null)
|
|
{
|
|
throw new DirectoryNotFoundException(string.Format(CultureInfo.InvariantCulture,
|
|
"The directory '{0}' was not found", path));
|
|
}
|
|
|
|
string resultPrefixPath = path;
|
|
if (IsRoot(path))
|
|
{
|
|
resultPrefixPath = @"\";
|
|
}
|
|
|
|
foreach (TDirEntry de in parentDir.AllEntries)
|
|
{
|
|
TDirEntry entry = de;
|
|
|
|
if (entry.IsSymlink)
|
|
{
|
|
entry = ResolveSymlink(entry, path + "\\" + entry.FileName);
|
|
}
|
|
|
|
bool isDir = entry.IsDirectory;
|
|
|
|
if ((isDir && dirs) || (!isDir && files))
|
|
{
|
|
if (regex.IsMatch(de.SearchName))
|
|
{
|
|
results.Add(Utilities.CombinePaths(resultPrefixPath, FormatFileName(entry.FileName)));
|
|
}
|
|
}
|
|
|
|
if (subFolders && isDir)
|
|
{
|
|
DoSearch(results, Utilities.CombinePaths(resultPrefixPath, FormatFileName(entry.FileName)), regex,
|
|
subFolders, dirs, files);
|
|
}
|
|
}
|
|
}
|
|
|
|
private TDirEntry ResolveSymlink(TDirEntry entry, string path)
|
|
{
|
|
TDirEntry currentEntry = entry;
|
|
if (path.Length > 0 && path[0] != '\\')
|
|
{
|
|
path = '\\' + path;
|
|
}
|
|
string currentPath = path;
|
|
int resolvesLeft = 20;
|
|
while (currentEntry.IsSymlink && resolvesLeft > 0)
|
|
{
|
|
IVfsSymlink<TDirEntry, TFile> symlink = GetFile(currentEntry) as IVfsSymlink<TDirEntry, TFile>;
|
|
if (symlink == null)
|
|
{
|
|
throw new FileNotFoundException("Unable to resolve symlink", path);
|
|
}
|
|
|
|
currentPath = Utilities.ResolvePath(currentPath.TrimEnd('\\'), symlink.TargetPath);
|
|
currentEntry = GetDirectoryEntry(currentPath);
|
|
if (currentEntry == null)
|
|
{
|
|
throw new FileNotFoundException("Unable to resolve symlink", path);
|
|
}
|
|
|
|
--resolvesLeft;
|
|
}
|
|
|
|
if (currentEntry.IsSymlink)
|
|
{
|
|
throw new FileNotFoundException("Unable to resolve symlink - too many links", path);
|
|
}
|
|
|
|
return currentEntry;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Delegate for processing directory entries.
|
|
/// </summary>
|
|
/// <param name="path">Full path to the directory entry.</param>
|
|
/// <param name="dirEntry">The directory entry itself.</param>
|
|
protected delegate void DirEntryHandler(string path, TDirEntry dirEntry);
|
|
}
|
|
} |