/* Misc utils for files. Copyright (C) 2006 Artica ST. Written by Esteban Sanchez. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "pandora_file.h" #include #include #include #include #include #include #include using namespace std; /** * Checks if a file exists. * * @param filepath Path of the file to check. * * @retval True if the file exists. **/ bool Pandora_File::fileExists (const string filepath) { ifstream myfile (filepath.c_str ()); if (! myfile.is_open ()) { return false; } myfile.close(); return true; } /** * Reads a file and returns its content. * * @param filepath Path of the file to read. * * @return File content. * * @exception File_Not_Found throwed if the path is incorrect or the * file does not exists or could not be opened. **/ int Pandora_File::readFile (const string filepath, string &result) { string line; ifstream myfile (filepath.c_str ()); if (! myfile.is_open ()) { return FILE_NOT_FOUND; } if (myfile.is_open()) { while (! myfile.eof()) { getline (myfile,line); result += line + '\n'; } myfile.close(); } return 0; } /** * Reads a binary file and returns its content. * * @param filepath Path of the file to read. * * @exception File_Not_Found throwed if the path is incorrect or the * file does not exists or could not be opened. * * @note Memory allocated by this function must be freed at some point. **/ int Pandora_File::readBinFile (const string filepath, char **buffer) { int length; ifstream file; if (buffer == NULL) { throw File_Exception (); } file.open (filepath.c_str(), ios::binary ); if (! file.is_open ()) { throw File_Not_Found (); } /* Get file length */ file.seekg (0, ios::end); length = file.tellg (); if (length < 1) { throw File_Exception (); } file.seekg (0, ios::beg); *buffer = new char [length]; if (*buffer == NULL) { throw File_Exception (); } /* Read data */ file.read (*buffer, length); file.close (); return length; } /** * Delete a file or a directory recursively. * * @param filepath Path of the file or directory to delete. * * @return non zero if the file could not be deleted. */ int Pandora_File::removeFile (const string filepath) { if (remove (filepath.c_str ()) == -1) { return DELETE_ERROR; } } /** * Delete a directory with all its content. * * @param filepath Path of the directory to delete. * * @return Delete_Error if the file could not be deleted. */ int Pandora_File::removeDir (const string filepath) { DIR *dir; struct dirent *dir_content; struct stat file; string tmp; /*Open the directory*/ dir = opendir (filepath.c_str ()); /*If not open*/ if (dir == NULL) { /**If not exists*/ if(errno == ENOENT) { return 0; } else { /*If its a file, delete it*/ if(errno == ENOTDIR) { if (remove (filepath.c_str ()) == -1) { return DELETE_ERROR; } return 0; } else { return DELETE_ERROR; } } } /*If open*/ /*Read the directory looking for files and folders*/ dir_content = readdir(dir); while (dir_content != NULL) { string tmp = filepath+"\\"+dir_content->d_name; stat(tmp.c_str(),&file); switch (file.st_mode & S_IFMT) { case S_IFDIR: /*If tmp is a folder, recursive delete*/ if ( (strcmp(dir_content->d_name,".") != 0) && (strcmp(dir_content->d_name,"..") != 0)){ if( removeDir(tmp) == DELETE_ERROR ) { /*Close dir oppened*/ closedir(dir); return DELETE_ERROR; } } break; default: /*If tmp is a file, it will be deleted*/ if (remove (tmp.c_str ()) == -1) { /*Close dir oppened*/ closedir(dir); return DELETE_ERROR; } break; } /*Next item*/ dir_content = readdir(dir); } /*When the folder is empty, delete the folder*/ if (rmdir (filepath.c_str ()) == -1) { /*Close dir oppened*/ closedir(dir); return DELETE_ERROR; } /*Close dir oppened*/ closedir(dir); return 0; } /** * Write data into a text file. * * @param filepath Path of the file to write in. * @param data Data to be written. * * @exception File_Not_Found throwed if the path is incorrect or the * file does not exists or could not be opened. */ void Pandora_File::writeFile (const string filepath, const string data) { ofstream file (filepath.c_str ()); if (! file.is_open ()) { throw File_Not_Found (); } file.write (data.c_str (), data.length ()); file.close (); } /** * Write binary data into a file. * * @param filepath Path of the file to write in. * @param data Data to be written. * @param size Data size in bytes. * * @exception File_Not_Found throwed if the path is incorrect or the * file does not exists or could not be opened. */ void Pandora_File::writeBinFile (const string filepath, const char *buffer, int size) { ofstream file; if (buffer == NULL) { throw File_Exception (); } file.open(filepath.c_str (), ios_base::binary | ios_base::trunc); if (! file.is_open ()) { throw File_Not_Found (); } file.write (buffer, size); file.close (); } /** * Returns the filename of a complete filepath. * * @param filepath */ string Pandora_File::fileName (const string filepath) { string filename; int pos; pos = filepath.find_last_of ("\\"); if (pos != string::npos) { filename = filepath.substr (pos + 1); } else { filename = filepath; } return filename; } /** * Returns the 32 digit hexadecimal representation of the md5 hash * of the given data. * * @param data Data. * @param data Data size. * @param buffer Buffer where the 32 digit hex md5 will be stored. * Must be big enough to hold it! */ void Pandora_File::md5 (const char *data, int size, char *buffer) { int i; md5_state_t pms; md5_byte_t digest[16]; if (buffer == NULL) { throw File_Exception (); } /* md5 hash */ md5_init (&pms); md5_append (&pms, (unsigned char *)data, size); md5_finish (&pms, digest); /* 32 digit hexadecimal representation */ for (i = 0; i < 16; i++) { snprintf (buffer + (i << 1), 3, "%.2x", (unsigned int)(digest[i])); } }