489 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
		
		
			
		
	
	
			489 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
|  | #include <pybind11/embed.h>
 | ||
|  | 
 | ||
|  | // Silence MSVC C++17 deprecation warning from Catch regarding std::uncaught_exceptions (up to
 | ||
|  | // catch 2.0.1; this should be fixed in the next catch release after 2.0.1).
 | ||
|  | PYBIND11_WARNING_DISABLE_MSVC(4996) | ||
|  | 
 | ||
|  | #include <catch.hpp>
 | ||
|  | #include <cstdlib>
 | ||
|  | #include <fstream>
 | ||
|  | #include <functional>
 | ||
|  | #include <thread>
 | ||
|  | #include <utility>
 | ||
|  | 
 | ||
|  | namespace py = pybind11; | ||
|  | using namespace py::literals; | ||
|  | 
 | ||
|  | size_t get_sys_path_size() { | ||
|  |     auto sys_path = py::module::import("sys").attr("path"); | ||
|  |     return py::len(sys_path); | ||
|  | } | ||
|  | 
 | ||
|  | class Widget { | ||
|  | public: | ||
|  |     explicit Widget(std::string message) : message(std::move(message)) {} | ||
|  |     virtual ~Widget() = default; | ||
|  | 
 | ||
|  |     std::string the_message() const { return message; } | ||
|  |     virtual int the_answer() const = 0; | ||
|  |     virtual std::string argv0() const = 0; | ||
|  | 
 | ||
|  | private: | ||
|  |     std::string message; | ||
|  | }; | ||
|  | 
 | ||
|  | class PyWidget final : public Widget { | ||
|  |     using Widget::Widget; | ||
|  | 
 | ||
|  |     int the_answer() const override { PYBIND11_OVERRIDE_PURE(int, Widget, the_answer); } | ||
|  |     std::string argv0() const override { PYBIND11_OVERRIDE_PURE(std::string, Widget, argv0); } | ||
|  | }; | ||
|  | 
 | ||
|  | class test_override_cache_helper { | ||
|  | 
 | ||
|  | public: | ||
|  |     virtual int func() { return 0; } | ||
|  | 
 | ||
|  |     test_override_cache_helper() = default; | ||
|  |     virtual ~test_override_cache_helper() = default; | ||
|  |     // Non-copyable
 | ||
|  |     test_override_cache_helper &operator=(test_override_cache_helper const &Right) = delete; | ||
|  |     test_override_cache_helper(test_override_cache_helper const &Copy) = delete; | ||
|  | }; | ||
|  | 
 | ||
|  | class test_override_cache_helper_trampoline : public test_override_cache_helper { | ||
|  |     int func() override { PYBIND11_OVERRIDE(int, test_override_cache_helper, func); } | ||
|  | }; | ||
|  | 
 | ||
|  | PYBIND11_EMBEDDED_MODULE(widget_module, m) { | ||
|  |     py::class_<Widget, PyWidget>(m, "Widget") | ||
|  |         .def(py::init<std::string>()) | ||
|  |         .def_property_readonly("the_message", &Widget::the_message); | ||
|  | 
 | ||
|  |     m.def("add", [](int i, int j) { return i + j; }); | ||
|  | } | ||
|  | 
 | ||
|  | PYBIND11_EMBEDDED_MODULE(trampoline_module, m) { | ||
|  |     py::class_<test_override_cache_helper, | ||
|  |                test_override_cache_helper_trampoline, | ||
|  |                std::shared_ptr<test_override_cache_helper>>(m, "test_override_cache_helper") | ||
|  |         .def(py::init_alias<>()) | ||
|  |         .def("func", &test_override_cache_helper::func); | ||
|  | } | ||
|  | 
 | ||
|  | PYBIND11_EMBEDDED_MODULE(throw_exception, ) { throw std::runtime_error("C++ Error"); } | ||
|  | 
 | ||
