messagerenderer_python.cc 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. // Copyright (C) 2009 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // Permission to use, copy, modify, and/or distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  8. // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  9. // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  10. // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  11. // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  12. // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  13. // PERFORMANCE OF THIS SOFTWARE.
  14. // $Id$
  15. #include <dns/messagerenderer.h>
  16. // For each class, we need a struct, a helper functions (init, destroy,
  17. // and static wrappers around the methods we export), a list of methods,
  18. // and a type description
  19. using namespace isc::dns;
  20. // MessageRenderer
  21. // since we don't use *Buffer in the python version (but work with
  22. // the already existing bytearray type where we use these custom buffers
  23. // in c++, we need to keep track of one here.
  24. typedef struct {
  25. PyObject_HEAD
  26. OutputBuffer* outputbuffer;
  27. MessageRenderer* messagerenderer;
  28. } s_MessageRenderer;
  29. static int MessageRenderer_init(s_MessageRenderer* self);
  30. static void MessageRenderer_destroy(s_MessageRenderer* self);
  31. static PyObject* MessageRenderer_getData(s_MessageRenderer* self);
  32. static PyObject* MessageRenderer_getLength(s_MessageRenderer* self);
  33. static PyObject* MessageRenderer_isTruncated(s_MessageRenderer* self);
  34. static PyObject* MessageRenderer_getLengthLimit(s_MessageRenderer* self);
  35. // TODO: set/get compressmode
  36. static PyObject* MessageRenderer_setTruncated(s_MessageRenderer* self);
  37. static PyObject* MessageRenderer_setLengthLimit(s_MessageRenderer* self, PyObject* args);
  38. static PyMethodDef MessageRenderer_methods[] = {
  39. { "get_data", (PyCFunction)MessageRenderer_getData, METH_NOARGS, "Return the data" },
  40. { "get_length", (PyCFunction)MessageRenderer_getLength, METH_NOARGS, "Return the length of the data" },
  41. { "is_truncated", (PyCFunction)MessageRenderer_isTruncated, METH_NOARGS, "Returns True if the data is truncated" },
  42. { "get_length_limit", (PyCFunction)MessageRenderer_getLengthLimit, METH_NOARGS, "Return the length limit of the data" },
  43. { "set_truncated", (PyCFunction)MessageRenderer_setTruncated, METH_NOARGS, "Set truncated to true" },
  44. { "set_length_limit", (PyCFunction)MessageRenderer_setLengthLimit, METH_VARARGS, "Set the length limit of the data" },
  45. { NULL, NULL, 0, NULL }
  46. };
  47. static PyTypeObject messagerenderer_type = {
  48. PyVarObject_HEAD_INIT(NULL, 0)
  49. "libdns_python.MessageRenderer",
  50. sizeof(s_MessageRenderer), /* tp_basicsize */
  51. 0, /* tp_itemsize */
  52. (destructor)MessageRenderer_destroy, /* tp_dealloc */
  53. NULL, /* tp_print */
  54. NULL, /* tp_getattr */
  55. NULL, /* tp_setattr */
  56. NULL, /* tp_reserved */
  57. NULL, /* tp_repr */
  58. NULL, /* tp_as_number */
  59. NULL, /* tp_as_sequence */
  60. NULL, /* tp_as_mapping */
  61. NULL, /* tp_hash */
  62. NULL, /* tp_call */
  63. NULL, /* tp_str */
  64. NULL, /* tp_getattro */
  65. NULL, /* tp_setattro */
  66. NULL, /* tp_as_buffer */
  67. Py_TPFLAGS_DEFAULT, /* tp_flags */
  68. "C++ MessageRenderer Object", /* tp_doc */
  69. NULL, /* tp_traverse */
  70. NULL, /* tp_clear */
  71. NULL, /* tp_richcompare */
  72. 0, /* tp_weaklistoffset */
  73. NULL, /* tp_iter */
  74. NULL, /* tp_iternext */
  75. MessageRenderer_methods, /* tp_methods */
  76. NULL, /* tp_members */
  77. NULL, /* tp_getset */
  78. NULL, /* tp_base */
  79. NULL, /* tp_dict */
  80. NULL, /* tp_descr_get */
  81. NULL, /* tp_descr_set */
  82. 0, /* tp_dictoffset */
  83. (initproc)MessageRenderer_init, /* tp_init */
  84. NULL, /* tp_alloc */
  85. PyType_GenericNew, /* tp_new */
  86. NULL, /* tp_free */
  87. NULL, /* tp_is_gc */
  88. NULL, /* tp_bases */
  89. NULL, /* tp_mro */
  90. NULL, /* tp_cache */
  91. NULL, /* tp_subclasses */
  92. NULL, /* tp_weaklist */
  93. // Note: not sure if the following are correct. Added them just to
  94. // make the compiler happy.
  95. NULL, /* tp_del */
  96. 0 /* tp_version_tag */
  97. };
  98. static int
  99. MessageRenderer_init(s_MessageRenderer* self)
  100. {
  101. self->outputbuffer = new OutputBuffer(4096);
  102. self->messagerenderer = new MessageRenderer(*self->outputbuffer);
  103. return 0;
  104. }
  105. static void
  106. MessageRenderer_destroy(s_MessageRenderer* self)
  107. {
  108. delete self->messagerenderer;
  109. delete self->outputbuffer;
  110. self->messagerenderer = NULL;
  111. Py_TYPE(self)->tp_free(self);
  112. }
  113. static PyObject*
  114. MessageRenderer_getData(s_MessageRenderer* self)
  115. {
  116. return Py_BuildValue("y#", self->messagerenderer->getData(), self->messagerenderer->getLength());
  117. }
  118. static PyObject*
  119. MessageRenderer_getLength(s_MessageRenderer* self)
  120. {
  121. return Py_BuildValue("I", self->messagerenderer->getLength());
  122. }
  123. static PyObject*
  124. MessageRenderer_isTruncated(s_MessageRenderer* self)
  125. {
  126. if (self->messagerenderer->isTruncated()) {
  127. Py_RETURN_TRUE;
  128. } else {
  129. Py_RETURN_FALSE;
  130. }
  131. }
  132. static PyObject*
  133. MessageRenderer_getLengthLimit(s_MessageRenderer* self)
  134. {
  135. return Py_BuildValue("I", self->messagerenderer->getLengthLimit());
  136. }
  137. static PyObject*
  138. MessageRenderer_setTruncated(s_MessageRenderer* self)
  139. {
  140. self->messagerenderer->setTruncated();
  141. Py_RETURN_NONE;
  142. }
  143. static PyObject*
  144. MessageRenderer_setLengthLimit(s_MessageRenderer* self, PyObject* args)
  145. {
  146. size_t lengthlimit;
  147. if (!PyArg_ParseTuple(args, "I", &lengthlimit)) {
  148. return NULL;
  149. }
  150. self->messagerenderer->setLengthLimit(lengthlimit);
  151. Py_RETURN_NONE;
  152. }
  153. // end of MessageRenderer
  154. // Module Initialization, all statics are initialized here
  155. bool
  156. initModulePart_MessageRenderer(PyObject* mod)
  157. {
  158. // Add the exceptions to the module
  159. // Add the enums to the module
  160. // Add the constants to the module
  161. // Add the classes to the module
  162. // We initialize the static description object with PyType_Ready(),
  163. // then add it to the module
  164. // NameComparisonResult
  165. if (PyType_Ready(&messagerenderer_type) < 0) {
  166. return false;
  167. }
  168. Py_INCREF(&messagerenderer_type);
  169. PyModule_AddObject(mod, "MessageRenderer",
  170. (PyObject*) &messagerenderer_type);
  171. return true;
  172. }