1 ###################################################
2 # Python function wrapper generator
3 # Copyright jelmer@samba.org 2007
4 # released under the GNU GPL
6 package Parse::Pidl::Samba4::Python;
12 use Parse::Pidl::Typelist;
13 use Parse::Pidl::Util qw(has_property ParseExpr);
15 use vars qw($VERSION);
20 my $self = { res => "", res_hdr => "", tabs => "", constants => {}};
27 $self->{res_hdr} .= shift;
34 $self->{res} .= $self->{tabs};
43 $self->{tabs} .= "\t";
49 $self->{tabs} = substr($self->{tabs}, 0, -1);
59 $self->pidl_hdr("#include \"librpc/gen_ndr/py_$_\.h\"\n");
65 my ($self, $const) = @_;
66 $self->{constants}->{$const->{NAME}} = [$const->{DATA}->{TYPE}, $const->{VALUE}];
69 sub FromTypeToPythonFunction($$)
71 my ($self, $type) = @_;
76 sub FromPythonToTypeFunction($$)
78 my ($self, $type) = @_;
83 sub TypeConstructor($$)
85 my ($self, $type) = @_;
87 $self->pidl("static PyObject *py_$type->{NAME}(PyObject *self, PyObject *args)");
92 $self->pidl("return Py_None;");
98 sub PythonFunction($$$)
100 my ($self, $fn, $iface) = @_;
102 $self->pidl("static PyObject *py_$fn->{NAME}(PyObject *self, PyObject *args)");
105 $self->pidl("$iface\_InterfaceObject *iface = ($iface\_InterfaceObject *)self;");
106 $self->pidl("NTSTATUS status;");
109 $self->handle_ntstatus("status", "NULL");
110 $self->pidl("return Py_None;");
116 sub handle_ntstatus($$$)
118 my ($self, $var, $retval) = @_;
120 $self->pidl("if (NT_STATUS_IS_ERR($var)) {");
122 $self->pidl("PyErr_SetString(PyExc_RuntimeError, nt_errstr($var));");
123 $self->pidl("return $retval;");
131 my($self,$interface) = @_;
133 $self->pidl_hdr("#ifndef _HEADER_PYTHON_$interface->{NAME}\n");
134 $self->pidl_hdr("#define _HEADER_PYTHON_$interface->{NAME}\n\n");
136 $self->pidl_hdr("\n");
138 $self->Const($_) foreach (@{$interface->{CONSTS}});
140 foreach (@{$interface->{TYPES}}) {
141 $self->FromTypeToPythonFunction($_);
142 $self->FromPythonToTypeFunction($_);
143 $self->TypeConstructor($_);
146 $self->pidl("staticforward PyTypeObject $interface->{NAME}_InterfaceType;");
147 $self->pidl("typedef struct {");
149 $self->pidl("PyObject_HEAD");
150 $self->pidl("struct dcerpc_pipe *pipe;");
152 $self->pidl("} $interface->{NAME}_InterfaceObject;");
156 foreach my $d (@{$interface->{FUNCTIONS}}) {
157 next if not defined($d->{OPNUM});
158 next if has_property($d, "nopython");
160 $self->PythonFunction($d, $interface->{NAME});
163 $self->pidl("static PyMethodDef interface_$interface->{NAME}\_methods[] = {");
165 foreach my $d (@{$interface->{FUNCTIONS}}) {
166 next if not defined($d->{OPNUM});
167 next if has_property($d, "nopython");
169 my $fn_name = $d->{NAME};
171 $fn_name =~ s/^$interface->{NAME}_//;
173 $self->pidl("{ (char *)\"$fn_name\", (PyCFunction)py_$d->{NAME}, METH_VARARGS|METH_KEYWORDS, NULL },");
175 $self->pidl("{ NULL, NULL, 0, NULL }");
180 $self->pidl("static void interface_$interface->{NAME}_dealloc(PyObject* self)");
183 $self->pidl("$interface->{NAME}_InterfaceObject *interface = ($interface->{NAME}_InterfaceObject *)self;");
184 $self->pidl("talloc_free(interface->pipe);");
185 $self->pidl("PyObject_Del(self);");
190 $self->pidl("static PyObject *interface_$interface->{NAME}_getattr(PyTypeObject *obj, char *name)");
193 $self->pidl("return Py_FindMethod(interface_$interface->{NAME}\_methods, (PyObject *)obj, name);");
199 $self->pidl("static PyTypeObject $interface->{NAME}_InterfaceType = {");
201 $self->pidl("PyObject_HEAD_INIT(NULL) 0,");
202 $self->pidl(".tp_name = \"$interface->{NAME}\",");
203 $self->pidl(".tp_basicsize = sizeof($interface->{NAME}_InterfaceObject),");
204 $self->pidl(".tp_dealloc = interface_$interface->{NAME}_dealloc,");
205 $self->pidl(".tp_getattr = interface_$interface->{NAME}_getattr,");
211 $self->pidl("static PyObject *interface_$interface->{NAME}(PyObject *self, PyObject *args)");
214 $self->pidl("$interface->{NAME}_InterfaceObject *ret;");
215 $self->pidl("const char *binding_string;");
216 $self->pidl("struct cli_credentials *credentials;");
217 $self->pidl("struct loadparm_context *lp_ctx;");
218 $self->pidl("NTSTATUS status;");
221 # FIXME: Arguments: binding string, credentials, loadparm context
222 $self->pidl("ret = PyObject_New($interface->{NAME}_InterfaceObject, &$interface->{NAME}_InterfaceType);");
225 $self->pidl("status = dcerpc_pipe_connect(NULL, &ret->pipe, binding_string, ");
226 $self->pidl(" &ndr_table_$interface->{NAME}, credentials, NULL, lp_ctx);");
227 $self->handle_ntstatus("status", "NULL");
229 $self->pidl("return (PyObject *)ret;");
235 $self->pidl_hdr("\n");
236 $self->pidl_hdr("#endif /* _HEADER_NDR_$interface->{NAME} */\n");
241 my($self,$basename,$ndr,$hdr) = @_;
244 $py_hdr =~ s/ndr_([^\/]+)$/py_$1/g;
246 $self->pidl_hdr("/* header auto-generated by pidl */\n\n");
249 /* Python wrapper functions auto-generated by pidl */
250 #include \"includes.h\"
252 #include \"librpc/rpc/dcerpc.h\"
258 foreach my $x (@$ndr) {
259 ($x->{TYPE} eq "INTERFACE") && $self->Interface($x);
260 ($x->{TYPE} eq "IMPORT") && $self->Import(@{$x->{PATHS}});
263 $self->pidl("static PyMethodDef $basename\_methods[] = {");
265 foreach my $x (@$ndr) {
266 next if ($x->{TYPE} ne "INTERFACE");
267 $self->pidl("{ (char *)\"$x->{NAME}\", (PyCFunction)interface_$x->{NAME}, METH_VARARGS|METH_KEYWORDS, NULL },");
269 foreach my $d (@{$x->{TYPES}}) {
270 next if has_property($d, "nopython");
271 next if ($d->{TYPE} eq "ENUM" or $d->{TYPE} eq "BITMAP");
273 my $fn_name = $d->{NAME};
275 $fn_name =~ s/^$x->{NAME}_//;
276 $fn_name =~ s/^$basename\_//;
278 $self->pidl("{ (char *)\"$fn_name\", (PyCFunction)py_$d->{NAME}, METH_VARARGS|METH_KEYWORDS, NULL },");
282 $self->pidl("{ NULL, NULL, 0, NULL }");
288 $self->pidl("void init$basename(void)");
291 $self->pidl("PyObject *m;");
292 $self->pidl("m = Py_InitModule((char *)\"$basename\", $basename\_methods);");
293 foreach (keys %{$self->{constants}}) {
294 # FIXME: Handle non-string constants
295 $self->pidl("PyModule_AddObject(m, \"$_\", PyString_FromString(" . $self->{constants}->{$_}->[1] . "));");
299 return ($self->{res_hdr}, $self->{res});