/SourseRestore/zip/core/FileSystemScanner.cs

# · C# · 405 lines · 212 code · 40 blank · 153 comment · 21 complexity · 5c92f84dc277e22ee1e3a5787b2f2158 MD5 · raw file

  1. // FileSystemScanner.cs
  2. //
  3. // Copyright 2005 John Reilly
  4. //
  5. // This program is free software; you can redistribute it and/or
  6. // modify it under the terms of the GNU General Public License
  7. // as published by the Free Software Foundation; either version 2
  8. // of the License, or (at your option) any later version.
  9. //
  10. // This program is distributed in the hope that it will be useful,
  11. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. // GNU General Public License for more details.
  14. //
  15. // You should have received a copy of the GNU General Public License
  16. // along with this program; if not, write to the Free Software
  17. // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  18. //
  19. // Linking this library statically or dynamically with other modules is
  20. // making a combined work based on this library. Thus, the terms and
  21. // conditions of the GNU General Public License cover the whole
  22. // combination.
  23. //
  24. // As a special exception, the copyright holders of this library give you
  25. // permission to link this library with independent modules to produce an
  26. // executable, regardless of the license terms of these independent
  27. // modules, and to copy and distribute the resulting executable under
  28. // terms of your choice, provided that you also meet, for each linked
  29. // independent module, the terms and conditions of the license of that
  30. // module. An independent module is a module which is not derived from
  31. // or based on this library. If you modify this library, you may extend
  32. // this exception to your version of the library, but you are not
  33. // obligated to do so. If you do not wish to do so, delete this
  34. // exception statement from your version.
  35. using System;
  36. using System.IO;
  37. namespace ICSharpCode.SharpZipLib.Core
  38. {
  39. /// <summary>
  40. /// Event arguments for scanning.
  41. /// </summary>
  42. public class ScanEventArgs : EventArgs
  43. {
  44. /// <summary>
  45. /// Initialise a new instance of <see cref="ScanEventArgs"/>
  46. /// </summary>
  47. /// <param name="name"></param>
  48. public ScanEventArgs(string name)
  49. {
  50. this.name = name;
  51. ContinueRunning = true;
  52. }
  53. string name;
  54. /// <summary>
  55. /// The name for this event.
  56. /// </summary>
  57. public string Name
  58. {
  59. get { return name; }
  60. }
  61. bool continueRunning;
  62. /// <summary>
  63. /// Get set a value indicating if scanning should continue or not.
  64. /// </summary>
  65. public bool ContinueRunning
  66. {
  67. get { return continueRunning; }
  68. set { continueRunning = value; }
  69. }
  70. }
  71. /// <summary>
  72. /// Event arguments for directories.
  73. /// </summary>
  74. public class DirectoryEventArgs : ScanEventArgs
  75. {
  76. /// <summary>
  77. /// Initialize an instance of <see cref="DirectoryEventArgs"></see>.
  78. /// </summary>
  79. /// <param name="name">The name for this directory.</param>
  80. /// <param name="hasMatchingFiles">Flag value indicating if any matching files are contained in this directory.</param>
  81. public DirectoryEventArgs(string name, bool hasMatchingFiles)
  82. : base(name)
  83. {
  84. this.hasMatchingFiles = hasMatchingFiles;
  85. }
  86. /// <summary>
  87. /// Get a value indicating if the directory contains any matching files or not.
  88. /// </summary>
  89. public bool HasMatchingFiles
  90. {
  91. get { return hasMatchingFiles; }
  92. }
  93. bool hasMatchingFiles;
  94. }
  95. /// <summary>
  96. /// Arguments passed when scan failures are detected.
  97. /// </summary>
  98. public class ScanFailureEventArgs
  99. {
  100. /// <summary>
  101. /// Initialise a new instance of <see cref="ScanFailureEventArgs"></see>
  102. /// </summary>
  103. /// <param name="name">The name to apply.</param>
  104. /// <param name="e">The exception to use.</param>
  105. public ScanFailureEventArgs(string name, Exception e)
  106. {
  107. this.name = name;
  108. this.exception = e;
  109. continueRunning = true;
  110. }
  111. string name;
  112. /// <summary>
  113. /// The applicable name.
  114. /// </summary>
  115. public string Name
  116. {
  117. get { return name; }
  118. }
  119. Exception exception;
  120. /// <summary>
  121. /// The applicable exception.
  122. /// </summary>
  123. public Exception Exception
  124. {
  125. get { return exception; }
  126. }
  127. bool continueRunning;
  128. /// <summary>
  129. /// Get / set a value indicating wether scanning should continue.
  130. /// </summary>
  131. public bool ContinueRunning
  132. {
  133. get { return continueRunning; }
  134. set { continueRunning = value; }
  135. }
  136. }
  137. /// <summary>
  138. /// Delegate invokked when a directory is processed.
  139. /// </summary>
  140. public delegate void ProcessDirectoryDelegate(object Sender, DirectoryEventArgs e);
  141. /// <summary>
  142. /// Delegate invoked when a file is processed.
  143. /// </summary>
  144. public delegate void ProcessFileDelegate(object sender, ScanEventArgs e);
  145. /// <summary>
  146. /// Delegate invoked when a directory failure is detected.
  147. /// </summary>
  148. public delegate void DirectoryFailureDelegate(object sender, ScanFailureEventArgs e);
  149. /// <summary>
  150. /// Delegate invoked when a file failure is detected.
  151. /// </summary>
  152. public delegate void FileFailureDelegate(object sender, ScanFailureEventArgs e);
  153. /// <summary>
  154. /// FileSystemScanner provides facilities scanning of files and directories.
  155. /// </summary>
  156. public class FileSystemScanner
  157. {
  158. /// <summary>
  159. /// Initialise a new instance of <see cref="FileSystemScanner"></see>
  160. /// </summary>
  161. /// <param name="filter">The file filter to apply when scanning.</param>
  162. public FileSystemScanner(string filter)
  163. {
  164. fileFilter = new PathFilter(filter);
  165. }
  166. /// <summary>
  167. /// Initialise a new instance of <see cref="FileSystemScanner"></see>
  168. /// </summary>
  169. /// <param name="fileFilter">The file <see cref="NameFilter"></see>filter to apply.</param>
  170. /// <param name="directoryFilter">The directory <see cref="NameFilter"></see>filter to apply.</param>
  171. public FileSystemScanner(string fileFilter, string directoryFilter)
  172. {
  173. this.fileFilter = new PathFilter(fileFilter);
  174. this.directoryFilter = new PathFilter(directoryFilter);
  175. }
  176. /// <summary>
  177. /// Initialise a new instance of <see cref="FileSystemScanner"></see>
  178. /// </summary>
  179. /// <param name="fileFilter">The file <see cref="NameFilter"></see>filter to apply.</param>
  180. public FileSystemScanner(IScanFilter fileFilter)
  181. {
  182. this.fileFilter = fileFilter;
  183. }
  184. /// <summary>
  185. /// Initialise a new instance of <see cref="FileSystemScanner"></see>
  186. /// </summary>
  187. /// <param name="fileFilter">The file <see cref="IScanFilter"></see>filter to apply.</param>
  188. /// <param name="directoryFilter">The directory <see cref="IScanFilter"></see>filter to apply.</param>
  189. public FileSystemScanner(IScanFilter fileFilter, IScanFilter directoryFilter)
  190. {
  191. this.fileFilter = fileFilter;
  192. this.directoryFilter = directoryFilter;
  193. }
  194. /// <summary>
  195. /// Delegate to invoke when a directory is processed.
  196. /// </summary>
  197. public ProcessDirectoryDelegate ProcessDirectory;
  198. /// <summary>
  199. /// Delegate to invoke when a file is processed.
  200. /// </summary>
  201. public ProcessFileDelegate ProcessFile;
  202. /// <summary>
  203. /// Delegate to invoke when a directory failure is detected.
  204. /// </summary>
  205. public DirectoryFailureDelegate DirectoryFailure;
  206. /// <summary>
  207. /// Delegate to invoke when a file failure is detected.
  208. /// </summary>
  209. public FileFailureDelegate FileFailure;
  210. /// <summary>
  211. /// Raise the DirectoryFailure event.
  212. /// </summary>
  213. /// <param name="directory">Rhe directory name.</param>
  214. /// <param name="e">The exception detected.</param>
  215. public void OnDirectoryFailure(string directory, Exception e)
  216. {
  217. if (DirectoryFailure == null)
  218. {
  219. alive = false;
  220. }
  221. else
  222. {
  223. ScanFailureEventArgs args = new ScanFailureEventArgs(directory, e);
  224. DirectoryFailure(this, args);
  225. alive = args.ContinueRunning;
  226. }
  227. }
  228. /// <summary>
  229. /// Raise the FileFailure event.
  230. /// </summary>
  231. /// <param name="file">The file name.</param>
  232. /// <param name="e">The exception detected.</param>
  233. public void OnFileFailure(string file, Exception e)
  234. {
  235. if (FileFailure == null)
  236. {
  237. alive = false;
  238. }
  239. else
  240. {
  241. ScanFailureEventArgs args = new ScanFailureEventArgs(file, e);
  242. FileFailure(this, args);
  243. alive = args.ContinueRunning;
  244. }
  245. }
  246. /// <summary>
  247. /// Raise the ProcessFile event.
  248. /// </summary>
  249. /// <param name="file">The file name.</param>
  250. public void OnProcessFile(string file)
  251. {
  252. if (ProcessFile != null)
  253. {
  254. ScanEventArgs args = new ScanEventArgs(file);
  255. ProcessFile(this, args);
  256. alive = args.ContinueRunning;
  257. }
  258. }
  259. /// <summary>
  260. /// Raise the ProcessDirectory event.
  261. /// </summary>
  262. /// <param name="directory">The directory name.</param>
  263. /// <param name="hasMatchingFiles">Flag indicating if the directory has matching files.</param>
  264. public void OnProcessDirectory(string directory, bool hasMatchingFiles)
  265. {
  266. if (ProcessDirectory != null)
  267. {
  268. DirectoryEventArgs args = new DirectoryEventArgs(directory, hasMatchingFiles);
  269. ProcessDirectory(this, args);
  270. alive = args.ContinueRunning;
  271. }
  272. }
  273. /// <summary>
  274. /// Scan a directory.
  275. /// </summary>
  276. /// <param name="directory">The base directory to scan.</param>
  277. /// <param name="recurse">True to recurse subdirectories, false to do a single directory.</param>
  278. public void Scan(string directory, bool recurse)
  279. {
  280. alive = true;
  281. ScanDir(directory, recurse);
  282. }
  283. void ScanDir(string directory, bool recurse)
  284. {
  285. try
  286. {
  287. string[] names = System.IO.Directory.GetFiles(directory);
  288. bool hasMatch = false;
  289. for (int fileIndex = 0; fileIndex < names.Length; ++fileIndex)
  290. {
  291. if (!fileFilter.IsMatch(names[fileIndex]))
  292. {
  293. names[fileIndex] = null;
  294. }
  295. else
  296. {
  297. hasMatch = true;
  298. }
  299. }
  300. OnProcessDirectory(directory, hasMatch);
  301. if (alive && hasMatch)
  302. {
  303. foreach (string fileName in names)
  304. {
  305. try
  306. {
  307. if (fileName != null)
  308. {
  309. OnProcessFile(fileName);
  310. if (!alive)
  311. {
  312. break;
  313. }
  314. }
  315. }
  316. catch (Exception e)
  317. {
  318. OnFileFailure(fileName, e);
  319. }
  320. }
  321. }
  322. }
  323. catch (Exception e)
  324. {
  325. OnDirectoryFailure(directory, e);
  326. }
  327. if (alive && recurse)
  328. {
  329. try
  330. {
  331. string[] names = System.IO.Directory.GetDirectories(directory);
  332. foreach (string fulldir in names)
  333. {
  334. if ((directoryFilter == null) || (directoryFilter.IsMatch(fulldir)))
  335. {
  336. ScanDir(fulldir, true);
  337. if (!alive)
  338. {
  339. break;
  340. }
  341. }
  342. }
  343. }
  344. catch (Exception e)
  345. {
  346. OnDirectoryFailure(directory, e);
  347. }
  348. }
  349. }
  350. #region Instance Fields
  351. /// <summary>
  352. /// The file filter currently in use.
  353. /// </summary>
  354. IScanFilter fileFilter;
  355. /// <summary>
  356. /// The directory filter currently in use.
  357. /// </summary>
  358. IScanFilter directoryFilter;
  359. /// <summary>
  360. /// Falg indicating if scanning is still alive. Used to cancel a scan.
  361. /// </summary>
  362. bool alive;
  363. #endregion
  364. }
  365. }