TDME2 1.9.121
ArchiveFileSystem.cpp
Go to the documentation of this file.
2
3#include <string.h>
4
5#include <algorithm>
6#include <cassert>
7#include <fstream>
8#include <string>
9#include <vector>
10
11#include <ext/sha256/sha256.h>
12#include <ext/zlib/zlib.h>
13
14#include <tdme/tdme.h>
15#include <tdme/math/Math.h>
23
24using std::ifstream;
25using std::ios;
26using std::sort;
27using std::string;
28using std::to_string;
29using std::vector;
30
38
39ArchiveFileSystem::ArchiveFileSystem(const string& fileName): fileName(fileName), ifsMutex("afs-ifs-mutex")
40{
41 // open
42 ifs.open(fileName.c_str(), ifstream::binary);
43 if (ifs.is_open() == false) {
44 throw FileSystemException("Unable to open file for reading(" + to_string(errno) + "): " + fileName);
45 }
46
47 // read toc offset
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);
52
53 // read toc
54 while (true == true) {
55 uint32_t nameSize;
56 ifs.read((char*)&nameSize, sizeof(nameSize));
57 if (nameSize == 0) break;
58
59 FileInformation fileInformation;
60 auto buffer = new char[nameSize];
61 ifs.read(buffer, nameSize);
62 fileInformation.name.append(buffer, nameSize);
63 delete [] buffer;
64 ifs.read((char*)&fileInformation.bytes, sizeof(fileInformation.bytes));
65 ifs.read((char*)&fileInformation.compressed, sizeof(fileInformation.compressed));
66 ifs.read((char*)&fileInformation.bytesCompressed, sizeof(fileInformation.bytesCompressed));
67 ifs.read((char*)&fileInformation.offset, sizeof(fileInformation.offset));
68 ifs.read((char*)&fileInformation.executable, sizeof(fileInformation.executable));
69 fileInformations[fileInformation.name] = fileInformation;
70 }
71}
72
74{
75 ifs.close();
76}
77
79 return fileName;
80}
81
82const string ArchiveFileSystem::getFileName(const string& pathName, const string& fileName) {
83 return pathName + "/" + fileName;
84}
85
86void ArchiveFileSystem::list(const string& pathName, vector<string>& files, FileNameFilter* filter, bool addDrives)
87{
88 // TODO: this currently lists all files beginning from given path, also files in sub folders
89 auto _pathName = pathName;
90 if (_pathName.empty() == false && StringTools::endsWith(pathName, "/") == false) _pathName+= "/";
91 for (auto& fileInformationIt: fileInformations) {
92 auto fileName = fileInformationIt.second.name;
93 if (StringTools::startsWith(fileName, _pathName) == true) {
94 try {
95 if (filter != nullptr && filter->accept(
98 ) == false) continue;
99 } catch (Exception& exception) {
100 Console::println("StandardFileSystem::list(): Filter::accept(): " + pathName + "/" + fileName + ": " + exception.what());
101 continue;
102 }
103 files.push_back(StringTools::substring(fileName, pathName.size()));
104 }
105 }
106 sort(files.begin(), files.end());
107}
108
109bool ArchiveFileSystem::isPath(const string& pathName) {
110 return false;
111}
112
113bool ArchiveFileSystem::isDrive(const string& pathName) {
114 return false;
115}
116
117bool ArchiveFileSystem::fileExists(const string& fileName) {
118 // compose relative file name and remove ./
119 auto relativeFileName = fileName;
120 if (StringTools::startsWith(relativeFileName, "./") == true) relativeFileName = StringTools::substring(relativeFileName, 2);
121
122 //
123 return fileInformations.find(relativeFileName) != fileInformations.end();
124}
125
126bool ArchiveFileSystem::isExecutable(const string& pathName, const string& fileName) {
127 // compose relative file name and remove ./
128 auto relativeFileName = pathName + "/" + fileName;
129 if (StringTools::startsWith(relativeFileName, "./") == true) relativeFileName = StringTools::substring(relativeFileName, 2);
130
131 // determine file
132 auto fileInformationIt = fileInformations.find(relativeFileName);
133 if (fileInformationIt == fileInformations.end()) {
134 throw FileSystemException("Unable to open file for reading: " + relativeFileName + ": " + pathName + "/" + fileName);
135 }
136 auto& fileInformation = fileInformationIt->second;
137 //
138 return fileInformation.executable;
139}
140
141void ArchiveFileSystem::setExecutable(const string& pathName, const string& fileName) {
142 throw FileSystemException("ArchiveFileSystem::createPath(): Not implemented yet");
143}
144
145uint64_t ArchiveFileSystem::getFileSize(const string& pathName, const string& fileName) {
146 // compose relative file name and remove ./
147 auto relativeFileName = pathName + "/" + fileName;
148 if (StringTools::startsWith(relativeFileName, "./") == true) relativeFileName = StringTools::substring(relativeFileName, 2);
149
150 // determine file
151 auto fileInformationIt = fileInformations.find(relativeFileName);
152 if (fileInformationIt == fileInformations.end()) {
153 throw FileSystemException("Unable to open file for reading: " + relativeFileName + ": " + pathName + "/" + fileName);
154 }
155 auto& fileInformation = fileInformationIt->second;
156 //
157 return fileInformation.bytes;
158}
159
160void ArchiveFileSystem::decompress(vector<uint8_t>& inContent, vector<uint8_t>& outContent) {
161 // see: https://www.zlib.net/zpipe.c
162
163 #define CHUNK 16384
164
165 int ret;
166 size_t have;
167 z_stream strm;
168 unsigned char in[CHUNK];
169 unsigned char out[CHUNK];
170
171 // allocate inflate state
172 strm.zalloc = Z_NULL;
173 strm.zfree = Z_NULL;
174 strm.opaque = Z_NULL;
175 strm.avail_in = 0;
176 strm.next_in = Z_NULL;
177 ret = inflateInit(&strm);
178 if (ret != Z_OK) {
179 throw FileSystemException("ArchiveFileSystem::decompress(): Error while decompressing: inflate init");
180 }
181
182 // decompress until deflate stream ends or end of file */
183 size_t outPosition = 0;
184 size_t inPosition = 0;
185 size_t inBytes = inContent.size();
186 do {
187 // see: https://www.zlib.net/zpipe.c
188 auto inStartPosition = inPosition;
189 for (size_t i = 0; i < CHUNK; i++) {
190 if (inPosition == inBytes) break;
191 in[i] = inContent[inPosition];
192 inPosition++;
193 }
194 strm.avail_in = inPosition - inStartPosition;
195 if (strm.avail_in == 0) break;
196 strm.next_in = in;
197
198 // run inflate() on input until output buffer not full
199 do {
200 strm.avail_out = CHUNK;
201 strm.next_out = out;
202 ret = inflate(&strm, Z_NO_FLUSH);
203 assert(ret != Z_STREAM_ERROR); // state not clobbered
204 switch (ret) {
205 case Z_NEED_DICT:
206 throw FileSystemException("ArchiveFileSystem::decompress(): Error while decompressing: Z_NEED_DICT");
207 case Z_DATA_ERROR:
208 case Z_MEM_ERROR:
209 (void)inflateEnd(&strm);
210 throw FileSystemException("ArchiveFileSystem::decompress(): Error while decompressing: Z_DATA_ERROR | Z_MEM_ERROR");
211 }
212 have = CHUNK - strm.avail_out;
213 for (size_t i = 0; i < have; i++) {
214 outContent[outPosition++] = out[i];
215 }
216 } while (strm.avail_out == 0);
217
218 // done when inflate() says it's done
219 } while (ret != Z_STREAM_END);
220
221 // clean up and return
222 (void) inflateEnd(&strm);
223
224 // check if eof
225 if (ret != Z_STREAM_END) {
226 throw FileSystemException("ArchiveFileSystem::decompress(): Error while decompressing: missing eof");
227 }
228}
229
230const string ArchiveFileSystem::getContentAsString(const string& pathName, const string& fileName) {
231 // compose relative file name and remove ./
232 auto relativeFileName = pathName + "/" + fileName;
233 if (StringTools::startsWith(relativeFileName, "./") == true) relativeFileName = StringTools::substring(relativeFileName, 2);
234
235 // determine file
236 auto fileInformationIt = fileInformations.find(relativeFileName);
237 if (fileInformationIt == fileInformations.end()) {
238 throw FileSystemException("Unable to open file for reading: " + relativeFileName + ": " + pathName + "/" + fileName);
239 }
240 auto& fileInformation = fileInformationIt->second;
241
242 //
243 ifsMutex.lock();
244
245 // seek
246 ifs.seekg(fileInformation.offset, ios::beg);
247
248 // result
249 string result;
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);
259 } else {
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);
265 }
266
267 // done
268 return result;
269}
270
271void ArchiveFileSystem::setContentFromString(const string& pathName, const string& fileName, const string& content) {
272 throw FileSystemException("ArchiveFileSystem::setContentFromString(): Not implemented yet");
273}
274
275void ArchiveFileSystem::getContent(const string& pathName, const string& fileName, vector<uint8_t>& content)
276{
277 // compose relative file name and remove ./
278 auto relativeFileName = pathName + "/" + fileName;
279 if (StringTools::startsWith(relativeFileName, "./") == true) relativeFileName = StringTools::substring(relativeFileName, 2);
280
281 // determine file
282 auto fileInformationIt = fileInformations.find(relativeFileName);
283 if (fileInformationIt == fileInformations.end()) {
284 throw FileSystemException("Unable to open file for reading: " + relativeFileName + ": " + pathName + "/" + fileName);
285 }
286 auto& fileInformation = fileInformationIt->second;
287
288 //
289 ifsMutex.lock();
290
291 // seek
292 ifs.seekg(fileInformation.offset, ios::beg);
293
294 // result
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);
301 decompress(compressedContent, content);
302 } else {
303 content.resize(fileInformation.bytes);
304 ifs.read((char*)content.data(), fileInformation.bytes);
306 }
307}
308
309void ArchiveFileSystem::setContent(const string& pathName, const string& fileName, const vector<uint8_t>& content) {
310 throw FileSystemException("ArchiveFileSystem::setContent(): Not implemented yet");
311}
312
313void ArchiveFileSystem::getContentAsStringArray(const string& pathName, const string& fileName, vector<string>& content)
314{
315 auto contentAsString = getContentAsString(pathName, fileName);
316 contentAsString = StringTools::replace(contentAsString, "\r", "");
318 t.tokenize(contentAsString, "\n");
319 while (t.hasMoreTokens() == true) {
320 content.push_back(t.nextToken());
321 }
322}
323
324void ArchiveFileSystem::setContentFromStringArray(const string& pathName, const string& fileName, const vector<string>& content)
325{
326 throw FileSystemException("ArchiveFileSystem::setContentFromStringArray(): Not implemented yet");
327}
328
329const string ArchiveFileSystem::getCanonicalPath(const string& pathName, const string& fileName) {
330 string unixPathName = StringTools::replace(pathName, "\\", "/");
331 string unixFileName = StringTools::replace(fileName, "\\", "/");
332
333 auto pathString = getFileName(unixPathName, unixFileName);
334
335 // separate into path components
336 vector<string> pathComponents;
338 t.tokenize(pathString, "/");
339 while (t.hasMoreTokens()) {
340 pathComponents.push_back(t.nextToken());
341 }
342
343 // process path components
344 for (auto i = 0; i < pathComponents.size(); i++) {
345 auto pathComponent = pathComponents[i];
346 if (pathComponent == ".") {
347 pathComponents[i] = "";
348 } else
349 if (pathComponent == "..") {
350 pathComponents[i]= "";
351 int j = i - 1;
352 for (int pathComponentReplaced = 0; pathComponentReplaced < 1 && j >= 0; ) {
353 if (pathComponents[j] != "") {
354 pathComponents[j] = "";
355 pathComponentReplaced++;
356 }
357 j--;
358 }
359 }
360 }
361
362 // process path components
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 == "") {
368 // no op
369 } else {
370 canonicalPath = canonicalPath + (slash == true?"/":"") + pathComponent;
371 slash = true;
372 }
373 }
374
375 // add cwd if required
376 auto canonicalPathString = canonicalPath;
377 if (canonicalPathString.length() == 0 ||
378 (StringTools::startsWith(canonicalPathString, "/") == false &&
379 StringTools::regexMatch(canonicalPathString, "^[a-zA-Z]\\:.*$") == false)) {
380 canonicalPathString = getCurrentWorkingPathName() + "/" + canonicalPathString;
381 }
382
383 //
384 return canonicalPathString;
385}
386
388 return ".";
389}
390
391void ArchiveFileSystem::changePath(const string& pathName) {
392}
393
394const string ArchiveFileSystem::getPathName(const string& fileName) {
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);
399}
400
401const string ArchiveFileSystem::getFileName(const string& fileName) {
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());
406}
407
408void ArchiveFileSystem::createPath(const string& pathName) {
409 throw FileSystemException("ArchiveFileSystem::createPath(): Not implemented yet");
410}
411
412void ArchiveFileSystem::removePath(const string& pathName, bool recursive) {
413 throw FileSystemException("ArchiveFileSystem::removePath(): Not implemented yet");
414}
415
416void ArchiveFileSystem::removeFile(const string& pathName, const string& fileName) {
417 throw FileSystemException("ArchiveFileSystem::removeFile(): Not implemented yet");
418}
419
420bool ArchiveFileSystem::getThumbnailAttachment(const string& pathName, const string& fileName, vector<uint8_t>& thumbnailAttachmentContent) {
421 throw FileSystemException("ArchiveFileSystem::removeFile(): Not implemented yet");
422}
423
424bool ArchiveFileSystem::getThumbnailAttachment(const vector<uint8_t>& content, vector<uint8_t>& thumbnailAttachmentContent) {
425 throw FileSystemException("ArchiveFileSystem::removeFile(): Not implemented yet");
426}
427
429 ifs.seekg(0, ios::end);
430 auto bytesTotal = ifs.tellg();
431 ifs.seekg(0, ios::beg);
432
433 uint8_t input[16384];
434 unsigned char digest[SHA256::DIGEST_SIZE];
435 memset(digest, 0, SHA256::DIGEST_SIZE);
436
437 auto ctx = SHA256();
438 ctx.init();
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;
445 }
446 ctx.final(digest);
447
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);
452}
453
#define CHUNK
Standard math functions.
Definition: Math.h:21
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.
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.
Mutex implementation.
Definition: Mutex.h:27
void unlock()
Unlocks this mutex.
Definition: Mutex.cpp:48
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
Definition: Mutex.cpp:39
Console class.
Definition: Console.h:26
String tokenizer class.
void tokenize(const string &str, const string &delimiters)
Tokenize.
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19
File system file name filter interface.
virtual bool accept(const string &path, const string &file)=0
Accept a file.