summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2009-01-23 01:53:19 -0400
committerJohn Wiegley <johnw@newartisans.com>2009-01-23 01:53:19 -0400
commit617dd0c2613531f559a0206afe9011df73ea1856 (patch)
tree7ade20f5c569e230a31847331faf44a18c73895d
parent8d4542d199a325f344f96aff94c8ca008f0ab98e (diff)
downloadfork-ledger-617dd0c2613531f559a0206afe9011df73ea1856.tar.gz
fork-ledger-617dd0c2613531f559a0206afe9011df73ea1856.tar.bz2
fork-ledger-617dd0c2613531f559a0206afe9011df73ea1856.zip
When Python support is present, use a session_t object which is derived from
python_interpreter_t, so that Python can provide value expression definitions.
-rw-r--r--Makefile.am6
-rw-r--r--python/pyinterp.cc91
-rw-r--r--python/pyinterp.h20
-rw-r--r--src/main.cc7
4 files changed, 95 insertions, 29 deletions
diff --git a/Makefile.am b/Makefile.am
index d2ac4b67..3384170d 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -160,8 +160,14 @@ endif
bin_PROGRAMS = ledger
ledger_CPPFLAGS = $(libledger_la_CPPFLAGS)
+if HAVE_BOOST_PYTHON
+ledger_CPPFLAGS += -DHAVE_BOOST_PYTHON=1 -I$(srcdir)/python
+endif
ledger_SOURCES = src/main.cc
ledger_LDADD = $(LIBOBJS) libamounts.la libledger.la
+if HAVE_BOOST_PYTHON
+ledger_LDADD += libpyledger.la
+endif
ledger_LDFLAGS =
info_TEXINFOS = doc/ledger.texi
diff --git a/python/pyinterp.cc b/python/pyinterp.cc
index f5c81142..36dd3e35 100644
--- a/python/pyinterp.cc
+++ b/python/pyinterp.cc
@@ -79,20 +79,28 @@ struct python_run
python_run(python_interpreter_t * intepreter,
const string& str, int input_mode)
: result(handle<>(borrowed(PyRun_String(str.c_str(), input_mode,
- intepreter->nspace.ptr(),
- intepreter->nspace.ptr())))) {}
+ intepreter->main_nspace.ptr(),
+ intepreter->main_nspace.ptr())))) {}
operator object() {
return result;
}
};
-python_interpreter_t::python_interpreter_t()
- : scope_t(), mmodule(borrowed(PyImport_AddModule("__main__"))),
- nspace(handle<>(borrowed(PyModule_GetDict(mmodule.get()))))
+python_interpreter_t::python_interpreter_t() : session_t(), main_nspace()
{
- TRACE_CTOR(python_interpreter_t, "expr_t::scope_t&");
+ TRACE_CTOR(python_interpreter_t, "");
+ DEBUG("python.interp", "Initializing Python");
Py_Initialize();
+
+ object main_module = boost::python::import("__main__");
+ if (! main_module)
+ throw_(std::logic_error, "Python failed to initialize");
+
+ main_nspace = main_module.attr("__dict__");
+ if (! main_nspace)
+ throw_(std::logic_error, "Python failed to initialize");
+
boost::python::detail::init_module("ledger", &initialize_for_python);
}
@@ -101,20 +109,19 @@ object python_interpreter_t::import(const string& str)
assert(Py_IsInitialized());
try {
- PyObject * mod = PyImport_Import(PyString_FromString(str.c_str()));
+ DEBUG("python.interp", "Importing Python module: " << str);
+
+ object mod = boost::python::import(str.c_str());
if (! mod)
throw_(std::logic_error, "Failed to import Python module " << str);
+
+ // Import all top-level entries directly into the main namespace
+ object nspace = mod.attr("__dict__");
- object newmod(handle<>(borrowed(mod)));
+ dict main_nspace_dict = extract<dict>(main_nspace);
+ main_nspace_dict.update(nspace);
-#if 1
- // Import all top-level entries directly into the main namespace
- dict m_nspace(handle<>(borrowed(PyModule_GetDict(mod))));
- nspace.update(m_nspace);
-#else
- nspace[string(PyModule_GetName(mod))] = newmod;
-#endif
- return newmod;
+ return mod;
}
catch (const error_already_set&) {
PyErr_Print();
@@ -125,7 +132,7 @@ object python_interpreter_t::import(const string& str)
object python_interpreter_t::eval(std::istream& in, py_eval_mode_t mode)
{
- bool first = true;
+ bool first = true;
string buffer;
buffer.reserve(4096);
@@ -149,6 +156,7 @@ object python_interpreter_t::eval(std::istream& in, py_eval_mode_t mode)
case PY_EVAL_MULTI: input_mode = Py_file_input; break;
}
assert(Py_IsInitialized());
+
return python_run(this, buffer, input_mode);
}
catch (const error_already_set&) {
@@ -177,11 +185,58 @@ object python_interpreter_t::eval(const string& str, py_eval_mode_t mode)
return object();
}
+expr_t::ptr_op_t python_interpreter_t::lookup(const string& name)
+{
+ if (expr_t::ptr_op_t op = session_t::lookup(name))
+ return op;
+
+ const char * p = name.c_str();
+ switch (*p) {
+ case 'o':
+ if (std::strncmp(p, "opt_", 4) == 0) {
+ p = p + 4;
+ switch (*p) {
+ case 'i':
+ if (std::strcmp(p, "import_") == 0)
+ return MAKE_FUNCTOR(python_interpreter_t::option_import_);
+ else if (std::strcmp(p, "import") == 0)
+ return expr_t::ptr_op_t();
+ break;
+ }
+ }
+ break;
+ }
+
+ DEBUG("python.interp", "Python eval: " << name);
+
+ try {
+ if (boost::python::object obj = eval(name))
+ return WRAP_FUNCTOR(functor_t(name, obj));
+ }
+ catch (...) {}
+
+ return expr_t::ptr_op_t();
+}
+
value_t python_interpreter_t::functor_t::operator()(call_scope_t& args)
{
try {
if (! PyCallable_Check(func.ptr())) {
- return extract<value_t>(func.ptr());
+ if (PyBool_Check(func.ptr()))
+ return extract<bool>(func)();
+ else if (PyInt_Check(func.ptr()))
+ return long(extract<int>(func)());
+ else if (PyString_Check(func.ptr()))
+ return string_value(extract<string>(func)());
+
+ extract<date_t> d(func);
+ if (d.check())
+ return value_t(d());
+ extract<datetime_t> dt(func);
+ if (dt.check())
+ return value_t(dt());
+
+ return extract<value_t>(func);
} else {
if (args.size() > 0) {
list arglist;
diff --git a/python/pyinterp.h b/python/pyinterp.h
index 5930cf8e..60aa9c20 100644
--- a/python/pyinterp.h
+++ b/python/pyinterp.h
@@ -33,22 +33,19 @@
#define _PYINTERP_H
#include "scope.h"
+#include "session.h"
#include <boost/python.hpp>
#include <Python.h>
namespace ledger {
-class python_interpreter_t : public noncopyable, public scope_t
+class python_interpreter_t : public session_t
{
- boost::python::handle<> mmodule;
-
- python_interpreter_t();
-
public:
- boost::python::dict nspace;
+ boost::python::object main_nspace;
- python_interpreter_t(scope_t& parent);
+ python_interpreter_t();
virtual ~python_interpreter_t() {
TRACE_DTOR(python_interpreter_t);
@@ -91,10 +88,11 @@ public:
virtual value_t operator()(call_scope_t& args);
};
- virtual expr_t::ptr_op_t lookup(const string& name) {
- if (boost::python::object func = eval(name))
- return WRAP_FUNCTOR(functor_t(name, func));
- return expr_t::ptr_op_t();
+ virtual expr_t::ptr_op_t lookup(const string& name);
+
+ value_t option_import_(call_scope_t& args) {
+ import(args[0].to_string());
+ return true;
}
class lambda_t : public functor_t {
diff --git a/src/main.cc b/src/main.cc
index a3fa5c18..5f8e7aeb 100644
--- a/src/main.cc
+++ b/src/main.cc
@@ -33,6 +33,9 @@
#include "report.h"
#include "option.h"
#include "help.h"
+#if defined(HAVE_BOOST_PYTHON)
+#include "pyinterp.h"
+#endif
#include "textual.h"
#include "qif.h"
@@ -455,7 +458,11 @@ int main(int argc, char * argv[], char * envp[])
INFO("Ledger starting");
+#if defined(HAVE_BOOST_PYTHON)
+ std::auto_ptr<ledger::session_t> session(new ledger::python_interpreter_t);
+#else
std::auto_ptr<ledger::session_t> session(new ledger::session_t);
+#endif
ledger::set_session_context(session.get());