|  | PYBIND11_EMBEDDED_MODULE(throw_error_already_set, ) { | ||
|  |     auto d = py::dict(); | ||
|  |     d["missing"].cast<py::object>(); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("PYTHONPATH is used to update sys.path") { | ||
|  |     // The setup for this TEST_CASE is in catch.cpp!
 | ||
|  |     auto sys_path = py::str(py::module_::import("sys").attr("path")).cast<std::string>(); | ||
|  |     REQUIRE_THAT(sys_path, | ||
|  |                  Catch::Matchers::Contains("pybind11_test_embed_PYTHONPATH_2099743835476552")); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Pass classes and data between modules defined in C++ and Python") { | ||
|  |     auto module_ = py::module_::import("test_interpreter"); | ||
|  |     REQUIRE(py::hasattr(module_, "DerivedWidget")); | ||
|  | 
 | ||
|  |     auto locals = py::dict("hello"_a = "Hello, World!", "x"_a = 5, **module_.attr("__dict__")); | ||
|  |     py::exec(R"( | ||
|  |         widget = DerivedWidget("{} - {}".format(hello, x)) | ||
|  |         message = widget.the_message | ||
|  |     )", | ||
|  |              py::globals(), | ||
|  |              locals); | ||
|  |     REQUIRE(locals["message"].cast<std::string>() == "Hello, World! - 5"); | ||
|  | 
 | ||
|  |     auto py_widget = module_.attr("DerivedWidget")("The question"); | ||
|  |     auto message = py_widget.attr("the_message"); | ||
|  |     REQUIRE(message.cast<std::string>() == "The question"); | ||
|  | 
 | ||
|  |     const auto &cpp_widget = py_widget.cast<const Widget &>(); | ||
|  |     REQUIRE(cpp_widget.the_answer() == 42); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Override cache") { | ||
|  |     auto module_ = py::module_::import("test_trampoline"); | ||
|  |     REQUIRE(py::hasattr(module_, "func")); | ||
|  |     REQUIRE(py::hasattr(module_, "func2")); | ||
|  | 
 | ||
|  |     auto locals = py::dict(**module_.attr("__dict__")); | ||
|  | 
 | ||
|  |     int i = 0; | ||
|  |     for (; i < 1500; ++i) { | ||
|  |         std::shared_ptr<test_override_cache_helper> p_obj; | ||
|  |         std::shared_ptr<test_override_cache_helper> p_obj2; | ||
|  | 
 | ||
|  |         py::object loc_inst = locals["func"](); | ||
|  |         p_obj = py::cast<std::shared_ptr<test_override_cache_helper>>(loc_inst); | ||
|  | 
 | ||
|  |         int ret = p_obj->func(); | ||
|  | 
 | ||
|  |         REQUIRE(ret == 42); | ||
|  | 
 | ||
|  |         loc_inst = locals["func2"](); | ||
|  | 
 | ||
|  |         p_obj2 = py::cast<std::shared_ptr<test_override_cache_helper>>(loc_inst); | ||
|  | 
 | ||
|  |         p_obj2->func(); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Import error handling") { | ||
|  |     REQUIRE_NOTHROW(py::module_::import("widget_module")); | ||
|  |     REQUIRE_THROWS_WITH(py::module_::import("throw_exception"), "ImportError: C++ Error"); | ||
|  |     REQUIRE_THROWS_WITH(py::module_::import("throw_error_already_set"), | ||
|  |                         Catch::Contains("ImportError: initialization failed")); | ||
|  | 
 | ||
|  |     auto locals = py::dict("is_keyerror"_a = false, "message"_a = "not set"); | ||
|  |     py::exec(R"( | ||
|  |         try: | ||
|  |             import throw_error_already_set | ||
|  |         except ImportError as e: | ||
|  |             is_keyerror = type(e.__cause__) == KeyError | ||
|  |             message = str(e.__cause__) | ||
|  |     )", | ||
|  |              py::globals(), | ||
|  |              locals); | ||
|  |     REQUIRE(locals["is_keyerror"].cast<bool>() == true); | ||
|  |     REQUIRE(locals["message"].cast<std::string>() == "'missing'"); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("There can be only one interpreter") { | ||
|  |     static_assert(std::is_move_constructible<py::scoped_interpreter>::value, ""); | ||
|  |     static_assert(!std::is_move_assignable<py::scoped_interpreter>::value, ""); | ||
|  |     static_assert(!std::is_copy_constructible<py::scoped_interpreter>::value, ""); | ||
|  |     static_assert(!std::is_copy_assignable<py::scoped_interpreter>::value, ""); | ||
|  | 
 | ||
|  |     REQUIRE_THROWS_WITH(py::initialize_interpreter(), "The interpreter is already running"); | ||
|  |     REQUIRE_THROWS_WITH(py::scoped_interpreter(), "The interpreter is already running"); | ||
|  | 
 | ||
|  |     py::finalize_interpreter(); | ||
|  |     REQUIRE_NOTHROW(py::scoped_interpreter()); | ||
|  |     { | ||
|  |         auto pyi1 = py::scoped_interpreter(); | ||
|  |         auto pyi2 = std::move(pyi1); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | 
 | ||
|  | #if PY_VERSION_HEX >= PYBIND11_PYCONFIG_SUPPORT_PY_VERSION_HEX
 | ||
|  | TEST_CASE("Custom PyConfig") { | ||
|  |     py::finalize_interpreter(); | ||
|  |     PyConfig config; | ||
|  |     PyConfig_InitPythonConfig(&config); | ||
|  |     REQUIRE_NOTHROW(py::scoped_interpreter{&config}); | ||
|  |     { | ||
|  |         py::scoped_interpreter p{&config}; | ||
|  |         REQUIRE(py::module_::import("widget_module").attr("add")(1, 41).cast<int>() == 42); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("scoped_interpreter with PyConfig_InitIsolatedConfig and argv") { | ||
|  |     py::finalize_interpreter(); | ||
|  |     { | ||
|  |         PyConfig config; | ||
|  |         PyConfig_InitIsolatedConfig(&config); | ||
|  |         char *argv[] = {strdup("a.out")}; | ||
|  |         py::scoped_interpreter argv_scope{&config, 1, argv}; | ||
|  |         std::free(argv[0]); | ||
|  |         auto module = py::module::import("test_interpreter"); | ||
|  |         auto py_widget = module.attr("DerivedWidget")("The question"); | ||
|  |         const auto &cpp_widget = py_widget.cast<const Widget &>(); | ||
|  |         REQUIRE(cpp_widget.argv0() == "a.out"); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("scoped_interpreter with PyConfig_InitPythonConfig and argv") { | ||
|  |     py::finalize_interpreter(); | ||
|  |     { | ||
|  |         PyConfig config; | ||
|  |         PyConfig_InitPythonConfig(&config); | ||
|  | 
 | ||
|  |         // `initialize_interpreter() overrides the default value for config.parse_argv (`1`) by
 | ||
|  |         // changing it to `0`. This test exercises `scoped_interpreter` with the default config.
 | ||
|  |         char *argv[] = {strdup("a.out"), strdup("arg1")}; | ||
|  |         py::scoped_interpreter argv_scope(&config, 2, argv); | ||
|  |         std::free(argv[0]); | ||
|  |         std::free(argv[1]); | ||
|  |         auto module = py::module::import("test_interpreter"); | ||
|  |         auto py_widget = module.attr("DerivedWidget")("The question"); | ||
|  |         const auto &cpp_widget = py_widget.cast<const Widget &>(); | ||
|  |         REQUIRE(cpp_widget.argv0() == "arg1"); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | TEST_CASE("Add program dir to path pre-PyConfig") { | ||
|  |     py::finalize_interpreter(); | ||
|  |     size_t path_size_add_program_dir_to_path_false = 0; | ||
|  |     { | ||
|  |         py::scoped_interpreter scoped_interp{true, 0, nullptr, false}; | ||
|  |         path_size_add_program_dir_to_path_false = get_sys_path_size(); | ||
|  |     } | ||
|  |     { | ||
|  |         py::scoped_interpreter scoped_interp{}; | ||
|  |         REQUIRE(get_sys_path_size() == path_size_add_program_dir_to_path_false + 1); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | 
 | ||
|  | #if PY_VERSION_HEX >= PYBIND11_PYCONFIG_SUPPORT_PY_VERSION_HEX
 | ||
|  | TEST_CASE("Add program dir to path using PyConfig") { | ||
|  |     py::finalize_interpreter(); | ||
|  |     size_t path_size_add_program_dir_to_path_false = 0; | ||
|  |     { | ||
|  |         PyConfig config; | ||
|  |         PyConfig_InitPythonConfig(&config); | ||
|  |         py::scoped_interpreter scoped_interp{&config, 0, nullptr, false}; | ||
|  |         path_size_add_program_dir_to_path_false = get_sys_path_size(); | ||
|  |     } | ||
|  |     { | ||
|  |         PyConfig config; | ||
|  |         PyConfig_InitPythonConfig(&config); | ||
|  |         py::scoped_interpreter scoped_interp{&config}; | ||
|  |         REQUIRE(get_sys_path_size() == path_size_add_program_dir_to_path_false + 1); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | bool has_state_dict_internals_obj() { | ||
|  |     return bool( | ||
|  |         py::detail::get_internals_obj_from_state_dict(py::detail::get_python_state_dict())); | ||
|  | } | ||
|  | 
 | ||
|  | bool has_pybind11_internals_static() { | ||
|  |     auto **&ipp = py::detail::get_internals_pp(); | ||
|  |     return (ipp != nullptr) && (*ipp != nullptr); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Restart the interpreter") { | ||
|  |     // Verify pre-restart state.
 | ||
|  |     REQUIRE(py::module_::import("widget_module").attr("add")(1, 2).cast<int>() == 3); | ||
|  |     REQUIRE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE(has_pybind11_internals_static()); | ||
|  |     REQUIRE(py::module_::import("external_module").attr("A")(123).attr("value").cast<int>() | ||
|  |             == 123); | ||
|  | 
 | ||
|  |     // local and foreign module internals should point to the same internals:
 | ||
|  |     REQUIRE(reinterpret_cast<uintptr_t>(*py::detail::get_internals_pp()) | ||
|  |             == py::module_::import("external_module").attr("internals_at")().cast<uintptr_t>()); | ||
|  | 
 | ||
|  |     // Restart the interpreter.
 | ||
|  |     py::finalize_interpreter(); | ||
|  |     REQUIRE(Py_IsInitialized() == 0); | ||
|  | 
 | ||
|  |     py::initialize_interpreter(); | ||
|  |     REQUIRE(Py_IsInitialized() == 1); | ||
|  | 
 | ||
|  |     // Internals are deleted after a restart.
 | ||
|  |     REQUIRE_FALSE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE_FALSE(has_pybind11_internals_static()); | ||
|  |     pybind11::detail::get_internals(); | ||
|  |     REQUIRE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE(has_pybind11_internals_static()); | ||
|  |     REQUIRE(reinterpret_cast<uintptr_t>(*py::detail::get_internals_pp()) | ||
|  |             == py::module_::import("external_module").attr("internals_at")().cast<uintptr_t>()); | ||
|  | 
 | ||
|  |     // Make sure that an interpreter with no get_internals() created until finalize still gets the
 | ||
|  |     // internals destroyed
 | ||
|  |     py::finalize_interpreter(); | ||
|  |     py::initialize_interpreter(); | ||
|  |     bool ran = false; | ||
|  |     py::module_::import("__main__").attr("internals_destroy_test") | ||
|  |         = py::capsule(&ran, [](void *ran) { | ||
|  |               py::detail::get_internals(); | ||
|  |               *static_cast<bool *>(ran) = true; | ||
|  |           }); | ||
|  |     REQUIRE_FALSE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE_FALSE(has_pybind11_internals_static()); | ||
|  |     REQUIRE_FALSE(ran); | ||
|  |     py::finalize_interpreter(); | ||
|  |     REQUIRE(ran); | ||
|  |     py::initialize_interpreter(); | ||
|  |     REQUIRE_FALSE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE_FALSE(has_pybind11_internals_static()); | ||
|  | 
 | ||
|  |     // C++ modules can be reloaded.
 | ||
|  |     auto cpp_module = py::module_::import("widget_module"); | ||
|  |     REQUIRE(cpp_module.attr("add")(1, 2).cast<int>() == 3); | ||
|  | 
 | ||
|  |     // C++ type information is reloaded and can be used in python modules.
 | ||
|  |     auto py_module = py::module_::import("test_interpreter"); | ||
|  |     auto py_widget = py_module.attr("DerivedWidget")("Hello after restart"); | ||
|  |     REQUIRE(py_widget.attr("the_message").cast<std::string>() == "Hello after restart"); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Subinterpreter") { | ||
|  |     // Add tags to the modules in the main interpreter and test the basics.
 | ||
|  |     py::module_::import("__main__").attr("main_tag") = "main interpreter"; | ||
|  |     { | ||
|  |         auto m = py::module_::import("widget_module"); | ||
|  |         m.attr("extension_module_tag") = "added to module in main interpreter"; | ||
|  | 
 | ||
|  |         REQUIRE(m.attr("add")(1, 2).cast<int>() == 3); | ||
|  |     } | ||
|  |     REQUIRE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE(has_pybind11_internals_static()); | ||
|  | 
 | ||
|  |     /// Create and switch to a subinterpreter.
 | ||
|  |     auto *main_tstate = PyThreadState_Get(); | ||
|  |     auto *sub_tstate = Py_NewInterpreter(); | ||
|  | 
 | ||
|  |     // Subinterpreters get their own copy of builtins. detail::get_internals() still
 | ||
|  |     // works by returning from the static variable, i.e. all interpreters share a single
 | ||
|  |     // global pybind11::internals;
 | ||
|  |     REQUIRE_FALSE(has_state_dict_internals_obj()); | ||
|  |     REQUIRE(has_pybind11_internals_static()); | ||
|  | 
 | ||
|  |     // Modules tags should be gone.
 | ||
|  |     REQUIRE_FALSE(py::hasattr(py::module_::import("__main__"), "tag")); | ||
|  |     { | ||
|  |         auto m = py::module_::import("widget_module"); | ||
|  |         REQUIRE_FALSE(py::hasattr(m, "extension_module_tag")); | ||
|  | 
 | ||
|  |         // Function bindings should still work.
 | ||
|  |         REQUIRE(m.attr("add")(1, 2).cast<int>() == 3); | ||
|  |     } | ||
|  | 
 | ||
|  |     // Restore main interpreter.
 | ||
|  |     Py_EndInterpreter(sub_tstate); | ||
|  |     PyThreadState_Swap(main_tstate); | ||
|  | 
 | ||
|  |     REQUIRE(py::hasattr(py::module_::import("__main__"), "main_tag")); | ||
|  |     REQUIRE(py::hasattr(py::module_::import("widget_module"), "extension_module_tag")); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Execution frame") { | ||
|  |     // When the interpreter is embedded, there is no execution frame, but `py::exec`
 | ||
|  |     // should still function by using reasonable globals: `__main__.__dict__`.
 | ||
|  |     py::exec("var = dict(number=42)"); | ||
|  |     REQUIRE(py::globals()["var"]["number"].cast<int>() == 42); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("Threads") { | ||
|  |     // Restart interpreter to ensure threads are not initialized
 | ||
|  |     py::finalize_interpreter(); | ||
|  |     py::initialize_interpreter(); | ||
|  |     REQUIRE_FALSE(has_pybind11_internals_static()); | ||
|  | 
 | ||
|  |     constexpr auto num_threads = 10; | ||
|  |     auto locals = py::dict("count"_a = 0); | ||
|  | 
 | ||
|  |     { | ||
|  |         py::gil_scoped_release gil_release{}; | ||
|  | 
 | ||
|  |         auto threads = std::vector<std::thread>(); | ||
|  |         for (auto i = 0; i < num_threads; ++i) { | ||
|  |             threads.emplace_back([&]() { | ||
|  |                 py::gil_scoped_acquire gil{}; | ||
|  |                 locals["count"] = locals["count"].cast<int>() + 1; | ||
|  |             }); | ||
|  |         } | ||
|  | 
 | ||
|  |         for (auto &thread : threads) { | ||
|  |             thread.join(); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     REQUIRE(locals["count"].cast<int>() == num_threads); | ||
|  | } | ||
|  | 
 | ||
|  | // Scope exit utility https://stackoverflow.com/a/36644501/7255855
 | ||
|  | struct scope_exit { | ||
|  |     std::function<void()> f_; | ||
|  |     explicit scope_exit(std::function<void()> f) noexcept : f_(std::move(f)) {} | ||
|  |     ~scope_exit() { | ||
|  |         if (f_) { | ||
|  |             f_(); | ||
|  |         } | ||
|  |     } | ||
|  | }; | ||
|  | 
 | ||
|  | TEST_CASE("Reload module from file") { | ||
|  |     // Disable generation of cached bytecode (.pyc files) for this test, otherwise
 | ||
|  |     // Python might pick up an old version from the cache instead of the new versions
 | ||
|  |     // of the .py files generated below
 | ||
|  |     auto sys = py::module_::import("sys"); | ||
|  |     bool dont_write_bytecode = sys.attr("dont_write_bytecode").cast<bool>(); | ||
|  |     sys.attr("dont_write_bytecode") = true; | ||
|  |     // Reset the value at scope exit
 | ||
|  |     scope_exit reset_dont_write_bytecode( | ||
|  |         [&]() { sys.attr("dont_write_bytecode") = dont_write_bytecode; }); | ||
|  | 
 | ||
|  |     std::string module_name = "test_module_reload"; | ||
|  |     std::string module_file = module_name + ".py"; | ||
|  | 
 | ||
|  |     // Create the module .py file
 | ||
|  |     std::ofstream test_module(module_file); | ||
|  |     test_module << "def test():\n"; | ||
|  |     test_module << "    return 1\n"; | ||
|  |     test_module.close(); | ||
|  |     // Delete the file at scope exit
 | ||
|  |     scope_exit delete_module_file([&]() { std::remove(module_file.c_str()); }); | ||
|  | 
 | ||
|  |     // Import the module from file
 | ||
|  |     auto module_ = py::module_::import(module_name.c_str()); | ||
|  |     int result = module_.attr("test")().cast<int>(); | ||
|  |     REQUIRE(result == 1); | ||
|  | 
 | ||
|  |     // Update the module .py file with a small change
 | ||
|  |     test_module.open(module_file); | ||
|  |     test_module << "def test():\n"; | ||
|  |     test_module << "    return 2\n"; | ||
|  |     test_module.close(); | ||
|  | 
 | ||
|  |     // Reload the module
 | ||
|  |     module_.reload(); | ||
|  |     result = module_.attr("test")().cast<int>(); | ||
|  |     REQUIRE(result == 2); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("sys.argv gets initialized properly") { | ||
|  |     py::finalize_interpreter(); | ||
|  |     { | ||
|  |         py::scoped_interpreter default_scope; | ||
|  |         auto module = py::module::import("test_interpreter"); | ||
|  |         auto py_widget = module.attr("DerivedWidget")("The question"); | ||
|  |         const auto &cpp_widget = py_widget.cast<const Widget &>(); | ||
|  |         REQUIRE(cpp_widget.argv0().empty()); | ||
|  |     } | ||
|  | 
 | ||
|  |     { | ||
|  |         char *argv[] = {strdup("a.out")}; | ||
|  |         py::scoped_interpreter argv_scope(true, 1, argv); | ||
|  |         std::free(argv[0]); | ||
|  |         auto module = py::module::import("test_interpreter"); | ||
|  |         auto py_widget = module.attr("DerivedWidget")("The question"); | ||
|  |         const auto &cpp_widget = py_widget.cast<const Widget &>(); | ||
|  |         REQUIRE(cpp_widget.argv0() == "a.out"); | ||
|  |     } | ||
|  |     py::initialize_interpreter(); | ||
|  | } | ||
|  | 
 | ||
|  | TEST_CASE("make_iterator can be called before then after finalizing an interpreter") { | ||
|  |     // Reproduction of issue #2101 (https://github.com/pybind/pybind11/issues/2101)
 | ||
|  |     py::finalize_interpreter(); | ||
|  | 
 | ||
|  |     std::vector<int> container; | ||
|  |     { | ||
|  |         pybind11::scoped_interpreter g; | ||
|  |         auto iter = pybind11::make_iterator(container.begin(), container.end()); | ||
|  |     } | ||
|  | 
 | ||
|  |     REQUIRE_NOTHROW([&]() { | ||
|  |         pybind11::scoped_interpreter g; | ||
|  |         auto iter = pybind11::make_iterator(container.begin(), container.end()); | ||
|  |     }()); | ||
|  | 
 | ||
|  |     py::initialize_interpreter(); | ||
|  | } |