| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092 |
- /*
- * conf.c -- handles:
- *
- * all of the conf handling
- */
- #include "common.h"
- #include "conf.h"
- #include "shell.h"
- #include "binary.h"
- #include "debug.h"
- #include "chanprog.h"
- #include "crypt.h"
- #include "main.h"
- #include "settings.h"
- #include "src/mod/irc.mod/irc.h"
- #include "misc.h"
- #include "users.h"
- #include "misc_file.h"
- #include "socket.h"
- #include "botnet.h"
- #include "userrec.h"
- #include <errno.h>
- #include "EncryptedStream.h"
- #include <bdlib/src/String.h>
- #ifdef HAVE_PATHS_H
- # include <paths.h>
- #endif /* HAVE_PATHS_H */
- #include <sys/types.h>
- #include <sys/wait.h>
- #include <sys/stat.h>
- #include <time.h>
- #include <sys/time.h>
- #include <signal.h>
- #ifdef HAVE_LIMITS_H
- # include <limits.h>
- #endif
- #ifdef CYGWIN_HACKS
- char cfile[DIRMAX] = "";
- #endif /* CYGWIN_HACKS */
- conf_t conf; /* global conf struct */
- static void
- tellconf()
- {
- conf_bot *bot = NULL;
- int i = 0;
- sdprintf(STR("tempdir: %s\n"), replace(tempdir, conf.homedir, "~"));
- sdprintf(STR("features: %d\n"), conf.features);
- sdprintf(STR("uid: %d\n"), conf.uid);
- sdprintf(STR("homedir: %s\n"), conf.homedir);
- sdprintf(STR("username: %s\n"), conf.username);
- sdprintf(STR("datadir: %s\n"), replace(conf.datadir, conf.homedir, "~"));
- sdprintf(STR("portmin: %d\n"), conf.portmin);
- sdprintf(STR("portmax: %d\n"), conf.portmax);
- sdprintf(STR("autocron: %d\n"), conf.autocron);
- sdprintf(STR("bots:\n"));
- for (bot = conf.bots; bot && bot->nick; bot = bot->next) {
- i++;
- sdprintf(STR("%d: %s%s IP: %s HOST: %s IP6: %s HOST6: %s v6: %d HUB: %d PID: %d\n"), i,
- bot->disabled ? "/" : "",
- bot->nick,
- bot->net.ip ? bot->net.ip : "",
- bot->net.host ? bot->net.host : "", bot->net.ip6 ? bot->net.ip6 : "", bot->net.host6 ? bot->net.host6 : "",
- bot->net.v6,
- bot->hub,
- bot->pid);
- }
- if (conf.bot && ((bot = conf.bot))) {
- sdprintf(STR("me:\n"));
- sdprintf(STR("%s%s IP: %s HOST: %s IP6: %s HOST6: %s v6: %d HUB: %d PID: %d\n"),
- bot->disabled ? "/" : "",
- bot->nick,
- bot->net.ip ? bot->net.ip : "",
- bot->net.host ? bot->net.host : "", bot->net.ip6 ? bot->net.ip6 : "", bot->net.host6 ? bot->net.host6 : "",
- bot->net.v6,
- bot->hub,
- bot->pid);
- }
- }
- void spawnbot(const char *nick)
- {
- int status = 0;
- sdprintf("Spawning '%s'", nick);
- const char* argv[] = {binname, nick, 0};
- status = simple_exec(argv);
- if (status == -1 || WEXITSTATUS(status))
- sdprintf("Failed to spawn '%s': %s", nick, strerror(errno));
- }
- /* spawn and kill bots accordingly
- * bots prefixxed with '/' will be killed auto if running.
- * if (updating) then we were called with -U or -u */
- void
- spawnbots(conf_bot *bots, bool rehashed)
- {
- conf_bot *bot = NULL;
- for (bot = bots; bot && bot->nick; bot = bot->next) {
- sdprintf("checking bot: %s", bot->nick);
- if (bot->disabled) {
- /* kill it if running */
- if (bot->pid) {
- kill(bot->pid, SIGKILL);
- bot->pid = 0;
- } else
- continue;
- /* if we're updating automatically, we were called with -u and are only supposed to kill non-localhubs
- -if updating and we find our nick, skip
- -if pid exists and not updating, bot is running and we have nothing more to do, skip.
- */
- } else if ((conf.bot && !strcasecmp(bot->nick, conf.bot->nick) &&
- (updating == UPDATE_AUTO || rehashed)) || (bot->pid && !updating)) {
- sdprintf(STR(" ... skipping. Updating: %d, pid: %d"), updating, bot->pid);
- continue;
- } else {
- /* if we are updating with -u then we need to restart ALL bots */
- if (updating == UPDATE_AUTO && bot->pid) {
- kill(bot->pid, SIGHUP);
- continue;
- }
- spawnbot(bot->nick);
- }
- }
- }
- int
- conf_killbot(conf_bot *bots, const char *botnick, conf_bot *bot, int signal, bool notbotnick)
- {
- int ret = -1;
- if (bot) {
- if (bot->pid)
- ret = kill(bot->pid, signal);
- } else {
- for (bot = bots; bot && bot->nick; bot = bot->next) {
- /* kill all bots but myself if botnick==NULL, otherwise just kill botnick */
- if ((!conf.bot) ||
- (!botnick &&
- (conf.bot->nick && strcasecmp(conf.bot->nick, bot->nick))) ||
- (botnick &&
- ((notbotnick == 0 && !strcasecmp(botnick, bot->nick)) ||
- (notbotnick == 1 && strcasecmp(botnick, bot->nick))
- )
- )
- ) {
- if (bot->pid)
- ret = kill(bot->pid, signal);
- if (botnick && !notbotnick)
- break;
- }
- }
- }
- return ret;
- }
- #ifndef CYGWIN_HACKS
- static int
- my_gettime(struct timespec *ts)
- {
- int rval;
- #if defined(HAVE_GETTIMEOFDAY) && (defined(HAVE_ST_MTIM) || defined(HAVE_ST_MTIMESPEC))
- struct timeval tv;
- rval = gettimeofday(&tv, NULL);
- ts->tv_sec = tv.tv_sec;
- ts->tv_nsec = tv.tv_usec * 1000;
- #else
- rval = (int)time(&ts->tv_sec);
- ts->tv_nsec = 0;
- #endif
- return (rval);
- }
- void
- confedit()
- {
- Tempfile tmpconf = Tempfile("conf");
- const char *editor = NULL;
- mode_t um;
- int waiter;
- pid_t pid, xpid;
- struct stat st, sn;
- struct timespec ts1, ts2; /* time before and after edit */
- bool autowrote = 0;
- conf_bot *oldbots = NULL;
- um = umask(077);
- autowrote = writeconf(NULL, tmpconf.fd, CONF_COMMENT);
- fstat(tmpconf.fd, &st); /* for file modification compares */
- // tmpconf.my_close();
- umask(um);
- if (!can_stat(tmpconf.file))
- fatal(STR("Cannot stat tempfile"), 0);
- /* Okay, edit the file */
- if ((!((editor = getenv(STR("EDITOR"))) && strlen(editor)))
- && (!((editor = getenv(STR("VISUAL"))) && strlen(editor)))
- ) {
- editor = STR("vi");
- /*
- #if defined(DEBIAN)
- editor = "/usr/bin/editor";
- #elif defined(_PATH_VI)
- editor = _PATH_VI;
- #else
- editor = "/usr/ucb/vi";
- #endif
- */
- }
- signal(SIGINT, SIG_IGN);
- signal(SIGQUIT, SIG_IGN);
- signal(SIGCONT, SIG_DFL);
- my_gettime(&ts1);
- switch (pid = fork()) {
- case -1:
- fatal(STR("Cannot fork"), 0);
- case 0:
- {
- /* child */
- execlp(editor, editor, tmpconf.file, (char*)NULL);
- perror(editor);
- exit(1);
- /*NOTREACHED*/}
- default:
- /* parent */
- break;
- }
- /* parent */
- while (1) {
- xpid = waitpid(pid, &waiter, WUNTRACED);
- my_gettime(&ts2);
- if (xpid == -1) {
- fprintf(stderr, STR("waitpid() failed waiting for PID %d from \"%s\": %s\n"), pid, editor, strerror(errno));
- } else if (xpid != pid) {
- fprintf(stderr, STR("wrong PID (%d != %d) from \"%s\"\n"), xpid, pid, editor);
- goto fatal;
- } else if (WIFSTOPPED(waiter)) {
- /* raise(WSTOPSIG(waiter)); Not needed and breaks in job control shell */
- } else if (WIFEXITED(waiter) && WEXITSTATUS(waiter)) {
- fprintf(stderr, STR("\"%s\" exited with status %d\n"), editor, WEXITSTATUS(waiter));
- goto fatal;
- } else if (WIFSIGNALED(waiter)) {
- fprintf(stderr, STR("\"%s\" killed; signal %d (%score dumped)\n"), editor, WTERMSIG(waiter),
- # ifdef CYGWIN_HACKS
- 0
- # else
- WCOREDUMP(waiter)
- # endif
- /* CYGWIN_HACKS */
- ? "" : "no ");
- goto fatal;
- } else {
- break;
- }
- }
- signal(SIGINT, SIG_DFL);
- signal(SIGQUIT, SIG_DFL);
- if (fstat(tmpconf.fd, &sn))
- fatal(STR("Error reading new config file"), 0);
- if (!autowrote && st.st_size == sn.st_size &&
- mtim_getsec(st) == mtim_getsec(sn) &&
- mtim_getnsec(st) == mtim_getnsec(sn)) {
- /*
- * If mtime and size match but the user spent no measurable
- * time in the editor we can't tell if the file was changed.
- */
- #ifdef HAVE_TIMESPECSUB2
- timespecsub(&ts1, &ts2);
- #else
- timespecsub(&ts1, &ts2, &ts2);
- #endif
- if (timespecisset(&ts2)) {
- printf(STR("* Config unchanged.\n"));
- tmpconf.my_close();
- unlink(tmpconf.file);
- exit(0);
- }
- }
- tmpconf.my_close();
- oldbots = conf_bots_dup(conf.bots);
- free_conf();
- readconf((const char *) tmpconf.file, 0); /* read cleartext conf tmp into &settings */
- expand_tilde(&conf.datadir);
- unlink(tmpconf.file);
- conf_to_bin(&conf, 0, -1);
- /* Now signal all of the old bots with SIGUSR1
- * They will auto die and determine new localhub, etc..
- */
- conf_checkpids(oldbots);
- conf_killbot(oldbots, NULL, NULL, SIGUSR1);
- /* Now spawn new bots */
- // spawnbots(conf.bots);
- exit(0);
- fatal:
- unlink(tmpconf.file);
- exit(1);
- }
- #endif /* !CYGWIN_HACKS */
- void
- init_conf()
- {
- // conf.bots = (conf_bot *) my_calloc(1, sizeof(conf_bot));
- // conf.bots->nick = NULL;
- // conf.bots->next = NULL;
- conf.bots = NULL;
- conf.bot = NULL;
- conf.localhub = NULL;
- #ifdef CYGWIN_HACKS
- conf.autocron = 0;
- #else
- conf.autocron = 1;
- #endif /* !CYGWIN_HACKS */
- conf.features = 0;
- conf.portmin = 0;
- conf.portmax = 0;
- conf.uid = -1;
- conf.username = NULL;
- conf.homedir = NULL;
- conf.datadir = strdup(STR("./..."));
- expand_tilde(&conf.datadir);
- }
- void conf_checkpids(conf_bot *bots, bool all)
- {
- conf_bot *bot = NULL;
- for (bot = bots; bot && bot->nick; bot = bot->next)
- if (all || (!all && bot->pid == 0))
- bot->pid = checkpid(bot->nick, bot);
- }
- static char* datafile(const char* prefix, const char* nick) {
- static char buf[DIRMAX] = "";
- char *tmpnick = NULL, *tmp_ptr = NULL;
- tmpnick = tmp_ptr = strdup(nick);
- strtolower(tmpnick);
- simple_snprintf(buf, sizeof buf, STR("%s/.%s.%s"), conf.datadir, prefix, tmpnick);
- free(tmp_ptr);
- return buf;
- }
- /*
- * Return the PID of a bot if it is running, otherwise return 0
- */
- pid_t
- checkpid(const char *nick, conf_bot *bot)
- {
- FILE *f = NULL;
- pid_t pid = 0;
- char buf[DIRMAX] = "";
- char *bufp = datafile("pid", nick);
- if (bot && !(bot->pid_file))
- bot->pid_file = strdup(bufp);
- else if (bot && strcasecmp(bot->pid_file, bufp))
- str_redup(&bot->pid_file, bufp);
- if ((f = fopen(bufp, "r"))) {
- char *pids = NULL;
- if (!fgets(buf, sizeof(buf), f)) {
- fclose(f);
- return 0;
- }
- fclose(f);
- remove_crlf(buf);
- if (!buf[0])
- return 0;
-
- bufp = buf;
- pids = newsplit(&bufp);
- if (str_isdigit(pids)) {
- pid = atoi(pids);
- if (kill(pid, SIGCHLD)) //Problem killing, most likely it's just not running.
- pid = 0;
- }
- if (bufp[0] && pid && can_stat(bufp) && (getpid() == pid) &&
- !strncasecmp(nick, origbotnick, HANDLEN)) {
- socksfile = strdup(bufp);
- return 0;
- }
- }
- return pid;
- }
- void
- conf_addbot(const char *nick, const char *ip, const char *host, const char *ip6)
- {
- conf_bot *bot = (conf_bot *) my_calloc(1, sizeof(conf_bot));
- bot->next = NULL;
- bot->pid_file = NULL;
- if (nick[0] == '/') {
- bot->disabled = 1;
- ++nick;
- sdprintf(STR("%s is disabled."), nick);
- }
- bot->nick = strldup(nick, HANDLEN);
- bot->net.ip = NULL;
- bot->net.host = NULL;
- bot->net.ip6 = NULL;
- bot->net.host6 = NULL;
- if (host && host[0] == '+') {
- ++host;
- bot->net.host6 = strdup(host);
- } else if (host && !strchr(".*", host[0])) {
- bot->net.host = strdup(host);
- }
- if (ip && !strchr(".*", ip[0])) {
- int aftype = is_dotted_ip(ip);
- if (aftype == AF_INET)
- bot->net.ip = strdup(ip);
- #ifdef USE_IPV6
- else if (aftype == AF_INET6)
- bot->net.ip6 = strdup(ip);
- #endif /* USE_IPV6 */
- else if (aftype == 0) { /* The idiot used a hostname in place of ip */
- if (bot->net.host)
- free(bot->net.host);
- bot->net.host = strdup(ip);
- }
- }
- #ifdef USE_IPV6
- if (ip6 && !strchr(".*", ip6[0]) && is_dotted_ip(ip6) == AF_INET6)
- bot->net.ip6 = strdup(ip6);
- if (bot->net.ip6 || bot->net.host6)
- bot->net.v6 = 1;
- #endif /* USE_IPV6 */
- if (userlist)
- bot->u = get_user_by_handle(userlist, (char*)nick);
- else
- bot->u = NULL;
- // bot->pid = checkpid(nick, bot);
- if (settings.hubs && is_hub(bot->nick))
- bot->hub = 1;
- /* not a hub
- AND
- * no bots added yet (first bot) yet, not disabled.
- OR
- * bots already listed but we dont have a localhub yet, so we're it!
- */
- if (!conf.localhub && !bot->hub && !bot->disabled) {
- bot->localhub = 1; /* first bot */
- conf.localhub = strdup(bot->nick);
- conf.localhub_socket = strdup(datafile("sock", conf.localhub));
- }
- list_append((struct list_type **) &(conf.bots), (struct list_type *) bot);
- }
- void
- free_bot(conf_bot *bot)
- {
- if (bot) {
- list_delete((struct list_type **) &(conf.bots), (struct list_type *) bot);
- free(bot->nick);
- free(bot->pid_file);
- if (bot->net.ip)
- free(bot->net.ip);
- if (bot->net.host)
- free(bot->net.host);
- if (bot->net.ip6)
- free(bot->net.ip6);
- if (bot->net.host6)
- free(bot->net.host6);
- free(bot);
- }
- }
- int
- conf_delbot(char *botn, bool kill)
- {
- conf_bot *bot = NULL;
- for (bot = conf.bots; bot && bot->nick; bot = bot->next) {
- if (!strcasecmp(bot->nick, botn)) { /* found it! */
- if (kill) {
- bot->pid = checkpid(bot->nick, bot);
- conf_killbot(conf.bots, NULL, bot, SIGKILL);
- }
- free_bot(bot);
- return 0;
- }
- }
- return 1;
- }
- void
- free_conf()
- {
- free_conf_bots(conf.bots);
- free_bot(conf.bot);
- conf.bot = NULL;
- if (conf.localhub)
- free(conf.localhub);
- if (conf.username)
- free(conf.username);
- if (conf.datadir)
- free(conf.datadir);
- if (conf.homedir)
- free(conf.homedir);
- init_conf();
- }
- void
- free_conf_bots(conf_bot *list)
- {
- if (list) {
- conf_bot *bot = NULL, *bot_n = NULL;
- for (bot = list; bot; bot = bot_n) {
- bot_n = bot->next;
- free_bot(bot);
- }
- list = NULL;
- }
- }
- void prep_homedir(bool error)
- {
- if (!conf.homedir)
- str_redup(&conf.homedir, homedir());
- if (error && (!conf.homedir || !conf.homedir[0]))
- werr(ERR_NOHOMEDIR);
- }
- int
- parseconf(bool error)
- {
- if (error && conf.uid == -1 && !conf.homedir)
- werr(ERR_NOTINIT);
- if (!conf.username)
- str_redup(&conf.username, my_username());
- if (error && (!conf.username || !conf.username[0]))
- werr(ERR_NOUSERNAME);
- #ifndef CYGWIN_HACKS
- if (error && conf.uid != (signed) myuid) {
- sdprintf(STR("wrong uid, conf: %d :: %d"), conf.uid, myuid);
- werr(ERR_WRONGUID);
- } else if (!conf.uid)
- conf.uid = myuid;
- #endif /* !CYGWIN_HACKS */
- return 0;
- }
- int
- readconf(const char *fname, int bits)
- {
- int enc = (bits & CONF_ENC) ? 1 : 0;
- bd::Stream* stream;
- if (enc) {
- const char salt1[] = SALT1;
- stream = new EncryptedStream(salt1);
- } else
- stream = new bd::Stream;
- sdprintf(STR("readconf(%s, %d)"), fname, enc);
- if (stream->loadFile(fname)) {
- delete stream;
- fatal(STR("Cannot read config"), 0);
- }
- free_conf_bots(conf.bots);
- bd::String line, option;
- while (stream->tell() < stream->length()) {
- line = stream->getline().chomp().trim();
- // Skip blank lines
- if (!line)
- continue;
- sdprintf(STR("CONF LINE: %s"), line.c_str());
- // !strchr("_`|}][{*/#-+!abcdefghijklmnopqrstuvwxyzABDEFGHIJKLMNOPWRSTUVWXYZ", line[0])) {
- /* - uid */
- if (line[0] == '!') {
- ++line;
- line.trim();
- option.clear();
- if (line.length())
- option = newsplit(line);
- if (!option || !line)
- continue;
- // option.toLower();
- if (option == STR("autocron")) { /* automatically check/create crontab? */
- if (egg_isdigit(line[0]))
- conf.autocron = atoi(line.c_str());
- } else if (option == STR("username")) { /* shell username */
- str_redup(&conf.username, line.c_str());
- } else if (option == STR("homedir")) { /* homedir */
- str_redup(&conf.homedir, line.c_str());
- } else if (option == STR("datadir")) { /* datadir */
- str_redup(&conf.datadir, line.c_str());
- } else if (option == STR("portmin")) {
- if (egg_isdigit(line[0]))
- conf.portmin = atoi(line.c_str());
- } else if (option == STR("portmax")) {
- if (egg_isdigit(line[0]))
- conf.portmax = atoi(line.c_str());
- } else if (option == STR("uid")) { /* new method uid */
- if (str_isdigit(line.c_str()))
- conf.uid = atoi(line.c_str());
- } else {
- putlog(LOG_MISC, "*", STR("Unrecognized config option '%s'"), option.c_str());
- }
- /* read in portmin */
- } else if (line[0] == '>') {
- conf.portmin = atoi(newsplit(line).c_str());
- } else if (line[0] == '<') {
- conf.portmax = atoi(newsplit(line).c_str());
- /* now to parse nick/hosts */
- } else if (line[0] != '#') {
- bd::String nick, host, ip, ipsix;
- nick = newsplit(line);
- if (!nick)
- werr(ERR_BADCONF);
- ip = newsplit(line);
- host = newsplit(line);
- ipsix = newsplit(line);
- conf_addbot(nick.c_str(), ip.c_str(), host.c_str(), ipsix.c_str());
- }
- } /* while(fgets()) */
- delete stream;
- return 0;
- }
- char s1_9[3] = "",s1_5[3] = "",s1_1[3] = "";
- int
- writeconf(char *filename, int fd, int bits)
- {
- conf_bot *bot = NULL;
- int autowrote = 0;
- bd::Stream* stream;
- bd::String buf;
- if (bits & CONF_ENC) {
- const char salt1[] = SALT1;
- stream = new EncryptedStream(salt1);
- } else
- stream = new bd::Stream;
- #define comment(text) do { \
- if (bits & CONF_COMMENT) \
- *stream << buf.printf(STR("%s\n"), text); \
- } while(0)
- #ifndef CYGWIN_HACKS
- char *p = NULL;
- comment("");
- #define conf_com() do { \
- if (do_confedit == CONF_AUTO) { \
- comment("# Automatically updated with -C"); \
- autowrote = 1; \
- } else \
- comment("#The correct line follows. (Not auto due to -c)"); \
- } while(0)
- if ((bits & CONF_COMMENT) && conf.uid != (signed) myuid) {
- conf_com();
- *stream << buf.printf(STR("%s! uid %d\n"), do_confedit == CONF_AUTO ? "" : "#", myuid);
- *stream << buf.printf(STR("%s! uid %d\n"), do_confedit == CONF_STATIC ? "" : "#", conf.uid);
- } else
- *stream << buf.printf(STR("! uid %d\n"), conf.uid);
- comment("");
- if (conf.username && my_username() && strcmp(conf.username, my_username())) {
- conf_com();
- *stream << buf.printf(STR("%s! username %s\n"), do_confedit == CONF_AUTO ? "" : "#", my_username());
- *stream << buf.printf(STR("%s! username %s\n"), do_confedit == CONF_STATIC ? "" : "#", conf.username);
- } else
- *stream << buf.printf(STR("! username %s\n"), conf.username ? conf.username : my_username() ? my_username() : "");
- if (conf.homedir && homedir(0) && strcmp(conf.homedir, homedir(0))) {
- conf_com();
- *stream << buf.printf(STR("%s! homedir %s\n"), do_confedit == CONF_AUTO ? "" : "#", homedir(0));
- *stream << buf.printf(STR("%s! homedir %s\n"), do_confedit == CONF_STATIC ? "" : "#", conf.homedir);
- } else
- *stream << buf.printf(STR("! homedir %s\n"), conf.homedir ? conf.homedir : homedir(0) ? homedir(0) : "");
- comment("");
- if (conf.datadir && strcmp(conf.datadir, "./...")) {
- comment("# datadir should be set to a static directory that is writable");
- if (homedir() && strstr(conf.datadir, homedir())) {
- p = replace(conf.datadir, homedir(), "~");
- *stream << buf.printf(STR("! datadir %s\n"), p);
- } else if (conf.homedir && strstr(conf.datadir, conf.homedir)) { /* Could be an older homedir */
- p = replace(conf.datadir, conf.homedir, "~");
- *stream << buf.printf(STR("! datadir %s\n"), p);
- } else
- *stream << buf.printf(STR("! datadir %s\n"), conf.datadir);
- comment("");
- }
- if (conf.portmin || conf.portmax) {
- comment("# portmin/max are for incoming connections (DCC) [0 for any] (These only make sense for HUBS)");
- *stream << buf.printf(STR("! portmin %d\n"), conf.portmin);
- *stream << buf.printf(STR("! portmax %d\n"), conf.portmax);
- comment("");
- }
- if (conf.autocron == 0) {
- comment("# Automatically add the bot to crontab?");
- *stream << buf.printf(STR("! autocron %d\n"), conf.autocron);
- comment("");
- }
- comment("# '|' means OR, [] means the enclosed is optional");
- comment("# A '+' in front of HOST means the HOST is ipv6");
- comment("# A '/' in front of BOT will disable that bot.");
- comment("#[/]BOT IP|* [+]HOST|* [IPV6-IP]");
- comment("#bot ip vhost");
- comment("#bot2 * vhost");
- comment("#bot3 ip");
- comment("#bot4 * +ipv6.vhost.com");
- comment("#bot5 * * ip:v6:ip:goes:here::");
- comment("### Hubs should have their own binary ###");
- #endif /* CYGWIN_HACKS */
- for (bot = conf.bots; bot && bot->nick; bot = bot->next) {
- *stream << buf.printf(STR("%s%s %s %s%s %s\n"),
- bot->disabled ? "/" : "", bot->nick,
- bot->net.ip ? bot->net.ip : "*", bot->net.host6 ? "+" : "",
- bot->net.host ? bot->net.host : (bot->net.host6 ? bot->net.host6 : "*"), bot->net.ip6 ? bot->net.ip6 : "");
- }
- if (fd != -1)
- stream->writeFile(fd);
- else
- stream->writeFile(filename);
- delete stream;
- return autowrote;
- }
- void
- conf_bot_dup(conf_bot *dest, conf_bot *src)
- {
- if (dest && src) {
- dest->nick = src->nick ? strdup(src->nick) : NULL;
- dest->pid_file = src->pid_file ? strdup(src->pid_file) : NULL;
- dest->net.ip = src->net.ip ? strdup(src->net.ip) : NULL;
- dest->net.host = src->net.host ? strdup(src->net.host) : NULL;
- dest->net.ip6 = src->net.ip6 ? strdup(src->net.ip6) : NULL;
- dest->net.host6 = src->net.host6 ? strdup(src->net.host6) : NULL;
- dest->net.v6 = src->net.v6;
- dest->u = src->u ? src->u : NULL;
- dest->pid = src->pid;
- dest->hub = src->hub;
- dest->localhub = src->localhub;
- dest->disabled = src->disabled;
- dest->next = NULL;
- }
- }
- conf_bot *conf_bots_dup(conf_bot *src)
- {
- conf_bot *ret = NULL;
- if (src) {
- conf_bot *bot = NULL, *newbot = NULL;
- for (bot = src; bot && bot->nick; bot = bot->next) {
- newbot = (conf_bot *) my_calloc(1, sizeof(conf_bot));
- conf_bot_dup(newbot, bot);
- list_append((struct list_type **) &(ret), (struct list_type *) newbot);
- }
- }
- return ret;
- }
- void deluser_removed_bots(conf_bot *oldlist, conf_bot *newlist)
- {
- if (oldlist) {
- conf_bot *botold = NULL, *botnew = NULL;
- bool found = 0;
- struct userrec *u = NULL;
- for (botold = oldlist; botold && botold->nick; botold = botold->next) {
- found = 0;
- for (botnew = newlist; botnew && botnew->nick; botnew = botnew->next) {
- if (!strcasecmp(botold->nick, botnew->nick)) {
- found = 1;
- break;
- }
- }
- if (!found && strcasecmp(botold->nick, origbotnick)) { /* Never kill ME.. will handle it elsewhere */
- /* No need to kill -- they are signalled and they will die on their own now */
- //botold->pid = checkpid(botold->nick, botold);
- //conf_killbot(conf.bots, NULL, botold, SIGKILL);
- if ((u = get_user_by_handle(userlist, botold->nick))) {
- putlog(LOG_MISC, "*", STR("Removing '%s' as it has been removed from the binary config."), botold->nick);
- if (server_online)
- check_this_user(botold->nick, 1, NULL);
- if (deluser(botold->nick)) {
- /* there is likely NO conf[]
- if (conf.bot->hub)
- write_userfile(-1);
- */
- }
- }
- }
- }
- }
- }
- void
- fill_conf_bot(bool fatal)
- {
- if (!conf.bots || !conf.bots->nick)
- return;
- char *mynick = NULL;
- conf_bot *me = NULL;
- /* This first clause should actually be obsolete */
- if (!used_B && conf.bots && conf.bots->nick) {
- mynick = strdup(conf.bots->nick);
- strlcpy(origbotnick, conf.bots->nick, sizeof(origbotnick));
- } else
- mynick = strldup(origbotnick, HANDLEN);
- sdprintf(STR("mynick: %s"), mynick);
- for (me = conf.bots; me && me->nick; me = me->next)
- if (!strcasecmp(me->nick, mynick))
- break;
- if (fatal && (!me || (me->nick && strcasecmp(me->nick, mynick))))
- werr(ERR_BADBOT);
- free(mynick);
- if (me) {
- if (!me->hub && me->localhub)
- sdprintf(STR("I am localhub!"));
- /* for future, we may just want to make this a pointer to ->bots if we do an emech style currentbot-> */
- conf.bot = (conf_bot *) my_calloc(1, sizeof(conf_bot));
- conf_bot_dup(conf.bot, me);
- }
- }
- void
- bin_to_conf(bool error)
- {
- /* printf("Converting binary data to conf struct\n"); */
- conf.features = atol(settings.features);
- conf.uid = atol(settings.uid);
- if (settings.username[0])
- str_redup(&conf.username, settings.username);
- str_redup(&conf.datadir, settings.datadir);
- if (settings.homedir[0])
- str_redup(&conf.homedir, settings.homedir);
- conf.portmin = atol(settings.portmin);
- conf.portmax = atol(settings.portmax);
- conf.autocron = atoi(settings.autocron);
- prep_homedir(error);
- expand_tilde(&conf.datadir);
- /* PARSE/ADD BOTS */
- {
- char *p = NULL, *tmp = NULL, *tmpp = NULL;
-
- tmp = tmpp = strdup(settings.bots);
- while ((p = strchr(tmp, ','))) {
- char *nick = NULL, *host = NULL, *ip = NULL, *ipsix = NULL;
- *p++ = 0;
- if (!tmp[0])
- break;
- nick = newsplit(&tmp);
- if (!nick || (nick && !nick[0]))
- werr(ERR_BADCONF);
- if (tmp[0])
- ip = newsplit(&tmp);
- if (tmp[0])
- host = newsplit(&tmp);
- if (tmp[0])
- ipsix = newsplit(&tmp);
- conf_addbot(nick, ip, host, ipsix);
- tmp = p++;
- }
- free(tmpp);
- }
- char datadir[PATH_MAX] = "";
- if (!realpath(conf.datadir, datadir)) {
- ;
- }
- str_redup(&conf.datadir, datadir);
- if (!mkdir_p(conf.datadir) && error)
- werr(ERR_DATADIR);
- str_redup(&conf.datadir, replace(datadir, dirname(binname), "."));
- if (Tempfile::FindDir() == ERROR)
- werr(ERR_TMPSTAT);
- if (clear_tmpdir)
- clear_tmp(); /* clear out the tmp dir, no matter if we are localhub or not */
- conf_checkpids(conf.bots);
- tellconf();
- }
- void conf_add_userlist_bots()
- {
- conf_bot *bot = NULL;
- struct userrec *u = NULL;
- struct bot_addr *bi = NULL;
- char tmp[81] = "", uhost[UHOSTLEN] = "";
- for (bot = conf.bots; bot && bot->nick; bot = bot->next) {
- /* Don't auto-add hubs. */
- if (!bot->hub && tands > 0 && !bot->disabled) {
- u = get_user_by_handle(userlist, bot->nick);
- if (!u) {
- putlog(LOG_MISC, "*", STR("Adding bot '%s' as it has been added to the binary config."), bot->nick);
- userlist = adduser(userlist, bot->nick, "none", "-", USER_OP, 1);
- u = get_user_by_handle(userlist, bot->nick);
- simple_snprintf(tmp, sizeof(tmp), "%li %s", (long)now, conf.bot->nick);
- set_user(&USERENTRY_ADDED, u, tmp);
- bi = (struct bot_addr *) my_calloc(1, sizeof(struct bot_addr));
- bi->address = (char *) my_calloc(1, 1);
- bi->uplink = (char *) my_calloc(1, 1);
- bi->telnet_port = bi->relay_port = 3333;
- bi->hublevel = 999;
- set_user(&USERENTRY_BOTADDR, u, bi);
- }
- if (bot->net.ip) {
- simple_snprintf(uhost, sizeof(uhost), "*!%s@%s", conf.username, bot->net.ip);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- simple_snprintf(uhost, sizeof(uhost), "*!~%s@%s", conf.username, bot->net.ip);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- }
- if (bot->net.host) {
- simple_snprintf(uhost, sizeof(uhost), "*!%s@%s", conf.username, bot->net.host);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- simple_snprintf(uhost, sizeof(uhost), "*!~%s@%s", conf.username, bot->net.host);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- }
- if (bot->net.host6) {
- simple_snprintf(uhost, sizeof(uhost), "*!%s@%s", conf.username, bot->net.host6);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- simple_snprintf(uhost, sizeof(uhost), "*!~%s@%s", conf.username, bot->net.host6);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- }
- if (bot->net.ip6 && strcmp(bot->net.ip6, "::")) {
- simple_snprintf(uhost, sizeof(uhost), "*!%s@%s", conf.username, bot->net.ip6);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- simple_snprintf(uhost, sizeof(uhost), "*!~%s@%s", conf.username, bot->net.ip6);
- if (!user_has_host(NULL, u, uhost) && !host_conflicts(uhost))
- addhost_by_handle(bot->nick, uhost);
- }
- }
- }
- }
- conf_bot *conf_getlocalhub(conf_bot *bots) {
- if (!bots)
- return NULL;
- conf_bot *localhub = bots;
- if (localhub->disabled)
- while (localhub && localhub->disabled)
- localhub = localhub->next;
- if (!localhub) return NULL;
- return !localhub->disabled ? localhub : NULL;
- }
- void conf_setmypid(pid_t pid) {
- conf.bot->pid = pid;
- conf_bot *bot = conf.bots;
- if (conf.bots) {
- for (; bot && strcasecmp(bot->nick, conf.bot->nick); bot = bot->next)
- ;
- if (bot)
- bot->pid = pid;
- }
- }
|