c# could not get part of file path error - c#
I'm attempting to recursively copy all files and directories and here is my class code. I am getting could not find part of path. I'm going to assume it may be because there is about 20 sub directories to directories (paths like \game\resour.zip\shared\effects\materials\ and such.) Any help would be appreciated.
static IEnumerable<string> GetFiles(string path)
{
Queue<string> queue = new Queue<string>();
queue.Enqueue(path);
while (queue.Count > 0)
{
path = queue.Dequeue();
try
{
foreach (string subDir in Directory.GetDirectories(path))
{
queue.Enqueue(subDir);
}
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
string[] files = null;
try
{
files = Directory.GetFiles(path);
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
if (files != null)
{
for (int i = 0; i < files.Length; i++)
{
yield return files[i];
}
}
}
}
In .NET the full path to a file can not be longer than 260 characters, if you wan't to use longer paths you must call the Win API directly using UNC paths and your file paths can be 32,767 characters long, I have added a .NET class that does this, I did not write this code originally but sadly I did not find the original article when looking for it today so I will just enter my modified code here for demonstration purposes, the code is slightly altered from the original code and has some extra functionalty.
/*
All sample code is provided by the Inge Henriksen for illustrative purposes only. These examples have not been thoroughly tested under all conditions. Inge Henriksen, therefore, cannot guarantee or imply reliability, serviceability, or function of these programs.
*/
using System;
using System.Collections.Generic;
using System.Text;
using Microsoft.Win32.SafeHandles;
using System.Runtime.InteropServices;
using System.IO;
using log4net;
namespace MyStuff.DAL.Services.IO.File
{
/// <summary>
/// Class for communicating with the Windows kernel library for low-level disk access.
/// The main purpose of this class is to allow for longer file paths than System.IO.File,
/// this class handles file paths up to 32,767 characters.
/// Note: Always be careful when accessing this class from a managed multi-threaded application
/// as the unmanaged Windows kernel is different, this "crash" causes application unstability
/// if not handled properly.
/// TODO: Look into if there are any significant gains on 64-bit systems using another kind of
/// core component than kernel32.dll.
/// </summary>
public class Win32File
{
ILog Log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
#region DLLImport's
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
static extern bool MoveFile(string lpExistingFileName, string lpNewFileName);
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
static extern bool DeleteFile(string lpFileName);
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
static extern bool CreateDirectoryW(string lpPathName, IntPtr lpSecurityAttributes);
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
static extern SafeFileHandle CreateFileW(string lpFileName, uint dwDesiredAccess,
uint dwShareMode, IntPtr lpSecurityAttributes, uint dwCreationDisposition,
uint dwFlagsAndAttributes, IntPtr hTemplateFile);
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
static extern uint SetFilePointer(SafeFileHandle hFile, long lDistanceToMove, IntPtr lpDistanceToMoveHigh, uint dwMoveMethod);
#endregion
// uint GetMode( FileMode mode )
// Converts the filemode constant to win32 constant
#region GetMode
private uint GetMode(FileMode mode)
{
Log.Debug("Get Win32 file mode");
uint umode = 0;
switch (mode)
{
case FileMode.CreateNew:
umode = Win32FileAttributes.CREATE_NEW;
break;
case FileMode.Create:
umode = Win32FileAttributes.CREATE_ALWAYS;
break;
case FileMode.Append:
umode = Win32FileAttributes.OPEN_ALWAYS;
break;
case FileMode.Open:
umode = Win32FileAttributes.OPEN_EXISTING;
break;
case FileMode.OpenOrCreate:
umode = Win32FileAttributes.OPEN_ALWAYS;
break;
case FileMode.Truncate:
umode = Win32FileAttributes.TRUNCATE_EXISTING;
break;
}
return umode;
}
#endregion
// uint GetAccess(FileAccess access)
// Converts the FileAccess constant to win32 constant
#region GetAccess
private uint GetAccess(FileAccess access)
{
Log.Debug("Get Win32 file access");
uint uaccess = 0;
switch (access)
{
case FileAccess.Read:
uaccess = Win32FileAttributes.GENERIC_READ;
break;
case FileAccess.ReadWrite:
uaccess = Win32FileAttributes.GENERIC_READ | Win32FileAttributes.GENERIC_WRITE;
break;
case FileAccess.Write:
uaccess = Win32FileAttributes.GENERIC_WRITE;
break;
}
return uaccess;
}
#endregion
// uint GetShare(FileShare share)
// Converts the FileShare constant to win32 constant
#region GetShare
private uint GetShare(FileShare share)
{
Log.Debug("Get Win32 file share");
uint ushare = 0;
switch (share)
{
case FileShare.Read:
ushare = Win32FileAttributes.FILE_SHARE_READ;
break;
case FileShare.ReadWrite:
ushare = Win32FileAttributes.FILE_SHARE_READ | Win32FileAttributes.FILE_SHARE_WRITE;
break;
case FileShare.Write:
ushare = Win32FileAttributes.FILE_SHARE_WRITE;
break;
case FileShare.Delete:
ushare = Win32FileAttributes.FILE_SHARE_DELETE;
break;
case FileShare.None:
ushare = 0;
break;
}
return ushare;
}
#endregion
public bool Move(string existingFile, string newFile)
{
Log.Debug(String.Format("Rename the file \"{0}\" to \"{1}\"", existingFile, newFile) );
return Win32File.MoveFile(existingFile, newFile);
}
public bool CreateDirectory(string filepath)
{
Log.Debug(String.Format("Create the directory \"{0}\"", filepath));
// If file path is disk file path then prepend it with \\?\
// if file path is UNC prepend it with \\?\UNC\ and remove \\ prefix in unc path.
if (filepath.StartsWith(#"\\"))
filepath = #"\\?\UNC\" + filepath.Substring(2, filepath.Length - 2);
else
filepath = #"\\?\" + filepath;
return CreateDirectoryW(filepath, IntPtr.Zero);
}
public FileStream Open(string filepath, FileMode mode, uint uaccess)
{
Log.Debug(String.Format("Open the file \"{0}\"", filepath));
//opened in the specified mode and path, with read/write access and not shared
FileStream fs = null;
uint umode = GetMode(mode);
uint ushare = 0; // Not shared
if (mode == FileMode.Append) uaccess = Win32FileAttributes.FILE_APPEND_DATA;
// If file path is disk file path then prepend it with \\?\
// if file path is UNC prepend it with \\?\UNC\ and remove \\ prefix in unc path.
if (filepath.StartsWith(#"\\"))
filepath = #"\\?\UNC\" + filepath.Substring(2, filepath.Length - 2);
else filepath = #"\\?\" + filepath;
SafeFileHandle sh = CreateFileW(filepath, uaccess, ushare, IntPtr.Zero, umode, Win32FileAttributes.FILE_ATTRIBUTE_NORMAL, IntPtr.Zero);
int iError = Marshal.GetLastWin32Error();
if ((iError > 0 && !(mode == FileMode.Append && iError == Win32FileAttributes.ERROR_ALREADY_EXISTS)) || sh.IsInvalid)
{
Log.Error(String.Format("Error opening file; Win32 Error: {0}", iError));
throw new Exception("Error opening file; Win32 Error:" + iError);
}
else
{
fs = new FileStream(sh, FileAccess.ReadWrite);
}
// if opened in append mode
if (mode == FileMode.Append)
{
if (!sh.IsInvalid)
{
SetFilePointer(sh, 0, IntPtr.Zero, Win32FileAttributes.FILE_END);
}
}
Log.Debug(String.Format("The file \"{0}\" is now open", filepath));
return fs;
}
public FileStream Open(string filepath, FileMode mode, FileAccess access)
{
Log.Debug(String.Format("Open the file \"{0}\"", filepath));
//opened in the specified mode and access and not shared
FileStream fs = null;
uint umode = GetMode(mode);
uint uaccess = GetAccess(access);
uint ushare = 0; // Exclusive lock of the file
if (mode == FileMode.Append) uaccess = Win32FileAttributes.FILE_APPEND_DATA;
// If file path is disk file path then prepend it with \\?\
// if file path is UNC prepend it with \\?\UNC\ and remove \\ prefix in unc path.
if (filepath.StartsWith(#"\\"))
filepath = #"\\?\UNC\" + filepath.Substring(2, filepath.Length - 2);
else
filepath = #"\\?\" + filepath;
SafeFileHandle sh = CreateFileW(filepath, uaccess, ushare, IntPtr.Zero, umode, Win32FileAttributes.FILE_ATTRIBUTE_NORMAL, IntPtr.Zero);
int iError = Marshal.GetLastWin32Error();
if ((iError > 0 && !(mode == FileMode.Append && iError != Win32FileAttributes.ERROR_ALREADY_EXISTS)) || sh.IsInvalid)
{
Log.Error(String.Format("Error opening file; Win32 Error: {0}", iError));
throw new Exception("Error opening file; Win32 Error:" + iError);
}
else
{
fs = new FileStream(sh, access);
}
// if opened in append mode
if (mode == FileMode.Append)
{
if (!sh.IsInvalid)
{
SetFilePointer(sh, 0, IntPtr.Zero, Win32FileAttributes.FILE_END);
}
}
Log.Debug(String.Format("The file \"{0}\" is now open", filepath));
return fs;
}
public FileStream Open(string filepath, FileMode mode, FileAccess access, FileShare share)
{
//opened in the specified mode , access and share
FileStream fs = null;
uint umode = GetMode(mode);
uint uaccess = GetAccess(access);
uint ushare = GetShare(share);
if (mode == FileMode.Append) uaccess = Win32FileAttributes.FILE_APPEND_DATA;
// If file path is disk file path then prepend it with \\?\
// if file path is UNC prepend it with \\?\UNC\ and remove \\ prefix in unc path.
if (filepath.StartsWith(#"\\"))
filepath = #"\\?\UNC\" + filepath.Substring(2, filepath.Length - 2);
else
filepath = #"\\?\" + filepath;
SafeFileHandle sh = CreateFileW(filepath, uaccess, ushare, IntPtr.Zero, umode, Win32FileAttributes.FILE_ATTRIBUTE_NORMAL, IntPtr.Zero);
int iError = Marshal.GetLastWin32Error();
if ((iError > 0 && !(mode == FileMode.Append && iError != Win32FileAttributes.ERROR_ALREADY_EXISTS)) || sh.IsInvalid)
{
throw new Exception("Error opening file Win32 Error:" + iError);
}
else
{
fs = new FileStream(sh, access);
}
// if opened in append mode
if (mode == FileMode.Append)
{
if (!sh.IsInvalid)
{
SetFilePointer(sh, 0, IntPtr.Zero, Win32FileAttributes.FILE_END);
}
}
return fs;
}
public FileStream OpenRead(string filepath)
{
Log.Debug(String.Format("Open the file \"{0}\"", filepath));
// Open readonly file mode open(String, FileMode.Open, FileAccess.Read, FileShare.Read)
return Open(filepath, FileMode.Open, FileAccess.Read, FileShare.Read);
}
public FileStream OpenWrite(string filepath)
{
Log.Debug(String.Format("Open the file \"{0}\" for writing", filepath));
//open writable open(String, FileMode.OpenOrCreate, FileAccess.Write, FileShare.None).
return Open(filepath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.None);
}
public bool Delete(string filepath)
{
Log.Debug(String.Format("Delete the file \"{0}\"", filepath));
// If file path is disk file path then prepend it with \\?\
// if file path is UNC prepend it with \\?\UNC\ and remove \\ prefix in unc path.
if (filepath.StartsWith(#"\\"))
filepath = #"\\?\UNC\" + filepath.Substring(2, filepath.Length - 2);
else
filepath = #"\\?\" + filepath;
Log.Debug(String.Format("The file \"{0}\" has been deleted", filepath));
return DeleteFile(filepath);
}
}
}
to copy recursively use this method
static public void CopyFolder( string sourceFolder, string destFolder )
{
if (!Directory.Exists( destFolder ))
Directory.CreateDirectory( destFolder );
string[] files = Directory.GetFiles( sourceFolder );
foreach (string file in files)
{
string name = Path.GetFileName( file );
string dest = Path.Combine( destFolder, name );
File.Copy( file, dest );
}
string[] folders = Directory.GetDirectories( sourceFolder );
foreach (string folder in folders)
{
string name = Path.GetFileName( folder );
string dest = Path.Combine( destFolder, name );
CopyFolder( folder, dest );
}
}
Not sure if this will be of assistance:
/// <summary>
/// Operates just like GetFileSystemInfos except it is recursive, operates on more than
/// one FileSystemInfo, ensures no paths are duplicateed, and includes the original
/// items in the return.
/// </summary>
/// <param name="files">A collection of files and/or directories to expand.</param>
/// <returns>A Dictionary of all file, directories, and subfiles and subdirectories of
/// those directories and subdirectories. The key is the fullpath, and the value
/// is the FileSystemInfo.</returns>
public static Dictionary<string, FileSystemInfo> ExpandFileSystemInfos(FileSystemInfo[] files)
{
Dictionary<string, FileSystemInfo> resultingFiles = new Dictionary<string,FileSystemInfo>();
//GetFileSystemInfosRecursively will expand everything, except it may contain duplicates
foreach (FileSystemInfo file in GetFileSystemInfosRecursively(files))
{//so we just go through adding them to the Dictionary
if (!resultingFiles.ContainsKey(file.FullName))
{
resultingFiles.Add(file.FullName, file);
}
}
return resultingFiles;
}
//helper method for ExpandFileSystemInfos
private static List<FileSystemInfo> GetFileSystemInfosRecursively(FileSystemInfo[] files)
{
List<FileSystemInfo> resultingFiles = new List<FileSystemInfo>();
foreach (FileSystemInfo file in files)
{
if (file is DirectoryInfo)
{
//get its sub items and pass to another function to process
DirectoryInfo dir = (DirectoryInfo)file;
//recursive call, passing in subdirectories and files of current directory.
//The result returned will be all subdirectories and files nested within the current
//dir. So we add them to our collection of results for each directory we encounter.
resultingFiles.AddRange(GetFileSystemInfosRecursively(dir.GetFileSystemInfos()));
}
//else do nothing, it is a file we already know about
}
//resultingFiles now contains all sub items
//It does not include the directories and files in files however, so we add them as well
resultingFiles.AddRange(files);
return resultingFiles;
}
I used this with a program where the user could choose multiple directories or files to be compressed. I had to handle the case where they explicitely added a directory, and then added a subdirectory/subfile(pointlessly) of that directory. So I wrote this so that you could pass in multiple directory and files in an array, and as it expands them out it will make sure none are duplicated.
Related
QueryFullProcessImageName returns error 31
I'm working on an application that should be able to show and retrieve information from all processes in the system. I try to get the process full path using the function QueryFullProcessImageName but it fails with error 31 (A device attached to the system is not functioning), this happens only with new processes created after the program's first enumeration of the running processes. This is the method that I use to get the full path: public static string GetProcessFullPath(SafeProcessHandle Handle) { uint MaxPathLength = 1024; StringBuilder ExePath = new StringBuilder(1024); if (NativeMethods.Win32ProcessFunctions.QueryFullProcessImageName(Handle.DangerousGetHandle(), 0, ExePath, ref MaxPathLength)) { return ExePath.ToString(); } else { Win32Exception ex = new Win32Exception(Marshal.GetLastWin32Error()); Logger.WriteEntry(new LogEntry("Non è stato possibile recuperare il percorso completo dell'eseguibile di un processo, codice di errore: " + ex.NativeErrorCode + " (" + ex.Message + ")", EventSource.WindowsAPI, EventSeverity.Error)); return "Non disponibile"; } } This is the declaration of the function: [DllImport("Kernel32.dll", EntryPoint = "QueryFullProcessImageNameW", SetLastError = true, CharSet = CharSet.Unicode)] [return: MarshalAs(UnmanagedType.Bool)] public static extern bool QueryFullProcessImageName(IntPtr ProcessHandle, uint PathType, StringBuilder ExePath, ref uint Characters); I don't know what could be causing this error, I get notifications of new processes from WMI (Win32_ProcessStartTrace). This is how I get the process handle that is used in the GetProcessFullPath method (the PID is from the WMI notification): public static SafeProcessHandle GetProcessHandle(uint ProcessID) { IntPtr UnsafeProcessHandle = NativeMethods.Win32ProcessFunctions.OpenProcess(NativeMethods.Win32Enumerations.ProcessAccessRights.PROCESS_ALL_ACCESS, false, ProcessID); if (UnsafeProcessHandle == IntPtr.Zero) { UnsafeProcessHandle = NativeMethods.Win32ProcessFunctions.OpenProcess(NativeMethods.Win32Enumerations.ProcessAccessRights.PROCESS_QUERY_LIMITED_INFORMATION, false, ProcessID); if (UnsafeProcessHandle == IntPtr.Zero) { Win32Exception ex = new Win32Exception(Marshal.GetLastWin32Error()); Logger.WriteEntry(new LogEntry("Non è stato possibile aprire un processo, codice di errore: " + ex.NativeErrorCode + " (" + ex.Message + ")", EventSource.WindowsAPI, EventSeverity.Error)); return new SafeProcessHandle(UnsafeProcessHandle, true); } else { return new SafeProcessHandle(UnsafeProcessHandle, true); } } else { return new SafeProcessHandle(UnsafeProcessHandle, true); } } Edit: Further testing indicated that the problem occurs even if the process whose full path I'm trying to get was already started when the function is called.
GetProcessImageFileName works but it returns the path in device form and I need a Win32 path. You can use GetLogicalDriveStrings and QueryDosDevice convert dos file path to windows file path. Some code: // dos file path => windows file path BOOL DosPathToNtPath(LPTSTR pszDosPath, LPTSTR pszNtPath) { TCHAR szDriveStr[500]; TCHAR szDrive[3]; TCHAR szDevName[100]; INT cchDevName; INT i; //Check the parameters if (!pszDosPath || !pszNtPath) return FALSE; //Get local disk string if (GetLogicalDriveStrings(sizeof(szDriveStr), szDriveStr)) { for (i = 0; szDriveStr[i]; i += 4) { if (!lstrcmpi(&(szDriveStr[i]), _T("A:\\")) || !lstrcmpi(&(szDriveStr[i]), _T("B:\\"))) { continue; } szDrive[0] = szDriveStr[i]; szDrive[1] = szDriveStr[i + 1]; szDrive[2] = '\0'; // Query Dos device name if (!QueryDosDevice(szDrive, szDevName, 100)) { return FALSE; } // Hit cchDevName = lstrlen(szDevName); if (_tcsnicmp(pszDosPath, szDevName, cchDevName) == 0) { // Copy drive lstrcpy(pszNtPath, szDrive); // Copy path lstrcat(pszNtPath, pszDosPath + cchDevName); return TRUE; } } } lstrcpy(pszNtPath, pszDosPath); return FALSE; } Then simply call, GetProcessImageFileName(hProcess, szImagePath, MAX_PATH); DosPathToNtPath(szImagePath,pszFullPath); You can convert it to C# code.
The QueryFullProcessImageName function fails with ERROR_GEN_FAILURE (31 or 0x1f, "A device attached to the system is not functioning") if the process is a "zombie" process, i.e. the process terminated but not all handles to it were closed. In this case, you can still use QueryFullProcessImageNameW with the PROCESS_NAME_NATIVE flag to get the native path, but you probably just want to skip it since it's no longer running.
How to fast delete many files
I have a folder in Windows Server with subfolders and ≈50000 files. When I click the right mouse button and choose delete (or shift+delete) – all files are deleted in 10-20 seconds. When I delete files using code – 1500-4000 seconds. Delete large number of files – don't work for me. My code: string folderPath = #"C://myFolder"; DirectoryInfo folderInfo = new DirectoryInfo(folderPath); folderInfo.Delete(true); // true - recursive, with sub-folders How to delete files faster?
A much faster way to delete files is to use the Windows functions instead of the .NET ones. You will need to first import the function: [DllImport("kernel32.dll", SetLastError = true)] [return: MarshalAs(UnmanagedType.Bool)] static extern bool DeleteFile(string lpFileName); And then you can do this: string[] files = Directory.EnumerateFiles(path, "*". SearchOption.AllDirectories); foreach (string file in files) { DeleteFile(file); } Once the files are deleted, which is the slowest part by using the managed APIs, you can call Directory.DeleteFolder(path, true) to delete the empty folders.
Since the question is actually about deleting network shared folders and it's stated that the explorer based delete is much faster than the C# internal delete mechanism, it might help to just invoke a windows shell based delete. ProcessStartInfo Info = new ProcessStartInfo(); Info.Arguments = "/C rd /s /q \"<your-path>\""; Info.WindowStyle = ProcessWindowStyle.Hidden; Info.CreateNoWindow = true; Info.FileName = "cmd.exe"; Process.Start(Info); Ofcourse, you have to replace <your-path>. However, I don't have the infrastructure and files available to test the performance myself right now.
Not quite sure why the method DirectoryInfo.Delete() takes too much time when deleting folders that have a lot of files and sub-folders. I suspect that the method may also do quite a few things that are unnecessary. I write a small class to to use Win API without doing too many unnecessary things to test my idea. It takes about 40 seconds to delete a folder that have 50,000 files and sub-folders. So, hope it helps. I use this PowerScript to generate the testing files. $folder = "d:\test1"; For ($i=0; $i -lt 50000; $i++) { New-Item -Path $folder -Name "test$i.txt" -ItemType "file" -Value $i.ToString(); } The following is the code in C#. using System; using System.Collections.Generic; // using System.Runtime.InteropServices; using System.IO; // namespace TestFileDelete { class FileDelete { [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)] struct WIN32_FIND_DATAW { public FileAttributes dwFileAttributes; public System.Runtime.InteropServices.ComTypes.FILETIME ftCreationTime; public System.Runtime.InteropServices.ComTypes.FILETIME ftLastAccessTime; public System.Runtime.InteropServices.ComTypes.FILETIME ftLastWriteTime; public UInt32 nFileSizeHigh; // DWORD public UInt32 nFileSizeLow; // DWORD public UInt32 dwReserved0; // DWORD public UInt32 dwReserved1; // DWORD [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 260)] public String cFileName; [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 14)] public String cAlternateFileName; }; static readonly IntPtr INVALID_HANDLE_VALUE = new IntPtr(-1); [DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] private static extern IntPtr FindFirstFileW(String lpFileName, out WIN32_FIND_DATAW lpFindFileData); [DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] private static extern Boolean FindNextFileW(IntPtr hFindFile, out WIN32_FIND_DATAW lpFindFileData); [DllImport("kernel32.dll")] private static extern Boolean FindClose(IntPtr handle); [DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] public static extern Boolean DeleteFileW(String lpFileName); // Deletes an existing file [DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)] private static extern Boolean RemoveDirectoryW(String lpPathName); // Deletes an existing empty directory // This method check to see if the given folder is empty or not. public static Boolean IsEmptyFolder(String folder) { Boolean res = true; if (folder == null && folder.Length == 0) { throw new Exception(folder + "is invalid"); } WIN32_FIND_DATAW findFileData; String searchFiles = folder + #"\*.*"; IntPtr searchHandle = FindFirstFileW(searchFiles, out findFileData); if (searchHandle == INVALID_HANDLE_VALUE) { throw new Exception("Cannot check folder " + folder); } do { if ((findFileData.dwFileAttributes & FileAttributes.Directory) == FileAttributes.Directory) { // found a sub folder if (findFileData.cFileName != "." && findFileData.cFileName != "..") { res = false; break; } } // if ((findFileData.dwFileAttributes & FileAttributes.Directory) == FileAttributes.Directory) else { // found a file res = false; break; } } while (FindNextFileW(searchHandle, out findFileData)); FindClose(searchHandle); return res; } // public static Boolean IsEmptyFolder(String folder) // This method deletes the given folder public static Boolean DeleteFolder(String folder) { Boolean res = true; // keep non-empty folders to delete later (after we delete everything inside) Stack<String> nonEmptyFolder = new Stack<String>(); String currentFolder = folder; do { Boolean isEmpty = false; try { isEmpty = IsEmptyFolder(currentFolder); } catch (Exception ex) { // Something wrong res = false; break; } if (!isEmpty) { nonEmptyFolder.Push(currentFolder); WIN32_FIND_DATAW findFileData; IntPtr searchHandle = FindFirstFileW(currentFolder + #"\*.*", out findFileData); if (searchHandle != INVALID_HANDLE_VALUE) { do { // for each folder, find all of its sub folders and files String foundPath = currentFolder + #"\" + findFileData.cFileName; if ((findFileData.dwFileAttributes & FileAttributes.Directory) == FileAttributes.Directory) { // found a sub folder if (findFileData.cFileName != "." && findFileData.cFileName != "..") { if (IsEmptyFolder(foundPath)) { // found an empty folder, delete it if (!(res = RemoveDirectoryW(foundPath))) { Int32 error = Marshal.GetLastWin32Error(); break; } } else { // found a non-empty folder nonEmptyFolder.Push(foundPath); } } // if (findFileData.cFileName != "." && findFileData.cFileName != "..") } // if ((findFileData.dwFileAttributes & FileAttributes.Directory) == FileAttributes.Directory) else { // found a file, delete it if (!(res = DeleteFileW(foundPath))) { Int32 error = Marshal.GetLastWin32Error(); break; } } } while (FindNextFileW(searchHandle, out findFileData)); FindClose(searchHandle); } // if (searchHandle != INVALID_HANDLE_VALUE) }// if (!IsEmptyFolder(folder)) else { if (!(res = RemoveDirectoryW(currentFolder))) { Int32 error = Marshal.GetLastWin32Error(); break; } } if (nonEmptyFolder.Count > 0) { currentFolder = nonEmptyFolder.Pop(); } else { currentFolder = null; } } while (currentFolder != null && res); return res; } // public static Boolean DeleteFolder(String folder) }; class Program { static void Main(string[] args) { DateTime t1 = DateTime.Now; try { Boolean b = FileDelete.DeleteFolder(#"d:\test1"); } catch (Exception ex) { Console.WriteLine(ex.Message); } DateTime t2 = DateTime.Now; TimeSpan ts = t2 - t1; Console.WriteLine(ts.Seconds); } } }
Any ideas why folders are extracted as file name when using Archive Utility?
I'm using ICSharpCode.SharpZipLib.Zip to pack files for my clients to download their files. However when using Archive Utility on Mac, the extracted files are all in a same folder with the original file path as folder name. But if using other program to unzip with no problems. Any ideas? Code: using System; using System.IO; using ICSharpCode.SharpZipLib.Checksums; using ICSharpCode.SharpZipLib.Zip; using ICSharpCode.SharpZipLib.GZip; using System.Runtime.InteropServices; public class ZipClass { /// <summary> /// Check file writing status /// </summary> /// <param name="lpPathName"></param> /// <param name="iReadWrite"></param> /// <returns></returns> [DllImport("kernel32.dll")] public static extern IntPtr _lopen(string lpPathName, int iReadWrite); [DllImport("kernel32.dll")] public static extern bool CloseHandle(IntPtr hObject); public const int OF_READWRITE = 2; public const int OF_SHARE_DENY_NONE = 0x40; public readonly IntPtr HFILE_ERROR = new IntPtr(-1); public bool FileOccupyStatus(string vFileName) { IntPtr vHandle = _lopen(vFileName, OF_READWRITE | OF_SHARE_DENY_NONE); if (vHandle == HFILE_ERROR) { //MessageBox.Show("File being used"); return false; } CloseHandle(vHandle); //MessageBox.Show("File free"); return true; } ZipOutputStream zos = null; string strBaseDir = ""; public bool ZipFolder(string folder, string zipFile) { zos = new ZipOutputStream(File.Create(zipFile)); zos.Password = ""; strBaseDir = folder + "\\"; addZipEntry(strBaseDir); //ZipFileDictory(strBaseDir, zos, ""); zos.Finish(); zos.Close(); return true; } void addZipEntry(string PathStr) { DirectoryInfo dir = new DirectoryInfo(PathStr); foreach (DirectoryInfo folder in dir.GetDirectories()) { addZipEntry(folder.FullName + "\\"); } foreach (FileInfo item in dir.GetFiles()) { if (FileOccupyStatus(item.FullName.ToString())) { if (item.Extension.ToLower() == ".pdf") { continue; } FileStream fs = File.OpenRead(item.FullName); byte[] buffer = new byte[fs.Length]; fs.Read(buffer, 0, buffer.Length); string strEntryName = item.FullName.Replace(strBaseDir, ""); ZipEntry entry = new ZipEntry(strEntryName); entry.IsUnicodeText = true; entry.Size = fs.Length; zos.PutNextEntry(entry); zos.Write(buffer, 0, buffer.Length); fs.Close(); } } } } Call function from webpage: .... string folder = dr["folder"].ToString().Replace("/", "\\"); ZipClass zip = new ZipClass(); folder = Server.MapPath("/data") + "\\" + folder; try { zip.ZipFolder(folder, folder + "\\media.zip"); Response.Write("{\"status\":\"true\",\"file\":\"/" + dr["zip"] + "\"}"); } catch {...}
Replace backwards slashes with forward slashes in the entry name. Check this question for example: How do you add a folder to a zip archive with ICSharpCode.SharpZipLib
An object reference is required for the nonstatic field, method, or property
Hi I am trying to writing a code which copy files from source folder to destination folder. If destination folder contains same file name then my program should store file with different name. e.g. Source Folder contains: C:\test\ test1.txt test2.txt Destination folder contains D:\test\ test1.txt test2.txt test3.txt then copy action should copy test1.txt and test2.txt from source to destination folder with name changed to test4.txt and test5.txt This is not complete code. But I am getting error An Object reference is required for nonstatic field,method, or property. at getFileName( ref destfileName, ref targetPath). Any help on this? class Program { static void Main(string[] args) { string sourcefileName = null; string destfileName = null; string sourcePath = #"C:\test"; string targetPath = #"D:\test"; List<int> seqNum = new List<int>(); // To copy a folder's contents to a new location: // Create a new target folder, if necessary. if (!System.IO.Directory.Exists(targetPath)) { System.IO.Directory.CreateDirectory(targetPath); } // To copy all the files in one directory to another directory. // Get the files in the source folder. (To recursively iterate through // all subfolders under the current directory, see // "How to: Iterate Through a Directory Tree.") // Note: Check for target path was performed previously // in this code example. if (System.IO.Directory.Exists(sourcePath)) { string[] files = System.IO.Directory.GetFiles(sourcePath); // Copy the files and overwrite destination files if they already exist. foreach (string s in files) { // Use static Path methods to extract only the file name from the path. //File name is like text1.txt sourcefileName = System.IO.Path.GetFileName(s); if (System.IO.Directory.GetFiles(targetPath).Count() > 0) { foreach (string file in System.IO.Directory.GetFiles(targetPath)) { if (file.Contains(sourcefileName)) { int num; string existingLatestFile = string.Empty; destfileName = sourcefileName.Replace(".txt", string.Empty); for (int i = 0; i < sourcefileName.Length; i++) { if (Char.IsDigit(sourcefileName[i])) { existingLatestFile += sourcefileName[i]; } } if (int.TryParse(existingLatestFile, out num)) { seqNum.Add(num); } destfileName = destfileName.Replace(existingLatestFile, string.Empty);//Remove existing number num = num + 1; destfileName = destfileName + num.ToString() + ".txt"; // Make a new file name while (!getFileName( ref destfileName, ref targetPath)) { } } else { destfileName = sourcefileName; } string destFile = System.IO.Path.Combine(targetPath, destfileName); System.IO.File.Copy(s, destFile, false); } } } } else { Console.WriteLine("Source path does not exist!"); } if (System.IO.Directory.GetFiles(targetPath).Count() > 0) { foreach (string file in System.IO.Directory.GetFiles(targetPath)) { /* if (file.Contains(dir + "\\" + filename)) { int num; existingLatestFile = file.Replace(dir + "\\" + filename, string.Empty); existingLatestFile = existingLatestFile.Replace(".txt", string.Empty); if (int.TryParse(existingLatestFile, out num)) { seqNum.Add(num); } }*/ Console.WriteLine(file); } } // Keep console window open in debug mode. Console.WriteLine("Press any key to exit."); Console.ReadKey(); } bool getFileName(ref string filename, ref string destFolder) { bool retValue =false; foreach (string file in System.IO.Directory.GetFiles(destFolder)) { if (file.Contains(filename)) { retValue = false; } else { retValue = true; } } return retValue; } }
Main() is a static method. It is not associated with any instance. You need to make the other method static as well.
main is a static method, to call the non-static method getFileName you need to create a instance first change while (!getFileName... to Program p = new Program(); while (!p.getFileName...
Copy the entire contents of a directory in C#
I want to copy the entire contents of a directory from one location to another in C#. There doesn't appear to be a way to do this using System.IO classes without lots of recursion. There is a method in VB that we can use if we add a reference to Microsoft.VisualBasic: new Microsoft.VisualBasic.Devices.Computer(). FileSystem.CopyDirectory( sourceFolder, outputFolder ); This seems like a rather ugly hack. Is there a better way?
Much easier private static void CopyFilesRecursively(string sourcePath, string targetPath) { //Now Create all of the directories foreach (string dirPath in Directory.GetDirectories(sourcePath, "*", SearchOption.AllDirectories)) { Directory.CreateDirectory(dirPath.Replace(sourcePath, targetPath)); } //Copy all the files & Replaces any files with the same name foreach (string newPath in Directory.GetFiles(sourcePath, "*.*",SearchOption.AllDirectories)) { File.Copy(newPath, newPath.Replace(sourcePath, targetPath), true); } }
Hmm, I think I misunderstand the question but I'm going to risk it. What's wrong with the following straightforward method? public static void CopyFilesRecursively(DirectoryInfo source, DirectoryInfo target) { foreach (DirectoryInfo dir in source.GetDirectories()) CopyFilesRecursively(dir, target.CreateSubdirectory(dir.Name)); foreach (FileInfo file in source.GetFiles()) file.CopyTo(Path.Combine(target.FullName, file.Name)); } EDIT Since this posting has garnered an impressive number of downvotes for such a simple answer to an equally simple question, let me add an explanation. Please read this before downvoting. First of all, this code is not intendend as a drop-in replacement to the code in the question. It is for illustration purpose only. Microsoft.VisualBasic.Devices.Computer.FileSystem.CopyDirectory does some additional correctness tests (e.g. whether the source and target are valid directories, whether the source is a parent of the target etc.) that are missing from this answer. That code is probably also more optimized. That said, the code works well. It has (almost identically) been used in a mature software for years. Apart from the inherent fickleness present with all IO handlings (e.g. what happens if the user manually unplugs the USB drive while your code is writing to it?), there are no known problems. In particular, I’d like to point out that the use of recursion here is absolutely not a problem. Neither in theory (conceptually, it’s the most elegant solution) nor in practice: this code will not overflow the stack. The stack is large enough to handle even deeply nested file hierarchies. Long before stack space becomes a problem, the folder path length limitation kicks in. Notice that a malicious user might be able to break this assumption by using deeply-nested directories of one letter each. I haven’t tried this. But just to illustrate the point: in order to make this code overflow on a typical computer, the directories would have to be nested a few thousand times. This is simply not a realistic scenario.
Copied from MSDN: using System; using System.IO; class CopyDir { public static void Copy(string sourceDirectory, string targetDirectory) { DirectoryInfo diSource = new DirectoryInfo(sourceDirectory); DirectoryInfo diTarget = new DirectoryInfo(targetDirectory); CopyAll(diSource, diTarget); } public static void CopyAll(DirectoryInfo source, DirectoryInfo target) { Directory.CreateDirectory(target.FullName); // Copy each file into the new directory. foreach (FileInfo fi in source.GetFiles()) { Console.WriteLine(#"Copying {0}\{1}", target.FullName, fi.Name); fi.CopyTo(Path.Combine(target.FullName, fi.Name), true); } // Copy each subdirectory using recursion. foreach (DirectoryInfo diSourceSubDir in source.GetDirectories()) { DirectoryInfo nextTargetSubDir = target.CreateSubdirectory(diSourceSubDir.Name); CopyAll(diSourceSubDir, nextTargetSubDir); } } public static void Main() { string sourceDirectory = #"c:\sourceDirectory"; string targetDirectory = #"c:\targetDirectory"; Copy(sourceDirectory, targetDirectory); } // Output will vary based on the contents of the source directory. }
Or, if you want to go the hard way, add a reference to your project for Microsoft.VisualBasic and then use the following: Microsoft.VisualBasic.FileIO.FileSystem.CopyDirectory(fromDirectory, toDirectory); However, using one of the recursive functions is a better way to go since it won't have to load the VB dll.
Try this: Process proc = new Process(); proc.StartInfo.UseShellExecute = true; proc.StartInfo.FileName = Path.Combine(Environment.SystemDirectory, "xcopy.exe"); proc.StartInfo.Arguments = #"C:\source C:\destination /E /I"; proc.Start(); Your xcopy arguments may vary but you get the idea.
This site always have helped me out a lot, and now it's my turn to help the others with what I know. I hope that my code below be useful for someone. string source_dir = #"E:\"; string destination_dir = #"C:\"; // substring is to remove destination_dir absolute path (E:\). // Create subdirectory structure in destination foreach (string dir in System.IO.Directory.GetDirectories(source_dir, "*", System.IO.SearchOption.AllDirectories)) { System.IO.Directory.CreateDirectory(System.IO.Path.Combine(destination_dir, dir.Substring(source_dir.Length + 1))); // Example: // > C:\sources (and not C:\E:\sources) } foreach (string file_name in System.IO.Directory.GetFiles(source_dir, "*", System.IO.SearchOption.AllDirectories)) { System.IO.File.Copy(file_name, System.IO.Path.Combine(destination_dir, file_name.Substring(source_dir.Length + 1))); }
Copy folder recursively without recursion to avoid stack overflow. public static void CopyDirectory(string source, string target) { var stack = new Stack<Folders>(); stack.Push(new Folders(source, target)); while (stack.Count > 0) { var folders = stack.Pop(); Directory.CreateDirectory(folders.Target); foreach (var file in Directory.GetFiles(folders.Source, "*.*")) { File.Copy(file, Path.Combine(folders.Target, Path.GetFileName(file))); } foreach (var folder in Directory.GetDirectories(folders.Source)) { stack.Push(new Folders(folder, Path.Combine(folders.Target, Path.GetFileName(folder)))); } } } public class Folders { public string Source { get; private set; } public string Target { get; private set; } public Folders(string source, string target) { Source = source; Target = target; } }
Here's a utility class I've used for IO tasks like this. using System; using System.Runtime.InteropServices; namespace MyNameSpace { public class ShellFileOperation { private static String StringArrayToMultiString(String[] stringArray) { String multiString = ""; if (stringArray == null) return ""; for (int i=0 ; i<stringArray.Length ; i++) multiString += stringArray[i] + '\0'; multiString += '\0'; return multiString; } public static bool Copy(string source, string dest) { return Copy(new String[] { source }, new String[] { dest }); } public static bool Copy(String[] source, String[] dest) { Win32.SHFILEOPSTRUCT FileOpStruct = new Win32.SHFILEOPSTRUCT(); FileOpStruct.hwnd = IntPtr.Zero; FileOpStruct.wFunc = (uint)Win32.FO_COPY; String multiSource = StringArrayToMultiString(source); String multiDest = StringArrayToMultiString(dest); FileOpStruct.pFrom = Marshal.StringToHGlobalUni(multiSource); FileOpStruct.pTo = Marshal.StringToHGlobalUni(multiDest); FileOpStruct.fFlags = (ushort)Win32.ShellFileOperationFlags.FOF_NOCONFIRMATION; FileOpStruct.lpszProgressTitle = ""; FileOpStruct.fAnyOperationsAborted = 0; FileOpStruct.hNameMappings = IntPtr.Zero; int retval = Win32.SHFileOperation(ref FileOpStruct); if(retval != 0) return false; return true; } public static bool Move(string source, string dest) { return Move(new String[] { source }, new String[] { dest }); } public static bool Delete(string file) { Win32.SHFILEOPSTRUCT FileOpStruct = new Win32.SHFILEOPSTRUCT(); FileOpStruct.hwnd = IntPtr.Zero; FileOpStruct.wFunc = (uint)Win32.FO_DELETE; String multiSource = StringArrayToMultiString(new string[] { file }); FileOpStruct.pFrom = Marshal.StringToHGlobalUni(multiSource); FileOpStruct.pTo = IntPtr.Zero; FileOpStruct.fFlags = (ushort)Win32.ShellFileOperationFlags.FOF_SILENT | (ushort)Win32.ShellFileOperationFlags.FOF_NOCONFIRMATION | (ushort)Win32.ShellFileOperationFlags.FOF_NOERRORUI | (ushort)Win32.ShellFileOperationFlags.FOF_NOCONFIRMMKDIR; FileOpStruct.lpszProgressTitle = ""; FileOpStruct.fAnyOperationsAborted = 0; FileOpStruct.hNameMappings = IntPtr.Zero; int retval = Win32.SHFileOperation(ref FileOpStruct); if(retval != 0) return false; return true; } public static bool Move(String[] source, String[] dest) { Win32.SHFILEOPSTRUCT FileOpStruct = new Win32.SHFILEOPSTRUCT(); FileOpStruct.hwnd = IntPtr.Zero; FileOpStruct.wFunc = (uint)Win32.FO_MOVE; String multiSource = StringArrayToMultiString(source); String multiDest = StringArrayToMultiString(dest); FileOpStruct.pFrom = Marshal.StringToHGlobalUni(multiSource); FileOpStruct.pTo = Marshal.StringToHGlobalUni(multiDest); FileOpStruct.fFlags = (ushort)Win32.ShellFileOperationFlags.FOF_NOCONFIRMATION; FileOpStruct.lpszProgressTitle = ""; FileOpStruct.fAnyOperationsAborted = 0; FileOpStruct.hNameMappings = IntPtr.Zero; int retval = Win32.SHFileOperation(ref FileOpStruct); if(retval != 0) return false; return true; } } }
tboswell 's replace Proof version (which is resilient to repeating pattern in filepath) public static void copyAll(string SourcePath , string DestinationPath ) { //Now Create all of the directories foreach (string dirPath in Directory.GetDirectories(SourcePath, "*", SearchOption.AllDirectories)) Directory.CreateDirectory(Path.Combine(DestinationPath ,dirPath.Remove(0, SourcePath.Length )) ); //Copy all the files & Replaces any files with the same name foreach (string newPath in Directory.GetFiles(SourcePath, "*.*", SearchOption.AllDirectories)) File.Copy(newPath, Path.Combine(DestinationPath , newPath.Remove(0, SourcePath.Length)) , true); }
My solution is basically a modification of #Termininja's answer, however I have enhanced it a bit and it appears to be more than 5 times faster than the accepted answer. public static void CopyEntireDirectory(string path, string newPath) { Parallel.ForEach(Directory.GetFileSystemEntries(path, "*", SearchOption.AllDirectories) ,(fileName) => { string output = Regex.Replace(fileName, "^" + Regex.Escape(path), newPath); if (File.Exists(fileName)) { Directory.CreateDirectory(Path.GetDirectoryName(output)); File.Copy(fileName, output, true); } else Directory.CreateDirectory(output); }); } EDIT: Modifying #Ahmed Sabry to full parallel foreach does produce a better result, however the code uses recursive function and its not ideal in some situation. public static void CopyEntireDirectory(DirectoryInfo source, DirectoryInfo target, bool overwiteFiles = true) { if (!source.Exists) return; if (!target.Exists) target.Create(); Parallel.ForEach(source.GetDirectories(), (sourceChildDirectory) => CopyEntireDirectory(sourceChildDirectory, new DirectoryInfo(Path.Combine(target.FullName, sourceChildDirectory.Name)))); Parallel.ForEach(source.GetFiles(), sourceFile => sourceFile.CopyTo(Path.Combine(target.FullName, sourceFile.Name), overwiteFiles)); }
It may not be performance-aware, but I'm using it for 30MB folders and it works flawlessly. Plus, I didn't like all the amount of code and recursion required for such an easy task. var src = "c:\src"; var dest = "c:\dest"; var cmp = CompressionLevel.NoCompression; var zip = source_folder + ".zip"; ZipFile.CreateFromDirectory(src, zip, cmp, includeBaseDirectory: false); ZipFile.ExtractToDirectory(zip, dest_folder); File.Delete(zip); Note: ZipFile is available on .NET 4.5+ in the System.IO.Compression namespace
Here is a concise and efficient solution: namespace System.IO { public static class ExtensionMethods { public static void CopyTo(this DirectoryInfo srcPath, string destPath) { Directory.CreateDirectory(destPath); Parallel.ForEach(srcPath.GetDirectories("*", SearchOption.AllDirectories), srcInfo => Directory.CreateDirectory($"{destPath}{srcInfo.FullName[srcPath.FullName.Length..]}")); Parallel.ForEach(srcPath.GetFiles("*", SearchOption.AllDirectories), srcInfo => File.Copy(srcInfo.FullName, $"{destPath}{srcInfo.FullName[srcPath.FullName.Length..]}", true)); }); } } } To use: new DirectoryInfo(sourcePath).CopyTo(destinationPath);
A minor improvement on d4nt's answer, as you probably want to check for errors and not have to change xcopy paths if you're working on a server and development machine: public void CopyFolder(string source, string destination) { string xcopyPath = Environment.GetEnvironmentVariable("WINDIR") + #"\System32\xcopy.exe"; ProcessStartInfo info = new ProcessStartInfo(xcopyPath); info.UseShellExecute = false; info.RedirectStandardOutput = true; info.Arguments = string.Format("\"{0}\" \"{1}\" /E /I", source, destination); Process process = Process.Start(info); process.WaitForExit(); string result = process.StandardOutput.ReadToEnd(); if (process.ExitCode != 0) { // Or your own custom exception, or just return false if you prefer. throw new InvalidOperationException(string.Format("Failed to copy {0} to {1}: {2}", source, destination, result)); } }
This is my code hope this help private void KCOPY(string source, string destination) { if (IsFile(source)) { string target = Path.Combine(destination, Path.GetFileName(source)); File.Copy(source, target, true); } else { string fileName = Path.GetFileName(source); string target = System.IO.Path.Combine(destination, fileName); if (!System.IO.Directory.Exists(target)) { System.IO.Directory.CreateDirectory(target); } List<string> files = GetAllFileAndFolder(source); foreach (string file in files) { KCOPY(file, target); } } } private List<string> GetAllFileAndFolder(string path) { List<string> allFile = new List<string>(); foreach (string dir in Directory.GetDirectories(path)) { allFile.Add(dir); } foreach (string file in Directory.GetFiles(path)) { allFile.Add(file); } return allFile; } private bool IsFile(string path) { if ((File.GetAttributes(path) & FileAttributes.Directory) == FileAttributes.Directory) { return false; } return true; }
If you like Konrad's popular answer, but you want the source itself to be a folder under target, rather than putting it's children under the target folder, here's the code for that. It returns the newly created DirectoryInfo, which is handy: public static DirectoryInfo CopyFilesRecursively(DirectoryInfo source, DirectoryInfo target) { var newDirectoryInfo = target.CreateSubdirectory(source.Name); foreach (var fileInfo in source.GetFiles()) fileInfo.CopyTo(Path.Combine(newDirectoryInfo.FullName, fileInfo.Name)); foreach (var childDirectoryInfo in source.GetDirectories()) CopyFilesRecursively(childDirectoryInfo, newDirectoryInfo); return newDirectoryInfo; }
You can always use this, taken from Microsofts website. static void Main() { // Copy from the current directory, include subdirectories. DirectoryCopy(".", #".\temp", true); } private static void DirectoryCopy(string sourceDirName, string destDirName, bool copySubDirs) { // Get the subdirectories for the specified directory. DirectoryInfo dir = new DirectoryInfo(sourceDirName); if (!dir.Exists) { throw new DirectoryNotFoundException( "Source directory does not exist or could not be found: " + sourceDirName); } DirectoryInfo[] dirs = dir.GetDirectories(); // If the destination directory doesn't exist, create it. if (!Directory.Exists(destDirName)) { Directory.CreateDirectory(destDirName); } // Get the files in the directory and copy them to the new location. FileInfo[] files = dir.GetFiles(); foreach (FileInfo file in files) { string temppath = Path.Combine(destDirName, file.Name); file.CopyTo(temppath, false); } // If copying subdirectories, copy them and their contents to new location. if (copySubDirs) { foreach (DirectoryInfo subdir in dirs) { string temppath = Path.Combine(destDirName, subdir.Name); DirectoryCopy(subdir.FullName, temppath, copySubDirs); } } }
Sorry for the previous code, it still had bugs :( (fell prey to the fastest gun problem) . Here it is tested and working. The key is the SearchOption.AllDirectories, which eliminates the need for explicit recursion. string path = "C:\\a"; string[] dirs = Directory.GetDirectories(path, "*.*", SearchOption.AllDirectories); string newpath = "C:\\x"; try { Directory.CreateDirectory(newpath); } catch (IOException ex) { Console.WriteLine(ex.Message); } for (int j = 0; j < dirs.Length; j++) { try { Directory.CreateDirectory(dirs[j].Replace(path, newpath)); } catch (IOException ex) { Console.WriteLine(ex.Message); } } string[] files = Directory.GetFiles(path, "*.*", SearchOption.AllDirectories); for (int j = 0; j < files.Length; j++) { try { File.Copy(files[j], files[j].Replace(path, newpath)); } catch (IOException ex) { Console.WriteLine(ex.Message); } }
Here is an extension method for DirectoryInfo a la FileInfo.CopyTo (note the overwrite parameter): public static DirectoryInfo CopyTo(this DirectoryInfo sourceDir, string destinationPath, bool overwrite = false) { var sourcePath = sourceDir.FullName; var destination = new DirectoryInfo(destinationPath); destination.Create(); foreach (var sourceSubDirPath in Directory.EnumerateDirectories(sourcePath, "*", SearchOption.AllDirectories)) Directory.CreateDirectory(sourceSubDirPath.Replace(sourcePath, destinationPath)); foreach (var file in Directory.EnumerateFiles(sourcePath, "*", SearchOption.AllDirectories)) File.Copy(file, file.Replace(sourcePath, destinationPath), overwrite); return destination; }
Use this class. public static class Extensions { public static void CopyTo(this DirectoryInfo source, DirectoryInfo target, bool overwiteFiles = true) { if (!source.Exists) return; if (!target.Exists) target.Create(); Parallel.ForEach(source.GetDirectories(), (sourceChildDirectory) => CopyTo(sourceChildDirectory, new DirectoryInfo(Path.Combine(target.FullName, sourceChildDirectory.Name)))); foreach (var sourceFile in source.GetFiles()) sourceFile.CopyTo(Path.Combine(target.FullName, sourceFile.Name), overwiteFiles); } public static void CopyTo(this DirectoryInfo source, string target, bool overwiteFiles = true) { CopyTo(source, new DirectoryInfo(target), overwiteFiles); } }
One variant with only one loop for copying of all folders and files: foreach (var f in Directory.GetFileSystemEntries(path, "*", SearchOption.AllDirectories)) { var output = Regex.Replace(f, #"^" + path, newPath); if (File.Exists(f)) File.Copy(f, output, true); else Directory.CreateDirectory(output); }
Better than any code (extension method to DirectoryInfo with recursion) public static bool CopyTo(this DirectoryInfo source, string destination) { try { foreach (string dirPath in Directory.GetDirectories(source.FullName)) { var newDirPath = dirPath.Replace(source.FullName, destination); Directory.CreateDirectory(newDirPath); new DirectoryInfo(dirPath).CopyTo(newDirPath); } //Copy all the files & Replaces any files with the same name foreach (string filePath in Directory.GetFiles(source.FullName)) { File.Copy(filePath, filePath.Replace(source.FullName,destination), true); } return true; } catch (IOException exp) { return false; } }
Copy and replace all files of the folder public static void CopyAndReplaceAll(string SourcePath, string DestinationPath, string backupPath) { foreach (string dirPath in Directory.GetDirectories(SourcePath, "*", SearchOption.AllDirectories)) { Directory.CreateDirectory($"{DestinationPath}{dirPath.Remove(0, SourcePath.Length)}"); Directory.CreateDirectory($"{backupPath}{dirPath.Remove(0, SourcePath.Length)}"); } foreach (string newPath in Directory.GetFiles(SourcePath, "*.*", SearchOption.AllDirectories)) { if (!File.Exists($"{ DestinationPath}{newPath.Remove(0, SourcePath.Length)}")) File.Copy(newPath, $"{ DestinationPath}{newPath.Remove(0, SourcePath.Length)}"); else File.Replace(newPath , $"{ DestinationPath}{newPath.Remove(0, SourcePath.Length)}" , $"{ backupPath}{newPath.Remove(0, SourcePath.Length)}", false); } }
The code below is microsoft suggestion how-to-copy-directories and it is shared by dear #iato but it just copies sub directories and files of source folder recursively and doesn't copy the source folder it self (like right click -> copy ). but there is a tricky way below this answer : private static void DirectoryCopy(string sourceDirName, string destDirName, bool copySubDirs = true) { // Get the subdirectories for the specified directory. DirectoryInfo dir = new DirectoryInfo(sourceDirName); if (!dir.Exists) { throw new DirectoryNotFoundException( "Source directory does not exist or could not be found: " + sourceDirName); } DirectoryInfo[] dirs = dir.GetDirectories(); // If the destination directory doesn't exist, create it. if (!Directory.Exists(destDirName)) { Directory.CreateDirectory(destDirName); } // Get the files in the directory and copy them to the new location. FileInfo[] files = dir.GetFiles(); foreach (FileInfo file in files) { string temppath = Path.Combine(destDirName, file.Name); file.CopyTo(temppath, false); } // If copying subdirectories, copy them and their contents to new location. if (copySubDirs) { foreach (DirectoryInfo subdir in dirs) { string temppath = Path.Combine(destDirName, subdir.Name); DirectoryCopy(subdir.FullName, temppath, copySubDirs); } } } if you want to copy contents of source folder and subfolders recursively you can simply use it like this : string source = #"J:\source\"; string dest= #"J:\destination\"; DirectoryCopy(source, dest); but if you want to copy the source directory it self (similar that you have right clicked on source folder and clicked copy then in the destination folder you clicked paste) you should use like this : string source = #"J:\source\"; string dest= #"J:\destination\"; DirectoryCopy(source, Path.Combine(dest, new DirectoryInfo(source).Name));
Below code to copy all files from source to destination of given pattern in same folder structure: public static void Copy() { string sourceDir = #"C:\test\source\"; string destination = #"C:\test\destination\"; string[] textFiles = Directory.GetFiles(sourceDir, "*.txt", SearchOption.AllDirectories); foreach (string textFile in textFiles) { string fileName = textFile.Substring(sourceDir.Length); string directoryPath = Path.Combine(destination, Path.GetDirectoryName(fileName)); if (!Directory.Exists(directoryPath)) Directory.CreateDirectory(directoryPath); File.Copy(textFile, Path.Combine(directoryPath, Path.GetFileName(textFile)), true); } }
Just wanted to add my version. It can handle both directories and files, and can overwrite or skip if destination file exists. public static void Copy( string source, string destination, string pattern = "*", bool includeSubFolders = true, bool overwrite = true, bool overwriteOnlyIfSourceIsNewer = false) { if (File.Exists(source)) { // Source is a file, copy and leave CopyFile(source, destination); return; } if (!Directory.Exists(source)) { throw new DirectoryNotFoundException($"Source directory does not exists: `{source}`"); } var files = Directory.GetFiles( source, pattern, includeSubFolders ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly); foreach (var file in files) { var newFile = file.Replace(source, destination); CopyFile(file, newFile, overwrite, overwriteOnlyIfSourceIsNewer); } } private static void CopyFile( string source, string destination, bool overwrite = true, bool overwriteIfSourceIsNewer = false) { if (!overwrite && File.Exists(destination)) { return; } if (overwriteIfSourceIsNewer && File.Exists(destination)) { var sourceLastModified = File.GetLastWriteTimeUtc(source); var destinationLastModified = File.GetLastWriteTimeUtc(destination); if (sourceLastModified <= destinationLastModified) { return; } CreateDirectory(destination); File.Copy(source, destination, overwrite); return; } CreateDirectory(destination); File.Copy(source, destination, overwrite); } private static void CreateDirectory(string filePath) { var targetDirectory = Path.GetDirectoryName(filePath); if (targetDirectory != null && !Directory.Exists(targetDirectory)) { Directory.CreateDirectory(targetDirectory); } }
Properties of this code: No parallel task, is less performant, but the idea is to treat file by file, so you can log or stop. Can skip hiddden files Can skip by modified date Can break or not (you chose) on a file copy error Uses Buffer of 64K for SMB and FileShare.ReadWrite to avoid locks Personalize your Exceptions Message For Windows Notes ExceptionToString() is a personal extension that tries to get inner exceptions and display stack. Replace it for ex.Message or any other code. log4net.ILog _log I use ==Log4net== You can make your Log in a different way. /// <summary> /// Recursive Directory Copy /// </summary> /// <param name="fromPath"></param> /// <param name="toPath"></param> /// <param name="continueOnException">on error, continue to copy next file</param> /// <param name="skipHiddenFiles">To avoid files like thumbs.db</param> /// <param name="skipByModifiedDate">Does not copy if the destiny file has the same or more recent modified date</param> /// <remarks> /// </remarks> public static void CopyEntireDirectory(string fromPath, string toPath, bool continueOnException = false, bool skipHiddenFiles = true, bool skipByModifiedDate = true) { log4net.ILog _log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); string nl = Environment.NewLine; string sourcePath = ""; string destPath = ""; string _exMsg = ""; void TreateException(Exception ex) { _log.Warn(_exMsg); if (continueOnException == false) { throw new Exception($"{_exMsg}{nl}----{nl}{ex.ExceptionToString()}"); } } try { foreach (string fileName in Directory.GetFileSystemEntries(fromPath, "*", SearchOption.AllDirectories)) { sourcePath = fileName; destPath = Regex.Replace(fileName, "^" + Regex.Escape(fromPath), toPath); Directory.CreateDirectory(Path.GetDirectoryName(destPath)); _log.Debug(FileCopyStream(sourcePath, destPath,skipHiddenFiles,skipByModifiedDate)); } } // Directory must be less than 148 characters, File must be less than 261 characters catch (PathTooLongException) { throw new Exception($"Both paths must be less than 148 characters:{nl}{sourcePath}{nl}{destPath}"); } // Not enough disk space. Cancel further copies catch (IOException ex) when ((ex.HResult & 0xFFFF) == 0x27 || (ex.HResult & 0xFFFF) == 0x70) { throw new Exception($"Not enough disk space:{nl}'{toPath}'"); } // used by another process catch (IOException ex) when ((uint)ex.HResult == 0x80070020) { _exMsg = $"File is being used by another process:{nl}'{destPath}'{nl}{ex.Message}"; TreateException(ex); } catch (UnauthorizedAccessException ex) { _exMsg = $"Unauthorized Access Exception:{nl}from:'{sourcePath}'{nl}to:{destPath}"; TreateException(ex); } catch (Exception ex) { _exMsg = $"from:'{sourcePath}'{nl}to:{destPath}"; TreateException(ex); } } /// <summary> /// File Copy using Stream 64K and trying to avoid locks with fileshare /// </summary> /// <param name="sourcePath"></param> /// <param name="destPath"></param> /// <param name="skipHiddenFiles">To avoid files like thumbs.db</param> /// <param name="skipByModifiedDate">Does not copy if the destiny file has the same or more recent modified date</param> public static string FileCopyStream(string sourcePath, string destPath, bool skipHiddenFiles = true, bool skipByModifiedDate = true) { // Buffer should be 64K = 65536 bytes // Increasing the buffer size beyond 64k will not help in any circunstance, // as the underlying SMB protocol does not support buffer lengths beyond 64k." byte[] buffer = new byte[65536]; if (!File.Exists(sourcePath)) return $"is not a file: '{sourcePath}'"; FileInfo sourcefileInfo = new FileInfo(sourcePath); FileInfo destFileInfo = null; if (File.Exists(destPath)) destFileInfo = new FileInfo(destPath); if (skipHiddenFiles) { if (sourcefileInfo.Attributes.HasFlag(FileAttributes.Hidden)) return $"Hidden File Not Copied: '{sourcePath}'"; } using (FileStream input = sourcefileInfo.Open(FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) using (FileStream output = new FileStream(destPath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.ReadWrite, buffer.Length)) { if (skipByModifiedDate && destFileInfo != null) { if (destFileInfo.LastWriteTime < sourcefileInfo.LastWriteTime) { input.CopyTo(output, buffer.Length); destFileInfo.LastWriteTime = sourcefileInfo.LastWriteTime; return $"Replaced: '{sourcePath}'"; } else { return $"NOT replaced (more recent or same file): '{sourcePath}'"; } } else { input.CopyTo(output, buffer.Length); destFileInfo = new FileInfo(destPath); destFileInfo.LastWriteTime = sourcefileInfo.LastWriteTime; return $"New File: '{sourcePath}'"; } } }
For UWP and Winui 3 (WindowsAppSdk) using Async API: public async Task CopyAsync(StorageFolder source, StorageFolder dest) { foreach (var item in await source.GetItemsAsync()) if (item is StorageFile file) await file.CopyAsync(dest); else if (item is StorageFolder folder) await CopyAsync(folder, await dest.CreateFolderAsync(folder.Name, CreationCollisionOption.OpenIfExists)); }
public static class Extensions { public static void Copy(this DirectoryInfo self, DirectoryInfo destination, bool recursively) { foreach (var file in self.GetFiles()) { file.CopyTo(Path.Combine(destination.FullName, file.Name)); } if (recursively) { foreach (var directory in self.GetDirectories()) { directory.Copy(destination.CreateSubdirectory(directory.Name), recursively); } } } } Example of use: var sourceDirectory = new DirectoryInfo(#"C:\source"); var destinationDirectory = new DirectoryInfo(#"C:\destination"); if (destinationDirectory.Exists == false) { sourceDirectory.Copy(destinationDirectory, recursively: true); }