ant2009
ant2009

Reputation: 22526

reading in lines of text from file

gcc 4.5.1 c89

I am using the following code to read in a line of text from a configuration file. The configuration file is small at the moment, will grow with new fields to add. And I can pretty much design what the configuration file will look like myself. So I have done it in the following way:

config.cfg

# Configuration file to be loaded 

# protocol to use either ISDN, SIP, 
protocol: ISDN

# run application in the following mode, makecall or waitcall
mode: makecall

I have used the colon as a way to search for the type of configuration that will be needed. I am just wondering if there a better way to do this?

My code I have used is as follows:

static int load_config(FILE *fp)
{
    char line_read[LINE_SIZE] = {0};
    char *type = NULL;
    char field[LINE_SIZE] = {0};
    char *carriage_return = NULL;

    /* Read each line */
    while(fgets(line_read, LINE_SIZE, fp) != NULL) {
        if(line_read != NULL) {
            /* Ignore any hashs and blank lines */
            if((line_read[0] != '#') && (strlen(line_read) > 1)) {
                /* I don't like the carriage return, so remove it. */
                carriage_return = strrchr(line_read, '\n');
                if(carriage_return != NULL) {
                    /* carriage return found so relace with a null */
                    *carriage_return = '\0';
                }

                /* Parse line_read to extract the field name i.e. protocol, mode, etc */
                parse_string(field, line_read);
                if(field != NULL) {
                    type = strchr(line_read, ':');
                    type+=2; /* Point to the first character after the space */

                    if(strcmp("protocol", field) == 0) {
                        /* Check the protocol type */
                        printf("protocol [ %s ]\n", type);
                    }
                    else if (strcmp("mode", field) == 0) {
                        /* Check the mode type */
                        printf("mode [ %s ]\n", type);
                    }
                }
            }
        }
    }

    return TRUE;
}

/* Extract the field name for the read in line from the configuration file. */
static void parse_string(char *dest, const char *string)
{
    /* Copy string up to the colon to determine configuration type */
    while(*string != ':') {
        *dest++ = *string++;
    }
    /* Insert nul terminator */
    *dest = '\0';
}

Upvotes: 0

Views: 446

Answers (4)

user411313
user411313

Reputation: 3990

easier should be:

static int load_config(FILE *fp)
{
  int r=0;
  char line[LINE_SIZE], field[LINE_SIZE], type[LINE_SIZE], dummy[LINE_SIZE];

  /* Read each line */
  while( fgets(line, LINE_SIZE, fp) )
  {
    if( strchr(line,'\n') ) *strchr(line,'\n')=0;
    if( 3==sscanf(line,"%[^: ]%[: ]%s,field,dummy,type) )
      ++r,printf("\n %s [ %s ]",field,type);
  }
  return r;
}

Upvotes: 1

wnoise
wnoise

Reputation: 9932

The standard answer to simple parsing problem is to use lex and yacc.

However, as you have freedom to set the form of the configuration file, you should use one of the numerous libraries that implement various configuration file formats, and just use that one.

http://www.google.com/search?q=configuration+file+parser

http://www.nongnu.org/confuse/ for instance, would seem to cover your needs adequately, but take a look at various other options that might be simpler as well.

Upvotes: 1

LumpN
LumpN

Reputation: 21

If you can design what the configuration file will look like i'd go for XML, parsing it with Expat. It's painless.

Upvotes: 1

Bret Savage
Bret Savage

Reputation: 201

<regex.h> is your friend!

http://www.gnu.org/s/libc/manual/html_node/Regular-Expressions.html

Upvotes: 1

Related Questions