2011-10-23 05:26:43 +00:00
using System ;
using System.IO ;
using System.Linq ;
2011-11-13 04:07:06 +00:00
using System.Runtime.InteropServices ;
2013-07-05 06:30:45 +00:00
using System.Security.AccessControl ;
2013-08-13 00:22:35 +00:00
using System.Security.Principal ;
2011-10-23 05:26:43 +00:00
using NLog ;
2013-05-24 22:18:37 +00:00
using NzbDrone.Common.EnsureThat ;
2013-06-28 00:04:52 +00:00
using NzbDrone.Common.EnvironmentInfo ;
2013-08-31 01:42:30 +00:00
using NzbDrone.Common.Instrumentation ;
2011-10-23 05:26:43 +00:00
2011-10-23 22:44:37 +00:00
namespace NzbDrone.Common
2011-10-23 05:26:43 +00:00
{
2013-05-10 23:53:50 +00:00
public interface IDiskProvider
{
DateTime GetLastFolderWrite ( string path ) ;
DateTime GetLastFileWrite ( string path ) ;
void EnsureFolder ( string path ) ;
bool FolderExists ( string path ) ;
bool FileExists ( string path ) ;
2013-07-24 06:26:10 +00:00
bool FileExists ( string path , bool caseSensitive ) ;
2013-05-10 23:53:50 +00:00
string [ ] GetDirectories ( string path ) ;
string [ ] GetFiles ( string path , SearchOption searchOption ) ;
2013-07-05 04:43:28 +00:00
long GetFolderSize ( string path ) ;
2013-05-29 04:10:23 +00:00
long GetFileSize ( string path ) ;
2013-09-12 06:52:37 +00:00
void CreateFolder ( string path ) ;
void CopyFolder ( string source , string destination ) ;
2013-07-05 04:43:28 +00:00
void MoveFolder ( string source , string destination ) ;
2013-05-10 23:53:50 +00:00
void DeleteFile ( string path ) ;
void MoveFile ( string source , string destination ) ;
void DeleteFolder ( string path , bool recursive ) ;
void InheritFolderPermissions ( string filename ) ;
2013-09-01 04:38:06 +00:00
long? GetAvailableSpace ( string path ) ;
2013-05-10 23:53:50 +00:00
string ReadAllText ( string filePath ) ;
void WriteAllText ( string filename , string contents ) ;
void FileSetLastWriteTimeUtc ( string path , DateTime dateTime ) ;
2013-07-05 04:43:28 +00:00
void FolderSetLastWriteTimeUtc ( string path , DateTime dateTime ) ;
2013-09-12 06:52:37 +00:00
bool IsFileLocked ( string path ) ;
2013-05-10 23:53:50 +00:00
string GetPathRoot ( string path ) ;
2013-08-13 00:22:35 +00:00
void SetPermissions ( string filename , WellKnownSidType accountSid , FileSystemRights rights , AccessControlType controlType ) ;
2013-08-11 22:55:26 +00:00
bool IsParent ( string parentPath , string childPath ) ;
2013-08-20 19:28:46 +00:00
FileAttributes GetFileAttributes ( string path ) ;
2013-09-08 17:13:46 +00:00
void EmptyFolder ( string path ) ;
2013-05-10 23:53:50 +00:00
}
public class DiskProvider : IDiskProvider
2011-10-23 05:26:43 +00:00
{
2013-08-20 19:28:46 +00:00
enum TransferAction
2011-11-18 06:52:50 +00:00
{
Copy ,
Move
}
2011-11-13 04:07:06 +00:00
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool GetDiskFreeSpaceEx ( string lpDirectoryName ,
out ulong lpFreeBytesAvailable ,
out ulong lpTotalNumberOfBytes ,
out ulong lpTotalNumberOfFreeBytes ) ;
2013-09-12 06:52:37 +00:00
private static readonly Logger Logger = NzbDroneLogger . GetLogger ( ) ;
2011-10-23 05:26:43 +00:00
2013-08-03 03:01:16 +00:00
public DateTime GetLastFolderWrite ( string path )
2012-01-23 04:34:30 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-01-23 04:34:30 +00:00
if ( ! FolderExists ( path ) )
{
throw new DirectoryNotFoundException ( "Directory doesn't exist. " + path ) ;
}
2012-01-23 04:59:23 +00:00
var dirFiles = GetFiles ( path , SearchOption . AllDirectories ) . ToList ( ) ;
if ( ! dirFiles . Any ( ) )
{
return new DirectoryInfo ( path ) . LastWriteTimeUtc ;
}
return dirFiles . Select ( f = > new FileInfo ( f ) )
. Max ( c = > c . LastWriteTimeUtc ) ;
2012-01-23 04:34:30 +00:00
}
2013-08-03 03:01:16 +00:00
public DateTime GetLastFileWrite ( string path )
2012-08-29 15:34:51 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-08-29 15:34:51 +00:00
if ( ! FileExists ( path ) )
2013-09-13 04:59:29 +00:00
{
2012-08-29 15:34:51 +00:00
throw new FileNotFoundException ( "File doesn't exist: " + path ) ;
2013-09-13 04:59:29 +00:00
}
2012-08-29 15:34:51 +00:00
return new FileInfo ( path ) . LastWriteTimeUtc ;
}
2013-08-03 03:01:16 +00:00
public void EnsureFolder ( string path )
2013-04-15 01:41:39 +00:00
{
if ( ! FolderExists ( path ) )
{
CreateFolder ( path ) ;
}
}
2013-08-03 03:01:16 +00:00
public bool FolderExists ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2011-10-23 05:26:43 +00:00
return Directory . Exists ( path ) ;
}
2013-07-26 05:55:19 +00:00
2013-08-03 03:01:16 +00:00
public bool FileExists ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2011-10-23 05:26:43 +00:00
return File . Exists ( path ) ;
}
2013-08-03 03:01:16 +00:00
public bool FileExists ( string path , bool caseSensitive )
2013-07-24 06:26:10 +00:00
{
if ( caseSensitive )
{
return FileExists ( path ) & & path = = path . GetActualCasing ( ) ;
}
return FileExists ( path ) ;
}
2013-08-03 03:01:16 +00:00
public string [ ] GetDirectories ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-01-23 06:43:11 +00:00
return Directory . GetDirectories ( path ) ;
2011-10-23 05:26:43 +00:00
}
2013-08-03 03:01:16 +00:00
public string [ ] GetFiles ( string path , SearchOption searchOption )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-01-23 06:43:11 +00:00
return Directory . GetFiles ( path , "*.*" , searchOption ) ;
2011-10-23 05:26:43 +00:00
}
2013-08-03 03:01:16 +00:00
public long GetFolderSize ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2011-10-23 05:26:43 +00:00
return GetFiles ( path , SearchOption . AllDirectories ) . Sum ( e = > new FileInfo ( e ) . Length ) ;
}
2013-08-03 03:01:16 +00:00
public long GetFileSize ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-10-20 01:42:42 +00:00
if ( ! FileExists ( path ) )
2013-09-13 04:59:29 +00:00
{
2012-10-20 01:42:42 +00:00
throw new FileNotFoundException ( "File doesn't exist: " + path ) ;
2013-09-13 04:59:29 +00:00
}
2012-10-20 01:42:42 +00:00
2011-10-23 05:26:43 +00:00
var fi = new FileInfo ( path ) ;
return fi . Length ;
}
2013-09-12 06:52:37 +00:00
public void CreateFolder ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2013-09-12 06:52:37 +00:00
Directory . CreateDirectory ( path ) ;
2011-10-23 05:26:43 +00:00
}
2013-09-12 06:52:37 +00:00
public void CopyFolder ( string source , string destination )
2011-11-13 04:07:06 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > source ) . IsValidPath ( ) ;
2013-09-12 06:52:37 +00:00
Ensure . That ( ( ) = > destination ) . IsValidPath ( ) ;
2013-05-24 22:18:37 +00:00
2013-09-12 06:52:37 +00:00
TransferFolder ( source , destination , TransferAction . Copy ) ;
2011-11-18 06:52:50 +00:00
}
2013-08-03 03:01:16 +00:00
public void MoveFolder ( string source , string destination )
2011-11-18 06:52:50 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > source ) . IsValidPath ( ) ;
Ensure . That ( ( ) = > destination ) . IsValidPath ( ) ;
2011-11-18 06:52:50 +00:00
try
{
2013-07-05 04:43:28 +00:00
TransferFolder ( source , destination , TransferAction . Move ) ;
2013-09-13 04:59:29 +00:00
DeleteFolder ( source , true ) ;
2011-11-18 06:52:50 +00:00
}
catch ( Exception e )
{
e . Data . Add ( "Source" , source ) ;
e . Data . Add ( "Destination" , destination ) ;
throw ;
}
}
2013-07-05 04:43:28 +00:00
private void TransferFolder ( string source , string target , TransferAction transferAction )
2011-11-18 06:52:50 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > source ) . IsValidPath ( ) ;
Ensure . That ( ( ) = > target ) . IsValidPath ( ) ;
2011-11-18 06:52:50 +00:00
Logger . Trace ( "{0} {1} -> {2}" , transferAction , source , target ) ;
2011-11-13 04:07:06 +00:00
var sourceFolder = new DirectoryInfo ( source ) ;
var targetFolder = new DirectoryInfo ( target ) ;
if ( ! targetFolder . Exists )
{
targetFolder . Create ( ) ;
}
2011-11-14 04:05:33 +00:00
foreach ( var subDir in sourceFolder . GetDirectories ( ) )
{
2013-07-05 04:43:28 +00:00
TransferFolder ( subDir . FullName , Path . Combine ( target , subDir . Name ) , transferAction ) ;
2011-11-14 04:05:33 +00:00
}
2011-11-18 07:16:05 +00:00
foreach ( var sourceFile in sourceFolder . GetFiles ( "*.*" , SearchOption . TopDirectoryOnly ) )
2011-11-13 04:07:06 +00:00
{
2011-11-18 07:16:05 +00:00
var destFile = Path . Combine ( target , sourceFile . Name ) ;
2011-11-18 06:52:50 +00:00
2013-07-18 23:41:11 +00:00
Logger . Trace ( "{0} {1} -> {2}" , transferAction , sourceFile , destFile ) ;
2011-11-18 06:52:50 +00:00
switch ( transferAction )
{
case TransferAction . Copy :
{
2011-11-18 07:16:05 +00:00
sourceFile . CopyTo ( destFile , true ) ;
2011-11-18 06:52:50 +00:00
break ;
}
case TransferAction . Move :
{
2011-11-18 07:16:05 +00:00
MoveFile ( sourceFile . FullName , destFile ) ;
2011-11-18 06:52:50 +00:00
break ;
}
}
2011-11-13 04:07:06 +00:00
}
}
2013-08-03 03:01:16 +00:00
public void DeleteFile ( string path )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2011-11-20 05:35:44 +00:00
Logger . Trace ( "Deleting file: {0}" , path ) ;
2013-09-13 04:59:29 +00:00
RemoveReadOnly ( path ) ;
2011-10-23 05:26:43 +00:00
File . Delete ( path ) ;
}
2013-08-03 03:01:16 +00:00
public void MoveFile ( string source , string destination )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > source ) . IsValidPath ( ) ;
Ensure . That ( ( ) = > destination ) . IsValidPath ( ) ;
2013-08-20 19:28:46 +00:00
if ( source . PathEquals ( destination ) )
2011-11-18 07:16:05 +00:00
{
2011-12-14 08:06:54 +00:00
Logger . Warn ( "Source and destination can't be the same {0}" , source ) ;
return ;
2011-11-18 07:16:05 +00:00
}
2011-12-14 08:06:54 +00:00
if ( FileExists ( destination ) )
{
DeleteFile ( destination ) ;
}
2013-09-13 04:59:29 +00:00
RemoveReadOnly ( source ) ;
2011-12-14 08:06:54 +00:00
File . Move ( source , destination ) ;
2011-10-23 05:26:43 +00:00
}
2013-08-03 03:01:16 +00:00
public void DeleteFolder ( string path , bool recursive )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2011-10-23 05:26:43 +00:00
Directory . Delete ( path , recursive ) ;
}
2013-08-03 03:01:16 +00:00
public void InheritFolderPermissions ( string filename )
2011-10-23 05:26:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > filename ) . IsValidPath ( ) ;
2011-10-23 05:26:43 +00:00
var fs = File . GetAccessControl ( filename ) ;
fs . SetAccessRuleProtection ( false , false ) ;
File . SetAccessControl ( filename , fs ) ;
}
2011-11-13 04:07:06 +00:00
2013-09-01 04:38:06 +00:00
public long? GetAvailableSpace ( string path )
2011-11-13 04:07:06 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2013-09-01 04:38:06 +00:00
var root = GetPathRoot ( path ) ;
if ( ! FolderExists ( root ) )
throw new DirectoryNotFoundException ( root ) ;
2013-07-12 00:07:21 +00:00
if ( OsInfo . IsLinux )
2013-04-30 06:11:49 +00:00
{
2013-09-01 04:38:06 +00:00
var drives = DriveInfo . GetDrives ( ) ;
2013-07-12 00:07:21 +00:00
2013-09-01 04:38:06 +00:00
foreach ( var drive in drives )
2013-04-30 06:11:49 +00:00
{
2013-09-01 04:38:06 +00:00
try
{
if ( drive . IsReady & & path . StartsWith ( drive . Name , StringComparison . CurrentCultureIgnoreCase ) )
{
return drive . AvailableFreeSpace ;
}
}
catch ( InvalidOperationException e )
{
Logger . ErrorException ( "Couldn't get free space for " + path , e ) ;
}
2013-04-30 06:11:49 +00:00
}
2013-05-10 23:53:50 +00:00
2013-09-01 04:38:06 +00:00
return null ;
2013-04-30 06:11:49 +00:00
}
2013-08-09 04:44:49 +00:00
2013-08-09 01:40:24 +00:00
return DriveFreeSpaceEx ( root ) ;
2013-08-20 19:28:46 +00:00
}
2013-04-30 06:11:49 +00:00
private static long DriveFreeSpaceEx ( string folderName )
{
if ( string . IsNullOrEmpty ( folderName ) )
{
throw new ArgumentNullException ( "folderName" ) ;
}
if ( ! folderName . EndsWith ( "\\" ) )
{
folderName + = '\\' ;
}
ulong free = 0 ;
ulong dummy1 = 0 ;
ulong dummy2 = 0 ;
if ( GetDiskFreeSpaceEx ( folderName , out free , out dummy1 , out dummy2 ) )
{
return ( long ) free ;
}
return 0 ;
2011-11-13 04:07:06 +00:00
}
2011-11-22 06:55:09 +00:00
2013-08-03 03:01:16 +00:00
public string ReadAllText ( string filePath )
2011-11-22 06:55:09 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > filePath ) . IsValidPath ( ) ;
2011-11-22 06:55:09 +00:00
return File . ReadAllText ( filePath ) ;
}
2011-12-10 19:22:47 +00:00
2013-08-03 03:01:16 +00:00
public void WriteAllText ( string filename , string contents )
2012-07-10 04:37:24 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > filename ) . IsValidPath ( ) ;
2013-09-13 04:59:29 +00:00
RemoveReadOnly ( filename ) ;
2012-07-10 04:37:24 +00:00
File . WriteAllText ( filename , contents ) ;
}
2011-12-10 19:22:47 +00:00
2013-08-03 03:01:16 +00:00
public void FileSetLastWriteTimeUtc ( string path , DateTime dateTime )
2012-09-04 06:49:04 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-09-04 06:49:04 +00:00
File . SetLastWriteTimeUtc ( path , dateTime ) ;
}
2013-08-03 03:01:16 +00:00
public void FolderSetLastWriteTimeUtc ( string path , DateTime dateTime )
2012-09-04 06:49:04 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-09-04 06:49:04 +00:00
Directory . SetLastWriteTimeUtc ( path , dateTime ) ;
}
2012-10-20 08:01:47 +00:00
2013-09-12 06:52:37 +00:00
public bool IsFileLocked ( string file )
2012-10-20 08:01:47 +00:00
{
try
{
2013-09-13 04:59:29 +00:00
using ( File . Open ( file , FileMode . Open , FileAccess . Read , FileShare . None ) )
{
return false ;
}
2012-10-20 08:01:47 +00:00
}
catch ( IOException )
{
return true ;
}
}
2013-08-03 03:01:16 +00:00
public string GetPathRoot ( string path )
2012-12-24 07:16:43 +00:00
{
2013-05-24 22:18:37 +00:00
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
2012-12-24 07:16:43 +00:00
return Path . GetPathRoot ( path ) ;
}
2013-07-05 06:30:45 +00:00
2013-08-13 00:22:35 +00:00
public void SetPermissions ( string filename , WellKnownSidType accountSid , FileSystemRights rights , AccessControlType controlType )
2013-07-05 06:30:45 +00:00
{
2013-07-26 05:55:19 +00:00
try
{
2013-08-13 00:22:35 +00:00
var sid = new SecurityIdentifier ( accountSid , null ) ;
2013-07-05 06:30:45 +00:00
2013-07-26 05:55:19 +00:00
var directoryInfo = new DirectoryInfo ( filename ) ;
var directorySecurity = directoryInfo . GetAccessControl ( ) ;
2013-08-13 00:22:35 +00:00
var accessRule = new FileSystemAccessRule ( sid , rights ,
2013-07-26 05:55:19 +00:00
InheritanceFlags . ContainerInherit | InheritanceFlags . ObjectInherit ,
PropagationFlags . None , controlType ) ;
directorySecurity . AddAccessRule ( accessRule ) ;
directoryInfo . SetAccessControl ( directorySecurity ) ;
}
catch ( Exception e )
{
2013-08-13 00:22:35 +00:00
Logger . WarnException ( string . Format ( "Couldn't set permission for {0}. account:{1} rights:{2} accessControlType:{3}" , filename , accountSid , rights , controlType ) , e ) ;
2013-07-26 05:55:19 +00:00
throw ;
}
2013-07-05 06:30:45 +00:00
}
2013-07-23 00:50:37 +00:00
2013-08-11 22:55:26 +00:00
public bool IsParent ( string parentPath , string childPath )
2013-07-23 00:50:37 +00:00
{
2013-08-11 22:55:26 +00:00
parentPath = parentPath . TrimEnd ( Path . DirectorySeparatorChar ) ;
childPath = childPath . TrimEnd ( Path . DirectorySeparatorChar ) ;
2013-07-23 00:50:37 +00:00
2013-08-11 22:55:26 +00:00
var parent = new DirectoryInfo ( parentPath ) ;
var child = new DirectoryInfo ( childPath ) ;
2013-07-23 00:50:37 +00:00
2013-08-11 22:55:26 +00:00
while ( child . Parent ! = null )
2013-07-23 00:50:37 +00:00
{
2013-08-11 22:55:26 +00:00
if ( child . Parent . FullName = = parent . FullName )
2013-07-23 00:50:37 +00:00
{
return true ;
}
2013-07-26 05:55:19 +00:00
2013-08-11 22:55:26 +00:00
child = child . Parent ;
2013-07-23 00:50:37 +00:00
}
return false ;
}
2013-07-30 00:09:48 +00:00
2013-09-13 04:59:29 +00:00
private static void RemoveReadOnly ( string path )
{
if ( File . Exists ( path ) )
{
var newAttributes = File . GetAttributes ( path ) & ~ ( FileAttributes . ReadOnly ) ;
File . SetAttributes ( path , newAttributes ) ;
}
}
2013-07-30 00:09:48 +00:00
public FileAttributes GetFileAttributes ( string path )
{
return File . GetAttributes ( path ) ;
}
2013-09-08 17:13:46 +00:00
public void EmptyFolder ( string path )
{
Ensure . That ( ( ) = > path ) . IsValidPath ( ) ;
foreach ( var file in GetFiles ( path , SearchOption . TopDirectoryOnly ) )
{
DeleteFile ( file ) ;
}
foreach ( var directory in GetDirectories ( path ) )
{
DeleteFolder ( directory , true ) ;
}
}
2011-10-23 05:26:43 +00:00
}
2011-11-13 04:07:06 +00:00
}