2017-09-13 02:46:56 -04:00
// The MIT License (MIT)
2017-08-31 20:47:09 -04:00
// Copyright (c) 2013-2017 Rapptz, ThePhD and contributors
// Permission is hereby granted, free of charge, to any person obtaining a copy of
// this software and associated documentation files (the "Software"), to deal in
// the Software without restriction, including without limitation the rights to
// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
// the Software, and to permit persons to whom the Software is furnished to do so,
// subject to the following conditions:
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
// FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
// COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
// IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
# ifndef SOL_ERROR_HANDLER_HPP
# define SOL_ERROR_HANDLER_HPP
# include "types.hpp"
2017-09-05 20:29:21 -04:00
# include "demangle.hpp"
2017-08-31 20:47:09 -04:00
namespace sol {
2017-10-26 12:53:52 -04:00
inline std : : string associated_type_name ( lua_State * L , int index , type t ) {
switch ( t ) {
case type : : poly :
return " anything " ;
case type : : userdata :
{
if ( lua_getmetatable ( L , index ) = = 0 ) {
break ;
}
lua_pushlstring ( L , " __name " , 6 ) ;
lua_rawget ( L , - 2 ) ;
size_t sz ;
const char * name = lua_tolstring ( L , - 1 , & sz ) ;
std : : string tn ( name , static_cast < std : : string : : size_type > ( sz ) ) ;
lua_pop ( L , 2 ) ;
return name ;
}
default :
break ;
}
return lua_typename ( L , static_cast < int > ( t ) ) ;
}
2017-08-31 20:47:09 -04:00
inline int type_panic_string ( lua_State * L , int index , type expected , type actual , const std : : string & message = " " ) noexcept ( false ) {
2017-10-26 12:53:52 -04:00
const char * err = message . empty ( ) ? " stack index %d, expected %s, received %s " : " stack index %d, expected %s, received %s: %s " ;
std : : string actualname = associated_type_name ( L , index , actual ) ;
2017-08-31 20:47:09 -04:00
return luaL_error ( L , err , index ,
expected = = type : : poly ? " anything " : lua_typename ( L , static_cast < int > ( expected ) ) ,
2017-10-26 12:53:52 -04:00
actualname . c_str ( ) ,
2017-09-13 02:46:56 -04:00
message . c_str ( ) ) ;
2017-08-31 20:47:09 -04:00
}
inline int type_panic_c_str ( lua_State * L , int index , type expected , type actual , const char * message = nullptr ) noexcept ( false ) {
2017-10-26 12:53:52 -04:00
const char * err = message = = nullptr | | ( std : : char_traits < char > : : length ( message ) = = 0 ) ? " stack index %d, expected %s, received %s " : " stack index %d, expected %s, received %s: %s " ;
std : : string actualname = associated_type_name ( L , index , actual ) ;
2017-08-31 20:47:09 -04:00
return luaL_error ( L , err , index ,
expected = = type : : poly ? " anything " : lua_typename ( L , static_cast < int > ( expected ) ) ,
2017-10-26 12:53:52 -04:00
actualname . c_str ( ) ,
2017-09-13 02:46:56 -04:00
message ) ;
2017-08-31 20:47:09 -04:00
}
struct type_panic_t {
int operator ( ) ( lua_State * L , int index , type expected , type actual ) const noexcept ( false ) {
return type_panic_c_str ( L , index , expected , actual , nullptr ) ;
}
int operator ( ) ( lua_State * L , int index , type expected , type actual , const char * message ) const noexcept ( false ) {
return type_panic_c_str ( L , index , expected , actual , message ) ;
}
int operator ( ) ( lua_State * L , int index , type expected , type actual , const std : : string & message ) const noexcept ( false ) {
return type_panic_string ( L , index , expected , actual , message ) ;
}
} ;
const type_panic_t type_panic = { } ;
struct constructor_handler {
int operator ( ) ( lua_State * L , int index , type expected , type actual , const std : : string & message ) const noexcept ( false ) {
return type_panic_string ( L , index , expected , actual , message + " (type check failed in constructor) " ) ;
}
} ;
2017-09-05 13:58:17 -04:00
template < typename F = void >
2017-08-31 20:47:09 -04:00
struct argument_handler {
int operator ( ) ( lua_State * L , int index , type expected , type actual , const std : : string & message ) const noexcept ( false ) {
return type_panic_string ( L , index , expected , actual , message + " (bad argument to variable or function call) " ) ;
}
} ;
2017-09-05 13:58:17 -04:00
template < typename R , typename . . . Args >
struct argument_handler < types < R , Args . . . > > {
int operator ( ) ( lua_State * L , int index , type expected , type actual , const std : : string & message ) const noexcept ( false ) {
2017-10-26 12:53:52 -04:00
std : : string addendum = " (bad argument into ' " ;
2017-09-05 13:58:17 -04:00
addendum + = detail : : demangle < R > ( ) ;
addendum + = " ( " ;
int marker = 0 ;
auto action = [ & addendum , & marker ] ( const std : : string & n ) {
if ( marker > 0 ) {
addendum + = " , " ;
}
addendum + = n ;
+ + marker ;
2017-09-05 21:07:33 -04:00
} ;
2017-09-13 02:46:56 -04:00
( void ) detail : : swallow { int ( ) , ( action ( detail : : demangle < Args > ( ) ) , int ( ) ) . . . } ;
2017-09-05 13:58:17 -04:00
addendum + = " )') " ;
return type_panic_string ( L , index , expected , actual , message + addendum ) ;
}
} ;
2017-08-31 20:47:09 -04:00
// Specify this function as the handler for lua::check if you know there's nothing wrong
inline int no_panic ( lua_State * , int , type , type , const char * = nullptr ) noexcept {
return 0 ;
}
inline void type_error ( lua_State * L , int expected , int actual ) noexcept ( false ) {
luaL_error ( L , " expected %s, received %s " , lua_typename ( L , expected ) , lua_typename ( L , actual ) ) ;
}
inline void type_error ( lua_State * L , type expected , type actual ) noexcept ( false ) {
type_error ( L , static_cast < int > ( expected ) , static_cast < int > ( actual ) ) ;
}
inline void type_assert ( lua_State * L , int index , type expected , type actual ) noexcept ( false ) {
if ( expected ! = type : : poly & & expected ! = actual ) {
type_panic_c_str ( L , index , expected , actual , nullptr ) ;
}
}
inline void type_assert ( lua_State * L , int index , type expected ) {
type actual = type_of ( L , index ) ;
type_assert ( L , index , expected , actual ) ;
}
2017-09-13 02:46:56 -04:00
} // namespace sol
2017-08-31 20:47:09 -04:00
# endif // SOL_ERROR_HANDLER_HPP