/* */ #include "File.h" #include #include #include #include #include #include #include #include "util.h" #include "A2STR.h" #include "array_fun.h" #include "Logger.h" #include "LogFactory.h" #include "fmt.h" namespace aria2 { #ifdef __MINGW32__ # define WIN32_LEAN_AND_MEAN # include #endif // __MINGW32__ File::File(const std::string& name) : name_(name) {} File::File(const File& c) : name_(c.name_) {} File::~File() {} File& File::operator=(const File& c) { if(this != &c) { name_ = c.name_; } return *this; } int File::fillStat(a2_struct_stat& fstat) { return a2stat(utf8ToWChar(name_).c_str(), &fstat); } bool File::exists() { a2_struct_stat fstat; return fillStat(fstat) == 0; } bool File::isFile() { a2_struct_stat fstat; if(fillStat(fstat) < 0) { return false; } return S_ISREG(fstat.st_mode) == 1; } bool File::isDir() { a2_struct_stat fstat; if(fillStat(fstat) < 0) { return false; } return S_ISDIR(fstat.st_mode) == 1; } bool File::remove() { if(isFile()) { return a2unlink(utf8ToWChar(name_).c_str()) == 0; } else if(isDir()) { return a2rmdir(utf8ToWChar(name_).c_str()) == 0; } else { return false; } } uint64_t File::size() { a2_struct_stat fstat; if(fillStat(fstat) < 0) { return 0; } return fstat.st_size; } bool File::mkdirs() { if(isDir()) { return false; } #ifdef __MINGW32__ std::string path = name_; for(std::string::iterator i = path.begin(), eoi = path.end(); i != eoi; ++i) { if(*i == '\\') { *i = '/'; } } std::string::iterator begin = path.begin(); std::string::iterator end = path.end(); #else // !__MINGW32__ std::string::iterator begin = name_.begin(); std::string::iterator end = name_.end(); #endif // !__MINGW32__ for(std::string::iterator i = begin; i != end;) { std::string::iterator j = std::find(i, end, '/'); if(std::distance(i, j) == 0) { ++i; continue; } i = j; if(i != end) { ++i; } #ifdef __MINGW32__ if(*(j-1) == ':') { // This is a drive letter, e.g. C:, so skip it. continue; } #endif // __MINGW32__ std::string dir = std::string(begin, j); A2_LOG_DEBUG(fmt("Making directory %s", utf8ToNative(dir).c_str())); if(File(dir).isDir()) { A2_LOG_DEBUG(fmt("%s exists and is a directory.", utf8ToNative(dir).c_str())); continue; } if(a2mkdir(utf8ToWChar(dir).c_str(), DIR_OPEN_MODE) == -1) { A2_LOG_DEBUG(fmt("Failed to create %s", utf8ToNative(dir).c_str())); return false; } } return true; } mode_t File::mode() { a2_struct_stat fstat; if(fillStat(fstat) < 0) { return 0; } return fstat.st_mode; } std::string File::getBasename() const { std::string::size_type lastSlashIndex = name_.find_last_of(A2STR::SLASH_C); if(lastSlashIndex == std::string::npos) { return name_; } else { return name_.substr(lastSlashIndex+1); } } std::string File::getDirname() const { std::string::size_type lastSlashIndex = name_.find_last_of(A2STR::SLASH_C); if(lastSlashIndex == std::string::npos) { if(name_.empty()) { return A2STR::NIL; } else { return A2STR::DOT_C; } } else if(lastSlashIndex == 0) { return A2STR::SLASH_C; } else { return name_.substr(0, lastSlashIndex); } } bool File::isDir(const std::string& filename) { return File(filename).isDir(); } bool File::renameTo(const std::string& dest) { #ifdef __MINGW32__ /* MinGW's rename() doesn't delete an existing destination */ if (_waccess(utf8ToWChar(dest).c_str(), 0) == 0) { if (a2unlink(utf8ToWChar(dest).c_str()) != 0) { return false; } } #endif // __MINGW32__ if(a2rename(utf8ToWChar(name_).c_str(), utf8ToWChar(dest).c_str()) == 0) { name_ = dest; return true; } else { return false; } } bool File::utime(const Time& actime, const Time& modtime) const { a2utimbuf ub; ub.actime = actime.getTime(); ub.modtime = modtime.getTime(); return a2utime(utf8ToWChar(name_).c_str(), &ub) == 0; } Time File::getModifiedTime() { a2_struct_stat fstat; if(fillStat(fstat) < 0) { return 0; } return Time(fstat.st_mtime); } std::string File::getCurrentDir() { #ifdef __MINGW32__ const size_t buflen = 2048; wchar_t buf[buflen]; if(_wgetcwd(buf, buflen)) { return wCharToUtf8(buf); } else { return A2STR::DOT_C; } #else // !__MINGW32__ const size_t buflen = 2048; char buf[buflen]; if(getcwd(buf, buflen)) { return std::string(buf); } else { return A2STR::DOT_C; } #endif // !__MINGW32__ } } // namespace aria2