2005-02-21 02:33:40 +08:00
|
|
|
/* bconfig.c - the config backend */
|
|
|
|
/* $OpenLDAP$ */
|
|
|
|
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* <http://www.OpenLDAP.org/license.html>.
|
|
|
|
*/
|
|
|
|
/* ACKNOWLEDGEMENTS:
|
|
|
|
* This work was originally developed by Howard Chu for inclusion
|
|
|
|
* in OpenLDAP Software.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "portable.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <ac/string.h>
|
|
|
|
|
|
|
|
#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;
|
2005-02-21 10:21:29 +08:00
|
|
|
int i, rc, buflen = 0;
|
2005-02-21 02:33:40 +08:00
|
|
|
char *buf = NULL;
|
|
|
|
BackendInfo *bi;
|
|
|
|
BackendDB *bptr;
|
2005-02-21 10:21:29 +08:00
|
|
|
ConfigArgs c;
|
|
|
|
ConfigTable *ct;
|
2005-02-21 02:33:40 +08:00
|
|
|
|
|
|
|
/* 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 );
|
2005-02-21 10:21:29 +08:00
|
|
|
c.be = be;
|
|
|
|
c.bi = be->bd_info;
|
|
|
|
ct = ce->ce_table;
|
|
|
|
for (ct=ce->ce_table; ct->name; ct++) {
|
|
|
|
if (!ct->ad) continue;
|
|
|
|
if (ct->arg_type & ARG_DB) continue;
|
|
|
|
rc = config_get_vals(ct, &c);
|
|
|
|
if (rc == LDAP_SUCCESS) {
|
|
|
|
attr_merge(e, ct->ad, c.rvalue_vals, c.rvalue_nvals);
|
|
|
|
}
|
|
|
|
}
|
2005-02-21 02:33:40 +08:00
|
|
|
|
|
|
|
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; i<nBackendInfo; i++, bi++) {
|
|
|
|
if (!bi->bi_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; i<nBackendDB; i++) {
|
|
|
|
if ( i == 0 ) {
|
|
|
|
bptr = frontendDB;
|
|
|
|
} else {
|
|
|
|
bptr = &backendDB[i];
|
|
|
|
}
|
|
|
|
bi = bptr->bd_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 );
|
2005-02-21 10:21:29 +08:00
|
|
|
c.be = bptr;
|
|
|
|
c.bi = bi;
|
|
|
|
ct = be->bd_info->bi_cf_table;
|
|
|
|
for (; ct->name; ct++) {
|
|
|
|
if (!ct->ad) continue;
|
|
|
|
if (!(ct->arg_type & (ARG_DB|ARG_MAY_DB))) continue;
|
|
|
|
rc = config_get_vals(ct, &c);
|
|
|
|
if (rc == LDAP_SUCCESS) {
|
|
|
|
attr_merge(e, ct->ad, c.rvalue_vals, c.rvalue_nvals);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-02-21 02:33:40 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2005-02-21 19:28:59 +08:00
|
|
|
|
2005-02-21 02:33:40 +08:00
|
|
|
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;
|
|
|
|
}
|