2 // System.IO.DirectoryInfo.cs
5 // Miguel de Icaza, miguel@ximian.com
6 // Jim Richardson, develop@wtfo-guru.com
7 // Dan Lewis, dihlewis@yahoo.co.uk
8 // Sebastien Pouliot <sebastien@ximian.com>
10 // Copyright (C) 2002 Ximian, Inc.
11 // Copyright (C) 2001 Moonlight Enterprises, All Rights Reserved
12 // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
14 // Permission is hereby granted, free of charge, to any person obtaining
15 // a copy of this software and associated documentation files (the
16 // "Software"), to deal in the Software without restriction, including
17 // without limitation the rights to use, copy, modify, merge, publish,
18 // distribute, sublicense, and/or sell copies of the Software, and to
19 // permit persons to whom the Software is furnished to do so, subject to
20 // the following conditions:
22 // The above copyright notice and this permission notice shall be
23 // included in all copies or substantial portions of the Software.
25 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
26 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
27 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
28 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
29 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
30 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
31 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
34 using System.Collections;
35 using System.Collections.Generic;
36 using System.Runtime.InteropServices;
37 using System.Runtime.Serialization;
40 using System.Security.AccessControl;
47 public sealed class DirectoryInfo : FileSystemInfo {
49 private string current;
50 private string parent;
52 public DirectoryInfo (string path) : this (path, false)
56 internal DirectoryInfo (string path, bool simpleOriginalPath)
60 FullPath = Path.GetFullPath (path);
61 if (simpleOriginalPath)
62 OriginalPath = Path.GetFileName (path);
69 private DirectoryInfo (SerializationInfo info, StreamingContext context)
70 : base (info, context)
77 int len = FullPath.Length - 1;
78 if ((len > 1) && (FullPath [len] == Path.DirectorySeparatorChar))
80 int last = FullPath.LastIndexOf (Path.DirectorySeparatorChar, len);
81 if ((last == -1) || ((last == 0) && (len == 0))) {
85 current = FullPath.Substring (last + 1, len - last);
86 if (last == 0 && !Environment.IsRunningOnWindows)
87 parent = Path.DirectorySeparatorStr;
89 parent = FullPath.Substring (0, last);
90 // adjust for drives, i.e. a special case for windows
91 if (Environment.IsRunningOnWindows) {
92 if ((parent.Length == 2) && (parent [1] == ':') && Char.IsLetter (parent [0]))
93 parent += Path.DirectorySeparatorChar;
100 public override bool Exists {
104 if (stat.Attributes == MonoIO.InvalidFileAttributes)
107 if ((stat.Attributes & FileAttributes.Directory) == 0)
114 public override string Name {
115 get { return current; }
118 public DirectoryInfo Parent {
120 if ((parent == null) || (parent.Length == 0))
122 return new DirectoryInfo (parent);
126 public DirectoryInfo Root {
128 string root = Path.GetPathRoot (FullPath);
132 return new DirectoryInfo (root);
136 // creational methods
138 public void Create ()
140 Directory.CreateDirectory (FullPath);
143 public DirectoryInfo CreateSubdirectory (string path)
147 path = Path.Combine (FullPath, path);
148 Directory.CreateDirectory (path);
149 return new DirectoryInfo (path);
152 // directory listing methods
154 public FileInfo [] GetFiles ()
156 return GetFiles ("*");
159 public FileInfo [] GetFiles (string searchPattern)
161 if (searchPattern == null)
162 throw new ArgumentNullException ("searchPattern");
164 string [] names = Directory.GetFiles (FullPath, searchPattern);
166 FileInfo[] infos = new FileInfo [names.Length];
168 foreach (string name in names)
169 infos [i++] = new FileInfo (name);
174 public DirectoryInfo [] GetDirectories ()
176 return GetDirectories ("*");
179 public DirectoryInfo [] GetDirectories (string searchPattern)
181 if (searchPattern == null)
182 throw new ArgumentNullException ("searchPattern");
184 string [] names = Directory.GetDirectories (FullPath, searchPattern);
186 DirectoryInfo[] infos = new DirectoryInfo [names.Length];
188 foreach (string name in names)
189 infos [i++] = new DirectoryInfo (name);
194 public FileSystemInfo [] GetFileSystemInfos ()
196 return GetFileSystemInfos ("*");
199 public FileSystemInfo [] GetFileSystemInfos (string searchPattern)
201 return GetFileSystemInfos (searchPattern, SearchOption.TopDirectoryOnly);
207 FileSystemInfo [] GetFileSystemInfos (string searchPattern, SearchOption searchOption)
209 if (searchPattern == null)
210 throw new ArgumentNullException ("searchPattern");
211 if (searchOption != SearchOption.TopDirectoryOnly && searchOption != SearchOption.AllDirectories)
212 throw new ArgumentOutOfRangeException ("searchOption", "Must be TopDirectoryOnly or AllDirectories");
213 if (!Directory.Exists (FullPath))
214 throw new IOException ("Invalid directory");
216 List<FileSystemInfo> infos = new List<FileSystemInfo> ();
217 InternalGetFileSystemInfos (searchPattern, searchOption, infos);
218 return infos.ToArray ();
221 void InternalGetFileSystemInfos (string searchPattern, SearchOption searchOption, List<FileSystemInfo> infos)
223 // UnauthorizedAccessExceptions might happen here and break everything for SearchOption.AllDirectories
224 string [] dirs = Directory.GetDirectories (FullPath, searchPattern);
225 string [] files = Directory.GetFiles (FullPath, searchPattern);
227 Array.ForEach<string> (dirs, (dir) => { infos.Add (new DirectoryInfo (dir)); });
228 Array.ForEach<string> (files, (file) => { infos.Add (new FileInfo (file)); });
229 if (dirs.Length == 0 || searchOption == SearchOption.TopDirectoryOnly)
232 foreach (string dir in dirs) {
233 DirectoryInfo dinfo = new DirectoryInfo (dir);
234 dinfo.InternalGetFileSystemInfos (searchPattern, searchOption, infos);
238 // directory management methods
240 public override void Delete ()
245 public void Delete (bool recursive)
247 Directory.Delete (FullPath, recursive);
250 public void MoveTo (string destDirName)
252 if (destDirName == null)
253 throw new ArgumentNullException ("destDirName");
254 if (destDirName.Length == 0)
255 throw new ArgumentException ("An empty file name is not valid.", "destDirName");
257 Directory.Move (FullPath, Path.GetFullPath (destDirName));
260 public override string ToString ()
266 public DirectoryInfo[] GetDirectories (string searchPattern, SearchOption searchOption)
268 switch (searchOption) {
269 case SearchOption.TopDirectoryOnly:
270 return GetDirectories (searchPattern);
271 case SearchOption.AllDirectories:
272 Queue workq = new Queue(GetDirectories(searchPattern));
273 Queue doneq = new Queue();
274 while (workq.Count > 0)
276 DirectoryInfo cinfo = (DirectoryInfo) workq.Dequeue();
277 DirectoryInfo[] cinfoDirs = cinfo.GetDirectories(searchPattern);
278 foreach (DirectoryInfo i in cinfoDirs) workq.Enqueue(i);
279 doneq.Enqueue(cinfo);
282 DirectoryInfo[] infos = new DirectoryInfo[doneq.Count];
283 doneq.CopyTo(infos, 0);
286 string msg = Locale.GetText ("Invalid enum value '{0}' for '{1}'.", searchOption, "SearchOption");
287 throw new ArgumentOutOfRangeException ("searchOption", msg);
291 internal int GetFilesSubdirs (ArrayList l, string pattern)
294 FileInfo [] thisdir = null;
297 thisdir = GetFiles (pattern);
298 } catch (System.UnauthorizedAccessException){
302 count = thisdir.Length;
305 foreach (DirectoryInfo subdir in GetDirectories ()){
306 count += subdir.GetFilesSubdirs (l, pattern);
311 public FileInfo[] GetFiles (string searchPattern, SearchOption searchOption)
313 switch (searchOption) {
314 case SearchOption.TopDirectoryOnly:
315 return GetFiles (searchPattern);
316 case SearchOption.AllDirectories: {
317 ArrayList groups = new ArrayList ();
318 int count = GetFilesSubdirs (groups, searchPattern);
321 FileInfo [] all = new FileInfo [count];
322 foreach (FileInfo [] p in groups){
323 p.CopyTo (all, current);
329 string msg = Locale.GetText ("Invalid enum value '{0}' for '{1}'.", searchOption, "SearchOption");
330 throw new ArgumentOutOfRangeException ("searchOption", msg);
334 // access control methods
336 [MonoLimitation ("DirectorySecurity isn't implemented")]
337 public void Create (DirectorySecurity directorySecurity)
339 if (directorySecurity != null)
340 throw new UnauthorizedAccessException ();
344 [MonoLimitation ("DirectorySecurity isn't implemented")]
345 public DirectoryInfo CreateSubdirectory (string path, DirectorySecurity directorySecurity)
347 if (directorySecurity != null)
348 throw new UnauthorizedAccessException ();
349 return CreateSubdirectory (path);
352 [MonoNotSupported ("DirectorySecurity isn't implemented")]
353 public DirectorySecurity GetAccessControl ()
355 throw new UnauthorizedAccessException ();
358 [MonoNotSupported ("DirectorySecurity isn't implemented")]
359 public DirectorySecurity GetAccessControl (AccessControlSections includeSections)
361 throw new UnauthorizedAccessException ();
364 [MonoLimitation ("DirectorySecurity isn't implemented")]
365 public void SetAccessControl (DirectorySecurity directorySecurity)
367 if (directorySecurity != null)
368 throw new ArgumentNullException ("directorySecurity");
369 throw new UnauthorizedAccessException ();
373 #if NET_4_0 || MOONLIGHT
375 public IEnumerable<DirectoryInfo> EnumerateDirectories ()
377 return EnumerateDirectories ("*", SearchOption.TopDirectoryOnly);
380 public IEnumerable<DirectoryInfo> EnumerateDirectories (string searchPattern)
382 return EnumerateDirectories (searchPattern, SearchOption.TopDirectoryOnly);
385 public IEnumerable<DirectoryInfo> EnumerateDirectories (string searchPattern, SearchOption searchOption)
387 foreach (string name in Directory.EnumerateDirectories (FullPath, searchPattern, searchOption))
388 yield return new DirectoryInfo (name);
391 public IEnumerable<FileInfo> EnumerateFiles ()
393 return EnumerateFiles ("*", SearchOption.TopDirectoryOnly);
396 public IEnumerable<FileInfo> EnumerateFiles (string searchPattern)
398 return EnumerateFiles (searchPattern, SearchOption.TopDirectoryOnly);
401 public IEnumerable<FileInfo> EnumerateFiles (string searchPattern, SearchOption searchOption)
403 foreach (string name in Directory.EnumerateFiles (FullPath, searchPattern, searchOption))
404 yield return new FileInfo (name);
407 public IEnumerable<FileSystemInfo> EnumerateFileSystemInfos ()
409 return EnumerateFileSystemInfos ("*", SearchOption.TopDirectoryOnly);
412 public IEnumerable<FileSystemInfo> EnumerateFileSystemInfos (string searchPattern)
414 return EnumerateFileSystemInfos (searchPattern, SearchOption.TopDirectoryOnly);
417 public IEnumerable<FileSystemInfo> EnumerateFileSystemInfos (string searchPattern, SearchOption searchOption)
419 if (searchPattern == null)
420 throw new ArgumentNullException ("searchPattern");
421 if (searchOption != SearchOption.TopDirectoryOnly && searchOption != SearchOption.AllDirectories)
422 throw new ArgumentOutOfRangeException ("searchoption");
424 return EnumerateFileSystemInfos (FullPath, searchPattern, searchOption);
427 static internal IEnumerable<FileSystemInfo> EnumerateFileSystemInfos (string full, string searchPattern, SearchOption searchOption)
429 string path_with_pattern = Path.Combine (full, searchPattern);
432 FileAttributes rattr;
433 bool subdirs = searchOption == SearchOption.AllDirectories;
435 string s = MonoIO.FindFirst (full, path_with_pattern, out rattr, out error, out handle);
439 throw MonoIO.GetException (Path.GetDirectoryName (path_with_pattern), (MonoIOError) error);
442 if (((rattr & FileAttributes.ReparsePoint) == 0)){
443 if ((rattr & FileAttributes.Directory) != 0)
444 yield return new DirectoryInfo (s);
446 yield return new FileInfo (s);
449 while ((s = MonoIO.FindNext (handle, out rattr, out error)) != null){
450 if ((rattr & FileAttributes.ReparsePoint) != 0)
452 if ((rattr & FileAttributes.Directory) != 0)
453 yield return new DirectoryInfo (s);
455 yield return new FileInfo (s);
457 if (((rattr & FileAttributes.Directory) != 0) && subdirs)
458 foreach (FileSystemInfo child in EnumerateFileSystemInfos (s, searchPattern, searchOption))
462 MonoIO.FindClose (handle);