miniupnpcmodule.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696
  1. /* $Id: miniupnpcmodule.c,v 1.29 2015/10/26 17:01:30 nanard Exp $*/
  2. /* Project : miniupnp
  3. * Author : Thomas BERNARD
  4. * website : http://miniupnp.tuxfamily.org/
  5. * copyright (c) 2007-2014 Thomas Bernard
  6. * This software is subjet to the conditions detailed in the
  7. * provided LICENCE file. */
  8. #include <Python.h>
  9. #define MINIUPNP_STATICLIB
  10. #include "structmember.h"
  11. #include "miniupnpc.h"
  12. #include "upnpcommands.h"
  13. #include "upnperrors.h"
  14. /* for compatibility with Python < 2.4 */
  15. #ifndef Py_RETURN_NONE
  16. #define Py_RETURN_NONE return Py_INCREF(Py_None), Py_None
  17. #endif
  18. #ifndef Py_RETURN_TRUE
  19. #define Py_RETURN_TRUE return Py_INCREF(Py_True), Py_True
  20. #endif
  21. #ifndef Py_RETURN_FALSE
  22. #define Py_RETURN_FALSE return Py_INCREF(Py_False), Py_False
  23. #endif
  24. /* for compatibility with Python < 3.0 */
  25. #ifndef PyVarObject_HEAD_INIT
  26. #define PyVarObject_HEAD_INIT(type, size) \
  27. PyObject_HEAD_INIT(type) size,
  28. #endif
  29. #ifndef Py_TYPE
  30. #define Py_TYPE(ob) (((PyObject*)(ob))->ob_type)
  31. #endif
  32. typedef struct {
  33. PyObject_HEAD
  34. /* Type-specific fields go here. */
  35. struct UPNPDev * devlist;
  36. struct UPNPUrls urls;
  37. struct IGDdatas data;
  38. unsigned int discoverdelay; /* value passed to upnpDiscover() */
  39. unsigned int localport; /* value passed to upnpDiscover() */
  40. char lanaddr[40]; /* our ip address on the LAN */
  41. char * multicastif;
  42. char * minissdpdsocket;
  43. } UPnPObject;
  44. static PyMemberDef UPnP_members[] = {
  45. {"lanaddr", T_STRING_INPLACE, offsetof(UPnPObject, lanaddr),
  46. READONLY, "ip address on the LAN"
  47. },
  48. {"discoverdelay", T_UINT, offsetof(UPnPObject, discoverdelay),
  49. 0/*READWRITE*/, "value in ms used to wait for SSDP responses"
  50. },
  51. {"localport", T_UINT, offsetof(UPnPObject, localport),
  52. 0/*READWRITE*/,
  53. "If localport is set to UPNP_LOCAL_PORT_SAME(1) "
  54. "SSDP packets will be sent from the source port "
  55. "1900 (same as destination port), if set to "
  56. "UPNP_LOCAL_PORT_ANY(0) system assign a source "
  57. "port, any other value will be attempted as the "
  58. "source port"
  59. },
  60. /* T_STRING is allways readonly :( */
  61. {"multicastif", T_STRING, offsetof(UPnPObject, multicastif),
  62. 0, "IP of the network interface to be used for multicast operations"
  63. },
  64. {"minissdpdsocket", T_STRING, offsetof(UPnPObject, minissdpdsocket),
  65. 0, "path of the MiniSSDPd unix socket"
  66. },
  67. {NULL}
  68. };
  69. static int UPnP_init(UPnPObject *self, PyObject *args, PyObject *kwds)
  70. {
  71. char* multicastif = NULL;
  72. char* minissdpdsocket = NULL;
  73. static char *kwlist[] = {
  74. "multicastif", "minissdpdsocket", "discoverdelay",
  75. "localport", NULL
  76. };
  77. if(!PyArg_ParseTupleAndKeywords(args, kwds, "|zzII", kwlist,
  78. &multicastif,
  79. &minissdpdsocket,
  80. &self->discoverdelay,
  81. &self->localport))
  82. return -1;
  83. if(self->localport>1 &&
  84. (self->localport>65534||self->localport<1024)) {
  85. PyErr_SetString(PyExc_Exception, "Invalid localport value");
  86. return -1;
  87. }
  88. if(multicastif)
  89. self->multicastif = strdup(multicastif);
  90. if(minissdpdsocket)
  91. self->minissdpdsocket = strdup(minissdpdsocket);
  92. return 0;
  93. }
  94. static void
  95. UPnPObject_dealloc(UPnPObject *self)
  96. {
  97. freeUPNPDevlist(self->devlist);
  98. FreeUPNPUrls(&self->urls);
  99. free(self->multicastif);
  100. free(self->minissdpdsocket);
  101. Py_TYPE(self)->tp_free((PyObject*)self);
  102. }
  103. static PyObject *
  104. UPnP_discover(UPnPObject *self)
  105. {
  106. struct UPNPDev * dev;
  107. int i;
  108. PyObject *res = NULL;
  109. if(self->devlist)
  110. {
  111. freeUPNPDevlist(self->devlist);
  112. self->devlist = 0;
  113. }
  114. Py_BEGIN_ALLOW_THREADS
  115. self->devlist = upnpDiscover((int)self->discoverdelay/*timeout in ms*/,
  116. self->multicastif,
  117. self->minissdpdsocket,
  118. (int)self->localport,
  119. 0/*ip v6*/,
  120. 2/* TTL */,
  121. 0/*error */);
  122. Py_END_ALLOW_THREADS
  123. /* Py_RETURN_NONE ??? */
  124. for(dev = self->devlist, i = 0; dev; dev = dev->pNext)
  125. i++;
  126. res = Py_BuildValue("i", i);
  127. return res;
  128. }
  129. static PyObject *
  130. UPnP_selectigd(UPnPObject *self)
  131. {
  132. int r;
  133. Py_BEGIN_ALLOW_THREADS
  134. r = UPNP_GetValidIGD(self->devlist, &self->urls, &self->data,
  135. self->lanaddr, sizeof(self->lanaddr));
  136. Py_END_ALLOW_THREADS
  137. if(r)
  138. {
  139. return Py_BuildValue("s", self->urls.controlURL);
  140. }
  141. else
  142. {
  143. /* TODO: have our own exception type ! */
  144. PyErr_SetString(PyExc_Exception, "No UPnP device discovered");
  145. return NULL;
  146. }
  147. }
  148. static PyObject *
  149. UPnP_totalbytesent(UPnPObject *self)
  150. {
  151. UNSIGNED_INTEGER i;
  152. Py_BEGIN_ALLOW_THREADS
  153. i = UPNP_GetTotalBytesSent(self->urls.controlURL_CIF,
  154. self->data.CIF.servicetype);
  155. Py_END_ALLOW_THREADS
  156. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  157. return Py_BuildValue("I", i);
  158. #else
  159. return Py_BuildValue("i", (int)i);
  160. #endif
  161. }
  162. static PyObject *
  163. UPnP_totalbytereceived(UPnPObject *self)
  164. {
  165. UNSIGNED_INTEGER i;
  166. Py_BEGIN_ALLOW_THREADS
  167. i = UPNP_GetTotalBytesReceived(self->urls.controlURL_CIF,
  168. self->data.CIF.servicetype);
  169. Py_END_ALLOW_THREADS
  170. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  171. return Py_BuildValue("I", i);
  172. #else
  173. return Py_BuildValue("i", (int)i);
  174. #endif
  175. }
  176. static PyObject *
  177. UPnP_totalpacketsent(UPnPObject *self)
  178. {
  179. UNSIGNED_INTEGER i;
  180. Py_BEGIN_ALLOW_THREADS
  181. i = UPNP_GetTotalPacketsSent(self->urls.controlURL_CIF,
  182. self->data.CIF.servicetype);
  183. Py_END_ALLOW_THREADS
  184. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  185. return Py_BuildValue("I", i);
  186. #else
  187. return Py_BuildValue("i", (int)i);
  188. #endif
  189. }
  190. static PyObject *
  191. UPnP_totalpacketreceived(UPnPObject *self)
  192. {
  193. UNSIGNED_INTEGER i;
  194. Py_BEGIN_ALLOW_THREADS
  195. i = UPNP_GetTotalPacketsReceived(self->urls.controlURL_CIF,
  196. self->data.CIF.servicetype);
  197. Py_END_ALLOW_THREADS
  198. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  199. return Py_BuildValue("I", i);
  200. #else
  201. return Py_BuildValue("i", (int)i);
  202. #endif
  203. }
  204. static PyObject *
  205. UPnP_statusinfo(UPnPObject *self)
  206. {
  207. char status[64];
  208. char lastconnerror[64];
  209. unsigned int uptime = 0;
  210. int r;
  211. status[0] = '\0';
  212. lastconnerror[0] = '\0';
  213. Py_BEGIN_ALLOW_THREADS
  214. r = UPNP_GetStatusInfo(self->urls.controlURL, self->data.first.servicetype,
  215. status, &uptime, lastconnerror);
  216. Py_END_ALLOW_THREADS
  217. if(r==UPNPCOMMAND_SUCCESS) {
  218. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  219. return Py_BuildValue("(s,I,s)", status, uptime, lastconnerror);
  220. #else
  221. return Py_BuildValue("(s,i,s)", status, (int)uptime, lastconnerror);
  222. #endif
  223. } else {
  224. /* TODO: have our own exception type ! */
  225. PyErr_SetString(PyExc_Exception, strupnperror(r));
  226. return NULL;
  227. }
  228. }
  229. static PyObject *
  230. UPnP_connectiontype(UPnPObject *self)
  231. {
  232. char connectionType[64];
  233. int r;
  234. connectionType[0] = '\0';
  235. Py_BEGIN_ALLOW_THREADS
  236. r = UPNP_GetConnectionTypeInfo(self->urls.controlURL,
  237. self->data.first.servicetype,
  238. connectionType);
  239. Py_END_ALLOW_THREADS
  240. if(r==UPNPCOMMAND_SUCCESS) {
  241. return Py_BuildValue("s", connectionType);
  242. } else {
  243. /* TODO: have our own exception type ! */
  244. PyErr_SetString(PyExc_Exception, strupnperror(r));
  245. return NULL;
  246. }
  247. }
  248. static PyObject *
  249. UPnP_externalipaddress(UPnPObject *self)
  250. {
  251. char externalIPAddress[40];
  252. int r;
  253. externalIPAddress[0] = '\0';
  254. Py_BEGIN_ALLOW_THREADS
  255. r = UPNP_GetExternalIPAddress(self->urls.controlURL,
  256. self->data.first.servicetype,
  257. externalIPAddress);
  258. Py_END_ALLOW_THREADS
  259. if(r==UPNPCOMMAND_SUCCESS) {
  260. return Py_BuildValue("s", externalIPAddress);
  261. } else {
  262. /* TODO: have our own exception type ! */
  263. PyErr_SetString(PyExc_Exception, strupnperror(r));
  264. return NULL;
  265. }
  266. }
  267. /* AddPortMapping(externalPort, protocol, internalHost, internalPort, desc,
  268. * remoteHost)
  269. * protocol is 'UDP' or 'TCP' */
  270. static PyObject *
  271. UPnP_addportmapping(UPnPObject *self, PyObject *args)
  272. {
  273. char extPort[6];
  274. unsigned short ePort;
  275. char inPort[6];
  276. unsigned short iPort;
  277. const char * proto;
  278. const char * host;
  279. const char * desc;
  280. const char * remoteHost;
  281. const char * leaseDuration = "0";
  282. int r;
  283. if (!PyArg_ParseTuple(args, "HssHss", &ePort, &proto,
  284. &host, &iPort, &desc, &remoteHost))
  285. return NULL;
  286. Py_BEGIN_ALLOW_THREADS
  287. sprintf(extPort, "%hu", ePort);
  288. sprintf(inPort, "%hu", iPort);
  289. r = UPNP_AddPortMapping(self->urls.controlURL, self->data.first.servicetype,
  290. extPort, inPort, host, desc, proto,
  291. remoteHost, leaseDuration);
  292. Py_END_ALLOW_THREADS
  293. if(r==UPNPCOMMAND_SUCCESS)
  294. {
  295. Py_RETURN_TRUE;
  296. }
  297. else
  298. {
  299. // TODO: RAISE an Exception. See upnpcommands.h for errors codes.
  300. // upnperrors.c
  301. //Py_RETURN_FALSE;
  302. /* TODO: have our own exception type ! */
  303. PyErr_SetString(PyExc_Exception, strupnperror(r));
  304. return NULL;
  305. }
  306. }
  307. /* AddAnyPortMapping(externalPort, protocol, internalHost, internalPort, desc,
  308. * remoteHost)
  309. * protocol is 'UDP' or 'TCP' */
  310. static PyObject *
  311. UPnP_addanyportmapping(UPnPObject *self, PyObject *args)
  312. {
  313. char extPort[6];
  314. unsigned short ePort;
  315. char inPort[6];
  316. unsigned short iPort;
  317. char reservedPort[6];
  318. const char * proto;
  319. const char * host;
  320. const char * desc;
  321. const char * remoteHost;
  322. const char * leaseDuration = "0";
  323. int r;
  324. if (!PyArg_ParseTuple(args, "HssHss", &ePort, &proto, &host, &iPort, &desc, &remoteHost))
  325. return NULL;
  326. Py_BEGIN_ALLOW_THREADS
  327. sprintf(extPort, "%hu", ePort);
  328. sprintf(inPort, "%hu", iPort);
  329. r = UPNP_AddAnyPortMapping(self->urls.controlURL, self->data.first.servicetype,
  330. extPort, inPort, host, desc, proto,
  331. remoteHost, leaseDuration, reservedPort);
  332. Py_END_ALLOW_THREADS
  333. if(r==UPNPCOMMAND_SUCCESS) {
  334. return Py_BuildValue("i", atoi(reservedPort));
  335. } else {
  336. /* TODO: have our own exception type ! */
  337. PyErr_SetString(PyExc_Exception, strupnperror(r));
  338. return NULL;
  339. }
  340. }
  341. /* DeletePortMapping(extPort, proto, removeHost='')
  342. * proto = 'UDP', 'TCP' */
  343. static PyObject *
  344. UPnP_deleteportmapping(UPnPObject *self, PyObject *args)
  345. {
  346. char extPort[6];
  347. unsigned short ePort;
  348. const char * proto;
  349. const char * remoteHost = "";
  350. int r;
  351. if(!PyArg_ParseTuple(args, "Hs|z", &ePort, &proto, &remoteHost))
  352. return NULL;
  353. Py_BEGIN_ALLOW_THREADS
  354. sprintf(extPort, "%hu", ePort);
  355. r = UPNP_DeletePortMapping(self->urls.controlURL, self->data.first.servicetype,
  356. extPort, proto, remoteHost);
  357. Py_END_ALLOW_THREADS
  358. if(r==UPNPCOMMAND_SUCCESS) {
  359. Py_RETURN_TRUE;
  360. } else {
  361. /* TODO: have our own exception type ! */
  362. PyErr_SetString(PyExc_Exception, strupnperror(r));
  363. return NULL;
  364. }
  365. }
  366. /* DeletePortMappingRange(extPort, proto, removeHost='')
  367. * proto = 'UDP', 'TCP' */
  368. static PyObject *
  369. UPnP_deleteportmappingrange(UPnPObject *self, PyObject *args)
  370. {
  371. char extPortStart[6];
  372. unsigned short ePortStart;
  373. char extPortEnd[6];
  374. unsigned short ePortEnd;
  375. const char * proto;
  376. unsigned char manage;
  377. char manageStr[1];
  378. int r;
  379. if(!PyArg_ParseTuple(args, "HHsb", &ePortStart, &ePortEnd, &proto, &manage))
  380. return NULL;
  381. Py_BEGIN_ALLOW_THREADS
  382. sprintf(extPortStart, "%hu", ePortStart);
  383. sprintf(extPortEnd, "%hu", ePortEnd);
  384. sprintf(manageStr, "%hhu", manage);
  385. r = UPNP_DeletePortMappingRange(self->urls.controlURL, self->data.first.servicetype,
  386. extPortStart, extPortEnd, proto, manageStr);
  387. Py_END_ALLOW_THREADS
  388. if(r==UPNPCOMMAND_SUCCESS) {
  389. Py_RETURN_TRUE;
  390. } else {
  391. /* TODO: have our own exception type ! */
  392. PyErr_SetString(PyExc_Exception, strupnperror(r));
  393. return NULL;
  394. }
  395. }
  396. static PyObject *
  397. UPnP_getportmappingnumberofentries(UPnPObject *self)
  398. {
  399. unsigned int n = 0;
  400. int r;
  401. Py_BEGIN_ALLOW_THREADS
  402. r = UPNP_GetPortMappingNumberOfEntries(self->urls.controlURL,
  403. self->data.first.servicetype,
  404. &n);
  405. Py_END_ALLOW_THREADS
  406. if(r==UPNPCOMMAND_SUCCESS) {
  407. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  408. return Py_BuildValue("I", n);
  409. #else
  410. return Py_BuildValue("i", (int)n);
  411. #endif
  412. } else {
  413. /* TODO: have our own exception type ! */
  414. PyErr_SetString(PyExc_Exception, strupnperror(r));
  415. return NULL;
  416. }
  417. }
  418. /* GetSpecificPortMapping(ePort, proto, remoteHost='')
  419. * proto = 'UDP' or 'TCP' */
  420. static PyObject *
  421. UPnP_getspecificportmapping(UPnPObject *self, PyObject *args)
  422. {
  423. char extPort[6];
  424. unsigned short ePort;
  425. const char * proto;
  426. const char * remoteHost = "";
  427. char intClient[40];
  428. char intPort[6];
  429. unsigned short iPort;
  430. char desc[80];
  431. char enabled[4];
  432. char leaseDuration[16];
  433. if(!PyArg_ParseTuple(args, "Hs|z", &ePort, &proto, &remoteHost))
  434. return NULL;
  435. extPort[0] = '\0'; intClient[0] = '\0'; intPort[0] = '\0';
  436. desc[0] = '\0'; enabled[0] = '\0'; leaseDuration[0] = '\0';
  437. Py_BEGIN_ALLOW_THREADS
  438. sprintf(extPort, "%hu", ePort);
  439. UPNP_GetSpecificPortMappingEntry(self->urls.controlURL,
  440. self->data.first.servicetype,
  441. extPort, proto, remoteHost,
  442. intClient, intPort,
  443. desc, enabled, leaseDuration);
  444. Py_END_ALLOW_THREADS
  445. if(intClient[0])
  446. {
  447. iPort = (unsigned short)atoi(intPort);
  448. return Py_BuildValue("(s,H,s,O,i)",
  449. intClient, iPort, desc,
  450. PyBool_FromLong(atoi(enabled)),
  451. atoi(leaseDuration));
  452. }
  453. else
  454. {
  455. Py_RETURN_NONE;
  456. }
  457. }
  458. /* GetGenericPortMapping(index) */
  459. static PyObject *
  460. UPnP_getgenericportmapping(UPnPObject *self, PyObject *args)
  461. {
  462. int i, r;
  463. char index[8];
  464. char intClient[40];
  465. char intPort[6];
  466. unsigned short iPort;
  467. char extPort[6];
  468. unsigned short ePort;
  469. char protocol[4];
  470. char desc[80];
  471. char enabled[6];
  472. char rHost[64];
  473. char duration[16]; /* lease duration */
  474. unsigned int dur;
  475. if(!PyArg_ParseTuple(args, "i", &i))
  476. return NULL;
  477. Py_BEGIN_ALLOW_THREADS
  478. snprintf(index, sizeof(index), "%d", i);
  479. rHost[0] = '\0'; enabled[0] = '\0';
  480. duration[0] = '\0'; desc[0] = '\0';
  481. extPort[0] = '\0'; intPort[0] = '\0'; intClient[0] = '\0';
  482. r = UPNP_GetGenericPortMappingEntry(self->urls.controlURL,
  483. self->data.first.servicetype,
  484. index,
  485. extPort, intClient, intPort,
  486. protocol, desc, enabled, rHost,
  487. duration);
  488. Py_END_ALLOW_THREADS
  489. if(r==UPNPCOMMAND_SUCCESS)
  490. {
  491. ePort = (unsigned short)atoi(extPort);
  492. iPort = (unsigned short)atoi(intPort);
  493. dur = (unsigned int)strtoul(duration, 0, 0);
  494. #if (PY_MAJOR_VERSION >= 3) || (PY_MAJOR_VERSION == 2 && PY_MINOR_VERSION > 3)
  495. return Py_BuildValue("(H,s,(s,H),s,s,s,I)",
  496. ePort, protocol, intClient, iPort,
  497. desc, enabled, rHost, dur);
  498. #else
  499. return Py_BuildValue("(i,s,(s,i),s,s,s,i)",
  500. (int)ePort, protocol, intClient, (int)iPort,
  501. desc, enabled, rHost, (int)dur);
  502. #endif
  503. }
  504. else
  505. {
  506. Py_RETURN_NONE;
  507. }
  508. }
  509. /* miniupnpc.UPnP object Method Table */
  510. static PyMethodDef UPnP_methods[] = {
  511. {"discover", (PyCFunction)UPnP_discover, METH_NOARGS,
  512. "discover UPnP IGD devices on the network"
  513. },
  514. {"selectigd", (PyCFunction)UPnP_selectigd, METH_NOARGS,
  515. "select a valid UPnP IGD among discovered devices"
  516. },
  517. {"totalbytesent", (PyCFunction)UPnP_totalbytesent, METH_NOARGS,
  518. "return the total number of bytes sent by UPnP IGD"
  519. },
  520. {"totalbytereceived", (PyCFunction)UPnP_totalbytereceived, METH_NOARGS,
  521. "return the total number of bytes received by UPnP IGD"
  522. },
  523. {"totalpacketsent", (PyCFunction)UPnP_totalpacketsent, METH_NOARGS,
  524. "return the total number of packets sent by UPnP IGD"
  525. },
  526. {"totalpacketreceived", (PyCFunction)UPnP_totalpacketreceived, METH_NOARGS,
  527. "return the total number of packets received by UPnP IGD"
  528. },
  529. {"statusinfo", (PyCFunction)UPnP_statusinfo, METH_NOARGS,
  530. "return status and uptime"
  531. },
  532. {"connectiontype", (PyCFunction)UPnP_connectiontype, METH_NOARGS,
  533. "return IGD WAN connection type"
  534. },
  535. {"externalipaddress", (PyCFunction)UPnP_externalipaddress, METH_NOARGS,
  536. "return external IP address"
  537. },
  538. {"addportmapping", (PyCFunction)UPnP_addportmapping, METH_VARARGS,
  539. "add a port mapping"
  540. },
  541. {"addanyportmapping", (PyCFunction)UPnP_addanyportmapping, METH_VARARGS,
  542. "add a port mapping, IGD to select alternative if necessary"
  543. },
  544. {"deleteportmapping", (PyCFunction)UPnP_deleteportmapping, METH_VARARGS,
  545. "delete a port mapping"
  546. },
  547. {"deleteportmappingrange", (PyCFunction)UPnP_deleteportmappingrange, METH_VARARGS,
  548. "delete a range of port mappings"
  549. },
  550. {"getportmappingnumberofentries", (PyCFunction)UPnP_getportmappingnumberofentries, METH_NOARGS,
  551. "-- non standard --"
  552. },
  553. {"getspecificportmapping", (PyCFunction)UPnP_getspecificportmapping, METH_VARARGS,
  554. "get details about a specific port mapping entry"
  555. },
  556. {"getgenericportmapping", (PyCFunction)UPnP_getgenericportmapping, METH_VARARGS,
  557. "get all details about the port mapping at index"
  558. },
  559. {NULL} /* Sentinel */
  560. };
  561. static PyTypeObject UPnPType = {
  562. PyVarObject_HEAD_INIT(NULL,
  563. 0) /*ob_size*/
  564. "miniupnpc.UPnP", /*tp_name*/
  565. sizeof(UPnPObject), /*tp_basicsize*/
  566. 0, /*tp_itemsize*/
  567. (destructor)UPnPObject_dealloc,/*tp_dealloc*/
  568. 0, /*tp_print*/
  569. 0, /*tp_getattr*/
  570. 0, /*tp_setattr*/
  571. 0, /*tp_compare*/
  572. 0, /*tp_repr*/
  573. 0, /*tp_as_number*/
  574. 0, /*tp_as_sequence*/
  575. 0, /*tp_as_mapping*/
  576. 0, /*tp_hash */
  577. 0, /*tp_call*/
  578. 0, /*tp_str*/
  579. 0, /*tp_getattro*/
  580. 0, /*tp_setattro*/
  581. 0, /*tp_as_buffer*/
  582. Py_TPFLAGS_DEFAULT, /*tp_flags*/
  583. "UPnP objects", /* tp_doc */
  584. 0, /* tp_traverse */
  585. 0, /* tp_clear */
  586. 0, /* tp_richcompare */
  587. 0, /* tp_weaklistoffset */
  588. 0, /* tp_iter */
  589. 0, /* tp_iternext */
  590. UPnP_methods, /* tp_methods */
  591. UPnP_members, /* tp_members */
  592. 0, /* tp_getset */
  593. 0, /* tp_base */
  594. 0, /* tp_dict */
  595. 0, /* tp_descr_get */
  596. 0, /* tp_descr_set */
  597. 0, /* tp_dictoffset */
  598. (initproc)UPnP_init, /* tp_init */
  599. 0, /* tp_alloc */
  600. #ifndef _WIN32
  601. PyType_GenericNew,/*UPnP_new,*/ /* tp_new */
  602. #else
  603. 0,
  604. #endif
  605. };
  606. /* module methods */
  607. static PyMethodDef miniupnpc_methods[] = {
  608. {NULL} /* Sentinel */
  609. };
  610. #if PY_MAJOR_VERSION >= 3
  611. static struct PyModuleDef moduledef = {
  612. PyModuleDef_HEAD_INIT,
  613. "miniupnpc", /* m_name */
  614. "miniupnpc module.", /* m_doc */
  615. -1, /* m_size */
  616. miniupnpc_methods, /* m_methods */
  617. NULL, /* m_reload */
  618. NULL, /* m_traverse */
  619. NULL, /* m_clear */
  620. NULL, /* m_free */
  621. };
  622. #endif
  623. #ifndef PyMODINIT_FUNC /* declarations for DLL import/export */
  624. #define PyMODINIT_FUNC void
  625. #endif
  626. PyMODINIT_FUNC
  627. #if PY_MAJOR_VERSION >= 3
  628. PyInit_miniupnpc(void)
  629. #else
  630. initminiupnpc(void)
  631. #endif
  632. {
  633. PyObject* m;
  634. #ifdef _WIN32
  635. UPnPType.tp_new = PyType_GenericNew;
  636. #endif
  637. if (PyType_Ready(&UPnPType) < 0)
  638. #if PY_MAJOR_VERSION >= 3
  639. return 0;
  640. #else
  641. return;
  642. #endif
  643. #if PY_MAJOR_VERSION >= 3
  644. m = PyModule_Create(&moduledef);
  645. #else
  646. m = Py_InitModule3("miniupnpc", miniupnpc_methods,
  647. "miniupnpc module.");
  648. #endif
  649. Py_INCREF(&UPnPType);
  650. PyModule_AddObject(m, "UPnP", (PyObject *)&UPnPType);
  651. #if PY_MAJOR_VERSION >= 3
  652. return m;
  653. #endif
  654. }