| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- /*
- * This source file is part of libRocket, the HTML/CSS Interface Middleware
- *
- * For the latest information, see http://www.librocket.com
- *
- * Copyright (c) 2008-2010 CodePoint Ltd, Shift Technology Ltd
- *
- * 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.
- *
- */
- #include "precompiled.h"
- #include "DataSourceWrapper.h"
- #include "../../../Include/Rocket/Core/Log.h"
- #include "../../../Include/Rocket/Core/Python/Utilities.h"
- namespace Rocket {
- namespace Controls {
- namespace Python {
- // Returns the string representation of a python class name
- static Core::String GetPythonClassName(PyObject* object)
- {
- Core::String full_name;
- // Check if the object is a class; either a standard Python class or a Boost Python class metatype. If so, we can
- // query the name of the class object directly; otherwise, it is an object that we have to query the class type
- // from.
- bool is_class = PyClass_Check(object) ||
- object->ob_type == python::objects::class_metatype().get();
- PyObject* py_class = NULL;
- if (!is_class)
- {
- py_class = PyObject_GetAttrString(object, "__class__");
- object = py_class;
- if (!object)
- {
- PyErr_Clear();
- return full_name;
- }
- }
- PyObject* py_class_name = PyObject_GetAttrString(object, "__name__");
- if (py_class_name != NULL)
- {
- const char* class_name = PyString_AsString(py_class_name);
- PyObject* py_module_name = PyObject_GetAttrString(object, "__module__");
- const char* module_name = PyString_AsString(py_module_name);
- full_name.FormatString(128, "%s.%s", module_name, class_name);
- Py_DECREF(py_module_name);
- Py_DECREF(py_class_name);
- }
- else
- {
- PyErr_Clear();
- }
- if (py_class != NULL)
- Py_DECREF(py_class);
- return full_name;
- }
- DataSourceWrapper::DataSourceWrapper(PyObject* _self, const char* name) : DataSource(name)
- {
- self = _self;
- }
- DataSourceWrapper::~DataSourceWrapper()
- {
- }
- void DataSourceWrapper::InitialisePythonInterface()
- {
- void (DataSourceWrapper::*NotifyRowChange)(const Core::String&) = &DataSourceWrapper::NotifyRowChange;
- void (DataSourceWrapper::*NotifyRowChangeParams)(const Core::String&, int, int) = &DataSourceWrapper::NotifyRowChange;
- python::scope datasource_scope = python::class_<DataSource, DataSourceWrapper, boost::noncopyable>("DataSource", python::init< const char* >())
- .def("NotifyRowAdd", &DataSourceWrapper::NotifyRowAdd)
- .def("NotifyRowRemove", &DataSourceWrapper::NotifyRowRemove)
- .def("NotifyRowChange", NotifyRowChange)
- .def("NotifyRowChange", NotifyRowChangeParams)
- ;
- // Register the column names for use in python
- python::scope().attr("COLUMN_CHILD_SOURCE") = DataSource::CHILD_SOURCE;
- python::scope().attr("COLUMN_DEPTH") = DataSource::DEPTH;
- python::scope().attr("COLUMN_NUM_CHILDREN") = DataSource::NUM_CHILDREN;
- }
- void DataSourceWrapper::GetRow(Core::StringList& row, const Core::String& table, int row_index, const Core::StringList& columns)
- {
- PyObject* callable = PyObject_GetAttrString(self, "GetRow");
- if (!callable)
- {
- Core::String error_message(128, "Function \"GetRow\" not found on python data source %s.", GetPythonClassName(self).CString());
- Core::Log::Message(Core::Log::LT_WARNING, "%s", error_message.CString());
- PyErr_SetString(PyExc_RuntimeError, error_message.CString());
- python::throw_error_already_set();
- return;
- }
- python::tuple t = python::make_tuple(table.CString(), row_index, columns);
- PyObject* result = PyObject_CallObject(callable, t.ptr());
- Py_DECREF(callable);
- // If it's a list, then just get the entries out of it
- if (result && PyList_Check(result))
- {
- int num_entries = PyList_Size(result);
- for (int i = 0; i < num_entries; i++)
- {
- Core::String entry;
- PyObject* entry_object = PyList_GetItem(result, i);
- if (PyString_Check(entry_object))
- {
- entry = PyString_AS_STRING(entry_object);
- }
- else if (PyInt_Check(entry_object))
- {
- int entry_int = (int)PyInt_AS_LONG(entry_object);
- Core::TypeConverter< int, Core::String >::Convert(entry_int, entry);
- }
- else if (PyFloat_Check(entry_object))
- {
- float entry_float = (float)PyFloat_AS_DOUBLE(entry_object);
- Core::TypeConverter< float, Core::String >::Convert(entry_float, entry);
- }
- else
- {
- Core::String error_message(128, "Failed to convert row %d entry %d on data source %s.", row_index, i, GetPythonClassName(self).CString());
- Core::Log::Message(Core::Log::LT_WARNING, "%s", error_message.CString());
- PyErr_SetString(PyExc_RuntimeError, error_message.CString());
- python::throw_error_already_set();
- }
- row.push_back(entry);
- }
- }
- else
- {
- // Print the error and restore it to the caller
- PyObject *type, *value, *traceback;
- PyErr_Fetch(&type, &value, &traceback);
- Py_XINCREF(type);
- Py_XINCREF(value);
- Py_XINCREF(traceback);
- Core::String error_message(128, "Failed to get entries for table %s row %d from python data source %s.", table.CString(), row_index, GetPythonClassName(self).CString());
- Core::Log::Message(Core::Log::LT_WARNING, "%s", error_message.CString());
- if (type == NULL)
- PyErr_SetString(PyExc_RuntimeError, error_message.CString());
- else
- PyErr_Restore(type, value, traceback);
- python::throw_error_already_set();
- }
- if (result)
- Py_DECREF(result);
- }
- int DataSourceWrapper::GetNumRows(const Core::String& table)
- {
- int num_rows = 0;
- PyObject* callable = PyObject_GetAttrString(self, "GetNumRows");
- if (!callable)
- {
- Core::String error_message(128, "Function \"GetNumRows\" not found on python data source %s.", GetPythonClassName(self).CString());
- Core::Log::Message(Core::Log::LT_WARNING, "%s", error_message.CString());
- PyErr_SetString(PyExc_RuntimeError, error_message.CString());
- python::throw_error_already_set();
- return 0;
- }
- PyObject* result = PyObject_CallObject(callable, python::make_tuple(table.CString()).ptr());
- Py_DECREF(callable);
- if (result && PyInt_Check(result))
- {
- num_rows = PyInt_AsLong(result);
- }
- else
- {
- // Print the error and restore it to the caller
- PyObject *type, *value, *traceback;
- PyErr_Fetch(&type, &value, &traceback);
- Py_XINCREF(type);
- Py_XINCREF(value);
- Py_XINCREF(traceback);
- Core::String error_message(128, "Failed to get number of rows from python data source %s.", GetPythonClassName(self).CString());
- Core::Log::Message(Core::Log::LT_WARNING, "%s", error_message.CString());
- if (type == NULL)
- PyErr_SetString(PyExc_RuntimeError, error_message.CString());
- else
- PyErr_Restore(type, value, traceback);
- python::throw_error_already_set();
- }
-
- if (result)
- Py_DECREF(result);
- return num_rows;
- }
- Core::ScriptObject DataSourceWrapper::GetScriptObject() const
- {
- return self;
- }
- }
- }
- }
|