//BEGIN_LICENSE // // Copyright (c) 2000 The Open Group, BMC Software, Tivoli Systems, IBM // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. // //END_LICENSE //BEGIN_HISTORY // // Author: // // $Log: FileSystem.cpp,v $ // Revision 1.2 2001/02/11 05:42:33 mike // new // // Revision 1.1.1.1 2001/01/14 19:51:35 mike // Pegasus import // // //END_HISTORY #include #ifdef PEGASUS_OS_TYPE_WINDOWS # include # include #else # include # include #endif #include #include #include #include "Destroyer.h" #include "FileSystem.h" #include "Dir.h" // ATTN-B: porting! PEGASUS_NAMESPACE_BEGIN #ifdef PEGASUS_OS_TYPE_WINDOWS static const int ACCESS_EXISTS = 0; static const int ACCESS_WRITE = 2; static const int ACCESS_READ = 4; static const int ACCESS_READ_AND_WRITE = 6; #endif // Clone the path but discard trailing slash if any: static char* _clonePath(const String& path) { char* p = path.allocateCString(); if (!*p) return p; char* last = p + path.getLength() - 1; if (*last == '/') *last = '\0'; return p; } Boolean FileSystem::exists(const String& path) { Destroyer p(_clonePath(path)); #ifdef PEGASUS_OS_TYPE_WINDOWS return _access(p.getPointer(), ACCESS_EXISTS) == 0; #else return access(p.getPointer(), F_OK) == 0; #endif } Boolean FileSystem::existsIgnoreCase(const String& path, String& realPath) { realPath.clear(); Destroyer destroyer(_clonePath(path)); char* p = destroyer.getPointer(); char* dirPath; char* fileName; char* slash = strrchr(p, '/'); if (slash) { *slash = '\0'; fileName = slash + 1; dirPath = p; if (*fileName == '\0') return false; } else { fileName = p; dirPath = "."; } for (Dir dir(dirPath); dir.more(); dir.next()) { #ifdef PEGASUS_OS_TYPE_WINDOWS if (stricmp(fileName, dir.getName()) == 0) #else if (strcasecmp(fileName, dir.getName()) == 0) #endif { if (strcmp(dirPath, ".") == 0) realPath = dir.getName(); else { realPath = dirPath; realPath += '/'; realPath += dir.getName(); } return true; } } return false; } Boolean FileSystem::canRead(const String& path) { Destroyer p(_clonePath(path)); #ifdef PEGASUS_OS_TYPE_WINDOWS return _access(p.getPointer(), ACCESS_READ) == 0; #else return access(p.getPointer(), R_OK) == 0; #endif } Boolean FileSystem::canWrite(const String& path) { Destroyer p(_clonePath(path)); #ifdef PEGASUS_OS_TYPE_WINDOWS return _access(p.getPointer(), ACCESS_WRITE) == 0; #else return access(p.getPointer(), W_OK) == 0; #endif } #if 0 // ATTN: not implemented for NT. But not used by Pegasus. Boolean FileSystem::canExecute(const String& path) { Destroyer p(_clonePath(path)); return access(p.getPointer(), X_OK) == 0; } #endif Boolean FileSystem::isDirectory(const String& path) { Destroyer p(_clonePath(path)); struct stat st; #ifdef PEGASUS_OS_TYPE_WINDOWS if (stat(p.getPointer(), &st) != 0) return false; Boolean result = (st.st_mode & _S_IFDIR) != 0; return result; #else if (stat(p.getPointer(), &st) != 0) return false; Boolean result = S_ISDIR(st.st_mode); return result; #endif } Boolean FileSystem::changeDirectory(const String& path) { Destroyer p(_clonePath(path)); return chdir(p.getPointer()) == 0; } Boolean FileSystem::makeDirectory(const String& path) { Destroyer p(_clonePath(path)); #ifdef PEGASUS_OS_TYPE_WINDOWS return _mkdir(p.getPointer()) == 0; #else return mkdir(p.getPointer(), 0777) == 0; #endif } Boolean FileSystem::getFileSize(const String& path, Uint32& size) { struct stat st; Destroyer p(_clonePath(path)); if (stat(p.getPointer(), &st) != 0) return false; size = st.st_size; return true; } Boolean FileSystem::removeDirectory(const String& path) { Destroyer p(_clonePath(path)); return rmdir(p.getPointer()) == 0; } Boolean FileSystem::removeFile(const String& path) { Destroyer p(_clonePath(path)); return unlink(p.getPointer()) == 0; } void FileSystem::loadFileToMemory( Array& array, const String& fileName) { Uint32 fileSize; if (!getFileSize(fileName, fileSize)) throw CannotOpenFile(fileName); char* tmp = fileName.allocateCString(); FILE* fp = fopen(tmp, "rb"); delete [] tmp; if (fp == NULL) throw CannotOpenFile(fileName); array.reserve(fileSize); char buffer[4096]; size_t n; while ((n = fread(buffer, 1, sizeof(buffer), fp)) > 0) array.append(buffer, n); fclose(fp); } Boolean FileSystem::compare( const String& fileName1, const String& fileName2) { Uint32 fileSize1; if (!getFileSize(fileName1, fileSize1)) throw CannotOpenFile(fileName1); Uint32 fileSize2; if (!getFileSize(fileName2, fileSize2)) throw CannotOpenFile(fileName2); if (fileSize1 != fileSize2) return false; char* tmp1 = fileName1.allocateCString(); FILE* fp1 = fopen(tmp1, "rb"); delete [] tmp1; if (fp1 == NULL) throw CannotOpenFile(fileName1); char* tmp2 = fileName2.allocateCString(); FILE* fp2 = fopen(tmp2, "rb"); delete [] tmp2; if (fp2 == NULL) { fclose(fp1); throw CannotOpenFile(fileName2); } int c1; int c2; while ((c1 = fgetc(fp1)) != EOF && (c2 = fgetc(fp2)) != EOF) { if (c1 != c2) { fclose(fp1); fclose(fp2); return false; } } fclose(fp1); fclose(fp2); return true; } Boolean FileSystem::getDirectoryContents( const String& path, Array& paths) { paths.clear(); for (Dir dir(path); dir.more(); dir.next()) { String name = dir.getName(); if (name == "." || name == "..") continue; paths.append(name); } return true; } PEGASUS_NAMESPACE_END