1 /* schema_init.c - init builtin schema */
4 * Copyright 1998-1999 The OpenLDAP Foundation, All Rights Reserved.
5 * COPYING RESTRICTIONS APPLY, see COPYRIGHT file
13 #include <ac/string.h>
14 #include <ac/socket.h>
24 /* any value allowed */
35 unsigned char *u = in->bv_val;
37 for( count = in->bv_len; count > 0; count+=len, u+=len ) {
38 /* get the length indicated by the first byte */
39 len = LDAP_UTF8_CHARLEN( u );
41 /* should not be zero */
42 if( len == 0 ) return -1;
44 /* make sure len corresponds with the offset
45 to the next character */
46 if( LDAP_UTF8_OFFSET( u ) != len ) return -1;
49 if( count != 0 ) return -1;
60 struct berval **normalized )
62 struct berval *newval;
65 newval = ch_malloc( sizeof( struct berval ) );
69 /* Ignore initial whitespace */
70 while ( ldap_utf8_isspace( p ) ) {
79 newval->bv_val = ch_strdup( p );
80 p = q = newval->bv_val;
86 if ( ldap_utf8_isspace( p ) ) {
87 len = LDAP_UTF8_COPY(q,p);
92 /* Ignore the extra whitespace */
93 while ( ldap_utf8_isspace( p ) ) {
97 len = LDAP_UTF8_COPY(q,p);
104 assert( *newval->bv_val );
105 assert( newval->bv_val < p );
108 /* cannot start with a space */
109 assert( !ldap_utf8_isspace(newval->bv_val) );
112 * If the string ended in space, backup the pointer one
113 * position. One is enough because the above loop collapsed
114 * all whitespace to a single space.
121 /* cannot end with a space */
122 assert( !ldap_utf8_isspace( LDAP_UTF8_PREV(q) ) );
127 newval->bv_len = q - newval->bv_val;
128 normalized = &newval;
140 for(i=0; i < val->bv_len; i++) {
141 if( !isascii(val->bv_val[i]) ) return -1;
151 struct berval **out )
154 struct berval *bv = ch_malloc( sizeof(struct berval) );
155 bv->bv_len = (in->bv_len+1) * sizeof( ldap_unicode_t );
156 bv->bv_val = ch_malloc( bv->bv_len );
158 for(i=0; i < in->bv_len; i++ ) {
160 * IA5StringValidate should have been called to ensure
161 * input is limited to IA5.
163 bv->bv_val[i] = in->bv_val[i];
176 struct berval **normalized )
178 struct berval *newval;
181 newval = ch_malloc( sizeof( struct berval ) );
185 /* Ignore initial whitespace */
186 while ( isspace( *p++ ) ) {
195 newval->bv_val = ch_strdup( p );
196 p = q = newval->bv_val;
199 if ( isspace( *p ) ) {
202 /* Ignore the extra whitespace */
203 while ( isspace( *p++ ) ) {
211 assert( *newval->bv_val );
212 assert( newval->bv_val < p );
215 /* cannot start with a space */
216 assert( !isspace(*newval->bv_val) );
219 * If the string ended in space, backup the pointer one
220 * position. One is enough because the above loop collapsed
221 * all whitespace to a single space.
224 if ( isspace( q[-1] ) ) {
228 /* cannot end with a space */
229 assert( !isspace( q[-1] ) );
234 newval->bv_len = q - newval->bv_val;
235 normalized = &newval;
245 struct berval *value,
246 void *assertedValue )
248 return strcmp( value->bv_val,
249 ((struct berval *) assertedValue)->bv_val );
257 struct berval *value,
258 void *assertedValue )
260 return strcasecmp( value->bv_val,
261 ((struct berval *) assertedValue)->bv_val );
264 struct syntax_defs_rec {
267 slap_syntax_validate_func *sd_validate;
268 slap_syntax_transform_func *sd_ber2str;
269 slap_syntax_transform_func *sd_str2ber;
272 struct syntax_defs_rec syntax_defs[] = {
273 {"( 1.3.6.1.4.1.1466.115.121.1.3 DESC 'AttributeTypeDescription' )",
274 0, NULL, NULL, NULL},
275 {"( 1.3.6.1.4.1.1466.115.121.1.4 DESC 'Audio' )",
276 0, NULL, NULL, NULL},
277 {"( 1.3.6.1.4.1.1466.115.121.1.5 DESC 'Binary' )",
280 {"( 1.3.6.1.4.1.1466.115.121.1.6 DESC 'BitString' )",
281 0, NULL, NULL, NULL},
282 {"( 1.3.6.1.4.1.1466.115.121.1.7 DESC 'Boolean' )",
283 0, NULL, NULL, NULL},
284 {"( 1.3.6.1.4.1.1466.115.121.1.8 DESC 'Certificate' )",
287 {"( 1.3.6.1.4.1.1466.115.121.1.9 DESC 'CertificateList' )",
290 {"( 1.3.6.1.4.1.1466.115.121.1.10 DESC 'CertificatePair' )",
293 {"( 1.3.6.1.4.1.1466.115.121.1.12 DESC 'DN' )",
294 0, NULL, NULL, NULL},
295 {"( 1.3.6.1.4.1.1466.115.121.1.14 DESC 'DeliveryMethod' )",
296 0, NULL, NULL, NULL},
297 {"( 1.3.6.1.4.1.1466.115.121.1.15 DESC 'DirectoryString' )",
298 0, UTF8StringValidate, NULL, NULL},
299 {"( 1.3.6.1.4.1.1466.115.121.1.16 DESC 'DITContentRuleDescription' )",
300 0, NULL, NULL, NULL},
301 {"( 1.3.6.1.4.1.1466.115.121.1.17 DESC 'DITStructureRuleDescription' )",
302 0, NULL, NULL, NULL},
303 {"( 1.3.6.1.4.1.1466.115.121.1.21 DESC 'EnhancedGuide' )",
304 0, NULL, NULL, NULL},
305 {"( 1.3.6.1.4.1.1466.115.121.1.22 DESC 'FacsimileTelephoneNumber' )",
306 0, NULL, NULL, NULL},
307 {"( 1.3.6.1.4.1.1466.115.121.1.24 DESC 'GeneralizedTime' )",
308 0, NULL, NULL, NULL},
309 {"( 1.3.6.1.4.1.1466.115.121.1.25 DESC 'Guide' )",
310 0, NULL, NULL, NULL},
311 {"( 1.3.6.1.4.1.1466.115.121.1.26 DESC 'IA5String' )",
312 0, IA5StringValidate, NULL, NULL},
313 {"( 1.3.6.1.4.1.1466.115.121.1.27 DESC 'Integer' )",
314 0, NULL, NULL, NULL},
315 {"( 1.3.6.1.4.1.1466.115.121.1.28 DESC 'JPEG' )",
316 0, NULL, NULL, NULL},
317 {"( 1.3.6.1.4.1.1466.115.121.1.30 DESC 'MatchingRuleDescription' )",
318 0, NULL, NULL, NULL},
319 {"( 1.3.6.1.4.1.1466.115.121.1.31 DESC 'MatchingRuleUseDescription' )",
320 0, NULL, NULL, NULL},
321 {"( 1.3.6.1.4.1.1466.115.121.1.32 DESC 'MailPreference' )",
322 0, NULL, NULL, NULL},
323 {"( 1.3.6.1.4.1.1466.115.121.1.34 DESC 'NameAndOptionalUID' )",
324 0, NULL, NULL, NULL},
325 {"( 1.3.6.1.4.1.1466.115.121.1.35 DESC 'NameFormDescription' )",
326 0, NULL, NULL, NULL},
327 {"( 1.3.6.1.4.1.1466.115.121.1.36 DESC 'NumericString' )",
328 0, NULL, NULL, NULL},
329 {"( 1.3.6.1.4.1.1466.115.121.1.37 DESC 'ObjectClassDescription' )",
330 0, NULL, NULL, NULL},
331 {"( 1.3.6.1.4.1.1466.115.121.1.38 DESC 'OID' )",
332 0, NULL, NULL, NULL},
333 {"( 1.3.6.1.4.1.1466.115.121.1.39 DESC 'OtherMailbox' )",
334 0, NULL, NULL, NULL},
335 {"( 1.3.6.1.4.1.1466.115.121.1.40 DESC 'OctetString' )",
336 0, octetStringValidate, NULL, NULL},
337 {"( 1.3.6.1.4.1.1466.115.121.1.41 DESC 'PostalAddress' )",
338 0, NULL, NULL, NULL},
339 {"( 1.3.6.1.4.1.1466.115.121.1.42 DESC 'ProtocolInformation' )",
340 0, NULL, NULL, NULL},
341 {"( 1.3.6.1.4.1.1466.115.121.1.43 DESC 'PresentationAddress' )",
342 0, NULL, NULL, NULL},
343 {"( 1.3.6.1.4.1.1466.115.121.1.44 DESC 'PrintableString' )",
344 0, NULL, NULL, NULL},
345 {"( 1.3.6.1.4.1.1466.115.121.1.49 DESC 'SupportedAlgorithm' )",
346 SLAP_SYNTAX_BINARY, NULL, NULL, NULL},
347 {"( 1.3.6.1.4.1.1466.115.121.1.50 DESC 'TelephoneNumber' )",
348 0, NULL, NULL, NULL},
349 {"( 1.3.6.1.4.1.1466.115.121.1.51 DESC 'TeletexTerminalIdentifier' )",
350 0, NULL, NULL, NULL},
351 {"( 1.3.6.1.4.1.1466.115.121.1.52 DESC 'TelexNumber' )",
352 0, NULL, NULL, NULL},
353 {"( 1.3.6.1.4.1.1466.115.121.1.53 DESC 'UTCTime' )",
354 0, NULL, NULL, NULL},
355 {"( 1.3.6.1.4.1.1466.115.121.1.54 DESC 'LDAPSyntaxDescription' )",
356 0, NULL, NULL, NULL},
357 {"( 1.3.6.1.4.1.1466.115.121.1.58 DESC 'SubstringAssertion' )",
358 0, NULL, NULL, NULL},
360 /* OpenLDAP Experimental Syntaxes */
361 {"( " SLAPD_OID_ACI_SYNTAX " DESC 'OpenLDAPexperimentalACI' )",
362 0, NULL, NULL, NULL},
364 {NULL, 0, NULL, NULL, NULL}
367 struct mrule_defs_rec {
370 slap_mr_convert_func * mrd_convert;
371 slap_mr_normalize_func * mrd_normalize;
372 slap_mr_match_func * mrd_match;
373 slap_mr_indexer_func * mrd_indexer;
374 slap_mr_filter_func * mrd_filter;
378 * Other matching rules in X.520 that we do not use:
380 * 2.5.13.9 numericStringOrderingMatch
381 * 2.5.13.12 caseIgnoreListSubstringsMatch
382 * 2.5.13.13 booleanMatch
383 * 2.5.13.15 integerOrderingMatch
384 * 2.5.13.18 octetStringOrderingMatch
385 * 2.5.13.19 octetStringSubstringsMatch
386 * 2.5.13.25 uTCTimeMatch
387 * 2.5.13.26 uTCTimeOrderingMatch
388 * 2.5.13.31 directoryStringFirstComponentMatch
389 * 2.5.13.32 wordMatch
390 * 2.5.13.33 keywordMatch
391 * 2.5.13.34 certificateExactMatch
392 * 2.5.13.35 certificateMatch
393 * 2.5.13.36 certificatePairExactMatch
394 * 2.5.13.37 certificatePairMatch
395 * 2.5.13.38 certificateListExactMatch
396 * 2.5.13.39 certificateListMatch
397 * 2.5.13.40 algorithmIdentifierMatch
398 * 2.5.13.41 storedPrefixMatch
399 * 2.5.13.42 attributeCertificateMatch
400 * 2.5.13.43 readerAndKeyIDMatch
401 * 2.5.13.44 attributeIntegrityMatch
404 /* recycled matching functions */
405 #define caseIgnoreMatch caseIgnoreIA5Match
406 #define caseExactMatch caseExactIA5Match
408 /* unimplemented matching functions */
409 #define objectIdentifierMatch NULL
410 #define distinguishedNameMatch NULL
411 #define caseIgnoreOrderingMatch NULL
412 #define caseIgnoreSubstringsMatch NULL
413 #define caseExactOrderingMatch NULL
414 #define caseExactSubstringsMatch NULL
415 #define numericStringMatch NULL
416 #define numericStringSubstringsMatch NULL
417 #define caseIgnoreListMatch NULL
418 #define integerMatch NULL
419 #define bitStringMatch NULL
420 #define octetStringMatch NULL
421 #define telephoneNumberMatch NULL
422 #define telephoneNumberSubstringsMatch NULL
423 #define presentationAddressMatch NULL
424 #define uniqueMemberMatch NULL
425 #define protocolInformationMatch NULL
426 #define generalizedTimeMatch NULL
427 #define generalizedTimeOrderingMatch NULL
428 #define integerFirstComponentMatch NULL
429 #define objectIdentifierFirstComponentMatch NULL
430 #define caseIgnoreIA5SubstringsMatch NULL
432 struct mrule_defs_rec mrule_defs[] = {
433 {"( 2.5.13.0 NAME 'objectIdentifierMatch' "
434 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.38 )",
435 SLAP_MR_EQUALITY | SLAP_MR_EXT,
436 NULL, NULL, objectIdentifierMatch, NULL, NULL},
438 {"( 2.5.13.1 NAME 'distinguishedNameMatch' "
439 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.12 )",
440 SLAP_MR_EQUALITY | SLAP_MR_EXT,
441 NULL, NULL, distinguishedNameMatch, NULL, NULL},
443 {"( 2.5.13.2 NAME 'caseIgnoreMatch' "
444 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.15 )",
445 SLAP_MR_EQUALITY | SLAP_MR_EXT,
446 NULL, UTF8StringNormalize, caseIgnoreMatch, NULL, NULL},
448 {"( 2.5.13.3 NAME 'caseIgnoreOrderingMatch' "
449 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.15 )",
451 NULL, UTF8StringNormalize, caseIgnoreOrderingMatch, NULL, NULL},
453 {"( 2.5.13.4 NAME 'caseIgnoreSubstringsMatch' "
454 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.58 )",
455 SLAP_MR_SUBSTR | SLAP_MR_EXT,
456 NULL, UTF8StringNormalize, caseIgnoreSubstringsMatch, NULL, NULL},
458 /* Next three are not in the RFC's, but are needed for compatibility */
459 {"( 2.5.13.5 NAME 'caseExactMatch' "
460 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.15 )",
461 SLAP_MR_EQUALITY | SLAP_MR_EXT,
462 NULL, UTF8StringNormalize, caseExactMatch, NULL, NULL},
464 {"( 2.5.13.6 NAME 'caseExactOrderingMatch' "
465 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.15 )",
467 NULL, UTF8StringNormalize, caseExactOrderingMatch, NULL, NULL},
469 {"( 2.5.13.7 NAME 'caseExactSubstringsMatch' "
470 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.58 )",
471 SLAP_MR_SUBSTR | SLAP_MR_EXT,
472 NULL, UTF8StringNormalize, caseExactSubstringsMatch, NULL, NULL},
474 {"( 2.5.13.8 NAME 'numericStringMatch' "
475 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.36 )",
476 SLAP_MR_EQUALITY | SLAP_MR_EXT,
477 NULL, NULL, numericStringMatch, NULL, NULL},
479 {"( 2.5.13.10 NAME 'numericStringSubstringsMatch' "
480 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.58 )",
481 SLAP_MR_SUBSTR | SLAP_MR_EXT,
482 NULL, NULL, numericStringSubstringsMatch, NULL, NULL},
484 {"( 2.5.13.11 NAME 'caseIgnoreListMatch' "
485 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.41 )",
486 SLAP_MR_EQUALITY | SLAP_MR_EXT,
487 NULL, NULL, caseIgnoreListMatch, NULL, NULL},
489 {"( 2.5.13.14 NAME 'integerMatch' "
490 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.27 )",
491 SLAP_MR_NONE | SLAP_MR_EXT,
492 NULL, NULL, integerMatch, NULL, NULL},
494 {"( 2.5.13.16 NAME 'bitStringMatch' "
495 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.6 )",
496 SLAP_MR_NONE | SLAP_MR_EXT,
497 NULL, NULL, bitStringMatch, NULL, NULL},
499 {"( 2.5.13.17 NAME 'octetStringMatch' "
500 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.40 )",
501 SLAP_MR_EQUALITY | SLAP_MR_EXT,
502 NULL, NULL, octetStringMatch, NULL, NULL},
504 {"( 2.5.13.20 NAME 'telephoneNumberMatch' "
505 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.50 )",
506 SLAP_MR_EQUALITY | SLAP_MR_EXT,
507 NULL, NULL, telephoneNumberMatch, NULL, NULL},
509 {"( 2.5.13.21 NAME 'telephoneNumberSubstringsMatch' "
510 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.58 )",
511 SLAP_MR_SUBSTR | SLAP_MR_EXT,
512 NULL, NULL, telephoneNumberSubstringsMatch, NULL, NULL},
514 {"( 2.5.13.22 NAME 'presentationAddressMatch' "
515 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.43 )",
516 SLAP_MR_NONE | SLAP_MR_EXT,
517 NULL, NULL, presentationAddressMatch, NULL, NULL},
519 {"( 2.5.13.23 NAME 'uniqueMemberMatch' "
520 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.34 )",
521 SLAP_MR_NONE | SLAP_MR_EXT,
522 NULL, NULL, uniqueMemberMatch, NULL, NULL},
524 {"( 2.5.13.24 NAME 'protocolInformationMatch' "
525 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.42 )",
526 SLAP_MR_NONE | SLAP_MR_EXT,
527 NULL, NULL, protocolInformationMatch, NULL, NULL},
529 {"( 2.5.13.27 NAME 'generalizedTimeMatch' "
530 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.24 )",
531 SLAP_MR_EQUALITY | SLAP_MR_EXT,
532 NULL, NULL, generalizedTimeMatch, NULL, NULL},
534 {"( 2.5.13.28 NAME 'generalizedTimeOrderingMatch' "
535 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.24 )",
537 NULL, NULL, generalizedTimeOrderingMatch, NULL, NULL},
539 {"( 2.5.13.29 NAME 'integerFirstComponentMatch' "
540 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.27 )",
541 SLAP_MR_EQUALITY | SLAP_MR_EXT,
542 NULL, NULL, integerFirstComponentMatch, NULL, NULL},
544 {"( 2.5.13.30 NAME 'objectIdentifierFirstComponentMatch' "
545 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.38 )",
546 SLAP_MR_EQUALITY | SLAP_MR_EXT,
547 NULL, NULL, objectIdentifierFirstComponentMatch, NULL, NULL},
549 {"( 1.3.6.1.4.1.1466.109.114.1 NAME 'caseExactIA5Match' "
550 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.26 )",
551 SLAP_MR_EQUALITY | SLAP_MR_EXT,
552 NULL, IA5StringNormalize, caseExactIA5Match, NULL, NULL},
554 {"( 1.3.6.1.4.1.1466.109.114.2 NAME 'caseIgnoreIA5Match' "
555 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.26 )",
556 SLAP_MR_EQUALITY | SLAP_MR_EXT,
557 NULL, IA5StringNormalize, caseIgnoreIA5Match, NULL, NULL},
559 {"( 1.3.6.1.4.1.1466.109.114.3 NAME 'caseIgnoreIA5SubstringsMatch' "
560 "SYNTAX 1.3.6.1.4.1.1466.115.121.1.26 )",
562 NULL, IA5StringNormalize, caseIgnoreIA5SubstringsMatch, NULL, NULL},
564 {NULL, SLAP_MR_NONE, NULL, NULL, NULL}
572 static int schema_init_done = 0;
574 /* We are called from read_config that is recursive */
575 if ( schema_init_done )
578 for ( i=0; syntax_defs[i].sd_desc != NULL; i++ ) {
579 res = register_syntax( syntax_defs[i].sd_desc,
580 syntax_defs[i].sd_flags,
581 syntax_defs[i].sd_validate,
582 syntax_defs[i].sd_ber2str,
583 syntax_defs[i].sd_str2ber );
586 fprintf( stderr, "schema_init: Error registering syntax %s\n",
587 syntax_defs[i].sd_desc );
588 exit( EXIT_FAILURE );
592 for ( i=0; mrule_defs[i].mrd_desc != NULL; i++ ) {
593 if( mrule_defs[i].mrd_usage == SLAP_MR_NONE ) {
595 "schema_init: Ingoring unusable matching rule %s\n",
596 mrule_defs[i].mrd_desc );
600 res = register_matching_rule(
601 mrule_defs[i].mrd_desc,
602 mrule_defs[i].mrd_usage,
603 mrule_defs[i].mrd_convert,
604 mrule_defs[i].mrd_normalize,
605 mrule_defs[i].mrd_match,
606 mrule_defs[i].mrd_indexer,
607 mrule_defs[i].mrd_filter );
611 "schema_init: Error registering matching rule %s\n",
612 mrule_defs[i].mrd_desc );
613 exit( EXIT_FAILURE );
616 schema_init_done = 1;