mirror of
https://github.com/pybind/pybind11.git
synced 2025-02-21 07:59:17 +00:00
* `#error BYE_BYE_GOLDEN_SNAKE` * Removing everything related to 2.7 from ci.yml * Commenting-out Centos7 * Removing `PYTHON: 27` from .appveyor.yml * "PY2" removal, mainly from tests. C++ code is not touched. * Systematic removal of `u` prefix from `u"..."` and `u'...'` literals. Collateral cleanup of a couple minor other things. * Cleaning up around case-insensitive hits for `[^a-z]py.*2` in tests/. * Removing obsolete Python 2 mention in compiling.rst * Proper `#error` for Python 2. * Using PY_VERSION_HEX to guard `#error "PYTHON 2 IS NO LONGER SUPPORTED.` * chore: bump pre-commit * style: run pre-commit for pyupgrade 3+ * tests: use sys.version_info, not PY * chore: more Python 2 removal * Uncommenting Centos7 block (PR #3691 showed that it is working again). * Update pre-commit hooks * Fix pre-commit hook * refactor: remove Python 2 from CMake * refactor: remove Python 2 from setup code * refactor: simplify, better static typing * feat: fail with nice messages * refactor: drop Python 2 C++ code * docs: cleanup for Python 3 * revert: intree revert: intree * docs: minor touchup to py2 statement Co-authored-by: Henry Schreiner <henryschreineriii@gmail.com> Co-authored-by: Aaron Gokaslan <skylion.aaron@gmail.com>
157 lines
4.6 KiB
C++
157 lines
4.6 KiB
C++
/*
|
|
pybind11/eval.h: Support for evaluating Python expressions and statements
|
|
from strings and files
|
|
|
|
Copyright (c) 2016 Klemens Morgenstern <klemens.morgenstern@ed-chemnitz.de> and
|
|
Wenzel Jakob <wenzel.jakob@epfl.ch>
|
|
|
|
All rights reserved. Use of this source code is governed by a
|
|
BSD-style license that can be found in the LICENSE file.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "pybind11.h"
|
|
|
|
#include <utility>
|
|
|
|
PYBIND11_NAMESPACE_BEGIN(PYBIND11_NAMESPACE)
|
|
PYBIND11_NAMESPACE_BEGIN(detail)
|
|
|
|
inline void ensure_builtins_in_globals(object &global) {
|
|
#if defined(PYPY_VERSION) || PY_VERSION_HEX < 0x03080000
|
|
// Running exec and eval adds `builtins` module under `__builtins__` key to
|
|
// globals if not yet present. Python 3.8 made PyRun_String behave
|
|
// similarly. Let's also do that for older versions, for consistency. This
|
|
// was missing from PyPy3.8 7.3.7.
|
|
if (!global.contains("__builtins__"))
|
|
global["__builtins__"] = module_::import(PYBIND11_BUILTINS_MODULE);
|
|
#else
|
|
(void) global;
|
|
#endif
|
|
}
|
|
|
|
PYBIND11_NAMESPACE_END(detail)
|
|
|
|
enum eval_mode {
|
|
/// Evaluate a string containing an isolated expression
|
|
eval_expr,
|
|
|
|
/// Evaluate a string containing a single statement. Returns \c none
|
|
eval_single_statement,
|
|
|
|
/// Evaluate a string containing a sequence of statement. Returns \c none
|
|
eval_statements
|
|
};
|
|
|
|
template <eval_mode mode = eval_expr>
|
|
object eval(const str &expr, object global = globals(), object local = object()) {
|
|
if (!local) {
|
|
local = global;
|
|
}
|
|
|
|
detail::ensure_builtins_in_globals(global);
|
|
|
|
/* PyRun_String does not accept a PyObject / encoding specifier,
|
|
this seems to be the only alternative */
|
|
std::string buffer = "# -*- coding: utf-8 -*-\n" + (std::string) expr;
|
|
|
|
int start = 0;
|
|
switch (mode) {
|
|
case eval_expr:
|
|
start = Py_eval_input;
|
|
break;
|
|
case eval_single_statement:
|
|
start = Py_single_input;
|
|
break;
|
|
case eval_statements:
|
|
start = Py_file_input;
|
|
break;
|
|
default:
|
|
pybind11_fail("invalid evaluation mode");
|
|
}
|
|
|
|
PyObject *result = PyRun_String(buffer.c_str(), start, global.ptr(), local.ptr());
|
|
if (!result) {
|
|
throw error_already_set();
|
|
}
|
|
return reinterpret_steal<object>(result);
|
|
}
|
|
|
|
template <eval_mode mode = eval_expr, size_t N>
|
|
object eval(const char (&s)[N], object global = globals(), object local = object()) {
|
|
/* Support raw string literals by removing common leading whitespace */
|
|
auto expr = (s[0] == '\n') ? str(module_::import("textwrap").attr("dedent")(s)) : str(s);
|
|
return eval<mode>(expr, global, local);
|
|
}
|
|
|
|
inline void exec(const str &expr, object global = globals(), object local = object()) {
|
|
eval<eval_statements>(expr, std::move(global), std::move(local));
|
|
}
|
|
|
|
template <size_t N>
|
|
void exec(const char (&s)[N], object global = globals(), object local = object()) {
|
|
eval<eval_statements>(s, global, local);
|
|
}
|
|
|
|
#if defined(PYPY_VERSION)
|
|
template <eval_mode mode = eval_statements>
|
|
object eval_file(str, object, object) {
|
|
pybind11_fail("eval_file not supported in PyPy3. Use eval");
|
|
}
|
|
template <eval_mode mode = eval_statements>
|
|
object eval_file(str, object) {
|
|
pybind11_fail("eval_file not supported in PyPy3. Use eval");
|
|
}
|
|
template <eval_mode mode = eval_statements>
|
|
object eval_file(str) {
|
|
pybind11_fail("eval_file not supported in PyPy3. Use eval");
|
|
}
|
|
#else
|
|
template <eval_mode mode = eval_statements>
|
|
object eval_file(str fname, object global = globals(), object local = object()) {
|
|
if (!local) {
|
|
local = global;
|
|
}
|
|
|
|
detail::ensure_builtins_in_globals(global);
|
|
|
|
int start = 0;
|
|
switch (mode) {
|
|
case eval_expr:
|
|
start = Py_eval_input;
|
|
break;
|
|
case eval_single_statement:
|
|
start = Py_single_input;
|
|
break;
|
|
case eval_statements:
|
|
start = Py_file_input;
|
|
break;
|
|
default:
|
|
pybind11_fail("invalid evaluation mode");
|
|
}
|
|
|
|
int closeFile = 1;
|
|
std::string fname_str = (std::string) fname;
|
|
FILE *f = _Py_fopen_obj(fname.ptr(), "r");
|
|
if (!f) {
|
|
PyErr_Clear();
|
|
pybind11_fail("File \"" + fname_str + "\" could not be opened!");
|
|
}
|
|
|
|
if (!global.contains("__file__")) {
|
|
global["__file__"] = std::move(fname);
|
|
}
|
|
|
|
PyObject *result
|
|
= PyRun_FileEx(f, fname_str.c_str(), start, global.ptr(), local.ptr(), closeFile);
|
|
|
|
if (!result) {
|
|
throw error_already_set();
|
|
}
|
|
return reinterpret_steal<object>(result);
|
|
}
|
|
#endif
|
|
|
|
PYBIND11_NAMESPACE_END(PYBIND11_NAMESPACE)
|