#ifndef COMPONENTS_FILES_ESCAPE_HPP #define COMPONENTS_FILES_ESCAPE_HPP #include #include #include #include #include /** * \namespace Files */ namespace Files { /** * \struct escape_hash_filter */ struct escape_hash_filter : public boost::iostreams::input_filter { static const int sEscape; static const int sHashIdentifier; static const int sEscapeIdentifier; escape_hash_filter(); virtual ~escape_hash_filter(); template int get(Source & src); private: std::queue mNext; int mPrevious; bool mSeenNonWhitespace; bool mFinishLine; }; template int escape_hash_filter::get(Source & src) { if (mNext.empty()) { int character = boost::iostreams::get(src); bool record = true; if (character == boost::iostreams::WOULD_BLOCK) { mNext.push(character); record = false; } else if (character == EOF) { mSeenNonWhitespace = false; mFinishLine = false; mNext.push(character); } else if (character == '\n') { mSeenNonWhitespace = false; mFinishLine = false; mNext.push(character); } else if (mFinishLine) { mNext.push(character); } else if (character == '#') { if (mSeenNonWhitespace) { mNext.push(sEscape); mNext.push(sHashIdentifier); } else { //it's fine being interpreted by Boost as a comment, and so is anything afterwards mNext.push(character); mFinishLine = true; } } else if (character == sEscape) { mNext.push(sEscape); mNext.push(sEscapeIdentifier); record = false; } else if (mPrevious == sEscape) { mNext.push(sEscape); mNext.push(sEscapeIdentifier); } else { mNext.push(character); } if (!mSeenNonWhitespace && !isspace(character)) mSeenNonWhitespace = true; if (record) mPrevious = character; } int retval = mNext.front(); mNext.pop(); return retval; } struct unescape_hash_filter : public boost::iostreams::input_filter { unescape_hash_filter(); virtual ~unescape_hash_filter(); template int get(Source & src); private: bool expectingIdentifier; }; template int unescape_hash_filter::get(Source & src) { int character; if (!expectingIdentifier) character = boost::iostreams::get(src); else { character = escape_hash_filter::sEscape; expectingIdentifier = false; } if (character == escape_hash_filter::sEscape) { int nextChar = boost::iostreams::get(src); int intended; if (nextChar == escape_hash_filter::sEscapeIdentifier) intended = escape_hash_filter::sEscape; else if (nextChar == escape_hash_filter::sHashIdentifier) intended = '#'; else if (nextChar == boost::iostreams::WOULD_BLOCK) { expectingIdentifier = true; intended = nextChar; } else intended = '?'; return intended; } else return character; } /** * \class EscapeHashString */ class EscapeHashString { private: std::string mData; public: static std::string processString(const std::string & str); EscapeHashString(); EscapeHashString(const std::string & str); EscapeHashString(const std::string & str, size_t pos, size_t len = std::string::npos); EscapeHashString(const char * s); EscapeHashString(const char * s, size_t n); EscapeHashString(size_t n, char c); template EscapeHashString(InputIterator first, InputIterator last); std::string toStdString() const; friend std::ostream & operator<< (std::ostream & os, const EscapeHashString & eHS); }; std::istream & operator>> (std::istream & is, EscapeHashString & eHS); struct EscapeStringVector { std::vector mVector; EscapeStringVector(); virtual ~EscapeStringVector(); std::vector toStdStringVector() const; }; //boost program options validation void validate(boost::any &v, const std::vector &tokens, Files::EscapeHashString * eHS, int a); void validate(boost::any &v, const std::vector &tokens, EscapeStringVector *, int); struct EscapePath { boost::filesystem::path mPath; static PathContainer toPathContainer(const std::vector & escapePathContainer); }; typedef std::vector EscapePathContainer; std::istream & operator>> (std::istream & istream, EscapePath & escapePath); } /* namespace Files */ #endif /* COMPONENTS_FILES_ESCAPE_HPP */