#ifndef PYPARAM_H #define PYPARAM_H #include #include #include "pyutils.h" #include "types/param.h" #include "types/paramtype.h" #include "loggingcategories.h" #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Winvalid-offsetof" #pragma GCC diagnostic ignored "-Wwrite-strings" #pragma GCC diagnostic ignored "-Wmissing-field-initializers" typedef struct _pyparam { PyObject_HEAD PyObject* pyParamTypeId = nullptr; PyObject* pyValue = nullptr; } PyParam; static PyMethodDef PyParam_methods[] = { {nullptr, nullptr, 0, nullptr} // sentinel }; static PyMemberDef PyParam_members[] = { {"paramTypeId", T_OBJECT_EX, offsetof(PyParam, pyParamTypeId), 0, "Param type ID"}, {"value", T_OBJECT_EX, offsetof(PyParam, pyValue), 0, "Param value"}, {nullptr, 0, 0, 0, nullptr} /* Sentinel */ }; static int PyParam_init(PyParam *self, PyObject *args, PyObject *kwds) { qCDebug(dcPythonIntegrations()) << "+++ PyParam"; static char *kwlist[] = {"paramTypeId", "value", nullptr}; PyObject *paramTypeId = nullptr, *value = nullptr; if (!PyArg_ParseTupleAndKeywords(args, kwds, "|OO", kwlist, ¶mTypeId, &value)) return -1; if (paramTypeId) { Py_INCREF(paramTypeId); self->pyParamTypeId = paramTypeId; } if (value) { Py_INCREF(value); self->pyValue = value; } return 0; } static void PyParam_dealloc(PyParam * self) { qCDebug(dcPythonIntegrations()) << "--- PyParam"; Py_XDECREF(self->pyParamTypeId); Py_XDECREF(self->pyValue); Py_TYPE(self)->tp_free(self); } static PyTypeObject PyParamType = { PyVarObject_HEAD_INIT(NULL, 0) "nymea.Param", /* tp_name */ sizeof(PyParam), /* tp_basicsize */ 0, /* tp_itemsize */ (destructor)PyParam_dealloc,/* tp_dealloc */ }; static PyParam* PyParam_fromParam(const Param ¶m) { PyObject *pyParamValue = QVariantToPyObject(param.value()); PyObject *args = Py_BuildValue("(sO)", param.paramTypeId().toString().toUtf8().data(), pyParamValue); PyParam *pyParam = (PyParam*)PyObject_CallObject((PyObject*)&PyParamType, args); Py_DECREF(pyParamValue); Py_DECREF(args); return pyParam; } static Param PyParam_ToParam(PyParam *pyParam) { ParamTypeId paramTypeId = ParamTypeId(PyUnicode_AsUTF8AndSize(pyParam->pyParamTypeId, nullptr)); QVariant value = PyObjectToQVariant(pyParam->pyValue); return Param(paramTypeId, value); } static PyObject* PyParams_FromParamList(const ParamList ¶ms) { PyObject* result = PyTuple_New(params.length()); for (int i = 0; i < params.count(); i++) { PyParam *pyParam = PyParam_fromParam(params.at(i)); PyTuple_SetItem(result, i, (PyObject*)pyParam); } return result; } static ParamList PyParams_ToParamList(PyObject *pyParams) { ParamList params; if (pyParams == nullptr) { return params; } PyObject *iter = PyObject_GetIter(pyParams); while (iter) { PyObject *next = PyIter_Next(iter); if (!next) { break; } if (next->ob_type != &PyParamType) { qCWarning(dcThingManager()) << "Invalid parameter passed in param list"; continue; } PyParam *pyParam = reinterpret_cast(next); params.append(PyParam_ToParam(pyParam)); Py_DECREF(next); } Py_DECREF(iter); return params; } static void registerParamType(PyObject *module) { PyParamType.tp_new = PyType_GenericNew; PyParamType.tp_init = reinterpret_cast(PyParam_init); PyParamType.tp_flags = Py_TPFLAGS_DEFAULT; PyParamType.tp_methods = PyParam_methods; PyParamType.tp_members = PyParam_members; PyParamType.tp_doc = "Param class"; if (PyType_Ready(&PyParamType) < 0) { return; } PyModule_AddObject(module, "Param", reinterpret_cast(&PyParamType)); } #pragma GCC diagnostic pop #endif // PYPARAM_H