diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 7107ae67..ff227d22 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -13,14 +13,14 @@ If you wish to submit a PR, please be aware that: others, add an include directive to `include/toml++/toml.h` 3. Run `python/generate_single_header.py` -### Building and testing +### Building and running the tests Testing is done using [Catch2], included in the respository as a submodule under `extern/Catch2`. The first time you want to begin testing you'll need to ensure submodules have been fetched: ```bash git submodule update --init --recursive extern/Catch2 ``` -#### Windows +#### Testing on Windows with Visual Studio Install [Visual Studio 2019] and [Test Adapter for Catch2], then open `vs/toml++.sln` and build the projects in the `tests` solution folder. Visual Studio's Test Explorer should pick these up and @@ -29,7 +29,7 @@ allow you to run the tests directly. If test discovery fails you can usually fix it by clicking enabling `Auto Detect runsettings Files` (settings gear icon > `Configure Run Settings`). -#### Linux +#### Testing on Linux (and WSL) Install [meson] and [ninja] if necessary, then test with both gcc and clang: ```bash CXX=g++ meson build-gcc diff --git a/README.md b/README.md index 727eedd5..dc515407 100644 --- a/README.md +++ b/README.md @@ -12,6 +12,7 @@ - Works with or without exceptions - Doesn't require RTTI - First-class support for serializing to JSON + - Tested on Clang and GCC and MSVC (VS2019)
diff --git a/docs/Doxyfile-mcss b/docs/Doxyfile-mcss index a2b25d94..485e011b 100644 --- a/docs/Doxyfile-mcss +++ b/docs/Doxyfile-mcss @@ -17,3 +17,4 @@ HTML_EXTRA_FILES = tomlplusplus.js \ ##! Report an issue \ ##!

Documentation generated using m.css ##! M_HTML_HEADER = +##! M_FAVICON = favicon.ico diff --git a/docs/favicon.ico b/docs/favicon.ico new file mode 100644 index 00000000..518213d1 Binary files /dev/null and b/docs/favicon.ico differ diff --git a/examples/parse_file.cpp b/examples/parse_file.cpp index f288241d..67206f25 100644 --- a/examples/parse_file.cpp +++ b/examples/parse_file.cpp @@ -27,12 +27,7 @@ int main(int argc, char** argv) } catch (const toml::parse_error& err) { - std::cerr - << "Error parsing file '"sv << *err.source().path - << "':\n"sv << err.description() - << "\n ("sv << err.source().begin << ")"sv - << std::endl; - + std::cerr << "Error parsing file:\n"sv << err << std::endl; return 1; } return 0; diff --git a/examples/toml_to_json_transcoder.cpp b/examples/toml_to_json_transcoder.cpp index 6111b5aa..5b460f28 100644 --- a/examples/toml_to_json_transcoder.cpp +++ b/examples/toml_to_json_transcoder.cpp @@ -29,13 +29,9 @@ int main(int argc, char** argv) const auto table = toml::parse(file, std::move(path)); std::cout << toml::json_formatter{ table } << std::endl; } - catch (const toml::parse_error & err) + catch (const toml::parse_error& err) { - std::cerr - << "Error parsing file '"sv << *err.source().path - << "':\n"sv << err.description() - << "\n ("sv << err.source().begin << ")"sv - << std::endl; + std::cerr << "Error parsing file:\n"sv << err << std::endl; return 1; } @@ -51,10 +47,7 @@ int main(int argc, char** argv) } catch (const toml::parse_error& err) { - std::cerr - << "Error parsing stdin:\n"sv << err.description() - << "\n ("sv << err.source().begin << ")"sv - << std::endl; + std::cerr << "Error parsing stdin:\n"sv << err << std::endl; return 1; } diff --git a/include/toml++/toml.h b/include/toml++/toml.h index ae64f801..325a9d0a 100644 --- a/include/toml++/toml.h +++ b/include/toml++/toml.h @@ -78,7 +78,8 @@ /// - Proper UTF-8 handling (incl. BOM) /// - Works with or without exceptions /// - Doesn't require RTTI -/// - First-class support for serializing to JSON +/// - First-class support for serializing to JSON +/// - Tested on Clang and GCC and MSVC (VS2019) /// /////////////////////////////////////////////////////////////////////// /// @@ -88,14 +89,15 @@ /// There's some minor configuration you can do to customize some basic library functionality, but that's totally /// optional. See the [README](https://github.com/marzer/tomlplusplus/blob/master/README.md) for more info. /// -///
-///

