]> git.sur5r.net Git - bacula/bacula/blobdiff - bacula/src/lib/parse_conf.c
This commit was manufactured by cvs2svn to create tag
[bacula/bacula] / bacula / src / lib / parse_conf.c
index 77590b139bbc329badb5e2b06c83b8357ca20e80..2c3ac1b43e3c0941b811c02427d695f284453cef 100755 (executable)
@@ -34,7 +34,7 @@
  */
 
 /*
-   Copyright (C) 2000, 2001, 2002 Kern Sibbald and John Walker
+   Copyright (C) 2000-2003 Kern Sibbald and John Walker
 
    This program is free software; you can redistribute it and/or
    modify it under the terms of the GNU General Public License as
@@ -115,6 +115,7 @@ static struct s_mtypes msg_types[] = {
    {"skipped",       M_SKIPPED},
    {"mount",         M_MOUNT},
    {"terminate",     M_TERM},
+   {"restored",      M_RESTORED},
    {"all",           M_MAX+1},
    {NULL,           0}
 };
@@ -130,7 +131,6 @@ static void prtmsg(void *sock, char *fmt, ...)
    va_end(arg_ptr);
 }
 
-#ifdef DEBUG
 char *res_to_str(int rcode)
 {
    if (rcode < r_first || rcode > r_last) {
@@ -139,7 +139,6 @@ char *res_to_str(int rcode)
       return resources[rcode-r_first].name;
    }
 }
-#endif /* DEBUG */
 
 
 /* 
@@ -167,14 +166,15 @@ void init_resource(int type, struct res_items *items)
            *(int *)(items[i].value) = items[i].default_value;
         } else if (items[i].handler == store_int64) {
            *(int64_t *)(items[i].value) = items[i].default_value;
-        } else if (items[i].handler == store_size ||
-                   items[i].handler == store_time) {
-           *(uint64_t *)(items[i].value) = items[i].default_value;
+        } else if (items[i].handler == store_size) {
+           *(uint64_t *)(items[i].value) = (uint64_t)items[i].default_value;
+        } else if (items[i].handler == store_time) {
+           *(utime_t *)(items[i].value) = (utime_t)items[i].default_value;
         }
       }
       /* If this triggers, take a look at lib/parse_conf.h */
       if (i >= MAX_RES_ITEMS) {
-         Emsg1(M_ABORT, 0, _("Too many items in %s resource\n"), resources[rindex]);
+         Emsg1(M_ERROR_TERM, 0, _("Too many items in %s resource\n"), resources[rindex]);
       }
    }
 }
@@ -186,7 +186,7 @@ void store_msgs(LEX *lc, struct res_items *item, int index, int pass)
    int token;
    char *cmd;
    POOLMEM *dest;
-   int dest_len;
+   int dest_len;    
 
    Dmsg2(200, "store_msgs pass=%d code=%d\n", pass, item->code);
    if (pass == 1) {
@@ -207,23 +207,20 @@ void store_msgs(LEX *lc, struct res_items *item, int index, int pass)
               cmd = res_all.res_msgs.mail_cmd;
            }
            dest = get_pool_memory(PM_MESSAGE);
-           dest_len = 0;
            dest[0] = 0;
+           dest_len = 0;
            /* Pick up comma separated list of destinations */
            for ( ;; ) {
-              token = lex_get_token(lc);    /* scan destination */
-              if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-                  scan_err1(lc, "expected a message destination, got: %s", lc->str);
-              }
-              dest = (char *) check_pool_memory_size(dest, dest_len + lc->str_len + 2);
+              token = lex_get_token(lc, T_NAME);   /* scan destination */
+              dest = check_pool_memory_size(dest, dest_len + lc->str_len + 2);
               if (dest[0] != 0) {
                   strcat(dest, " ");  /* separate multiple destinations with space */
                  dest_len++;
               }
               strcat(dest, lc->str);
               dest_len += lc->str_len;
-               Dmsg2(100, "store_msgs newdest=%s: dest=%s:\n", lc->str, dest);
-              token = lex_get_token(lc);
+               Dmsg2(100, "store_msgs newdest=%s: dest=%s:\n", lc->str, NPRT(dest));
+              token = lex_get_token(lc, T_ALL);
               if (token == T_COMMA) { 
                  continue;           /* get another destination */
               }
