1 | /*
|
---|
2 | Python wrappers for DCERPC/SMB client routines.
|
---|
3 |
|
---|
4 | Copyright (C) Tim Potter, 2002
|
---|
5 |
|
---|
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 2 of the License, or
|
---|
9 | (at your option) any later version.
|
---|
10 |
|
---|
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.
|
---|
15 |
|
---|
16 | You should have received a copy of the GNU General Public License
|
---|
17 | along with this program; if not, write to the Free Software
|
---|
18 | Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
---|
19 | */
|
---|
20 |
|
---|
21 | #include "python/py_spoolss.h"
|
---|
22 |
|
---|
23 | /* Open a printer */
|
---|
24 |
|
---|
25 | PyObject *spoolss_openprinter(PyObject *self, PyObject *args, PyObject *kw)
|
---|
26 | {
|
---|
27 | char *unc_name, *server, *errstr;
|
---|
28 | TALLOC_CTX *mem_ctx = NULL;
|
---|
29 | POLICY_HND hnd;
|
---|
30 | WERROR werror;
|
---|
31 | PyObject *result = NULL, *creds = NULL;
|
---|
32 | static char *kwlist[] = { "printername", "creds", "access", NULL };
|
---|
33 | uint32 desired_access = MAXIMUM_ALLOWED_ACCESS;
|
---|
34 | struct cli_state *cli;
|
---|
35 |
|
---|
36 | if (!PyArg_ParseTupleAndKeywords(
|
---|
37 | args, kw, "s|Oi", kwlist, &unc_name, &creds,
|
---|
38 | &desired_access))
|
---|
39 | return NULL;
|
---|
40 |
|
---|
41 | if (unc_name[0] != '\\' || unc_name[1] != '\\') {
|
---|
42 | PyErr_SetString(PyExc_ValueError, "UNC name required");
|
---|
43 | return NULL;
|
---|
44 | }
|
---|
45 |
|
---|
46 | server = SMB_STRDUP(unc_name + 2);
|
---|
47 |
|
---|
48 | if (strchr(server, '\\')) {
|
---|
49 | char *c = strchr(server, '\\');
|
---|
50 | *c = 0;
|
---|
51 | }
|
---|
52 |
|
---|
53 | if (creds && creds != Py_None && !PyDict_Check(creds)) {
|
---|
54 | PyErr_SetString(PyExc_TypeError,
|
---|
55 | "credentials must be dictionary or None");
|
---|
56 | return NULL;
|
---|
57 | }
|
---|
58 |
|
---|
59 | if (!(cli = open_pipe_creds(server, creds, PI_SPOOLSS, &errstr))) {
|
---|
60 | PyErr_SetString(spoolss_error, errstr);
|
---|
61 | free(errstr);
|
---|
62 | goto done;
|
---|
63 | }
|
---|
64 |
|
---|
65 | if (!(mem_ctx = talloc_init("spoolss_openprinter"))) {
|
---|
66 | PyErr_SetString(spoolss_error,
|
---|
67 | "unable to init talloc context\n");
|
---|
68 | goto done;
|
---|
69 | }
|
---|
70 |
|
---|
71 | werror = rpccli_spoolss_open_printer_ex(
|
---|
72 | cli->pipe_list, mem_ctx, unc_name, "", desired_access, server,
|
---|
73 | "", &hnd);
|
---|
74 |
|
---|
75 | if (!W_ERROR_IS_OK(werror)) {
|
---|
76 | PyErr_SetObject(spoolss_werror, py_werror_tuple(werror));
|
---|
77 | goto done;
|
---|
78 | }
|
---|
79 |
|
---|
80 | result = new_spoolss_policy_hnd_object(cli, mem_ctx, &hnd);
|
---|
81 |
|
---|
82 | done:
|
---|
83 | if (!result) {
|
---|
84 | if (cli)
|
---|
85 | cli_shutdown(cli);
|
---|
86 |
|
---|
87 | if (mem_ctx)
|
---|
88 | talloc_destroy(mem_ctx);
|
---|
89 | }
|
---|
90 |
|
---|
91 | SAFE_FREE(server);
|
---|
92 |
|
---|
93 | return result;
|
---|
94 | }
|
---|
95 |
|
---|
96 | /* Close a printer */
|
---|
97 |
|
---|
98 | PyObject *spoolss_closeprinter(PyObject *self, PyObject *args)
|
---|
99 | {
|
---|
100 | PyObject *po;
|
---|
101 | spoolss_policy_hnd_object *hnd;
|
---|
102 | WERROR result;
|
---|
103 |
|
---|
104 | /* Parse parameters */
|
---|
105 |
|
---|
106 | if (!PyArg_ParseTuple(args, "O!", &spoolss_policy_hnd_type, &po))
|
---|
107 | return NULL;
|
---|
108 |
|
---|
109 | hnd = (spoolss_policy_hnd_object *)po;
|
---|
110 |
|
---|
111 | /* Call rpc function */
|
---|
112 |
|
---|
113 | result = rpccli_spoolss_close_printer(
|
---|
114 | hnd->cli, hnd->mem_ctx, &hnd->pol);
|
---|
115 |
|
---|
116 | /* Return value */
|
---|
117 |
|
---|
118 | Py_INCREF(Py_None);
|
---|
119 | return Py_None;
|
---|
120 | }
|
---|
121 |
|
---|
122 | /* Fetch printer information */
|
---|
123 |
|
---|
124 | PyObject *spoolss_hnd_getprinter(PyObject *self, PyObject *args, PyObject *kw)
|
---|
125 | {
|
---|
126 | spoolss_policy_hnd_object *hnd = (spoolss_policy_hnd_object *)self;
|
---|
127 | WERROR werror;
|
---|
128 | PyObject *result = NULL;
|
---|
129 | PRINTER_INFO_CTR ctr;
|
---|
130 | int level = 1;
|
---|
131 | static char *kwlist[] = {"level", NULL};
|
---|
132 |
|
---|
133 | /* Parse parameters */
|
---|
134 |
|
---|
135 | if (!PyArg_ParseTupleAndKeywords(args, kw, "|i", kwlist, &level))
|
---|
136 | return NULL;
|
---|
137 |
|
---|
138 | ZERO_STRUCT(ctr);
|
---|
139 |
|
---|
140 | /* Call rpc function */
|
---|
141 |
|
---|
142 | werror = rpccli_spoolss_getprinter(
|
---|
143 | hnd->cli, hnd->mem_ctx, &hnd->pol, level, &ctr);
|
---|
144 |
|
---|
145 | /* Return value */
|
---|
146 |
|
---|
147 | if (!W_ERROR_IS_OK(werror)) {
|
---|
148 | PyErr_SetObject(spoolss_werror, py_werror_tuple(werror));
|
---|
149 | return NULL;
|
---|
150 | }
|
---|
151 |
|
---|
152 | result = Py_None;
|
---|
153 |
|
---|
154 | switch (level) {
|
---|
155 |
|
---|
156 | case 0:
|
---|
157 | py_from_PRINTER_INFO_0(&result, ctr.printers_0);
|
---|
158 | break;
|
---|
159 |
|
---|
160 | case 1:
|
---|
161 | py_from_PRINTER_INFO_1(&result, ctr.printers_1);
|
---|
162 | break;
|
---|
163 |
|
---|
164 | case 2:
|
---|
165 | py_from_PRINTER_INFO_2(&result, ctr.printers_2);
|
---|
166 | break;
|
---|
167 |
|
---|
168 | case 3:
|
---|
169 | py_from_PRINTER_INFO_3(&result, ctr.printers_3);
|
---|
170 | break;
|
---|
171 | }
|
---|
172 |
|
---|
173 | Py_INCREF(result);
|
---|
174 | return result;
|
---|
175 | }
|
---|
176 |
|
---|
177 | /* Set printer information */
|
---|
178 |
|
---|
179 | PyObject *spoolss_hnd_setprinter(PyObject *self, PyObject *args, PyObject *kw)
|
---|
180 | {
|
---|
181 | spoolss_policy_hnd_object *hnd = (spoolss_policy_hnd_object *)self;
|
---|
182 | WERROR werror;
|
---|
183 | PyObject *info;
|
---|
184 | PRINTER_INFO_CTR ctr;
|
---|
185 | uint32 level;
|
---|
186 | static char *kwlist[] = {"dict", NULL};
|
---|
187 | union {
|
---|
188 | PRINTER_INFO_1 printers_1;
|
---|
189 | PRINTER_INFO_2 printers_2;
|
---|
190 | PRINTER_INFO_3 printers_3;
|
---|
191 | } pinfo;
|
---|
192 |
|
---|
193 | /* Parse parameters */
|
---|
194 |
|
---|
195 | if (!PyArg_ParseTupleAndKeywords(
|
---|
196 | args, kw, "O!", kwlist, &PyDict_Type, &info))
|
---|
197 | return NULL;
|
---|
198 |
|
---|
199 | if (!get_level_value(info, &level)) {
|
---|
200 | PyErr_SetString(spoolss_error, "invalid info level");
|
---|
201 | return NULL;
|
---|
202 | }
|
---|
203 |
|
---|
204 | if (level < 1 && level > 3) {
|
---|
205 | PyErr_SetString(spoolss_error, "unsupported info level");
|
---|
206 | return NULL;
|
---|
207 | }
|
---|
208 |
|
---|
209 | /* Fill in printer info */
|
---|
210 |
|
---|
211 | ZERO_STRUCT(ctr);
|
---|
212 |
|
---|
213 | switch (level) {
|
---|
214 | case 1:
|
---|
215 | ctr.printers_1 = &pinfo.printers_1;
|
---|
216 |
|
---|
217 | if (!py_to_PRINTER_INFO_1(ctr.printers_1, info)){
|
---|
218 | PyErr_SetString(spoolss_error,
|
---|
219 | "error converting printer to info 1");
|
---|
220 | return NULL;
|
---|
221 | }
|
---|
222 |
|
---|
223 | break;
|
---|
224 | case 2:
|
---|
225 | ctr.printers_2 = &pinfo.printers_2;
|
---|
226 |
|
---|
227 | if (!py_to_PRINTER_INFO_2(ctr.printers_2, info,
|
---|
228 | hnd->mem_ctx)){
|
---|
229 | PyErr_SetString(spoolss_error,
|
---|
230 | "error converting printer to info 2");
|
---|
231 | return NULL;
|
---|
232 | }
|
---|
233 |
|
---|
234 | break;
|
---|
235 | case 3:
|
---|
236 | ctr.printers_3 = &pinfo.printers_3;
|
---|
237 |
|
---|
238 | if (!py_to_PRINTER_INFO_3(ctr.printers_3, info,
|
---|
239 | hnd->mem_ctx)) {
|
---|
240 | PyErr_SetString(spoolss_error,
|
---|
241 | "error converting to printer info 3");
|
---|
242 | return NULL;
|
---|
243 | }
|
---|
244 |
|
---|
245 | break;
|
---|
246 | default:
|
---|
247 | PyErr_SetString(spoolss_error, "unsupported info level");
|
---|
248 | return NULL;
|
---|
249 | }
|
---|
250 |
|
---|
251 | /* Call rpc function */
|
---|
252 |
|
---|
253 | werror = rpccli_spoolss_setprinter(
|
---|
254 | hnd->cli, hnd->mem_ctx, &hnd->pol, level, &ctr, 0);
|
---|
255 |
|
---|
256 | /* Return value */
|
---|
257 |
|
---|
258 | if (!W_ERROR_IS_OK(werror)) {
|
---|
259 | PyErr_SetObject(spoolss_werror, py_werror_tuple(werror));
|
---|
260 | return NULL;
|
---|
261 | }
|
---|
262 |
|
---|
263 | Py_INCREF(Py_None);
|
---|
264 | return Py_None;
|
---|
265 | }
|
---|
266 |
|
---|
267 | /* Enumerate printers */
|
---|
268 |
|
---|
269 | PyObject *spoolss_enumprinters(PyObject *self, PyObject *args, PyObject *kw)
|
---|
270 | {
|
---|
271 | WERROR werror;
|
---|
272 | PyObject *result = NULL, *creds = NULL;
|
---|
273 | PRINTER_INFO_CTR ctr;
|
---|
274 | int level = 1, flags = PRINTER_ENUM_LOCAL, i;
|
---|
275 | uint32 num_printers;
|
---|
276 | static char *kwlist[] = {"server", "name", "level", "flags",
|
---|
277 | "creds", NULL};
|
---|
278 | TALLOC_CTX *mem_ctx = NULL;
|
---|
279 | struct cli_state *cli = NULL;
|
---|
280 | char *server, *errstr, *name = NULL;
|
---|
281 |
|
---|
282 | /* Parse parameters */
|
---|
283 |
|
---|
284 | if (!PyArg_ParseTupleAndKeywords(
|
---|
285 | args, kw, "s|siiO", kwlist, &server, &name, &level,
|
---|
286 | &flags, &creds))
|
---|
287 | return NULL;
|
---|
288 |
|
---|
289 | if (server[0] != '\\' || server[1] != '\\') {
|
---|
290 | PyErr_SetString(PyExc_ValueError, "UNC name required");
|
---|
291 | return NULL;
|
---|
292 | }
|
---|
293 |
|
---|
294 | server += 2;
|
---|
295 |
|
---|
296 | if (creds && creds != Py_None && !PyDict_Check(creds)) {
|
---|
297 | PyErr_SetString(PyExc_TypeError,
|
---|
298 | "credentials must be dictionary or None");
|
---|
299 | return NULL;
|
---|
300 | }
|
---|
301 |
|
---|
302 | if (!(cli = open_pipe_creds(server, creds, PI_SPOOLSS, &errstr))) {
|
---|
303 | PyErr_SetString(spoolss_error, errstr);
|
---|
304 | free(errstr);
|
---|
305 | goto done;
|
---|
306 | }
|
---|
307 |
|
---|
308 | if (!(mem_ctx = talloc_init("spoolss_enumprinters"))) {
|
---|
309 | PyErr_SetString(
|
---|
310 | spoolss_error, "unable to init talloc context\n");
|
---|
311 | goto done;
|
---|
312 | }
|
---|
313 |
|
---|
314 | /* This RPC is weird. By setting the server name to different
|
---|
315 | values we can get different behaviour. If however the server
|
---|
316 | name is not specified, we default it to being the full server
|
---|
317 | name as this is probably what the caller intended. To pass a
|
---|
318 | NULL name, pass a value of "" */
|
---|
319 |
|
---|
320 | if (!name)
|
---|
321 | name = server;
|
---|
322 | else {
|
---|
323 | if (!name[0])
|
---|
324 | name = NULL;
|
---|
325 | }
|
---|
326 |
|
---|
327 | /* Call rpc function */
|
---|
328 |
|
---|
329 | werror = rpccli_spoolss_enum_printers(
|
---|
330 | cli->pipe_list, mem_ctx, name, flags, level, &num_printers, &ctr);
|
---|
331 |
|
---|
332 | if (!W_ERROR_IS_OK(werror)) {
|
---|
333 | PyErr_SetObject(spoolss_werror, py_werror_tuple(werror));
|
---|
334 | goto done;
|
---|
335 | }
|
---|
336 |
|
---|
337 | /* Return value */
|
---|
338 |
|
---|
339 | switch (level) {
|
---|
340 | case 0:
|
---|
341 | result = PyDict_New();
|
---|
342 |
|
---|
343 | for (i = 0; i < num_printers; i++) {
|
---|
344 | PyObject *value;
|
---|
345 | fstring s;
|
---|
346 |
|
---|
347 | rpcstr_pull(s, ctr.printers_0[i].printername.buffer,
|
---|
348 | sizeof(fstring), -1, STR_TERMINATE);
|
---|
349 |
|
---|
350 | py_from_PRINTER_INFO_0(&value, &ctr.printers_0[i]);
|
---|
351 |
|
---|
352 | PyDict_SetItemString(
|
---|
353 | value, "level", PyInt_FromLong(0));
|
---|
354 |
|
---|
355 | PyDict_SetItemString(result, s, value);
|
---|
356 | }
|
---|
357 |
|
---|
358 | break;
|
---|
359 | case 1:
|
---|
360 | result = PyDict_New();
|
---|
361 |
|
---|
362 | for(i = 0; i < num_printers; i++) {
|
---|
363 | PyObject *value;
|
---|
364 | fstring s;
|
---|
365 |
|
---|
366 | rpcstr_pull(s, ctr.printers_1[i].name.buffer,
|
---|
367 | sizeof(fstring), -1, STR_TERMINATE);
|
---|
368 |
|
---|
369 | py_from_PRINTER_INFO_1(&value, &ctr.printers_1[i]);
|
---|
370 |
|
---|
371 | PyDict_SetItemString(
|
---|
372 | value, "level", PyInt_FromLong(1));
|
---|
373 |
|
---|
374 | PyDict_SetItemString(result, s, value);
|
---|
375 | }
|
---|
376 |
|
---|
377 | break;
|
---|
378 | case 2:
|
---|
379 | result = PyDict_New();
|
---|
380 |
|
---|
381 | for(i = 0; i < num_printers; i++) {
|
---|
382 | PyObject *value;
|
---|
383 | fstring s;
|
---|
384 |
|
---|
385 | rpcstr_pull(s, ctr.printers_2[i].printername.buffer,
|
---|
386 | sizeof(fstring), -1, STR_TERMINATE);
|
---|
387 |
|
---|
388 | py_from_PRINTER_INFO_2(&value, &ctr.printers_2[i]);
|
---|
389 |
|
---|
390 | PyDict_SetItemString(
|
---|
391 | value, "level", PyInt_FromLong(2));
|
---|
392 |
|
---|
393 | PyDict_SetItemString(result, s, value);
|
---|
394 | }
|
---|
395 |
|
---|
396 | break;
|
---|
397 | default:
|
---|
398 | PyErr_SetString(spoolss_error, "unknown info level");
|
---|
399 | goto done;
|
---|
400 | }
|
---|
401 |
|
---|
402 | done:
|
---|
403 | if (cli)
|
---|
404 | cli_shutdown(cli);
|
---|
405 |
|
---|
406 | if (mem_ctx)
|
---|
407 | talloc_destroy(mem_ctx);
|
---|
408 |
|
---|
409 | return result;
|
---|
410 | }
|
---|
411 |
|
---|
412 | /* Add a printer */
|
---|
413 |
|
---|
414 | PyObject *spoolss_addprinterex(PyObject *self, PyObject *args, PyObject *kw)
|
---|
415 | {
|
---|
416 | static char *kwlist[] = { "server", "printername", "info", "creds",
|
---|
417 | NULL};
|
---|
418 | char *printername, *server, *errstr;
|
---|
419 | PyObject *info, *result = NULL, *creds = NULL;
|
---|
420 | struct cli_state *cli = NULL;
|
---|
421 | TALLOC_CTX *mem_ctx = NULL;
|
---|
422 | PRINTER_INFO_CTR ctr;
|
---|
423 | PRINTER_INFO_2 info2;
|
---|
424 | WERROR werror;
|
---|
425 |
|
---|
426 | if (!PyArg_ParseTupleAndKeywords(
|
---|
427 | args, kw, "ssO!|O!", kwlist, &server, &printername,
|
---|
428 | &PyDict_Type, &info, &PyDict_Type, &creds))
|
---|
429 | return NULL;
|
---|
430 |
|
---|
431 | if (!(cli = open_pipe_creds(server, creds, PI_SPOOLSS, &errstr))) {
|
---|
432 | PyErr_SetString(spoolss_error, errstr);
|
---|
433 | free(errstr);
|
---|
434 | goto done;
|
---|
435 | }
|
---|
436 |
|
---|
437 | if (!(mem_ctx = talloc_init("spoolss_addprinterex"))) {
|
---|
438 | PyErr_SetString(
|
---|
439 | spoolss_error, "unable to init talloc context\n");
|
---|
440 | goto done;
|
---|
441 | }
|
---|
442 |
|
---|
443 | if (!py_to_PRINTER_INFO_2(&info2, info, mem_ctx)) {
|
---|
444 | PyErr_SetString(spoolss_error,
|
---|
445 | "error converting to printer info 2");
|
---|
446 | goto done;
|
---|
447 | }
|
---|
448 |
|
---|
449 | ctr.printers_2 = &info2;
|
---|
450 |
|
---|
451 | werror = rpccli_spoolss_addprinterex(cli->pipe_list, mem_ctx, 2, &ctr);
|
---|
452 |
|
---|
453 | Py_INCREF(Py_None);
|
---|
454 | result = Py_None;
|
---|
455 |
|
---|
456 | done:
|
---|
457 | if (cli)
|
---|
458 | cli_shutdown(cli);
|
---|
459 |
|
---|
460 | if (mem_ctx)
|
---|
461 | talloc_destroy(mem_ctx);
|
---|
462 |
|
---|
463 | return result;
|
---|
464 | }
|
---|