termmodule.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. #define PY_SSIZE_T_CLEAN
  2. #include <Python.h>
  3. #include <structmember.h>
  4. #include "term.h"
  5. typedef struct {
  6. PyObject_HEAD
  7. LSUP_Term * ob_struct;
  8. } TermObject;
  9. static PyModuleDef termmodule = {
  10. PyModuleDef_HEAD_INIT,
  11. .m_name = "term",
  12. .m_doc = "RDF term module.",
  13. .m_size = -1,
  14. };
  15. static int
  16. Term_init (TermObject *self, PyObject *args, PyObject *kwargs)
  17. {
  18. LSUP_term_type term_type;
  19. char *data = NULL, *datatype = NULL, *lang = NULL;
  20. static char *kwlist[] = {"data", "datatype", "lang", NULL};
  21. if (!PyArg_ParseTupleAndKeywords (
  22. args, kwargs, "bs|ss", kwlist,
  23. &term_type, &data, &datatype, &lang))
  24. return -1;
  25. self->ob_struct = LSUP_term_new (term_type, data, datatype, lang);
  26. if (!self->ob_struct) {
  27. PyErr_SetString (PyExc_ValueError, "Could not create term.");
  28. return -1;
  29. }
  30. return 0;
  31. }
  32. static void
  33. Term_dealloc (TermObject *self)
  34. {
  35. LSUP_term_free (self->ob_struct);
  36. Py_TYPE (self)->tp_free ((PyObject *) self);
  37. }
  38. static PyObject *
  39. Term_get_type (TermObject *self)
  40. {
  41. PyObject *type = PyLong_FromSize_t ((size_t)self->ob_struct->type);
  42. Py_INCREF (type);
  43. return type;
  44. }
  45. static PyObject *
  46. Term_get_data (TermObject *self)
  47. {
  48. PyObject *data = PyUnicode_FromString (self->ob_struct->data);
  49. Py_INCREF (data);
  50. return data;
  51. }
  52. static PyObject *
  53. Term_get_datatype (TermObject *self)
  54. {
  55. if (!self->ob_struct->datatype) Py_RETURN_NONE;
  56. PyObject *datatype = PyUnicode_FromString (self->ob_struct->datatype);
  57. Py_INCREF (datatype);
  58. return datatype;
  59. }
  60. static PyObject *
  61. Term_get_lang (TermObject *self)
  62. {
  63. if (
  64. !self->ob_struct->datatype || !self->ob_struct->lang ||
  65. strlen (self->ob_struct->lang) == 0)
  66. Py_RETURN_NONE;
  67. PyObject *lang = PyUnicode_FromString (self->ob_struct->lang);
  68. Py_INCREF (lang);
  69. return lang;
  70. }
  71. static PyGetSetDef Term_getsetters[] = {
  72. {"type", (getter) Term_get_type, NULL, "Term type.", NULL},
  73. {"data", (getter) Term_get_data, NULL, "Term data.", NULL},
  74. {
  75. "datatype", (getter) Term_get_datatype,
  76. NULL, "Literal term data type.", NULL
  77. },
  78. {
  79. "lang", (getter) Term_get_lang,
  80. NULL, "Literal term language tag.", NULL
  81. },
  82. {NULL}
  83. };
  84. static PyTypeObject TermType = {
  85. PyVarObject_HEAD_INIT(NULL, 0)
  86. .tp_name = "term.Term",
  87. .tp_doc = "RDF term",
  88. .tp_basicsize = sizeof(TermObject),
  89. .tp_itemsize = 0,
  90. .tp_flags = Py_TPFLAGS_DEFAULT,
  91. .tp_new = PyType_GenericNew,
  92. .tp_init = (initproc) Term_init,
  93. .tp_dealloc = (destructor) Term_dealloc,
  94. .tp_getset = Term_getsetters,
  95. };
  96. static PyObject *
  97. Term_richcmp (PyObject *obj1, PyObject *obj2, int op)
  98. {
  99. PyObject *result = NULL;
  100. if (
  101. ! PyObject_TypeCheck (obj1, &TermType) ||
  102. ! PyObject_TypeCheck (obj2, &TermType)
  103. ) return NULL;
  104. int c = 0;
  105. LSUP_Term *t1 = ((TermObject *) obj1)->ob_struct;
  106. LSUP_Term *t2 = ((TermObject *) obj2)->ob_struct;
  107. switch (op) {
  108. case Py_LT: result = Py_NotImplemented; break;
  109. case Py_LE: result = Py_NotImplemented; break;
  110. case Py_EQ: c = LSUP_term_equals (t1, t2); break;
  111. case Py_NE: c = ! LSUP_term_equals (t1, t2); break;
  112. case Py_GT: result = Py_NotImplemented; break;
  113. case Py_GE: result = Py_NotImplemented; break;
  114. }
  115. if (!result) result = c ? Py_True : Py_False;
  116. Py_INCREF(result);
  117. return result;
  118. }
  119. PyMODINIT_FUNC
  120. PyInit_term(void)
  121. {
  122. if (PyType_Ready (&TermType) < 0) return NULL;
  123. PyObject *m = PyModule_Create(&termmodule);
  124. if (m == NULL) return NULL;
  125. #define ENTRY(a, b) \
  126. if (PyModule_AddIntConstant (m, "TERM_" #a, b) < 0) return NULL;
  127. TTYPE_TABLE
  128. #undef ENTRY
  129. TermType.tp_richcompare = Term_richcmp;
  130. Py_INCREF(&TermType);
  131. if (PyModule_AddObject(m, "Term", (PyObject *) &TermType) < 0) {
  132. Py_DECREF(&TermType);
  133. Py_DECREF(m);
  134. return NULL;
  135. }
  136. return m;
  137. }