2003-12-09 01:41:40 +08:00
|
|
|
/* $OpenLDAP$ */
|
|
|
|
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
|
2001-05-12 08:51:28 +08:00
|
|
|
*
|
2011-01-05 08:42:37 +08:00
|
|
|
* Copyright 2000-2011 The OpenLDAP Foundation.
|
2001-05-12 08:51:28 +08:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
2003-12-09 01:41:40 +08:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted only as authorized by the OpenLDAP
|
|
|
|
* Public License.
|
2001-05-12 08:51:28 +08:00
|
|
|
*
|
2003-12-09 01:41:40 +08:00
|
|
|
* A copy of this license is available in the file LICENSE in the
|
|
|
|
* top-level directory of the distribution or, alternatively, at
|
|
|
|
* <http://www.OpenLDAP.org/license.html>.
|
|
|
|
*/
|
|
|
|
/* ACKNOWLEDGEMENT:
|
|
|
|
* This work was initially developed by Pierangelo Masarati for
|
|
|
|
* inclusion in OpenLDAP Software.
|
|
|
|
*/
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
#include <portable.h>
|
|
|
|
|
|
|
|
#include "rewrite-int.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Appends a rule to the double linked list of rules
|
|
|
|
* Helper for rewrite_rule_compile
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
append_rule(
|
|
|
|
struct rewrite_context *context,
|
|
|
|
struct rewrite_rule *rule
|
|
|
|
)
|
|
|
|
{
|
|
|
|
struct rewrite_rule *r;
|
|
|
|
|
|
|
|
assert( context != NULL );
|
|
|
|
assert( context->lc_rule != NULL );
|
|
|
|
assert( rule != NULL );
|
|
|
|
|
|
|
|
for ( r = context->lc_rule; r->lr_next != NULL; r = r->lr_next );
|
|
|
|
r->lr_next = rule;
|
|
|
|
rule->lr_prev = r;
|
|
|
|
|
|
|
|
return REWRITE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Appends an action to the linked list of actions
|
|
|
|
* Helper for rewrite_rule_compile
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
append_action(
|
2003-12-30 00:14:47 +08:00
|
|
|
struct rewrite_action **pbase,
|
2001-05-12 08:51:28 +08:00
|
|
|
struct rewrite_action *action
|
|
|
|
)
|
|
|
|
{
|
2003-12-30 00:14:47 +08:00
|
|
|
struct rewrite_action **pa;
|
2001-05-12 08:51:28 +08:00
|
|
|
|
2003-12-30 00:14:47 +08:00
|
|
|
assert( pbase != NULL );
|
2001-05-12 08:51:28 +08:00
|
|
|
assert( action != NULL );
|
|
|
|
|
2003-12-30 00:14:47 +08:00
|
|
|
for ( pa = pbase; *pa != NULL; pa = &(*pa)->la_next );
|
|
|
|
*pa = action;
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
return REWRITE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2003-11-15 18:02:44 +08:00
|
|
|
static int
|
|
|
|
destroy_action(
|
|
|
|
struct rewrite_action **paction
|
|
|
|
)
|
|
|
|
{
|
|
|
|
struct rewrite_action *action;
|
|
|
|
|
2005-07-18 14:22:33 +08:00
|
|
|
assert( paction != NULL );
|
|
|
|
assert( *paction != NULL );
|
2003-11-15 18:02:44 +08:00
|
|
|
|
|
|
|
action = *paction;
|
|
|
|
|
|
|
|
/* do something */
|
|
|
|
switch ( action->la_type ) {
|
2003-12-30 00:14:47 +08:00
|
|
|
case REWRITE_FLAG_GOTO:
|
|
|
|
case REWRITE_FLAG_USER: {
|
2003-11-15 18:02:44 +08:00
|
|
|
int *pi = (int *)action->la_args;
|
|
|
|
|
|
|
|
if ( pi ) {
|
|
|
|
free( pi );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
free( action );
|
|
|
|
*paction = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-04-03 03:55:05 +08:00
|
|
|
static void
|
2006-04-01 04:06:49 +08:00
|
|
|
destroy_actions(
|
|
|
|
struct rewrite_action *paction
|
|
|
|
)
|
|
|
|
{
|
|
|
|
struct rewrite_action *next;
|
|
|
|
|
|
|
|
for (; paction; paction = next) {
|
|
|
|
next = paction->la_next;
|
|
|
|
destroy_action( &paction );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-05-12 08:51:28 +08:00
|
|
|
/*
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
rewrite_rule_compile(
|
|
|
|
struct rewrite_info *info,
|
|
|
|
struct rewrite_context *context,
|
|
|
|
const char *pattern,
|
|
|
|
const char *result,
|
|
|
|
const char *flagstring
|
|
|
|
)
|
|
|
|
{
|
|
|
|
int flags = REWRITE_REGEX_EXTENDED | REWRITE_REGEX_ICASE;
|
|
|
|
int mode = REWRITE_RECURSE;
|
2003-12-30 02:02:49 +08:00
|
|
|
int max_passes = info->li_max_passes_per_rule;
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
struct rewrite_rule *rule = NULL;
|
|
|
|
struct rewrite_subst *subst = NULL;
|
|
|
|
struct rewrite_action *action = NULL, *first_action = NULL;
|
|
|
|
|
|
|
|
const char *p;
|
|
|
|
|
|
|
|
assert( info != NULL );
|
|
|
|
assert( context != NULL );
|
|
|
|
assert( pattern != NULL );
|
|
|
|
assert( result != NULL );
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A null flagstring should be allowed
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Take care of substitution string
|
|
|
|
*/
|
|
|
|
subst = rewrite_subst_compile( info, result );
|
|
|
|
if ( subst == NULL ) {
|
|
|
|
return REWRITE_ERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Take care of flags
|
|
|
|
*/
|
|
|
|
for ( p = flagstring; p[ 0 ] != '\0'; p++ ) {
|
|
|
|
switch( p[ 0 ] ) {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* REGEX flags
|
|
|
|
*/
|
|
|
|
case REWRITE_FLAG_HONORCASE: /* 'C' */
|
|
|
|
/*
|
|
|
|
* Honor case (default is case insensitive)
|
|
|
|
*/
|
|
|
|
flags &= ~REWRITE_REGEX_ICASE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REWRITE_FLAG_BASICREGEX: /* 'R' */
|
|
|
|
/*
|
|
|
|
* Use POSIX Basic Regular Expression syntax
|
|
|
|
* instead of POSIX Extended Regular Expression
|
|
|
|
* syntax (default)
|
|
|
|
*/
|
|
|
|
flags &= ~REWRITE_REGEX_EXTENDED;
|
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Execution mode flags
|
|
|
|
*/
|
|
|
|
case REWRITE_FLAG_EXECONCE: /* ':' */
|
|
|
|
/*
|
|
|
|
* Apply rule once only
|
|
|
|
*/
|
|
|
|
mode &= ~REWRITE_RECURSE;
|
|
|
|
mode |= REWRITE_EXEC_ONCE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Special action flags
|
|
|
|
*/
|
|
|
|
case REWRITE_FLAG_STOP: /* '@' */
|
|
|
|
/*
|
|
|
|
* Bail out after applying rule
|
|
|
|
*/
|
|
|
|
action = calloc( sizeof( struct rewrite_action ), 1 );
|
|
|
|
if ( action == NULL ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
2004-01-09 00:20:32 +08:00
|
|
|
|
2001-05-12 08:51:28 +08:00
|
|
|
action->la_type = REWRITE_ACTION_STOP;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REWRITE_FLAG_UNWILLING: /* '#' */
|
|
|
|
/*
|
|
|
|
* Matching objs will be marked as gone!
|
|
|
|
*/
|
|
|
|
action = calloc( sizeof( struct rewrite_action ), 1 );
|
|
|
|
if ( action == NULL ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
mode &= ~REWRITE_RECURSE;
|
|
|
|
mode |= REWRITE_EXEC_ONCE;
|
|
|
|
action->la_type = REWRITE_ACTION_UNWILLING;
|
|
|
|
break;
|
|
|
|
|
2003-12-30 00:14:47 +08:00
|
|
|
case REWRITE_FLAG_GOTO: /* 'G' */
|
2001-05-12 08:51:28 +08:00
|
|
|
/*
|
|
|
|
* After applying rule, jump N rules
|
|
|
|
*/
|
|
|
|
|
2003-12-30 00:14:47 +08:00
|
|
|
case REWRITE_FLAG_USER: { /* 'U' */
|
|
|
|
/*
|
|
|
|
* After applying rule, return user-defined
|
|
|
|
* error code
|
|
|
|
*/
|
2003-12-30 01:06:43 +08:00
|
|
|
char *next = NULL;
|
2001-05-12 08:51:28 +08:00
|
|
|
int *d;
|
|
|
|
|
|
|
|
if ( p[ 1 ] != '{' ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
2003-12-30 01:06:43 +08:00
|
|
|
d = malloc( sizeof( int ) );
|
|
|
|
if ( d == NULL ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
2003-12-30 01:06:43 +08:00
|
|
|
d[ 0 ] = strtol( &p[ 2 ], &next, 0 );
|
2005-11-24 09:10:05 +08:00
|
|
|
if ( next == &p[ 2 ] || next[0] != '}' ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
free( d );
|
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
action = calloc( sizeof( struct rewrite_action ), 1 );
|
|
|
|
if ( action == NULL ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
free( d );
|
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
2003-12-30 00:14:47 +08:00
|
|
|
switch ( p[ 0 ] ) {
|
|
|
|
case REWRITE_FLAG_GOTO:
|
|
|
|
action->la_type = REWRITE_ACTION_GOTO;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REWRITE_FLAG_USER:
|
|
|
|
action->la_type = REWRITE_ACTION_USER;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
|
2001-05-12 08:51:28 +08:00
|
|
|
action->la_args = (void *)d;
|
|
|
|
|
2003-12-30 01:06:43 +08:00
|
|
|
p = next; /* p is incremented by the for ... */
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2003-12-30 02:02:49 +08:00
|
|
|
case REWRITE_FLAG_MAX_PASSES: { /* 'U' */
|
|
|
|
/*
|
|
|
|
* Set the number of max passes per rule
|
|
|
|
*/
|
|
|
|
char *next = NULL;
|
|
|
|
|
|
|
|
if ( p[ 1 ] != '{' ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2003-12-30 02:02:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
max_passes = strtol( &p[ 2 ], &next, 0 );
|
2005-11-24 09:10:05 +08:00
|
|
|
if ( next == &p[ 2 ] || next[0] != '}' ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2003-12-30 02:02:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( max_passes < 1 ) {
|
|
|
|
/* FIXME: nonsense ... */
|
|
|
|
max_passes = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
p = next; /* p is incremented by the for ... */
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2001-05-12 08:51:28 +08:00
|
|
|
case REWRITE_FLAG_IGNORE_ERR: /* 'I' */
|
|
|
|
/*
|
|
|
|
* Ignore errors!
|
|
|
|
*/
|
|
|
|
action = calloc( sizeof( struct rewrite_action ), 1 );
|
|
|
|
if ( action == NULL ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
action->la_type = REWRITE_ACTION_IGNORE_ERR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Other flags ...
|
|
|
|
*/
|
|
|
|
default:
|
|
|
|
/*
|
|
|
|
* Unimplemented feature (complain only)
|
|
|
|
*/
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stupid way to append to a list ...
|
|
|
|
*/
|
|
|
|
if ( action != NULL ) {
|
2003-12-30 00:14:47 +08:00
|
|
|
append_action( &first_action, action );
|
2001-05-12 08:51:28 +08:00
|
|
|
action = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Finally, rule allocation
|
|
|
|
*/
|
|
|
|
rule = calloc( sizeof( struct rewrite_rule ), 1 );
|
|
|
|
if ( rule == NULL ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* REGEX compilation (luckily I don't need to take care of this ...)
|
|
|
|
*/
|
|
|
|
if ( regcomp( &rule->lr_regex, ( char * )pattern, flags ) != 0 ) {
|
2006-04-01 04:06:49 +08:00
|
|
|
goto fail;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Just to remember them ...
|
|
|
|
*/
|
|
|
|
rule->lr_pattern = strdup( pattern );
|
|
|
|
rule->lr_subststring = strdup( result );
|
|
|
|
rule->lr_flagstring = strdup( flagstring );
|
2008-09-10 03:58:47 +08:00
|
|
|
if ( rule->lr_pattern == NULL
|
|
|
|
|| rule->lr_subststring == NULL
|
|
|
|
|| rule->lr_flagstring == NULL )
|
|
|
|
{
|
|
|
|
goto fail;
|
|
|
|
}
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Load compiled data into rule
|
|
|
|
*/
|
|
|
|
rule->lr_subst = subst;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set various parameters
|
|
|
|
*/
|
|
|
|
rule->lr_flags = flags; /* don't really need any longer ... */
|
|
|
|
rule->lr_mode = mode;
|
2003-12-30 02:02:49 +08:00
|
|
|
rule->lr_max_passes = max_passes;
|
2001-05-12 08:51:28 +08:00
|
|
|
rule->lr_action = first_action;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Append rule at the end of the rewrite context
|
|
|
|
*/
|
|
|
|
append_rule( context, rule );
|
|
|
|
|
|
|
|
return REWRITE_SUCCESS;
|
2006-04-01 04:06:49 +08:00
|
|
|
|
|
|
|
fail:
|
2008-09-10 03:58:47 +08:00
|
|
|
if ( rule ) {
|
|
|
|
if ( rule->lr_pattern ) free( rule->lr_pattern );
|
|
|
|
if ( rule->lr_subststring ) free( rule->lr_subststring );
|
|
|
|
if ( rule->lr_flagstring ) free( rule->lr_flagstring );
|
|
|
|
free( rule );
|
|
|
|
}
|
2006-04-01 04:06:49 +08:00
|
|
|
destroy_actions( first_action );
|
|
|
|
free( subst );
|
|
|
|
return REWRITE_ERR;
|
2001-05-12 08:51:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Rewrites string according to rule; may return:
|
|
|
|
* OK: fine; if *result != NULL rule matched and rewrite succeeded.
|
|
|
|
* STOP: fine, rule matched; stop processing following rules
|
|
|
|
* UNWILL: rule matched; force 'unwilling to perform'
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
rewrite_rule_apply(
|
|
|
|
struct rewrite_info *info,
|
|
|
|
struct rewrite_op *op,
|
|
|
|
struct rewrite_rule *rule,
|
|
|
|
const char *arg,
|
|
|
|
char **result
|
|
|
|
)
|
|
|
|
{
|
|
|
|
size_t nmatch = REWRITE_MAX_MATCH;
|
|
|
|
regmatch_t match[ REWRITE_MAX_MATCH ];
|
|
|
|
|
|
|
|
int rc = REWRITE_SUCCESS;
|
|
|
|
|
|
|
|
char *string;
|
2003-11-15 02:39:18 +08:00
|
|
|
int strcnt = 0;
|
|
|
|
struct berval val = { 0, NULL };
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
assert( info != NULL );
|
|
|
|
assert( op != NULL );
|
|
|
|
assert( rule != NULL );
|
|
|
|
assert( arg != NULL );
|
|
|
|
assert( result != NULL );
|
|
|
|
|
|
|
|
*result = NULL;
|
|
|
|
|
2003-11-15 02:39:18 +08:00
|
|
|
string = (char *)arg;
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* In case recursive match is required (default)
|
|
|
|
*/
|
|
|
|
recurse:;
|
|
|
|
|
|
|
|
Debug( LDAP_DEBUG_TRACE, "==> rewrite_rule_apply"
|
2004-01-20 00:56:15 +08:00
|
|
|
" rule='%s' string='%s' [%d pass(es)]\n",
|
2003-12-30 02:02:49 +08:00
|
|
|
rule->lr_pattern, string, strcnt + 1 );
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
op->lo_num_passes++;
|
2005-08-14 02:43:20 +08:00
|
|
|
|
|
|
|
rc = regexec( &rule->lr_regex, string, nmatch, match, 0 );
|
|
|
|
if ( rc != 0 ) {
|
2005-01-02 00:10:25 +08:00
|
|
|
if ( *result == NULL && string != arg ) {
|
2001-05-12 08:51:28 +08:00
|
|
|
free( string );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* No match is OK; *result = NULL means no match
|
|
|
|
*/
|
|
|
|
return REWRITE_REGEXEC_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = rewrite_subst_apply( info, op, rule->lr_subst, string,
|
|
|
|
match, &val );
|
|
|
|
|
|
|
|
*result = val.bv_val;
|
2003-11-15 02:39:18 +08:00
|
|
|
val.bv_val = NULL;
|
2005-01-02 00:10:25 +08:00
|
|
|
if ( string != arg ) {
|
2003-11-15 02:39:18 +08:00
|
|
|
free( string );
|
|
|
|
string = NULL;
|
|
|
|
}
|
2001-05-12 08:51:28 +08:00
|
|
|
|
|
|
|
if ( rc != REWRITE_REGEXEC_OK ) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ( rule->lr_mode & REWRITE_RECURSE ) == REWRITE_RECURSE
|
2003-12-30 02:02:49 +08:00
|
|
|
&& op->lo_num_passes < info->li_max_passes
|
|
|
|
&& ++strcnt < rule->lr_max_passes ) {
|
2001-05-12 08:51:28 +08:00
|
|
|
string = *result;
|
2003-11-15 02:39:18 +08:00
|
|
|
|
2001-05-12 08:51:28 +08:00
|
|
|
goto recurse;
|
|
|
|
}
|
|
|
|
|
|
|
|
return REWRITE_REGEXEC_OK;
|
|
|
|
}
|
|
|
|
|
2003-11-15 02:39:18 +08:00
|
|
|
int
|
|
|
|
rewrite_rule_destroy(
|
|
|
|
struct rewrite_rule **prule
|
|
|
|
)
|
|
|
|
{
|
|
|
|
struct rewrite_rule *rule;
|
|
|
|
|
2005-07-18 14:22:33 +08:00
|
|
|
assert( prule != NULL );
|
|
|
|
assert( *prule != NULL );
|
2003-11-15 02:39:18 +08:00
|
|
|
|
|
|
|
rule = *prule;
|
|
|
|
|
|
|
|
if ( rule->lr_pattern ) {
|
|
|
|
free( rule->lr_pattern );
|
|
|
|
rule->lr_pattern = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( rule->lr_subststring ) {
|
|
|
|
free( rule->lr_subststring );
|
|
|
|
rule->lr_subststring = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( rule->lr_flagstring ) {
|
|
|
|
free( rule->lr_flagstring );
|
|
|
|
rule->lr_flagstring = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( rule->lr_subst ) {
|
|
|
|
rewrite_subst_destroy( &rule->lr_subst );
|
|
|
|
}
|
|
|
|
|
|
|
|
regfree( &rule->lr_regex );
|
|
|
|
|
2006-04-01 04:06:49 +08:00
|
|
|
destroy_actions( rule->lr_action );
|
2003-11-15 18:02:44 +08:00
|
|
|
|
2003-11-15 02:39:18 +08:00
|
|
|
free( rule );
|
|
|
|
*prule = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|