@ -29,6 +29,8 @@
# include <OgreArchiveManager.h>
# include <OgreArchiveManager.h>
# include "bsa_file.hpp"
# include "bsa_file.hpp"
# include "../files/constrainedfiledatastream.hpp"
namespace
namespace
{
{
@ -60,104 +62,64 @@ public:
static bool fsstrict = false ;
static bool fsstrict = false ;
/// An OGRE Archive wrapping a BSAFile archive
/// An OGRE Archive wrapping a BSAFile archive
class DirArchive : public Ogre : : FileSystem Archive
class DirArchive : public Ogre : : Archive
{
{
boost : : filesystem : : path currentdir ;
typedef std : : map < std : : string , std : : string > index ;
std : : map < std : : string , std : : vector < std : : string > , ciLessBoost > m ;
unsigned int cutoff ;
bool findFile ( const String & filename , std : : string & copy ) const
{
copy = filename ;
std : : replace ( copy . begin ( ) , copy . end ( ) , ' \\ ' , ' / ' ) ;
if ( copy . at ( 0 ) = = ' / ' )
copy . erase ( 0 , 1 ) ;
if ( fsstrict = = true )
index mIndex ;
return true ;
std : : string folder ;
static char strict_normalize_char ( char ch )
//int delimiter = 0;
{
size_t lastSlash = copy . rfind ( ' / ' ) ;
return ch = = ' \\ ' ? ' / ' : ch ;
if ( lastSlash ! = std : : string : : npos )
}
{
folder = copy . substr ( 0 , lastSlash ) ;
//delimiter = lastSlash+1;
}
std : : vector < std : : string > current ;
static char nonstrict_normalize_char ( char ch )
{
{
std : : map < std : : string , std : : vector < std : : string > , ciLessBoost > : : const_iterator found = m . find ( folder ) ;
return ch = = ' \\ ' ? ' / ' : std : : tolower ( ch ) ;
}
if ( found = = m . end ( ) )
{
return false ;
}
else
current = found - > second ;
}
std : : vector < std : : string > : : iterator find = std : : lower_bound ( current . begin ( ) , current . end ( ) , copy , ciLessBoost ( ) ) ;
static std : : string normalize_path ( std : : string : : const_iterator begin , std : : string : : const_iterator end )
if ( find ! = current . end ( ) & & ! ciLessBoost ( ) ( copy , current . front ( ) ) )
{
{
std : : string normalized ;
if ( ! boost : : iequals ( copy , * find ) )
normalized . reserve ( end - begin ) ;
if ( ( find = std : : find_if ( current . begin ( ) , current . end ( ) , pathComparer ( copy ) ) ) = = current . end ( ) ) //\todo Check if this line is actually needed
char ( * normalize_char ) ( char ) = fsstrict ? & strict_normalize_char : & nonstrict_normalize_char ;
return false ;
std : : transform ( begin , end , std : : back_inserter ( normalized ) , normalize_char ) ;
return normalized ;
}
copy = * find ;
index : : const_iterator lookup_filename ( std : : string const & filename ) const
return true ;
{
}
std : : string normalized = normalize_path ( filename . begin ( ) , filename . end ( ) ) ;
return false ;
return mIndex . find ( normalized ) ;
}
}
public :
public :
DirArchive ( const String & name )
DirArchive ( const String & name )
: FileSystem Archive( name , " Dir " ) , currentdir ( name )
: Archive( name , " Dir " )
{
{
mType = " Dir " ;
typedef boost : : filesystem : : recursive_directory_iterator directory_iterator ;
std : : string s = name ;
cutoff = s . size ( ) + 1 ;
if ( fsstrict = = false )
populateMap ( currentdir ) ;
}
directory_iterator end ;
void populateMap ( boost : : filesystem : : path d ) {
//need to cut off first
size_t prefix = name . size ( ) ;
boost : : filesystem : : directory_iterator dir_iter ( d ) , dir_end ;
std : : vector < std : : string > filesind ;
if ( name . size ( ) > 0 & & name [ prefix - 1 ] ! = ' \\ ' & & name [ prefix - 1 ] ! = ' / ' )
for ( ; dir_iter ! = dir_end ; dir_iter + + )
+ + prefix ;
{
if ( boost : : filesystem : : is_directory ( * dir_iter ) )
for ( directory_iterator i ( name ) ; i ! = end ; + + i )
populateMap ( * dir_iter ) ;
else
{
{
std : : string s = dir_iter - > path ( ) . string ( ) ;
if ( boost : : filesystem : : is_directory ( * i ) )
std : : replace ( s . begin ( ) , s . end ( ) , ' \\ ' , ' / ' ) ;
continue ;
std : : string proper = i - > path ( ) . string ( ) ;
std : : string small ;
std : : string searchable = normalize_path ( proper . begin ( ) + prefix , proper . end ( ) ) ;
if ( cutoff < s . size ( ) )
small = s . substr ( cutoff , s . size ( ) - cutoff ) ;
else
small = s . substr ( cutoff - 1 , s . size ( ) - cutoff ) ;
filesind. push_back ( small ) ;
mIndex . insert ( std : : make_pair ( std : : move ( searchable ) , std : : move ( proper ) ) ) ;
}
}
}
}
std : : sort ( filesind . begin ( ) , filesind . end ( ) , ciLessBoost ( ) ) ;
std : : string small ;
std : : string original = d . string ( ) ;
std : : replace ( original . begin ( ) , original . end ( ) , ' \\ ' , ' / ' ) ;
if ( cutoff < original . size ( ) )
small = original . substr ( cutoff , original . size ( ) - cutoff ) ;
else
small = original . substr ( cutoff - 1 , original . size ( ) - cutoff ) ;
m [ small ] = filesind ;
}
bool isCaseSensitive ( ) const { return fsstrict ; }
bool isCaseSensitive ( ) const { return fsstrict ; }
@ -165,26 +127,76 @@ class DirArchive: public Ogre::FileSystemArchive
void load ( ) { }
void load ( ) { }
void unload ( ) { }
void unload ( ) { }
bool exists ( const String & filename ) {
DataStreamPtr open ( const String & filename , bool readonly = true ) const
std : : string copy ;
{
index : : const_iterator i = lookup_filename ( filename ) ;
if ( findFile ( filename , copy ) )
if ( i = = mIndex . end ( ) )
return FileSystemArchive : : exists ( copy ) ;
{
std : : ostringstream os ;
os < < " The file ' " < < filename < < " ' could not be found. " ;
throw std : : runtime_error ( os . str ( ) ) ;
}
return false ;
return openConstrainedFileDataStream ( i - > second . c_str ( ) ) ;
}
}
DataStreamPtr open ( const String & filename , bool readonly = true ) const
StringVectorPtr list ( bool recursive = true , bool dirs = false )
{
{
std : : string copy ;
StringVectorPtr ptr = StringVectorPtr ( new StringVector ( ) ) ;
std : : cout < < " DirArchive< " < < getName ( ) < < " >::list " < < std : : endl ;
return ptr ;
}
if ( findFile ( filename , copy ) )
FileInfoListPtr listFileInfo ( bool recursive = true , bool dirs = false )
return FileSystemArchive : : open ( copy , readonly ) ;
{
FileInfoListPtr ptr = FileInfoListPtr ( new FileInfoList ( ) ) ;
std : : cout < < " DirArchive< " < < getName ( ) < < " >::listFileInfo " < < std : : endl ;
return ptr ;
}
DataStreamPtr p ;
StringVectorPtr find ( const String & pattern , bool recursive = true ,
return p ;
bool dirs = false )
}
{
StringVectorPtr ptr = StringVectorPtr ( new StringVector ( ) ) ;
if ( pattern = = " * " )
for ( index : : const_iterator i = mIndex . begin ( ) ; i ! = mIndex . end ( ) ; + + i )
ptr - > push_back ( i - > first ) ;
return ptr ;
}
bool exists ( const String & filename )
{
return lookup_filename ( filename ) ! = mIndex . end ( ) ;
}
time_t getModifiedTime ( const String & ) { return 0 ; }
FileInfoListPtr findFileInfo ( const String & pattern , bool recursive = true ,
bool dirs = false ) const
{
FileInfoListPtr ptr = FileInfoListPtr ( new FileInfoList ( ) ) ;
index : : const_iterator i = lookup_filename ( pattern ) ;
if ( i ! = mIndex . end ( ) )
{
FileInfo fi ;
std : : size_t npos = i - > first . rfind ( ' / ' ) ;
fi . archive = this ;
fi . filename = npos ! = - 1 ? i - > first . substr ( npos ) : i - > first ;
fi . path = npos ! = - 1 ? i - > first . substr ( 0 , npos ) : " " ;
fi . compressedSize = fi . uncompressedSize = 0 ;
ptr - > push_back ( fi ) ;
}
return ptr ;
}
} ;
} ;
class BSAArchive : public Archive
class BSAArchive : public Archive