xref: /netbsd-src/external/bsd/openldap/dist/libraries/librewrite/parse.c (revision 549b59ed3ccf0d36d3097190a0db27b770f3a839)
1*549b59edSchristos /*	$NetBSD: parse.c,v 1.3 2021/08/14 16:14:58 christos Exp $	*/
24e6df137Slukem 
3d11b170bStron /* $OpenLDAP$ */
42de962bdSlukem /* This work is part of OpenLDAP Software <http://www.openldap.org/>.
52de962bdSlukem  *
6*549b59edSchristos  * Copyright 2000-2021 The OpenLDAP Foundation.
72de962bdSlukem  * All rights reserved.
82de962bdSlukem  *
92de962bdSlukem  * Redistribution and use in source and binary forms, with or without
102de962bdSlukem  * modification, are permitted only as authorized by the OpenLDAP
112de962bdSlukem  * Public License.
122de962bdSlukem  *
132de962bdSlukem  * A copy of this license is available in the file LICENSE in the
142de962bdSlukem  * top-level directory of the distribution or, alternatively, at
152de962bdSlukem  * <http://www.OpenLDAP.org/license.html>.
162de962bdSlukem  */
172de962bdSlukem /* ACKNOWLEDGEMENT:
182de962bdSlukem  * This work was initially developed by Pierangelo Masarati for
192de962bdSlukem  * inclusion in OpenLDAP Software.
202de962bdSlukem  */
212de962bdSlukem 
222de962bdSlukem #include <portable.h>
232de962bdSlukem 
242de962bdSlukem #include <stdio.h>
252de962bdSlukem 
262de962bdSlukem #include "rewrite-int.h"
272de962bdSlukem 
282de962bdSlukem static int
parse_line(char ** argv,int * argc,int maxargs,char * buf)292de962bdSlukem parse_line(
302de962bdSlukem 		char **argv,
312de962bdSlukem 		int *argc,
322de962bdSlukem 		int maxargs,
332de962bdSlukem 		char *buf
342de962bdSlukem )
352de962bdSlukem {
362de962bdSlukem 	char *p, *begin;
372de962bdSlukem 	int in_quoted_field = 0, cnt = 0;
382de962bdSlukem 	char quote = '\0';
392de962bdSlukem 
402de962bdSlukem 	for ( p = buf; isspace( (unsigned char) p[ 0 ] ); p++ );
412de962bdSlukem 
422de962bdSlukem 	if ( p[ 0 ] == '#' ) {
432de962bdSlukem 		return 0;
442de962bdSlukem 	}
452de962bdSlukem 
462de962bdSlukem 	for ( begin = p;  p[ 0 ] != '\0'; p++ ) {
472de962bdSlukem 		if ( p[ 0 ] == '\\' && p[ 1 ] != '\0' ) {
482de962bdSlukem 			p++;
492de962bdSlukem 		} else if ( p[ 0 ] == '\'' || p[ 0 ] == '\"') {
502de962bdSlukem 			if ( in_quoted_field && p[ 0 ] == quote ) {
512de962bdSlukem 				in_quoted_field = 1 - in_quoted_field;
522de962bdSlukem 				quote = '\0';
532de962bdSlukem 				p[ 0 ] = '\0';
542de962bdSlukem 				argv[ cnt ] = begin;
552de962bdSlukem 				if ( ++cnt == maxargs ) {
562de962bdSlukem 					*argc = cnt;
572de962bdSlukem 					return 1;
582de962bdSlukem 				}
592de962bdSlukem 				for ( p++; isspace( (unsigned char) p[ 0 ] ); p++ );
602de962bdSlukem 				begin = p;
612de962bdSlukem 				p--;
622de962bdSlukem 
632de962bdSlukem 			} else if ( !in_quoted_field ) {
642de962bdSlukem 				if ( p != begin ) {
652de962bdSlukem 					return -1;
662de962bdSlukem 				}
672de962bdSlukem 				begin++;
682de962bdSlukem 				in_quoted_field = 1 - in_quoted_field;
692de962bdSlukem 				quote = p[ 0 ];
702de962bdSlukem 			}
712de962bdSlukem 		} else if ( isspace( (unsigned char) p[ 0 ] ) && !in_quoted_field ) {
722de962bdSlukem 			p[ 0 ] = '\0';
732de962bdSlukem 			argv[ cnt ] = begin;
742de962bdSlukem 
752de962bdSlukem 			if ( ++cnt == maxargs ) {
762de962bdSlukem 				*argc = cnt;
772de962bdSlukem 				return 1;
782de962bdSlukem 			}
792de962bdSlukem 
802de962bdSlukem 			for ( p++; isspace( (unsigned char) p[ 0 ] ); p++ );
812de962bdSlukem 			begin = p;
822de962bdSlukem 			p--;
832de962bdSlukem 		}
842de962bdSlukem 	}
852de962bdSlukem 
862de962bdSlukem 	*argc = cnt;
872de962bdSlukem 
882de962bdSlukem 	return 1;
892de962bdSlukem }
902de962bdSlukem 
912de962bdSlukem int
rewrite_read(FILE * fin,struct rewrite_info * info)922de962bdSlukem rewrite_read(
932de962bdSlukem 		FILE *fin,
942de962bdSlukem 		struct rewrite_info *info
952de962bdSlukem )
962de962bdSlukem {
972de962bdSlukem 	char buf[ 1024 ];
982de962bdSlukem 	char *argv[11];
992de962bdSlukem 	int argc, lineno;
1002de962bdSlukem 
1012de962bdSlukem 	/*
1022de962bdSlukem 	 * Empty rule at the beginning of the context
1032de962bdSlukem 	 */
1042de962bdSlukem 
1052de962bdSlukem 	for ( lineno = 0; fgets( buf, sizeof( buf ), fin ); lineno++ ) {
1062de962bdSlukem 		switch ( parse_line( argv, &argc, sizeof( argv ) - 1, buf ) ) {
1072de962bdSlukem 		case -1:
1082de962bdSlukem 			return REWRITE_ERR;
1092de962bdSlukem 		case 0:
1102de962bdSlukem 			break;
1112de962bdSlukem 		case 1:
1122de962bdSlukem 			if ( strncasecmp( argv[ 0 ], "rewrite", 7 ) == 0 ) {
1132de962bdSlukem 				int rc;
1142de962bdSlukem 				rc = rewrite_parse( info, "file", lineno,
1152de962bdSlukem 						argc, argv );
1162de962bdSlukem 				if ( rc != REWRITE_SUCCESS ) {
1172de962bdSlukem 					return rc;
1182de962bdSlukem 				}
1192de962bdSlukem 			}
1202de962bdSlukem 			break;
1212de962bdSlukem 		}
1222de962bdSlukem 	}
1232de962bdSlukem 
1242de962bdSlukem 	return REWRITE_SUCCESS;
1252de962bdSlukem }
1262de962bdSlukem 
127