1998-08-09 08:43:13 +08:00
|
|
|
#if !defined(lint)
|
|
|
|
static char copyright[] = "Copyright 1992 The University of Adelaide";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* C O M M A N D
|
|
|
|
*
|
|
|
|
* Author: Mark R. Prior
|
|
|
|
* Communications and Systems Branch
|
|
|
|
* Information Technology Division
|
|
|
|
* The University of Adelaide
|
|
|
|
* E-mail: mrp@itd.adelaide.edu.au
|
|
|
|
* Date: October 1992
|
|
|
|
* Version: 1.8
|
|
|
|
* Description:
|
|
|
|
* Interpret the command sent by the client
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms are permitted
|
|
|
|
* provided that the above copyright notice and this paragraph are
|
|
|
|
* duplicated in all such forms and that any documentation,
|
|
|
|
* advertising materials, and other materials related to such
|
|
|
|
* distribution and use acknowledge that the software was developed
|
|
|
|
* by the University of Adelaide. The name of the University may not
|
|
|
|
* be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "whois++.h"
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
#include <stdlib.h>
|
|
|
|
extern int getdtablesize (void);
|
1998-08-09 08:43:13 +08:00
|
|
|
|
|
|
|
#define isspecial(c) ( (c) == ',' || (c) == ';' || (c) == ':' || (c) == '=' )
|
|
|
|
|
|
|
|
static char **component = NULL;
|
|
|
|
static int numberOfComponents;
|
|
|
|
static int components = 10;
|
|
|
|
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
static int
|
|
|
|
getToken( char *token )
|
1998-08-09 08:43:13 +08:00
|
|
|
{
|
|
|
|
static char *buffer = NULL;
|
|
|
|
static int idx;
|
|
|
|
char ch;
|
|
|
|
fd_set readfds;
|
|
|
|
struct timeval timeout;
|
|
|
|
int i, status, tablesize;
|
|
|
|
|
|
|
|
if ( buffer == NULL ) {
|
|
|
|
tablesize = getdtablesize();
|
1998-08-19 02:19:49 +08:00
|
|
|
|
|
|
|
#ifdef FD_SETSIZE
|
|
|
|
if ( tablesize > FD_SETSIZE ) {
|
|
|
|
tablesize = FD_SETSIZE;
|
|
|
|
}
|
|
|
|
#endif /* FD_SETSIZE */
|
|
|
|
|
1998-08-09 08:43:13 +08:00
|
|
|
timeout.tv_sec = 60;
|
|
|
|
timeout.tv_usec = 0;
|
|
|
|
FD_ZERO( &readfds );
|
|
|
|
FD_SET( fileno( stdin ), &readfds );
|
|
|
|
|
|
|
|
if ( (status = select( tablesize, &readfds, 0, 0, &timeout )) <= 0 ) {
|
|
|
|
if ( status < 0 )
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"select: %s", strerror( errno ) );
|
|
|
|
else
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Connection timed out waiting for input." );
|
|
|
|
exit( 1 );
|
|
|
|
}
|
|
|
|
/**/ /*
|
|
|
|
* We really should determine how many characters are
|
|
|
|
* waiting for us and then malloc that amount rather than
|
|
|
|
* just guessing!
|
|
|
|
*/
|
|
|
|
if ( ( buffer = (char *)malloc(BUFSIZ) ) == NULL
|
|
|
|
|| fgets( buffer, BUFSIZ, stdin ) == NULL ) {
|
|
|
|
*token = '\0';
|
|
|
|
return EOF;
|
|
|
|
}
|
|
|
|
idx = 0;
|
|
|
|
i = strlen( buffer );
|
|
|
|
while ( i-- > 0 && ( buffer[i] == '\r' || buffer[i] == '\n' ) )
|
|
|
|
buffer[i] = '\0';
|
|
|
|
if ( log )
|
|
|
|
syslog( LOG_INFO, "Whois++ Query: %s", buffer );
|
|
|
|
}
|
1999-02-23 01:57:22 +08:00
|
|
|
while ( buffer[idx] != '\0' && isspace( (unsigned char) buffer[idx] ) )
|
1998-08-09 08:43:13 +08:00
|
|
|
idx++;
|
|
|
|
token[0] = buffer[idx++];
|
|
|
|
token[1] = '\0';
|
|
|
|
switch ( token[0] ) {
|
|
|
|
case '\0':
|
|
|
|
strcpy( token, "<end of line>" );
|
|
|
|
free( buffer );
|
|
|
|
buffer = NULL;
|
|
|
|
return EOF;
|
|
|
|
|
|
|
|
case '^':
|
|
|
|
return TEMPLATE;
|
|
|
|
|
|
|
|
case '!':
|
|
|
|
return HANDLE;
|
|
|
|
|
|
|
|
case '.':
|
|
|
|
return ATTRIBUTE;
|
|
|
|
|
|
|
|
case '#':
|
|
|
|
return VALUE;
|
|
|
|
|
|
|
|
case '*':
|
|
|
|
return SEARCH_ALL;
|
|
|
|
|
|
|
|
case '?':
|
|
|
|
return HELP;
|
|
|
|
|
|
|
|
case ':':
|
|
|
|
return COLON;
|
|
|
|
|
|
|
|
case ';':
|
|
|
|
return SEMICOLON;
|
|
|
|
|
|
|
|
case ',':
|
|
|
|
return COMMA;
|
|
|
|
|
|
|
|
case '=':
|
|
|
|
return EQUALS;
|
|
|
|
|
|
|
|
case '"':
|
|
|
|
i = 0;
|
|
|
|
do {
|
|
|
|
ch = buffer[idx++];
|
|
|
|
if ( ch == '\\' && buffer[idx] != '\0' )
|
|
|
|
token[i++] = buffer[idx++];
|
|
|
|
else
|
|
|
|
token[i++] = ch;
|
|
|
|
} while ( ch != '\0' && ch != '"' );
|
|
|
|
if ( ch == '\0' ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Trailing \" missing" );
|
|
|
|
idx--;
|
|
|
|
}
|
|
|
|
token[--i] = '\0';
|
|
|
|
return SEARCH;
|
|
|
|
|
|
|
|
default:
|
|
|
|
i = 1;
|
|
|
|
do {
|
|
|
|
ch = buffer[idx++];
|
|
|
|
if ( ch == '\\' && buffer[idx] != '\0' )
|
|
|
|
token[i++] = buffer[idx++];
|
|
|
|
else
|
|
|
|
token[i++] = ch;
|
1999-02-23 01:57:22 +08:00
|
|
|
} while ( ch != '\0' &&
|
|
|
|
!isspace( (unsigned char) ch ) && !isspecial( ch ) );
|
1998-08-09 08:43:13 +08:00
|
|
|
token[--i] = '\0';
|
|
|
|
idx--;
|
|
|
|
/**/ /*
|
|
|
|
* The following is a brute force lookup, once the names
|
|
|
|
* have settled down this should change to a hash table,
|
|
|
|
* or something similar.
|
|
|
|
*/
|
|
|
|
if ( EQ( token, "help" ) )
|
|
|
|
return HELP;
|
|
|
|
else if ( EQ( token, "list" ) )
|
|
|
|
return LIST;
|
|
|
|
else if ( EQ( token, "show" ) )
|
|
|
|
return SHOW;
|
|
|
|
else if ( EQ( token, "constraints" ) )
|
|
|
|
return CONSTRAINTS;
|
|
|
|
else if ( EQ( token, "describe" ) )
|
|
|
|
return DESCRIBE;
|
|
|
|
else if ( EQ( token, "version" ) )
|
|
|
|
return VERSION;
|
|
|
|
else if ( EQ( token, "template" ) )
|
|
|
|
return TEMPLATE;
|
|
|
|
else if ( EQ( token, "handle" ) )
|
|
|
|
return HANDLE;
|
|
|
|
else if ( EQ( token, "attribute" ) )
|
|
|
|
return ATTRIBUTE;
|
|
|
|
else if ( EQ( token, "value" ) )
|
|
|
|
return VALUE;
|
|
|
|
else if ( EQ( token, "full" ) )
|
|
|
|
return FULL;
|
|
|
|
else if ( EQ( token, "abridged" ) )
|
|
|
|
return ABRIDGED;
|
|
|
|
else if ( EQ( token, "summary" ) )
|
|
|
|
return SUMMARY;
|
|
|
|
else if ( EQ( token, "format" ) )
|
|
|
|
return FORMAT;
|
|
|
|
else if ( EQ( token, "hold" ) )
|
|
|
|
return HOLD;
|
|
|
|
else if ( EQ( token, "maxhits" ) )
|
|
|
|
return MAXHITS;
|
|
|
|
else if ( EQ( token, "match" ) )
|
|
|
|
return MATCH;
|
|
|
|
else if ( EQ( token, "linelength" ) )
|
|
|
|
return LINE_LENGTH;
|
|
|
|
else if ( EQ( token, "command" ) )
|
|
|
|
return COMMAND;
|
|
|
|
else if ( EQ( token, "trace" ) )
|
|
|
|
return TRACE;
|
|
|
|
else
|
|
|
|
return SEARCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
static int
|
|
|
|
term(
|
|
|
|
int token,
|
|
|
|
char *value,
|
|
|
|
char *attribute,
|
|
|
|
int *specifier,
|
|
|
|
int *soundex
|
|
|
|
)
|
1998-08-09 08:43:13 +08:00
|
|
|
{
|
|
|
|
char buffer[BUFSIZ], temp[BUFSIZ];
|
|
|
|
int iterations;
|
|
|
|
|
|
|
|
*soundex = FALSE;
|
|
|
|
switch ( token ) {
|
|
|
|
case ATTRIBUTE: /* . */
|
|
|
|
case VALUE: /* # */
|
|
|
|
case HANDLE: /* ! */
|
|
|
|
case TEMPLATE: /* ^ */
|
|
|
|
case SEARCH_ALL:/* * */
|
|
|
|
*specifier = token;
|
|
|
|
if ( strlen( value ) > 1 ) {
|
|
|
|
/* fullname used, so expect an equals sign */
|
|
|
|
if ( getToken( buffer ) != EQUALS ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"\"=\" expected" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
token = getToken( value );
|
|
|
|
} else
|
|
|
|
token = getToken( value );
|
|
|
|
if ( token != COMMA && token != SEMICOLON && token != EQUALS
|
|
|
|
&& token != COLON && token != EOF ) {
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case COMMA:
|
|
|
|
case SEMICOLON:
|
|
|
|
case EQUALS:
|
|
|
|
case COLON:
|
|
|
|
case EOF:
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Expected search string but got \"%s\"", buffer );
|
|
|
|
return ERROR;
|
|
|
|
|
|
|
|
default:
|
|
|
|
*specifier = SEARCH_ALL;
|
|
|
|
if ( ( token = getToken( buffer ) ) == EQUALS ) {
|
|
|
|
strcpy( attribute, value );
|
|
|
|
token = getToken( value );
|
|
|
|
if ( token == COMMA || token == SEMICOLON
|
|
|
|
|| token == COLON || token == EOF ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Syntax error, string expected." );
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
token = getToken( buffer );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while ( token != COMMA && token != SEMICOLON && token != COLON
|
|
|
|
&& token != EOF ) {
|
|
|
|
if ( *value != '\0' )
|
|
|
|
strcat( value, " " );
|
|
|
|
strcat( value, buffer );
|
|
|
|
token = getToken( buffer );
|
|
|
|
}
|
|
|
|
iterations = 2;
|
|
|
|
while ( token == COMMA ) {
|
|
|
|
token = getToken( buffer );
|
|
|
|
switch ( token ) {
|
|
|
|
case MATCH:
|
|
|
|
iterations = 0;
|
|
|
|
if ( ( token = getToken( buffer ) ) != EQUALS ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"\"=\" expected" );
|
|
|
|
} else
|
|
|
|
token = getToken( buffer );
|
|
|
|
if ( EQ( buffer, "exact" ) )
|
|
|
|
*soundex = FALSE;
|
|
|
|
else if ( EQ( buffer, "fuzzy" ) )
|
|
|
|
*soundex = TRUE;
|
|
|
|
else
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Unrecognised search type" );
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if ( iterations == 0 ) {
|
|
|
|
/* obviously an unrecognised constraint */
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Constraint \"%s\" not supported",
|
|
|
|
buffer );
|
|
|
|
while ( ( token = getToken( buffer ) ) != EOF
|
|
|
|
&& token != COMMA && token != COLON
|
|
|
|
&& token != SEMICOLON )
|
|
|
|
;
|
|
|
|
} else {
|
|
|
|
strcpy( temp, buffer );
|
|
|
|
token = getToken( buffer );
|
|
|
|
if ( token == EQUALS ) {
|
|
|
|
iterations = 0;
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Constraint \"%s\" not supported",
|
|
|
|
buffer );
|
|
|
|
}
|
|
|
|
while ( token != EOF && token != SEMICOLON
|
|
|
|
&& token != COLON && token != COMMA ) {
|
|
|
|
if ( iterations > 0 ) {
|
|
|
|
strcat( temp, " " );
|
|
|
|
strcat( temp, buffer );
|
|
|
|
}
|
|
|
|
token = getToken( buffer );
|
|
|
|
}
|
|
|
|
if ( iterations > 0 ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Assuming \"%s\" part of query and not an unrecognised constraint.", temp );
|
|
|
|
strcat( value, "," );
|
|
|
|
strcat( value, temp );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
iterations--;
|
|
|
|
}
|
|
|
|
if ( *value == '\0' ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Value not specified" );
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
if ( *specifier == NULL )
|
|
|
|
*specifier = SEARCH_ALL;
|
|
|
|
return token;
|
|
|
|
}
|
|
|
|
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
static int
|
|
|
|
processTerm(
|
|
|
|
int specifier,
|
|
|
|
int soundex,
|
|
|
|
char *buffer,
|
|
|
|
char *attribute,
|
|
|
|
char *value
|
|
|
|
)
|
1998-08-09 08:43:13 +08:00
|
|
|
{
|
|
|
|
char *s, *t;
|
|
|
|
char query[BUFSIZ];
|
|
|
|
char **reallocResult;
|
|
|
|
|
|
|
|
switch ( specifier ) {
|
|
|
|
case SEARCH_ALL:
|
|
|
|
if ( numberOfComponents+3 >= components ) {
|
|
|
|
components += 10;
|
|
|
|
reallocResult = (char **)realloc(component, sizeof(char **)*components);
|
|
|
|
if ( reallocResult == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Realloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
component = reallocResult;
|
|
|
|
}
|
|
|
|
if ( attribute != NULL && *attribute != '\0' ) {
|
|
|
|
/* The user obviously knows what they are doing */
|
|
|
|
sprintf( query, "(%s%s%s)", attribute,
|
|
|
|
(soundex)?"~=":"=", buffer );
|
|
|
|
} else {
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
if ( ( s = strchr( buffer, ',' ) ) != NULL ) {
|
1998-08-09 08:43:13 +08:00
|
|
|
*s++ = '\0';
|
1999-02-23 01:57:22 +08:00
|
|
|
while ( *s && isspace( (unsigned char) *s ) )
|
1998-08-09 08:43:13 +08:00
|
|
|
s++;
|
|
|
|
sprintf( query, "(sn%s%s)",
|
|
|
|
(soundex)?"~=":"=", buffer );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
/* let's just make sure there is no title */
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
if ( ( t = strrchr( s, ',' ) ) != NULL ) {
|
1998-08-09 08:43:13 +08:00
|
|
|
*t++ = '\0';
|
1999-02-23 01:57:22 +08:00
|
|
|
while ( *t && isspace( (unsigned char) *t ) )
|
1998-08-09 08:43:13 +08:00
|
|
|
t++;
|
|
|
|
sprintf( query, "(personalTitle%s%s)",
|
|
|
|
(soundex)?"~=":"=", t );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
}
|
|
|
|
sprintf( query, "%s %s", s, buffer );
|
|
|
|
strcpy( buffer, query );
|
|
|
|
} else if ( strncasecmp( buffer, "first ", 6 ) == 0 ) {
|
|
|
|
sprintf( query, "%s *", &buffer[6] );
|
|
|
|
strcpy( buffer, query );
|
|
|
|
}
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
if ( ( s = strchr( buffer, '@' ) ) != NULL ) {
|
1998-08-09 08:43:13 +08:00
|
|
|
*s++ = '\0';
|
|
|
|
if ( *buffer == '\0' ) /* no username */
|
|
|
|
sprintf( query, "(mail=*@%s)", s );
|
|
|
|
else if ( *s == '\0' ) /* no host */
|
|
|
|
sprintf( query, "(|(mail=%s@*)(userid=%s))",
|
|
|
|
buffer, buffer );
|
|
|
|
else
|
|
|
|
sprintf( query, "(mail=%s@%s)",
|
|
|
|
buffer, s );
|
|
|
|
if ( soundex )
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Fuzzy matching not supported on e-mail address queries" );
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
} else if ( strchr( buffer, ' ' ) == NULL ) {
|
1998-08-09 08:43:13 +08:00
|
|
|
sprintf( query,
|
|
|
|
"(|(sn%s%s)(userid%s%s)(l%s%s)(ou%s%s)\
|
|
|
|
(&(cn%s%s)(!(objectClass=person))))",
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer );
|
|
|
|
} else {
|
|
|
|
#if defined(UOFA)
|
|
|
|
sprintf( query, "(|(l%s%s)(ou%s%s)(preferredName%s%s)",
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer );
|
|
|
|
#else
|
|
|
|
sprintf( query, "(|(l%s%s)(ou%s%s)",
|
|
|
|
(soundex)?"~=":"=", buffer,
|
|
|
|
(soundex)?"~=":"=", buffer );
|
|
|
|
#endif
|
|
|
|
/*
|
|
|
|
* If LDAP and/or Quipu didn't strip spaces
|
|
|
|
* then this would be different but as it does
|
|
|
|
* this is easy :-) but it also means we might
|
|
|
|
* get false hits.
|
|
|
|
*/
|
|
|
|
if ( soundex ) {
|
|
|
|
strcat( query, "(cn~=" );
|
|
|
|
strcat( query, buffer );
|
|
|
|
} else {
|
|
|
|
strcat( query, "(cn=*" );
|
|
|
|
strcat( query, strtok( buffer, " " ) );
|
|
|
|
while ( ( s = strtok( NULL, " " ) ) != NULL ) {
|
|
|
|
strcat( query, " * " );
|
|
|
|
strcat( query, s );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
strcat( query, "))" );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ATTRIBUTE:
|
|
|
|
if ( numberOfComponents+1 >= components ) {
|
|
|
|
components += 10;
|
|
|
|
reallocResult = (char **)realloc(component, sizeof(char **)*components);
|
|
|
|
if ( reallocResult == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Realloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
component = reallocResult;
|
|
|
|
}
|
|
|
|
if ( *value != '\0' ) {
|
|
|
|
sprintf( query, "(%s%s%s)", buffer,
|
|
|
|
(soundex)?"~=":"=", value );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
*value = '\0';
|
|
|
|
} else {
|
|
|
|
if ( *attribute != '\0' ) {
|
|
|
|
sprintf( query, "(%s%s*)", attribute,
|
|
|
|
(soundex)?"~=":"=" );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
}
|
|
|
|
strcpy( attribute, buffer );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TEMPLATE:
|
|
|
|
if ( numberOfComponents+1 >= components ) {
|
|
|
|
components += 10;
|
|
|
|
reallocResult = (char **)realloc(component, sizeof(char **)*components);
|
|
|
|
if ( reallocResult == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Realloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
component = reallocResult;
|
|
|
|
}
|
|
|
|
sprintf( query, "(objectClass%s%s)",
|
|
|
|
(soundex)?"~=":"=", templateToObjectClass( buffer ) );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VALUE:
|
|
|
|
if ( *attribute != '\0' ) {
|
|
|
|
if ( numberOfComponents+1 >= components ) {
|
|
|
|
components += 10;
|
|
|
|
reallocResult = (char **)realloc(component, sizeof(char **)*components);
|
|
|
|
if ( reallocResult == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Realloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
component = reallocResult;
|
|
|
|
}
|
|
|
|
sprintf( query, "(%s%s%s)", attribute,
|
|
|
|
(soundex)?"~=":"=", buffer );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
*attribute = '\0';
|
|
|
|
} else {
|
|
|
|
if ( *value != '\0' )
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Ignoring old value (%s)", value );
|
|
|
|
strcpy( value, buffer );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HANDLE:
|
|
|
|
if ( numberOfComponents+1 >= components ) {
|
|
|
|
components += 10;
|
|
|
|
reallocResult = (char **)realloc(component, sizeof(char **)*components);
|
|
|
|
if ( reallocResult == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Realloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
component = reallocResult;
|
|
|
|
}
|
|
|
|
component[numberOfComponents++] = strdup( buffer );
|
|
|
|
return READ;
|
|
|
|
|
|
|
|
}
|
|
|
|
return SEARCH;
|
|
|
|
}
|
|
|
|
|
Protoized, moved extern definitions to .h files, fixed related bugs.
Most function and variable definitions are now preceded by its extern
definition, for error checking. Retyped a number of functions, usually
to return void. Fixed a number of printf format errors.
API changes (in ldap/include):
Added avl_dup_ok, avl_prefixapply, removed ber_fatten (probably typo
for ber_flatten), retyped ldap_sort_strcasecmp, grew lutil.h.
A number of `extern' declarations are left (some added by protoize), to
be cleaned away later. Mostly strdup(), strcasecmp(), mktemp(), optind,
optarg, errno.
1998-11-16 06:40:11 +08:00
|
|
|
int
|
|
|
|
parseCommand( char *query )
|
1998-08-09 08:43:13 +08:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* This procedure reads the string sent by the user and breaks it
|
|
|
|
* down into command to execute.
|
|
|
|
*/
|
|
|
|
char buffer[BUFSIZ], attribute[BUFSIZ], objectClass[BUFSIZ],
|
|
|
|
value[BUFSIZ];
|
|
|
|
char **reallocResult;
|
|
|
|
int command, specificName, length, token, i, j, specifier, soundex;
|
|
|
|
int trace = FALSE;
|
|
|
|
|
|
|
|
switch ( command = getToken( buffer ) ) {
|
|
|
|
case COMMAND:
|
|
|
|
case CONSTRAINTS:
|
|
|
|
case DESCRIBE:
|
|
|
|
case VERSION:
|
|
|
|
/* <command> */
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HELP:
|
|
|
|
case LIST:
|
|
|
|
/* <command> [ <string> ] */
|
|
|
|
if ( ( token = getToken( buffer ) ) != EOF && token != COLON ) {
|
|
|
|
strcpy( query, buffer );
|
|
|
|
token = getToken( buffer );
|
|
|
|
} else
|
|
|
|
*query = '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SHOW:
|
|
|
|
/* "show" <string> */
|
|
|
|
if ( ( token = getToken( buffer ) ) != EOF && token != COLON ) {
|
|
|
|
strcpy( query, buffer );
|
|
|
|
token = getToken( buffer );
|
|
|
|
} else {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Show must have a parameter" );
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* <term> [ ";" <term> ] */
|
|
|
|
*attribute = '\0';
|
|
|
|
*value = '\0';
|
|
|
|
soundex = FALSE;
|
|
|
|
numberOfComponents = 0;
|
|
|
|
if ( ( component = (char **)malloc(sizeof(char **)*components) ) == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Malloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
if ( ( token = term( command, buffer, attribute, &specifier,
|
|
|
|
&soundex ) ) != ERROR )
|
|
|
|
command = processTerm( specifier, soundex, buffer,
|
|
|
|
attribute, value );
|
|
|
|
else
|
|
|
|
return ERROR;
|
|
|
|
if ( token == SEMICOLON ) {
|
|
|
|
if ( command == READ ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Multiple components on a Handle query not supported." );
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
do {
|
|
|
|
soundex = FALSE;
|
|
|
|
token = getToken( buffer );
|
|
|
|
token = term( token, buffer, attribute,
|
|
|
|
&specifier, &soundex );
|
|
|
|
command = processTerm( specifier, soundex,
|
|
|
|
buffer, attribute, value );
|
|
|
|
if ( command == READ ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Multiple components on a Handle query not supported." );
|
|
|
|
return ERROR;
|
|
|
|
} else if ( command == ERROR )
|
|
|
|
return ERROR;
|
|
|
|
} while ( token == SEMICOLON );
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Need to tidy up outstanding single value or attribute terms
|
|
|
|
*/
|
|
|
|
if ( *attribute != '\0' ) {
|
|
|
|
if ( numberOfComponents+1 >= components ) {
|
|
|
|
components += 10;
|
|
|
|
reallocResult = (char **)realloc(component, sizeof(char **)*components);
|
|
|
|
if ( reallocResult == NULL ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Realloc failed" );
|
|
|
|
return ERROR;
|
|
|
|
} else
|
|
|
|
component = reallocResult;
|
|
|
|
}
|
|
|
|
sprintf( query, "(%s%s*)", attribute,
|
|
|
|
(soundex)?"~=":"=" );
|
|
|
|
component[numberOfComponents++] = strdup( query );
|
|
|
|
}
|
|
|
|
if ( *value != '\0' )
|
|
|
|
if ( processTerm( SEARCH_ALL, soundex, value, NULL, NULL ) == ERROR )
|
|
|
|
return ERROR;
|
|
|
|
if ( numberOfComponents == 0 ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"NULL query." );
|
|
|
|
return ERROR;
|
|
|
|
} else if ( numberOfComponents == 1 )
|
|
|
|
strcpy( query, component[0] );
|
|
|
|
else {
|
|
|
|
strcpy( query, "(&" );
|
|
|
|
for ( i = 0; i < numberOfComponents; i++ )
|
|
|
|
strcat( query, component[i] );
|
|
|
|
strcat( query, ")" );
|
|
|
|
}
|
|
|
|
free( component );
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
if ( token == COLON ) { /* global constraints */
|
|
|
|
do {
|
|
|
|
token = getToken( buffer );
|
|
|
|
switch ( token ) {
|
|
|
|
case FORMAT:
|
|
|
|
if ( ( token = getToken( buffer ) ) != EQUALS ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "\"=\" expected" );
|
|
|
|
} else
|
|
|
|
token = getToken( buffer );
|
|
|
|
switch ( token ) {
|
|
|
|
case FULL:
|
|
|
|
case ABRIDGED:
|
|
|
|
case HANDLE:
|
|
|
|
case SUMMARY:
|
|
|
|
if ( outputFormat != NULL )
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "Only one response format can be specified." );
|
|
|
|
else
|
|
|
|
outputFormat = token;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "Unrecognised format specifier" );
|
|
|
|
}
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HOLD:
|
|
|
|
holdConnection = TRUE;
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MAXHITS:
|
|
|
|
if ( ( token = getToken( buffer ) ) != EQUALS ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "\"=\" expected" );
|
|
|
|
} else
|
|
|
|
token = getToken( buffer );
|
|
|
|
if ( (maxHits = atoi( buffer )) < 1
|
|
|
|
|| maxHits > maximumSize ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "Invalid maxhits value, defaulting to %s", maximumSize );
|
|
|
|
maxHits = maximumSize;
|
|
|
|
}
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LANGUAGE:
|
|
|
|
if ( ( token = getToken( buffer ) ) != EQUALS ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "\"=\" expected" );
|
|
|
|
} else
|
|
|
|
token = getToken( buffer );
|
|
|
|
/**/ /* need to save this value and lookup locale */
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Language not currently implemented" );
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LINE_LENGTH:
|
|
|
|
if ( ( token = getToken( buffer ) ) != EQUALS ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "\"=\" expected" );
|
|
|
|
} else
|
|
|
|
token = getToken( buffer );
|
|
|
|
lineLength = atoi( buffer );
|
|
|
|
if ( lineLength < MIN_LINE_LENGTH
|
|
|
|
|| lineLength > MAX_LINE_LENGTH ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "Invalid line length, using default %d", DEFAULT_LINE_LENGTH );
|
|
|
|
lineLength = DEFAULT_LINE_LENGTH;
|
|
|
|
}
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TRACE:
|
|
|
|
trace = TRUE;
|
|
|
|
token = getToken( buffer );
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printFormatted( lineLength, TRUE, stdout, "Unrecognised global constraint \"%s\"", buffer );
|
|
|
|
while ( ( token = getToken( buffer ) ) != EOF
|
|
|
|
&& token != COMMA )
|
|
|
|
;
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
} while ( token == COMMA );
|
|
|
|
}
|
|
|
|
if ( token != EOF ) {
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Data following \"%s\" ignored.", buffer );
|
|
|
|
while ( ( token = getToken( buffer ) ) != EOF )
|
|
|
|
;
|
|
|
|
}
|
|
|
|
if ( trace && ( command == READ || command == SEARCH ) )
|
|
|
|
switch (command) {
|
|
|
|
case READ:
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Attempting to read \"%s\"", query );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SEARCH:
|
|
|
|
printFormatted( lineLength, TRUE, stdout,
|
|
|
|
"Searching using LDAP query %s", query );
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
return command;
|
|
|
|
}
|