]> git.sur5r.net Git - openldap/blob - tests/progs/slapd-tester.c
handle exceptions; spotted by coverity
[openldap] / tests / progs / slapd-tester.c
1 /* $OpenLDAP$ */
2 /* This work is part of OpenLDAP Software <http://www.openldap.org/>.
3  *
4  * Copyright 1999-2007 The OpenLDAP Foundation.
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted only as authorized by the OpenLDAP
9  * Public License.
10  *
11  * A copy of this license is available in file LICENSE in the
12  * top-level directory of the distribution or, alternatively, at
13  * <http://www.OpenLDAP.org/license.html>.
14  */
15 /* ACKNOWLEDGEMENTS:
16  * This work was initially developed by Kurt Spanier for inclusion
17  * in OpenLDAP Software.
18  */
19
20 #include "portable.h"
21
22 #include <stdio.h>
23
24 #include "ac/stdlib.h"
25
26 #include "ac/ctype.h"
27 #include "ac/dirent.h"
28 #include "ac/param.h"
29 #include "ac/socket.h"
30 #include "ac/string.h"
31 #include "ac/unistd.h"
32 #include "ac/wait.h"
33
34
35 #include "ldap_defaults.h"
36 #include "lutil.h"
37
38 #include "ldap.h"
39 #include "ldap_pvt.h"
40 #include "lber_pvt.h"
41 #include "slapd-common.h"
42
43 #define SEARCHCMD               "slapd-search"
44 #define READCMD                 "slapd-read"
45 #define ADDCMD                  "slapd-addel"
46 #define MODRDNCMD               "slapd-modrdn"
47 #define MODIFYCMD               "slapd-modify"
48 #define BINDCMD                 "slapd-bind"
49 #define MAXARGS                 100
50 #define MAXREQS                 5000
51 #define LOOPS                   100
52 #define OUTERLOOPS              "1"
53 #define RETRIES                 "0"
54
55 #define TSEARCHFILE             "do_search.0"
56 #define TREADFILE               "do_read.0"
57 #define TADDFILE                "do_add."
58 #define TMODRDNFILE             "do_modrdn.0"
59 #define TMODIFYFILE             "do_modify.0"
60 #define TBINDFILE               "do_bind.0"
61
62 static char *get_file_name( char *dirname, char *filename );
63 static int  get_search_filters( char *filename, char *filters[], char *attrs[], char *bases[], LDAPURLDesc *luds[] );
64 static int  get_read_entries( char *filename, char *entries[], char *filters[] );
65 static void fork_child( char *prog, char **args );
66 static void     wait4kids( int nkidval );
67
68 static int      maxkids = 20;
69 static int      nkids;
70
71 #ifdef HAVE_WINSOCK
72 static HANDLE   *children;
73 static char argbuf[BUFSIZ];
74 #define ArgDup(x) strdup(strcat(strcat(strcpy(argbuf,"\""),x),"\""))
75 #else
76 #define ArgDup(x) strdup(x)
77 #endif
78
79 static void
80 usage( char *name, char opt )
81 {
82         if ( opt ) {
83                 fprintf( stderr, "%s: unable to handle option \'%c\'\n\n",
84                         name, opt );
85         }
86
87         fprintf( stderr,
88                 "usage: %s "
89                 "-H <uri> | ([-h <host>] -p <port>) "
90                 "-D <manager> "
91                 "-w <passwd> "
92                 "-d <datadir> "
93                 "[-i <ignore>] "
94                 "[-j <maxchild>] "
95                 "[-l {<loops>|<type>=<loops>[,...]}] "
96                 "[-L <outerloops>] "
97                 "-P <progdir> "
98                 "[-r <maxretries>] "
99                 "[-t <delay>] "
100                 "[-C] "
101                 "[-F] "
102                 "[-I] "
103                 "[-N]\n",
104                 name );
105         exit( EXIT_FAILURE );
106 }
107
108 int
109 main( int argc, char **argv )
110 {
111         int             i, j;
112         char            *uri = NULL;
113         char            *host = "localhost";
114         char            *port = NULL;
115         char            *manager = NULL;
116         char            *passwd = NULL;
117         char            *dirname = NULL;
118         char            *progdir = NULL;
119         int             loops = LOOPS;
120         char            *outerloops = OUTERLOOPS;
121         char            *retries = RETRIES;
122         char            *delay = "0";
123         DIR             *datadir;
124         struct dirent   *file;
125         int             friendly = 0;
126         int             chaserefs = 0;
127         int             noattrs = 0;
128         int             nobind = 0;
129         int             noinit = 1;
130         char            *ignore = NULL;
131         /* search */
132         char            *sfile = NULL;
133         char            *sreqs[MAXREQS];
134         char            *sattrs[MAXREQS];
135         char            *sbase[MAXREQS];
136         LDAPURLDesc     *slud[MAXREQS];
137         int             snum = 0;
138         char            *sargs[MAXARGS];
139         int             sanum;
140         int             sextra_args = 0;
141         char            scmd[MAXPATHLEN];
142         /* static so that its address can be used in initializer below. */
143         static char     sloops[] = "18446744073709551615UL";
144         /* read */
145         char            *rfile = NULL;
146         char            *rreqs[MAXREQS];
147         int             rnum = 0;
148         char            *rargs[MAXARGS];
149         char            *rflts[MAXREQS];
150         int             ranum;
151         int             rextra_args = 0;
152         char            rcmd[MAXPATHLEN];
153         static char     rloops[] = "18446744073709551615UL";
154         /* addel */
155         char            *afiles[MAXREQS];
156         int             anum = 0;
157         char            *aargs[MAXARGS];
158         int             aanum;
159         char            acmd[MAXPATHLEN];
160         static char     aloops[] = "18446744073709551615UL";
161         /* modrdn */
162         char            *nfile = NULL;
163         char            *nreqs[MAXREQS];
164         int             nnum = 0;
165         char            *nargs[MAXARGS];
166         int             nanum;
167         char            ncmd[MAXPATHLEN];
168         static char     nloops[] = "18446744073709551615UL";
169         /* modify */
170         char            *mfile = NULL;
171         char            *mreqs[MAXREQS];
172         char            *mdn[MAXREQS];
173         int             mnum = 0;
174         char            *margs[MAXARGS];
175         int             manum;
176         char            mcmd[MAXPATHLEN];
177         static char     mloops[] = "18446744073709551615UL";
178         /* bind */
179         char            *bfile = NULL;
180         char            *breqs[MAXREQS];
181         char            *bcreds[MAXREQS];
182         char            *battrs[MAXREQS];
183         int             bnum = 0;
184         char            *bargs[MAXARGS];
185         int             banum;
186         char            bcmd[MAXPATHLEN];
187         static char     bloops[] = "18446744073709551615UL";
188         char            **bargs_extra = NULL;
189
190         char            *friendlyOpt = NULL;
191         int             pw_ask = 0;
192         char            *pw_file = NULL;
193
194         /* extra action to do after bind... */
195         typedef struct extra_t {
196                 char            *action;
197                 struct extra_t  *next;
198         }               extra_t;
199
200         extra_t         *extra = NULL;
201         int             nextra = 0;
202
203         tester_init( "slapd-tester", TESTER_TESTER );
204
205         sloops[0] = '\0';
206         rloops[0] = '\0';
207         aloops[0] = '\0';
208         nloops[0] = '\0';
209         mloops[0] = '\0';
210         bloops[0] = '\0';
211
212         while ( ( i = getopt( argc, argv, "AB:CD:d:FH:h:Ii:j:L:l:NP:p:r:t:Ww:y:" ) ) != EOF )
213         {
214                 switch ( i ) {
215                 case 'A':
216                         noattrs++;
217                         break;
218
219                 case 'B': {
220                         char    **p,
221                                 **b = ldap_str2charray( optarg, "," );
222                         extra_t **epp;
223
224                         for ( epp = &extra; *epp; epp = &(*epp)->next )
225                                 ;
226
227                         for ( p = b; p[0]; p++ ) {
228                                 *epp = calloc( 1, sizeof( extra_t ) );
229                                 (*epp)->action = p[0];
230                                 epp = &(*epp)->next;
231                                 nextra++;
232                         }
233
234                         ldap_memfree( b );
235                         } break;
236
237                 case 'C':
238                         chaserefs++;
239                         break;
240
241                 case 'D':               /* slapd manager */
242                         manager = ArgDup( optarg );
243                         break;
244
245                 case 'd':               /* data directory */
246                         dirname = strdup( optarg );
247                         break;
248
249                 case 'F':
250                         friendly++;
251                         break;
252
253                 case 'H':               /* slapd uri */
254                         uri = strdup( optarg );
255                         break;
256
257                 case 'h':               /* slapd host */
258                         host = strdup( optarg );
259                         break;
260
261                 case 'I':
262                         noinit = 0;
263                         break;
264
265                 case 'i':
266                         ignore = optarg;
267                         break;
268
269                 case 'j':               /* the number of parallel clients */
270                         if ( lutil_atoi( &maxkids, optarg ) != 0 ) {
271                                 usage( argv[0], 'j' );
272                         }
273                         break;
274
275                 case 'l':               /* the number of loops per client */
276                         if ( !isdigit( (unsigned char) optarg[0] ) ) {
277                                 char    **p,
278                                         **l = ldap_str2charray( optarg, "," );
279
280                                 for ( p = l; p[0]; p++) {
281                                         struct {
282                                                 struct berval   type;
283                                                 char            *buf;
284                                         } types[] = {
285                                                 { BER_BVC( "add=" ),    aloops },
286                                                 { BER_BVC( "bind=" ),   bloops },
287                                                 { BER_BVC( "modify=" ), mloops },
288                                                 { BER_BVC( "modrdn=" ), nloops },
289                                                 { BER_BVC( "read=" ),   rloops },
290                                                 { BER_BVC( "search=" ), sloops },
291                                                 { BER_BVNULL,           NULL }
292                                         };
293                                         int     c, n;
294
295                                         for ( c = 0; types[c].type.bv_val; c++ ) {
296                                                 if ( strncasecmp( p[0], types[c].type.bv_val, types[c].type.bv_len ) == 0 ) {
297                                                         break;
298                                                 }
299                                         }
300
301                                         if ( types[c].type.bv_val == NULL ) {
302                                                 usage( argv[0], 'l' );
303                                         }
304
305                                         if ( lutil_atoi( &n, &p[0][types[c].type.bv_len] ) != 0 ) {
306                                                 usage( argv[0], 'l' );
307                                         }
308
309                                         snprintf( types[c].buf, sizeof( aloops ), "%d", n );
310                                 }
311
312                                 ldap_charray_free( l );
313
314                         } else if ( lutil_atoi( &loops, optarg ) != 0 ) {
315                                 usage( argv[0], 'l' );
316                         }
317                         break;
318
319                 case 'L':               /* the number of outerloops per client */
320                         outerloops = strdup( optarg );
321                         break;
322
323                 case 'N':
324                         nobind++;
325                         break;
326
327                 case 'P':               /* prog directory */
328                         progdir = strdup( optarg );
329                         break;
330
331                 case 'p':               /* the servers port number */
332                         port = strdup( optarg );
333                         break;
334
335                 case 'r':               /* the number of retries in case of error */
336                         retries = strdup( optarg );
337                         break;
338
339                 case 't':               /* the delay in seconds between each retry */
340                         delay = strdup( optarg );
341                         break;
342
343                 case 'w':               /* the managers passwd */
344                         passwd = ArgDup( optarg );
345                         memset( optarg, '*', strlen( optarg ) );
346                         break;
347
348                 case 'W':
349                         pw_ask++;
350                         break;
351
352                 case 'y':
353                         pw_file = optarg;
354                         break;
355
356                 default:
357                         usage( argv[0], '\0' );
358                         break;
359                 }
360         }
361
362         if (( dirname == NULL ) || ( port == NULL && uri == NULL ) ||
363                         ( manager == NULL ) || ( passwd == NULL ) || ( progdir == NULL ))
364         {
365                 usage( argv[0], '\0' );
366         }
367
368 #ifdef HAVE_WINSOCK
369         children = malloc( maxkids * sizeof(HANDLE) );
370 #endif
371         /* get the file list */
372         if ( ( datadir = opendir( dirname )) == NULL ) {
373                 fprintf( stderr, "%s: couldn't open data directory \"%s\".\n",
374                                         argv[0], dirname );
375                 exit( EXIT_FAILURE );
376         }
377
378         /*  look for search, read, modrdn, and add/delete files */
379         for ( file = readdir( datadir ); file; file = readdir( datadir )) {
380
381                 if ( !strcasecmp( file->d_name, TSEARCHFILE )) {
382                         sfile = get_file_name( dirname, file->d_name );
383                         continue;
384                 } else if ( !strcasecmp( file->d_name, TREADFILE )) {
385                         rfile = get_file_name( dirname, file->d_name );
386                         continue;
387                 } else if ( !strcasecmp( file->d_name, TMODRDNFILE )) {
388                         nfile = get_file_name( dirname, file->d_name );
389                         continue;
390                 } else if ( !strcasecmp( file->d_name, TMODIFYFILE )) {
391                         mfile = get_file_name( dirname, file->d_name );
392                         continue;
393                 } else if ( !strncasecmp( file->d_name, TADDFILE, strlen( TADDFILE ))
394                         && ( anum < MAXREQS )) {
395                         afiles[anum++] = get_file_name( dirname, file->d_name );
396                         continue;
397                 } else if ( !strcasecmp( file->d_name, TBINDFILE )) {
398                         bfile = get_file_name( dirname, file->d_name );
399                         continue;
400                 }
401         }
402
403         closedir( datadir );
404
405         if ( pw_ask ) {
406                 passwd = getpassphrase( _("Enter LDAP Password: ") );
407
408         } else if ( pw_file ) {
409                 struct berval   pw;
410
411                 if ( lutil_get_filed_password( pw_file, &pw ) ) {
412                         exit( EXIT_FAILURE );
413                 }
414
415                 passwd = pw.bv_val;
416         }
417
418         /* look for search requests */
419         if ( sfile ) {
420                 snum = get_search_filters( sfile, sreqs, sattrs, sbase, slud );
421                 if ( snum == -1 ) {
422                         fprintf( stderr, "unable to parse file \"%s\"\n", sfile );
423                         exit( EXIT_FAILURE );
424                 }
425         }
426
427         /* look for read requests */
428         if ( rfile ) {
429                 rnum = get_read_entries( rfile, rreqs, rflts );
430                 if ( rnum == -1 ) {
431                         fprintf( stderr, "unable to parse file \"%s\"\n", rfile );
432                         exit( EXIT_FAILURE );
433                 }
434         }
435
436         /* look for modrdn requests */
437         if ( nfile ) {
438                 nnum = get_read_entries( nfile, nreqs, NULL );
439                 if ( nnum == -1 ) {
440                         fprintf( stderr, "unable to parse file \"%s\"\n", nfile );
441                         exit( EXIT_FAILURE );
442                 }
443         }
444
445         /* look for modify requests */
446         if ( mfile ) {
447                 mnum = get_search_filters( mfile, mreqs, NULL, mdn, NULL );
448                 if ( mnum == -1 ) {
449                         fprintf( stderr, "unable to parse file \"%s\"\n", mfile );
450                         exit( EXIT_FAILURE );
451                 }
452         }
453
454         /* look for bind requests */
455         if ( bfile ) {
456                 bnum = get_search_filters( bfile, bcreds, battrs, breqs, NULL );
457                 if ( bnum == -1 ) {
458                         fprintf( stderr, "unable to parse file \"%s\"\n", bfile );
459                         exit( EXIT_FAILURE );
460                 }
461         }
462
463         /* setup friendly option */
464
465         switch ( friendly ) {
466         case 0:
467                 break;
468
469         case 1:
470                 friendlyOpt = "-F";
471                 break;
472
473         default:
474                 /* NOTE: right now we don't need it more than twice */
475         case 2:
476                 friendlyOpt = "-FF";
477                 break;
478         }
479
480         if ( sloops[0] == '\0' ) snprintf( sloops, sizeof( sloops ), "%d", 10 * loops );
481         if ( rloops[0] == '\0' ) snprintf( rloops, sizeof( rloops ), "%d", 20 * loops );
482         if ( aloops[0] == '\0' ) snprintf( aloops, sizeof( aloops ), "%d", loops );
483         if ( nloops[0] == '\0' ) snprintf( nloops, sizeof( nloops ), "%d", loops );
484         if ( mloops[0] == '\0' ) snprintf( mloops, sizeof( mloops ), "%d", loops );
485         if ( bloops[0] == '\0' ) snprintf( bloops, sizeof( bloops ), "%d", 20 * loops );
486
487         /*
488          * generate the search clients
489          */
490
491         sanum = 0;
492         snprintf( scmd, sizeof scmd, "%s" LDAP_DIRSEP SEARCHCMD,
493                 progdir );
494         sargs[sanum++] = scmd;
495         if ( uri ) {
496                 sargs[sanum++] = "-H";
497                 sargs[sanum++] = uri;
498         } else {
499                 sargs[sanum++] = "-h";
500                 sargs[sanum++] = host;
501                 sargs[sanum++] = "-p";
502                 sargs[sanum++] = port;
503         }
504         sargs[sanum++] = "-D";
505         sargs[sanum++] = manager;
506         sargs[sanum++] = "-w";
507         sargs[sanum++] = passwd;
508         sargs[sanum++] = "-l";
509         sargs[sanum++] = sloops;
510         sargs[sanum++] = "-L";
511         sargs[sanum++] = outerloops;
512         sargs[sanum++] = "-r";
513         sargs[sanum++] = retries;
514         sargs[sanum++] = "-t";
515         sargs[sanum++] = delay;
516         if ( friendly ) {
517                 sargs[sanum++] = friendlyOpt;
518         }
519         if ( chaserefs ) {
520                 sargs[sanum++] = "-C";
521         }
522         if ( noattrs ) {
523                 sargs[sanum++] = "-A";
524         }
525         if ( nobind ) {
526                 sargs[sanum++] = "-N";
527         }
528         if ( ignore ) {
529                 sargs[sanum++] = "-i";
530                 sargs[sanum++] = ignore;
531         }
532         sargs[sanum++] = "-b";
533         sargs[sanum++] = NULL;          /* will hold the search base */
534         sargs[sanum++] = "-s";
535         sargs[sanum++] = NULL;          /* will hold the search scope */
536         sargs[sanum++] = "-f";
537         sargs[sanum++] = NULL;          /* will hold the search request */
538
539         sargs[sanum++] = NULL;
540         sargs[sanum++] = NULL;          /* might hold the "attr" request */
541         sextra_args += 2;
542
543         sargs[sanum] = NULL;
544
545         /*
546          * generate the read clients
547          */
548
549         ranum = 0;
550         snprintf( rcmd, sizeof rcmd, "%s" LDAP_DIRSEP READCMD,
551                 progdir );
552         rargs[ranum++] = rcmd;
553         if ( uri ) {
554                 rargs[ranum++] = "-H";
555                 rargs[ranum++] = uri;
556         } else {
557                 rargs[ranum++] = "-h";
558                 rargs[ranum++] = host;
559                 rargs[ranum++] = "-p";
560                 rargs[ranum++] = port;
561         }
562         rargs[ranum++] = "-D";
563         rargs[ranum++] = manager;
564         rargs[ranum++] = "-w";
565         rargs[ranum++] = passwd;
566         rargs[ranum++] = "-l";
567         rargs[ranum++] = rloops;
568         rargs[ranum++] = "-L";
569         rargs[ranum++] = outerloops;
570         rargs[ranum++] = "-r";
571         rargs[ranum++] = retries;
572         rargs[ranum++] = "-t";
573         rargs[ranum++] = delay;
574         if ( friendly ) {
575                 rargs[ranum++] = friendlyOpt;
576         }
577         if ( chaserefs ) {
578                 rargs[ranum++] = "-C";
579         }
580         if ( noattrs ) {
581                 rargs[ranum++] = "-A";
582         }
583         if ( ignore ) {
584                 rargs[ranum++] = "-i";
585                 rargs[ranum++] = ignore;
586         }
587         rargs[ranum++] = "-e";
588         rargs[ranum++] = NULL;          /* will hold the read entry */
589
590         rargs[ranum++] = NULL;
591         rargs[ranum++] = NULL;          /* might hold the filter arg */
592         rextra_args += 2;
593
594         rargs[ranum] = NULL;
595
596         /*
597          * generate the modrdn clients
598          */
599
600         nanum = 0;
601         snprintf( ncmd, sizeof ncmd, "%s" LDAP_DIRSEP MODRDNCMD,
602                 progdir );
603         nargs[nanum++] = ncmd;
604         if ( uri ) {
605                 nargs[nanum++] = "-H";
606                 nargs[nanum++] = uri;
607         } else {
608                 nargs[nanum++] = "-h";
609                 nargs[nanum++] = host;
610                 nargs[nanum++] = "-p";
611                 nargs[nanum++] = port;
612         }
613         nargs[nanum++] = "-D";
614         nargs[nanum++] = manager;
615         nargs[nanum++] = "-w";
616         nargs[nanum++] = passwd;
617         nargs[nanum++] = "-l";
618         nargs[nanum++] = nloops;
619         nargs[nanum++] = "-L";
620         nargs[nanum++] = outerloops;
621         nargs[nanum++] = "-r";
622         nargs[nanum++] = retries;
623         nargs[nanum++] = "-t";
624         nargs[nanum++] = delay;
625         if ( friendly ) {
626                 nargs[nanum++] = friendlyOpt;
627         }
628         if ( chaserefs ) {
629                 nargs[nanum++] = "-C";
630         }
631         if ( ignore ) {
632                 nargs[nanum++] = "-i";
633                 nargs[nanum++] = ignore;
634         }
635         nargs[nanum++] = "-e";
636         nargs[nanum++] = NULL;          /* will hold the modrdn entry */
637         nargs[nanum] = NULL;
638         
639         /*
640          * generate the modify clients
641          */
642
643         manum = 0;
644         snprintf( mcmd, sizeof mcmd, "%s" LDAP_DIRSEP MODIFYCMD,
645                 progdir );
646         margs[manum++] = mcmd;
647         if ( uri ) {
648                 margs[manum++] = "-H";
649                 margs[manum++] = uri;
650         } else {
651                 margs[manum++] = "-h";
652                 margs[manum++] = host;
653                 margs[manum++] = "-p";
654                 margs[manum++] = port;
655         }
656         margs[manum++] = "-D";
657         margs[manum++] = manager;
658         margs[manum++] = "-w";
659         margs[manum++] = passwd;
660         margs[manum++] = "-l";
661         margs[manum++] = mloops;
662         margs[manum++] = "-L";
663         margs[manum++] = outerloops;
664         margs[manum++] = "-r";
665         margs[manum++] = retries;
666         margs[manum++] = "-t";
667         margs[manum++] = delay;
668         if ( friendly ) {
669                 margs[manum++] = friendlyOpt;
670         }
671         if ( chaserefs ) {
672                 margs[manum++] = "-C";
673         }
674         if ( ignore ) {
675                 margs[manum++] = "-i";
676                 margs[manum++] = ignore;
677         }
678         margs[manum++] = "-e";
679         margs[manum++] = NULL;          /* will hold the modify entry */
680         margs[manum++] = "-a";;
681         margs[manum++] = NULL;          /* will hold the ava */
682         margs[manum] = NULL;
683
684         /*
685          * generate the add/delete clients
686          */
687
688         aanum = 0;
689         snprintf( acmd, sizeof acmd, "%s" LDAP_DIRSEP ADDCMD,
690                 progdir );
691         aargs[aanum++] = acmd;
692         if ( uri ) {
693                 aargs[aanum++] = "-H";
694                 aargs[aanum++] = uri;
695         } else {
696                 aargs[aanum++] = "-h";
697                 aargs[aanum++] = host;
698                 aargs[aanum++] = "-p";
699                 aargs[aanum++] = port;
700         }
701         aargs[aanum++] = "-D";
702         aargs[aanum++] = manager;
703         aargs[aanum++] = "-w";
704         aargs[aanum++] = passwd;
705         aargs[aanum++] = "-l";
706         aargs[aanum++] = aloops;
707         aargs[aanum++] = "-L";
708         aargs[aanum++] = outerloops;
709         aargs[aanum++] = "-r";
710         aargs[aanum++] = retries;
711         aargs[aanum++] = "-t";
712         aargs[aanum++] = delay;
713         if ( friendly ) {
714                 aargs[aanum++] = friendlyOpt;
715         }
716         if ( chaserefs ) {
717                 aargs[aanum++] = "-C";
718         }
719         if ( ignore ) {
720                 aargs[aanum++] = "-i";
721                 aargs[aanum++] = ignore;
722         }
723         aargs[aanum++] = "-f";
724         aargs[aanum++] = NULL;          /* will hold the add data file */
725         aargs[aanum] = NULL;
726
727         /*
728          * generate the bind clients
729          */
730
731         banum = 0;
732         snprintf( bcmd, sizeof bcmd, "%s" LDAP_DIRSEP BINDCMD,
733                 progdir );
734         bargs[banum++] = bcmd;
735         if ( !noinit ) {
736                 bargs[banum++] = "-I";  /* init on each bind */
737         }
738         if ( uri ) {
739                 bargs[banum++] = "-H";
740                 bargs[banum++] = uri;
741         } else {
742                 bargs[banum++] = "-h";
743                 bargs[banum++] = host;
744                 bargs[banum++] = "-p";
745                 bargs[banum++] = port;
746         }
747         bargs[banum++] = "-l";
748         bargs[banum++] = bloops;
749         bargs[banum++] = "-L";
750         bargs[banum++] = outerloops;
751 #if 0
752         bargs[banum++] = "-r";
753         bargs[banum++] = retries;
754         bargs[banum++] = "-t";
755         bargs[banum++] = delay;
756 #endif
757         if ( friendly ) {
758                 bargs[banum++] = friendlyOpt;
759         }
760         if ( chaserefs ) {
761                 bargs[banum++] = "-C";
762         }
763         if ( ignore ) {
764                 bargs[banum++] = "-i";
765                 bargs[banum++] = ignore;
766         }
767         if ( nextra ) {
768                 bargs[banum++] = "-B";
769                 bargs_extra = &bargs[banum++];
770         }
771         bargs[banum++] = "-D";
772         bargs[banum++] = NULL;
773         bargs[banum++] = "-w";
774         bargs[banum++] = NULL;
775         bargs[banum] = NULL;
776
777 #define DOREQ(n,j) ((n) && ((maxkids > (n)) ? ((j) < maxkids ) : ((j) < (n))))
778
779         for ( j = 0; j < MAXREQS; j++ ) {
780                 /* search */
781                 if ( DOREQ( snum, j ) ) {
782                         int     jj = j % snum;
783                         int     x = sanum - sextra_args;
784
785                         /* base */
786                         if ( sbase[jj] != NULL ) {
787                                 sargs[sanum - 7] = sbase[jj];
788
789                         } else {
790                                 sargs[sanum - 7] = slud[jj]->lud_dn;
791                         }
792
793                         /* scope */
794                         if ( slud[jj] != NULL ) {
795                                 sargs[sanum - 5] = (char *)ldap_pvt_scope2str( slud[jj]->lud_scope );
796
797                         } else {
798                                 sargs[sanum - 5] = "sub";
799                         }
800
801                         /* filter */
802                         if ( sreqs[jj] != NULL ) {
803                                 sargs[sanum - 3] = sreqs[jj];
804
805                         } else if ( slud[jj]->lud_filter != NULL ) {
806                                 sargs[sanum - 3] = slud[jj]->lud_filter;
807
808                         } else {
809                                 sargs[sanum - 3] = "(objectClass=*)";
810                         }
811
812                         /* extras */
813                         sargs[x] = NULL;
814
815                         /* attr */
816                         if ( sattrs[jj] != NULL ) {
817                                 sargs[x++] = "-a";
818                                 sargs[x++] = sattrs[jj];
819                         }
820
821                         /* attrs */
822                         if ( slud[jj] != NULL && slud[jj]->lud_attrs != NULL ) {
823                                 int     i;
824
825                                 for ( i = 0; slud[jj]->lud_attrs[ i ] != NULL && x + i < MAXARGS - 1; i++ ) {
826                                         sargs[x + i] = slud[jj]->lud_attrs[ i ];
827                                 }
828                                 sargs[x + i] = NULL;
829                         }
830
831                         fork_child( scmd, sargs );
832                 }
833
834                 /* read */
835                 if ( DOREQ( rnum, j ) ) {
836                         int     jj = j % rnum;
837                         int     x = ranum - rextra_args;
838
839                         rargs[ranum - 3] = rreqs[jj];
840                         if ( rflts[jj] != NULL ) {
841                                 rargs[x++] = "-f";
842                                 rargs[x++] = rflts[jj];
843                         }
844                         rargs[x] = NULL;
845                         fork_child( rcmd, rargs );
846                 }
847
848                 /* rename */
849                 if ( j < nnum ) {
850                         nargs[nanum - 1] = nreqs[j];
851                         fork_child( ncmd, nargs );
852                 }
853
854                 /* modify */
855                 if ( j < mnum ) {
856                         margs[manum - 3] = mdn[j];
857                         margs[manum - 1] = mreqs[j];
858                         fork_child( mcmd, margs );
859                 }
860
861                 /* add/delete */
862                 if ( j < anum ) {
863                         aargs[aanum - 1] = afiles[j];
864                         fork_child( acmd, aargs );
865                 }
866
867                 /* bind */
868                 if ( DOREQ( bnum, j ) ) {
869                         int     jj = j % bnum;
870
871                         if ( nextra ) {
872                                 int     n = ((double)nextra)*rand()/(RAND_MAX + 1.0);
873                                 extra_t *e;
874
875                                 for ( e = extra; n-- > 0; e = e->next )
876                                         ;
877                                 *bargs_extra = e->action;
878                         }
879
880                         if ( battrs[jj] != NULL ) {
881                                 bargs[banum - 3] = manager ? manager : "";
882                                 bargs[banum - 1] = passwd ? passwd : "";
883
884                                 bargs[banum - 2] = "-b";
885                                 bargs[banum - 1] = breqs[jj];
886                                 bargs[banum + 0] = "-f";
887                                 bargs[banum + 1] = bcreds[jj];
888                                 bargs[banum + 2] = "-a";
889                                 bargs[banum + 3] = battrs[jj];
890
891                         } else {
892                                 bargs[banum - 3] = breqs[jj];
893                                 bargs[banum - 1] = bcreds[jj];
894                                 bargs[banum] = NULL;
895                         }
896
897                         fork_child( bcmd, bargs );
898                         bargs[banum] = NULL;
899                 }
900         }
901
902         wait4kids( -1 );
903
904         exit( EXIT_SUCCESS );
905 }
906
907 static char *
908 get_file_name( char *dirname, char *filename )
909 {
910         char buf[MAXPATHLEN];
911
912         snprintf( buf, sizeof buf, "%s" LDAP_DIRSEP "%s",
913                 dirname, filename );
914         return( strdup( buf ));
915 }
916
917
918 static int
919 get_search_filters( char *filename, char *filters[], char *attrs[], char *bases[], LDAPURLDesc *luds[] )
920 {
921         FILE    *fp;
922         int     filter = 0;
923
924         if ( (fp = fopen( filename, "r" )) != NULL ) {
925                 char  line[BUFSIZ];
926
927                 while (( filter < MAXREQS ) && ( fgets( line, BUFSIZ, fp ))) {
928                         char    *nl;
929                         int     got_URL = 0;
930
931                         if (( nl = strchr( line, '\r' )) || ( nl = strchr( line, '\n' )))
932                                 *nl = '\0';
933
934                         if ( luds ) luds[filter] = NULL;
935
936                         if ( luds && strncmp( line, "ldap:///", STRLENOF( "ldap:///" ) ) == 0 ) {
937                                 LDAPURLDesc     *lud;
938
939                                 got_URL = 1;
940                                 bases[filter] = NULL;
941                                 if ( ldap_url_parse( line, &lud ) != LDAP_URL_SUCCESS ) {
942                                         filter = -1;
943                                         break;
944                                 }
945
946                                 if ( lud->lud_dn == NULL || lud->lud_exts != NULL ) {
947                                         filter = -1;
948                                         break;
949                                 }
950
951                                 luds[filter] = lud;
952
953                         } else {
954                                 bases[filter] = ArgDup( line );
955                         }
956                         fgets( line, BUFSIZ, fp );
957                         if (( nl = strchr( line, '\r' )) || ( nl = strchr( line, '\n' )))
958                                 *nl = '\0';
959
960                         filters[filter] = ArgDup( line );
961                         if ( attrs ) {
962                                 if ( filters[filter][0] == '+') {
963                                         char    *sep = strchr( filters[filter], ':' );
964
965                                         attrs[ filter ] = &filters[ filter ][ 1 ];
966                                         if ( sep != NULL ) {
967                                                 sep[ 0 ] = '\0';
968                                                 /* NOTE: don't free this! */
969                                                 filters[ filter ] = &sep[ 1 ];
970                                         }
971
972                                 } else {
973                                         attrs[ filter ] = NULL;
974                                 }
975                         }
976                         filter++;
977
978                 }
979                 fclose( fp );
980         }
981
982         return filter;
983 }
984
985
986 static int
987 get_read_entries( char *filename, char *entries[], char *filters[] )
988 {
989         FILE    *fp;
990         int     entry = 0;
991
992         if ( (fp = fopen( filename, "r" )) != NULL ) {
993                 char  line[BUFSIZ];
994
995                 while (( entry < MAXREQS ) && ( fgets( line, BUFSIZ, fp ))) {
996                         char *nl;
997
998                         if (( nl = strchr( line, '\r' )) || ( nl = strchr( line, '\n' )))
999                                 *nl = '\0';
1000                         if ( filters != NULL && line[0] == '+' ) {
1001                                 LDAPURLDesc     *lud;
1002
1003                                 if ( ldap_url_parse( &line[1], &lud ) != LDAP_URL_SUCCESS ) {
1004                                         entry = -1;
1005                                         break;
1006                                 }
1007
1008                                 if ( lud->lud_dn == NULL || lud->lud_dn[ 0 ] == '\0' ) {
1009                                         ldap_free_urldesc( lud );
1010                                         entry = -1;
1011                                         break;
1012                                 }
1013
1014                                 entries[entry] = ArgDup( lud->lud_dn );
1015
1016                                 if ( lud->lud_filter ) {
1017                                         filters[entry] = ArgDup( lud->lud_filter );
1018
1019                                 } else {
1020                                         filters[entry] = ArgDup( "(objectClass=*)" );
1021                                 }
1022                                 ldap_free_urldesc( lud );
1023
1024                         } else {
1025                                 entries[entry] = ArgDup( line );
1026                         }
1027
1028                         entry++;
1029
1030                 }
1031                 fclose( fp );
1032         }
1033
1034         return( entry );
1035 }
1036
1037 #ifndef HAVE_WINSOCK
1038 static void
1039 fork_child( char *prog, char **args )
1040 {
1041         /* note: obscures global pid var; intended */
1042         pid_t   pid;
1043
1044         wait4kids( maxkids );
1045
1046         switch ( pid = fork() ) {
1047         case 0:         /* child */
1048 #ifdef HAVE_EBCDIC
1049                 /* The __LIBASCII execvp only handles ASCII "prog",
1050                  * we still need to translate the arg vec ourselves.
1051                  */
1052                 { char *arg2[MAXREQS];
1053                 int i;
1054
1055                 for (i=0; args[i]; i++) {
1056                         arg2[i] = ArgDup(args[i]);
1057                         __atoe(arg2[i]);
1058                 }
1059                 arg2[i] = NULL;
1060                 args = arg2; }
1061 #endif
1062                 execvp( prog, args );
1063                 tester_perror( "execvp", NULL );
1064                 exit( EXIT_FAILURE );
1065                 break;
1066
1067         case -1:        /* trouble */
1068                 tester_perror( "fork", NULL );
1069                 break;
1070
1071         default:        /* parent */
1072                 nkids++;
1073                 break;
1074         }
1075 }
1076
1077 static void
1078 wait4kids( int nkidval )
1079 {
1080         int             status;
1081
1082         while ( nkids >= nkidval ) {
1083                 wait( &status );
1084
1085                 if ( WIFSTOPPED(status) ) {
1086                         fprintf( stderr,
1087                             "stopping: child stopped with signal %d\n",
1088                             (int) WSTOPSIG(status) );
1089
1090                 } else if ( WIFSIGNALED(status) ) {
1091                         fprintf( stderr, 
1092                             "stopping: child terminated with signal %d%s\n",
1093                             (int) WTERMSIG(status),
1094 #ifdef WCOREDUMP
1095                                 WCOREDUMP(status) ? ", core dumped" : ""
1096 #else
1097                                 ""
1098 #endif
1099                                 );
1100                         exit( WEXITSTATUS(status)  );
1101
1102                 } else if ( WEXITSTATUS(status) != 0 ) {
1103                         fprintf( stderr, 
1104                             "stopping: child exited with status %d\n",
1105                             (int) WEXITSTATUS(status) );
1106                         exit( WEXITSTATUS(status) );
1107
1108                 } else {
1109                         nkids--;
1110                 }
1111         }
1112 }
1113 #else
1114
1115 static void
1116 wait4kids( int nkidval )
1117 {
1118         int rc, i;
1119
1120         while ( nkids >= nkidval ) {
1121                 rc = WaitForMultipleObjects( nkids, children, FALSE, INFINITE );
1122                 for ( i=rc - WAIT_OBJECT_0; i<nkids-1; i++)
1123                         children[i] = children[i+1];
1124                 nkids--;
1125         }
1126 }
1127
1128 static void
1129 fork_child( char *prog, char **args )
1130 {
1131         int rc;
1132
1133         wait4kids( maxkids );
1134
1135         rc = _spawnvp( _P_NOWAIT, prog, args );
1136
1137         if ( rc == -1 ) {
1138                 tester_perror( "_spawnvp", NULL );
1139         } else {
1140                 children[nkids++] = (HANDLE)rc;
1141         }
1142 }
1143 #endif