@@ -232,7 +229,7 @@ void store_msgs(LEX *lc, struct res_items *item, int index, int pass)
               }
               break;
            }
-            Dmsg1(200, "mail_cmd=%s\n", cmd);
+            Dmsg1(200, "mail_cmd=%s\n", NPRT(cmd));
            scan_types(lc, (MSGS *)(item->value), item->code, dest, cmd);
            free_pool_memory(dest);
             Dmsg0(200, "done with dest codes\n");
@@ -241,15 +238,11 @@ void store_msgs(LEX *lc, struct res_items *item, int index, int pass)
         case MD_APPEND:              /* append */
            dest = get_pool_memory(PM_MESSAGE);
            /* Pick up a single destination */
-           token = lex_get_token(lc);    /* scan destination */
-           if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-               scan_err1(lc, "expected a message destination, got: %s", lc->str);
-           }
-           dest = check_pool_memory_size(dest, dest_len + lc->str_len + 2);
-           strcpy(dest, lc->str);
+           token = lex_get_token(lc, T_NAME);   /* scan destination */
+           pm_strcpy(&dest, lc->str);
            dest_len = lc->str_len;
-           token = lex_get_token(lc);
-            Dmsg1(200, "store_msgs dest=%s:\n", dest);
+           token = lex_get_token(lc, T_ALL);
+            Dmsg1(200, "store_msgs dest=%s:\n", NPRT(dest));
            if (token != T_EQUALS) {
                scan_err1(lc, "expected an =, got: %s", lc->str); 
            }
