11#include <ext/sha256/sha256.h>
12#include <ext/zlib/zlib.h>
39ArchiveFileSystem::ArchiveFileSystem(
const string& fileName): fileName(fileName), ifsMutex(
"afs-ifs-mutex")
43 if (
ifs.is_open() ==
false) {
48 uint64_t fileInformationOffset;
49 ifs.seekg(-
static_cast<int64_t
>(
sizeof(fileInformationOffset)), ios::end);
50 ifs.read((
char*)&fileInformationOffset,
sizeof(fileInformationOffset));
51 ifs.seekg(fileInformationOffset, ios::beg);
54 while (
true ==
true) {
56 ifs.read((
char*)&nameSize,
sizeof(nameSize));
57 if (nameSize == 0)
break;
60 auto buffer =
new char[nameSize];
61 ifs.read(buffer, nameSize);
62 fileInformation.
name.append(buffer, nameSize);
64 ifs.read((
char*)&fileInformation.
bytes,
sizeof(fileInformation.
bytes));
67 ifs.read((
char*)&fileInformation.
offset,
sizeof(fileInformation.
offset));
89 auto _pathName = pathName;
90 if (_pathName.empty() ==
false && StringTools::endsWith(pathName,
"/") ==
false) _pathName+=
"/";
92 auto fileName = fileInformationIt.second.name;
93 if (StringTools::startsWith(
fileName, _pathName) ==
true) {
95 if (filter !=
nullptr && filter->
accept(
100 Console::println(
"StandardFileSystem::list(): Filter::accept(): " + pathName +
"/" +
fileName +
": " + exception.what());
103 files.push_back(StringTools::substring(
fileName, pathName.size()));
106 sort(files.begin(), files.end());
120 if (StringTools::startsWith(relativeFileName,
"./") ==
true) relativeFileName = StringTools::substring(relativeFileName, 2);
128 auto relativeFileName = pathName +
"/" +
fileName;
129 if (StringTools::startsWith(relativeFileName,
"./") ==
true) relativeFileName = StringTools::substring(relativeFileName, 2);
136 auto& fileInformation = fileInformationIt->second;
138 return fileInformation.executable;
147 auto relativeFileName = pathName +
"/" +
fileName;
148 if (StringTools::startsWith(relativeFileName,
"./") ==
true) relativeFileName = StringTools::substring(relativeFileName, 2);
155 auto& fileInformation = fileInformationIt->second;
157 return fileInformation.bytes;
168 unsigned char in[
CHUNK];
169 unsigned char out[
CHUNK];
172 strm.zalloc = Z_NULL;
174 strm.opaque = Z_NULL;
176 strm.next_in = Z_NULL;
177 ret = inflateInit(&strm);
179 throw FileSystemException(
"ArchiveFileSystem::decompress(): Error while decompressing: inflate init");
183 size_t outPosition = 0;
184 size_t inPosition = 0;
185 size_t inBytes = inContent.size();
188 auto inStartPosition = inPosition;
189 for (
size_t i = 0; i <
CHUNK; i++) {
190 if (inPosition == inBytes)
break;
191 in[i] = inContent[inPosition];
194 strm.avail_in = inPosition - inStartPosition;
195 if (strm.avail_in == 0)
break;
200 strm.avail_out =
CHUNK;
202 ret = inflate(&strm, Z_NO_FLUSH);
203 assert(ret != Z_STREAM_ERROR);
206 throw FileSystemException(
"ArchiveFileSystem::decompress(): Error while decompressing: Z_NEED_DICT");
209 (void)inflateEnd(&strm);
210 throw FileSystemException(
"ArchiveFileSystem::decompress(): Error while decompressing: Z_DATA_ERROR | Z_MEM_ERROR");
212 have =
CHUNK - strm.avail_out;
213 for (
size_t i = 0; i < have; i++) {
214 outContent[outPosition++] = out[i];
216 }
while (strm.avail_out == 0);
219 }
while (ret != Z_STREAM_END);
222 (void) inflateEnd(&strm);
225 if (ret != Z_STREAM_END) {
226 throw FileSystemException(
"ArchiveFileSystem::decompress(): Error while decompressing: missing eof");
232 auto relativeFileName = pathName +
"/" +
fileName;
233 if (StringTools::startsWith(relativeFileName,
"./") ==
true) relativeFileName = StringTools::substring(relativeFileName, 2);
240 auto& fileInformation = fileInformationIt->second;
246 ifs.seekg(fileInformation.offset, ios::beg);
250 if (fileInformation.compressed == 1) {
251 vector<uint8_t> compressedBuffer;
252 compressedBuffer.resize(fileInformation.bytesCompressed);
253 ifs.read((
char*)compressedBuffer.data(), fileInformation.bytesCompressed);
255 vector<uint8_t> decompressedBuffer;
256 decompressedBuffer.resize(fileInformation.bytes);
257 decompress(compressedBuffer, decompressedBuffer);
258 result.append((
char*)decompressedBuffer.data(), fileInformation.bytes);
260 vector<uint8_t> buffer;
261 buffer.resize(fileInformation.bytes);
262 ifs.read((
char*)buffer.data(), fileInformation.bytes);
264 result.append((
char*)buffer.data(), fileInformation.bytes);
272 throw FileSystemException(
"ArchiveFileSystem::setContentFromString(): Not implemented yet");
278 auto relativeFileName = pathName +
"/" +
fileName;
279 if (StringTools::startsWith(relativeFileName,
"./") ==
true) relativeFileName = StringTools::substring(relativeFileName, 2);
286 auto& fileInformation = fileInformationIt->second;
292 ifs.seekg(fileInformation.offset, ios::beg);
295 if (fileInformation.compressed == 1) {
296 vector<uint8_t> compressedContent;
297 compressedContent.resize(fileInformation.bytesCompressed);
298 ifs.read((
char*)compressedContent.data(), fileInformation.bytesCompressed);
300 content.resize(fileInformation.bytes);
303 content.resize(fileInformation.bytes);
304 ifs.read((
char*)content.data(), fileInformation.bytes);
316 contentAsString = StringTools::replace(contentAsString,
"\r",
"");
326 throw FileSystemException(
"ArchiveFileSystem::setContentFromStringArray(): Not implemented yet");
330 string unixPathName = StringTools::replace(pathName,
"\\",
"/");
331 string unixFileName = StringTools::replace(
fileName,
"\\",
"/");
333 auto pathString =
getFileName(unixPathName, unixFileName);
336 vector<string> pathComponents;
344 for (
auto i = 0; i < pathComponents.size(); i++) {
345 auto pathComponent = pathComponents[i];
346 if (pathComponent ==
".") {
347 pathComponents[i] =
"";
349 if (pathComponent ==
"..") {
350 pathComponents[i]=
"";
352 for (
int pathComponentReplaced = 0; pathComponentReplaced < 1 && j >= 0; ) {
353 if (pathComponents[j] !=
"") {
354 pathComponents[j] =
"";
355 pathComponentReplaced++;
363 string canonicalPath =
"";
364 bool slash = StringTools::startsWith(pathString,
"/");
365 for (
auto i = 0; i < pathComponents.size(); i++) {
366 auto pathComponent = pathComponents[i];
367 if (pathComponent ==
"") {
370 canonicalPath = canonicalPath + (slash ==
true?
"/":
"") + pathComponent;
376 auto canonicalPathString = canonicalPath;
377 if (canonicalPathString.length() == 0 ||
378 (StringTools::startsWith(canonicalPathString,
"/") ==
false &&
379 StringTools::regexMatch(canonicalPathString,
"^[a-zA-Z]\\:.*$") ==
false)) {
384 return canonicalPathString;
395 string unixFileName = StringTools::replace(
fileName, L
'\\', L
'/');
396 int32_t lastPathSeparator = StringTools::lastIndexOf(unixFileName, L
'/');
397 if (lastPathSeparator == -1)
return ".";
398 return StringTools::substring(unixFileName, 0, lastPathSeparator);
402 string unixFileName = StringTools::replace(
fileName, L
'\\', L
'/');
403 int32_t lastPathSeparator = StringTools::lastIndexOf(unixFileName, L
'/');
404 if (lastPathSeparator == -1)
return fileName;
405 return StringTools::substring(unixFileName, lastPathSeparator + 1, unixFileName.length());
429 ifs.seekg(0, ios::end);
430 auto bytesTotal =
ifs.tellg();
431 ifs.seekg(0, ios::beg);
433 uint8_t input[16384];
434 unsigned char digest[SHA256::DIGEST_SIZE];
435 memset(digest, 0, SHA256::DIGEST_SIZE);
439 int64_t bytesRead = 0LL;
440 while (bytesRead < bytesTotal) {
441 auto bytesToRead = Math::min(
static_cast<int64_t
>(bytesTotal) - bytesRead,
sizeof(input));
442 ifs.read((
char*)input, bytesToRead);
443 ctx.update((
const uint8_t*)input, bytesToRead);
444 bytesRead+= bytesToRead;
448 char buf[2 * SHA256::DIGEST_SIZE + 1];
449 buf[2 * SHA256::DIGEST_SIZE] = 0;
450 for (
int i = 0; i < SHA256::DIGEST_SIZE; i++) sprintf(buf + i * 2,
"%02x", digest[i]);
451 return std::string(buf);
Archive file system implementation.
const string getPathName(const string &fileName) override
Get path name.
map< string, FileInformation > fileInformations
bool isPath(const string &pathName) override
Check if file is a path.
virtual ~ArchiveFileSystem()
Public destructor.
uint64_t getFileSize(const string &pathName, const string &fileName) override
Return file size of given file.
void setContent(const string &pathName, const string &fileName, const vector< uint8_t > &content) override
Set file content.
void removeFile(const string &pathName, const string &fileName) override
Remove file.
void removePath(const string &pathName, bool recursive) override
Remove path.
void decompress(vector< uint8_t > &inContent, vector< uint8_t > &outContent)
Decompress from archive.
void createPath(const string &pathName) override
Create path.
const string getCurrentWorkingPathName() override
Get current working path name.
const string getCanonicalPath(const string &pathName, const string &fileName) override
Get canonical path name.
void setExecutable(const string &pathName, const string &fileName) override
Set up file to be an executable file.
const string & getArchiveFileName()
void changePath(const string &pathName) override
Change path.
void setContentFromString(const string &pathName, const string &fileName, const string &content) override
Set content from string.
const string computeSHA256Hash()
Compute SHA256 hash.
void getContentAsStringArray(const string &pathName, const string &fileName, vector< string > &content) override
Get file content as string array.
void setContentFromStringArray(const string &pathName, const string &fileName, const vector< string > &content) override
Set file content as string array.
const string getFileName(const string &path, const string &fileName) override
Get file name.
bool fileExists(const string &fileName) override
Check if file exists.
bool isDrive(const string &pathName) override
Check if file is a drive (applies to Microsoft Windows only)
bool isExecutable(const string &pathName, const string &fileName) override
Returns if file is a executable file.
void list(const string &pathName, vector< string > &files, FileNameFilter *filter=nullptr, bool addDrives=false) override
List files for given path and filter by a file name filter if not null.
void getContent(const string &pathName, const string &fileName, vector< uint8_t > &content) override
Get file content.
bool getThumbnailAttachment(const string &pathName, const string &fileName, vector< uint8_t > &thumbnailAttachmentContent) override
Reads a thumbnail attachment from binary file.
const string getContentAsString(const string &pathName, const string &fileName) override
Get content as string.
void unlock()
Unlocks this mutex.
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
const string & nextToken()
void tokenize(const string &str, const string &delimiters)
Tokenize.
std::exception Exception
Exception base class.
File system file name filter interface.
virtual bool accept(const string &path, const string &file)=0
Accept a file.