/* X-Chat * Copyright (C) 1998 Peter Zelezny. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include #ifndef WIN32 #include #endif #include "hexchat.h" #include "cfgfiles.h" #include "fe.h" #include "server.h" #include "text.h" #include "util.h" /* token_foreach */ #include "hexchatc.h" #include "servlist.h" struct defaultserver { char *network; char *host; char *channel; char *charset; int loginmode; /* default authentication type */ char *connectcmd; /* default connect command - should only be used for rare login types, paired with LOGIN_CUSTOM */ gboolean ssl; }; static const struct defaultserver def[] = { {"2600net", 0}, /* Invalid hostname in cert */ {0, "irc.2600.net"}, {"2ch", 0, 0, "iso-2022-jp", 0, 0}, {0, "irc.2ch.sc"}, {0, "irc.nurs.or.jp"}, {0, "irc.juggler.jp"}, {"AccessIRC", 0}, /* Self signed */ {0, "irc.accessirc.net"}, {"AfterNET", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.afternet.org"}, {"Aitvaras", 0}, #ifdef USE_OPENSSL {0, "irc.data.lt/+6668"}, {0, "irc.omnitel.net/+6668"}, {0, "irc.ktu.lt/+6668"}, {0, "irc.kis.lt/+6668"}, {0, "irc.vub.lt/+6668"}, #endif {0, "irc.data.lt"}, {0, "irc.omnitel.net"}, {0, "irc.ktu.lt"}, {0, "irc.kis.lt"}, {0, "irc.vub.lt"}, {"Anthrochat", 0, 0, 0, 0, 0, TRUE}, {0, "irc.anthrochat.net"}, {"ARCNet", 0}, {0, "arcnet-irc.org"}, {"AustNet", 0}, {0, "irc.austnet.org"}, {"AzzurraNet", 0}, {0, "irc.azzurra.org"}, {"BetaChat", 0, 0, 0, LOGIN_SASL}, {0, "irc.betachat.net"}, {"Canternet", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.canternet.org"}, {"Chat4all", 0, 0, 0, 0, 0, TRUE}, {0, "irc.chat4all.org"}, {"ChatJunkies", 0}, {0, "irc.chatjunkies.org"}, {"ChatNet", 0}, {0, "irc.chatnet.org"}, {"ChatSpike", 0, 0, 0, LOGIN_SASL}, {0, "irc.chatspike.net"}, {"ChattingAway", 0}, {0, "irc.chattingaway.com"}, {"Criten", 0}, /* Self signed */ {0, "irc.criten.net"}, {"DALnet", 0}, /* Self signed */ {0, "us.dal.net"}, {"DarkMyst", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.darkmyst.org"}, {"Dark-Tou-Net", 0}, {0, "irc.d-t-net.de"}, {"DeltaAnime", 0}, {0, "irc.deltaanime.net"}, {"EFnet", 0}, {0, "irc.choopa.net"}, {0, "irc.paraphysics.net"}, {0, "efnet.port80.se"}, {0, "irc.underworld.no"}, {0, "irc.inet.tele.dk"}, {"ElectroCode", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.electrocode.net"}, {"EnterTheGame", 0}, {0, "irc.enterthegame.com"}, {"EntropyNet", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.entropynet.net"}, {"EsperNet", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.esper.net"}, {"EUIrc", 0}, {0, "irc.euirc.net"}, {"EuropNet", 0}, /* Self signed */ {0, "irc.europnet.org"}, {"FDFNet", 0}, /* Self signed */ {0, "irc.fdfnet.net"}, {"freenode", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "chat.freenode.net"}, /* irc. points to chat. but many users and urls still reference it */ {0, "irc.freenode.net"}, {"Furnet", 0, 0, 0, 0, 0, TRUE}, {0, "irc.furnet.org"}, {"GalaxyNet", 0}, {0, "irc.galaxynet.org"}, {"GameSurge", 0}, {0, "irc.gamesurge.net"}, {"GeeksIRC", 0, 0, 0, LOGIN_SASL}, /* Self signed */ {0, "irc.geeksirc.net"}, {"GeekShed", 0, 0, 0, 0, 0, TRUE}, {0, "irc.geekshed.net"}, {"German-Elite", 0, 0, "CP1252"}, {0, "irc.german-elite.net"}, {"GIMPNet", 0}, /* Invalid hostname in cert */ {0, "irc.gimp.org"}, {0, "irc.gnome.org"}, {"Hashmark", 0}, {0, "irc.hashmark.net"}, {"IdleMonkeys", 0}, {0, "irc.idlemonkeys.net"}, {"IndirectIRC", 0, 0, 0, LOGIN_SASL}, /* Self signed */ {0, "irc.indirectirc.com"}, {"Interlinked", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.interlinked.me"}, {"IRC4Fun", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.irc4fun.net"}, {"IRCHighWay", 0, 0, 0, 0, 0, TRUE}, {0, "irc.irchighway.net"}, {"IRCNet", 0}, {0, "open.ircnet.net"}, {"Irctoo.net", 0}, {0, "irc.irctoo.net"}, {"iZ-smart.net", 0, 0, "CP1252"}, {0, "irc.iz-smart.net"}, {"KBFail", 0}, /* SSL is self-signed */ {0, "irc.kbfail.net"}, {"Krstarica", 0}, {0, "irc.krstarica.com"}, {"LibraIRC", 0}, /* Self signed */ {0, "irc.librairc.net"}, #ifdef USE_OPENSSL {"LinkNet", 0}, {0, "irc.link-net.org/+7000"}, #endif {"MindForge", 0, 0, 0, LOGIN_SASL}, {0, "irc.mindforge.org"}, {"MIXXnet", 0}, {0, "irc.mixxnet.net"}, {"Moznet", 0, 0, 0, 0, 0, TRUE}, {0, "irc.mozilla.org"}, {"ObsidianIRC", 0}, /* Self signed */ {0, "irc.obsidianirc.net"}, {"Oceanius", 0, 0, 0, LOGIN_SASL}, /* Self signed */ {0, "irc.oceanius.com"}, {"OFTC", 0}, /* Uncommon CA? */ {0, "irc.oftc.net"}, {"OtherNet", 0}, {0, "irc.othernet.org"}, {"OzNet", 0}, {0, "irc.oz.org"}, {"PIRC.PL", 0, 0, 0, 0, 0, TRUE}, {0, "irc.pirc.pl"}, {"PonyChat", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.ponychat.net"}, {"PTNet.org", 0, 0, "ISO-8859-1"}, {0, "uevora.ptnet.org"}, {0, "vianetworks.ptnet.org"}, {"QuakeNet", 0, 0, 0, LOGIN_CHALLENGEAUTH}, {0, "irc.quakenet.org"}, {"Rizon", 0, 0, 0, 0, 0, TRUE}, {0, "irc.rizon.net"}, {"RusNet", 0, 0, "KOI8-R (Cyrillic)"}, /* Self signed */ {0, "irc.tomsk.net"}, {0, "irc.run.net"}, {0, "irc.ru"}, {0, "irc.lucky.net"}, {"SceneNet", 0}, {0, "irc.scene.org"}, {"SeilEn.de", 0, 0, "CP1252"}, {0, "irc.seilen.de"}, {"Serenity-IRC", 0}, {0, "irc.serenity-irc.net"}, {"SlashNET", 0}, /* Self signed */ {0, "irc.slashnet.org"}, {"Snoonet", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.snoonet.org"}, {"Sohbet.Net", 0, 0, "CP1254"}, {0, "irc.sohbet.net"}, {"SolidIRC", 0}, /* Self signed */ {0, "irc.solidirc.com"}, {"SorceryNet", 0, 0, 0, LOGIN_SASL}, /* Self signed */ {0, "irc.sorcery.net"}, {"SpotChat", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.spotchat.org"}, {"StarChat", 0}, {0, "irc.starchat.net"}, {"Station51", 0}, /* Self signed */ {0, "irc.station51.net"}, {"StormBit", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.stormbit.net"}, {"SwiftIRC", 0}, /* Expired cert */ {0, "irc.swiftirc.net"}, {"synIRC", 0}, /* Self signed */ {0, "irc.synirc.net"}, {"Techtronix", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.techtronix.net"}, {"TURLINet", 0, 0, 0, 0, 0, TRUE}, /* Other servers use CP1251 and invalid certs */ {0, "irc.servx.ru"}, {"UnderNet", 0, 0, 0, LOGIN_CUSTOM, "MSG x@channels.undernet.org login %u %p"}, {0, "us.undernet.org"}, {"UniBG", 0, 0, "CP1251", LOGIN_CUSTOM, "MSG NS IDENTIFY %p"}, {0, "irc.lirex.com"}, {0, "irc.naturella.com"}, {0, "irc.techno-link.com"}, {"Worldnet", 0}, {0, "irc.worldnet.net"}, {"Xertion", 0, 0, 0, LOGIN_SASL, 0, TRUE}, {0, "irc.xertion.org"}, {0,0} }; GSList *network_list = 0; #if !GLIB_CHECK_VERSION(2,34,0) #define g_slist_copy_deep servlist_slist_copy_deep /* FIXME copy-paste from gslist.c, should be dumped sometime */ static GSList* servlist_slist_copy_deep (GSList *list, GCopyFunc func, gpointer user_data) { GSList *new_list = NULL; if (list) { GSList *last; new_list = g_slice_new (GSList); if (func) new_list->data = func (list->data, user_data); else new_list->data = list->data; last = new_list; list = list->next; while (list) { last->next = g_slice_new (GSList); last = last->next; if (func) last->data = func (list->data, user_data); else last->data = list->data; list = list->next; } last->next = NULL; } return new_list; } #endif favchannel * servlist_favchan_copy (favchannel *fav) { favchannel *newfav; newfav = g_new (favchannel, 1); newfav->name = g_strdup (fav->name); newfav->key = g_strdup (fav->key); /* g_strdup() can handle NULLs so no need to check it */ return newfav; } void servlist_connect (session *sess, ircnet *net, gboolean join) { ircserver *ircserv; GSList *list; char *port; server *serv; if (!sess) sess = new_ircwindow (NULL, NULL, SESS_SERVER, TRUE); serv = sess->server; /* connect to the currently selected Server-row */ list = g_slist_nth (net->servlist, net->selected); if (!list) list = net->servlist; if (!list) return; ircserv = list->data; /* in case a protocol switch is added to the servlist gui */ server_fill_her_up (sess->server); if (join) { sess->willjoinchannel[0] = 0; if (net->favchanlist) { if (serv->favlist) { g_slist_free_full (serv->favlist, (GDestroyNotify) servlist_favchan_free); } serv->favlist = g_slist_copy_deep (net->favchanlist, (GCopyFunc) servlist_favchan_copy, NULL); } } if (net->logintype) { serv->loginmethod = net->logintype; } else { serv->loginmethod = LOGIN_DEFAULT_REAL; } serv->password[0] = 0; if (net->pass) { safe_strcpy (serv->password, net->pass, sizeof (serv->password)); } if (net->flags & FLAG_USE_GLOBAL) { strcpy (serv->nick, prefs.hex_irc_nick1); } else { if (net->nick) strcpy (serv->nick, net->nick); } serv->dont_use_proxy = (net->flags & FLAG_USE_PROXY) ? FALSE : TRUE; #ifdef USE_OPENSSL serv->use_ssl = (net->flags & FLAG_USE_SSL) ? TRUE : FALSE; serv->accept_invalid_cert = (net->flags & FLAG_ALLOW_INVALID) ? TRUE : FALSE; #endif serv->network = net; port = strrchr (ircserv->hostname, '/'); if (port) { *port = 0; /* support "+port" to indicate SSL (like mIRC does) */ if (port[1] == '+') { #ifdef USE_OPENSSL serv->use_ssl = TRUE; #endif serv->connect (serv, ircserv->hostname, atoi (port + 2), FALSE); } else { serv->connect (serv, ircserv->hostname, atoi (port + 1), FALSE); } *port = '/'; } else serv->connect (serv, ircserv->hostname, -1, FALSE); server_set_encoding (serv, net->encoding); } int servlist_connect_by_netname (session *sess, char *network, gboolean join) { ircnet *net; GSList *list = network_list; while (list) { net = list->data; if (g_ascii_strcasecmp (net->name, network) == 0) { servlist_connect (sess, net, join); return 1; } list = list->next; } return 0; } int servlist_have_auto (void) { GSList *list = network_list; ircnet *net; while (list) { net = list->data; if (net->flags & FLAG_AUTO_CONNECT) return 1; list = list->next; } return 0; } int servlist_auto_connect (session *sess) { GSList *list = network_list; ircnet *net; int ret = 0; while (list) { net = list->data; if (net->flags & FLAG_AUTO_CONNECT) { servlist_connect (sess, net, TRUE); ret = 1; } list = list->next; } return ret; } static gint servlist_cycle_cb (server *serv) { if (serv->network) { PrintTextf (serv->server_session, _("Cycling to next server in %s...\n"), ((ircnet *)serv->network)->name); servlist_connect (serv->server_session, serv->network, TRUE); } return 0; } int servlist_cycle (server *serv) { ircnet *net; int max, del; net = serv->network; if (net) { max = g_slist_length (net->servlist); if (max > 0) { /* try the next server, if that option is on */ if (net->flags & FLAG_CYCLE) { net->selected++; if (net->selected >= max) net->selected = 0; } del = prefs.hex_net_reconnect_delay * 1000; if (del < 1000) del = 500; /* so it doesn't block the gui */ if (del) serv->recondelay_tag = fe_timeout_add (del, servlist_cycle_cb, serv); else servlist_connect (serv->server_session, net, TRUE); return TRUE; } } return FALSE; } ircserver * servlist_server_find (ircnet *net, char *name, int *pos) { GSList *list = net->servlist; ircserver *serv; int i = 0; while (list) { serv = list->data; if (strcmp (serv->hostname, name) == 0) { if (pos) { *pos = i; } return serv; } i++; list = list->next; } return NULL; } favchannel * servlist_favchan_find (ircnet *net, char *channel, int *pos) { GSList *list; favchannel *favchan; int i = 0; if (net == NULL) return NULL; list = net->favchanlist; while (list) { favchan = list->data; if (g_ascii_strcasecmp (favchan->name, channel) == 0) { if (pos) { *pos = i; } return favchan; } i++; list = list->next; } return NULL; } commandentry * servlist_command_find (ircnet *net, char *cmd, int *pos) { GSList *list = net->commandlist; commandentry *entry; int i = 0; while (list) { entry = list->data; if (strcmp (entry->command, cmd) == 0) { if (pos) { *pos = i; } return entry; } i++; list = list->next; } return NULL; } /* find a network (e.g. (ircnet *) to "FreeNode") from a hostname (e.g. "irc.eu.freenode.net") */ ircnet * servlist_net_find_from_server (char *server_name) { GSList *list = network_list; GSList *slist; ircnet *net; ircserver *serv; while (list) { net = list->data; slist = net->servlist; while (slist) { gsize hostname_len; const char *hostname, *p; serv = slist->data; hostname = serv->hostname; /* Ignore port when comparing */ if ((p = strchr (hostname, '/'))) hostname_len = p - hostname; else hostname_len = strlen (hostname); if (g_ascii_strncasecmp (hostname, server_name, hostname_len) == 0) return net; slist = slist->next; } list = list->next; } return NULL; } ircnet * servlist_net_find (char *name, int *pos, int (*cmpfunc) (const char *, const char *)) { GSList *list = network_list; ircnet *net; int i = 0; while (list) { net = list->data; if (cmpfunc (net->name, name) == 0) { if (pos) *pos = i; return net; } i++; list = list->next; } return NULL; } ircserver * servlist_server_add (ircnet *net, char *name) { ircserver *serv; serv = g_new (ircserver, 1); serv->hostname = g_strdup (name); net->servlist = g_slist_append (net->servlist, serv); return serv; } commandentry * servlist_command_add (ircnet *net, char *cmd) { commandentry *entry; entry = g_new (commandentry, 1); entry->command = g_strdup (cmd); net->commandlist = g_slist_append (net->commandlist, entry); return entry; } GSList * servlist_favchan_listadd (GSList *chanlist, char *channel, char *key) { favchannel *chan; chan = g_new (favchannel, 1); chan->name = g_strdup (channel); chan->key = g_strdup (key); chanlist = g_slist_append (chanlist, chan); return chanlist; } void servlist_favchan_add (ircnet *net, char *channel) { int pos; char *name; char *key; if (strchr (channel, ',') != NULL) { pos = (int) (strchr (channel, ',') - channel); name = g_strndup (channel, pos); key = g_strdup (channel + pos + 1); } else { name = g_strdup (channel); key = NULL; } net->favchanlist = servlist_favchan_listadd (net->favchanlist, name, key); g_free (name); g_free (key); } void servlist_server_remove (ircnet *net, ircserver *serv) { g_free (serv->hostname); g_free (serv); net->servlist = g_slist_remove (net->servlist, serv); } static void servlist_server_remove_all (ircnet *net) { ircserver *serv; while (net->servlist) { serv = net->servlist->data; servlist_server_remove (net, serv); } } void servlist_command_free (commandentry *entry) { g_free (entry->command); g_free (entry); } void servlist_command_remove (ircnet *net, commandentry *entry) { servlist_command_free (entry); net->commandlist = g_slist_remove (net->commandlist, entry); } void servlist_favchan_free (favchannel *channel) { g_free (channel->name); g_free (channel->key); g_free (channel); } void servlist_favchan_remove (ircnet *net, favchannel *channel) { servlist_favchan_free (channel); net->favchanlist = g_slist_remove (net->favchanlist, channel); } static void free_and_clear (char *str) { if (str) { char *orig = str; while (*str) *str++ = 0; g_free (orig); } } /* executed on exit: Clear any password strings */ void servlist_cleanup (void) { GSList *list; ircnet *net; for (list = network_list; list; list = list->next) { net = list->data; free_and_clear (net->pass); } } void servlist_net_remove (ircnet *net) { GSList *list; server *serv; servlist_server_remove_all (net); network_list = g_slist_remove (network_list, net); g_free (net->nick); g_free (net->nick2); g_free (net->user); g_free (net->real); free_and_clear (net->pass); if (net->favchanlist) g_slist_free_full (net->favchanlist, (GDestroyNotify) servlist_favchan_free); if (net->commandlist) g_slist_free_full (net->commandlist, (GDestroyNotify) servlist_command_free); g_free (net->encoding); g_free (net->name); g_free (net); /* for safety */ list = serv_list; while (list) { serv = list->data; if (serv->network == net) { serv->network = NULL; } list = list->next; } } ircnet * servlist_net_add (char *name, char *comment, int prepend) { ircnet *net; net = g_new0 (ircnet, 1); net->name = g_strdup (name); net->flags = FLAG_CYCLE | FLAG_USE_GLOBAL | FLAG_USE_PROXY; if (prepend) network_list = g_slist_prepend (network_list, net); else network_list = g_slist_append (network_list, net); return net; } static void servlist_load_defaults (void) { int i = 0, j = 0; ircnet *net = NULL; guint def_hash = g_str_hash ("freenode"); while (1) { if (def[i].network) { net = servlist_net_add (def[i].network, def[i].host, FALSE); if (def[i].channel) { servlist_favchan_add (net, def[i].channel); } if (def[i].charset) { net->encoding = g_strdup (def[i].charset); } else { net->encoding = g_strdup (IRC_DEFAULT_CHARSET); } if (def[i].loginmode) { net->logintype = def[i].loginmode; } if (def[i].connectcmd) { servlist_command_add (net, def[i].connectcmd); } if (def[i].ssl) { net->flags |= FLAG_USE_SSL; } if (g_str_hash (def[i].network) == def_hash) { prefs.hex_gui_slist_select = j; } j++; } else { servlist_server_add (net, def[i].host); if (!def[i+1].host && !def[i+1].network) { break; } } i++; } } static int servlist_load (void) { FILE *fp; char buf[2048]; int len; ircnet *net = NULL; /* simple migration we will keep for a short while */ char *oldfile = g_build_filename (get_xdir (), "servlist_.conf", NULL); char *newfile = g_build_filename (get_xdir (), "servlist.conf", NULL); if (g_file_test (oldfile, G_FILE_TEST_EXISTS) && !g_file_test (newfile, G_FILE_TEST_EXISTS)) { g_rename (oldfile, newfile); } g_free (oldfile); g_free (newfile); fp = hexchat_fopen_file ("servlist.conf", "r", 0); if (!fp) return FALSE; while (fgets (buf, sizeof (buf) - 2, fp)) { len = strlen (buf); buf[len] = 0; buf[len-1] = 0; /* remove the trailing \n */ if (net) { switch (buf[0]) { case 'I': net->nick = g_strdup (buf + 2); break; case 'i': net->nick2 = g_strdup (buf + 2); break; case 'U': net->user = g_strdup (buf + 2); break; case 'R': net->real = g_strdup (buf + 2); break; case 'P': net->pass = g_strdup (buf + 2); break; case 'L': net->logintype = atoi (buf + 2); break; case 'E': net->encoding = servlist_check_encoding (buf + 2) ? g_strdup (buf + 2) : g_strdup ("UTF-8"); break; case 'F': net->flags = atoi (buf + 2); break; case 'S': /* new server/hostname for this network */ servlist_server_add (net, buf + 2); break; case 'C': servlist_command_add (net, buf + 2); break; case 'J': servlist_favchan_add (net, buf + 2); break; case 'D': net->selected = atoi (buf + 2); break; /* FIXME Migration code. In 2.9.5 the order was: * * P=serverpass, A=saslpass, B=nickservpass * * So if server password was unset, we can safely use SASL * password for our new universal password, or if that's also * unset, use NickServ password. * * Should be removed at some point. */ case 'A': if (!net->pass) { net->pass = g_strdup (buf + 2); if (!net->logintype) { net->logintype = LOGIN_SASL; } } case 'B': if (!net->pass) { net->pass = g_strdup (buf + 2); if (!net->logintype) { net->logintype = LOGIN_NICKSERV; } } } } if (buf[0] == 'N') net = servlist_net_add (buf + 2, /* comment */ NULL, FALSE); } fclose (fp); return TRUE; } void servlist_init (void) { if (!network_list) if (!servlist_load ()) servlist_load_defaults (); } /* check if a charset is known by Iconv */ int servlist_check_encoding (char *charset) { GIConv gic; char *c; c = strchr (charset, ' '); if (c) c[0] = 0; gic = g_iconv_open (charset, "UTF-8"); if (c) c[0] = ' '; if (gic != (GIConv)-1) { g_iconv_close (gic); return TRUE; } return FALSE; } int servlist_save (void) { FILE *fp; char *buf; ircnet *net; ircserver *serv; commandentry *cmd; favchannel *favchan; GSList *list; GSList *netlist; GSList *cmdlist; GSList *favlist; #ifndef WIN32 int first = FALSE; buf = g_build_filename (get_xdir (), "servlist.conf", NULL); if (g_access (buf, F_OK) != 0) first = TRUE; #endif fp = hexchat_fopen_file ("servlist.conf", "w", 0); if (!fp) { #ifndef WIN32 g_free (buf); #endif return FALSE; } #ifndef WIN32 if (first) g_chmod (buf, 0600); g_free (buf); #endif fprintf (fp, "v=" PACKAGE_VERSION "\n\n"); list = network_list; while (list) { net = list->data; fprintf (fp, "N=%s\n", net->name); if (net->nick) fprintf (fp, "I=%s\n", net->nick); if (net->nick2) fprintf (fp, "i=%s\n", net->nick2); if (net->user) fprintf (fp, "U=%s\n", net->user); if (net->real) fprintf (fp, "R=%s\n", net->real); if (net->pass) fprintf (fp, "P=%s\n", net->pass); if (net->logintype) fprintf (fp, "L=%d\n", net->logintype); if (net->encoding) { fprintf (fp, "E=%s\n", net->encoding); if (!servlist_check_encoding (net->encoding)) { buf = g_strdup_printf (_("Warning: \"%s\" character set is unknown. No conversion will be applied for network %s."), net->encoding, net->name); fe_message (buf, FE_MSG_WARN); g_free (buf); } } fprintf (fp, "F=%d\nD=%d\n", net->flags, net->selected); netlist = net->servlist; while (netlist) { serv = netlist->data; fprintf (fp, "S=%s\n", serv->hostname); netlist = netlist->next; } cmdlist = net->commandlist; while (cmdlist) { cmd = cmdlist->data; fprintf (fp, "C=%s\n", cmd->command); cmdlist = cmdlist->next; } favlist = net->favchanlist; while (favlist) { favchan = favlist->data; if (favchan->key) { fprintf (fp, "J=%s,%s\n", favchan->name, favchan->key); } else { fprintf (fp, "J=%s\n", favchan->name); } favlist = favlist->next; } if (fprintf (fp, "\n") < 1) { fclose (fp); return FALSE; } list = list->next; } fclose (fp); return TRUE; } static int joinlist_find_chan (favchannel *curr_item, const char *channel) { if (!g_ascii_strcasecmp (curr_item->name, channel)) { return 0; } else { return 1; } } gboolean joinlist_is_in_list (server *serv, char *channel) { if (!serv->network || !((ircnet *)serv->network)->favchanlist) { return FALSE; } if (g_slist_find_custom (((ircnet *)serv->network)->favchanlist, channel, (GCompareFunc) joinlist_find_chan)) { return TRUE; } else { return FALSE; } }