diff --git a/NEWS.md b/NEWS.md index c6fd11130..0012595ac 100644 --- a/NEWS.md +++ b/NEWS.md @@ -12,7 +12,7 @@ - Added support for Python 3.13. Note that Python 3.13 removed support for `classmethod` descriptors, which may affect the S3 class of some Python objects that use metaclass properties to resolve a class’s - `__module__` or `__name__` attribute. (#1686) + `__module__` or `__name__` attribute. (#1686, #1698) - `py_is_null_xptr()` and `[[` now load delayed modules (#1688). diff --git a/R/RcppExports.R b/R/RcppExports.R index a0900d864..c321c2fea 100644 --- a/R/RcppExports.R +++ b/R/RcppExports.R @@ -295,8 +295,8 @@ py_eval_impl <- function(code, convert = TRUE) { .Call(`_reticulate_py_eval_impl`, code, convert) } -py_convert_pandas_series <- function(series) { - .Call(`_reticulate_py_convert_pandas_series`, series) +py_convert_pandas_series <- function(series_) { + .Call(`_reticulate_py_convert_pandas_series`, series_) } py_convert_pandas_df <- function(df) { diff --git a/src/RcppExports.cpp b/src/RcppExports.cpp index 60a1663fa..9f8441b71 100644 --- a/src/RcppExports.cpp +++ b/src/RcppExports.cpp @@ -650,13 +650,13 @@ BEGIN_RCPP END_RCPP } // py_convert_pandas_series -SEXP py_convert_pandas_series(PyObjectRef series); -RcppExport SEXP _reticulate_py_convert_pandas_series(SEXP seriesSEXP) { +SEXP py_convert_pandas_series(PyObjectRef series_); +RcppExport SEXP _reticulate_py_convert_pandas_series(SEXP series_SEXP) { BEGIN_RCPP Rcpp::RObject rcpp_result_gen; Rcpp::RNGScope rcpp_rngScope_gen; - Rcpp::traits::input_parameter< PyObjectRef >::type series(seriesSEXP); - rcpp_result_gen = Rcpp::wrap(py_convert_pandas_series(series)); + Rcpp::traits::input_parameter< PyObjectRef >::type series_(series_SEXP); + rcpp_result_gen = Rcpp::wrap(py_convert_pandas_series(series_)); return rcpp_result_gen; END_RCPP } diff --git a/src/libpython.cpp b/src/libpython.cpp index 4d62ce801..e0aac3ee0 100644 --- a/src/libpython.cpp +++ b/src/libpython.cpp @@ -61,7 +61,7 @@ bool loadLibrary(const std::string& libPath, void** ppLib, std::string* pError) *ppLib = (void*)::LoadLibraryEx(libPath.c_str(), NULL, 0); #else if (libPath == "NA") { - *ppLib = ::dlopen(NULL, RTLD_NOW|RTLD_GLOBAL); + *ppLib = ::dlopen(NULL, RTLD_NOW|RTLD_GLOBAL); // on linux, should we also do: | RTLD_DEEPBIND ?? } else { *ppLib = ::dlopen(libPath.c_str(), RTLD_NOW|RTLD_GLOBAL); } diff --git a/src/python.cpp b/src/python.cpp index e3400dd21..c645b2fbf 100644 --- a/src/python.cpp +++ b/src/python.cpp @@ -1312,8 +1312,21 @@ void set_string_element(SEXP rArray, int i, PyObject* pyStr) { SET_STRING_ELT(rArray, i, strSEXP); } +static inline +bool py_has_attr(PyObject* x_, const char* name) { + switch (PyObject_HasAttrStringWithError(x_, name)) { + case 1: return true; + case 0: return false; + case -1: + default: + PyErr_Clear(); + return false; + } +} + + bool py_is_callable(PyObject* x) { - return PyCallable_Check(x) == 1 || PyObject_HasAttrString(x, "__call__"); + return PyCallable_Check(x) == 1 || py_has_attr(x, "__call__"); } // [[Rcpp::export]] @@ -1546,7 +1559,7 @@ SEXP py_to_r_cpp(PyObject* x, bool convert, bool simple) { } // tuple (but don't convert namedtuple as it's often a custom class) - if (PyTuple_CheckExact(x) && !PyObject_HasAttrString(x, "_fields")) { + if (PyTuple_CheckExact(x) && !py_has_attr(x, "_fields")) { Py_ssize_t len = PyTuple_Size(x); Rcpp::List list(len); for (Py_ssize_t i = 0; i