On Linkers and the One-Definition-Rule

-///

Header-only libraries are great for minimal setup, but can cause ODR violations and complex linker errors +/// \remark +/// \parblock +/// Header-only libraries are great for minimal setup, but can cause ODR violations and complex linker errors /// in situations where multiple modules include them separately, each with different versions, configuration options, -/// exception handling modes, et cetera.

-///

`toml++` attempts to combat this problem by nesting everything inside an additional inline namespace that -/// changes according to the library's major version and the compiler's exception-handling mode.

-///
+/// exception handling modes, et cetera. +/// +/// `toml++` attempts to combat this problem by nesting everything inside an additional inline namespace that +/// changes according to the library's major version and the compiler's exception-handling mode. +/// \endparblock /// /////////////////////////////////////////////////////////////////////// /// @@ -173,23 +175,47 @@ /// } /// \ecpp /// +/// Instances of toml::parse_error can be printed directly to streams: +/// \cpp +/// try +/// { +/// auto tbl = toml::parse("enabled = trUe"sv); //fails; TOML booleans are case-sensitive +/// } +/// catch (const toml::parse_error & err) +/// { +/// std::cerr << "Parsing failed:\n"sv << err << std::endl; +/// return 1; +/// } +/// \ecpp +/// +/// \out +/// Parsing failed: +/// Encountered unexpected character while parsing boolean; expected 'true', saw 'trU' +/// (error occurred at line 1, column 13) +/// \eout +/// +/// If the default error formatting is not be suitable for your use-case you can access the error's +/// toml::source_region and description directly from the error object (as in the examples above). +/// /// \see /// - toml::parse_file() /// - toml::parse_result -/// - toml::parse_error +/// - toml::parse_error +/// - toml::source_region +/// - toml::source_position /// /////////////////////////////////// /// -/// \subsection mainpage-example-parsing-strings Parsing TOML directly from strings -/// +/// \subsection mainpage-example-parsing-strings Parsing TOML directly from strings and streams +/// Strings and std::istreams can be read directly using toml::parse(): /// \cpp /// #include +/// #include /// #include /// using namespace std::string_view_literals; /// /// int main() /// { -/// // parse error handling omitted for brevity. /// static constexpr auto source = R"( /// [library] /// name = "toml++" @@ -199,8 +225,20 @@ /// [dependencies] /// cpp = 17 /// )"sv; -/// auto tbl = toml::parse(source); -/// std::cout << tbl << std::endl; +/// +/// // parse directly from a string view: +/// { +/// auto tbl = toml::parse(source); +/// std::cout << tbl << std::endl; +/// } +/// +/// // parse from a string stream: +/// { +/// std::stringstream ss{ std::string{ source } }; +/// auto tbl = toml::parse(ss); +/// std::cout << tbl << std::endl; +/// } +/// /// return 0; /// } /// \ecpp @@ -213,7 +251,10 @@ /// authors = ["Mark Gillard "] /// name = "toml++" /// version = "0.1.0" +/// +/// ... exactly as above, but twice /// \eout +/// /// \see toml::parse() /// /////////////////////////////////// @@ -353,7 +394,9 @@ /////////////////////////////////////////////////////////////////////// /// /// \section mainpage-contributing Contributing -/// See the [Contributing](https://github.com/marzer/tomlplusplus/blob/master/README.md#contributing) section of the repository README. +/// Contributions are very welcome! Either by [reporting issues](https://github.com/marzer/tomlplusplus/issues) or submitting pull requests. +/// If you wish to submit a pull request, please see [CONTRIBUTING](https://github.com/marzer/tomlplusplus/blob/master/CONTRIBUTING.md) +/// for all the details you need to get going. /// /////////////////////////////////////////////////////////////////////// /// @@ -366,6 +409,6 @@ /// 'Flexible and Economical UTF - 8 Decoder', which is also subject to the terms of the MIT license - see /// [LICENSE-utf8-decoder](https://github.com/marzer/tomlplusplus/blob/master/LICENSE-utf8-decoder). /// -/// \remark Note that if you're using the single-header version of the library you don't need to distribute these files; +/// \remark If you're using the single-header version of the library you don't need to distribute these files; /// their contents is included in the preamble at the top of the file. /// diff --git a/include/toml++/toml_common.h b/include/toml++/toml_common.h index 777ef7f5..c8589076 100644 --- a/include/toml++/toml_common.h +++ b/include/toml++/toml_common.h @@ -594,6 +594,33 @@ TOML_START { return source_; } + + /// \brief Prints a parse_error to a stream. + /// + /// \detail \cpp + /// try + /// { + /// auto tbl = toml::parse("enabled = trUe"sv); + /// } + /// catch (const toml::parse_error & err) + /// { + /// std::cerr << "Parsing failed:\n"sv << err << std::endl; + /// } + /// \ecpp + /// + /// \out + /// Parsing failed: + /// Encountered unexpected character while parsing boolean; expected 'true', saw 'trU' + /// (error occurred at line 1, column 13) + /// \eout + /// + /// \param lhs The stream. + /// \param rhs The parse_error. + /// + /// \returns The input stream. + template + friend std::basic_ostream& operator << (std::basic_ostream& lhs, const parse_error& rhs) + TOML_MAY_THROW; }; #else @@ -633,6 +660,10 @@ TOML_START { return source_; } + + template + friend std::basic_ostream& operator << (std::basic_ostream& lhs, const parse_error& rhs) + TOML_MAY_THROW; }; #endif diff --git a/include/toml++/toml_parser.h b/include/toml++/toml_parser.h index f5b4d81b..ff1abc60 100644 --- a/include/toml++/toml_parser.h +++ b/include/toml++/toml_parser.h @@ -9,6 +9,26 @@ TOML_START #if TOML_DOXYGEN || !TOML_EXCEPTIONS /// \brief The result of a parsing operation. + /// + /// \detail A parse_result is effectively a discriminated union containing either a toml::table + /// or a toml::parse_error. Most member functions assume a particular one of these two states, + /// and calling them when in the wrong state will cause errors (e.g. attempting to access the + /// error object when parsing was successful). \cpp + /// parse_result result = toml::parse_file("config.toml"); + /// if (result) + /// do_stuff_with_a_table(result); //implicitly converts to table& + /// else + /// std::cerr << "Parse failed:\n"sv << result.error() << std::endl; + /// + /// \ecpp + /// + /// \out + /// example output: + /// + /// Parse failed: + /// Encountered unexpected character while parsing boolean; expected 'true', saw 'trU' + /// (error occurred at line 1, column 13 of 'config.toml') + /// \eout /// /// \warning This type only exists when exceptions are disabled. /// Otherwise parse_result is just an alias for toml::table: \cpp @@ -26,22 +46,6 @@ TOML_START /// /// #endif /// \ecpp - /// - /// \detail A parse_result is effectively a discriminated union containing either a toml::table - /// or a toml::parse_error. Most member functions assume a particular one of these two states, - /// and calling them when in the wrong state will cause errors (e.g. attempting to access the - /// error object when parsing was successful). \cpp - /// parse_result result = toml::parse_file("config.toml"); - /// if (result) - /// do_stuff_with_a_table(result); //implicitly converts to table& - /// else - /// std::cerr - /// << "Error parsing file '"sv << *result.error().source().path - /// << "':\n"sv << result.error().description() - /// << "\n ("sv << result.error().source().begin << ")"sv - /// << std::endl; - /// - /// \ecpp class parse_result final { private: @@ -971,6 +975,7 @@ TOML_IMPL_START TOML_ERROR_CHECK({}); TOML_ASSERT(cp && (*cp == U't' || *cp == U'f')); + start_recording(true); auto result = *cp == U't'; if (!consume_expected_sequence(result ? U"true"sv : U"false"sv)) { @@ -979,9 +984,10 @@ TOML_IMPL_START else abort_with_error( "Encountered unexpected character while parsing "sv, node_type::boolean, - "; expected 'true' or 'false', saw '"sv, *cp, '\'' + "; expected '"sv, result ? "true"sv : "false"sv, "', saw '"sv, recording_buffer, '\'' ); } + stop_recording(); TOML_ERROR_CHECK({}); if (cp && !is_value_terminator(*cp)) @@ -1008,6 +1014,7 @@ TOML_IMPL_START abort_with_error("Encountered EOF while parsing "sv, node_type::floating_point); }; + start_recording(true); const int sign = *cp == U'-' ? -1 : 1; if (*cp == U'+' || *cp == U'-') { @@ -1022,9 +1029,10 @@ TOML_IMPL_START eof_check(); abort_with_error( "Encountered unexpected character while parsing "sv, node_type::floating_point, - "; expected '"sv, inf ? "inf"sv : "nan"sv, "', saw '"sv, *cp, '\'' + "; expected '"sv, inf ? "inf"sv : "nan"sv, "', saw '"sv, recording_buffer, '\'' ); } + stop_recording(); TOML_ERROR_CHECK({}); if (cp && !is_value_terminator(*cp)) diff --git a/include/toml++/toml_print_to_stream.h b/include/toml++/toml_print_to_stream.h index 1327be9a..01ef94e6 100644 --- a/include/toml++/toml_print_to_stream.h +++ b/include/toml++/toml_print_to_stream.h @@ -302,7 +302,7 @@ TOML_START ); impl::print_to_stream("line "sv, lhs); impl::print_to_stream(rhs.line, lhs); - impl::print_to_stream(", column ", lhs); + impl::print_to_stream(", column "sv, lhs); impl::print_to_stream(rhs.column, lhs); return lhs; } @@ -324,5 +324,16 @@ TOML_START } return lhs; } + + template + std::basic_ostream& operator << (std::basic_ostream& lhs, const parse_error& rhs) + TOML_MAY_THROW + { + impl::print_to_stream(rhs.description(), lhs); + impl::print_to_stream("\n\t(error occurred at "sv, lhs); + lhs << rhs.source(); + impl::print_to_stream(")"sv, lhs); + return lhs; + } } TOML_END diff --git a/include/toml++/toml_version.h b/include/toml++/toml_version.h index 63f8d8de..202c65e7 100644 --- a/include/toml++/toml_version.h +++ b/include/toml++/toml_version.h @@ -2,7 +2,7 @@ #define TOML_LIB_MAJOR 0 #define TOML_LIB_MINOR 2 -#define TOML_LIB_PATCH 0 +#define TOML_LIB_PATCH 1 #define TOML_LANG_MAJOR 0 #define TOML_LANG_MINOR 5 diff --git a/python/generate_documentation.py b/python/generate_documentation.py index 34436084..896f1264 100644 --- a/python/generate_documentation.py +++ b/python/generate_documentation.py @@ -517,9 +517,16 @@ def __call__(self, file, doc): # adds some additional colouring to the syntax highlighting in code blocks. class SyntaxHighlightingFix(object): - __missing_keywords = [ + __keywords = [ 'constexpr', - 'if' + 'if', + 'else', + 'true', + 'false', + 'const', + 'noexcept', + 'template', + 'typename' ] def __call__(self, file, doc): @@ -570,11 +577,12 @@ def __call__(self, file, doc): changed = True # misidentifed keywords - nfs = code_block('span', class_='nf') - for nf in nfs: - if (nf.string is not None and nf.string in self.__missing_keywords): - nf['class'] = 'k' - changed = True + for keywordClass in ['nf', 'nb']: + kws = code_block('span', class_=keywordClass) + for kw in kws: + if (kw.string is not None and kw.string in self.__keywords): + kw['class'] = 'k' + changed = True return changed @@ -677,7 +685,7 @@ class ExtDocLinksFix(object): # toml-specific (r'toml::values?', 'classtoml_1_1value.html'), (r'(toml::)?date_times?', 'structtoml_1_1date__time.html'), - (r'(toml::)?times?', 'structtoml_1_1time.html'), + (r'(toml::)?time', 'structtoml_1_1time.html'), (r'(toml::)?dates?', 'structtoml_1_1date.html') ] __allowedNames = ['dd', 'p', 'dt', 'h3', 'td'] diff --git a/toml.hpp b/toml.hpp index 1c2151e6..12a44eee 100644 --- a/toml.hpp +++ b/toml.hpp @@ -1,6 +1,6 @@ //---------------------------------------------------------------------------------------------------------------------- // -// toml++ v0.2.0 +// toml++ v0.2.1 // https://github.com/marzer/tomlplusplus // SPDX-License-Identifier: MIT // @@ -295,7 +295,7 @@ #define TOML_LIB_MAJOR 0 #define TOML_LIB_MINOR 2 -#define TOML_LIB_PATCH 0 +#define TOML_LIB_PATCH 1 #define TOML_LANG_MAJOR 0 #define TOML_LANG_MINOR 5 @@ -532,6 +532,10 @@ TOML_START { return source_; } + + template + friend std::basic_ostream& operator << (std::basic_ostream& lhs, const parse_error& rhs) + TOML_MAY_THROW; }; #else @@ -571,6 +575,10 @@ TOML_START { return source_; } + + template + friend std::basic_ostream& operator << (std::basic_ostream& lhs, const parse_error& rhs) + TOML_MAY_THROW; }; #endif @@ -1308,7 +1316,7 @@ TOML_START ); impl::print_to_stream("line "sv, lhs); impl::print_to_stream(rhs.line, lhs); - impl::print_to_stream(", column ", lhs); + impl::print_to_stream(", column "sv, lhs); impl::print_to_stream(rhs.column, lhs); return lhs; } @@ -1330,6 +1338,17 @@ TOML_START } return lhs; } + + template + std::basic_ostream& operator << (std::basic_ostream& lhs, const parse_error& rhs) + TOML_MAY_THROW + { + impl::print_to_stream(rhs.description(), lhs); + impl::print_to_stream("\n\t(error occurred at "sv, lhs); + lhs << rhs.source(); + impl::print_to_stream(")"sv, lhs); + return lhs; + } } TOML_END @@ -5590,6 +5609,7 @@ TOML_IMPL_START TOML_ERROR_CHECK({}); TOML_ASSERT(cp && (*cp == U't' || *cp == U'f')); + start_recording(true); auto result = *cp == U't'; if (!consume_expected_sequence(result ? U"true"sv : U"false"sv)) { @@ -5598,9 +5618,10 @@ TOML_IMPL_START else abort_with_error( "Encountered unexpected character while parsing "sv, node_type::boolean, - "; expected 'true' or 'false', saw '"sv, *cp, '\'' + "; expected '"sv, result ? "true"sv : "false"sv, "', saw '"sv, recording_buffer, '\'' ); } + stop_recording(); TOML_ERROR_CHECK({}); if (cp && !is_value_terminator(*cp)) @@ -5627,6 +5648,7 @@ TOML_IMPL_START abort_with_error("Encountered EOF while parsing "sv, node_type::floating_point); }; + start_recording(true); const int sign = *cp == U'-' ? -1 : 1; if (*cp == U'+' || *cp == U'-') { @@ -5641,9 +5663,10 @@ TOML_IMPL_START eof_check(); abort_with_error( "Encountered unexpected character while parsing "sv, node_type::floating_point, - "; expected '"sv, inf ? "inf"sv : "nan"sv, "', saw '"sv, *cp, '\'' + "; expected '"sv, inf ? "inf"sv : "nan"sv, "', saw '"sv, recording_buffer, '\'' ); } + stop_recording(); TOML_ERROR_CHECK({}); if (cp && !is_value_terminator(*cp)) diff --git a/vs/toml++.vcxproj b/vs/toml++.vcxproj index dc6be779..6360905e 100644 --- a/vs/toml++.vcxproj +++ b/vs/toml++.vcxproj @@ -75,6 +75,7 @@ + diff --git a/vs/toml++.vcxproj.filters b/vs/toml++.vcxproj.filters index 087c99ee..812176d5 100644 --- a/vs/toml++.vcxproj.filters +++ b/vs/toml++.vcxproj.filters @@ -77,6 +77,7 @@ doc +