TDME2 1.9.121
StandardFileSystem.cpp
Go to the documentation of this file.
2
3#if defined(_WIN32) && defined(_MSC_VER)
4 #include <direct.h>
5# else
6 #include <unistd.h>
7#endif
8
9#include <algorithm>
10#include <array>
11#include <dirent.h>
12#include <limits.h>
13#include <stdlib.h>
14#include <sys/stat.h>
15
16#include <fstream>
17#include <iostream>
18#include <sstream>
19#include <string>
20#include <vector>
21
22#include <tdme/tdme.h>
29
30using std::array;
31using std::getline;
32using std::ifstream;
33using std::ios;
34using std::ofstream;
35using std::sort;
36using std::string;
37using std::stringstream;
38using std::to_string;
39using std::vector;
40
42
48
49StandardFileSystem::StandardFileSystem()
50{
51}
52
54{
55}
56
57const string StandardFileSystem::getFileName(const string& pathName, const string& fileName) {
58 return pathName + "/" + fileName;
59}
60
61void StandardFileSystem::list(const string& pathName, vector<string>& files, FileNameFilter* filter, bool addDrives)
62{
63 auto _pathName = pathName;
64 if (StringTools::endsWith(pathName, "/") == false) _pathName+= "/";
65
66 DIR* dir = nullptr;
67 struct dirent* dirent = nullptr;
68 if ((dir = opendir(_pathName.c_str())) == nullptr) {
69 throw FileSystemException("Unable to list path(" + to_string(errno) + "): " + _pathName);
70 }
71 while ((dirent = readdir(dir)) != nullptr) {
72 string fileName = dirent->d_name;
73 if (fileName == ".") continue;
74 try {
75 if (filter != nullptr && filter->accept(pathName, fileName) == false) continue;
76 } catch (Exception& exception) {
77 Console::println("StandardFileSystem::list(): Filter::accept(): " + pathName + "/" + fileName + ": " + exception.what());
78 continue;
79 }
80 files.push_back(fileName);
81 }
82 sort(files.begin(), files.end());
83
84 #if defined(_WIN32)
85 if (addDrives == true) {
86 for (char drive = 'A'; drive <= 'Z'; drive++) {
87 string fileName;
88 fileName+= drive;
89 fileName+= ":";
90 try {
91 if (fileExists(fileName + "/") == true) files.insert(files.begin() + (drive - 'C'), fileName);
92 } catch (Exception& exception) {
93 Console::println("StandardFileSystem::list(): fileExists(): " + pathName + "/" + fileName + ": " + exception.what());
94 }
95 }
96 }
97 #endif
98
99 closedir(dir);
100}
101
102bool StandardFileSystem::isPath(const string& pathName) {
103 struct stat s;
104 if (stat(pathName.c_str(), &s) == 0) {
105 return (s.st_mode & S_IFDIR) == S_IFDIR;
106 } else {
107 throw FileSystemException("Unable to check if path(" + to_string(errno) + "): " + pathName);
108 }
109}
110
111bool StandardFileSystem::isDrive(const string& pathName) {
112 return StringTools::regexMatch(pathName, "^[a-zA-Z]\\:[\\/\\\\]?$");
113}
114
115bool StandardFileSystem::fileExists(const string& fileName) {
116 struct stat s;
117 return stat(fileName.c_str(), &s) == 0;
118}
119
120bool StandardFileSystem::isExecutable(const string& pathName, const string& fileName) {
121 struct stat s;
122 if (stat((pathName + "/" + fileName).c_str(), &s) != 0) return false;
123 return (s.st_mode & S_IXUSR) == S_IXUSR;
124}
125
126void StandardFileSystem::setExecutable(const string& pathName, const string& fileName) {
127 #if !defined(_MSC_VER)
128 struct stat s;
129 if (stat((pathName + "/" + fileName).c_str(), &s) != 0) {
130 throw FileSystemException("Unable to set file to executable(" + to_string(errno) + "): " + pathName + "/" + fileName);
131 }
132 auto newMode = s.st_mode;
133 if ((s.st_mode & S_IRUSR) == S_IRUSR && (s.st_mode & S_IXUSR) == 0) newMode|= S_IXUSR;
134 if ((s.st_mode & S_IRGRP) == S_IRGRP && (s.st_mode & S_IXGRP) == 0) newMode|= S_IXGRP;
135 if ((s.st_mode & S_IROTH) == S_IROTH && (s.st_mode & S_IXOTH) == 0) newMode|= S_IXOTH;
136 if (chmod((pathName + "/" + fileName).c_str(), newMode) < 0) {
137 throw FileSystemException("Unable to set file to executable(" + to_string(errno) + "): " + pathName + "/" + fileName);
138 }
139 #endif
140}
141
142uint64_t StandardFileSystem::getFileSize(const string& pathName, const string& fileName) {
143 ifstream ifs(getFileName(pathName, fileName).c_str(), ifstream::binary);
144 if (ifs.is_open() == false) {
145 throw FileSystemException("Unable to open file for reading(" + to_string(errno) + "): " + pathName + "/" + fileName);
146 }
147 ifs.seekg( 0, ios::end );
148 size_t size = ifs.tellg();
149 ifs.close();
150 return size;
151}
152
153const string StandardFileSystem::getContentAsString(const string& pathName, const string& fileName) {
154 ifstream ifs(getFileName(pathName, fileName).c_str());
155 if (ifs.is_open() == false) {
156 throw FileSystemException("Unable to open file for reading(" + to_string(errno) + "): " + pathName + "/" + fileName);
157 }
158 stringstream stringStream;
159 stringStream << ifs.rdbuf();
160 ifs.close();
161 return (stringStream.str());
162}
163
164void StandardFileSystem::setContentFromString(const string& pathName, const string& fileName, const string& content) {
165 ofstream ofs(getFileName(pathName, fileName).c_str());
166 if (ofs.is_open() == false) {
167 throw FileSystemException("Unable to open file for writing(" + to_string(errno) + "): " + pathName + "/" + fileName);
168 }
169 ofs << (content);
170 ofs.close();
171 return;
172}
173
174void StandardFileSystem::getContent(const string& pathName, const string& fileName, vector<uint8_t>& content)
175{
176 ifstream ifs(getFileName(pathName, fileName).c_str(), ifstream::binary);
177 if (ifs.is_open() == false) {
178 throw FileSystemException("Unable to open file for reading(" + to_string(errno) + "): " + pathName + "/" + fileName);
179 }
180 ifs.seekg( 0, ios::end );
181 size_t size = ifs.tellg();
182 content.resize(size);
183 ifs.seekg(0, ios::beg);
184 ifs.read((char*)content.data(), size);
185 ifs.close();
186}
187
188void StandardFileSystem::setContent(const string& pathName, const string& fileName, const vector<uint8_t>& content) {
189 ofstream ofs(getFileName(pathName, fileName).c_str(), ofstream::binary);
190 if (ofs.is_open() == false) {
191 throw FileSystemException("Unable to open file for writing(" + to_string(errno) + "): " + pathName + "/" + fileName);
192 }
193 ofs.write((char*)content.data(), content.size());
194 ofs.close();
195}
196
197void StandardFileSystem::getContentAsStringArray(const string& pathName, const string& fileName, vector<string>& content)
198{
199 ifstream ifs(getFileName(pathName, fileName).c_str());
200 if(ifs.is_open() == false) {
201 throw FileSystemException("Unable to open file for reading(" + to_string(errno) + "): " + pathName + "/" + fileName);
202 }
203
204 string line;
205 while (getline(ifs, line)) {
206 content.push_back((line));
207 }
208
209 ifs.close();
210}
211
212void StandardFileSystem::setContentFromStringArray(const string& pathName, const string& fileName, const vector<string>& content)
213{
214 ofstream ofs(getFileName(pathName, fileName).c_str(), ofstream::binary);
215 if(ofs.is_open() == false) {
216 throw FileSystemException("Unable to open file for writing(" + to_string(errno) + "): " + pathName + "/" + fileName);
217 }
218
219 for (auto i = 0; i < content.size(); i++) {
220 ofs << (content.at(i)) << "\n";
221 }
222
223 ofs.close();
224 return;
225}
226
227const string StandardFileSystem::getCanonicalPath(const string& pathName, const string& fileName) {
228 string unixPathName = StringTools::replace(pathName, "\\", "/");
229 string unixFileName = StringTools::replace(fileName, "\\", "/");
230
231 auto pathString = getFileName(unixPathName, unixFileName);
232
233 // separate into path components
234 vector<string> pathComponents;
236 t.tokenize(pathString, "/");
237 while (t.hasMoreTokens()) {
238 pathComponents.push_back(t.nextToken());
239 }
240
241 // process path components
242 for (auto i = 0; i < pathComponents.size(); i++) {
243 auto pathComponent = pathComponents[i];
244 if (pathComponent == ".") {
245 pathComponents[i] = "";
246 } else
247 if (pathComponent == "..") {
248 pathComponents[i]= "";
249 int j = i - 1;
250 for (int pathComponentReplaced = 0; pathComponentReplaced < 1 && j >= 0; ) {
251 if (pathComponents[j] != "") {
252 pathComponents[j] = "";
253 pathComponentReplaced++;
254 }
255 j--;
256 }
257 }
258 }
259
260 // process path components
261 string canonicalPath = "";
262 bool slash = StringTools::startsWith(pathString, "/");
263 for (auto i = 0; i < pathComponents.size(); i++) {
264 auto pathComponent = pathComponents[i];
265 if (pathComponent == "") {
266 // no op
267 } else {
268 canonicalPath = canonicalPath + (slash == true?"/":"") + pathComponent;
269 slash = true;
270 }
271 }
272
273 // add cwd if required
274 auto canonicalPathString = canonicalPath;
275 if (canonicalPathString.length() == 0 ||
276 (StringTools::startsWith(canonicalPathString, "/") == false &&
277 StringTools::regexMatch(canonicalPathString, "^[a-zA-Z]\\:.*$") == false)) {
278 canonicalPathString = getCurrentWorkingPathName() + "/" + canonicalPathString;
279 }
280
281 //
282 return canonicalPathString;
283}
284
286 char cwdBuffer[PATH_MAX + 1];
287 char* cwdPtr = getcwd(cwdBuffer, sizeof(cwdBuffer));
288 if (cwdPtr == nullptr) {
289 throw FileSystemException("Unable to get current working path(" + to_string(errno) + ")");
290 }
291 auto cwd = string(cwdPtr);
292 return StringTools::replace(cwd, "\\", "/");
293}
294
295void StandardFileSystem::changePath(const string& pathName) {
296 if (chdir(pathName.c_str()) != 0) {
297 throw FileSystemException("Unable to change path(" + to_string(errno) + "): " + pathName);
298 }
299}
300
301const string StandardFileSystem::getPathName(const string& fileName) {
302 string unixFileName = StringTools::replace(fileName, '\\', '/');
303 int32_t lastPathSeparator = StringTools::lastIndexOf(unixFileName, L'/');
304 if (lastPathSeparator == -1) return ".";
305 return StringTools::substring(unixFileName, 0, lastPathSeparator);
306}
307
308const string StandardFileSystem::getFileName(const string& fileName) {
309 string unixFileName = StringTools::replace(fileName, '\\', '/');
310 int32_t lastPathSeparator = StringTools::lastIndexOf(unixFileName, L'/');
311 if (lastPathSeparator == -1) return fileName;
312 return StringTools::substring(unixFileName, lastPathSeparator + 1, unixFileName.length());
313}
314
315void StandardFileSystem::createPath(const string& pathName) {
316 #if defined(_WIN32)
317 int32_t status = mkdir(pathName.c_str());
318 #else
319 int32_t status = mkdir(pathName.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH);
320 #endif
321 if (status == -1) {
322 throw FileSystemException("Unable to create path(" + to_string(errno) + "): " + pathName);
323 }
324}
325
326void StandardFileSystem::removePath(const string& pathName, bool recursive) {
327 if (recursive == true) {
328 vector<string> files;
329 list(pathName, files, nullptr);
330 for (auto i = 0; i < files.size(); i++) {
331 auto file = files[i];
332 if (file == "." || file == "..") {
333 continue;
334 }
335 auto completeFileName = getFileName(pathName, file);
336 if (isPath(completeFileName)) {
337 removePath(completeFileName, true);
338 } else {
339 removeFile(pathName, file);
340 }
341 }
342 }
343 Console::println(string("StandardFileSystem::removePath(): Removing ") + pathName);
344 int32_t status = rmdir(pathName.c_str());
345 if (status == -1) {
346 throw FileSystemException("Unable to delete folder(" + to_string(errno) + "): " + pathName);
347 }
348}
349
350void StandardFileSystem::removeFile(const string& pathName, const string& fileName) {
351 Console::println(string("StandardFileSystem::removeFile(): Removing ") + getFileName(pathName, fileName));
352 int32_t status = unlink(getFileName(pathName, fileName).c_str());
353 if (status == -1) {
354 throw FileSystemException("Unable to delete file(" + to_string(errno) + "): " + pathName + "/" + fileName);
355 }
356}
357
358bool StandardFileSystem::getThumbnailAttachment(const string& pathName, const string& fileName, vector<uint8_t>& thumbnailAttachmentContent) {
359 ifstream ifs(getFileName(pathName, fileName).c_str(), ifstream::binary);
360 if (ifs.is_open() == false) {
361 throw FileSystemException("Unable to open file for reading(" + to_string(errno) + "): " + pathName + "/" + fileName);
362 }
363
364 // check size
365 ifs.seekg( 0, ios::end );
366 size_t size = ifs.tellg();
367 if (size < 12) return false;
368
369 array<uint8_t, 12> id;
370
371 ifs.seekg(size - 12, ios::beg);
372 ifs.read((char*)id.data(), 12);
373
374 // attachment
375 if (id[8] != 'A' || id[9] != 'T' || id[10] != 'M' || id[11] != 'T') {
376 return false;
377 }
378 // thumbnail
379 if (id[4] != 'T' || id[5] != 'M' || id[6] != 'B' || id[7] != 'N') {
380 return false;
381 }
382 // attachment size
383 int32_t attachmentSize =
384 ((static_cast<int32_t>(id[0]) & 0xFF) << 24) +
385 ((static_cast<int32_t>(id[1]) & 0xFF) << 16) +
386 ((static_cast<int32_t>(id[2]) & 0xFF) << 8) +
387 ((static_cast<int32_t>(id[3]) & 0xFF) << 0);
388
389 // do read the attachment
390 thumbnailAttachmentContent.resize(attachmentSize);
391 ifs.seekg(size - 12 - attachmentSize, ios::beg);
392 ifs.read((char*)thumbnailAttachmentContent.data(), attachmentSize);
393 ifs.close();
394
395 //
396 return thumbnailAttachmentContent.empty() == false;
397}
398
399bool StandardFileSystem::getThumbnailAttachment(const vector<uint8_t>& content, vector<uint8_t>& thumbnailAttachmentContent) {
400 if (content.size() < 12) return false;
401
402 array<uint8_t, 12> id;
403
404 //
405 for (auto i = 0; i < 12; i++) id[i] = content[content.size() - 12 + i];
406
407 // attachment
408 if (id[8] != 'A' || id[9] != 'T' || id[10] != 'M' || id[11] != 'T') {
409 return false;
410 }
411 // thumbnail
412 if (id[4] != 'T' || id[5] != 'M' || id[6] != 'B' || id[7] != 'N') {
413 return false;
414 }
415 // attachment size
416 int32_t attachmentSize =
417 ((static_cast<int32_t>(id[0]) & 0xFF) << 24) +
418 ((static_cast<int32_t>(id[1]) & 0xFF) << 16) +
419 ((static_cast<int32_t>(id[2]) & 0xFF) << 8) +
420 ((static_cast<int32_t>(id[3]) & 0xFF) << 0);
421
422 // do read the attachment
423 thumbnailAttachmentContent.resize(attachmentSize);
424 for (auto i = 0; i < attachmentSize; i++) thumbnailAttachmentContent[i] = content[content.size() - 12 - attachmentSize + i];
425
426 //
427 return thumbnailAttachmentContent.empty() == false;
428}
Standard file system implementation.
const string getPathName(const string &fileName) override
Get path name.
bool isPath(const string &pathName) override
Check if file is a path.
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.
virtual ~StandardFileSystem()
Public destructor.
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.
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.
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.