| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504 | /* * $Id: dict.c,v 1.10 2007/07/11 17:29:29 cparker Exp $ * * Copyright (C) 1995,1996,1997 Lars Fenneberg * * Copyright 1992 Livingston Enterprises, Inc. * * Copyright 1992,1993, 1994,1995 The Regents of the University of Michigan * and Merit Network, Inc. All Rights Reserved * * See the file COPYRIGHT for the respective terms and conditions. * If the file is missing contact me at lf@elemental.net * and I'll send you a copy. * */#include <radius_config.h>#include <includes.h>#include <freeradius-client.h>#if defined ( __GNUC__ )#include <my_strings.h>#endif/** Initialize the dictionary * * Read all ATTRIBUTES into the dictionary_attributes list. * Read all VALUES into the dictionary_values list. * * @param rh a handle to parsed configuration. * @param filename the name of the dictionary file. * @return 0 on success, -1 on failure. */int rc_read_dictionary (rc_handle *rh, char const *filename){	FILE           *dictfd;	char            dummystr[AUTH_ID_LEN];	char            namestr[AUTH_ID_LEN];	char            valstr[AUTH_ID_LEN];	char            attrstr[AUTH_ID_LEN];	char            typestr[AUTH_ID_LEN];	char		optstr[AUTH_ID_LEN];	char		*cp, *ifilename;	int             line_no;	DICT_ATTR      *attr;	DICT_VALUE     *dval;	DICT_VENDOR    *dvend;	char            buffer[256];	int             value;	int             type;	unsigned attr_vendorspec = 0;	if ((dictfd = fopen (filename, "r")) == NULL)	{		rc_log(LOG_ERR, "rc_read_dictionary couldn't open dictionary %s: %s",				filename, strerror(errno));		return -1;	}	line_no = 0;	while (fgets (buffer, sizeof (buffer), dictfd) != NULL)	{		line_no++;		/* Skip empty space */		if (*buffer == '#' || *buffer == '\0' || *buffer == '\n' || \		    *buffer == '\r')		{			continue;		}		/* Strip out comments */		cp = strchr(buffer, '#');		if (cp != NULL)		{			*cp = '\0';		}		if (strncmp (buffer, "ATTRIBUTE", 9) == 0)		{			optstr[0] = '\0';			/* Read the ATTRIBUTE line */			if (sscanf (buffer, "%63s%63s%63s%63s%63s", dummystr, namestr,				    valstr, typestr, optstr) < 4)			{				rc_log(LOG_ERR, "rc_read_dictionary: invalid attribute on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			/*			 * Validate all entries			 */			if (strlen (namestr) > NAME_LENGTH)			{				rc_log(LOG_ERR, "rc_read_dictionary: invalid name length on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			if (!isdigit (*valstr))			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid value on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			value = atoi (valstr);			if (strcmp (typestr, "string") == 0)			{				type = PW_TYPE_STRING;			}			else if (strcmp (typestr, "integer") == 0)			{				type = PW_TYPE_INTEGER;			}			else if (strcmp (typestr, "ipaddr") == 0)			{				type = PW_TYPE_IPADDR;			}			else if (strcmp (typestr, "ipv6addr") == 0)			{				type = PW_TYPE_IPV6ADDR;			}			else if (strcmp (typestr, "ipv6prefix") == 0)			{				type = PW_TYPE_IPV6PREFIX;			}			else if (strcmp (typestr, "date") == 0)			{				type = PW_TYPE_DATE;			}			else			{				rc_log(LOG_ERR,				  "rc_read_dictionary: invalid type on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			dvend = NULL;			if (optstr[0] != '\0') {				char *cp1;				for (cp1 = optstr; cp1 != NULL; cp1 = cp) {					cp = strchr(cp1, ',');					if (cp != NULL) {						*cp = '\0';						cp++;					}					if (strncmp(cp1, "vendor=", 7) == 0)						cp1 += 7;					dvend = rc_dict_findvend(rh, cp1);					if (dvend == NULL) {						rc_log(LOG_ERR,						 "rc_read_dictionary: unknown Vendor-Id %s on line %d of dictionary %s",							 cp1, line_no, filename);						fclose(dictfd);						return -1;					}				}			}			/* Create a new attribute for the list */			if ((attr = malloc (sizeof (DICT_ATTR))) == NULL)			{				rc_log(LOG_CRIT, "rc_read_dictionary: out of memory");				fclose(dictfd);				return -1;			}			strcpy (attr->name, namestr);			attr->value = value | (attr_vendorspec << 16);			attr->type = type;			if (dvend != NULL) {				attr->value = value | (dvend->vendorpec << 16);			} else {				attr->value = value | (attr_vendorspec << 16);			}			/* Insert it into the list */			attr->next = rh->dictionary_attributes;			rh->dictionary_attributes = attr;		}		else if (strncmp (buffer, "VALUE", 5) == 0)		{			/* Read the VALUE line */			if (sscanf (buffer, "%63s%63s%63s%63s", dummystr, attrstr,				    namestr, valstr) != 4)			{				rc_log(LOG_ERR,			   "rc_read_dictionary: invalid value entry on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			/*			 * Validate all entries			 */			if (strlen (attrstr) > NAME_LENGTH)			{				rc_log(LOG_ERR,		      "rc_read_dictionary: invalid attribute length on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			if (strlen (namestr) > NAME_LENGTH)			{				rc_log(LOG_ERR,			   "rc_read_dictionary: invalid name length on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			if (!isdigit (*valstr))			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid value on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			value = atoi (valstr);			/* Create a new VALUE entry for the list */			if ((dval = malloc (sizeof (DICT_VALUE))) == NULL)			{				rc_log(LOG_CRIT, "rc_read_dictionary: out of memory");				fclose(dictfd);				return -1;			}			strcpy (dval->attrname, attrstr);			strcpy (dval->name, namestr);			dval->value = value;			/* Insert it into the list */			dval->next = rh->dictionary_values;			rh->dictionary_values = dval;		}                else if (strncmp (buffer, "$INCLUDE", 8) == 0)                {			/* Read the $INCLUDE line */			if (sscanf (buffer, "%63s%63s", dummystr, namestr) != 2)			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid include entry on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			ifilename = namestr;			/* Append directory if necessary */			if (namestr[0] != '/') {				cp = strrchr(filename, '/');				if (cp != NULL) {					ifilename = malloc(AUTH_ID_LEN);					*cp = '\0';					snprintf(ifilename, AUTH_ID_LEN, "%s/%s", filename, namestr);					*cp = '/';				}			}			if (rc_read_dictionary(rh, ifilename) < 0)			{				fclose(dictfd);				return -1;			}		}		else if (strncmp (buffer, "END-VENDOR", 10) == 0)		{			attr_vendorspec = 0;		}		else if (strncmp (buffer, "BEGIN-VENDOR", 12) == 0)		{			DICT_VENDOR *v;			/* Read the vendor name */			if (sscanf (buffer+12, "%63s", dummystr) != 1)			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid Vendor-Id on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			v = rc_dict_findvend(rh, dummystr);			if (v == NULL) {				rc_log(LOG_ERR,				 "rc_read_dictionary: unknown Vendor %s on line %d of dictionary %s",					 dummystr, line_no, filename);				fclose(dictfd);				return -1;			}			attr_vendorspec = v->vendorpec;		}		else if (strncmp (buffer, "VENDOR", 6) == 0)		{			/* Read the VALUE line */			if (sscanf (buffer, "%63s%63s%63s", dummystr, attrstr, valstr) != 3)			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid Vendor-Id on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			/* Validate all entries */			if (strlen (attrstr) > NAME_LENGTH)			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid attribute length on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			if (!isdigit (*valstr))			{				rc_log(LOG_ERR,				 "rc_read_dictionary: invalid Vendor-Id on line %d of dictionary %s",					 line_no, filename);				fclose(dictfd);				return -1;			}			value = atoi (valstr);			/* Create a new VENDOR entry for the list */			dvend = malloc(sizeof(DICT_VENDOR));			if (dvend == NULL)			{				rc_log(LOG_CRIT, "rc_read_dictionary: out of memory");				fclose(dictfd);				return -1;			}			strcpy (dvend->vendorname, attrstr);			dvend->vendorpec = value;			/* Insert it into the list */			dvend->next = rh->dictionary_vendors;			rh->dictionary_vendors = dvend;                }	}	fclose (dictfd);	return 0;}/** Lookup a DICT_ATTR by attribute number * * @param rh a handle to parsed configuration. * @param attribute the attribute ID. * @return the full attribute structure based on the attribute id number. */DICT_ATTR *rc_dict_getattr(rc_handle const *rh, int attribute){	DICT_ATTR      *attr;	attr = rh->dictionary_attributes;	while (attr != NULL)	{		if (attr->value == attribute)		{			return attr;		}		attr = attr->next;	}	return NULL;}/** Lookup a DICT_ATTR by its name * * @param rh a handle to parsed configuration. * @param attrname the attribute name. * * @return the full attribute structure based on the attribute name. */DICT_ATTR *rc_dict_findattr(rc_handle const *rh, char const *attrname){	DICT_ATTR      *attr;	attr = rh->dictionary_attributes;	while (attr != NULL)	{		if (strcasecmp (attr->name, attrname) == 0)		{			return attr;		}		attr = attr->next;	}	return NULL;}/** Lookup a DICT_VALUE by its name * * @param rh a handle to parsed configuration. * @param valname the value name. * @return the full value structure based on the value name. */DICT_VALUE *rc_dict_findval(rc_handle const *rh, char const *valname){	DICT_VALUE     *val;	val = rh->dictionary_values;	while (val != NULL)	{		if (strcasecmp (val->name, valname) == 0)		{			return val;		}		val = val->next;	}	return NULL;}/** Lookup a DICT_VENDOR by its name * * @param rh a handle to parsed configuration. * @param valname the vendor name. * @return the full vendor structure based on the vendor name. */DICT_VENDOR *rc_dict_findvend(rc_handle const *rh, char const *vendorname){	DICT_VENDOR	*vend;	for (vend = rh->dictionary_vendors; vend != NULL; vend = vend->next)		if (strcasecmp(vend->vendorname, vendorname) == 0)			return vend;	return NULL;}/** Lookup a DICT_VENDOR by its IANA number * * @param rh a handle to parsed configuration. * @param vendorpec the vendor ID. * @return the full vendor structure based on the vendor id number. */DICT_VENDOR *rc_dict_getvend (rc_handle const *rh, int vendorpec){        DICT_VENDOR      *vend;	for (vend = rh->dictionary_vendors; vend != NULL; vend = vend->next)		if (vend->vendorpec == vendorpec)			return vend;	return NULL;}/** Get DICT_VALUE based on attribute name and integer value number * * @param rh a handle to parsed configuration. * @param value the attribute value. * @param attrname the attribute name. * @return the full value structure based on the actual value and the associated attribute name. */DICT_VALUE *rc_dict_getval(rc_handle const *rh, uint32_t value, char const *attrname){	DICT_VALUE     *val;	val = rh->dictionary_values;	while (val != NULL)	{		if (strcmp (val->attrname, attrname) == 0 &&				val->value == value)		{			return val;		}		val = val->next;	}	return NULL;}/** Frees the allocated av lists * * @param rh a handle to parsed configuration. */void rc_dict_free(rc_handle *rh){	DICT_ATTR	*attr, *nattr;	DICT_VALUE	*val, *nval;	DICT_VENDOR	*vend, *nvend;	for (attr = rh->dictionary_attributes; attr != NULL; attr = nattr) {		nattr = attr->next;		free(attr);	}	for (val = rh->dictionary_values; val != NULL; val = nval) {		nval = val->next;		free(val);	}	for (vend = rh->dictionary_vendors; vend != NULL; vend = nvend) {		nvend = vend->next;		free(vend);	}	rh->dictionary_attributes = NULL;	rh->dictionary_values = NULL;	rh->dictionary_vendors = NULL;}
 |