3#if defined(_WIN32) && defined(_MSC_VER)
37using std::stringstream;
49StandardFileSystem::StandardFileSystem()
58 return pathName +
"/" + fileName;
63 auto _pathName = pathName;
64 if (StringTools::endsWith(pathName,
"/") ==
false) _pathName+=
"/";
67 struct dirent* dirent =
nullptr;
68 if ((dir = opendir(_pathName.c_str())) ==
nullptr) {
71 while ((dirent = readdir(dir)) !=
nullptr) {
72 string fileName = dirent->d_name;
73 if (fileName ==
".")
continue;
75 if (filter !=
nullptr && filter->
accept(pathName, fileName) ==
false)
continue;
77 Console::println(
"StandardFileSystem::list(): Filter::accept(): " + pathName +
"/" + fileName +
": " + exception.what());
80 files.push_back(fileName);
82 sort(files.begin(), files.end());
85 if (addDrives ==
true) {
86 for (
char drive =
'A'; drive <=
'Z'; drive++) {
91 if (
fileExists(fileName +
"/") ==
true) files.insert(files.begin() + (drive -
'C'), fileName);
93 Console::println(
"StandardFileSystem::list(): fileExists(): " + pathName +
"/" + fileName +
": " + exception.what());
104 if (stat(pathName.c_str(), &s) == 0) {
105 return (s.st_mode & S_IFDIR) == S_IFDIR;
107 throw FileSystemException(
"Unable to check if path(" + to_string(errno) +
"): " + pathName);
112 return StringTools::regexMatch(pathName,
"^[a-zA-Z]\\:[\\/\\\\]?$");
117 return stat(fileName.c_str(), &s) == 0;
122 if (stat((pathName +
"/" + fileName).c_str(), &s) != 0)
return false;
123 return (s.st_mode & S_IXUSR) == S_IXUSR;
127 #if !defined(_MSC_VER)
129 if (stat((pathName +
"/" + fileName).c_str(), &s) != 0) {
130 throw FileSystemException(
"Unable to set file to executable(" + to_string(errno) +
"): " + pathName +
"/" + fileName);
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);
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);
147 ifs.seekg( 0, ios::end );
148 size_t size = ifs.tellg();
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);
158 stringstream stringStream;
159 stringStream << ifs.rdbuf();
161 return (stringStream.str());
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);
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);
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);
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);
193 ofs.write((
char*)content.data(), content.size());
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);
205 while (getline(ifs, line)) {
206 content.push_back((line));
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);
219 for (
auto i = 0; i < content.size(); i++) {
220 ofs << (content.at(i)) <<
"\n";
228 string unixPathName = StringTools::replace(pathName,
"\\",
"/");
229 string unixFileName = StringTools::replace(fileName,
"\\",
"/");
231 auto pathString =
getFileName(unixPathName, unixFileName);
234 vector<string> pathComponents;
242 for (
auto i = 0; i < pathComponents.size(); i++) {
243 auto pathComponent = pathComponents[i];
244 if (pathComponent ==
".") {
245 pathComponents[i] =
"";
247 if (pathComponent ==
"..") {
248 pathComponents[i]=
"";
250 for (
int pathComponentReplaced = 0; pathComponentReplaced < 1 && j >= 0; ) {
251 if (pathComponents[j] !=
"") {
252 pathComponents[j] =
"";
253 pathComponentReplaced++;
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 ==
"") {
268 canonicalPath = canonicalPath + (slash ==
true?
"/":
"") + pathComponent;
274 auto canonicalPathString = canonicalPath;
275 if (canonicalPathString.length() == 0 ||
276 (StringTools::startsWith(canonicalPathString,
"/") ==
false &&
277 StringTools::regexMatch(canonicalPathString,
"^[a-zA-Z]\\:.*$") ==
false)) {
282 return canonicalPathString;
286 char cwdBuffer[PATH_MAX + 1];
287 char* cwdPtr = getcwd(cwdBuffer,
sizeof(cwdBuffer));
288 if (cwdPtr ==
nullptr) {
291 auto cwd = string(cwdPtr);
292 return StringTools::replace(cwd,
"\\",
"/");
296 if (chdir(pathName.c_str()) != 0) {
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);
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());
317 int32_t status = mkdir(pathName.c_str());
319 int32_t status = mkdir(pathName.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH);
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 ==
"..") {
335 auto completeFileName =
getFileName(pathName, file);
336 if (
isPath(completeFileName)) {
343 Console::println(
string(
"StandardFileSystem::removePath(): Removing ") + pathName);
344 int32_t status = rmdir(pathName.c_str());
346 throw FileSystemException(
"Unable to delete folder(" + to_string(errno) +
"): " + pathName);
351 Console::println(
string(
"StandardFileSystem::removeFile(): Removing ") +
getFileName(pathName, fileName));
352 int32_t status = unlink(
getFileName(pathName, fileName).c_str());
354 throw FileSystemException(
"Unable to delete file(" + to_string(errno) +
"): " + pathName +
"/" + fileName);
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);
365 ifs.seekg( 0, ios::end );
366 size_t size = ifs.tellg();
367 if (size < 12)
return false;
369 array<uint8_t, 12> id;
371 ifs.seekg(size - 12, ios::beg);
372 ifs.read((
char*)
id.data(), 12);
375 if (
id[8] !=
'A' ||
id[9] !=
'T' ||
id[10] !=
'M' ||
id[11] !=
'T') {
379 if (
id[4] !=
'T' ||
id[5] !=
'M' ||
id[6] !=
'B' ||
id[7] !=
'N') {
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);
390 thumbnailAttachmentContent.resize(attachmentSize);
391 ifs.seekg(size - 12 - attachmentSize, ios::beg);
392 ifs.read((
char*)thumbnailAttachmentContent.data(), attachmentSize);
396 return thumbnailAttachmentContent.empty() ==
false;
400 if (content.size() < 12)
return false;
402 array<uint8_t, 12> id;
405 for (
auto i = 0; i < 12; i++)
id[i] = content[content.size() - 12 + i];
408 if (
id[8] !=
'A' ||
id[9] !=
'T' ||
id[10] !=
'M' ||
id[11] !=
'T') {
412 if (
id[4] !=
'T' ||
id[5] !=
'M' ||
id[6] !=
'B' ||
id[7] !=
'N') {
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);
423 thumbnailAttachmentContent.resize(attachmentSize);
424 for (
auto i = 0; i < attachmentSize; i++) thumbnailAttachmentContent[i] = content[content.size() - 12 - attachmentSize + i];
427 return thumbnailAttachmentContent.empty() ==
false;
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.
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.