/* ****************************************************************************** Project: OWA EPANET Version: 2.2 Module: util/filemanager.c Description: Provides a simple interface for managing files Authors: see AUTHORS Copyright: see AUTHORS License: see LICENSE Last Updated: 04/01/2019 ****************************************************************************** */ // MSVC ONLY #ifdef _DEBUG #define _CRTDBG_MAP_ALLOC #include #include #else #include #endif #include #include "filemanager.h" typedef struct file_s { char *filename; // Assumes this is a null terminated string FILE *file; } file_handle_t; // local (private) functions int _fopen(FILE **f, const char *name, const char *mode); int _get_temp_filename(char **tempname); file_handle_t *create_file_manager() { file_handle_t *file_handle; file_handle = (file_handle_t *)calloc(1, sizeof(file_handle_t)); file_handle->filename = NULL; file_handle->file = NULL; return file_handle; } void delete_file_manager(file_handle_t *file_handle) { if (file_handle->file != NULL) close_file(file_handle); free(file_handle->filename); free(file_handle); } int get_filename(file_handle_t *file_handle, char **filename) // // BE AWARE: The memory allocated here must be freed by the caller // { return cstr_duplicate(filename, file_handle->filename); } int open_file(file_handle_t *file_handle, const char *filename, const char *file_mode) { int error = 0; if (filename == NULL) _get_temp_filename(&(file_handle->filename)); else cstr_duplicate(&(file_handle->filename), filename); if (file_mode == NULL) error = -1; else { error = _fopen(&(file_handle->file), file_handle->filename, file_mode); } return error; } int seek_file(file_handle_t *file_handle, F_OFF offset, int whence) { #ifdef _MSC_VER // Windows (32-bit and 64-bit) #define FSEEK64 _fseeki64 #else // Other platforms #define FSEEK64 fseeko #endif return FSEEK64(file_handle->file, offset, whence); } F_OFF tell_file(file_handle_t *file_handle) { #ifdef _MSC_VER // Windows (32-bit and 64-bit) #define FTELL64 _ftelli64 #else // Other platforms #define FTELL64 ftello #endif return FTELL64(file_handle->file); } // Read and write to a binary file size_t read_file(void *ptr, size_t size, size_t nmemb, file_handle_t *file_handle) { return fread(ptr, size, nmemb, file_handle->file); } size_t write_file(const void * ptr, size_t size, size_t count, file_handle_t *file_handle) { return fwrite(ptr, size, count, file_handle->file); } // print and get from a text file int printf_file(file_handle_t *file_handle, const char *format, ... ) { int error = 0; va_list args; va_start(args, format); error = vfprintf(file_handle->file, format, args); va_end(args); return error; } int gets_file(char *str, int num, file_handle_t *file_handle) { fgets(str, num, file_handle->file); return 0; } int close_file(file_handle_t *file_handle) { int error = 0; if (file_handle->file != NULL) { error = fclose(file_handle->file); file_handle->file = NULL; } return error; } int remove_file(file_handle_t *file_handle) { return remove(file_handle->filename); } bool is_valid(file_handle_t *file_handle) { if ((file_handle->filename == NULL && file_handle->file == NULL) || (cstr_isnullterm(file_handle->filename) && file_handle != NULL)) return true; else return false; } int _fopen(FILE **f, const char *name, const char *mode) // // Purpose: Substitute for fopen_s on platforms where it doesn't exist // Note: fopen_s is part of C++11 standard // { int ret = 0; #ifdef _MSC_VER ret = (int)fopen_s(f, name, mode); #else *f = fopen(name, mode); if (!*f) ret = -1; #endif return ret; } int _get_temp_filename(char **tempname) { int error = 0; #ifdef _MSC_VER char *name = NULL; // --- use Windows _tempnam function to get a pointer to an // unused file name that begins with "en" if ((name = _tempnam(name, "en")) == NULL) { error = -1; return error; } else cstr_duplicate(tempname, name); // --- free the pointer returned by _tempnam if (name) free(name); // --- for non-Windows systems: #else // --- use system function mkstemp() to create a temporary file name cstr_duplicate(tempname, "enXXXXXX"); error = mkstemp(*tempname); #endif return error; }