2 Unix SMB/CIFS implementation.
4 Simple LDB NTPTR backend
6 Copyright (C) Stefan (metze) Metzmacher 2005
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>.
22 This implements a NTPTR backend that store
23 all objects (Printers, Ports, Monitors, PrinterDrivers ...)
24 in a ldb database, but doesn't do real printing.
26 This is just used for testing how some of
27 the SPOOLSS protocol details should work
31 #include "ntptr/ntptr.h"
32 #include "librpc/gen_ndr/ndr_spoolss.h"
33 #include "lib/ldb/include/ldb.h"
34 #include "auth/auth.h"
35 #include "dsdb/samdb/samdb.h"
37 #include "../lib/util/util_ldb.h"
38 #include "librpc/gen_ndr/dcerpc.h"
39 #include "rpc_server/dcerpc_server.h"
40 #include "rpc_server/common/common.h"
41 #include "param/param.h"
44 connect to the SPOOLSS database
45 return a ldb_context pointer on success, or NULL on failure
47 static struct ldb_context *sptr_db_connect(TALLOC_CTX *mem_ctx, struct tevent_context *ev_ctx, struct loadparm_context *lp_ctx)
49 return ldb_wrap_connect(mem_ctx, ev_ctx, lp_ctx, lpcfg_spoolss_url(lp_ctx), system_session(lp_ctx),
53 static int sptr_db_search(struct ldb_context *ldb,
55 struct ldb_dn *basedn,
56 struct ldb_message ***res,
57 const char * const *attrs,
58 const char *format, ...) PRINTF_ATTRIBUTE(6,7);
60 static int sptr_db_search(struct ldb_context *ldb,
62 struct ldb_dn *basedn,
63 struct ldb_message ***res,
64 const char * const *attrs,
65 const char *format, ...)
71 count = gendb_search_v(ldb, mem_ctx, basedn, res, attrs, format, ap);
77 #define SET_STRING(ldb, mod, attr, value) do { \
78 if (value == NULL) return WERR_INVALID_PARAM; \
79 if (samdb_msg_add_string(ldb, (TALLOC_CTX *)mod, mod, attr, value) != LDB_SUCCESS) { \
84 #define SET_UINT(ldb, mod, attr, value) do { \
85 if (samdb_msg_add_uint(ldb, (TALLOC_CTX *)mod, mod, attr, value) != LDB_SUCCESS) { \
90 static NTSTATUS sptr_init_context(struct ntptr_context *ntptr)
92 struct ldb_context *sptr_db = sptr_db_connect(ntptr, ntptr->ev_ctx, ntptr->lp_ctx);
93 NT_STATUS_HAVE_NO_MEMORY(sptr_db);
95 ntptr->private_data = sptr_db;
100 /* PrintServer functions */
101 static WERROR sptr_OpenPrintServer(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
102 struct spoolss_OpenPrinterEx *r,
103 const char *server_name,
104 struct ntptr_GenericHandle **_server)
106 struct ntptr_GenericHandle *server;
108 /* TODO: do access check here! */
110 server = talloc(mem_ctx, struct ntptr_GenericHandle);
111 W_ERROR_HAVE_NO_MEMORY(server);
113 server->type = NTPTR_HANDLE_SERVER;
114 server->ntptr = ntptr;
115 server->object_name = talloc_strdup(server, server_name);
116 W_ERROR_HAVE_NO_MEMORY(server->object_name);
117 server->access_mask = 0;
118 server->private_data = NULL;
125 * PrintServer PrinterData functions
128 static WERROR sptr_PrintServerData(struct ntptr_GenericHandle *server,
130 const char *value_name,
131 union spoolss_PrinterData *r,
132 enum winreg_Type *type)
134 struct dcerpc_server_info *server_info = lpcfg_dcerpc_server_info(mem_ctx, server->ntptr->lp_ctx);
135 if (strcmp("W3SvcInstalled", value_name) == 0) {
139 } else if (strcmp("BeepEnabled", value_name) == 0) {
143 } else if (strcmp("EventLog", value_name) == 0) {
147 } else if (strcmp("NetPopup", value_name) == 0) {
151 } else if (strcmp("NetPopupToComputer", value_name) == 0) {
155 } else if (strcmp("MajorVersion", value_name) == 0) {
159 } else if (strcmp("MinorVersion", value_name) == 0) {
163 } else if (strcmp("DefaultSpoolDirectory", value_name) == 0) {
165 r->string = "C:\\PRINTERS";
167 } else if (strcmp("Architecture", value_name) == 0) {
169 r->string = SPOOLSS_ARCHITECTURE_NT_X86;
171 } else if (strcmp("DsPresent", value_name) == 0) {
175 } else if (strcmp("OSVersion", value_name) == 0) {
177 enum ndr_err_code ndr_err;
178 struct spoolss_OSVersion os;
180 os.major = server_info->version_major;
181 os.minor = server_info->version_minor;
182 os.build = server_info->version_build;
183 os.extra_string = "";
185 ndr_err = ndr_push_struct_blob(&blob, mem_ctx, &os, (ndr_push_flags_fn_t)ndr_push_spoolss_OSVersion);
186 if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
187 return WERR_GENERAL_FAILURE;
193 } else if (strcmp("OSVersionEx", value_name) == 0) {
195 enum ndr_err_code ndr_err;
196 struct spoolss_OSVersionEx os_ex;
198 os_ex.major = server_info->version_major;
199 os_ex.minor = server_info->version_minor;
200 os_ex.build = server_info->version_build;
201 os_ex.extra_string = "";
202 os_ex.service_pack_major= 0;
203 os_ex.service_pack_minor= 0;
204 os_ex.suite_mask = 0;
205 os_ex.product_type = 0;
208 ndr_err = ndr_push_struct_blob(&blob, mem_ctx, &os_ex, (ndr_push_flags_fn_t)ndr_push_spoolss_OSVersionEx);
209 if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
210 return WERR_GENERAL_FAILURE;
216 } else if (strcmp("DNSMachineName", value_name) == 0) {
217 const char *dnsdomain = lpcfg_dnsdomain(server->ntptr->lp_ctx);
219 if (dnsdomain == NULL) return WERR_INVALID_PARAM;
222 r->string = talloc_asprintf(mem_ctx, "%s.%s",
223 lpcfg_netbios_name(server->ntptr->lp_ctx),
225 W_ERROR_HAVE_NO_MEMORY(r->string);
229 return WERR_INVALID_PARAM;
232 static WERROR sptr_GetPrintServerData(struct ntptr_GenericHandle *server, TALLOC_CTX *mem_ctx,
233 struct spoolss_GetPrinterData *r)
236 union spoolss_PrinterData data;
238 enum ndr_err_code ndr_err;
240 result = sptr_PrintServerData(server, mem_ctx, r->in.value_name, &data, r->out.type);
241 if (!W_ERROR_IS_OK(result)) {
245 ndr_err = ndr_push_union_blob(&blob, mem_ctx,
246 &data, *r->out.type, (ndr_push_flags_fn_t)ndr_push_spoolss_PrinterData);
247 if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
248 return WERR_GENERAL_FAILURE;
251 *r->out.needed = blob.length;
253 if (r->in.offered >= *r->out.needed) {
254 memcpy(r->out.data, blob.data, blob.length);
260 /* PrintServer Form functions */
261 static WERROR sptr_EnumPrintServerForms(struct ntptr_GenericHandle *server, TALLOC_CTX *mem_ctx,
262 struct spoolss_EnumForms *r)
264 struct ldb_context *sptr_db = talloc_get_type(server->ntptr->private_data, struct ldb_context);
265 struct ldb_message **msgs;
268 union spoolss_FormInfo *info;
270 count = sptr_db_search(sptr_db, mem_ctx,
271 ldb_dn_new(mem_ctx, sptr_db, "CN=Forms,CN=PrintServer"),
272 &msgs, NULL, "(&(objectClass=form))");
274 if (count == 0) return WERR_OK;
275 if (count < 0) return WERR_GENERAL_FAILURE;
277 info = talloc_array(mem_ctx, union spoolss_FormInfo, count);
278 W_ERROR_HAVE_NO_MEMORY(info);
280 switch (r->in.level) {
282 for (i=0; i < count; i++) {
283 info[i].info1.flags = ldb_msg_find_attr_as_uint(msgs[i], "flags", SPOOLSS_FORM_BUILTIN);
285 info[i].info1.form_name = ldb_msg_find_attr_as_string(msgs[i], "form-name", NULL);
286 W_ERROR_HAVE_NO_MEMORY(info[i].info1.form_name);
288 info[i].info1.size.width = ldb_msg_find_attr_as_uint(msgs[i], "size-width", 0);
289 info[i].info1.size.height = ldb_msg_find_attr_as_uint(msgs[i], "size-height", 0);
291 info[i].info1.area.left = ldb_msg_find_attr_as_uint(msgs[i], "area-left", 0);
292 info[i].info1.area.top = ldb_msg_find_attr_as_uint(msgs[i], "area-top", 0);
293 info[i].info1.area.right = ldb_msg_find_attr_as_uint(msgs[i], "area-right", 0);
294 info[i].info1.area.bottom = ldb_msg_find_attr_as_uint(msgs[i], "area-bottom", 0);
298 return WERR_UNKNOWN_LEVEL;
302 *r->out.count = count;
306 static WERROR sptr_AddPrintServerForm(struct ntptr_GenericHandle *server, TALLOC_CTX *mem_ctx,
307 struct spoolss_AddForm *r)
309 struct ldb_context *sptr_db = talloc_get_type(server->ntptr->private_data, struct ldb_context);
310 struct ldb_message *msg,**msgs;
311 const char * const attrs[] = {"flags", NULL };
314 /* TODO: do checks access here
315 * if (!(server->access_mask & desired_access)) {
316 * return WERR_FOOBAR;
320 switch (r->in.level) {
322 if (!r->in.info.info1) {
325 count = sptr_db_search(sptr_db, mem_ctx,
326 ldb_dn_new(mem_ctx, sptr_db, "CN=Forms,CN=PrintServer"),
327 &msgs, attrs, "(&(form-name=%s)(objectClass=form))",
328 r->in.info.info1->form_name);
330 if (count == 1) return WERR_FOOBAR;
331 if (count > 1) return WERR_FOOBAR;
332 if (count < 0) return WERR_GENERAL_FAILURE;
334 if (r->in.info.info1->flags != SPOOLSS_FORM_USER) {
338 msg = ldb_msg_new(mem_ctx);
339 W_ERROR_HAVE_NO_MEMORY(msg);
341 /* add core elements to the ldb_message for the Form */
342 msg->dn = ldb_dn_new_fmt(msg, sptr_db, "form-name=%s,CN=Forms,CN=PrintServer", r->in.info.info1->form_name);
343 SET_STRING(sptr_db, msg, "objectClass", "form");
345 SET_UINT(sptr_db, msg, "flags", r->in.info.info1->flags);
347 SET_STRING(sptr_db, msg, "form-name", r->in.info.info1->form_name);
349 SET_UINT(sptr_db, msg, "size-width", r->in.info.info1->size.width);
350 SET_UINT(sptr_db, msg, "size-height", r->in.info.info1->size.height);
352 SET_UINT(sptr_db, msg, "area-left", r->in.info.info1->area.left);
353 SET_UINT(sptr_db, msg, "area-top", r->in.info.info1->area.top);
354 SET_UINT(sptr_db, msg, "area-right", r->in.info.info1->area.right);
355 SET_UINT(sptr_db, msg, "area-bottom", r->in.info.info1->area.bottom);
358 return WERR_UNKNOWN_LEVEL;
361 ret = ldb_add(sptr_db, msg);
369 static WERROR sptr_SetPrintServerForm(struct ntptr_GenericHandle *server, TALLOC_CTX *mem_ctx,
370 struct spoolss_SetForm *r)
372 struct ldb_context *sptr_db = talloc_get_type(server->ntptr->private_data, struct ldb_context);
373 struct ldb_message *msg,**msgs;
374 const char * const attrs[] = { "flags", NULL};
376 enum spoolss_FormFlags flags;
378 /* TODO: do checks access here
379 * if (!(server->access_mask & desired_access)) {
380 * return WERR_FOOBAR;
384 switch (r->in.level) {
386 if (!r->in.info.info1) {
390 count = sptr_db_search(sptr_db, mem_ctx,
391 ldb_dn_new(mem_ctx, sptr_db, "CN=Forms,CN=PrintServer"),
392 &msgs, attrs, "(&(form-name=%s)(objectClass=form))",
393 r->in.info.info1->form_name);
395 if (count == 0) return WERR_FOOBAR;
396 if (count > 1) return WERR_FOOBAR;
397 if (count < 0) return WERR_GENERAL_FAILURE;
399 flags = ldb_msg_find_attr_as_uint(msgs[0], "flags", SPOOLSS_FORM_BUILTIN);
400 if (flags != SPOOLSS_FORM_USER) {
404 msg = ldb_msg_new(mem_ctx);
405 W_ERROR_HAVE_NO_MEMORY(msg);
407 /* add core elements to the ldb_message for the user */
408 msg->dn = msgs[0]->dn;
410 SET_UINT(sptr_db, msg, "flags", r->in.info.info1->flags);
412 SET_STRING(sptr_db, msg, "form-name", r->in.info.info1->form_name);
414 SET_UINT(sptr_db, msg, "size-width", r->in.info.info1->size.width);
415 SET_UINT(sptr_db, msg, "size-height", r->in.info.info1->size.height);
417 SET_UINT(sptr_db, msg, "area-left", r->in.info.info1->area.left);
418 SET_UINT(sptr_db, msg, "area-top", r->in.info.info1->area.top);
419 SET_UINT(sptr_db, msg, "area-right", r->in.info.info1->area.right);
420 SET_UINT(sptr_db, msg, "area-bottom", r->in.info.info1->area.bottom);
423 return WERR_UNKNOWN_LEVEL;
426 ret = dsdb_replace(sptr_db, msg, 0);
434 static WERROR sptr_DeletePrintServerForm(struct ntptr_GenericHandle *server, TALLOC_CTX *mem_ctx,
435 struct spoolss_DeleteForm *r)
437 struct ldb_context *sptr_db = talloc_get_type(server->ntptr->private_data, struct ldb_context);
438 struct ldb_message **msgs;
439 const char * const attrs[] = { "flags", NULL};
441 enum spoolss_FormFlags flags;
443 /* TODO: do checks access here
444 * if (!(server->access_mask & desired_access)) {
445 * return WERR_FOOBAR;
449 if (!r->in.form_name) {
453 count = sptr_db_search(sptr_db, mem_ctx,
454 ldb_dn_new(mem_ctx, sptr_db, "CN=Forms,CN=PrintServer"),
455 &msgs, attrs, "(&(form-name=%s)(objectclass=form))",
458 if (count == 0) return WERR_FOOBAR;
459 if (count > 1) return WERR_FOOBAR;
460 if (count < 0) return WERR_GENERAL_FAILURE;
462 flags = ldb_msg_find_attr_as_uint(msgs[0], "flags", SPOOLSS_FORM_BUILTIN);
463 if (flags != SPOOLSS_FORM_USER) {
467 ret = ldb_delete(sptr_db, msgs[0]->dn);
475 /* PrintServer Driver functions */
476 static WERROR sptr_EnumPrinterDrivers(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
477 struct spoolss_EnumPrinterDrivers *r)
482 static WERROR sptr_GetPrinterDriverDirectory(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
483 struct spoolss_GetPrinterDriverDirectory *r)
485 union spoolss_DriverDirectoryInfo *info;
490 * NOTE: normally r->in.level is 1, but both w2k3 and nt4 sp6a
491 * are ignoring the r->in.level completely, so we do :-)
495 * TODO: check the server name is ours
496 * - if it's a invalid UNC then return WERR_INVALID_NAME
497 * - if it's the wrong host name return WERR_INVALID_PARAM
498 * - if it's "" then we need to return a local WINDOWS path
500 if (!r->in.server || !r->in.server[0]) {
501 prefix = "C:\\DRIVERS";
503 prefix = talloc_asprintf(mem_ctx, "%s\\print$", r->in.server);
504 W_ERROR_HAVE_NO_MEMORY(prefix);
507 if (r->in.environment && strcmp(SPOOLSS_ARCHITECTURE_NT_X86, r->in.environment) == 0) {
510 return WERR_INVALID_ENVIRONMENT;
513 info = talloc(mem_ctx, union spoolss_DriverDirectoryInfo);
514 W_ERROR_HAVE_NO_MEMORY(info);
516 info->info1.directory_name = talloc_asprintf(mem_ctx, "%s\\%s", prefix, postfix);
517 W_ERROR_HAVE_NO_MEMORY(info->info1.directory_name);
523 /* Printer functions */
524 static WERROR sptr_EnumPrinters(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
525 struct spoolss_EnumPrinters *r)
527 struct ldb_context *sptr_db = talloc_get_type(ntptr->private_data, struct ldb_context);
528 struct ldb_message **msgs;
531 union spoolss_PrinterInfo *info;
533 count = sptr_db_search(sptr_db, mem_ctx, NULL, &msgs, NULL,
534 "(&(objectclass=printer))");
536 if (count == 0) return WERR_OK;
537 if (count < 0) return WERR_GENERAL_FAILURE;
539 info = talloc_array(mem_ctx, union spoolss_PrinterInfo, count);
540 W_ERROR_HAVE_NO_MEMORY(info);
542 switch(r->in.level) {
544 for (i = 0; i < count; i++) {
545 info[i].info1.flags = ldb_msg_find_attr_as_uint(msgs[i], "flags", 0);
547 info[i].info1.name = ldb_msg_find_attr_as_string(msgs[i], "name", "");
548 W_ERROR_HAVE_NO_MEMORY(info[i].info1.name);
550 info[i].info1.description = ldb_msg_find_attr_as_string(msgs[i], "description", "");
551 W_ERROR_HAVE_NO_MEMORY(info[i].info1.description);
553 info[i].info1.comment = ldb_msg_find_attr_as_string(msgs[i], "comment", NULL);
557 for (i = 0; i < count; i++) {
558 info[i].info2.servername = ldb_msg_find_attr_as_string(msgs[i], "servername", "");
559 W_ERROR_HAVE_NO_MEMORY(info[i].info2.servername);
561 info[i].info2.printername = ldb_msg_find_attr_as_string(msgs[i], "printername", "");
562 W_ERROR_HAVE_NO_MEMORY(info[i].info2.printername);
564 info[i].info2.sharename = ldb_msg_find_attr_as_string(msgs[i], "sharename", "");
565 W_ERROR_HAVE_NO_MEMORY(info[i].info2.sharename);
567 info[i].info2.portname = ldb_msg_find_attr_as_string(msgs[i], "portname", "");
568 W_ERROR_HAVE_NO_MEMORY(info[i].info2.portname);
570 info[i].info2.drivername = ldb_msg_find_attr_as_string(msgs[i], "drivername", "");
571 W_ERROR_HAVE_NO_MEMORY(info[i].info2.drivername);
573 info[i].info2.comment = ldb_msg_find_attr_as_string(msgs[i], "comment", NULL);
575 info[i].info2.location = ldb_msg_find_attr_as_string(msgs[i], "location", NULL);
577 info[i].info2.devmode = NULL;
579 info[i].info2.sepfile = ldb_msg_find_attr_as_string(msgs[i], "sepfile", NULL);
581 info[i].info2.printprocessor = ldb_msg_find_attr_as_string(msgs[i], "printprocessor", "");
582 W_ERROR_HAVE_NO_MEMORY(info[i].info2.printprocessor);
584 info[i].info2.datatype = ldb_msg_find_attr_as_string(msgs[i], "datatype", "");
585 W_ERROR_HAVE_NO_MEMORY(info[i].info2.datatype);
587 info[i].info2.parameters = ldb_msg_find_attr_as_string(msgs[i], "parameters", NULL);
589 info[i].info2.secdesc = NULL;
591 info[i].info2.attributes = ldb_msg_find_attr_as_uint(msgs[i], "attributes", 0);
592 info[i].info2.priority = ldb_msg_find_attr_as_uint(msgs[i], "priority", 0);
593 info[i].info2.defaultpriority = ldb_msg_find_attr_as_uint(msgs[i], "defaultpriority", 0);
594 info[i].info2.starttime = ldb_msg_find_attr_as_uint(msgs[i], "starttime", 0);
595 info[i].info2.untiltime = ldb_msg_find_attr_as_uint(msgs[i], "untiltime", 0);
596 info[i].info2.status = ldb_msg_find_attr_as_uint(msgs[i], "status", 0);
597 info[i].info2.cjobs = ldb_msg_find_attr_as_uint(msgs[i], "cjobs", 0);
598 info[i].info2.averageppm = ldb_msg_find_attr_as_uint(msgs[i], "averageppm", 0);
602 for (i = 0; i < count; i++) {
603 info[i].info4.printername = ldb_msg_find_attr_as_string(msgs[i], "printername", "");
604 W_ERROR_HAVE_NO_MEMORY(info[i].info2.printername);
606 info[i].info4.servername = ldb_msg_find_attr_as_string(msgs[i], "servername", "");
607 W_ERROR_HAVE_NO_MEMORY(info[i].info2.servername);
609 info[i].info4.attributes = ldb_msg_find_attr_as_uint(msgs[i], "attributes", 0);
613 for (i = 0; i < count; i++) {
614 info[i].info5.printername = ldb_msg_find_attr_as_string(msgs[i], "name", "");
615 W_ERROR_HAVE_NO_MEMORY(info[i].info5.printername);
617 info[i].info5.portname = ldb_msg_find_attr_as_string(msgs[i], "port", "");
618 W_ERROR_HAVE_NO_MEMORY(info[i].info5.portname);
620 info[i].info5.attributes = ldb_msg_find_attr_as_uint(msgs[i], "attributes", 0);
621 info[i].info5.device_not_selected_timeout = ldb_msg_find_attr_as_uint(msgs[i], "device_not_selected_timeout", 0);
622 info[i].info5.transmission_retry_timeout = ldb_msg_find_attr_as_uint(msgs[i], "transmission_retry_timeout", 0);
626 return WERR_UNKNOWN_LEVEL;
630 *r->out.count = count;
634 static WERROR sptr_OpenPrinter(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
635 struct spoolss_OpenPrinterEx *r,
636 const char *printer_name,
637 struct ntptr_GenericHandle **printer)
639 return WERR_INVALID_PRINTER_NAME;
643 static WERROR sptr_EnumPorts(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
644 struct spoolss_EnumPorts *r)
646 struct ldb_context *sptr_db = talloc_get_type(ntptr->private_data, struct ldb_context);
647 struct ldb_message **msgs;
650 union spoolss_PortInfo *info;
652 count = sptr_db_search(sptr_db, mem_ctx, NULL, &msgs, NULL,
653 "(&(objectclass=port))");
655 if (count == 0) return WERR_OK;
656 if (count < 0) return WERR_GENERAL_FAILURE;
658 info = talloc_array(mem_ctx, union spoolss_PortInfo, count);
659 W_ERROR_HAVE_NO_MEMORY(info);
661 switch (r->in.level) {
663 for (i = 0; i < count; i++) {
664 info[i].info1.port_name = ldb_msg_find_attr_as_string(msgs[i], "port-name", "");
665 W_ERROR_HAVE_NO_MEMORY(info[i].info1.port_name);
669 for (i=0; i < count; i++) {
670 info[i].info2.port_name = ldb_msg_find_attr_as_string(msgs[i], "port-name", "");
671 W_ERROR_HAVE_NO_MEMORY(info[i].info2.port_name);
673 info[i].info2.monitor_name = ldb_msg_find_attr_as_string(msgs[i], "monitor-name", "");
674 W_ERROR_HAVE_NO_MEMORY(info[i].info2.monitor_name);
676 info[i].info2.description = ldb_msg_find_attr_as_string(msgs[i], "description", "");
677 W_ERROR_HAVE_NO_MEMORY(info[i].info2.description);
679 info[i].info2.port_type = ldb_msg_find_attr_as_uint(msgs[i], "port-type", SPOOLSS_PORT_TYPE_WRITE);
680 info[i].info2.reserved = ldb_msg_find_attr_as_uint(msgs[i], "reserved", 0);
684 return WERR_UNKNOWN_LEVEL;
688 *r->out.count = count;
692 /* monitor functions */
693 static WERROR sptr_EnumMonitors(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
694 struct spoolss_EnumMonitors *r)
696 struct ldb_context *sptr_db = talloc_get_type(ntptr->private_data, struct ldb_context);
697 struct ldb_message **msgs;
700 union spoolss_MonitorInfo *info;
702 count = sptr_db_search(sptr_db, mem_ctx, NULL, &msgs, NULL,
703 "(&(objectclass=monitor))");
705 if (count == 0) return WERR_OK;
706 if (count < 0) return WERR_GENERAL_FAILURE;
708 info = talloc_array(mem_ctx, union spoolss_MonitorInfo, count);
709 W_ERROR_HAVE_NO_MEMORY(info);
711 switch (r->in.level) {
713 for (i = 0; i < count; i++) {
714 info[i].info1.monitor_name = ldb_msg_find_attr_as_string(msgs[i], "monitor-name", "");
715 W_ERROR_HAVE_NO_MEMORY(info[i].info1.monitor_name);
719 for (i=0; i < count; i++) {
720 info[i].info2.monitor_name = ldb_msg_find_attr_as_string(msgs[i], "monitor-name", "");
721 W_ERROR_HAVE_NO_MEMORY(info[i].info2.monitor_name);
723 info[i].info2.environment = ldb_msg_find_attr_as_string(msgs[i], "environment", "");
724 W_ERROR_HAVE_NO_MEMORY(info[i].info2.environment);
726 info[i].info2.dll_name = ldb_msg_find_attr_as_string(msgs[i], "dll-name", "");
727 W_ERROR_HAVE_NO_MEMORY(info[i].info2.dll_name);
731 return WERR_UNKNOWN_LEVEL;
735 *r->out.count = count;
739 /* Printer Form functions */
740 static WERROR sptr_GetPrinterForm(struct ntptr_GenericHandle *printer, TALLOC_CTX *mem_ctx,
741 struct spoolss_GetForm *r)
743 struct ldb_context *sptr_db = talloc_get_type(printer->ntptr->private_data, struct ldb_context);
744 struct ldb_message **msgs;
745 struct ldb_dn *base_dn;
747 union spoolss_FormInfo *info;
749 /* TODO: do checks access here
750 * if (!(printer->access_mask & desired_access)) {
751 * return WERR_FOOBAR;
755 base_dn = ldb_dn_new_fmt(mem_ctx, sptr_db, "CN=Forms,CN=%s,CN=Printers", printer->object_name);
756 W_ERROR_HAVE_NO_MEMORY(base_dn);
758 count = sptr_db_search(sptr_db, mem_ctx, base_dn, &msgs, NULL,
759 "(&(form-name=%s)(objectClass=form))",
762 if (count == 0) return WERR_FOOBAR;
763 if (count > 1) return WERR_FOOBAR;
764 if (count < 0) return WERR_GENERAL_FAILURE;
766 info = talloc(mem_ctx, union spoolss_FormInfo);
767 W_ERROR_HAVE_NO_MEMORY(info);
769 switch (r->in.level) {
771 info->info1.flags = ldb_msg_find_attr_as_uint(msgs[0], "flags", SPOOLSS_FORM_BUILTIN);
773 info->info1.form_name = ldb_msg_find_attr_as_string(msgs[0], "form-name", NULL);
774 W_ERROR_HAVE_NO_MEMORY(info->info1.form_name);
776 info->info1.size.width = ldb_msg_find_attr_as_uint(msgs[0], "size-width", 0);
777 info->info1.size.height = ldb_msg_find_attr_as_uint(msgs[0], "size-height", 0);
779 info->info1.area.left = ldb_msg_find_attr_as_uint(msgs[0], "area-left", 0);
780 info->info1.area.top = ldb_msg_find_attr_as_uint(msgs[0], "area-top", 0);
781 info->info1.area.right = ldb_msg_find_attr_as_uint(msgs[0], "area-right", 0);
782 info->info1.area.bottom = ldb_msg_find_attr_as_uint(msgs[0], "area-bottom", 0);
785 return WERR_UNKNOWN_LEVEL;
792 static WERROR sptr_GetPrintProcessorDirectory(struct ntptr_context *ntptr, TALLOC_CTX *mem_ctx,
793 struct spoolss_GetPrintProcessorDirectory *r)
795 union spoolss_PrintProcessorDirectoryInfo *info;
800 * NOTE: normally r->in.level is 1, but both w2k3 and nt4 sp6a
801 * are ignoring the r->in.level completely, so we do :-)
805 * TODO: check the server name is ours
806 * - if it's a invalid UNC then return WERR_INVALID_NAME
807 * - if it's the wrong host name return WERR_INVALID_PARAM
808 * - if it's "" then we need to return a local WINDOWS path
810 if (!r->in.server || !r->in.server[0]) {
811 prefix = "C:\\PRTPROCS";
813 prefix = talloc_asprintf(mem_ctx, "%s\\prnproc$", r->in.server);
814 W_ERROR_HAVE_NO_MEMORY(prefix);
817 if (r->in.environment && strcmp(SPOOLSS_ARCHITECTURE_NT_X86, r->in.environment) == 0) {
820 return WERR_INVALID_ENVIRONMENT;
823 info = talloc(mem_ctx, union spoolss_PrintProcessorDirectoryInfo);
824 W_ERROR_HAVE_NO_MEMORY(info);
826 info->info1.directory_name = talloc_asprintf(mem_ctx, "%s\\%s", prefix, postfix);
827 W_ERROR_HAVE_NO_MEMORY(info->info1.directory_name);
835 initialialise the simble ldb backend, registering ourselves with the ntptr subsystem
837 static const struct ntptr_ops ntptr_simple_ldb_ops = {
838 .name = "simple_ldb",
839 .init_context = sptr_init_context,
841 /* PrintServer functions */
842 .OpenPrintServer = sptr_OpenPrintServer,
843 /* .XcvDataPrintServer = sptr_XcvDataPrintServer,
845 /* PrintServer PrinterData functions */
846 /* .EnumPrintServerData = sptr_EnumPrintServerData,
847 */ .GetPrintServerData = sptr_GetPrintServerData,
848 /* .SetPrintServerData = sptr_SetPrintServerData,
849 .DeletePrintServerData = sptr_DeletePrintServerData,
851 /* PrintServer Form functions */
852 .EnumPrintServerForms = sptr_EnumPrintServerForms,
853 .AddPrintServerForm = sptr_AddPrintServerForm,
854 .SetPrintServerForm = sptr_SetPrintServerForm,
855 .DeletePrintServerForm = sptr_DeletePrintServerForm,
857 /* PrintServer Driver functions */
858 .EnumPrinterDrivers = sptr_EnumPrinterDrivers,
859 /* .AddPrinterDriver = sptr_AddPrinterDriver,
860 .DeletePrinterDriver = sptr_DeletePrinterDriver,
861 */ .GetPrinterDriverDirectory = sptr_GetPrinterDriverDirectory,
864 .EnumPorts = sptr_EnumPorts,
865 /* .OpenPort = sptr_OpenPort,
866 .XcvDataPort = sptr_XcvDataPort,
868 /* Monitor functions */
869 .EnumMonitors = sptr_EnumMonitors,
870 /* .OpenMonitor = sptr_OpenMonitor,
871 .XcvDataMonitor = sptr_XcvDataMonitor,
873 /* PrintProcessor functions */
874 /* .EnumPrintProcessors = sptr_EnumPrintProcessors,
876 .GetPrintProcessorDirectory = sptr_GetPrintProcessorDirectory,
878 /* Printer functions */
879 .EnumPrinters = sptr_EnumPrinters,
880 .OpenPrinter = sptr_OpenPrinter,
881 /* .AddPrinter = sptr_AddPrinter,
882 .GetPrinter = sptr_GetPrinter,
883 .SetPrinter = sptr_SetPrinter,
884 .DeletePrinter = sptr_DeletePrinter,
885 .XcvDataPrinter = sptr_XcvDataPrinter,
887 /* Printer Driver functions */
888 /* .GetPrinterDriver = sptr_GetPrinterDriver,
890 /* Printer PrinterData functions */
891 /* .EnumPrinterData = sptr_EnumPrinterData,
892 .GetPrinterData = sptr_GetPrinterData,
893 .SetPrinterData = sptr_SetPrinterData,
894 .DeletePrinterData = sptr_DeletePrinterData,
896 /* Printer Form functions */
897 /* .EnumPrinterForms = sptr_EnumPrinterForms,
898 .AddPrinterForm = sptr_AddPrinterForm,
899 */ .GetPrinterForm = sptr_GetPrinterForm,
900 /* .SetPrinterForm = sptr_SetPrinterForm,
901 .DeletePrinterForm = sptr_DeletePrinterForm,
903 /* Printer Job functions */
904 /* .EnumJobs = sptr_EnumJobs,
905 .AddJob = sptr_AddJob,
906 .ScheduleJob = sptr_ScheduleJob,
907 .GetJob = sptr_GetJob,
908 .SetJob = sptr_SetJob,
910 /* Printer Printing functions */
911 /* .StartDocPrinter = sptr_StartDocPrinter,
912 .EndDocPrinter = sptr_EndDocPrinter,
913 .StartPagePrinter = sptr_StartPagePrinter,
914 .EndPagePrinter = sptr_EndPagePrinter,
915 .WritePrinter = sptr_WritePrinter,
916 .ReadPrinter = sptr_ReadPrinter,
919 NTSTATUS ntptr_simple_ldb_init(void)
923 ret = ntptr_register(&ntptr_simple_ldb_ops);
924 if (!NT_STATUS_IS_OK(ret)) {
925 DEBUG(0,("Failed to register NTPTR '%s' backend!\n",
926 ntptr_simple_ldb_ops.name));