You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			162 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			162 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="FileEnumerator.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright>                                                                 | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | /* | ||
|  |  * FileEnumerator class | ||
|  |  *  | ||
|  |  * Copyright (c) 2003 Microsoft Corporation | ||
|  |  * | ||
|  |  * Class to efficiently enumerate the files in a directory.  The only thing the framework provides | ||
|  |  * to do this is Directory.GetFiles(), which is unusable on large directories because it returns an | ||
|  |  * array containing all the file names at once (huge memory allocation). | ||
|  |  * | ||
|  |  * An efficient alternative is to use FindFirstFile/FindNextFile, which works but requires a lot | ||
|  |  * more code.  Also, it makes our code base harder to port to non-windows platforms. | ||
|  |  * | ||
|  |  * This FileEnumerator class solves both problem, by providing a simple and efficient wrapper. | ||
|  |  * By working with a single object, it is almost as efficient as calling FindFirstFile/FindNextFile, | ||
|  |  * but is much easier to use.  e.g. instead of: | ||
|  |  * | ||
|  |  *      UnsafeNativeMethods.WIN32_FIND_DATA wfd; | ||
|  |  *      IntPtr hFindFile = UnsafeNativeMethods.FindFirstFile(physicalDir + @"\*.*", out wfd); | ||
|  |  * | ||
|  |  *      if (hFindFile == INVALID_HANDLE_VALUE) | ||
|  |  *          return; | ||
|  |  * | ||
|  |  *      try { | ||
|  |  *          for (bool more=true; more; more=UnsafeNativeMethods.FindNextFile(hFindFile, out wfd)) { | ||
|  |  * | ||
|  |  *              // Skip false directories | ||
|  |  *              if (wfd.cFileName == "." || wfd.cFileName == "..") | ||
|  |  *                  continue; | ||
|  |  *               | ||
|  |  *              string fullPath = Path.Combine(physicalDir, wfd.cFileName); | ||
|  |  * | ||
|  |  *              ProcessFile(fullPath); | ||
|  |  *          } | ||
|  |  *      } | ||
|  |  *      finally { | ||
|  |  *          UnsafeNativeMethods.FindClose(hFindFile); | ||
|  |  *      } | ||
|  |  * | ||
|  |  * we can simply write | ||
|  |  * | ||
|  |  *      foreach (FileData fileData in FileEnumerator.Create(physicalDir)) { | ||
|  |  *          ProcessFile(fileData.FullName); | ||
|  |  *      } | ||
|  |  */ | ||
|  | 
 | ||
|  | 
 | ||
|  | namespace System.Web.Util { | ||
|  | 
 | ||
|  | using System.IO; | ||
|  | using System.Collections; | ||
|  | 
 | ||
|  | /* | ||
|  |  * This is a somewhat artificial base class for FileEnumerator.  The main reason | ||
|  |  * for it is to allow user code to be more readable, by looking like: | ||
|  |  *      foreach (FileData fileData in FileEnumerator.Create(path)) { ... } | ||
|  |  * instead of | ||
|  |  *      foreach (FileEnumerator fileData in FileEnumerator.Create(path)) { ... } | ||
|  |  */ | ||
|  | internal abstract class FileData { | ||
|  | 
 | ||
|  |     protected string _path; | ||
|  |     protected UnsafeNativeMethods.WIN32_FIND_DATA _wfd; | ||
|  | 
 | ||
|  |     internal string Name { | ||
|  |         get { return _wfd.cFileName; } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal string FullName { | ||
|  |         get { return _path + @"\" + _wfd.cFileName; } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal bool IsDirectory { | ||
|  |         get { return (_wfd.dwFileAttributes & UnsafeNativeMethods.FILE_ATTRIBUTE_DIRECTORY) != 0; } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal bool IsHidden { | ||
|  |         get { return (_wfd.dwFileAttributes & UnsafeNativeMethods.FILE_ATTRIBUTE_HIDDEN) != 0; } | ||
|  |     } | ||
|  | 
 | ||
|  |     internal FindFileData GetFindFileData() { | ||
|  |         return new FindFileData(ref _wfd); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | internal class FileEnumerator: FileData, IEnumerable, IEnumerator, IDisposable { | ||
|  |     private IntPtr _hFindFile = UnsafeNativeMethods.INVALID_HANDLE_VALUE; | ||
|  | 
 | ||
|  |     internal static FileEnumerator Create(string path) { | ||
|  |         return new FileEnumerator(path); | ||
|  |     } | ||
|  | 
 | ||
|  |     private FileEnumerator(string path) { | ||
|  |         _path = Path.GetFullPath(path); | ||
|  |     } | ||
|  | 
 | ||
|  |     ~FileEnumerator() { | ||
|  |         ((IDisposable)this).Dispose(); | ||
|  |     } | ||
|  | 
 | ||
|  |     // Should the current file be excluded from the enumeration | ||
|  |     private bool SkipCurrent() { | ||
|  |      | ||
|  |         // Skip false directories | ||
|  |         if (_wfd.cFileName == "." || _wfd.cFileName == "..") | ||
|  |             return true; | ||
|  | 
 | ||
|  |         return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     // We just return ourselves for the enumerator, to avoid creating a new object | ||
|  |     IEnumerator IEnumerable.GetEnumerator() { | ||
|  |         return this; | ||
|  |     } | ||
|  | 
 | ||
|  |     bool IEnumerator.MoveNext() { | ||
|  | 
 | ||
|  |         for (;;) { | ||
|  |             if (_hFindFile == UnsafeNativeMethods.INVALID_HANDLE_VALUE) { | ||
|  |                 _hFindFile = UnsafeNativeMethods.FindFirstFile(_path + @"\*.*", out _wfd); | ||
|  |                  | ||
|  |                 // Empty enumeration case | ||
|  |                 if (_hFindFile == UnsafeNativeMethods.INVALID_HANDLE_VALUE) | ||
|  |                     return false; | ||
|  |             } | ||
|  |             else { | ||
|  |                 bool hasMoreFiles = UnsafeNativeMethods.FindNextFile(_hFindFile, out _wfd); | ||
|  |                 if (!hasMoreFiles) | ||
|  |                     return false; | ||
|  |             } | ||
|  |              | ||
|  |             if (!SkipCurrent()) | ||
|  |                 return true; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     // The current object of the enumeration is always ourselves.  No new object created. | ||
|  |     object IEnumerator.Current { | ||
|  |         get { return this; } | ||
|  |     } | ||
|  | 
 | ||
|  |     void IEnumerator.Reset() { | ||
|  |         // We don't support reset, though it would be easy to add if needed | ||
|  |         throw new InvalidOperationException(); | ||
|  |     } | ||
|  | 
 | ||
|  |     void IDisposable.Dispose() { | ||
|  |         if (_hFindFile != UnsafeNativeMethods.INVALID_HANDLE_VALUE) { | ||
|  |             UnsafeNativeMethods.FindClose(_hFindFile); | ||
|  |             _hFindFile = UnsafeNativeMethods.INVALID_HANDLE_VALUE; | ||
|  |         } | ||
|  |         System.GC.SuppressFinalize(this); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | } |