From 63ccfabe75be0a75d7b0ac1644816f2e8d8e3d69 Mon Sep 17 00:00:00 2001 From: ManifoldFR <wilson.jallet@polytechnique.org> Date: Thu, 6 Apr 2023 19:01:59 +0200 Subject: [PATCH] unittest: add test for std::optional if available + unittest: make bind_optional and test_optional.py into template files + cmake: instantiate boost and std optional tests using configure_file --- unittest/CMakeLists.txt | 22 ++++-- ...bind_optional.cpp => bind_optional.cpp.in} | 24 ++++--- .../{test_optional.py => test_optional.py.in} | 14 ++-- unittest/python/test_optional_boost.py | 67 +++++++++++++++++++ unittest/python/test_optional_std.py | 67 +++++++++++++++++++ 5 files changed, 174 insertions(+), 20 deletions(-) rename unittest/{bind_optional.cpp => bind_optional.cpp.in} (73%) rename unittest/python/{test_optional.py => test_optional.py.in} (88%) create mode 100644 unittest/python/test_optional_boost.py create mode 100644 unittest/python/test_optional_std.py diff --git a/unittest/CMakeLists.txt b/unittest/CMakeLists.txt index 9d62420f..7b171d71 100644 --- a/unittest/CMakeLists.txt +++ b/unittest/CMakeLists.txt @@ -39,7 +39,24 @@ if(NOT NUMPY_WITH_BROKEN_UFUNC_SUPPORT) endif() add_lib_unit_test(std_vector) add_lib_unit_test(user_struct) -add_lib_unit_test(bind_optional) + +function(config_bind_optional tagname opttype) + set(MODNAME bind_optional_${tagname}) + set(OPTIONAL ${opttype}) + configure_file(bind_optional.cpp.in ${MODNAME}.cpp) + + set(py_file test_optional_${tagname}.py) + configure_file(python/test_optional.py.in ${CMAKE_CURRENT_SOURCE_DIR}/python/${py_file}) + add_lib_unit_test(${MODNAME}) + message(STATUS "Adding unit test py-optional-${tagname} with file ${py_file} and module ${MODNAME}") + add_python_unit_test("py-optional-${tagname}" "unittest/python/${py_file}" + "unittest") +endfunction() + +config_bind_optional(boost "boost::optional") +if(CMAKE_CXX_STANDARD GREATER 14 AND CMAKE_CXX_STANDARD LESS 98) + config_bind_optional(std "std::optional") +endif() add_python_unit_test("py-matrix" "unittest/python/test_matrix.py" "unittest") @@ -98,6 +115,3 @@ set_tests_properties("py-std-vector" PROPERTIES DEPENDS ${PYWRAP}) add_python_unit_test("py-user-struct" "unittest/python/test_user_struct.py" "python;unittest") set_tests_properties("py-std-vector" PROPERTIES DEPENDS ${PYWRAP}) - -add_python_unit_test("py-optional" "unittest/python/test_optional.py" - "unittest") diff --git a/unittest/bind_optional.cpp b/unittest/bind_optional.cpp.in similarity index 73% rename from unittest/bind_optional.cpp rename to unittest/bind_optional.cpp.in index 6d85795c..f6827872 100644 --- a/unittest/bind_optional.cpp +++ b/unittest/bind_optional.cpp.in @@ -1,16 +1,20 @@ #include "eigenpy/eigenpy.hpp" #include "eigenpy/optional.hpp" +#ifdef EIGENPY_WITH_CXX17_SUPPORT +#include <optional> +#endif -#define OPTIONAL boost::optional -#define OPT_NONE boost::none +#cmakedefine OPTIONAL @OPTIONAL@ -using opt_dbl = OPTIONAL<double>; +typedef eigenpy::detail::nullopt_helper<OPTIONAL> none_helper; +static auto OPT_NONE = none_helper::value(); +typedef OPTIONAL<double> opt_dbl; struct mystruct { OPTIONAL<int> a; opt_dbl b; OPTIONAL<std::string> msg{"i am struct"}; - mystruct() : a(OPT_NONE), b(boost::none) {} + mystruct() : a(OPT_NONE), b(OPT_NONE) {} mystruct(int a, const opt_dbl &b = OPT_NONE) : a(a), b(b) {} }; @@ -36,13 +40,13 @@ OPTIONAL<Eigen::MatrixXd> random_mat_if_true(bool flag) { return OPT_NONE; } -BOOST_PYTHON_MODULE(bind_optional) { +BOOST_PYTHON_MODULE(@MODNAME@) { using namespace eigenpy; - OptionalConverter<int>::registration(); - OptionalConverter<double>::registration(); - OptionalConverter<std::string>::registration(); - OptionalConverter<mystruct>::registration(); - OptionalConverter<Eigen::MatrixXd>::registration(); + OptionalConverter<int, OPTIONAL>::registration(); + OptionalConverter<double, OPTIONAL>::registration(); + OptionalConverter<std::string, OPTIONAL>::registration(); + OptionalConverter<mystruct, OPTIONAL>::registration(); + OptionalConverter<Eigen::MatrixXd, OPTIONAL>::registration(); enableEigenPy(); bp::class_<mystruct>("mystruct", bp::no_init) diff --git a/unittest/python/test_optional.py b/unittest/python/test_optional.py.in similarity index 88% rename from unittest/python/test_optional.py rename to unittest/python/test_optional.py.in index d6a29738..bd5e085e 100644 --- a/unittest/python/test_optional.py +++ b/unittest/python/test_optional.py.in @@ -1,4 +1,6 @@ -import bind_optional +import importlib + +bind_optional = importlib.import_module("@MODNAME@") def test_none_if_zero(): @@ -58,8 +60,8 @@ def test_random_mat(): assert M.shape == (4, 4) -if __name__ == "__main__": - import pytest - import sys - - sys.exit(pytest.main(sys.argv)) +test_none_if_zero() +test_struct_ctors() +test_struct_setters() +test_factory() +test_random_mat() diff --git a/unittest/python/test_optional_boost.py b/unittest/python/test_optional_boost.py new file mode 100644 index 00000000..fc818739 --- /dev/null +++ b/unittest/python/test_optional_boost.py @@ -0,0 +1,67 @@ +import importlib + +bind_optional = importlib.import_module("bind_optional_boost") + + +def test_none_if_zero(): + x = bind_optional.none_if_zero(0) + y = bind_optional.none_if_zero(-1) + assert x is None + assert y == -1 + + +def test_struct_ctors(): + # test struct ctors + + struct = bind_optional.mystruct() + assert struct.a is None + assert struct.b is None + assert struct.msg == "i am struct" + + ## no 2nd arg automatic overload using bp::optional + struct = bind_optional.mystruct(2) + assert struct.a == 2 + assert struct.b is None + + struct = bind_optional.mystruct(13, -1.0) + assert struct.a == 13 + assert struct.b == -1.0 + + +def test_struct_setters(): + struct = bind_optional.mystruct() + struct.a = 1 + assert struct.a == 1 + + struct.b = -3.14 + assert struct.b == -3.14 + + # set to None + struct.a = None + struct.b = None + struct.msg = None + assert struct.a is None + assert struct.b is None + assert struct.msg is None + + +def test_factory(): + struct = bind_optional.create_if_true(False, None) + assert struct is None + struct = bind_optional.create_if_true(True, None) + assert struct.a == 0 + assert struct.b is None + + +def test_random_mat(): + M = bind_optional.random_mat_if_true(False) + assert M is None + M = bind_optional.random_mat_if_true(True) + assert M.shape == (4, 4) + + +test_none_if_zero() +test_struct_ctors() +test_struct_setters() +test_factory() +test_random_mat() diff --git a/unittest/python/test_optional_std.py b/unittest/python/test_optional_std.py new file mode 100644 index 00000000..69949a44 --- /dev/null +++ b/unittest/python/test_optional_std.py @@ -0,0 +1,67 @@ +import importlib + +bind_optional = importlib.import_module("bind_optional_std") + + +def test_none_if_zero(): + x = bind_optional.none_if_zero(0) + y = bind_optional.none_if_zero(-1) + assert x is None + assert y == -1 + + +def test_struct_ctors(): + # test struct ctors + + struct = bind_optional.mystruct() + assert struct.a is None + assert struct.b is None + assert struct.msg == "i am struct" + + ## no 2nd arg automatic overload using bp::optional + struct = bind_optional.mystruct(2) + assert struct.a == 2 + assert struct.b is None + + struct = bind_optional.mystruct(13, -1.0) + assert struct.a == 13 + assert struct.b == -1.0 + + +def test_struct_setters(): + struct = bind_optional.mystruct() + struct.a = 1 + assert struct.a == 1 + + struct.b = -3.14 + assert struct.b == -3.14 + + # set to None + struct.a = None + struct.b = None + struct.msg = None + assert struct.a is None + assert struct.b is None + assert struct.msg is None + + +def test_factory(): + struct = bind_optional.create_if_true(False, None) + assert struct is None + struct = bind_optional.create_if_true(True, None) + assert struct.a == 0 + assert struct.b is None + + +def test_random_mat(): + M = bind_optional.random_mat_if_true(False) + assert M is None + M = bind_optional.random_mat_if_true(True) + assert M.shape == (4, 4) + + +test_none_if_zero() +test_struct_ctors() +test_struct_setters() +test_factory() +test_random_mat() -- GitLab