2017-11-02 00:29:20 -07:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using SabreTools.Library.Data ;
2017-11-02 15:44:15 -07:00
using SabreTools.Library.DatItems ;
2017-11-02 00:29:20 -07:00
using SabreTools.Library.Tools ;
#if MONO
using System.IO ;
using Directory = Alphaleonis . Win32 . Filesystem . Directory ;
using PathFormat = Alphaleonis . Win32 . Filesystem . PathFormat ;
#else
using Alphaleonis.Win32.Filesystem ;
using BinaryReader = System . IO . BinaryReader ;
using FileStream = System . IO . FileStream ;
using MemoryStream = System . IO . MemoryStream ;
using SearchOption = System . IO . SearchOption ;
using SeekOrigin = System . IO . SeekOrigin ;
using Stream = System . IO . Stream ;
#endif
namespace SabreTools.Library.FileTypes
{
/// <summary>
/// Represents a folder for reading and writing
/// </summary>
2018-02-15 23:38:55 -08:00
public class Folder : BaseFile
2017-11-02 00:29:20 -07:00
{
2018-02-15 23:38:55 -08:00
#region Protected instance variables
protected List < BaseFile > _children ;
#endregion
2017-11-02 00:29:20 -07:00
#region Constructors
/// <summary>
/// Create a new folder with no base file
/// </summary>
public Folder ( )
: base ( )
{
2018-02-15 22:06:20 -08:00
_fileType = FileType . Folder ;
2017-11-02 00:29:20 -07:00
}
/// <summary>
/// Create a new folder from the given file
/// </summary>
/// <param name="filename">Name of the file to use as an archive</param>
/// <param name="read">True for opening file as read, false for opening file as write</param>
2018-02-23 11:56:18 -08:00
/// <param name="getHashes">True if hashes for this file should be calculated, false otherwise (default)</param>
public Folder ( string filename , bool getHashes = false )
: base ( filename , getHashes )
2017-11-02 00:29:20 -07:00
{
2018-02-15 22:06:20 -08:00
_fileType = FileType . Folder ;
2017-11-02 00:29:20 -07:00
}
#endregion
#region Extraction
/// <summary>
/// Attempt to extract a file as an archive
/// </summary>
/// <param name="outDir">Output directory for archive extraction</param>
/// <returns>True if the extraction was a success, false otherwise</returns>
2018-02-15 23:52:57 -08:00
public virtual bool CopyAll ( string outDir )
2017-11-02 00:29:20 -07:00
{
// Copy all files from the current folder to the output directory recursively
try
{
// Make sure the folders exist
Directory . CreateDirectory ( _filename ) ;
Directory . CreateDirectory ( outDir ) ;
Directory . Copy ( _filename , outDir , true , PathFormat . FullPath ) ;
}
catch ( Exception ex )
{
Globals . Logger . Error ( ex . ToString ( ) ) ;
return false ;
}
return true ;
}
/// <summary>
/// Attempt to extract a file from an archive
/// </summary>
/// <param name="entryName">Name of the entry to be extracted</param>
/// <param name="outDir">Output directory for archive extraction</param>
/// <returns>Name of the extracted file, null on error</returns>
2018-02-15 23:52:57 -08:00
public virtual string CopyToFile ( string entryName , string outDir )
2017-11-02 00:29:20 -07:00
{
string realentry = null ;
// Copy single file from the current folder to the output directory, if exists
try
{
// Make sure the folders exist
Directory . CreateDirectory ( _filename ) ;
Directory . CreateDirectory ( outDir ) ;
// Get all files from the input directory
2017-11-08 00:27:00 -08:00
List < string > files = Utilities . RetrieveFiles ( _filename , new List < string > ( ) ) ;
2017-11-02 00:29:20 -07:00
// Now sort through to find the first file that matches
string match = files . Where ( s = > s . EndsWith ( entryName ) ) . FirstOrDefault ( ) ;
// If we had a file, copy that over to the new name
2017-11-08 13:15:44 -08:00
if ( ! String . IsNullOrWhiteSpace ( match ) )
2017-11-02 00:29:20 -07:00
{
realentry = match ;
File . Copy ( match , Path . Combine ( outDir , entryName ) ) ;
}
}
catch ( Exception ex )
{
Globals . Logger . Error ( ex . ToString ( ) ) ;
return realentry ;
}
return realentry ;
}
/// <summary>
/// Attempt to extract a stream from an archive
/// </summary>
/// <param name="entryName">Name of the entry to be extracted</param>
/// <param name="realEntry">Output representing the entry name that was found</param>
/// <returns>MemoryStream representing the entry, null on error</returns>
2018-02-15 23:52:57 -08:00
public virtual ( MemoryStream , string ) CopyToStream ( string entryName )
2017-11-02 00:29:20 -07:00
{
MemoryStream ms = new MemoryStream ( ) ;
string realentry = null ;
// Copy single file from the current folder to the output directory, if exists
try
{
// Make sure the folders exist
Directory . CreateDirectory ( _filename ) ;
// Get all files from the input directory
2017-11-08 00:27:00 -08:00
List < string > files = Utilities . RetrieveFiles ( _filename , new List < string > ( ) ) ;
2017-11-02 00:29:20 -07:00
// Now sort through to find the first file that matches
string match = files . Where ( s = > s . EndsWith ( entryName ) ) . FirstOrDefault ( ) ;
// If we had a file, copy that over to the new name
2017-11-08 13:15:44 -08:00
if ( ! String . IsNullOrWhiteSpace ( match ) )
2017-11-02 00:29:20 -07:00
{
2017-11-08 00:27:00 -08:00
Utilities . TryOpenRead ( match ) . CopyTo ( ms ) ;
2017-11-02 00:29:20 -07:00
realentry = match ;
}
}
catch ( Exception ex )
{
Globals . Logger . Error ( ex . ToString ( ) ) ;
return ( ms , realentry ) ;
}
return ( ms , realentry ) ;
}
#endregion
#region Information
/// <summary>
2018-02-15 23:38:55 -08:00
/// Generate a list of immediate children from the current folder
2017-11-02 00:29:20 -07:00
/// </summary>
/// <param name="omitFromScan">Hash representing the hashes that should be skipped</param>
/// <param name="date">True if entry dates should be included, false otherwise (default)</param>
2018-02-15 23:38:55 -08:00
/// <returns>List of BaseFile objects representing the found data</returns>
2017-11-02 00:29:20 -07:00
/// <remarks>TODO: All instances of Hash.DeepHashes should be made into 0x0 eventually</remarks>
2018-02-15 23:52:57 -08:00
public virtual List < BaseFile > GetChildren ( Hash omitFromScan = Hash . DeepHashes , bool date = false )
2017-11-02 00:29:20 -07:00
{
2018-02-15 23:38:55 -08:00
if ( _children = = null | | _children . Count = = 0 )
{
_children = new List < BaseFile > ( ) ;
foreach ( string file in Directory . EnumerateFiles ( _filename , "*" , SearchOption . TopDirectoryOnly ) )
{
BaseFile nf = Utilities . GetFileInfo ( file , omitFromScan : omitFromScan , date : date ) ;
_children . Add ( nf ) ;
}
foreach ( string dir in Directory . EnumerateDirectories ( _filename , "*" , SearchOption . TopDirectoryOnly ) )
{
Folder fl = new Folder ( dir ) ;
_children . Add ( fl ) ;
}
}
return _children ;
2017-11-02 00:29:20 -07:00
}
/// <summary>
/// Generate a list of empty folders in an archive
/// </summary>
/// <param name="input">Input file to get data from</param>
2018-02-15 23:38:55 -08:00
/// <returns>List of empty folders in the folder</returns>
2018-02-15 23:52:57 -08:00
public virtual List < string > GetEmptyFolders ( )
2017-11-02 00:29:20 -07:00
{
2018-02-15 23:38:55 -08:00
return Utilities . GetEmptyDirectories ( _filename ) . ToList ( ) ;
2017-11-02 01:03:36 -07:00
}
2017-11-02 00:29:20 -07:00
#endregion
#region Writing
/// <summary>
2018-02-15 23:52:57 -08:00
/// Write an input file to an output folder
2017-11-02 00:29:20 -07:00
/// </summary>
/// <param name="inputFile">Input filename to be moved</param>
/// <param name="outDir">Output directory to build to</param>
/// <param name="rom">DatItem representing the new information</param>
/// <param name="date">True if the date from the DAT should be used if available, false otherwise (default)</param>
/// <param name="romba">True if files should be output in Romba depot folders, false otherwise</param>
/// <returns>True if the write was a success, false otherwise</returns>
/// <remarks>This works for now, but it can be sped up by using Ionic.Zip or another zlib wrapper that allows for header values built-in. See edc's code.</remarks>
2018-02-15 23:52:57 -08:00
public virtual bool Write ( string inputFile , string outDir , Rom rom , bool date = false , bool romba = false )
2017-11-02 00:29:20 -07:00
{
2018-03-05 11:37:41 -08:00
FileStream fs = Utilities . TryOpenRead ( inputFile ) ;
return Write ( fs , outDir , rom , date , romba ) ;
2017-11-02 00:29:20 -07:00
}
/// <summary>
2018-02-15 23:52:57 -08:00
/// Write an input stream to an output folder
2017-11-02 00:29:20 -07:00
/// </summary>
/// <param name="inputStream">Input stream to be moved</param>
/// <param name="outDir">Output directory to build to</param>
/// <param name="rom">DatItem representing the new information</param>
/// <param name="date">True if the date from the DAT should be used if available, false otherwise (default)</param>
/// <param name="romba">True if files should be output in Romba depot folders, false otherwise</param>
/// <returns>True if the write was a success, false otherwise</returns>
/// <remarks>This works for now, but it can be sped up by using Ionic.Zip or another zlib wrapper that allows for header values built-in. See edc's code.</remarks>
2018-02-15 23:52:57 -08:00
public virtual bool Write ( Stream inputStream , string outDir , Rom rom , bool date = false , bool romba = false )
2017-11-02 00:29:20 -07:00
{
bool success = false ;
// If either input is null or empty, return
if ( inputStream = = null | | rom = = null | | rom . Name = = null )
{
return success ;
}
// If the stream is not readable, return
if ( ! inputStream . CanRead )
{
return success ;
}
// Set internal variables
FileStream outputStream = null ;
// Get the output folder name from the first rebuild rom
2017-11-08 00:27:00 -08:00
string fileName = Path . Combine ( outDir , Utilities . RemovePathUnsafeCharacters ( rom . MachineName ) , Utilities . RemovePathUnsafeCharacters ( rom . Name ) ) ;
2017-11-02 00:29:20 -07:00
try
{
// If the full output path doesn't exist, create it
if ( ! Directory . Exists ( Path . GetDirectoryName ( fileName ) ) )
{
Directory . CreateDirectory ( Path . GetDirectoryName ( fileName ) ) ;
}
// Overwrite output files by default
2017-11-08 00:27:00 -08:00
outputStream = Utilities . TryCreate ( fileName ) ;
2017-11-02 00:29:20 -07:00
// If the output stream isn't null
if ( outputStream ! = null )
{
// Copy the input stream to the output
inputStream . Seek ( 0 , SeekOrigin . Begin ) ;
int bufferSize = 4096 * 128 ;
byte [ ] ibuffer = new byte [ bufferSize ] ;
int ilen ;
while ( ( ilen = inputStream . Read ( ibuffer , 0 , bufferSize ) ) > 0 )
{
outputStream . Write ( ibuffer , 0 , ilen ) ;
outputStream . Flush ( ) ;
}
outputStream . Dispose ( ) ;
if ( rom . Type = = ItemType . Rom )
{
2017-11-08 13:15:44 -08:00
if ( date & & ! String . IsNullOrWhiteSpace ( ( ( Rom ) rom ) . Date ) )
2017-11-02 00:29:20 -07:00
{
File . SetCreationTime ( fileName , DateTime . Parse ( ( ( Rom ) rom ) . Date ) ) ;
}
}
success = true ;
}
}
catch ( Exception ex )
{
Console . WriteLine ( ex ) ;
success = false ;
}
finally
{
inputStream . Dispose ( ) ;
outputStream ? . Dispose ( ) ;
}
return success ;
}
/// <summary>
2018-02-15 23:52:57 -08:00
/// Write a set of input files to an output folder (assuming the same output archive name)
2017-11-02 00:29:20 -07:00
/// </summary>
/// <param name="inputFiles">Input files to be moved</param>
/// <param name="outDir">Output directory to build to</param>
/// <param name="rom">DatItem representing the new information</param>
/// <param name="date">True if the date from the DAT should be used if available, false otherwise (default)</param>
/// <param name="romba">True if files should be output in Romba depot folders, false otherwise</param>
/// <returns>True if the archive was written properly, false otherwise</returns>
2018-02-15 23:52:57 -08:00
public virtual bool Write ( List < string > inputFiles , string outDir , List < Rom > roms , bool date = false , bool romba = false )
2017-11-02 00:29:20 -07:00
{
throw new NotImplementedException ( ) ;
}
#endregion
}
}