2 Unix SMB/CIFS implementation.
3 Samba utility functions
4 Copyright © Jelmer Vernooij <jelmer@samba.org> 2008
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
22 #include "libcli/util/pyerrors.h"
23 #include "scripting/python/modules.h"
24 #include "../libcli/nbt/libnbt.h"
25 #include "lib/events/events.h"
27 #ifndef Py_RETURN_NONE
28 #define Py_RETURN_NONE return Py_INCREF(Py_None), Py_None
31 PyAPI_DATA(PyTypeObject
) nbt_node_Type
;
36 struct nbt_name_socket
*socket
;
39 static void py_nbt_node_dealloc(PyObject
*obj
)
41 talloc_free(((nbt_node_Object
*)obj
)->mem_ctx
);
45 static PyObject
*py_nbt_node_init(PyTypeObject
*self
, PyObject
*args
, PyObject
*kwargs
)
47 struct tevent_context
*ev
;
48 nbt_node_Object
*ret
= PyObject_New(nbt_node_Object
, &nbt_node_Type
);
50 ret
->mem_ctx
= talloc_new(NULL
);
51 if (ret
->mem_ctx
== NULL
)
54 ev
= s4_event_context_init(ret
->mem_ctx
);
55 ret
->socket
= nbt_name_socket_init(ret
->mem_ctx
, ev
, py_iconv_convenience(ret
->mem_ctx
));
56 return (PyObject
*)ret
;
59 static bool PyObject_AsDestinationTuple(PyObject
*obj
, const char **dest_addr
, uint16_t *dest_port
)
61 if (PyString_Check(obj
)) {
62 *dest_addr
= PyString_AsString(obj
);
63 *dest_port
= NBT_NAME_SERVICE_PORT
;
67 if (PyTuple_Check(obj
)) {
68 if (PyTuple_Size(obj
) < 1) {
69 PyErr_SetString(PyExc_TypeError
, "Destination tuple size invalid");
73 if (!PyString_Check(PyTuple_GetItem(obj
, 0))) {
74 PyErr_SetString(PyExc_TypeError
, "Destination tuple first element not string");
78 *dest_addr
= PyString_AsString(obj
);
80 if (PyTuple_Size(obj
) == 1) {
81 *dest_port
= NBT_NAME_SERVICE_PORT
;
83 } else if (PyInt_Check(PyTuple_GetItem(obj
, 1))) {
84 *dest_port
= PyInt_AsLong(PyTuple_GetItem(obj
, 1));
87 PyErr_SetString(PyExc_TypeError
, "Destination tuple second element not a port");
92 PyErr_SetString(PyExc_TypeError
, "Destination tuple second element not a port");
96 static bool PyObject_AsNBTName(PyObject
*obj
, struct nbt_name_socket
*name_socket
, struct nbt_name
*name
)
98 if (PyTuple_Check(obj
)) {
99 if (PyTuple_Size(obj
) == 2) {
100 name
->name
= PyString_AsString(PyTuple_GetItem(obj
, 0));
101 name
->type
= PyInt_AsLong(PyTuple_GetItem(obj
, 1));
104 } else if (PyTuple_Size(obj
) == 3) {
105 name
->name
= PyString_AsString(PyTuple_GetItem(obj
, 0));
106 name
->scope
= PyString_AsString(PyTuple_GetItem(obj
, 1));
107 name
->type
= PyInt_AsLong(PyTuple_GetItem(obj
, 2));
110 PyErr_SetString(PyExc_TypeError
, "Invalid tuple size");
115 if (PyString_Check(obj
)) {
116 /* FIXME: Parse string to be able to interpret things like RHONWYN<02> ? */
117 name
->name
= PyString_AsString(obj
);
123 PyErr_SetString(PyExc_TypeError
, "Invalid type for object");
127 static PyObject
*PyObject_FromNBTName(struct nbt_name_socket
*name_socket
, struct smb_iconv_convenience
*ic
,
128 struct nbt_name
*name
)
131 return Py_BuildValue("(ssi)", name
->name
, name
->scope
, name
->type
);
133 return Py_BuildValue("(si)", name
->name
, name
->type
);
137 static PyObject
*py_nbt_name_query(PyObject
*self
, PyObject
*args
, PyObject
*kwargs
)
139 nbt_node_Object
*node
= (nbt_node_Object
*)self
;
140 PyObject
*ret
, *reply_addrs
, *py_dest
, *py_name
;
141 struct nbt_name_query io
;
145 const char *kwnames
[] = { "name", "dest", "broadcast", "wins", "timeout",
147 io
.in
.broadcast
= true;
148 io
.in
.wins_lookup
= false;
152 if (!PyArg_ParseTupleAndKeywords(args
, kwargs
, "OO|bbii:query_name",
153 discard_const_p(char *, kwnames
),
155 &io
.in
.broadcast
, &io
.in
.wins_lookup
,
156 &io
.in
.timeout
, &io
.in
.retries
)) {
160 if (!PyObject_AsDestinationTuple(py_dest
, &io
.in
.dest_addr
, &io
.in
.dest_port
))
163 if (!PyObject_AsNBTName(py_name
, node
->socket
, &io
.in
.name
))
166 status
= nbt_name_query(node
->socket
, NULL
, &io
);
168 if (NT_STATUS_IS_ERR(status
)) {
169 PyErr_SetNTSTATUS(status
);
173 ret
= PyTuple_New(3);
176 PyTuple_SetItem(ret
, 0, PyString_FromString(io
.out
.reply_from
));
178 py_name
= PyObject_FromNBTName(node
->socket
, py_iconv_convenience(node
->socket
), &io
.out
.name
);
182 PyTuple_SetItem(ret
, 1, py_name
);
184 reply_addrs
= PyList_New(io
.out
.num_addrs
);
185 if (reply_addrs
== NULL
) {
190 for (i
= 0; i
< io
.out
.num_addrs
; i
++) {
191 PyList_SetItem(reply_addrs
, i
, PyString_FromString(io
.out
.reply_addrs
[i
]));
194 PyTuple_SetItem(ret
, 2, reply_addrs
);
198 static PyObject
*py_nbt_name_status(PyObject
*self
, PyObject
*args
, PyObject
*kwargs
)
200 nbt_node_Object
*node
= (nbt_node_Object
*)self
;
201 PyObject
*ret
, *py_dest
, *py_name
, *py_names
;
202 struct nbt_name_status io
;
206 const char *kwnames
[] = { "name", "dest", "timeout", "retries", NULL
};
211 if (!PyArg_ParseTupleAndKeywords(args
, kwargs
, "OO|ii:name_status",
212 discard_const_p(char *, kwnames
),
214 &io
.in
.timeout
, &io
.in
.retries
)) {
218 if (!PyObject_AsDestinationTuple(py_dest
, &io
.in
.dest_addr
, &io
.in
.dest_port
))
221 if (!PyObject_AsNBTName(py_name
, node
->socket
, &io
.in
.name
))
224 status
= nbt_name_status(node
->socket
, NULL
, &io
);
226 if (NT_STATUS_IS_ERR(status
)) {
227 PyErr_SetNTSTATUS(status
);
231 ret
= PyTuple_New(3);
234 PyTuple_SetItem(ret
, 0, PyString_FromString(io
.out
.reply_from
));
236 py_name
= PyObject_FromNBTName(node
->socket
, py_iconv_convenience(NULL
), &io
.out
.name
);
240 PyTuple_SetItem(ret
, 1, py_name
);
242 py_names
= PyList_New(io
.out
.status
.num_names
);
244 for (i
= 0; i
< io
.out
.status
.num_names
; i
++) {
245 PyList_SetItem(py_names
, i
, Py_BuildValue("(sii)",
246 io
.out
.status
.names
[i
].name
,
247 io
.out
.status
.names
[i
].nb_flags
,
248 io
.out
.status
.names
[i
].type
));
251 PyTuple_SetItem(ret
, 2, py_names
);
256 static PyObject
*py_nbt_name_register(PyObject
*self
, PyObject
*args
, PyObject
*kwargs
)
258 nbt_node_Object
*node
= (nbt_node_Object
*)self
;
259 PyObject
*ret
, *py_dest
, *py_name
;
260 struct nbt_name_register io
;
263 const char *kwnames
[] = { "name", "address", "dest", "register_demand", "broadcast",
264 "multi_homed", "ttl", "timeout", "retries", NULL
};
266 io
.in
.broadcast
= true;
267 io
.in
.multi_homed
= true;
268 io
.in
.register_demand
= true;
272 if (!PyArg_ParseTupleAndKeywords(args
, kwargs
, "OsO|bbbiii:query_name",
273 discard_const_p(char *, kwnames
),
274 &py_name
, &io
.in
.address
, &py_dest
,
275 &io
.in
.register_demand
,
276 &io
.in
.broadcast
, &io
.in
.multi_homed
,
277 &io
.in
.ttl
, &io
.in
.timeout
, &io
.in
.retries
)) {
281 if (!PyObject_AsDestinationTuple(py_dest
, &io
.in
.dest_addr
, &io
.in
.dest_port
))
284 if (!PyObject_AsNBTName(py_name
, node
->socket
, &io
.in
.name
))
287 status
= nbt_name_register(node
->socket
, NULL
, &io
);
289 if (NT_STATUS_IS_ERR(status
)) {
290 PyErr_SetNTSTATUS(status
);
294 ret
= PyTuple_New(3);
297 PyTuple_SetItem(ret
, 0, PyString_FromString(io
.out
.reply_from
));
299 py_name
= PyObject_FromNBTName(node
->socket
, py_iconv_convenience(NULL
), &io
.out
.name
);
303 PyTuple_SetItem(ret
, 1, py_name
);
305 PyTuple_SetItem(ret
, 2, PyString_FromString(io
.out
.reply_addr
));
307 PyTuple_SetItem(ret
, 3, PyInt_FromLong(io
.out
.rcode
));
312 static PyObject
*py_nbt_name_refresh(PyObject
*self
, PyObject
*args
, PyObject
*kwargs
)
314 nbt_node_Object
*node
= (nbt_node_Object
*)self
;
315 PyObject
*ret
, *py_dest
, *py_name
;
316 struct nbt_name_refresh io
;
319 const char *kwnames
[] = { "name", "address", "dest", "nb_flags", "broadcast",
320 "ttl", "timeout", "retries", NULL
};
322 io
.in
.broadcast
= true;
327 if (!PyArg_ParseTupleAndKeywords(args
, kwargs
, "OsO|ibiii:query_name",
328 discard_const_p(char *, kwnames
),
329 &py_name
, &io
.in
.address
, &py_dest
,
332 &io
.in
.ttl
, &io
.in
.timeout
, &io
.in
.retries
)) {
336 if (!PyObject_AsDestinationTuple(py_dest
, &io
.in
.dest_addr
, &io
.in
.dest_port
))
339 if (!PyObject_AsNBTName(py_name
, node
->socket
, &io
.in
.name
))
342 status
= nbt_name_refresh(node
->socket
, NULL
, &io
);
344 if (NT_STATUS_IS_ERR(status
)) {
345 PyErr_SetNTSTATUS(status
);
349 ret
= PyTuple_New(3);
352 PyTuple_SetItem(ret
, 0, PyString_FromString(io
.out
.reply_from
));
354 py_name
= PyObject_FromNBTName(node
->socket
, py_iconv_convenience(NULL
), &io
.out
.name
);
358 PyTuple_SetItem(ret
, 1, py_name
);
360 PyTuple_SetItem(ret
, 2, PyString_FromString(io
.out
.reply_addr
));
362 PyTuple_SetItem(ret
, 3, PyInt_FromLong(io
.out
.rcode
));
367 static PyObject
*py_nbt_name_release(PyObject
*self
, PyObject
*args
, PyObject
*kwargs
)
369 Py_RETURN_NONE
; /* FIXME */
372 static PyMethodDef py_nbt_methods
[] = {
373 { "query_name", (PyCFunction
)py_nbt_name_query
, METH_VARARGS
|METH_KEYWORDS
,
374 "S.query_name(name, dest, broadcast=True, wins=False, timeout=0, retries=3) -> (reply_from, name, reply_addr)\n"
375 "Query for a NetBIOS name" },
376 { "register_name", (PyCFunction
)py_nbt_name_register
, METH_VARARGS
|METH_KEYWORDS
,
377 "S.register_name(name, address, dest, register_demand=True, broadcast=True, multi_homed=True, ttl=0, timeout=0, retries=0) -> (reply_from, name, reply_addr, rcode)\n"
378 "Register a new name" },
379 { "release_name", (PyCFunction
)py_nbt_name_release
, METH_VARARGS
|METH_KEYWORDS
, "S.release_name(name, address, dest, nb_flags=0, broadcast=true, timeout=0, retries=3) -> (reply_from, name, reply_addr, rcode)\n"
380 "release a previously registered name" },
381 { "refresh_name", (PyCFunction
)py_nbt_name_refresh
, METH_VARARGS
|METH_KEYWORDS
, "S.refresh_name(name, address, dest, nb_flags=0, broadcast=True, ttl=0, timeout=0, retries=0) -> (reply_from, name, reply_addr, rcode)\n"
382 "release a previously registered name" },
383 { "name_status", (PyCFunction
)py_nbt_name_status
, METH_VARARGS
|METH_KEYWORDS
,
384 "S.name_status(name, dest, timeout=0, retries=0) -> (reply_from, name, status)\n"
385 "Find the status of a name" },
390 PyTypeObject nbt_node_Type
= {
391 PyObject_HEAD_INIT(NULL
) 0,
392 .tp_name
= "netbios.Node",
393 .tp_basicsize
= sizeof(nbt_node_Object
),
394 .tp_flags
= Py_TPFLAGS_DEFAULT
|Py_TPFLAGS_BASETYPE
,
395 .tp_new
= py_nbt_node_init
,
396 .tp_dealloc
= py_nbt_node_dealloc
,
397 .tp_methods
= py_nbt_methods
,
399 "Create a new NetBIOS node\n"
402 void initnetbios(void)
405 if (PyType_Ready(&nbt_node_Type
) < 0)
408 mod
= Py_InitModule3("netbios", NULL
, "NetBIOS over TCP/IP support");
410 Py_INCREF((PyObject
*)&nbt_node_Type
);
411 PyModule_AddObject(mod
, "Node", (PyObject
*)&nbt_node_Type
);