messagerenderer_python.cc 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. // Copyright (C) 2010 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. #include <dns/messagerenderer.h>
  15. // For each class, we need a struct, a helper functions (init, destroy,
  16. // and static wrappers around the methods we export), a list of methods,
  17. // and a type description
  18. using namespace isc::dns;
  19. // MessageRenderer
  20. // since we don't use *Buffer in the python version (but work with
  21. // the already existing bytearray type where we use these custom buffers
  22. // in c++, we need to keep track of one here.
  23. class s_MessageRenderer : public PyObject {
  24. public:
  25. OutputBuffer* outputbuffer;
  26. MessageRenderer* messagerenderer;
  27. };
  28. static int MessageRenderer_init(s_MessageRenderer* self);
  29. static void MessageRenderer_destroy(s_MessageRenderer* self);
  30. static PyObject* MessageRenderer_getData(s_MessageRenderer* self);
  31. static PyObject* MessageRenderer_getLength(s_MessageRenderer* self);
  32. static PyObject* MessageRenderer_isTruncated(s_MessageRenderer* self);
  33. static PyObject* MessageRenderer_getLengthLimit(s_MessageRenderer* self);
  34. static PyObject* MessageRenderer_getCompressMode(s_MessageRenderer* self);
  35. static PyObject* MessageRenderer_setTruncated(s_MessageRenderer* self);
  36. static PyObject* MessageRenderer_setLengthLimit(s_MessageRenderer* self, PyObject* args);
  37. static PyObject* MessageRenderer_setCompressMode(s_MessageRenderer* self, PyObject* args);
  38. static PyObject* MessageRenderer_clear(s_MessageRenderer* self);
  39. static PyMethodDef MessageRenderer_methods[] = {
  40. { "get_data", reinterpret_cast<PyCFunction>(MessageRenderer_getData), METH_NOARGS,
  41. "Returns the data as a bytes() object" },
  42. { "get_length", reinterpret_cast<PyCFunction>(MessageRenderer_getLength), METH_NOARGS,
  43. "Returns the length of the data" },
  44. { "is_truncated", reinterpret_cast<PyCFunction>(MessageRenderer_isTruncated), METH_NOARGS,
  45. "Returns True if the data is truncated" },
  46. { "get_length_limit", reinterpret_cast<PyCFunction>(MessageRenderer_getLengthLimit), METH_NOARGS,
  47. "Returns the length limit of the data" },
  48. { "get_compress_mode", reinterpret_cast<PyCFunction>(MessageRenderer_getCompressMode), METH_NOARGS,
  49. "Returns the current compression mode" },
  50. { "set_truncated", reinterpret_cast<PyCFunction>(MessageRenderer_setTruncated), METH_NOARGS,
  51. "Sets truncated to true" },
  52. { "set_length_limit", reinterpret_cast<PyCFunction>(MessageRenderer_setLengthLimit), METH_VARARGS,
  53. "Sets the length limit of the data to the given number" },
  54. { "set_compress_mode", reinterpret_cast<PyCFunction>(MessageRenderer_setCompressMode), METH_VARARGS,
  55. "Sets the compression mode of the MessageRenderer" },
  56. { "clear", reinterpret_cast<PyCFunction>(MessageRenderer_clear),
  57. METH_NOARGS,
  58. "Clear the internal buffer and other internal resources." },
  59. { NULL, NULL, 0, NULL }
  60. };
  61. static PyTypeObject messagerenderer_type = {
  62. PyVarObject_HEAD_INIT(NULL, 0)
  63. "pydnspp.MessageRenderer",
  64. sizeof(s_MessageRenderer), // tp_basicsize
  65. 0, // tp_itemsize
  66. (destructor)MessageRenderer_destroy,// tp_dealloc
  67. NULL, // tp_print
  68. NULL, // tp_getattr
  69. NULL, // tp_setattr
  70. NULL, // tp_reserved
  71. NULL, // tp_repr
  72. NULL, // tp_as_number
  73. NULL, // tp_as_sequence
  74. NULL, // tp_as_mapping
  75. NULL, // tp_hash
  76. NULL, // tp_call
  77. NULL, // tp_str
  78. NULL, // tp_getattro
  79. NULL, // tp_setattro
  80. NULL, // tp_as_buffer
  81. Py_TPFLAGS_DEFAULT, // tp_flags
  82. "The MessageRenderer class encapsulates implementation details "
  83. "of rendering a DNS message into a buffer in wire format. "
  84. "In effect, it's simply responsible for name compression at least in the "
  85. "current implementation. A MessageRenderer class object manages the "
  86. "positions of names rendered in a buffer and uses that information to render "
  87. "subsequent names with compression.",
  88. NULL, // tp_traverse
  89. NULL, // tp_clear
  90. NULL, // tp_richcompare
  91. 0, // tp_weaklistoffset
  92. NULL, // tp_iter
  93. NULL, // tp_iternext
  94. MessageRenderer_methods, // tp_methods
  95. NULL, // tp_members
  96. NULL, // tp_getset
  97. NULL, // tp_base
  98. NULL, // tp_dict
  99. NULL, // tp_descr_get
  100. NULL, // tp_descr_set
  101. 0, // tp_dictoffset
  102. (initproc)MessageRenderer_init, // tp_init
  103. NULL, // tp_alloc
  104. PyType_GenericNew, // tp_new
  105. NULL, // tp_free
  106. NULL, // tp_is_gc
  107. NULL, // tp_bases
  108. NULL, // tp_mro
  109. NULL, // tp_cache
  110. NULL, // tp_subclasses
  111. NULL, // tp_weaklist
  112. NULL, // tp_del
  113. 0 // tp_version_tag
  114. };
  115. static int
  116. MessageRenderer_init(s_MessageRenderer* self) {
  117. self->outputbuffer = new OutputBuffer(4096);
  118. self->messagerenderer = new MessageRenderer(*self->outputbuffer);
  119. return (0);
  120. }
  121. static void
  122. MessageRenderer_destroy(s_MessageRenderer* self) {
  123. delete self->messagerenderer;
  124. delete self->outputbuffer;
  125. self->messagerenderer = NULL;
  126. self->outputbuffer = NULL;
  127. Py_TYPE(self)->tp_free(self);
  128. }
  129. static PyObject*
  130. MessageRenderer_getData(s_MessageRenderer* self) {
  131. return (Py_BuildValue("y#",
  132. self->messagerenderer->getData(),
  133. self->messagerenderer->getLength()));
  134. }
  135. static PyObject*
  136. MessageRenderer_getLength(s_MessageRenderer* self) {
  137. return (Py_BuildValue("I", self->messagerenderer->getLength()));
  138. }
  139. static PyObject*
  140. MessageRenderer_isTruncated(s_MessageRenderer* self) {
  141. if (self->messagerenderer->isTruncated()) {
  142. Py_RETURN_TRUE;
  143. } else {
  144. Py_RETURN_FALSE;
  145. }
  146. }
  147. static PyObject*
  148. MessageRenderer_getLengthLimit(s_MessageRenderer* self) {
  149. return (Py_BuildValue("I", self->messagerenderer->getLengthLimit()));
  150. }
  151. static PyObject*
  152. MessageRenderer_getCompressMode(s_MessageRenderer* self) {
  153. return (Py_BuildValue("I", self->messagerenderer->getCompressMode()));
  154. }
  155. static PyObject*
  156. MessageRenderer_setTruncated(s_MessageRenderer* self) {
  157. self->messagerenderer->setTruncated();
  158. Py_RETURN_NONE;
  159. }
  160. static PyObject*
  161. MessageRenderer_setLengthLimit(s_MessageRenderer* self,
  162. PyObject* args)
  163. {
  164. long lengthlimit;
  165. if (!PyArg_ParseTuple(args, "l", &lengthlimit)) {
  166. PyErr_Clear();
  167. PyErr_SetString(PyExc_TypeError,
  168. "No valid type in set_length_limit argument");
  169. return (NULL);
  170. }
  171. if (lengthlimit < 0 || lengthlimit > 0xffff) {
  172. PyErr_SetString(PyExc_ValueError,
  173. "MessageRenderer length limit out of range");
  174. return (NULL);
  175. }
  176. self->messagerenderer->setLengthLimit(lengthlimit);
  177. Py_RETURN_NONE;
  178. }
  179. static PyObject*
  180. MessageRenderer_setCompressMode(s_MessageRenderer* self,
  181. PyObject* args)
  182. {
  183. int mode;
  184. if (!PyArg_ParseTuple(args, "i", &mode)) {
  185. return (NULL);
  186. }
  187. if (mode == MessageRenderer::CASE_INSENSITIVE) {
  188. self->messagerenderer->setCompressMode(MessageRenderer::CASE_INSENSITIVE);
  189. // If we return NULL it is seen as an error, so use this for
  190. // None returns, it also applies to CASE_SENSITIVE.
  191. Py_RETURN_NONE;
  192. } else if (mode == MessageRenderer::CASE_SENSITIVE) {
  193. self->messagerenderer->setCompressMode(MessageRenderer::CASE_SENSITIVE);
  194. Py_RETURN_NONE;
  195. } else {
  196. PyErr_SetString(PyExc_TypeError,
  197. "MessageRenderer compress mode must be MessageRenderer.CASE_INSENSITIVE"
  198. "or MessageRenderer.CASE_SENSITIVE");
  199. return (NULL);
  200. }
  201. }
  202. static PyObject*
  203. MessageRenderer_clear(s_MessageRenderer* self) {
  204. self->messagerenderer->clear();
  205. Py_RETURN_NONE;
  206. }
  207. // end of MessageRenderer
  208. // Module Initialization, all statics are initialized here
  209. bool
  210. initModulePart_MessageRenderer(PyObject* mod) {
  211. // Add the exceptions to the module
  212. // Add the enums to the module
  213. // Add the constants to the module
  214. // Add the classes to the module
  215. // We initialize the static description object with PyType_Ready(),
  216. // then add it to the module
  217. // NameComparisonResult
  218. if (PyType_Ready(&messagerenderer_type) < 0) {
  219. return (false);
  220. }
  221. Py_INCREF(&messagerenderer_type);
  222. // Class variables
  223. // These are added to the tp_dict of the type object
  224. addClassVariable(messagerenderer_type, "CASE_INSENSITIVE",
  225. Py_BuildValue("I", MessageRenderer::CASE_INSENSITIVE));
  226. addClassVariable(messagerenderer_type, "CASE_SENSITIVE",
  227. Py_BuildValue("I", MessageRenderer::CASE_SENSITIVE));
  228. PyModule_AddObject(mod, "MessageRenderer",
  229. reinterpret_cast<PyObject*>(&messagerenderer_type));
  230. return (true);
  231. }