/* bconfig.c - the config backend */
/* $OpenLDAP$ */
/* This work is part of OpenLDAP Software .
*
* Copyright 2005 The OpenLDAP Foundation.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted only as authorized by the OpenLDAP
* Public License.
*
* A copy of this license is available in the file LICENSE in the
* top-level directory of the distribution or, alternatively, at
* .
*/
/* ACKNOWLEDGEMENTS:
* This work was originally developed by Howard Chu for inclusion
* in OpenLDAP Software.
*/
#include "portable.h"
#include
#include
#include "slap.h"
#include "config.h"
#define CONFIG_DN "cn=config"
typedef struct CfEntryInfo {
struct CfEntryInfo *ce_sibs;
struct CfEntryInfo *ce_kids;
Entry *ce_entry;
ConfigTable *ce_table;
} CfEntryInfo;
typedef struct {
ConfigFile *cb_config;
CfEntryInfo *cb_root;
} CfBackInfo;
static int
config_back_bind( Operation *op, SlapReply *rs )
{
if ( op->orb_method == LDAP_AUTH_SIMPLE && be_isroot_pw( op )) {
ber_dupbv( &op->orb_edn, be_root_dn( op->o_bd ));
/* frontend sends result */
return LDAP_SUCCESS;
}
rs->sr_err = LDAP_INVALID_CREDENTIALS;
send_ldap_result( op, rs );
return rs->sr_err;
}
static CfEntryInfo *
config_find_base( CfEntryInfo *root, struct berval *dn, CfEntryInfo **last )
{
struct berval cdn;
char *c;
if ( dn_match( &root->ce_entry->e_nname, dn ))
return root;
c = dn->bv_val+dn->bv_len;
for (;*c != ',';c--);
while(root) {
*last = root;
for (--c;c>dn->bv_val && *c != ',';c--);
if ( *c == ',' )
c++;
cdn.bv_val = c;
cdn.bv_len = dn->bv_len - (c-dn->bv_val);
root = root->ce_kids;
for (;root;root=root->ce_sibs) {
if ( dn_match( &root->ce_entry->e_nname, &cdn )) {
if ( cdn.bv_val == dn->bv_val ) {
return root;
}
break;
}
}
}
return root;
}
static int
config_send( Operation *op, SlapReply *rs, CfEntryInfo *ce, int depth )
{
int rc = 0;
if ( test_filter( op, ce->ce_entry, op->ors_filter ) == LDAP_COMPARE_TRUE )
{
rs->sr_attrs = op->ors_attrs;
rs->sr_entry = ce->ce_entry;
rc = send_search_entry( op, rs );
}
if ( op->ors_scope == LDAP_SCOPE_SUBTREE ) {
if ( ce->ce_kids ) {
rc = config_send( op, rs, ce->ce_kids, 1 );
if ( rc ) return rc;
}
if ( depth ) {
for (ce=ce->ce_sibs; ce; ce=ce->ce_sibs) {
rc = config_send( op, rs, ce, 0 );
if ( rc ) break;
}
}
}
return rc;
}
static int
config_back_search( Operation *op, SlapReply *rs )
{
CfBackInfo *cfb;
CfEntryInfo *ce, *last;
int rc;
if ( !be_isroot( op ) ) {
rs->sr_err = LDAP_INSUFFICIENT_ACCESS;
send_ldap_result( op, rs );
}
cfb = (CfBackInfo *)op->o_bd->be_private;
ce = config_find_base( cfb->cb_root, &op->o_req_ndn, &last );
if ( !ce ) {
if ( last )
rs->sr_matched = last->ce_entry->e_name.bv_val;
rs->sr_err = LDAP_NO_SUCH_OBJECT;
goto out;
}
switch ( op->ors_scope ) {
case LDAP_SCOPE_BASE:
case LDAP_SCOPE_SUBTREE:
config_send( op, rs, ce, 0 );
break;
case LDAP_SCOPE_ONELEVEL:
for (ce = ce->ce_kids; ce; ce=ce->ce_sibs) {
config_send( op, rs, ce, 1 );
}
break;
}
rs->sr_err = LDAP_SUCCESS;
out:
send_ldap_result( op, rs );
return 0;
}
static Entry *
config_alloc_entry( struct berval *pdn, struct berval *rdn )
{
Entry *e = ch_calloc( 1, sizeof(Entry) );
CfEntryInfo *ce = ch_calloc( 1, sizeof(CfEntryInfo) );
e->e_private = ce;
ce->ce_entry = e;
build_new_dn( &e->e_name, pdn, rdn, NULL );
ber_dupbv( &e->e_nname, &e->e_name );
return e;
}
static int
config_build_entry( Entry *e, void *private, char *oc, struct berval *rdn )
{
struct berval vals[2];
struct berval ad_name;
AttributeDescription *ad = NULL;
int rc;
char *ptr;
const char *text;
BER_BVZERO( &vals[1] );
ber_str2bv( oc, 0, 0, &vals[0] );
attr_merge(e, slap_schema.si_ad_objectClass, vals, NULL );
ptr = strchr(rdn->bv_val, '=');
ad_name.bv_val = rdn->bv_val;
ad_name.bv_len = ptr - rdn->bv_val;
rc = slap_bv2ad( &ad_name, &ad, &text );
if ( rc ) {
return rc;
}
vals[0].bv_val = ptr+1;
vals[0].bv_len = rdn->bv_len - (vals[0].bv_val - rdn->bv_val);
attr_merge(e, ad, vals, NULL );
return 0;
}
static int
config_back_db_open( BackendDB *be )
{
CfBackInfo *cfb = be->be_private;
struct berval rdn;
Entry *e, *parent;
CfEntryInfo *ce, *ceparent, *ceprev;
int i, buflen = 0;
char *buf = NULL;
BackendInfo *bi;
BackendDB *bptr;
/* create root of tree */
ber_str2bv( CONFIG_DN, STRLENOF( CONFIG_DN ), 0, &rdn );
e = config_alloc_entry( NULL, &rdn );
ce = e->e_private;
ce->ce_table = be->bd_info->bi_cf_table;
cfb->cb_root = ce;
config_build_entry( e, be->be_private, "olcGlobal", &rdn );
parent = e;
ceparent = ce;
/* Create backend nodes. Skip if they don't provide a cf_table.
* There usually aren't any of these.
*/
bi = backendInfo;
for (i=0; ibi_cf_table) continue;
if (!bi->bi_private) continue;
if ( buflen < STRLENOF("olcbackend=")+strlen(bi->bi_type)+1) {
buflen = STRLENOF("olcbackend=") + strlen(bi->bi_type)+1;
buf = realloc(buf, buflen);
}
rdn.bv_val = buf;
rdn.bv_len = sprintf(buf, "olcBackend=%s", bi->bi_type);
e = config_alloc_entry( &parent->e_nname, &rdn );
ce = e->e_private;
ce->ce_table = bi->bi_cf_table;
config_build_entry( e, bi->bi_private, "olcBackendConfig",
&rdn );
if ( !ceparent->ce_kids ) {
ceparent->ce_kids = ce;
} else {
ceprev->ce_sibs = ce;
}
ceprev = ce;
}
/* Create database nodes... */
for (i=0; ibd_info;
if ( buflen < STRLENOF("olcdatabase={xxxxxxxx}")+strlen(bi->bi_type)+1) {
buflen = STRLENOF("olcdatabase={xxxxxxxx}")+strlen(bi->bi_type)+1;
buf = realloc(buf, buflen);
}
rdn.bv_val = buf;
rdn.bv_len = sprintf(buf, "olcDatabase={%0x}%s", i, bi->bi_type);
e = config_alloc_entry( &parent->e_nname, &rdn );
ce = e->e_private;
ce->ce_table = bptr->be_cf_table;
config_build_entry( e, bptr->be_private, "olcDatabaseConfig",
&rdn );
if ( !ceparent->ce_kids ) {
ceparent->ce_kids = ce;
} else {
ceprev->ce_sibs = ce;
}
ceprev = ce;
/* Iterate through overlays */
}
/* Create includeFile nodes... */
return 0;
}
static int
config_back_db_destroy( Backend *be )
{
free( be->be_private );
return 0;
}
int
config_back_initialize( BackendInfo *bi )
{
bi->bi_open = 0;
bi->bi_close = 0;
bi->bi_config = 0;
bi->bi_destroy = 0;
bi->bi_db_init = 0;
bi->bi_db_config = 0;
bi->bi_db_open = config_back_db_open;
bi->bi_db_close = 0;
bi->bi_db_destroy = config_back_db_destroy;
bi->bi_op_bind = config_back_bind;
bi->bi_op_unbind = 0;
bi->bi_op_search = config_back_search;
bi->bi_op_compare = 0;
bi->bi_op_modify = 0;
bi->bi_op_modrdn = 0;
bi->bi_op_add = 0;
bi->bi_op_delete = 0;
bi->bi_op_abandon = 0;
bi->bi_extended = 0;
bi->bi_chk_referrals = 0;
bi->bi_connection_init = 0;
bi->bi_connection_destroy = 0;
return 0;
}
void config_back_init( ConfigFile *cfp, ConfigTable *ct )
{
BackendInfo bi = {0};
BackendDB *be;
struct berval dn;
CfBackInfo *cfb;
bi.bi_type = "config";
bi.bi_init = config_back_initialize;
bi.bi_cf_table = ct;
backend_add( &bi );
be = backend_db_init( bi.bi_type );
ber_str2bv( CONFIG_DN, 0, 1, &be->be_rootdn );
ber_dupbv( &be->be_rootndn, &be->be_rootdn );
ber_dupbv( &dn, &be->be_rootdn );
ber_bvarray_add( &be->be_suffix, &dn );
ber_dupbv( &dn, &be->be_rootdn );
ber_bvarray_add( &be->be_nsuffix, &dn );
cfb = ch_calloc( 1, sizeof(CfBackInfo));
cfb->cb_config = cfp;
be->be_private = cfb;
}