actube/src/cw/cfg.c

610 lines
8.9 KiB
C
Raw Normal View History

2022-07-31 01:22:11 +02:00
#include <string.h>
2022-07-31 09:49:14 +02:00
#include <ctype.h>
2022-07-31 01:22:11 +02:00
#include <mavl.h>
#include "cw.h"
#include "cfg.h"
2022-07-31 17:15:32 +02:00
#include "val.h"
2022-07-31 01:22:11 +02:00
static const char *nextc(const char *s)
{
while (isdigit(*s))
s++;
return s;
}
static int cmp0(const char *s1, const char *s2)
{
const char *d1, *d2;
int i1, i2, i;
d1 = strchr(s1, '.');
if (d1 == NULL)
return strcmp(s1, s2);
d2 = strchr(s2, '.');
if (d2 == NULL)
return strcmp(s1, s2);
if ((d1 - s1) != (d2 - s2))
return strcmp(s1, s2);
if (strncmp(s1, s2, (d1 - s1)) != 0)
return strcmp(s1, s2);
if (isdigit(d1[1])) {
i1 = atoi(d1 + 1);
} else {
return cmp0(d1 + 1, d2 + 1);
}
if (isdigit(d2[1])) {
i2 = atoi(d2 + 1);
} else {
return cmp0(d1 + 1, d2 + 1);
}
i = i1 - i2;
if (i == 0) {
return cmp0(nextc(d1 + 1), nextc(d2 + 1));
}
return i;
}
static int cmp(const void *k1, const void *k2)
{
struct cw_Cfg_entry *e1, *e2;
e1 = (struct cw_Cfg_entry *) k1;
e2 = (struct cw_Cfg_entry *) k2;
return cmp0(e1->key, e2->key);
/* return strcmp(e1->key,e2->key);*/
2022-07-31 01:22:11 +02:00
}
static void del(void *ptr)
{
struct cw_Cfg_entry *e;
e = (struct cw_Cfg_entry *) ptr;
free((void *) e->key);
free((void *) e->val);
2022-07-31 01:22:11 +02:00
}
cw_Cfg_t *cw_cfg_create()
2022-07-31 01:22:11 +02:00
{
return mavl_create(cmp, del, sizeof(struct cw_Cfg_entry));
}
2022-07-31 01:22:11 +02:00
int cw_cfg_set(cw_Cfg_t * cfg, const char *key, const char *val)
2022-07-31 01:22:11 +02:00
{
struct cw_Cfg_entry e;
2022-07-31 01:22:11 +02:00
int replaced;
e.key = cw_strdup(key);
if (!e.key)
return 0;
e.val = cw_strdup(val);
if (!e.val) {
free((void *) e.key);
2022-07-31 01:22:11 +02:00
return 0;
}
void *rc = mavl_replace(cfg, &e, &replaced);
if (!rc) {
2022-07-31 01:22:11 +02:00
del(&e);
return 0;
}
if (!replaced)
return 1;
return -1;
}
const char *cw_cfg_get(cw_Cfg_t * cfg, const char *key, const char *def)
{
struct cw_Cfg_entry e, *r;
e.key = key;
r = mavl_get(cfg, &e);
if (!r)
return def;
return r->val;
}
2022-07-31 17:15:32 +02:00
void cw_cfg_dump(cw_Cfg_t * cfg)
2022-07-31 01:22:11 +02:00
{
mavliter_t it;
struct cw_Cfg_entry *e;
mavliter_init(&it, cfg);
mavliter_foreach(&it) {
2022-07-31 01:22:11 +02:00
e = mavliter_get(&it);
printf("%s: '%s'\n", e->key, e->val);
2022-07-31 01:22:11 +02:00
//cw_dbg(dbglevel,"%s%s :%s: %s",prefix,data->key,type->get_type_name(data), value);
}
}
2022-07-31 09:49:14 +02:00
struct parser {
int line;
int pos;
int prevpos;
char error[256];
int quote;
FILE *f;
int (*getc)(struct parser *);
void (*ungetc)(struct parser *);
};
static int get_char(struct parser *p)
{
int c;
c = fgetc(p->f);
2022-07-31 09:49:14 +02:00
p->pos++;
if (c == '\n') {
p->prevpos = p->pos;
p->line++;
p->pos = 0;
2022-07-31 09:49:14 +02:00
}
return c;
}
static void unget_char(struct parser *p, int c)
{
ungetc(c, p->f);
if (c == '\n') {
2022-07-31 09:49:14 +02:00
p->line--;
p->pos = p->prevpos;
} else
2022-07-31 09:49:14 +02:00
p->pos--;
}
static int get_char_q(struct parser *p)
{
int c;
while (1) {
2022-07-31 09:49:14 +02:00
c = get_char(p);
if (c == EOF || c == '\n')
2022-07-31 09:49:14 +02:00
return c;
if (c == '"' && !p->quote) {
p->quote = 1;
2022-07-31 09:49:14 +02:00
continue;
}
if (c == '"' && p->quote) {
p->quote = 0;
2022-07-31 09:49:14 +02:00
continue;
}
break;
}
2022-07-31 09:49:14 +02:00
if (!p->quote)
return c;
if (c != '\\')
2022-07-31 09:49:14 +02:00
return c;
2022-07-31 09:49:14 +02:00
c = get_char(p);
switch (c) {
2022-07-31 09:49:14 +02:00
case EOF:
return c;
case 'n':
return '\n';
2022-07-31 09:49:14 +02:00
case '\\':
return '\\';
case '"':
return '"';
default:
unget_char(p, c);
2022-07-31 09:49:14 +02:00
return '\\';
}
2022-07-31 09:49:14 +02:00
/* We will never reach here */
/* return c; */
2022-07-31 09:49:14 +02:00
}
static int skip_chars(struct parser *p, const char *chars)
2022-07-31 09:49:14 +02:00
{
int c;
while ((c = get_char(p)) != EOF) {
if (strchr(chars, c))
2022-07-31 09:49:14 +02:00
continue;
return c;
}
return c;
}
static int skip_to_chars(struct parser *p, const char *chars)
2022-07-31 09:49:14 +02:00
{
int c;
while ((c = get_char(p)) != EOF) {
if (strchr(chars, c))
2022-07-31 09:49:14 +02:00
return c;
}
return c;
}
static int read_key(struct parser *p, char *key, int max_len)
2022-07-31 09:49:14 +02:00
{
int c, n;
2022-07-31 09:49:14 +02:00
do {
c = skip_chars(p, " \t\n\a\v");
2022-07-31 09:49:14 +02:00
if (c == '#') {
c = skip_to_chars(p, "\n\a");
2022-07-31 09:49:14 +02:00
} else {
break;
}
} while (c != EOF);
unget_char(p, c);
c = get_char_q(p);
n = 0;
while (c != EOF && n < max_len) {
if (!p->quote && !isalnum(c)
&& !strchr("._/-()@#|{}[]\\", c) /*strchr(": \t\n\a",c) */ ) {
unget_char(p, c);
2022-07-31 09:49:14 +02:00
break;
}
key[n] = c;
2022-07-31 09:49:14 +02:00
c = get_char_q(p);
2022-07-31 09:49:14 +02:00
n++;
}
key[n] = 0;
2022-07-31 09:49:14 +02:00
return n;
}
static int skip_to_colon(FILE * f, struct parser *p)
2022-07-31 09:49:14 +02:00
{
int c;
c = skip_chars(p, " \t");
if (c != ':') {
if (c == '\n') {
unget_char(p, c);
sprintf(p->error,
"Error at line %d, pos %d: Unexpected EOL, collon expected.",
p->line, p->pos);
2022-07-31 09:49:14 +02:00
return 0;
}
sprintf(p->error, "Error at line %d, pos %d: Collon expected.",
p->line, p->pos);
2022-07-31 09:49:14 +02:00
return 0;
}
return 1;
}
static int read_val(struct parser *p, char *val, int max_len)
{
int c, n, quote;
if (!skip_to_colon(p->f, p))
2022-07-31 09:49:14 +02:00
return -1;
c = skip_chars(p, " \t");
if (c == '"') {
quote = 1;
c = get_char(p);
} else {
quote = 0;
2022-07-31 09:49:14 +02:00
}
n = 0;
while (c != EOF && n < max_len) {
if (quote && c == '"') {
2022-07-31 09:49:14 +02:00
break;
}
if (c == '\n') {
2022-07-31 09:49:14 +02:00
break;
}
if (quote) {
if (c == '\\') {
2022-07-31 09:49:14 +02:00
c = get_char(p);
switch (c) {
2022-07-31 09:49:14 +02:00
case 'n':
c = '\n';
2022-07-31 09:49:14 +02:00
break;
case '\\':
break;
case '"':
break;
default:
unget_char(p, c);
c = '\\';
2022-07-31 09:49:14 +02:00
}
}
}
val[n++] = c;
c = get_char(p);
2022-07-31 09:49:14 +02:00
}
if (!quote && n > 0) {
while (n > 0) {
if (isspace(val[n - 1]))
2022-07-31 09:49:14 +02:00
n--;
else
break;
}
}
val[n] = 0;
2022-07-31 09:49:14 +02:00
return n;
}
int cw_cfg_read_line(FILE * f, struct parser *p, char *key, char *val)
2022-07-31 09:49:14 +02:00
{
int n;
n = read_key(p, key, CW_CFG_MAX_KEY_LEN);
if (n == 0)
2022-07-31 09:49:14 +02:00
return 1;
if (n == -1) {
2022-07-31 09:49:14 +02:00
return -1;
}
n = read_val(p, val, CW_CFG_MAX_KEY_LEN);
if (n == -1) {
2022-07-31 09:49:14 +02:00
return -1;
}
return 0;
}
int cw_cfg_read_from_file(FILE * f, cw_Cfg_t * cfg)
{
char key[CW_CFG_MAX_KEY_LEN];
char val[2048];
struct parser p;
p.line = 1;
p.pos = 0;
p.prevpos = 0;
p.quote = 0;
p.f = f;
2022-07-31 09:49:14 +02:00
int rc;
int errs = 0;
2022-07-31 09:49:14 +02:00
do {
rc = cw_cfg_read_line(f, &p, key, val);
if (rc == -1) {
fprintf(stderr, "Error: %s\n", p.error);
2022-07-31 21:42:48 +02:00
errs++;
2022-07-31 09:49:14 +02:00
}
if (rc != 0) {
2022-07-31 09:49:14 +02:00
continue;
}
cw_cfg_set(cfg, key, val);
} while (rc == 0);
2022-07-31 09:49:14 +02:00
2022-07-31 21:42:48 +02:00
return errs;
2022-07-31 09:49:14 +02:00
}
int cw_cfg_load(const char *filename, cw_Cfg_t * cfg)
2022-07-31 09:49:14 +02:00
{
2022-07-31 21:42:48 +02:00
int errs;
FILE *f = fopen(filename, "rb");
2022-07-31 09:49:14 +02:00
if (!f)
return errno;
errs = cw_cfg_read_from_file(f, cfg);
2022-07-31 21:42:48 +02:00
fclose(f);
2022-07-31 23:58:43 +02:00
if (errs)
errno = EINVAL;
return errno;
2022-07-31 09:49:14 +02:00
}
static int cw_cfg_get_next_idx(cw_Cfg_t * cfg, const char *key, int n)
{
char ikey[CW_CFG_MAX_KEY_LEN];
struct cw_Cfg_entry search, *result;
const char *d;
int i;
sprintf(ikey, "%s.%d", key, n);
search.key = ikey;
result = mavl_get_first(cfg, &search);
printf("KEY: %s\n", search.key);
printf("NNNNN: %s\n", result->key);
if (result == NULL)
return -1;
d = NULL;
for (i = strlen(ikey); i >= 0; i--) {
if (ikey[i] == '.') {
d = result->key + i;
break;
}
}
if (d == NULL) {
return -1;
}
if (result->key[i] != '.') {
return -1;
}
if (strncmp(result->key, ikey, i) != 0)
return -1;
printf("TRANSFER %s\n", result->key + i + 1);
return atoi(result->key + i + 1);
}
static void pcb(char *dst, struct mavlnode *node)
{
struct cw_Cfg_entry *e = mavlnode_dataptr(node);
sprintf(dst, "%s", e->key);
}
void cw_cfg_iter_init(cw_Cfg_t * cfg, struct cw_Cfg_iter *cfi, const char *base)
{
struct cw_Cfg_entry search;
search.key = base;
mavliter_init(&(cfi->it), cfg);
mavliter_seek(&(cfi->it), &search, 0);
cfi->base = base;
}
const char *cw_cfg_iter_next(struct cw_Cfg_iter *cfi, const char *key)
{
struct cw_Cfg_entry *e;
int bl, kl;
const char *d;
e = mavliter_get(&(cfi->it));
if (e == NULL)
return NULL;
bl = strlen(cfi->base);
kl = strlen(e->key);
if (bl > kl)
return NULL;
if (bl == kl) {
if (strcmp(cfi->base, e->key) != 0)
return NULL;
else {
mavliter_next(&(cfi->it));
return e->val;
}
}
d = strchr(e->key, '.');
if (d == NULL)
return NULL;
if (d - e->key != bl)
return NULL;
if (strncmp(cfi->base, e->key, bl) != 0)
return NULL;
mavliter_next(&(cfi->it));
return e->val;
}
void cw_cfg_iterate(cw_Cfg_t * cfg, const char *key)
{
printf("Iterate\n");
struct cw_Cfg_entry *e;
struct cw_Cfg_entry search;
search.key = key;
struct mavliter it;
struct mavlnode *first;
mavl_print(cfg,pcb,180);
printf("SEEK TO %s\n", search.key);
struct cw_Cfg_iter cfi;
cw_cfg_iter_init(cfg, &cfi, key);
const char *kee;
while ((kee = cw_cfg_iter_next(&cfi, NULL)) != NULL) {
printf("KEY===%s\n", kee);
}
return;
mavliter_init(&it, cfg);
mavliter_seek(&it, &search, 0);
struct cw_Cfg_entry *en;
return;
int i = 0;
i = cw_cfg_get_next_idx(cfg, "actube/listen", i);
printf("This i %d\n", i);
while ((i = cw_cfg_get_next_idx(cfg, "actube/listen", i)) != -1) {
printf("Here i %d\n", i);
printf("we have key: %s.%d\n", "actube/listen", i);
printf("Next=%d\n", i);
i++;
};
e = mavl_get_first(cfg, &search);
if (!e) {
printf("NULL\n");
return;
}
printf("%s : %s\n", e->key, e->val);
}
int cw_cfg_get_bool(cw_Cfg_t * cfg, const char * key, const char *def)
{
struct cw_Val v;
const char *s = cw_cfg_get(cfg,key,def);
CW_TYPE_BOOL->from_str(&v,s);
return v.val.boolean;
}
uint16_t cw_cfg_get_word(cw_Cfg_t * cfg, char *key, const char * def)
{
struct cw_Val v;
const char *s = cw_cfg_get(cfg,key,def);
CW_TYPE_WORD->from_str(&v,s);
return v.val.word;
}