1 /* convert-conf.c - Convert ircu2.10.11 ircd.conf to ircu2.10.12 format.
2 * Copyright 2005 Michael Poole
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License as
6 * published by the Free Software Foundation; either version 2 of the
7 * License, or (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
20 #include <ctype.h> /* tolower(), toupper(), isdigit() */
21 #include <stdio.h> /* *printf(), fgets() */
22 #include <stdlib.h> /* free(), strtol() */
23 #include <string.h> /* strlen(), memcpy(), strchr(), strspn() */
27 const char *admin_names[] = { "location", "contact", "contact", 0 },
28 *connect_names[] = { "host", "password", "name", "#port", "class", 0 },
29 *crule_names[] = { "server", "", "rule", 0 },
30 *general_names[] = { "name", "vhost", "description", "", "#numeric", 0 },
31 *motd_names[] = { "host", "file", 0 },
32 *class_names[] = { "name", "#pingfreq", "#connectfreq", "#maxlinks", "#sendq", 0 },
33 *removed_features[] = { "VIRTUAL_HOST", "OPERS_SEE_IN_SECRET_CHANNELS", "LOCOP_SEE_IN_SECRET_CHANNELS", 0 };
34 char orig_line[512], line[512], dbuf[512];
35 char *fields[MAX_FIELDS + 1];
39 /*** GENERIC SUPPORT CODE ***/
41 static int split_line(char *input, char **output)
43 size_t quoted = 0, jj;
44 char *dest = dbuf, ch;
48 while (*input != '\0' && *input != '#') switch (ch = *input++) {
54 if (nfields >= MAX_FIELDS)
56 output[nfields++] = dest;
60 switch (ch = *input++) {
61 case 'b': *dest++ = '\b'; break;
62 case 'f': *dest++ = '\f'; break;
63 case 'n': *dest++ = '\n'; break;
64 case 'r': *dest++ = '\r'; break;
65 case 't': *dest++ = '\t'; break;
66 case 'v': *dest++ = '\v'; break;
67 default: *dest++ = ch; break;
70 case '"': quoted = !quoted; break;
71 default: *dest++ = ch; break;
75 for (jj = nfields; jj < MAX_FIELDS; ++jj)
80 static void simple_line(const char *block, const char **names, const char *extra)
84 /* Print the current line and start the new block. */
85 fprintf(stdout, "# %s\n%s {\n", orig_line, block);
87 /* Iterate over fields in input line, formatting each. */
88 for (ii = 0; ii < nfields && names[ii]; ++ii) {
89 if (!fields[ii][0] || !names[ii][0])
91 else if (names[ii][0] == '#')
92 fprintf(stdout, "\t%s = %s;\n", names[ii] + 1, fields[ii]);
94 fprintf(stdout, "\t%s = \"%s\";\n", names[ii], fields[ii]);
97 /* Close the new block (including any fixed-form text). */
99 fprintf(stdout, "\t%s\n", extra);
100 fputs("};\n", stdout);
103 #define dupstring(TARGET, SOURCE) do { free(TARGET); if (SOURCE) { size_t len = strlen(SOURCE); (TARGET) = malloc(len+1); memcpy((TARGET), (SOURCE), len); } else (TARGET) = 0; } while(0)
105 /*** MANAGING LISTS OF STRINGS ***/
108 struct string_list *next;
114 static struct string_list *string_get(struct string_list **list, const char *value)
116 struct string_list *curr;
117 size_t len = strlen(value), ii;
119 while ((curr = *list)) {
120 for (ii = 0; tolower(curr->value[ii]) == tolower(value[ii]) && ii < len; ++ii) ;
121 if (curr->value[ii] == '\0' && value[ii] == '\0')
126 *list = calloc(1, sizeof(**list) + len);
127 memcpy((*list)->value, value, len);
131 /*** SERVER CONNECTION RELATED CODE ***/
140 struct connect *next;
141 struct string_list *origins;
145 static struct connect *get_connect(const char *name)
147 struct connect *conn;
150 /* Look for a pre-existing connection with the same name. */
152 for (conn = connects; conn; conn = conn->next)
154 for (ii = 0; tolower(name[ii]) == conn->name[ii] && ii < nlen; ++ii) ;
155 if (conn->name[ii] == '\0' && name[ii] == '\0')
159 /* If none was found, create a new one. */
162 conn = calloc(1, sizeof(*conn) + nlen);
163 for (ii = 0; ii < nlen; ++ii)
164 conn->name[ii] = tolower(name[ii]);
165 conn->next = connects;
169 /* Return the connection. */
173 static void do_connect(void)
175 struct connect *conn = get_connect(fields[2]);
176 dupstring(conn->host, fields[0]);
177 dupstring(conn->password, fields[1]);
178 dupstring(conn->port, fields[3]);
179 dupstring(conn->class, fields[4]);
180 string_get(&conn->origins, orig_line);
183 static void do_hub(void)
185 struct connect *conn = get_connect(fields[2]);
186 dupstring(conn->hub, fields[0]);
187 dupstring(conn->maximum, fields[3]);
188 string_get(&conn->origins, orig_line);
191 static void do_leaf(void)
193 struct connect *conn = get_connect(fields[2]);
196 string_get(&conn->origins, orig_line);
199 static void finish_connects(void)
201 struct connect *conn;
202 struct string_list *sl;
204 for (conn = connects; conn; conn = conn->next)
206 for (sl = conn->origins; sl; sl = sl->next)
207 fprintf(stdout, "# %s\n", sl->value);
209 "Connect {\n\tname =\"%s\";\n\thost = \"%s\";\n"
210 "\tpassword = \"%s\";\n\tclass = \"%s\";\n",
211 conn->name, conn->host, conn->password, conn->class);
212 if (conn->port && conn->port[0] != '\0')
213 fprintf(stdout, "\tport = %s;\n", conn->port);
216 "# Every Connect block should have a port number.\n"
217 "# To prevent autoconnects, set autoconnect = no.\n"
219 "\tautoconnect = no;\n");
220 if (conn->maximum && conn->maximum[0] != '\0')
221 fprintf(stdout, "\tmaxhops = %s;\n", conn->maximum);
222 if (conn->hub && conn->hub[0] != '\0')
223 fprintf(stdout, "\thub = \"%s\";\n", conn->hub);
224 fprintf(stdout, "};\n\n");
229 /*** FEATURE MANAGEMENT CODE ***/
232 struct string_list *values;
233 struct string_list *origins;
234 struct feature *next;
238 struct remapped_feature {
240 const char *privilege;
241 int flags; /* 2 = global, 1 = local */
242 struct feature *feature;
243 } remapped_features[] = {
244 /* Specially handled privileges: If you change the index of
245 * anything with NULL privilege, change the code in
246 * finish_operators() to match!
248 { "CRYPT_OPER_PASSWORD", NULL, 0, 0 }, /* default: true */
249 { "OPER_KILL", NULL, 2, 0 }, /* default: true */
250 { "LOCAL_KILL_ONLY", NULL, 2, 0 }, /* default: false */
251 /* remapped features that affect all opers */
252 { "OPER_NO_CHAN_LIMIT", "chan_limit", 3, 0 },
253 { "OPER_MODE_LCHAN", "mode_lchan", 3, 0 },
254 { "OPER_WALK_THROUGH_LMODES", "walk_lchan", 3, 0 },
255 { "NO_OPER_DEOP_LCHAN", "deop_lchan", 3, 0 },
256 { "SHOW_INVISIBLE_USERS", "show_invis", 3, 0 },
257 { "SHOW_ALL_INVISIBLE_USERS", "show_all_invis", 3, 0 },
258 { "UNLIMIT_OPER_QUERY", "unlimit_query", 3, 0 },
259 /* remapped features affecting only global opers */
260 { "OPER_REHASH", "rehash", 2, 0 },
261 { "OPER_RESTART", "restart", 2, 0 },
262 { "OPER_DIE", "die", 2, 0 },
263 { "OPER_GLINE", "gline", 2, 0 },
264 { "OPER_LGLINE", "local_gline", 2, 0 },
265 { "OPER_JUPE", "jupe", 2, 0 },
266 { "OPER_LJUPE", "local_jupe", 2, 0 },
267 { "OPER_OPMODE", "opmode", 2, 0 },
268 { "OPER_LOPMODE", "local_opmode", 2, 0 },
269 { "OPER_FORCE_OPMODE", "force_opmode", 2, 0 },
270 { "OPER_FORCE_LOPMODE", "force_local_opmode", 2, 0 },
271 { "OPER_BADCHAN", "badchan", 2, 0 },
272 { "OPER_LBADCHAN", "local_badchan", 2, 0 },
273 { "OPER_SET", "set", 2, 0 },
274 { "OPER_WIDE_GLINE", "wide_gline", 2, 0 },
275 /* remapped features affecting only local opers */
276 { "LOCOP_KILL", "kill", 1, 0 },
277 { "LOCOP_REHASH", "rehash", 1, 0 },
278 { "LOCOP_RESTART", "restart", 1, 0 },
279 { "LOCOP_DIE", "die", 1, 0 },
280 { "LOCOP_LGLINE", "local_gline", 1, 0 },
281 { "LOCOP_LJUPE", "local_jupe", 1, 0 },
282 { "LOCOP_LOPMODE", "local_opmode", 1, 0 },
283 { "LOCOP_FORCE_LOPMODE", "force_local_opmode", 1, 0 },
284 { "LOCOP_LBADCHAN", "local_badchan", 1, 0 },
285 { "LOCOP_WIDE_GLINE", "wide_gline", 1, 0 },
289 static void do_feature(void)
291 struct feature *feat;
294 ii = strlen(fields[0]);
295 feat = calloc(1, sizeof(*feat) + ii);
297 feat->name[ii] = toupper(fields[0][ii]);
298 feat->next = features;
300 string_get(&feat->origins, orig_line);
301 for (ii = 1; fields[ii] && fields[ii][0]; ++ii)
302 string_get(&feat->values, fields[ii]);
305 static void finish_features(void)
307 struct remapped_feature *rmf;
308 struct string_list *sl;
309 struct feature *feat;
312 fputs("Features {\n\t\"OPLEVELS\" = \"FALSE\";\n", stdout);
314 for (feat = features; feat; feat = feat->next) {
315 /* See if the feature was remapped to an oper privilege. */
316 for (rmf = remapped_features; rmf->name; rmf++)
317 if (0 == strcmp(feat->name, rmf->name))
321 fprintf(stdout, "# Above feature mapped to an oper privilege.\n");
325 /* Was it removed? */
326 for (ii = 0; removed_features[ii]; ++ii)
327 if (0 == strcmp(feat->name, removed_features[ii]))
329 if (removed_features[ii]) {
330 fprintf(stdout, "# Above feature no longer exists.\n");
334 /* Wasn't remapped, wasn't removed: print it out. */
335 fprintf(stdout, "\t\"%s\" =", feat->name);
336 for (sl = feat->values; sl; sl = sl->next)
337 fprintf(stdout, " \"%s\"", sl->value);
338 fprintf(stdout, ";\n");
340 fputs("};\n\n", stdout);
344 /*** OPERATOR BLOCKS ***/
353 struct operator *next;
356 static void do_operator(int is_local)
358 struct operator *oper;
360 oper = calloc(1, sizeof(*oper));
361 dupstring(oper->host, fields[0]);
362 dupstring(oper->password, fields[1]);
363 dupstring(oper->name, fields[2]);
364 dupstring(oper->class, fields[4]);
365 dupstring(oper->origin, orig_line);
366 oper->is_local = is_local;
367 oper->next = operators;
371 static void finish_operators(void)
373 struct remapped_feature *remap;
374 struct operator *oper;
375 struct feature *feat;
377 int global_kill = 0, mask = 0;
380 if ((feat = remapped_features[0].feature) && feat->values
381 && 0 == strcmp(feat->values->value, "FALSE"))
384 if ((feat = remapped_features[1].feature) && feat->values
385 && 0 == strcmp(feat->values->value, "FALSE"))
387 else if ((feat = remapped_features[2].feature) && feat->values
388 && 0 == strcmp(feat->values->value, "FALSE"))
391 for (oper = operators; oper; oper = oper->next) {
392 fprintf(stdout, "# %s\nOperator {\n\tname = \"%s\";\n"
393 "\thost = \"%s\";\n\tpassword = \"%s%s\";\n"
394 "\tclass = \"%s\";\n",
395 oper->origin, oper->name, oper->host, pw_salt,
396 oper->password, oper->class);
397 if (oper->is_local) {
398 fputs("\tlocal = yes;\n", stdout);
401 fputs("\tlocal = no;\n", stdout);
402 if (global_kill == 1)
403 fputs("\tkill = no;\n\tlocal_kill = no;\n", stdout);
404 else if (global_kill == 2)
405 fputs("\tkill = no;\n\tlocal_kill = yes;\n", stdout);
408 for (ii = 0; (remap = &remapped_features[ii++])->name; ) {
409 if (!remap->feature || !remap->privilege
410 || !remap->feature->values || !remap->flags & mask)
412 fprintf(stdout, "\t%s = %s;\n", remap->privilege,
413 strcmp(remap->feature->values->value, "TRUE") ? "no" : "yes");
415 fputs("};\n\n", stdout);
419 /*** OTHER CONFIG TRANSFORMS ***/
421 static void do_kill(void)
423 const char *host = fields[0], *reason = fields[1], *user = fields[2];
425 if (!memcmp(host, "$R", 3)) {
426 fprintf(stderr, "Empty realname K: line at line %u.\n", lineno);
430 /* Print the current line and start the new block. */
431 fprintf(stdout, "# %s\nKill {\n", orig_line);
433 /* Translate the user-matching portions. */
434 if (host[0] == '$' && host[1] == 'R') {
435 /* Realname kill, possibly with a username */
436 fprintf(stdout, "\trealname = \"%s\";\n", host + 2);
437 if (user[0] != '\0' && (user[0] != '*' || user[1] != '\0'))
438 fprintf(stdout, "\thost = \"%s@*\";\n", user);
440 /* Normal host or IP-based kill */
441 if (user[0] != '\0' && (user[0] != '*' || user[1] != '\0'))
442 fprintf(stdout, "\thost = \"%s@%s\";\n", user, host);
444 fprintf(stdout, "\thost = \"%s\";\n", host);
447 /* Translate the reason section. */
448 if (reason[0] == '!')
449 fprintf(stdout, "\tfile = \"%s\";\n", reason + 1);
451 fprintf(stdout, "\treason = \"%s\";\n", reason);
453 /* Close the block. */
454 fprintf(stdout, "};\n");
457 static void do_port(void)
459 const char *ipmask = fields[0], *iface = fields[1], *flags = fields[2], *port = fields[3];
461 /* Print the current line and start the new block. */
462 fprintf(stdout, "# %s\nPort {\n", orig_line);
464 /* Print the easy fields. */
465 fprintf(stdout, "\tport = %s;\n", port);
466 if (iface && iface[0] != '\0')
467 fprintf(stdout, "\tvhost = \"%s\";\n", iface);
468 if (ipmask && ipmask[0] != '\0')
469 fprintf(stdout, "\tmask = \"%s\";\n", ipmask);
471 /* Translate flag field. */
472 while (*flags) switch (*flags++) {
473 case 'C': case 'c': /* client port is default state */; break;
474 case 'S': case 's': fprintf(stdout, "\tserver = yes;\n"); break;
475 case 'H': case 'h': fprintf(stdout, "\thidden = yes;\n"); break;
478 /* Close the block. */
479 fprintf(stdout, "};\n");
482 struct string_list *quarantines;
484 static void do_quarantine(void)
486 struct string_list *q;
487 q = string_get(&quarantines, fields[0]);
488 dupstring(q->origin, orig_line);
489 dupstring(q->extra, fields[1]);
492 static void finish_quarantines(void)
494 struct string_list *sl;
498 fputs("Quarantine {\n", stdout);
499 for (sl = quarantines; sl; sl = sl->next)
500 fprintf(stdout, "# %s\n\t\"%s\" = \"%s\";\n", sl->origin, sl->value, sl->extra);
501 fputs("};\n\n", stdout);
505 static void do_uworld(void)
507 fprintf(stdout, "# %s\n", orig_line);
508 if (fields[0] && fields[0][0])
509 fprintf(stdout, "Uworld { name = \"%s\"; };\n", fields[0]);
510 if (fields[1] && fields[1][0])
511 fprintf(stdout, "Jupe { nick = \"%s\"; };\n", fields[1]);
514 static void emit_client(const char *mask, const char *passwd, const char *class, long maxlinks, int is_ip)
519 delim = strchr(mask, '@');
523 len = strspn(delim, "0123456789.*");
525 fprintf(stderr, "Invalid IP mask on line %u.\n", lineno);
528 fprintf(stdout, "Client {\n\tusername = \"%s\";\n\tip = \"%s\";\n", mask, delim);
530 fprintf(stdout, "Client {\n\tusername =\"%s\";\n\thost = \"%s\";\n", mask, delim);
533 len = strspn(mask, "0123456789.*");
536 fprintf(stdout, "Client {\n\tip = \"%s\";\n", mask);
538 if (!strchr(mask, '.') && !strchr(mask, '*'))
540 fprintf(stdout, "Client {\n\thost = \"%s\";\n", mask);
544 fprintf(stdout, "\tpassword = \"%s\";\n", passwd);
547 fprintf(stdout, "\tmaxlinks = %ld;\n", maxlinks);
549 fprintf(stdout, "\tclass = \"%s\";\n};\n", class);
552 static void do_client(void)
554 char *passwd = NULL, *delim;
557 /* Print the current line. */
558 fprintf(stdout, "# %s\n", orig_line);
560 /* See if the password is really a maxlinks count. */
561 maxlinks = strtol(fields[1], &delim, 10);
562 if (fields[1][0] == '\0')
564 else if (maxlinks < 0 || maxlinks > 99 || *delim != '\0')
567 /* Translate the IP and host mask fields into blocks. */
568 emit_client(fields[0], passwd, fields[4], maxlinks, 1);
569 emit_client(fields[2], passwd, fields[4], maxlinks, 0);
572 int main(int argc, char *argv[])
578 else if (!(ifile = fopen(argv[1], "rt"))) {
579 fprintf(stderr, "Unable to open file %s for input.\n", argv[1]);
583 for (lineno = 1; fgets(line, sizeof(line), ifile); ++lineno) {
584 /* Read line and pass comments through. */
585 size_t len = strlen(line);
586 if (line[0] == '#') {
590 /* Strip EOL character(s) and pass blank lines through. */
591 while (len > 0 && (line[len-1] == '\n' || line[len-1] == '\r'))
597 /* Skip but report invalid lines. */
598 if (line[1] != ':') {
599 fprintf(stdout, "# %s\n", line);
600 fprintf(stderr, "Invalid input line %d.\n", lineno);
603 /* Copy the original line into a reusable variable. */
604 strcpy(orig_line, line);
605 /* Split line into fields. */
606 nfields = split_line(line + 2, fields);
608 /* Process the input line. */
610 case 'A': case 'a': simple_line("Admin", admin_names, NULL); break;
611 case 'C': case 'c': do_connect(); break;
612 case 'D': simple_line("CRule", crule_names, "all = yes;"); break;
613 case 'd': simple_line("CRule", crule_names, NULL); break;
614 case 'F': case 'f': do_feature(); break;
615 case 'H': case 'h': do_hub(); break;
616 case 'I': case 'i': do_client(); break;
617 case 'K': case 'k': do_kill(); break;
618 case 'L': case 'l': do_leaf(); break;
619 case 'M': case 'm': simple_line("General", general_names, NULL); break;
620 case 'O': do_operator(0); break;
621 case 'o': do_operator(1); break;
622 case 'P': case 'p': do_port(); break;
623 case 'Q': case 'q': do_quarantine(); break;
624 case 'T': case 't': simple_line("Motd", motd_names, NULL); break;
625 case 'U': case 'u': do_uworld(); break;
626 case 'Y': case 'y': simple_line("Class", class_names, NULL); break;
628 fprintf(stderr, "Unknown line %u with leading character '%c'.\n", lineno, line[0]);
635 fputs("\n# The following lines were intentionally moved and rearranged."
636 "\n# Our apologies for any inconvenience this may cause."
639 finish_quarantines();