2017-11-02 00:29:20 -07:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using SabreTools.Library.Data ;
using SabreTools.Library.Items ;
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
using SharpCompress.Common ;
namespace SabreTools.Library.FileTypes
{
/// <summary>
/// Represents a folder for reading and writing
/// </summary>
public class Folder : BaseArchive
{
#region Constructors
/// <summary>
/// Create a new folder with no base file
/// </summary>
public Folder ( )
: base ( )
{
}
/// <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>
public Folder ( string filename )
: base ( filename )
{
}
#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>
public override bool ExtractAll ( string outDir )
{
// 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>
public override string ExtractEntry ( string entryName , string outDir )
{
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
List < string > files = FileTools . RetrieveFiles ( _filename , new List < string > ( ) ) ;
// 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
if ( ! String . IsNullOrEmpty ( match ) )
{
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>
public override ( MemoryStream , string ) ExtractEntryStream ( string entryName )
{
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
List < string > files = FileTools . RetrieveFiles ( _filename , new List < string > ( ) ) ;
// 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
if ( ! String . IsNullOrEmpty ( match ) )
{
FileTools . TryOpenRead ( match ) . CopyTo ( ms ) ;
realentry = match ;
}
}
catch ( Exception ex )
{
Globals . Logger . Error ( ex . ToString ( ) ) ;
return ( ms , realentry ) ;
}
return ( ms , realentry ) ;
}
#endregion
#region Information
/// <summary>
/// Generate a list of DatItem objects from the header values in an archive
/// </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>
/// <returns>List of DatItem objects representing the found data</returns>
/// <remarks>TODO: All instances of Hash.DeepHashes should be made into 0x0 eventually</remarks>
public override List < Rom > GetArchiveFileInfo ( Hash omitFromScan = Hash . DeepHashes , bool date = false )
{
throw new NotImplementedException ( ) ;
}
/// <summary>
/// Generate a list of empty folders in an archive
/// </summary>
/// <param name="input">Input file to get data from</param>
/// <returns>List of empty folders in the archive</returns>
public override List < string > GetEmptyFolders ( )
{
throw new NotImplementedException ( ) ;
}
2017-11-02 01:03:36 -07:00
/// <summary>
/// Check whether the input file is a standardized format
/// </summary>
public override bool IsTorrent ( )
{
throw new NotImplementedException ( ) ;
}
2017-11-02 00:29:20 -07:00
#endregion
#region Writing
/// <summary>
/// Write an input file to a torrent LRZip file
/// </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>
public override bool Write ( string inputFile , string outDir , Rom rom , bool date = false , bool romba = false )
{
throw new NotImplementedException ( ) ;
}
/// <summary>
/// Write an input stream to a torrent LRZip file
/// </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>
public override bool Write ( Stream inputStream , string outDir , Rom rom , bool date = false , bool romba = false )
{
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
string fileName = Path . Combine ( outDir , Style . RemovePathUnsafeCharacters ( rom . MachineName ) , Style . RemovePathUnsafeCharacters ( rom . Name ) ) ;
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
outputStream = FileTools . TryCreate ( fileName ) ;
// 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 )
{
if ( date & & ! String . IsNullOrEmpty ( ( ( Rom ) rom ) . Date ) )
{
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>
/// Write a set of input files to a torrent LRZip archive (assuming the same output archive name)
/// </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>
public override bool Write ( List < string > inputFiles , string outDir , List < Rom > roms , bool date = false , bool romba = false )
{
throw new NotImplementedException ( ) ;
}
#endregion
}
}