mirror of
https://github.com/ToruNiina/toml11.git
synced 2025-09-16 16:28:09 +08:00
fix(#139): Merge branch 'auto-conversion-macro'
This commit is contained in:
16
.github/workflows/main.yml
vendored
16
.github/workflows/main.yml
vendored
@@ -26,11 +26,9 @@ jobs:
|
|||||||
sudo apt-add-repository ppa:mhier/libboost-latest
|
sudo apt-add-repository ppa:mhier/libboost-latest
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install boost1.70
|
sudo apt-get install boost1.70
|
||||||
if [[ "${{ matrix.compiler }}" == "g++-6" || "${{ matrix.compiler }}" == "g++-5" ]] ; then
|
sudo apt-add-repository ppa:ubuntu-toolchain-r/test
|
||||||
sudo apt-add-repository ppa:ubuntu-toolchain-r/test
|
sudo apt-get update
|
||||||
sudo apt-get update
|
sudo apt-get install ${{ matrix.compiler }}
|
||||||
sudo apt-get install ${{ matrix.compiler }}
|
|
||||||
fi
|
|
||||||
- name: Configure
|
- name: Configure
|
||||||
run: |
|
run: |
|
||||||
mkdir build && cd build
|
mkdir build && cd build
|
||||||
@@ -70,11 +68,9 @@ jobs:
|
|||||||
sudo apt-add-repository ppa:mhier/libboost-latest
|
sudo apt-add-repository ppa:mhier/libboost-latest
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install boost1.70
|
sudo apt-get install boost1.70
|
||||||
if [[ "${{ matrix.compiler }}" != "6" && "${{ matrix.compiler }}" != "8" && "${{ matrix.compiler }}" != "9" ]] ; then
|
sudo apt-add-repository ppa:ubuntu-toolchain-r/test
|
||||||
sudo apt-add-repository ppa:ubuntu-toolchain-r/test
|
sudo apt-get update
|
||||||
sudo apt-get update
|
sudo apt-get install clang-${{ matrix.compiler }}
|
||||||
sudo apt-get install clang-${{ matrix.compiler }}
|
|
||||||
fi
|
|
||||||
- name: Configure
|
- name: Configure
|
||||||
run: |
|
run: |
|
||||||
mkdir build && cd build
|
mkdir build && cd build
|
||||||
|
@@ -38,8 +38,16 @@ else()
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(MSVC)
|
if(MSVC)
|
||||||
add_definitions("/Zc:__cplusplus") # define __cplusplus value correctly
|
add_definitions("/Zc:__cplusplus") # define __cplusplus value correctly
|
||||||
add_definitions("/utf-8") # enable to use u8"" literal
|
add_definitions("/utf-8") # enable to use u8"" literal
|
||||||
|
if(MSVC_VERSION LESS 1910)
|
||||||
|
message(STATUS "MSVC < 1910. DEFINE_CONVERSION_NON_INTRUSIVE is disabled")
|
||||||
|
add_definitions(-DTOML11_WITHOUT_DEFINE_NON_INTRUSIVE)
|
||||||
|
elseif(MSVC_VERSION LESS 1920)
|
||||||
|
add_definitions("/experimental:preprocessor") # MSVC 2017
|
||||||
|
else()
|
||||||
|
add_definitions("/Zc:preprocessor") # MSVC 2019
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Set some common directories
|
# Set some common directories
|
||||||
|
28
README.md
28
README.md
@@ -1371,6 +1371,34 @@ struct into<ext::foo>
|
|||||||
But note that, if this `basic_value` would be assigned into other `toml::value`
|
But note that, if this `basic_value` would be assigned into other `toml::value`
|
||||||
that discards `comments`, the comments would be dropped.
|
that discards `comments`, the comments would be dropped.
|
||||||
|
|
||||||
|
### Macro to automatically define conversion functions
|
||||||
|
|
||||||
|
There is a helper macro that automatically generates conversion functions `from` and `into` for a simple struct.
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
namespace foo
|
||||||
|
{
|
||||||
|
struct Foo
|
||||||
|
{
|
||||||
|
std::string s;
|
||||||
|
double d;
|
||||||
|
int i;
|
||||||
|
};
|
||||||
|
} // foo
|
||||||
|
|
||||||
|
TOML11_DEFINE_CONVERSION_NON_INTRUSIVE(foo::Foo, s, d, i)
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
const auto file = toml::parse("example.toml");
|
||||||
|
auto f = toml::find<foo::Foo>(file, "foo");
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
And then you can use `toml::find<foo::Foo>(file, "foo");`
|
||||||
|
|
||||||
|
**Note** that, because of a slight difference in implementation of preprocessor between gcc/clang and MSVC, [you need to define `/Zc:preprocessor`](https://github.com/ToruNiina/toml11/issues/139#issuecomment-803683682) to use it in MSVC (Thank you @glebm !).
|
||||||
|
|
||||||
## Formatting user-defined error messages
|
## Formatting user-defined error messages
|
||||||
|
|
||||||
When you encounter an error after you read the toml value, you may want to
|
When you encounter an error after you read the toml value, you may want to
|
||||||
|
@@ -545,3 +545,87 @@ BOOST_AUTO_TEST_CASE(test_recursive_conversion)
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ===========================================================================
|
||||||
|
|
||||||
|
#ifndef TOML11_WITHOUT_DEFINE_NON_INTRUSIVE
|
||||||
|
|
||||||
|
namespace extlib3
|
||||||
|
{
|
||||||
|
struct foo
|
||||||
|
{
|
||||||
|
int a;
|
||||||
|
std::string b;
|
||||||
|
};
|
||||||
|
struct bar
|
||||||
|
{
|
||||||
|
int a;
|
||||||
|
std::string b;
|
||||||
|
foo f;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // extlib3
|
||||||
|
|
||||||
|
TOML11_DEFINE_CONVERSION_NON_INTRUSIVE(extlib3::foo, a, b)
|
||||||
|
TOML11_DEFINE_CONVERSION_NON_INTRUSIVE(extlib3::bar, a, b, f)
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(test_conversion_via_macro)
|
||||||
|
{
|
||||||
|
{
|
||||||
|
const toml::value v{{"a", 42}, {"b", "baz"}};
|
||||||
|
|
||||||
|
const auto foo = toml::get<extlib3::foo>(v);
|
||||||
|
BOOST_TEST(foo.a == 42);
|
||||||
|
BOOST_TEST(foo.b == "baz");
|
||||||
|
|
||||||
|
const toml::value v2(foo);
|
||||||
|
BOOST_TEST(v2 == v);
|
||||||
|
}
|
||||||
|
{
|
||||||
|
const toml::basic_value<toml::discard_comments, std::map, std::deque> v{
|
||||||
|
{"a", 42}, {"b", "baz"}
|
||||||
|
};
|
||||||
|
|
||||||
|
const auto foo = toml::get<extlib3::foo>(v);
|
||||||
|
BOOST_TEST(foo.a == 42);
|
||||||
|
BOOST_TEST(foo.b == "baz");
|
||||||
|
|
||||||
|
const toml::basic_value<toml::discard_comments, std::map, std::deque> v2(foo);
|
||||||
|
BOOST_TEST(v2 == v);
|
||||||
|
}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
|
||||||
|
{
|
||||||
|
const toml::value v{
|
||||||
|
{"a", 42},
|
||||||
|
{"b", "bar.b"},
|
||||||
|
{"f", toml::table{{"a", 42}, {"b", "foo.b"}}}
|
||||||
|
};
|
||||||
|
|
||||||
|
const auto bar = toml::get<extlib3::bar>(v);
|
||||||
|
BOOST_TEST(bar.a == 42);
|
||||||
|
BOOST_TEST(bar.b == "bar.b");
|
||||||
|
BOOST_TEST(bar.f.a == 42);
|
||||||
|
BOOST_TEST(bar.f.b == "foo.b");
|
||||||
|
|
||||||
|
const toml::value v2(bar);
|
||||||
|
BOOST_TEST(v2 == v);
|
||||||
|
}
|
||||||
|
{
|
||||||
|
const toml::basic_value<toml::discard_comments, std::map, std::deque> v{
|
||||||
|
{"a", 42},
|
||||||
|
{"b", "bar.b"},
|
||||||
|
{"f", toml::table{{"a", 42}, {"b", "foo.b"}}}
|
||||||
|
};
|
||||||
|
|
||||||
|
const auto bar = toml::get<extlib3::bar>(v);
|
||||||
|
BOOST_TEST(bar.a == 42);
|
||||||
|
BOOST_TEST(bar.b == "bar.b");
|
||||||
|
BOOST_TEST(bar.f.a == 42);
|
||||||
|
BOOST_TEST(bar.f.b == "foo.b");
|
||||||
|
|
||||||
|
const toml::basic_value<toml::discard_comments, std::map, std::deque> v2(bar);
|
||||||
|
BOOST_TEST(v2 == v);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // TOML11_WITHOUT_DEFINE_NON_INTRUSIVE
|
||||||
|
1
toml.hpp
1
toml.hpp
@@ -41,5 +41,6 @@
|
|||||||
#include "toml/literal.hpp"
|
#include "toml/literal.hpp"
|
||||||
#include "toml/serializer.hpp"
|
#include "toml/serializer.hpp"
|
||||||
#include "toml/get.hpp"
|
#include "toml/get.hpp"
|
||||||
|
#include "toml/macros.hpp"
|
||||||
|
|
||||||
#endif// TOML_FOR_MODERN_CPP
|
#endif// TOML_FOR_MODERN_CPP
|
||||||
|
121
toml/macros.hpp
Normal file
121
toml/macros.hpp
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
#ifndef TOML11_MACROS_HPP
|
||||||
|
#define TOML11_MACROS_HPP
|
||||||
|
|
||||||
|
#define TOML11_STRINGIZE_AUX(x) #x
|
||||||
|
#define TOML11_STRINGIZE(x) TOML11_STRINGIZE_AUX(x)
|
||||||
|
|
||||||
|
#define TOML11_CONCATENATE_AUX(x, y) x##y
|
||||||
|
#define TOML11_CONCATENATE(x, y) TOML11_CONCATENATE_AUX(x, y)
|
||||||
|
|
||||||
|
// ============================================================================
|
||||||
|
// TOML11_DEFINE_CONVERSION_NON_INTRUSIVE
|
||||||
|
|
||||||
|
#ifndef TOML11_WITHOUT_DEFINE_NON_INTRUSIVE
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// TOML11_ARGS_SIZE
|
||||||
|
|
||||||
|
#define TOML11_INDEX_RSEQ() \
|
||||||
|
32, 31, 30, 29, 28, 27, 26, 25, 24, 23, 22, 21, 20, 19, 18, 17, \
|
||||||
|
16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0
|
||||||
|
#define TOML11_ARGS_SIZE_IMPL(\
|
||||||
|
ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7, ARG8, ARG9, ARG10, \
|
||||||
|
ARG11, ARG12, ARG13, ARG14, ARG15, ARG16, ARG17, ARG18, ARG19, ARG20, \
|
||||||
|
ARG21, ARG22, ARG23, ARG24, ARG25, ARG26, ARG27, ARG28, ARG29, ARG30, \
|
||||||
|
ARG31, ARG32, N, ...) N
|
||||||
|
#define TOML11_ARGS_SIZE_AUX(...) TOML11_ARGS_SIZE_IMPL(__VA_ARGS__)
|
||||||
|
#define TOML11_ARGS_SIZE(...) TOML11_ARGS_SIZE_AUX(__VA_ARGS__, TOML11_INDEX_RSEQ())
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// TOML11_FOR_EACH_VA_ARGS
|
||||||
|
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_1( FUNCTOR, ARG1 ) FUNCTOR(ARG1)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_2( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_1( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_3( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_2( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_4( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_3( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_5( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_4( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_6( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_5( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_7( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_6( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_8( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_7( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_9( FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_8( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_10(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_9( FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_11(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_10(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_12(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_11(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_13(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_12(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_14(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_13(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_15(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_14(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_16(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_15(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_17(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_16(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_18(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_17(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_19(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_18(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_20(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_19(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_21(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_20(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_22(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_21(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_23(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_22(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_24(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_23(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_25(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_24(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_26(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_25(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_27(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_26(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_28(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_27(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_29(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_28(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_30(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_29(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_31(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_30(FUNCTOR, __VA_ARGS__)
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS_AUX_32(FUNCTOR, ARG1, ...) FUNCTOR(ARG1) TOML11_FOR_EACH_VA_ARGS_AUX_31(FUNCTOR, __VA_ARGS__)
|
||||||
|
|
||||||
|
#define TOML11_FOR_EACH_VA_ARGS(FUNCTOR, ...)\
|
||||||
|
TOML11_CONCATENATE(TOML11_FOR_EACH_VA_ARGS_AUX_, TOML11_ARGS_SIZE(__VA_ARGS__))(FUNCTOR, __VA_ARGS__)
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// TOML11_DEFINE_CONVERSION_NON_INTRUSIVE
|
||||||
|
|
||||||
|
// use it in the following way.
|
||||||
|
// ```cpp
|
||||||
|
// namespace foo
|
||||||
|
// {
|
||||||
|
// struct Foo
|
||||||
|
// {
|
||||||
|
// std::string s;
|
||||||
|
// double d;
|
||||||
|
// int i;
|
||||||
|
// };
|
||||||
|
// } // foo
|
||||||
|
//
|
||||||
|
// TOML11_DEFINE_CONVERSION_NON_INTRUSIVE(foo::Foo, s, d, i)
|
||||||
|
// ```
|
||||||
|
// And then you can use `toml::find<foo::Foo>(file, "foo");`
|
||||||
|
//
|
||||||
|
#define TOML11_FIND_MEMBER_VARIABLE_FROM_VALUE(VAR_NAME)\
|
||||||
|
obj.VAR_NAME = toml::find<decltype(obj.VAR_NAME)>(v, TOML11_STRINGIZE(VAR_NAME));
|
||||||
|
|
||||||
|
#define TOML11_ASSIGN_MEMBER_VARIABLE_TO_VALUE(VAR_NAME)\
|
||||||
|
v[TOML11_STRINGIZE(VAR_NAME)] = obj.VAR_NAME;
|
||||||
|
|
||||||
|
#define TOML11_DEFINE_CONVERSION_NON_INTRUSIVE(NAME, ...)\
|
||||||
|
namespace toml { \
|
||||||
|
template<> \
|
||||||
|
struct from<NAME> \
|
||||||
|
{ \
|
||||||
|
template<typename C, template<typename ...> class T, \
|
||||||
|
template<typename ...> class A> \
|
||||||
|
static NAME from_toml(const basic_value<C, T, A>& v) \
|
||||||
|
{ \
|
||||||
|
NAME obj; \
|
||||||
|
TOML11_FOR_EACH_VA_ARGS(TOML11_FIND_MEMBER_VARIABLE_FROM_VALUE, __VA_ARGS__) \
|
||||||
|
return obj; \
|
||||||
|
} \
|
||||||
|
}; \
|
||||||
|
template<> \
|
||||||
|
struct into<NAME> \
|
||||||
|
{ \
|
||||||
|
static value into_toml(const NAME& obj) \
|
||||||
|
{ \
|
||||||
|
::toml::value v = ::toml::table{}; \
|
||||||
|
TOML11_FOR_EACH_VA_ARGS(TOML11_ASSIGN_MEMBER_VARIABLE_TO_VALUE, __VA_ARGS__) \
|
||||||
|
return v; \
|
||||||
|
} \
|
||||||
|
}; \
|
||||||
|
} /* toml */
|
||||||
|
|
||||||
|
#endif// TOML11_WITHOUT_DEFINE_NON_INTRUSIVE
|
||||||
|
|
||||||
|
#endif// TOML11_MACROS_HPP
|
Reference in New Issue
Block a user