mirror of
https://git.openldap.org/openldap/openldap.git
synced 2024-12-15 03:01:09 +08:00
349 lines
7.7 KiB
C
349 lines
7.7 KiB
C
|
/*
|
||
|
* query.c: for rcpt500 (X.500 email query responder)
|
||
|
*
|
||
|
* 18 June 1992 by Mark C Smith
|
||
|
* Copyright (c) 1992 The Regents of The University of Michigan
|
||
|
* All Rights Reserved
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <syslog.h>
|
||
|
#include <string.h>
|
||
|
#include <ctype.h>
|
||
|
#include <sys/time.h>
|
||
|
|
||
|
#include "lber.h"
|
||
|
#include "ldap.h"
|
||
|
#include "portable.h"
|
||
|
#include "ldapconfig.h"
|
||
|
#include "disptmpl.h"
|
||
|
#include "rcpt500.h"
|
||
|
|
||
|
extern int dosyslog;
|
||
|
extern int do_cldap;
|
||
|
extern int rdncount;
|
||
|
extern int derefaliases;
|
||
|
extern int sizelimit;
|
||
|
extern int ldapport;
|
||
|
extern char *ldaphost;
|
||
|
extern char *searchbase;
|
||
|
extern char *dapuser;
|
||
|
extern char *filterfile;
|
||
|
extern char *templatefile;
|
||
|
|
||
|
static char buf[ MAXSIZE ];
|
||
|
static char *errpreface = "Your query failed: ";
|
||
|
|
||
|
extern int strcasecmp();
|
||
|
|
||
|
void close_ldap();
|
||
|
|
||
|
|
||
|
int
|
||
|
query_cmd( msgp, reply )
|
||
|
struct msginfo *msgp;
|
||
|
char *reply;
|
||
|
{
|
||
|
LDAP *ldp;
|
||
|
LDAPMessage *ldmsgp, *entry;
|
||
|
char *s, *dn;
|
||
|
int matches, rc, ufn;
|
||
|
LDAPFiltDesc *lfdp;
|
||
|
LDAPFiltInfo *lfi;
|
||
|
struct ldap_disptmpl *tmpllist = NULL;
|
||
|
static char *attrs[] = { "cn", "title",
|
||
|
#ifdef RCPT500_SORT_ATTR
|
||
|
RCPT500_SORT_ATTR,
|
||
|
#endif
|
||
|
NULL };
|
||
|
|
||
|
ufn = 0;
|
||
|
|
||
|
if ( msgp->msg_arg == NULL ) {
|
||
|
return( help_cmd( msgp, reply ));
|
||
|
}
|
||
|
|
||
|
remove_trailing_space( msgp->msg_arg );
|
||
|
if ( *msgp->msg_arg == '\0' ) {
|
||
|
return( help_cmd( msgp, reply ));
|
||
|
}
|
||
|
|
||
|
if (( lfdp = ldap_init_getfilter( filterfile )) == NULL ) {
|
||
|
strcat( reply, errpreface );
|
||
|
strcat( reply, "filter file configuration error. Try again later." );
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* open connection to LDAP server and bind as dapuser
|
||
|
*/
|
||
|
#ifdef CLDAP
|
||
|
if ( do_cldap )
|
||
|
ldp = cldap_open( ldaphost, ldapport );
|
||
|
else
|
||
|
#endif /* CLDAP */
|
||
|
ldp = ldap_open( ldaphost, ldapport );
|
||
|
|
||
|
if ( ldp == NULL ) {
|
||
|
strcat( reply, errpreface );
|
||
|
strcat( reply, "X.500 service unavailable. Try again later." );
|
||
|
ldap_getfilter_free( lfdp );
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
#ifdef CLDAP
|
||
|
if ( !do_cldap )
|
||
|
#endif /* CLDAP */
|
||
|
if ( ldap_simple_bind_s( ldp, dapuser, NULL ) != LDAP_SUCCESS ) {
|
||
|
report_ldap_err( ldp, reply );
|
||
|
close_ldap( ldp );
|
||
|
ldap_getfilter_free( lfdp );
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* set options for search and build filter
|
||
|
*/
|
||
|
ldp->ld_deref = derefaliases;
|
||
|
ldp->ld_sizelimit = sizelimit;
|
||
|
|
||
|
matches = 0;
|
||
|
|
||
|
#ifdef RCPT500_UFN
|
||
|
#ifdef CLDAP
|
||
|
if ( !do_cldap && strchr( msgp->msg_arg, ',' ) != NULL ) {
|
||
|
#else /* CLDAP */
|
||
|
if ( strchr( msgp->msg_arg, ',' ) != NULL ) {
|
||
|
#endif /* CLDAP */
|
||
|
struct timeval tv;
|
||
|
|
||
|
ldap_ufn_setprefix( ldp, searchbase );
|
||
|
if (( rc = ldap_ufn_search_s( ldp, msgp->msg_arg, attrs, 0, &ldmsgp ))
|
||
|
!= LDAP_SUCCESS && rc != LDAP_SIZELIMIT_EXCEEDED
|
||
|
&& rc != LDAP_TIMELIMIT_EXCEEDED ) {
|
||
|
report_ldap_err( ldp, reply );
|
||
|
close_ldap( ldp );
|
||
|
ldap_getfilter_free( lfdp );
|
||
|
return( 0 );
|
||
|
}
|
||
|
matches = ldap_count_entries( ldp, ldmsgp );
|
||
|
ufn = 1;
|
||
|
} else {
|
||
|
#endif /* RCPT500_UFN */
|
||
|
|
||
|
for ( lfi = ldap_getfirstfilter( lfdp, "rcpt500", msgp->msg_arg );
|
||
|
lfi != NULL; lfi = ldap_getnextfilter( lfdp )) {
|
||
|
#ifdef CLDAP
|
||
|
if ( do_cldap )
|
||
|
rc = cldap_search_s( ldp, searchbase, LDAP_SCOPE_SUBTREE,
|
||
|
lfi->lfi_filter, attrs, 0, &ldmsgp, dapuser );
|
||
|
else
|
||
|
#endif /* CLDAP */
|
||
|
rc = ldap_search_s( ldp, searchbase, LDAP_SCOPE_SUBTREE,
|
||
|
lfi->lfi_filter, attrs, 0, &ldmsgp );
|
||
|
|
||
|
if ( rc != LDAP_SUCCESS && rc != LDAP_SIZELIMIT_EXCEEDED
|
||
|
&& rc != LDAP_TIMELIMIT_EXCEEDED ) {
|
||
|
report_ldap_err( ldp, reply );
|
||
|
close_ldap( ldp );
|
||
|
ldap_getfilter_free( lfdp );
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
if (( matches = ldap_count_entries( ldp, ldmsgp )) != 0 ) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if ( ldmsgp != NULL ) {
|
||
|
ldap_msgfree( ldmsgp );
|
||
|
}
|
||
|
}
|
||
|
#ifdef RCPT500_UFN
|
||
|
}
|
||
|
#endif /* RCPT500_UFN */
|
||
|
|
||
|
if ( matches == 0 ) {
|
||
|
sprintf( buf, "No matches were found for '%s'\n", msgp->msg_arg );
|
||
|
strcat( reply, buf );
|
||
|
close_ldap( ldp );
|
||
|
ldap_getfilter_free( lfdp );
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
if ( ldp->ld_errno == LDAP_TIMELIMIT_EXCEEDED
|
||
|
|| ldp->ld_errno == LDAP_SIZELIMIT_EXCEEDED ) {
|
||
|
strcat( reply, "(Partial results only - a limit was exceeded)\n" );
|
||
|
}
|
||
|
|
||
|
if ( matches <= RCPT500_LISTLIMIT ) {
|
||
|
sprintf( buf, "%d %s match%s found for '%s':\n\n", matches,
|
||
|
ufn ? "UFN" : lfi->lfi_desc,
|
||
|
( matches > 1 ) ? "es" : "", msgp->msg_arg );
|
||
|
strcat( reply, buf );
|
||
|
|
||
|
if (( rc = ldap_init_templates( templatefile, &tmpllist )) != 0 ) {
|
||
|
sprintf( buf, "%s ldap_init_templates( %s ) failed (error %d)\n",
|
||
|
errpreface, templatefile, rc );
|
||
|
strcat( reply, buf );
|
||
|
}
|
||
|
|
||
|
for ( entry = ldap_first_entry( ldp, ldmsgp ); entry != NULL; ) {
|
||
|
dn = ldap_get_dn( ldp, entry );
|
||
|
if ( do_read( ldp, dn, reply, tmpllist ) != LDAP_SUCCESS ) {
|
||
|
report_ldap_err( ldp, reply );
|
||
|
}
|
||
|
free( dn );
|
||
|
if (( entry = ldap_next_entry( ldp, entry )) != NULL ) {
|
||
|
strcat( reply, "\n-------\n\n" );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( tmpllist != NULL ) {
|
||
|
ldap_free_templates( tmpllist );
|
||
|
}
|
||
|
ldap_msgfree( ldmsgp );
|
||
|
|
||
|
} else {
|
||
|
sprintf( buf, "%d %s matches were found for '%s':\n",
|
||
|
matches, ufn ? "UFN" : lfi->lfi_desc, msgp->msg_arg );
|
||
|
strcat( reply, buf );
|
||
|
append_entry_list( reply, msgp->msg_arg, ldp, ldmsgp );
|
||
|
ldap_msgfree( ldmsgp );
|
||
|
}
|
||
|
|
||
|
close_ldap( ldp );
|
||
|
ldap_getfilter_free( lfdp );
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
close_ldap( LDAP *ld )
|
||
|
{
|
||
|
#ifdef CLDAP
|
||
|
if ( do_cldap )
|
||
|
cldap_close( ld );
|
||
|
else
|
||
|
#endif /* CLDAP */
|
||
|
ldap_unbind( ld );
|
||
|
}
|
||
|
|
||
|
|
||
|
append_entry_list( reply, query, ldp, ldmsgp )
|
||
|
char *reply;
|
||
|
char *query;
|
||
|
LDAP *ldp;
|
||
|
LDAPMessage *ldmsgp;
|
||
|
{
|
||
|
LDAPMessage *e;
|
||
|
char *dn, *rdn, *s, **title;
|
||
|
int free_rdn = 0;
|
||
|
|
||
|
#ifdef RCPT500_SORT_ATTR
|
||
|
ldap_sort_entries( ldp, &ldmsgp, RCPT500_SORT_ATTR, strcasecmp );
|
||
|
#endif
|
||
|
|
||
|
for ( e = ldap_first_entry( ldp, ldmsgp ); e != NULL;
|
||
|
e = ldap_next_entry( ldp, e )) {
|
||
|
dn = ldap_get_dn( ldp, e );
|
||
|
if (( s = strchr( dn, ',' )) != NULL ) {
|
||
|
*s = '\0';
|
||
|
}
|
||
|
if (( s = strchr( dn, '=' )) == NULL ) {
|
||
|
rdn = dn;
|
||
|
} else {
|
||
|
rdn = s + 1;
|
||
|
}
|
||
|
|
||
|
#ifdef UOFM
|
||
|
/*
|
||
|
* if this entry's rdn is an exact match for the thing looked up, we
|
||
|
* return the CN that has a digit after it, so that the user is
|
||
|
* returned something guaranteed to yield exactly one match if they
|
||
|
* pick it from the list and query it
|
||
|
*/
|
||
|
|
||
|
if ( strcasecmp( rdn, query ) == 0 ) {
|
||
|
char **cn;
|
||
|
int i;
|
||
|
|
||
|
if (( cn = ldap_get_values( ldp, e, "cn" )) != NULL ) {
|
||
|
for ( i = 0; cn[i] != NULL; i++ ) {
|
||
|
if ( isdigit( *( cn[i] + strlen( cn[i] ) - 1 ))) {
|
||
|
rdn = strdup( cn[i] );
|
||
|
free_rdn = 1;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
ldap_value_free( cn );
|
||
|
}
|
||
|
}
|
||
|
#endif /* UOFM */
|
||
|
|
||
|
title = ldap_get_values( ldp, e, "title" );
|
||
|
sprintf( buf, " %-20s %s\n", rdn, title ? title[0] : "" );
|
||
|
strcat( reply, buf );
|
||
|
if ( title != NULL ) {
|
||
|
ldap_value_free( title );
|
||
|
}
|
||
|
free( dn );
|
||
|
if ( free_rdn ) {
|
||
|
free( rdn );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
int
|
||
|
append_text( reply, text, len )
|
||
|
char *reply;
|
||
|
char *text;
|
||
|
int len;
|
||
|
{
|
||
|
strcat( reply, text );
|
||
|
return( len );
|
||
|
}
|
||
|
|
||
|
|
||
|
int
|
||
|
do_read( ldp, dn, reply, tmpll )
|
||
|
LDAP *ldp;
|
||
|
char *dn;
|
||
|
char *reply;
|
||
|
struct ldap_disptmpl *tmpll;
|
||
|
{
|
||
|
int rc;
|
||
|
static char *maildefvals[] = { "None registered in this service", NULL };
|
||
|
static char *defattrs[] = { "mail", NULL };
|
||
|
static char **defvals[] = { maildefvals, NULL };
|
||
|
|
||
|
|
||
|
rc = ldap_entry2text_search( ldp, dn, searchbase, NULLMSG, tmpll,
|
||
|
defattrs, defvals, (void *)append_text, (void *)reply, "\n",
|
||
|
rdncount, LDAP_DISP_OPT_DOSEARCHACTIONS );
|
||
|
|
||
|
return( rc );
|
||
|
}
|
||
|
|
||
|
|
||
|
report_ldap_err( ldp, reply )
|
||
|
LDAP *ldp;
|
||
|
char *reply;
|
||
|
{
|
||
|
strcat( reply, errpreface );
|
||
|
strcat( reply, ldap_err2string( ldp->ld_errno ));
|
||
|
strcat( reply, "\n" );
|
||
|
}
|
||
|
|
||
|
|
||
|
remove_trailing_space( s )
|
||
|
char *s;
|
||
|
{
|
||
|
char *p = s + strlen( s ) - 1;
|
||
|
|
||
|
while ( isspace( *p ) && p > s ) {
|
||
|
--p;
|
||
|
}
|
||
|
*(++p) = '\0';
|
||
|
}
|