]> mj.ucw.cz Git - libucw.git/blob - lib/conf2-test.c
conf2: dumper of the configuration polished; support printing names of user types
[libucw.git] / lib / conf2-test.c
1 /*
2  *      Insane tester of reading configuration files
3  *
4  *      (c) 2006 Robert Spalek <robert@ucw.cz>
5  */
6
7 #include "lib/lib.h"
8 #include "lib/conf2.h"
9 #include "lib/clists.h"
10 #include "lib/fastbuf.h"
11
12 #include <stdlib.h>
13 #include <stdio.h>
14 #include <time.h>
15 #include <getopt.h>
16
17 static int verbose;
18
19 struct sub_sect_1 {
20   struct cnode n;
21   byte *name;
22   byte *level;
23   int confidence[2];
24   double *list;
25 };
26
27 static struct sub_sect_1 sec1 = { {}, "Charlie", "WBAFC", { 0, -1}, DARY_ALLOC(double, 3, 1e4, -1e-4, 8) };
28
29 static byte *
30 init_sec_1(struct sub_sect_1 *s)
31 {
32   if (s == &sec1) {                     // this is a static variable; skip clearing
33     DARY_LEN(sec1.list) = 3;            // XXX: fix for the bug in DARY_ALLOC()
34     return NULL;
35   }
36   s->name = "unknown";
37   s->level = "default";
38   s->confidence[0] = 5;
39   s->confidence[1] = 6;
40   // leave s->list==NULL
41   return NULL;
42 }
43
44 static byte *
45 commit_sec_1(struct sub_sect_1 *s)
46 {
47   if (s->confidence[0] < 0 || s->confidence[0] > 10)
48     return "Well, this can't be";
49   return NULL;
50 }
51
52 static struct cf_section cf_sec_1 = {
53   CF_TYPE(struct sub_sect_1),
54   CF_INIT(init_sec_1),
55   CF_COMMIT(commit_sec_1),
56 #define F(x)    PTR_TO(struct sub_sect_1, x)
57   CF_ITEMS {
58     CF_STRING("name", F(name)),
59     CF_STRING("level", F(level)),
60     CF_INT_ARY("confidence", F(confidence[0]), 2),              // XXX: the [0] is needed for the sake of type checking
61     CF_DOUBLE_DYN("list", F(list), 100),
62     CF_END
63   }
64 #undef F
65 };
66
67 static uns nr1 = 15;
68 static int *nrs1 = DARY_ALLOC(int, 5, 5, 4, 3, 2, 1);
69 static int nrs2[5];
70 static byte *str1 = "no worries";
71 static byte **str2 = DARY_ALLOC(byte *, 2, "Alice", "Bob");
72 static u64 u1 = 0xCafeBeefDeadC00ll;
73 static double d1 = -1.1;
74 static struct clist secs;
75 static time_t t1, t2;
76 static u32 ip;
77 static int *look = DARY_ALLOC(int, 2, 2, 1);
78 static u16 numbers[10] = { 2, 100, 1, 5 };
79
80 static byte *
81 parse_u16(byte *string, u16 *ptr)
82 {
83   uns a;
84   byte *msg = cf_parse_int(string, &a);
85   if (msg)
86     return msg;
87   if (a >= (1<<16))
88     return "Come on, man, this doesn't fit to 16 bits";
89   *ptr = a;
90   return NULL;
91 }
92
93 static void
94 dump_u16(struct fastbuf *fb, u16 *ptr)
95 {
96   bprintf(fb, "%d ", *ptr);
97 }
98
99 static struct cf_user_type u16_type = {
100   .size = sizeof(u16),
101   .name = "u16",
102   .parser = (cf_parser1*) parse_u16,
103   .dumper = (cf_dumper1*) dump_u16
104 };
105
106 static byte *
107 init_top(void *ptr UNUSED)
108 {
109   for (uns i=0; i<5; i++)
110   {
111     struct sub_sect_1 *s = xmalloc(sizeof(struct sub_sect_1));  // XXX: cannot by cf_malloc(), because it's deleted when cf_reload()'ed
112     cf_init_section("slaves", &cf_sec_1, s, 1);
113     s->confidence[1] = i;
114     clist_add_tail(&secs, &s->n);
115   }
116   return NULL;
117 }
118
119 static byte *
120 commit_top(void *ptr UNUSED)
121 {
122   if (nr1 != 15)
123     return "Don't touch my variable!";
124   return NULL;
125 }
126
127 static byte *
128 time_parser(uns number, byte **pars, time_t *ptr)
129 {
130   *ptr = number ? atoi(pars[0]) : time(NULL);
131   return NULL;
132 }
133
134 static byte *alphabet[] = { "alpha", "beta", "gamma", "delta", NULL };
135 static struct cf_section cf_top = {
136   CF_INIT(init_top),
137   CF_COMMIT(commit_top),
138   CF_ITEMS {
139     CF_UNS("nr1", &nr1),
140     CF_INT_DYN("nrs1", &nrs1, 1000),
141     CF_INT_ARY("nrs2", nrs2, 5),
142     CF_STRING("str1", &str1),
143     CF_STRING_DYN("str2", &str2, 20),
144     CF_U64("u1", &u1),
145     CF_DOUBLE("d1", &d1),
146     CF_PARSER("FirstTime", &t1, time_parser, -1),
147     CF_PARSER("SecondTime", &t2, time_parser, 1),
148     CF_SECTION("master", &sec1, &cf_sec_1),
149     CF_LIST("slaves", &secs, &cf_sec_1),
150     CF_IP("ip", &ip),
151     CF_LOOKUP_DYN("look", &look, alphabet, 1000),
152     CF_USER_ARY("numbers", numbers, &u16_type, 10),
153     CF_END
154   }
155 };
156
157 static byte short_opts[] = CF_SHORT_OPTS "v";
158 static struct option long_opts[] = {
159         CF_LONG_OPTS
160         {"verbose",     0, 0, 'v'},
161         {NULL,          0, 0, 0}
162 };
163
164 static char *help = "\
165 Usage: conf2-test <options>\n\
166 \n\
167 Options:\n"
168 CF_USAGE
169 "-v\t\t\tBe verbose\n\
170 ";
171
172 static void NONRET
173 usage(byte *msg, ...)
174 {
175   va_list va;
176   va_start(va, msg);
177   if (msg)
178     vfprintf(stderr, msg, va);
179   fputs(help, stderr);
180   exit(1);
181 }
182
183 int
184 main(int argc, char *argv[])
185 {
186   log_init(argv[0]);
187   cf_declare_section("top", &cf_top, 0);
188   cf_def_file = "lib/conf2.t";
189
190   int opt;
191   while ((opt = cf_get_opt(argc, argv, short_opts, long_opts, NULL)) >= 0)
192     switch (opt) {
193       case 'v': verbose++; break;
194       default: usage("unknown option %c\n", opt);
195     }
196   if (optind < argc)
197     usage("too many parameters (%d more)\n", argc-optind);
198
199   /*
200   cf_load("non-existent file");
201   //cf_reload("non-existent file");
202   cf_load("non-existent file");
203   cf_set("top.d1 -1.1; top.master b");
204   */
205
206   struct fastbuf *out = bfdopen(1, 1<<14);
207   cf_dump_sections(out);
208   bclose(out);
209
210   return 0;
211 }