1
0
mirror of https://github.com/samba-team/samba.git synced 2025-11-07 12:23:51 +03:00
Files
samba-mirror/source/cldap_server/rootdse.c
Simo Sorce a580c871d3 r19831: Big ldb_dn optimization and interfaces enhancement patch
This patch changes a lot of the code in ldb_dn.c, and also
removes and add a number of manipulation functions around.

The aim is to avoid validating a dn if not necessary as the
validation code is necessarily slow. This is mainly to speed up
internal operations where input is not user generated and so we
can assume the DNs need no validation. The code is designed to
keep the data as a string if possible.

The code is not yet 100% perfect, but pass all the tests so far.
A memleak is certainly present, I'll work on that next.

Simo.
2007-10-10 14:28:22 -05:00

186 lines
5.1 KiB
C

/*
Unix SMB/CIFS implementation.
CLDAP server - rootdse handling
Copyright (C) Stefan Metzmacher 2006
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "includes.h"
#include "libcli/ldap/ldap.h"
#include "lib/ldb/include/ldb.h"
#include "lib/ldb/include/ldb_errors.h"
#include "lib/events/events.h"
#include "lib/socket/socket.h"
#include "smbd/service_task.h"
#include "cldap_server/cldap_server.h"
#include "librpc/gen_ndr/ndr_misc.h"
#include "dsdb/samdb/samdb.h"
#include "auth/auth.h"
#include "db_wrap.h"
#include "system/network.h"
#include "lib/socket/netif.h"
static void cldapd_rootdse_fill(struct cldapd_server *cldapd,
TALLOC_CTX *mem_ctx,
struct ldap_SearchRequest *search,
struct ldap_SearchResEntry **response,
struct ldap_Result *result)
{
struct ldap_SearchResEntry *ent = NULL;
struct ldb_dn *basedn;
struct ldb_result *res = NULL;
struct ldb_request *lreq;
enum ldb_scope scope = LDB_SCOPE_DEFAULT;
const char **attrs = NULL;
const char *errstr = NULL;
int ret = 0;
int ldb_ret = -1;
basedn = ldb_dn_new(mem_ctx, cldapd->samctx, NULL);
if (basedn == NULL) goto nomem;
scope = LDB_SCOPE_BASE;
if (search->num_attributes >= 1) {
int i;
attrs = talloc_array(mem_ctx, const char *, search->num_attributes+1);
if (attrs == NULL) goto nomem;
for (i=0; i < search->num_attributes; i++) {
attrs[i] = search->attributes[i];
}
attrs[i] = NULL;
}
lreq = talloc(mem_ctx, struct ldb_request);
if (lreq == NULL) goto nomem;
res = talloc_zero(mem_ctx, struct ldb_result);
if (res == NULL) goto nomem;
lreq->operation = LDB_SEARCH;
lreq->op.search.base = basedn;
lreq->op.search.scope = scope;
lreq->op.search.tree = search->tree;
lreq->op.search.attrs = attrs;
lreq->controls = NULL;
lreq->context = res;
lreq->callback = ldb_search_default_callback;
/* Copy the timeout from the incoming call */
ldb_set_timeout(cldapd->samctx, lreq, search->timelimit);
ldb_ret = ldb_request(cldapd->samctx, lreq);
if (ldb_ret != LDB_SUCCESS) {
goto reply;
}
ldb_ret = ldb_wait(lreq->handle, LDB_WAIT_ALL);
if (ldb_ret != LDB_SUCCESS) {
goto reply;
}
if (res->count > 1) {
errstr = "Internal error: to much replies";
ldb_ret = LDB_ERR_OTHER;
goto reply;
} else if (res->count == 1) {
int j;
ent = talloc(mem_ctx, struct ldap_SearchResEntry);
if (ent == NULL) goto nomem;
ent->dn = ldb_dn_linearize(ent, res->msgs[0]->dn);
if (ent->dn == NULL) goto nomem;
ent->num_attributes = 0;
ent->attributes = NULL;
if (res->msgs[0]->num_elements == 0) {
goto reply;
}
ent->num_attributes = res->msgs[0]->num_elements;
ent->attributes = talloc_array(ent, struct ldb_message_element, ent->num_attributes);
if (ent->attributes == NULL) goto nomem;
for (j=0; j < ent->num_attributes; j++) {
ent->attributes[j].name = talloc_steal(ent->attributes, res->msgs[0]->elements[j].name);
ent->attributes[j].num_values = 0;
ent->attributes[j].values = NULL;
if (search->attributesonly && (res->msgs[0]->elements[j].num_values == 0)) {
continue;
}
ent->attributes[j].num_values = res->msgs[0]->elements[j].num_values;
ent->attributes[j].values = res->msgs[0]->elements[j].values;
talloc_steal(ent->attributes, res->msgs[0]->elements[j].values);
}
}
reply:
if (ret) {
/* nothing ... */
} else if (ldb_ret == LDB_SUCCESS) {
ret = LDAP_SUCCESS;
errstr = NULL;
} else {
ret = ldb_ret;
errstr = ldb_errstring(cldapd->samctx);
}
goto done;
nomem:
talloc_free(ent);
ret = LDAP_OPERATIONS_ERROR;
errstr = "No memory";
done:
*response = ent;
result->resultcode = ret;
result->errormessage = (errstr?talloc_strdup(mem_ctx, errstr):NULL);
}
/*
handle incoming cldap requests
*/
void cldapd_rootdse_request(struct cldap_socket *cldap,
uint32_t message_id,
struct ldap_SearchRequest *search,
struct socket_address *src)
{
struct cldapd_server *cldapd = talloc_get_type(cldap->incoming.private, struct cldapd_server);
NTSTATUS status;
struct cldap_reply reply;
struct ldap_Result result;
TALLOC_CTX *tmp_ctx = talloc_new(cldap);
ZERO_STRUCT(result);
reply.messageid = message_id;
reply.dest = src;
reply.response = NULL;
reply.result = &result;
cldapd_rootdse_fill(cldapd, tmp_ctx, search, &reply.response, reply.result);
status = cldap_reply_send(cldap, &reply);
if (!NT_STATUS_IS_OK(status)) {
DEBUG(2,("cldap rootdse query failed '%s' - %s\n",
ldb_filter_from_tree(tmp_ctx, search->tree), nt_errstr(status)));
}
talloc_free(tmp_ctx);
return;
}