From 59b196a1db28fd1ab2a2c40ca4864c76447d24c2 Mon Sep 17 00:00:00 2001 From: ThePhD Date: Fri, 31 Mar 2017 17:38:04 -0400 Subject: [PATCH] add a new kind of error handling script call, updating docs and examples --- docs/source/api/state.rst | 27 +++++- docs/source/errors.rst | 4 + docs/source/tutorial/all-the-things.rst | 29 +++++- docs/source/tutorial/getting-started.rst | 10 +- examples/basic.cpp | 26 +++++- single/sol/sol.hpp | 112 ++++++++++++++++++++++- sol/state_view.hpp | 49 +++++++++- sol/types.hpp | 59 +++++++++++- 8 files changed, 301 insertions(+), 15 deletions(-) diff --git a/docs/source/api/state.rst b/docs/source/api/state.rst index b1b17a3c..d9411757 100644 --- a/docs/source/api/state.rst +++ b/docs/source/api/state.rst @@ -58,9 +58,34 @@ This function takes a number of :ref:`sol::lib` as arguments and opens sol::function_result script(const std::string& code); sol::function_result script_file(const std::string& filename); + template + sol::protected_function_result script(const std::string& code, Func&& func); + template + sol::protected_function_result script_file(const std::string& filename, Func&& func); + These functions run the desired blob of either code that is in a string, or code that comes from a filename, on the ``lua_State*``. It will not run isolated: any scripts or code run will affect code in the ``lua_State*`` the object uses as well (unless ``local`` is applied to a variable declaration, as specified by the Lua language). Code ran in this fashion is not isolated. If you need isolation, consider creating a new state or traditional Lua sandboxing techniques. -If your script returns a value, you can capture it from the returned :ref:`function_result`. +If your script returns a value, you can capture it from the returned :ref:`sol::function_result`/:ref:`sol::protected_function_result`. + +To handle errors when using the second overload, provide a callable function/object that takes a ``lua_State*`` as its first argument and a ``sol::protected_function_result`` as its second argument. Then, handle the errors any way you like: + +.. code-block:: cpp + :caption: running code safely + :name: state-script-safe + + int main () { + sol::state lua; + // the default handler panics or throws, depending on your settings + auto result1 = lua.script("bad.code", &sol::default_on_error); + auto result2 = lua.script("123 bad.code", [](lua_State* L, sol::protected_function_result pfr) { + // pfr will contain things that went wrong, for either loading or executing the script + // the user can do whatever they like here, including throw. Otherwise, + // they need to return the protected_function_result + + // You can also just return it, and let the call-site handle the error if necessary. + return pfr; + }); + } .. code-block:: cpp :caption: function: require / require_file diff --git a/docs/source/errors.rst b/docs/source/errors.rst index 47eaf27e..bda26a64 100644 --- a/docs/source/errors.rst +++ b/docs/source/errors.rst @@ -5,6 +5,10 @@ how to handle exceptions or other errors Here is some advice and some tricks for common errors about iteration, compile time / linker errors, and other pitfalls, especially when dealing with thrown exceptions, error conditions and the like in Sol. +Running Scripts +--------------- + +Scripts can have syntax errors, can load from the file system wrong, or have runtime issues. Knowing which one can be troublesome. There are various small building blocks to load and run code, but to check errors you can use the overloaded :ref:`script/script_file functions on sol::state/sol::state_view` Linker Errors ------------- diff --git a/docs/source/tutorial/all-the-things.rst b/docs/source/tutorial/all-the-things.rst index cd7ec63a..ca2f0707 100644 --- a/docs/source/tutorial/all-the-things.rst +++ b/docs/source/tutorial/all-the-things.rst @@ -47,6 +47,23 @@ running lua code int value = lua.script("return 54"); // value == 54 +To run Lua code but have an error handler in case things go wrong: + +.. code-block:: cpp + + sol::state lua; + + // the default handler panics or throws, depending on your settings + auto result1 = lua.script("bad.code", &sol::default_on_error); + + auto result2 = lua.script("123 herp.derp", [](lua_State* L, sol::protected_function_result pfr) { + // pfr will contain things that went wrong, for either loading or executing the script + // Can throw your own custom error + // You can also just return it, and let the call-site handle the error if necessary. + return pfr; + }); + + To check the success of a loading operation: .. code-block:: cpp @@ -54,16 +71,24 @@ To check the success of a loading operation: // load file without execute sol::load_result script1 = lua.load_file("path/to/luascript.lua"); script1(); //execute + // load string without execute sol::load_result script2 = lua.load("a = 'test'"); - script2(); //execute + sol::protected_function_result script2result = script2(); //execute + // optionally, check if it worked + if (script2result.valid()) { + // yay! + } + else { + // aww + } sol::load_result script3 = lua.load("return 24"); int value2 = script3(); // execute, get return value // value2 == 24 -To check whether a script was successfully run or not (after loading is assumed to be successful): +To check whether a script was successfully run or not (if the actual loading is successful): .. code-block:: cpp diff --git a/docs/source/tutorial/getting-started.rst b/docs/source/tutorial/getting-started.rst index 02fc2daa..caab0d81 100644 --- a/docs/source/tutorial/getting-started.rst +++ b/docs/source/tutorial/getting-started.rst @@ -38,7 +38,9 @@ Or using your favorite IDE / tool after setting up your include paths and librar If you get an avalanche of errors (particularly referring to ``auto``), you may not have enabled C++14 / C++17 mode for your compiler. Add one of ``std=c++14``, ``std=c++1z`` OR ``std=c++1y`` to your compiler options. By default, this is always-on for VC++ compilers in Visual Studio and friends, but g++ and clang++ require a flag (unless you're on `GCC 6.0`_). -If this works, you're ready to start! The first line creates the ``lua_State`` and will hold onto it for the duration of the scope its declared in (e.g., from the opening ``{`` to the closing ``}``). It will automatically close / cleanup that lua state when it gets destructed. The second line opens a single lua-provided library, "base". There are several other libraries that come with lua that you can open by default, and those are included in the :ref:`sol::lib` enumeration. You can open multiple base libraries by specifying multiple ``sol::lib`` arguments: +If this works, you're ready to start! The first line creates the ``lua_State`` and will hold onto it for the duration of the scope its declared in (e.g., from the opening ``{`` to the closing ``}``). It will automatically close / cleanup that lua state when it gets destructed. + +The second line opens a single lua-provided library, "base". There are several other libraries that come with lua that you can open by default, and those are included in the :ref:`sol::lib` enumeration. You can open multiple base libraries by specifying multiple ``sol::lib`` arguments: .. code-block:: cpp :linenos: @@ -59,6 +61,8 @@ If this works, you're ready to start! The first line creates the ``lua_State`` a If you're interested in integrating Sol with a project that already uses some other library or Lua in the codebase, check out the :doc:`existing example` to see how to work with Sol when you add it to a project (the existing example covers ``require`` as well)! +Some more ways of loading scripts and handling errors is shown `in this example`_! + Next, let's start :doc:`reading/writing some variables` from Lua into C++, and vice-versa! @@ -76,4 +80,6 @@ Next, let's start :doc:`reading/writing some variables` from Lua into .. _github repository here: https://github.com/ThePhD/sol2 -.. _Lua page on getting started: https://www.lua.org/start.html \ No newline at end of file +.. _Lua page on getting started: https://www.lua.org/start.html + +.. _in this example: https://github.com/ThePhD/sol2/blob/develop/examples/basic.cpp \ No newline at end of file diff --git a/examples/basic.cpp b/examples/basic.cpp index 8d9d62c3..7178caf1 100644 --- a/examples/basic.cpp +++ b/examples/basic.cpp @@ -3,17 +3,39 @@ #include int main() { - // create an empty lua state + std::cout << "=== basic example ===" << std::endl; + // create an empty lua state sol::state lua; // by default, libraries are not opened // you can open libraries by using open_libraries // the libraries reside in the sol::lib enum class lua.open_libraries(sol::lib::base); + // you can open all libraries by passing no arguments + //lua.open_libraries(); // call lua code directly - std::cout << "=== basic example ===" << std::endl; lua.script("print('hello world')"); + // call lua code, and check to make sure it has loaded and run properly: + auto handler = &sol::default_on_error; + lua.script("print('hello again, world')", handler); + + // Use a custom error handler if you need it + // This gets called when the result is bad + auto simple_handler = [](lua_State*, sol::protected_function_result result) { + // You can just pass it through to let the call-site handle it + return result; + }; + auto result = lua.script("print('hello hello again, world') \n return 24", simple_handler); + if (result.valid()) { + std::cout << "the code worked, and a double-hello statement should appear above this one!" << std::endl; + int value = result; + assert(value == 24); + } + else { + std::cout << "the code failed, check the result type for more information!" << std::endl; + } + std::cout << std::endl; } \ No newline at end of file diff --git a/single/sol/sol.hpp b/single/sol/sol.hpp index 3e4c42c5..0ca98b6e 100644 --- a/single/sol/sol.hpp +++ b/single/sol/sol.hpp @@ -20,8 +20,8 @@ // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // This file was generated with a script. -// Generated 2017-03-30 18:11:26.562402 UTC -// This header was generated with sol v2.16.0 (revision 92d31b3) +// Generated 2017-03-31 21:37:49.518087 UTC +// This header was generated with sol v2.16.0 (revision 1e367ab) // https://github.com/ThePhD/sol2 #ifndef SOL_SINGLE_INCLUDE_HPP @@ -3324,7 +3324,9 @@ namespace sol { runtime = LUA_ERRRUN, memory = LUA_ERRMEM, handler = LUA_ERRERR, - gc = LUA_ERRGCMM + gc = LUA_ERRGCMM, + syntax = LUA_ERRSYNTAX, + file = LUA_ERRFILE, }; enum class thread_status : int { @@ -3363,6 +3365,61 @@ namespace sol { table | boolean | function | userdata | lightuserdata }; + inline const std::string& to_string(call_status c) { + static const std::array names{{ + "ok", + "yielded", + "runtime", + "memory", + "handler", + "gc", + "syntax", + "file", + }}; + switch (c) { + case call_status::ok: + return names[0]; + case call_status::yielded: + return names[1]; + case call_status::runtime: + return names[2]; + case call_status::memory: + return names[3]; + case call_status::handler: + return names[4]; + case call_status::gc: + return names[5]; + case call_status::syntax: + return names[6]; + case call_status::file: + return names[7]; + } + return names[0]; + } + + inline const std::string& to_string(load_status c) { + static const std::array names{ { + "ok", + "memory", + "gc", + "syntax", + "file", + } }; + switch (c) { + case load_status::ok: + return names[0]; + case load_status::memory: + return names[1]; + case load_status::gc: + return names[2]; + case load_status::syntax: + return names[3]; + case load_status::file: + return names[4]; + } + return names[0]; + } + enum class meta_function { construct, index, @@ -12674,6 +12731,25 @@ namespace sol { return kb; } + inline protected_function_result default_on_error( lua_State* L, const protected_function_result& pfr ) { + type t = type_of(L, pfr.stack_index()); + std::string err = to_string(pfr.status()) + " error"; + if (t == type::string) { + err += " "; + err += stack::get(L, pfr.stack_index()); + } +#ifdef SOL_NO_EXCEPTIONS + if (t != type::nil) { + lua_pop(L, 1); + } + stack::push(L, err); + lua_error(L); +#else + throw error(detail::direct_error, err); +#endif + return pfr; + } + class state_view { private: lua_State* L; @@ -12856,15 +12932,41 @@ namespace sol { } protected_function_result do_string(const std::string& code) { - sol::protected_function pf = load(code); + load_status x = static_cast(luaL_loadstring(L, code.c_str())); + if (x != load_status::ok) { + return protected_function_result(L, -1, 0, 1, static_cast(x)); + } + protected_function pf(L, -1); return pf(); } protected_function_result do_file(const std::string& filename) { - sol::protected_function pf = load_file(filename); + load_status x = static_cast(luaL_loadfile(L, filename.c_str())); + if (x != load_status::ok) { + return protected_function_result(L, -1, 0, 1, static_cast(x)); + } + protected_function pf(L, -1); return pf(); } + template + protected_function_result script(const std::string& code, Fx&& on_error) { + protected_function_result pfr = do_string(code); + if (!pfr.valid()) { + return on_error(L, pfr); + } + return pfr; + } + + template + protected_function_result script_file(const std::string& filename, Fx&& on_error) { + protected_function_result pfr = do_file(filename); + if (!pfr.valid()) { + return on_error(L, pfr); + } + return pfr; + } + function_result script(const std::string& code) { int index = lua_gettop(L); stack::script(L, code); diff --git a/sol/state_view.hpp b/sol/state_view.hpp index 14032d65..34dfb33b 100644 --- a/sol/state_view.hpp +++ b/sol/state_view.hpp @@ -52,6 +52,25 @@ namespace sol { return kb; } + inline protected_function_result default_on_error( lua_State* L, const protected_function_result& pfr ) { + type t = type_of(L, pfr.stack_index()); + std::string err = to_string(pfr.status()) + " error"; + if (t == type::string) { + err += " "; + err += stack::get(L, pfr.stack_index()); + } +#ifdef SOL_NO_EXCEPTIONS + if (t != type::nil) { + lua_pop(L, 1); + } + stack::push(L, err); + lua_error(L); +#else + throw error(detail::direct_error, err); +#endif + return pfr; + } + class state_view { private: lua_State* L; @@ -234,15 +253,41 @@ namespace sol { } protected_function_result do_string(const std::string& code) { - sol::protected_function pf = load(code); + load_status x = static_cast(luaL_loadstring(L, code.c_str())); + if (x != load_status::ok) { + return protected_function_result(L, -1, 0, 1, static_cast(x)); + } + protected_function pf(L, -1); return pf(); } protected_function_result do_file(const std::string& filename) { - sol::protected_function pf = load_file(filename); + load_status x = static_cast(luaL_loadfile(L, filename.c_str())); + if (x != load_status::ok) { + return protected_function_result(L, -1, 0, 1, static_cast(x)); + } + protected_function pf(L, -1); return pf(); } + template + protected_function_result script(const std::string& code, Fx&& on_error) { + protected_function_result pfr = do_string(code); + if (!pfr.valid()) { + return on_error(L, pfr); + } + return pfr; + } + + template + protected_function_result script_file(const std::string& filename, Fx&& on_error) { + protected_function_result pfr = do_file(filename); + if (!pfr.valid()) { + return on_error(L, pfr); + } + return pfr; + } + function_result script(const std::string& code) { int index = lua_gettop(L); stack::script(L, code); diff --git a/sol/types.hpp b/sol/types.hpp index 92a61d52..354e7e94 100644 --- a/sol/types.hpp +++ b/sol/types.hpp @@ -345,7 +345,9 @@ namespace sol { runtime = LUA_ERRRUN, memory = LUA_ERRMEM, handler = LUA_ERRERR, - gc = LUA_ERRGCMM + gc = LUA_ERRGCMM, + syntax = LUA_ERRSYNTAX, + file = LUA_ERRFILE, }; enum class thread_status : int { @@ -384,6 +386,61 @@ namespace sol { table | boolean | function | userdata | lightuserdata }; + inline const std::string& to_string(call_status c) { + static const std::array names{{ + "ok", + "yielded", + "runtime", + "memory", + "handler", + "gc", + "syntax", + "file", + }}; + switch (c) { + case call_status::ok: + return names[0]; + case call_status::yielded: + return names[1]; + case call_status::runtime: + return names[2]; + case call_status::memory: + return names[3]; + case call_status::handler: + return names[4]; + case call_status::gc: + return names[5]; + case call_status::syntax: + return names[6]; + case call_status::file: + return names[7]; + } + return names[0]; + } + + inline const std::string& to_string(load_status c) { + static const std::array names{ { + "ok", + "memory", + "gc", + "syntax", + "file", + } }; + switch (c) { + case load_status::ok: + return names[0]; + case load_status::memory: + return names[1]; + case load_status::gc: + return names[2]; + case load_status::syntax: + return names[3]; + case load_status::file: + return names[4]; + } + return names[0]; + } + enum class meta_function { construct, index,