diff --git a/dumux/io/rasterimagereader.hh b/dumux/io/rasterimagereader.hh index d3ff963332ae7406e14aba731b0bc42002b25c09..4ead3192f7a2ebbed232fee6396bdcced365605b 100644 --- a/dumux/io/rasterimagereader.hh +++ b/dumux/io/rasterimagereader.hh @@ -21,20 +21,21 @@ * \ingroup InputOutput * \brief A simple reader class for raster images. */ - #ifndef DUMUX_RASTER_IMAGE_READER_HH - #define DUMUX_RASTER_IMAGE_READER_HH - - #include <cassert> - #include <string> - #include <vector> - #include <fstream> - #include <sstream> - #include <algorithm> - #include <map> - #include <iterator> - #include <iostream> - - #include <dune/common/exceptions.hh> +#ifndef DUMUX_RASTER_IMAGE_READER_HH +#define DUMUX_RASTER_IMAGE_READER_HH + +#include <cassert> +#include <string> +#include <vector> +#include <fstream> +#include <sstream> +#include <algorithm> +#include <map> +#include <iterator> +#include <iostream> + +#include <dune/common/exceptions.hh> +#include <dumux/common/stringutilities.hh> namespace Dumux { @@ -108,11 +109,11 @@ public: }; /*! - * \brief A helper function to retrieve the format data from a given magic number. + * \brief A helper function to retrieve the format from tokens of the file's first line. * - * \param magicNumber The magic number contained in the header data of the file. + * \param firstLineTokes The tokens extracted from the first line of the file */ - static Format getFormat(const std::string& magicNumber) + static Format getFormat(const std::vector<std::string_view>& firstLineTokes) { static const auto format = []{ std::map<std::string, Format> format; @@ -121,10 +122,12 @@ public: format["P3"] = Format{"P3", "Portable PixMap", "ASCII"}; format["P4"] = Format{"P4", "Portable BitMap", "binary"}; format["P5"] = Format{"P5", "Portable GrayMap", "binary"}; - format["P5"] = Format{"P5", "Portable PixMap", "binary"}; + format["P6"] = Format{"P6", "Portable PixMap", "binary"}; return format; }(); + const std::string& magicNumber = std::string(firstLineTokes[0]); + if (!format.count(magicNumber)) DUNE_THROW(Dune::IOError, magicNumber << " is not a valid magic number for the Netpbm format"); @@ -213,36 +216,70 @@ public: { HeaderData headerData; std::string inputLine; + std::size_t lineNumber = 0; // First line : get format. std::getline(infile, inputLine); - headerData.format = getFormat(inputLine); + ++lineNumber; + + const auto firstLineTokens = tokenize(inputLine, " "); + headerData.format = getFormat(firstLineTokens); const auto magicNumber = headerData.format.magicNumber; - // Read dimensions and maximum value (for non-b/w images). - while (!infile.eof()) + // dimensions could be given right after magic number (in same line) + if (firstLineTokens.size() > 2) { - std::getline(infile, inputLine); + if (isBlackAndWhite_(magicNumber) && firstLineTokens.size() != 3) + DUNE_THROW(Dune::IOError, "Could not read first line for B/W image"); - auto isComment = [](const auto& s) - { return (s.find("#") != std::string::npos); }; + headerData.nCols = std::stoi(std::string(firstLineTokens[1])); + headerData.nRows = std::stoi(std::string(firstLineTokens[2])); - // Skip comments. - if (isComment(inputLine)) - continue; - - // The first line after the comments contains the dimensions. - headerData.nCols = std::stoi(inputLine.substr(0, inputLine.find(" "))); - headerData.nRows = std::stoi(inputLine.substr(inputLine.find(" ") + 1)); - - // Grayscale images additionaly contain a maxium value in the header. - if (magicNumber != "P1" && magicNumber != "P4") + if (isGrayScale_(magicNumber)) + { + if (firstLineTokens.size() == 4) + headerData.maxValue = std::stoi(std::string(firstLineTokens[3])); + if (firstLineTokens.size() > 4) + DUNE_THROW(Dune::IOError, "Could not read first line for grayscale image"); + } + } + else + { + // Read dimensions and maximum value (for non-b/w images). + while (!infile.eof()) { std::getline(infile, inputLine); - headerData.maxValue = std::stoi(inputLine); + ++lineNumber; + + // Skip comments. + if (isComment_(inputLine)) + continue; + + const auto tokens = tokenize(inputLine, " "); + + // The first line after the comments contains the dimensions. + if (tokens.size() != 2) + DUNE_THROW(Dune::IOError, "Expecting " << [](auto size){ return size < 2 ? "both" : "only"; }(tokens.size()) << " dimensions (2 numbers) in line " << lineNumber); + + headerData.nCols = std::stoi(std::string(tokens[0])); + headerData.nRows = std::stoi(std::string(tokens[1])); + + // Grayscale images additionaly contain a maxium value in the header. + if (isGrayScale_(magicNumber)) + { + std::getline(infile, inputLine); + ++lineNumber; + + const auto token = tokenize(inputLine, " "); + if (token.size() != 1) + DUNE_THROW(Dune::IOError, "Expecting" << [](auto size){ return size == 0 ? "" : " only"; }(token.size()) << " intensity (one number) in line " << lineNumber); + + headerData.maxValue = std::stoi(std::string(token[0])); + } + break; } - break; } + return headerData; } @@ -324,6 +361,17 @@ public: } private: + + static bool isBlackAndWhite_(const std::string& magicNumber) + { + return magicNumber == "P1" || magicNumber == "P4"; + } + + static bool isGrayScale_(const std::string& magicNumber) + { + return magicNumber == "P2" || magicNumber == "P5"; + } + /*! * \brief Reads the data block of a *.pbm (black and white) file in ASCII encoding. * Returns a vector that contains the pixel values. @@ -342,14 +390,16 @@ private: while (!infile.eof()) { std::getline(infile, inputLine); - inputLine.erase(std::remove(inputLine.begin(), inputLine.end(), '\n'), inputLine.end()); - inputLine.erase(std::remove(inputLine.begin(), inputLine.end(), ' '), inputLine.end()); - if (!inputLine.empty()) + if (!isComment_(inputLine)) { - for (const auto& value : inputLine) + inputLine.erase(std::remove_if(inputLine.begin(), inputLine.end(), [](unsigned char c){ return std::isspace(c); }), inputLine.end()); + if (!inputLine.empty()) { - assert(value == '0' || value == '1'); - data.push_back(value - '0'); // convert char to int + for (const auto& value : inputLine) + { + assert(value == '0' || value == '1'); + data.push_back(value - '0'); // convert char to int + } } } } @@ -370,6 +420,28 @@ private: { std::vector<bool> data(numPixel_(headerData)); + // Skip potentially remaining comments in header section + // before reading binary content. We detect a comment by + // reading a line with std::getline and checking the resulting string. + // We continue reading new lines until no more comments are found. Then we + // need to set infile's current position to one line before the actual binary + // content, otherwise the following steps will fail. + std::string inputLine; + while (!infile.eof()) + { + // store position before calling std::getline + const auto lastPos = infile.tellg(); + std::getline(infile, inputLine); + + // stop the loop if no more comment is found and go back one line + if (!isComment_(inputLine)) + { + infile.seekg(lastPos); + break; + } + } + + // read actual binary content std::size_t nBytes = 0; std::size_t bitIndex = 0; using Bit = std::uint8_t; @@ -489,6 +561,14 @@ private: { return headerData.nRows*headerData.nCols; } + + /*! + * \brief Returns true if a given line is a comment (starting with #) + */ + static bool isComment_(const std::string_view line) + { + return line[0] == '#'; + } }; } // namespace Dumux diff --git a/test/io/rasterimagereader/CMakeLists.txt b/test/io/rasterimagereader/CMakeLists.txt index 5dbae8b4668c1c53fbcbde51e0c7c5098e8801f7..3488e608f14d42942adc6dbc8921d052c344d68c 100644 --- a/test/io/rasterimagereader/CMakeLists.txt +++ b/test/io/rasterimagereader/CMakeLists.txt @@ -1,4 +1,7 @@ -dune_symlink_to_source_files(FILES blackwhite_j.pbm blackwhite_binary_j.pbm blackwhite_j.txt grayscale_j.pgm grayscale_binary_j.pgm grayscale_j.txt) +dune_symlink_to_source_files(FILES blackwhite_j.pbm blackwhite_binary_j.pbm blackwhite_j.txt + grayscale_j.pgm grayscale_binary_j.pgm grayscale_j.txt + blackwhite_dim_firstline.pbm blackwhite_dim_firstline_binary.pbm + blackwhite_fail.pbm grayscale_fail_binary.pgm) dumux_add_test(NAME test_io_rasterimagereader SOURCES test_rasterimagereader.cc diff --git a/test/io/rasterimagereader/blackwhite_dim_firstline.pbm b/test/io/rasterimagereader/blackwhite_dim_firstline.pbm new file mode 100644 index 0000000000000000000000000000000000000000..adfff86bbadc3c71ce58dc34a0f8244f693e126e --- /dev/null +++ b/test/io/rasterimagereader/blackwhite_dim_firstline.pbm @@ -0,0 +1,5 @@ +P1 2 2 +# delete this +1 0 +# delete this +0 1 diff --git a/test/io/rasterimagereader/blackwhite_dim_firstline_binary.pbm b/test/io/rasterimagereader/blackwhite_dim_firstline_binary.pbm new file mode 100644 index 0000000000000000000000000000000000000000..2dddbeb48381d3907f80f9230cbd198b30d8ce2d --- /dev/null +++ b/test/io/rasterimagereader/blackwhite_dim_firstline_binary.pbm @@ -0,0 +1,3 @@ +P4 2 2 +# dimensions given in first line +€@ diff --git a/test/io/rasterimagereader/blackwhite_fail.pbm b/test/io/rasterimagereader/blackwhite_fail.pbm new file mode 100644 index 0000000000000000000000000000000000000000..b3f1c44cdb15ca4e3ef3758308a9d7bf92cd64b2 --- /dev/null +++ b/test/io/rasterimagereader/blackwhite_fail.pbm @@ -0,0 +1,5 @@ +P1 +# we expect this to fail because line three contains more than just the two dimensions +2 2 1 +0 +0 1 diff --git a/test/io/rasterimagereader/grayscale_fail_binary.pgm b/test/io/rasterimagereader/grayscale_fail_binary.pgm new file mode 100644 index 0000000000000000000000000000000000000000..225e05dc7e2dfb44c6d4f509addeeb22d36cfa0c Binary files /dev/null and b/test/io/rasterimagereader/grayscale_fail_binary.pgm differ diff --git a/test/io/rasterimagereader/test_rasterimagereader.cc b/test/io/rasterimagereader/test_rasterimagereader.cc index f4cc8b7028f3532f232cba0b77e37ef8e2c6774d..491a4fab09ea6e18cfe7ea19315121b5f60fbba3 100644 --- a/test/io/rasterimagereader/test_rasterimagereader.cc +++ b/test/io/rasterimagereader/test_rasterimagereader.cc @@ -90,6 +90,35 @@ int main(int argc, char** argv) std::cout << std::endl; + // test file where the dimensions are given in the same line as the magic number and comments are present + const std::vector<bool> reference{1,0,0,1}; + + if (!isEqual(reference, NetPBMReader::readPBM("blackwhite_dim_firstline.pbm", false))) + { + std::cout << "Reading black/white with dimension in first line failed" << std::endl; + return 1; + } + if (!isEqual(reference, NetPBMReader::readPBM("blackwhite_dim_firstline.pbm", false))) + { + std::cout << "Reading black/white (binary) with dimension in first line failed" << std::endl; + return 1; + } + + // test error message for poorly formatted file + try + { + NetPBMReader::readPBM("blackwhite_fail.pbm", false); + } + catch(const Dune::IOError& e) + { + const auto tokens = tokenize(e.what(), "]:"); + if (tokens.back() != " Expecting only dimensions (2 numbers) in line 3") + { + std::cout << e.what() << std::endl; + return 1; + } + } + ////////////////////////////////////////////////// // Test the gray scale image reader ////////////////////////////////////////////////// @@ -126,5 +155,20 @@ int main(int argc, char** argv) return 1; } + // test error message for poorly formatted file + try + { + NetPBMReader::readPBM("grayscale_fail_binary.pgm", false); + } + catch(const Dune::IOError& e) + { + const auto tokens = tokenize(e.what(), "]:"); + if (tokens.back() != " Expecting only intensity (one number) in line 4") + { + std::cout << e.what() << std::endl; + return 1; + } + } + return 0; }