From a43c32452e7ef6c27cbe1d30ddd136a55bd76b20 Mon Sep 17 00:00:00 2001 From: Giovanni La Mura <giovanni.lamura@inaf.it> Date: Sun, 26 May 2024 19:15:35 +0200 Subject: [PATCH] Implement VirtualAsciiFile local interface --- src/include/file_io.h | 57 +++++++++++++++++++++++++++++++-------- src/libnptm/file_io.cpp | 60 +++++++++++++++++++++++++++++++++-------- 2 files changed, 95 insertions(+), 22 deletions(-) diff --git a/src/include/file_io.h b/src/include/file_io.h index 312517f1..fc319e82 100644 --- a/src/include/file_io.h +++ b/src/include/file_io.h @@ -165,41 +165,76 @@ class HDFFile { */ class VirtualAsciiFile { protected: + //! \brief The number of lines. + int32_t _num_lines; //! \brief A vector of strings representing the file lines. - std::vector<std::string> *file_lines; - //! \brief The name of the file. - std::string _file_name; + std::vector<std::string> *_file_lines; public: - const std::string& file_name = _file_name; - + const int32_t &num_lines = _num_lines; /*! \brief VirtualAsciiFile instance constructor. * - * \param name: `const string&` Reference to a string for the file name. + * \param lines: `int32_t` Number of lines, if known in advance (optional, default is 0). */ - VirtualAsciiFile(const std::string& name); + VirtualAsciiFile(int32_t lines = 0); /*! \brief VirtualAsciiFile copy constructor. * * \param rhs: `const VirtualAsciiFile&` Reference to a VirtualAsciiFile instance. - * \param name: `const string&` Name of the copy (optional, default is the same as original). */ - VirtualAsciiFile(const VirtualAsciiFile& rhs, const std::string& name = ""); + VirtualAsciiFile(const VirtualAsciiFile& rhs); /*! \brief VirtualAsciiFile instance destroyer. */ ~VirtualAsciiFile(); + /*! \brief Append another VirtualAsciiFile at the end of the current instance. + * + * \param rhs: `const VirtualAsciiFile&` Reference to the VirtualAsciiFile to be appended. + */ + void append(const VirtualAsciiFile& rhs); + /*! \brief Append a line at the end of the file. * * \param line: `const string&` Reference to a string representing the line. */ - void append(const std::string& line); + void append_line(const std::string& line); + + /*! \brief Append the contents of the VirtualAsciiFile to a physical file on disk. + * + * \param file_name: `const string&` Name of the file to append contents to. + * \return result: `int` A result code (0 if successful). + */ + int append_to_disk(const std::string& file_name); + /*! \brief Insert another VirtualAsciiFile at a given position. + * + * This function inserts a target VirtualAsciiFile in the current one at the given + * position. Optionally, a range of lines to be inserted can be specified, otherwise + * the full content of the target file is inserted. This function DOES NOT increase + * the size of the inner storage and it can only be used if the inner storage has + * already been adjusted to contain the insertion target. + * + * \param position: `int32_t` The position at which the other file is inserted in this one. + * \param rhs: `const VirtualAsciiFile&` The refence to the VirtualAsciiFile to be inserted. + * \param start: `int32_t` The first line to be inserted (optional, default is 0). + * \param end: `int32_t` The last line to be inserted (optional, default is 0 to read all). + * \param line: `const string&` Reference to a string representing the line. + * \return result: `int` A result code (0 if successful). + */ + int insert(int32_t position, VirtualAsciiFile& rhs, int32_t start = 0, int32_t end = 0); + + /*! \brief Get the number of lines in the current instance. + * + * \return size: `int32_t` The number of lines in the VirtualAsciiFile instance. + */ + int32_t number_of_lines() { return _file_lines->size(); } + /*! \brief Write virtual file contents to a real file on disk. * + * \param file_name: `const string&` Name of the file to append contents to. * \return result: `int` A result code (0 if successful). */ - int write_to_disk(); + int write_to_disk(const std::string& file_name); }; #endif diff --git a/src/libnptm/file_io.cpp b/src/libnptm/file_io.cpp index b7a4b404..37cddd20 100644 --- a/src/libnptm/file_io.cpp +++ b/src/libnptm/file_io.cpp @@ -229,17 +229,15 @@ herr_t HDFFile::write( /* >>> End of HDFFile class implementation <<< */ /* >>> VirtualAsciiFile class implementation <<< */ -VirtualAsciiFile::VirtualAsciiFile(const std::string& name) { - _file_name = name; +VirtualAsciiFile::VirtualAsciiFile(int32_t lines) { _file_lines = new vector<string>(); + for (int32_t li = 0; li < lines; li++) { + _file_lines->push_back(""); + } } -VirtualAsciiFile::VirtualAsciiFile(const VirtualAsciiFile& rhs, const std::string& name) { - if (name.compare("") == 0) { - _file_name = rhs._file_name; - } else { - _file_name = name; - } +VirtualAsciiFile::VirtualAsciiFile(const VirtualAsciiFile& rhs) { + _num_lines = rhs._num_lines; _file_lines = new vector<string>(); for (vector<string>::iterator it = rhs._file_lines->begin(); it != rhs._file_lines->end(); ++it) { _file_lines->push_back(*it); @@ -247,14 +245,54 @@ VirtualAsciiFile::VirtualAsciiFile(const VirtualAsciiFile& rhs, const std::strin } VirtualAsciiFile::~VirtualAsciiFile() { + while (!_file_lines->size() > 0) { + _file_lines->pop_back(); + } if (_file_lines != NULL) delete _file_lines; } -void VirtualAsciiFile::append(const string& line) { - _file_lines.push_back(line); +void VirtualAsciiFile::append(const VirtualAsciiFile& rhs) { + for (vector<string>::iterator it = rhs._file_lines->begin(); it != rhs._file_lines->end(); ++it) { + _file_lines->push_back(*it); + } +} + +void VirtualAsciiFile::append_line(const string& line) { + _file_lines->push_back(line); +} + +int VirtualAsciiFile::append_to_disk(const std::string& file_name) { + int result = 0; + fstream output_file; + output_file.open(file_name, ios::app); + if (output_file.is_open()) { + for (vector<string>::iterator it = _file_lines->begin(); it != _file_lines->end(); ++it) { + output_file << *it; + } + } else { + result = 1; + } + return result; +} + +int VirtualAsciiFile::insert(int32_t position, VirtualAsciiFile& rhs, int32_t start, int32_t end) { + int result = 0; + if (start == 0 && end == 0) { + end = rhs.number_of_lines(); + } + int32_t final_index = position + end - start; + if (final_index <= number_of_lines()) { + for (int32_t li = start; li < end; li++) { + _file_lines->at(position++) = rhs._file_lines->at(li); + } + } else { + // ERROR: target file is too long; + result = 1; + } + return result; } -int VirtualAsciiFile::write_to_disk() { +int VirtualAsciiFile::write_to_disk(const std::string& file_name) { int result = 0; fstream output_file; output_file.open(file_name, ios::out); -- GitLab