eval.h 4.62 KB
Newer Older
1
/*
NaDDu's avatar
NaDDu committed
2
    pybind11/eval.h: Support for evaluating Python expressions and statements
3
4
5
6
7
8
9
10
11
12
13
    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

Wenzel Jakob's avatar
Wenzel Jakob committed
14
#include "pybind11.h"
15

16
17
#include <utility>

18
PYBIND11_NAMESPACE_BEGIN(PYBIND11_NAMESPACE)
19
20
21
PYBIND11_NAMESPACE_BEGIN(detail)

inline void ensure_builtins_in_globals(object &global) {
22
#if defined(PYPY_VERSION) || PY_VERSION_HEX < 0x03080000
23
24
25
26
    // 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.
27
28
29
30
31
    if (!global.contains("__builtins__"))
        global["__builtins__"] = module_::import(PYBIND11_BUILTINS_MODULE);
#else
    (void) global;
#endif
32
33
34
}

PYBIND11_NAMESPACE_END(detail)
35
36
37
38
39
40
41
42
43
44
45
46
47

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>
48
object eval(const str &expr, object global = globals(), object local = object()) {
49
    if (!local) {
50
        local = global;
51
    }
52

53
54
    detail::ensure_builtins_in_globals(global);

Wenzel Jakob's avatar
Wenzel Jakob committed
55
56
57
58
    /* 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;

59
    int start = 0;
60
    switch (mode) {
61
62
63
64
65
66
67
68
69
70
71
        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");
72
73
    }

74
    PyObject *result = PyRun_String(buffer.c_str(), start, global.ptr(), local.ptr());
75
    if (!result) {
76
        throw error_already_set();
77
    }
78
    return reinterpret_steal<object>(result);
79
80
}

81
template <eval_mode mode = eval_expr, size_t N>
82
object eval(const char (&s)[N], object global = globals(), object local = object()) {
83
    /* Support raw string literals by removing common leading whitespace */
84
    auto expr = (s[0] == '\n') ? str(module_::import("textwrap").attr("dedent")(s)) : str(s);
85
    return eval<mode>(expr, std::move(global), std::move(local));
86
87
}

88
89
inline void exec(const str &expr, object global = globals(), object local = object()) {
    eval<eval_statements>(expr, std::move(global), std::move(local));
90
91
92
}

template <size_t N>
93
void exec(const char (&s)[N], object global = globals(), object local = object()) {
94
    eval<eval_statements>(s, std::move(global), std::move(local));
95
96
}

97
#if defined(PYPY_VERSION)
Isuru Fernando's avatar
Isuru Fernando committed
98
99
100
101
102
103
104
105
106
107
108
109
110
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
111
template <eval_mode mode = eval_statements>
112
object eval_file(str fname, object global = globals(), object local = object()) {
113
    if (!local) {
114
        local = global;
115
    }
116

117
118
    detail::ensure_builtins_in_globals(global);

119
    int start = 0;
120
    switch (mode) {
121
122
123
124
125
126
127
128
129
130
131
        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");
132
133
    }

Wenzel Jakob's avatar
Wenzel Jakob committed
134
135
136
137
138
139
140
    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!");
    }
141

142
143
144
145
    if (!global.contains("__file__")) {
        global["__file__"] = std::move(fname);
    }

146
147
    PyObject *result
        = PyRun_FileEx(f, fname_str.c_str(), start, global.ptr(), local.ptr(), closeFile);
Wenzel Jakob's avatar
Wenzel Jakob committed
148

149
    if (!result) {
150
        throw error_already_set();
151
    }
152
    return reinterpret_steal<object>(result);
153
}
Isuru Fernando's avatar
Isuru Fernando committed
154
#endif
155

156
PYBIND11_NAMESPACE_END(PYBIND11_NAMESPACE)