@@ -276,15 +269,12 @@ void store_msgs(LEX *lc, struct res_items *item, int index, int pass)
  */
 static void scan_types(LEX *lc, MSGS *msg, int dest_code, char *where, char *cmd)
 {
-   int token, i, found, quit, is_not;
-   int msg_type;
+   int i, found, quit, is_not;
+   int msg_type = 0;
    char *str;
 
    for (quit=0; !quit;) {
-      token = lex_get_token(lc);            /* expect at least one type */       
-      if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-         scan_err1(lc, "expected a message type, got: %s", lc->str);
-      }
+      lex_get_token(lc, T_NAME);           /* expect at least one type */       
       found = FALSE;
       if (lc->str[0] == '!') {
         is_not = TRUE;
@@ -294,7 +284,7 @@ static void scan_types(LEX *lc, MSGS *msg, int dest_code, char *where, char *cmd
         str = &lc->str[0];
       }
       for (i=0; msg_types[i].name; i++) {
-        if (strcmp(str, msg_types[i].name) == 0) {
+        if (strcasecmp(str, msg_types[i].name) == 0) {
            msg_type = msg_types[i].token;
            found = TRUE;
            break;
@@ -302,6 +292,7 @@ static void scan_types(LEX *lc, MSGS *msg, int dest_code, char *where, char *cmd
       }
       if (!found) {
          scan_err1(lc, "message type: %s not found", str);
+        /* NOT REACHED */
       }
 
       if (msg_type == M_MAX+1) {        /* all? */
@@ -319,7 +310,7 @@ static void scan_types(LEX *lc, MSGS *msg, int dest_code, char *where, char *cmd
         break;
       }
       Dmsg0(200, "call lex_get_token() to eat comma\n");
-      token = lex_get_token(lc);         /* eat comma */
+      lex_get_token(lc, T_ALL);         /* eat comma */
    }
    Dmsg0(200, "Done scan_types()\n");
 }
@@ -331,18 +322,9 @@ static void scan_types(LEX *lc, MSGS *msg, int dest_code, char *where, char *cmd
  */
 void store_name(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected an identifier or string, got: %s", lc->str);
-   } else if (lc->str_len > MAX_RES_NAME_LENGTH) {
-      scan_err3(lc, "name %s length %d too long, max is %d\n", lc->str, 
-        lc->str_len, MAX_RES_NAME_LENGTH);
-   } else {
-      /* Store the name both pass 1 and pass 2 */
-      *(item->value) = bstrdup(lc->str);
-   }
+   lex_get_token(lc, T_NAME);
+   /* Store the name both pass 1 and pass 2 */
+   *(item->value) = bstrdup(lc->str);
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
 }
@@ -354,18 +336,10 @@ void store_name(LEX *lc, struct res_items *item, int index, int pass)
  */
 void store_strname(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected an identifier or string, got: %s", lc->str);
-   } else if (lc->str_len > MAX_RES_NAME_LENGTH) {
-      scan_err3(lc, "name %s length %d too long, max is %d\n", lc->str, 
-        lc->str_len, MAX_RES_NAME_LENGTH);
-   } else {
-      /* Store the name */
-      if (pass == 1)
-        *(item->value) = bstrdup(lc->str);
+   lex_get_token(lc, T_NAME);
+   /* Store the name */
+   if (pass == 1) {
+      *(item->value) = bstrdup(lc->str);
    }
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
@@ -376,33 +350,27 @@ void store_strname(LEX *lc, struct res_items *item, int index, int pass)
 /* Store a string at specified address */
 void store_str(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected an identifier or string, got: %s", lc->str);
-   } else {
-      if (pass == 1) {
-        *(item->value) = bstrdup(lc->str);
-      }
+   lex_get_token(lc, T_STRING);
+   if (pass == 1) {
+      *(item->value) = bstrdup(lc->str);
    }
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
 }
 
-/* Store a directory name at specified address */
+/*
+ * Store a directory name at specified address. Note, we do
+ *   shell expansion except if the string begins with a vertical
+ *   bar (i.e. it will likely be passed to the shell later).
+ */
 void store_dir(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected an identifier or string, got: %s", lc->str);
-   } else {
-      if (pass == 1) {
-        do_shell_expansion(lc->str);
-        *(item->value) = bstrdup(lc->str);
+   lex_get_token(lc, T_STRING);
+   if (pass == 1) {
+      if (lc->str[0] != '|') {
+        do_shell_expansion(lc->str, sizeof(lc->str));
       }
+      *(item->value) = bstrdup(lc->str);
    }
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
@@ -412,26 +380,22 @@ void store_dir(LEX *lc, struct res_items *item, int index, int pass)
 /* Store a password specified address in MD5 coding */
 void store_password(LEX *lc, struct res_items *item, int index, int pass)
 {
-   unsigned int token, i, j;
+   unsigned int i, j;
    struct MD5Context md5c;
    unsigned char signature[16];
    char sig[100];
 
 
-   token = lex_get_token(lc);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected an identifier or string, got: %s\n", lc->str);
-   } else {
-      if (pass == 1) {
-        MD5Init(&md5c);
-        MD5Update(&md5c, (unsigned char *) (lc->str), lc->str_len);
-        MD5Final(signature, &md5c);
-        for (i = j = 0; i < sizeof(signature); i++) {
-            sprintf(&sig[j], "%02x", signature[i]); 
-           j += 2;
-        }
-        *(item->value) = bstrdup(sig);
+   lex_get_token(lc, T_STRING);
+   if (pass == 1) {
+      MD5Init(&md5c);
+      MD5Update(&md5c, (unsigned char *) (lc->str), lc->str_len);
+      MD5Final(signature, &md5c);
+      for (i = j = 0; i < sizeof(signature); i++) {
+         sprintf(&sig[j], "%02x", signature[i]); 
+        j += 2;
       }
+      *(item->value) = bstrdup(sig);
    }
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
@@ -444,13 +408,9 @@ void store_password(LEX *lc, struct res_items *item, int index, int pass)
  */
 void store_res(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
    RES *res;
 
-   token = lex_get_token(lc);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected a Resource name, got: %s", lc->str);
-   }
+   lex_get_token(lc, T_NAME);
    if (pass == 2) {
      res = GetResWithName(item->code, lc->str);
      if (res == NULL) {
@@ -467,18 +427,8 @@ void store_res(LEX *lc, struct res_items *item, int index, int pass)
 /* Store an integer at specified address */
 void store_int(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   if (token != T_NUMBER || !is_a_number(lc->str)) {
-      scan_err1(lc, "expected an integer number, got: %s", lc->str);
-   } else {
-      errno = 0;
-      *(int *)(item->value) = (int)strtod(lc->str, NULL);
-      if (errno != 0) {
-         scan_err1(lc, "expected an integer number, got: %s", lc->str);
-      }
-   }
+   lex_get_token(lc, T_INT32);
+   *(int *)(item->value) = lc->int32_val;
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
 }
@@ -486,19 +436,8 @@ void store_int(LEX *lc, struct res_items *item, int index, int pass)
 /* Store a positive integer at specified address */
 void store_pint(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   if (token != T_NUMBER || !is_a_number(lc->str)) {
-      scan_err1(lc, "expected a positive integer number, got: %s", lc->str);
-   } else {
-      errno = 0;
-      token = (int)strtod(lc->str, NULL);
-      if (errno != 0 || token < 0) {
-         scan_err1(lc, "expected a postive integer number, got: %s", lc->str);
-      }
-      *(int *)(item->value) = token;
-   }
+   lex_get_token(lc, T_PINT32);
+   *(int *)(item->value) = lc->pint32_val;
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
 }
@@ -507,18 +446,8 @@ void store_pint(LEX *lc, struct res_items *item, int index, int pass)
 /* Store an 64 bit integer at specified address */
 void store_int64(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   Dmsg2(400, "int64=:%s: %f\n", lc->str, strtod(lc->str, NULL)); 
-   if (token != T_NUMBER || !is_a_number(lc->str)) {
-      scan_err1(lc, "expected an integer number, got: %s", lc->str);
-   } else {
-      errno = 0;
-      *(int64_t *)(item->value) = (int64_t)strtod(lc->str, NULL);
-      if (errno != 0)
-         scan_err1(lc, "expected an integer number, got: %s", lc->str);
-   }
+   lex_get_token(lc, T_INT64);
+   *(int64_t *)(item->value) = lc->int64_val;
    scan_to_eol(lc);
    set_bit(index, res_all.hdr.item_present);
 }
@@ -526,65 +455,20 @@ void store_int64(LEX *lc, struct res_items *item, int index, int pass)
 /* Store a size in bytes */
 void store_size(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token, i, ch;
-   uint64_t value;
-   int mod[]  = {'*', 'k', 'm', 'g', 0}; /* first item * not used */
-   uint64_t mult[] = {1,            /* byte */
-                     1024,          /* kilobyte */
-                     1048576,       /* megabyte */
-                     1073741824};   /* gigabyte */
-
-#ifdef we_have_a_compiler_that_works
-   int mod[]  = {'*', 'k', 'm', 'g', 't', 0};
-   uint64_t mult[] = {1,            /* byte */
-                     1024,          /* kilobyte */
-                     1048576,       /* megabyte */
-                     1073741824,    /* gigabyte */
-                     1099511627776};/* terabyte */
-#endif
+   int token;
+   uint64_t uvalue;
 
    Dmsg0(400, "Enter store_size\n");
-   token = lex_get_token(lc);
+   token = lex_get_token(lc, T_ALL);
    errno = 0;
    switch (token) {
    case T_NUMBER:
-      Dmsg2(400, "size num=:%s: %f\n", lc->str, strtod(lc->str, NULL)); 
-      value = (uint64_t)strtod(lc->str, NULL);
-      if (errno != 0 || token < 0) {
-         scan_err1(lc, "expected a size number, got: %s", lc->str);
-      }
-      *(uint64_t *)(item->value) = value;
-      break;
    case T_IDENTIFIER:
-   case T_STRING:
-      /* Look for modifier */
-      ch = lc->str[lc->str_len - 1];
-      i = 0;
-      if (ISALPHA(ch)) {
-        if (ISUPPER(ch)) {
-           ch = tolower(ch);
-        }
-        while (mod[++i] != 0) {
-           if (ch == mod[i]) {
-              lc->str_len--;
-              lc->str[lc->str_len] = 0; /* strip modifier */
-              break;
-           }
-        }
-      }
-      if (mod[i] == 0 || !is_a_number(lc->str)) {
-         scan_err1(lc, "expected a size number, got: %s", lc->str);
-      }
-      Dmsg3(400, "size str=:%s: %f i=%d\n", lc->str, strtod(lc->str, NULL), i);
-
-      value = (uint64_t)strtod(lc->str, NULL);
-      Dmsg1(400, "Int value = %d\n", (int)value);
-      if (errno != 0 || value < 0) {
+   case T_UNQUOTED_STRING:
+      if (!size_to_uint64(lc->str, lc->str_len, &uvalue)) {
          scan_err1(lc, "expected a size number, got: %s", lc->str);
       }
-      *(uint64_t *)(item->value) = value * mult[i];
-      Dmsg2(400, "Full value = %f %" lld "\n", strtod(lc->str, NULL) * mult[i],
-         value *mult[i]);
+      *(uint64_t *)(item->value) = uvalue;
       break;
    default:
       scan_err1(lc, "expected a size, got: %s", lc->str);
@@ -600,30 +484,37 @@ void store_size(LEX *lc, struct res_items *item, int index, int pass)
 void store_time(LEX *lc, struct res_items *item, int index, int pass)
 {
    int token; 
-   btime_t value;
+   utime_t utime;
+   char period[500];
 
-   token = lex_get_token(lc);
+   token = lex_get_token(lc, T_ALL);
    errno = 0;
    switch (token) {
    case T_NUMBER:
-      value = (btime_t)strtod(lc->str, NULL);
-      if (errno != 0 || value < 0) {
-         scan_err1(lc, "expected a time period, got: %s", lc->str);
-      }
-      *(btime_t *)(item->value) = value;
-      break;
    case T_IDENTIFIER:
-   case T_STRING:
-      if (!string_to_btime(lc->str, &value)) {
-         scan_err1(lc, "expected a time period, got: %s", lc->str);
+   case T_UNQUOTED_STRING:
+      bstrncpy(period, lc->str, sizeof(period));
+      if (lc->ch == ' ') {
+        token = lex_get_token(lc, T_ALL);
+        switch (token) {
+        case T_IDENTIFIER:
+        case T_UNQUOTED_STRING:
+           bstrncat(period, lc->str, sizeof(period));
+           break;
+        }
       }
-      *(btime_t *)(item->value) = value;
+      if (!duration_to_utime(period, &utime)) {
+         scan_err1(lc, "expected a time period, got: %s", period);
+      }
+      *(utime_t *)(item->value) = utime;
       break;
    default:
       scan_err1(lc, "expected a time period, got: %s", lc->str);
       break;
    }
-   scan_to_eol(lc);
+   if (token != T_EOL) {
+      scan_to_eol(lc);
+   }
    set_bit(index, res_all.hdr.item_present);
 }
 
@@ -631,15 +522,10 @@ void store_time(LEX *lc, struct res_items *item, int index, int pass)
 /* Store a yes/no in a bit field */
 void store_yesno(LEX *lc, struct res_items *item, int index, int pass)
 {
-   int token;
-
-   token = lex_get_token(lc);
-   lcase(lc->str);
-   if (token != T_IDENTIFIER && token != T_STRING && token != T_QUOTED_STRING) {
-      scan_err1(lc, "expected an identifier or string, got: %s", lc->str);
-   } else if (strcmp(lc->str, "yes") == 0) {
+   lex_get_token(lc, T_NAME);
+   if (strcasecmp(lc->str, "yes") == 0) {
       *(int *)(item->value) |= item->code;
-   } else if (strcmp(lc->str, "no") == 0) {
+   } else if (strcasecmp(lc->str, "no") == 0) {
       *(int *)(item->value) &= ~(item->code);
    } else {
       scan_err1(lc, "Expect a YES or NO, got: %s", lc->str);
@@ -649,36 +535,6 @@ void store_yesno(LEX *lc, struct res_items *item, int index, int pass)
 }
 
 
-/*
- * Scan to "logical" end of line. I.e. end of line,
- * or semicolon.
- */
-void scan_to_eol(LEX *lc)
-{
-   int token;
-   Dmsg0(150, "start scan to eof\n");
-   while ((token = lex_get_token(lc)) != T_EOL) {
-   }
-   Dmsg0(150, "done scan to eof\n");
-}
-
-   
-/*
- * Format a scanner error message 
- */
-void s_err(char *file, int line, LEX *lc, char *msg, ...)
-{
-   va_list arg_ptr;
-   char buf[MAXSTRING];
-
-   va_start(arg_ptr, msg);
-   bvsnprintf(buf, sizeof(buf), msg, arg_ptr);
-   va_end(arg_ptr);
-     
-   e_msg(file, line, M_ABORT, 0, "Config error: %s,\n\
-            : Line %d, col %d of file %s\n%s\n",
-      buf, lc->line_no, lc->col_no, lc->fname, lc->line);
-}
 
 void LockRes()
 {
@@ -753,9 +609,10 @@ void
 parse_config(char *cf)
 {
    LEX *lc = NULL;
-   int token, i, res_type, pass;
+   int token, i, pass;
+   int res_type = 0;
    enum parse_state state = p_none;
-   struct res_items *items;
+   struct res_items *items = NULL;
    int level = 0;
 
    /* Make two passes. The first builds the name symbol table,
@@ -764,8 +621,8 @@ parse_config(char *cf)
    Dmsg0(200, "Enter parse_config()\n");
    for (pass=1; pass <= 2; pass++) {
       Dmsg1(200, "parse_config pass %d\n", pass);
-      lc = lex_open_file(lc, cf);
-      while ((token=lex_get_token(lc)) != T_EOF) {
+      lc = lex_open_file(lc, cf, NULL);
+      while ((token=lex_get_token(lc, T_ALL)) != T_EOF) {
          Dmsg1(150, "parse got token=%s\n", lex_tok_to_str(token));
         switch (state) {
            case p_none:
@@ -774,10 +631,10 @@ parse_config(char *cf)
               }
               if (token != T_IDENTIFIER) {
                   scan_err1(lc, "Expected a Resource name identifier, got: %s", lc->str);
+                 /* NOT REACHED */
               }
-              lcase(lc->str);
               for (i=0; resources[i].name; i++)
-                 if (strcmp(resources[i].name, lc->str) == 0) {
+                 if (strcasecmp(resources[i].name, lc->str) == 0) {
                     state = p_resource;
                     items = resources[i].items;
                     res_type = resources[i].rcode;
@@ -786,6 +643,7 @@ parse_config(char *cf)
                  }
               if (state == p_none) {
                   scan_err1(lc, "expected resource name, got: %s", lc->str);
+                 /* NOT REACHED */
               }
               break;
            case p_resource:
@@ -796,14 +654,19 @@ parse_config(char *cf)
                  case T_IDENTIFIER:
                     if (level != 1) {
                         scan_err1(lc, "not in resource definition: %s", lc->str);
+                       /* NOT REACHED */
                     }
-                    lcase(lc->str);
                     for (i=0; items[i].name; i++) {
-                       if (strcmp(items[i].name, lc->str) == 0) {
-                          token = lex_get_token(lc);
-                           Dmsg1 (150, "in T_IDENT got token=%s\n", lex_tok_to_str(token));
-                          if (token != T_EQUALS) {
-                              scan_err1(lc, "expected an equals, got: %s", lc->str);
+                       if (strcasecmp(items[i].name, lc->str) == 0) {
+                          /* If the ITEM_NO_EQUALS flag is set we do NOT              
+                           *   scan for = after the keyword  */
+                          if (!(items[i].flags & ITEM_NO_EQUALS)) {
+                             token = lex_get_token(lc, T_ALL);
+                              Dmsg1 (150, "in T_IDENT got token=%s\n", lex_tok_to_str(token));
+                             if (token != T_EQUALS) {
+                                 scan_err1(lc, "expected an equals, got: %s", lc->str);
+                                /* NOT REACHED */
+                             }
                           }
                            Dmsg1(150, "calling handler for %s\n", items[i].name);
                           /* Call item handler */
@@ -815,7 +678,9 @@ parse_config(char *cf)
                     if (i >= 0) {
                         Dmsg2(150, "level=%d id=%s\n", level, lc->str);
                         Dmsg1(150, "Keyword = %s\n", lc->str);
-                        scan_err1(lc, "Keyword %s not permitted in this resource", lc->str);
+                        scan_err1(lc, "Keyword \"%s\" not permitted in this resource.\n"
+                           "Perhaps you left the trailing brace off of the previous resource.", lc->str);
+                       /* NOT REACHED */
                     }
                     break;
 
@@ -832,12 +697,17 @@ parse_config(char *cf)
                  default:
                      scan_err2(lc, "unexpected token %d %s in resource definition",    
                        token, lex_tok_to_str(token));
+                    /* NOT REACHED */
               }
               break;
            default:
                scan_err1(lc, "Unknown parser state %d\n", state);
+              /* NOT REACHED */
         }
       }
+      if (state != p_none) {
+         scan_err0(lc, "End of conf file reached with unclosed resource.");
+      }
       if (debug_level > 50 && pass == 2) {
         int i;
         for (i=r_first; i<=r_last; i++) {