/**
 * -----------------------------------------------------------------------------
 * Name    : ns_ajoin
 * Author  : Dj-RuFfy ( Founder & CEO EasyNets Corp )
 * Date    : 23/10/2006  
 * Version : 4.2.1
 * -----------------------------------------------------------------------------
 * Limitations: Any IRCd which supports SVSJOIN
 * Tested     : Anope-1.7.21 + UnrealIRCd 3.2.6
 * Requires   : Anope-1.7.20
 * -----------------------------------------------------------------------------
 * This module adds the AJOIN command to nickserv allowing users to maintain
 * a server side auto-join list.
 * The ajoin settings can be configured for each nick group through SET AJOIN.
 *
 * This module has been designed and tested for Anope-1.7.21 and will not
 * properly work on any versions prior to 1.7.20! Changes have been made to
 * SVSJOIN support for  both inspircd and UnrealIRCd. Without these users
 * may be able to pass through channels bans (insp) or not get into +k chans.
 *
 * This module requires the IRCd to support SVSJOIN.
 * At the time of writing this module will only properly work with following IRCDs:
 *   - InspIRCd
 *   - Plexus 3.0 [or later]
 *   - PTLink
 *   - UltimateIRCd 2.8.2 [or later] [Excluding 3.x]
 *   - Unreal 3.1.1 [or later]
 *   - Unreal 3.2 [or later]
 *   - ViagraIRCd 1.3.x [or later]
 *
 * I wrote this module because the other 2 alternatives (addon module by
 * DukePyrolator and ns_ajoin by Scott) are outdated  and require MySQL, which
 * i - like many other people probably - do not use / want to use.
 *
 * The database subroutines have been ported from swhois by Trystan,
 * I wouldn't have been able to pull this off without those...
 *
 * Note that this module should be placed in ModuleDelayedAutoload.
 * -----------------------------------------------------------------------------
 * Translations:
 *
 *     - English and Dutch Languages provided and maintained by myself.
 *     - German translation 	: SNU.
 *     - Turkish translation	: ice
 *     - Russian translation	: Kein
 * -----------------------------------------------------------------------------
 *
 * Changelog:
 *
 *    4.2.1  Fixed AJOIN not requiring password identify (Reported by ysfm).
 *           Fixed ajoining suspended channels on identify.
 *           Fixed adding forbidden/suspended channels to the ajoin list.
 *           Fixed ChanServ replying to AJOIN ADD if the channel is not registered.
 *
 *    4.2.0  Going to next minor version and considering stable.
 *           Added ability to delete and list entries by number/list.
 *           Added russian translation (Provided by Kein)
 *           Fixed a few typos.
 *           Fixed crashbug when autojoining to dropped/expired channels. (Thanks to "Aded")
 *           Increased default number of entries allowed in AutoJoin list to 12
 *
 *    4.1.15 Fixed module causing a crash when ajoin issued by an unregistered user. (Thanks to "aerolife")
 *
 *    4.1.14 Fixed ajoin failing under several conditions on a channel with +k..
 *           Added ban testing and unbanning if possible...
 *           Because UnrealIRCd is the only IRCd supporting SVSJOIN'ing to +k chans
 *               we now invite before joining on all IRCd's except unreal.
 *           We no longer unban the user if he is excepted...
 *           Added hooks for NS ID.
 *
 *    4.1.13 Limited channels on ajoin list to registered channels only.
 *               This should resolve issues with invalid channel names. (Reported by "Avenger")
 *           Added check before executing the SVSJOIN to make sure channel isn't forbidden.
 *           Fixed bug causing remaining ajoins to be aborted if user is already on a ajoin chan.
 *
 *    4.1.12 Removed remaining bit of debug code
 *           Fixed some minor language details  (Complained about by "SNU" ;-) )
 *           Added german translation (Provided by SNU)
 *
 *    4.1.11 Fixed possible segfault when loading is aborted because of unsupported ircd
 *           Fixed missing help for SET AJOIN.
 *
 *    4.1.10 Fixed segfault when saving DB after nick was dropped/expired. (Reported by "hexa")
 *               (Rewrote part of DB saving routine and the way AjoinEntries are addressed.(ugly!))
 *           Added InspIRCd as supported IRCd. (Their DOCs say it implements SVSJOIN so..)
 *           Added turkish translation (Provided by ice)
 *
 *    4.1.9  Added win32 support
 *           Fixed segfault when AJOIN is used by an unregistered user
 *
 *    4.1.8  Fixed vhost being set after ajoining (Reported by "CuttingEdge")
 *           Now also performing ajoin on UPDATE
 *
 *    4.1.7  Fixed crashes in ajoin add/del
 *           Added Anope version check
 *           Default ajoin flags moved to a const
 *           No longer storing empty entries with default configuration to the DB
 *
 *    4.1.6  Added SET AJOIN option
 *           Changed database scheme to include version info.
 *           Fixed a memleak in the AjoinEntry handlers.
 *
 *    4.1.5  Fixed segfaults on load & unload
 *           AnopeFini now goes over the ajoinTable.. should speed unloading up.
 *           Anope now also backs the ajoin database up.
 *
 *    4.1.4  Finished runtime storing and database writing and reading.
 *           Implemented ajoining on identify.
 *
 *    4.1.3  Replaced hardcoded replies with langtables.
 *
 *    4.1.2  Replaced database system
 *
 *    4.1.1  First 'working' Alpha Version
 *
 *    4.1    Module Development taken over by me (Dj-RuFfy)
 *           Beginning with clean development version.
 *
 * -----------------------------------------------------------------------------
 **/
/**
 * TODO:
 *
 *    - default entries in ajoin: a /msg nickserv ajoin default (add,del,list,clear)
 *         which is only available to opers
 *
 **/
/**
 * Configuration directives that should be copy-pasted to services.conf
# AJoinDB [OPTIONAL]
# Module: ns_ajoin
#
# Use the given filename as database to store the AJOINs.
# If not given, the default of "ns_ajoin.db" will be used.
#
#AJoinDB "ns_ajoin.db"
 *
 **/
/*------------------------------Configuration Block----------------------------*/
/**
 * Determines the maximum entries in an ajoin list.
 *
 * If this value is changed, it will not affect AjoinEntries allready in the
 * database, however now ones can't be added if the limit is exceeded.
 **/
int NSAjoinMax = 12;
/*-------------------------End of Configuration Block--------------------------*/
#include "module.h"
#define AUTHOR "Dj-RuFfy"
#define VERSION "4.2.1"
#define AJOINDBVERSION 1
/* Language defines */
#define LANG_NUM_STRINGS 					29
#define LANG_AJOIN_DESC						0
#define LANG_AJOIN_SYNTAX					1
#define LANG_AJOIN_SYNTAX_EXT				2
#define LANG_AJOIN_DISABLED					3
#define LANG_NO_LOCAL_CHAN					4
#define LANG_CHAN_SYMB_REQUIRED				5
#define LANG_CHAN_UPDATED					6
#define LANG_AJOIN_LIST_FULL				7
#define LANG_CHAN_ADDED						8
#define LANG_NO_AJOINS						9
#define LANG_AJOIN_LIST_EMPTY				10
#define LANG_NO_ENTRY						11
#define LANG_CHAN_DELETED					12
#define LANG_AJOIN_ENTRY					13
#define LANG_AJOIN_ENTRIES					14
#define LANG_AJOIN_LIST_CLEARED				15
#define LANG_UNKWN_AJOIN_OPTION				16
#define LANG_AJOINING						17
#define LANG_AJOINING_FAILED				18
#define LANG_SET_AJOIN_DESC					19
#define LANG_SET_AJOIN_SYNTAX				20
#define LANG_SET_AJOIN_SYNTAX_EXT			21
#define LANG_SET_AJOIN_ON					22
#define LANG_SET_AJOIN_SILENT				23
#define LANG_SET_AJOIN_OFF					24
#define LANG_AJOINING_SUM_SUCCESS			25
#define LANG_AJOINING_SUM_FAILED			26
#define LANG_AJOIN_DELETED_NR_1				27
#define LANG_AJOIN_DELETED_NR				28
/* Flags to set in database */
#define AJOIN_ON        (1 << 0)
#define AJOIN_SILENT    (1 << 1)
/* Database seperators */
#define SEPARATOR  ':'          /* End of a key, seperates keys from values */
#define BLOCKEND   '\n'         /* End of a block, e.g. a whole nick/channel or a subblock */
#define VALUEEND   '\000'       /* End of a value */
#define SUBSTART   '\010'       /* Beginning of a new subblock, closed by a BLOCKEND */
/* Database reading return values */
#define DB_READ_SUCCESS   0
#define DB_READ_ERROR     1
#define DB_EOF_ERROR      2
#define DB_VERSION_ERROR  3
#define DB_READ_BLOCKEND  4
#define DB_READ_SUBSTART  5
#define DB_WRITE_SUCCESS  0
#define DB_WRITE_ERROR    1
#define DB_WRITE_NOVAL    2
/* Database Key, Value max length */
#define MAXKEYLEN 128
#define MAXVALLEN 1024
/* Structs */
typedef struct db_file_ DBFile;
typedef struct ajoinchan_ AjoinChan;
typedef struct ajoinentry_ AjoinEntry;
struct db_file_ {
	FILE *fptr;             /* Pointer to the opened file */
	int db_version;         /* The db version of the datafiles (only needed for reading) */
	int core_db_version;    /* The current db version of this anope source */
	char service[256];      /* StatServ/etc. */
	char filename[256];     /* Filename of the database */
	char temp_name[262];    /* Temp filename of the database */
};
struct ajoinchan_ {
	AjoinChan *prev, *next;
	char *channel;
	char *key;
	int nr;
};
struct ajoinentry_ {
	/* Specifies storage location in ajoinTable */
	int row, col;
	AjoinEntry *prev, *next;
	int ajchannels;
	AjoinChan *channels;
	/* Flags store AJOIN settings.
	 * Bits are used as followed:
	 *   User wants to be autojoined            {bit 0}
	 *   User wants autojoin to be silent       {bit 1}
	 */
	uint16 flags;
	/* Used to check during the cleanup whether the entry is still being used.
	 * This is always 0 except when the DB saving function has completed going
	 * over all NickCore's. The ajoin entries with in_use set to 0 at that time
	 * are no longer used and cleared. In the other entries in_use is reset to 0. */
	int in_use;
};
/* Constants */
char *DefAJoinDB = "ns_ajoin.db";
char *ModDataKey = "ajoin";
uint16 DefAjoinFlags = 1;
/* Variables */
char *AJoinDB;
/* This stores the last used row in teh ajoinTable so the module can attempt to
 * spread the entries out as evenly as possible... */
int counter;
/* We still need a table cause we won't be able to reach the AjoinEntry if
 * the nick is dropped... this sucks i know, but it s just a list of references */
/* Table will be cleaned of invalid entries every time the database is saved */
AjoinEntry *ajoinTable[1024];
/* Functions */
void do_help_list(User *u);
int do_help(User *u);
int add_ajoin_option(User *u);
int do_set_ajoin_help(User *u);
int do_ajoin(User *u);
int do_identify(User *u);
int ns_set(User *u);
int null_func(User *u);
int is_banned(ChannelInfo *ci, User *u);
int new_open_db_read(DBFile *dbptr, char **key, char **value);
int new_open_db_write(DBFile *dbptr);
void new_close_db(FILE *fptr, char **key, char **value);
int new_read_db_entry(char **key, char **value, FILE * fptr);
int new_write_db_entry(const char *key, DBFile *dbptr, const char *fmt, ...);
int new_write_db_endofblock(DBFile *dbptr);
void fill_db_ptr(DBFile *dbptr, int version, int core_version, char service[256], char filename[256]);
static int ajoin_del_callback(User * u, int num, va_list args);
static int ajoin_list_callback(User * u, int num, va_list args);
static AjoinChan *addAjoinChan(AjoinEntry *ae, char *chan, char *key);
AjoinChan *findAjoinChan(AjoinEntry *ae, char *chan);
AjoinChan *findAjoinChanNr(AjoinEntry *ae, int nr);
static int deleteAjoinChan(AjoinEntry *ae, AjoinChan *ac);
static int clearAjoinChans(AjoinEntry *ae);
static int numberAjoinChans(AjoinEntry *ae);
static AjoinEntry *createAjoinEntry(NickCore *nc);
AjoinEntry *getAjoinEntry(NickCore *nc);
static int deleteAjoinEntry(AjoinEntry *ae);
void freeUnusedEntries();
int do_save(int argc, char **argv);
int db_backup(int argc, char **argv);
void load_ajoin_db(void);
void save_ajoin_db(void);
AjoinEntry *go_to_next_entry(uint16 skipped, AjoinEntry *next, AjoinEntry *ae);
int valid_ircd(void);
void load_config(void);
int reload_config(int argc, char **argv);
void add_languages(void);
/* ------------------------------------------------------------------------------- */
/**
 * Create the command, and tell anope about it.
 * @param argc Argument count
 * @param argv Argument list
 * @return MOD_CONT to allow the module, MOD_STOP to stop it
 **/
int AnopeInit(int argc, char **argv) {
	Command *c;
	EvtHook *hook;
	alog("[\002ns_ajoin\002] Loading module...");
	counter = 0;
	if (!valid_ircd()) {
		alog("[\002ns_ajoin\002] ERROR: IRCd not supported by this module");
		alog("[\002ns_ajoin\002] Unloading module...");
		return MOD_STOP;
	}
	if (!moduleMinVersion(1,7,21,1341)) {
		alog("[\002ns_ajoin\002] Your version of Anope isn't supported. Please update to a newer release.");
		return MOD_STOP;
	}
	/* Create AJOIN command.. */
	c = createCommand("AJOIN", do_ajoin, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV,c,MOD_HEAD) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot create AJOIN command...");
		return MOD_STOP;
	}
	moduleAddHelp(c,do_help);
	moduleSetNickHelp(do_help_list);
	/* Hook to the IDENTIFY and UPDATE command */
	c = createCommand("IDENTIFY", do_identify, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV,c,MOD_TAIL) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot hook to IDENTIFY command...");
		return MOD_STOP;
	}
	c = createCommand("ID", do_identify, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV,c,MOD_TAIL) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot hook to ID command...");
		return MOD_STOP;
	}
	c = createCommand("UPDATE", do_identify, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV,c,MOD_TAIL) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot hook to UPDATE command...");
		return MOD_STOP;
	}
	/* Create the SET AJOIN command, its help and add it to cmd listing.. */
	c = createCommand("SET", ns_set, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV, c, MOD_HEAD) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot create SET AJOIN command...");
		return MOD_STOP;
	}
	c = createCommand("SET AJOIN", NULL, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV, c, MOD_TAIL) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot create help for the SET AJOIN command...");
		return MOD_STOP;
	}
	moduleAddHelp(c, do_set_ajoin_help);
	c = createCommand("SET", null_func, NULL, -1, -1, -1, -1, -1);
	if (moduleAddCommand(NICKSERV, c, MOD_TAIL) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Cannot add AJOIN to SET options...");
		return MOD_STOP;
	}
	moduleAddHelp(c, add_ajoin_option);
	/* Hook to some events.. */
	hook = createEventHook(EVENT_RELOAD, reload_config);
	if (moduleAddEventHook(hook) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Can't hook to EVENT_RELOAD event");
		return MOD_STOP;
	}
	hook = createEventHook(EVENT_DB_SAVING, do_save);
	if (moduleAddEventHook(hook) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Can't hook to EVENT_DB_SAVING event");
		return MOD_STOP;
	}
	hook = createEventHook(EVENT_DB_BACKUP, db_backup);
	if (moduleAddEventHook(hook) != MOD_ERR_OK) {
		alog("[\002ns_ajoin\002] Can't hook to EVENT_DB_BACKUP event");
		return MOD_STOP;
	}
	load_config();
	add_languages();
	load_ajoin_db();
	moduleAddAuthor(AUTHOR);
	moduleAddVersion(VERSION);
	alog("[\002ns_ajoin\002] Module loaded successfully...");
	return MOD_CONT;
}
/**
 * Unload the module
 **/
void AnopeFini(void) {
	AjoinEntry *ae = NULL, *next = NULL;
	int i;
	if (AJoinDB)
		save_ajoin_db();
	/* clear the memory.... */
	for (i = 0; i < 1024; i++) {
		for (ae = ajoinTable[i]; ae; ae = next) {
			next = ae->next;
			deleteAjoinEntry(ae);
		}
	}
	if (AJoinDB)
		free(AJoinDB);
	alog("[\002ns_ajoin\002] Unloading module...");
}
/* ------------------------------------------------------------------------------- */
/**
 * Add the AJOIN command to the NickServ HELP listing.
 **/
void do_help_list(User *u) {
	moduleNoticeLang(s_NickServ, u, LANG_AJOIN_DESC);
}
/**
 * Show the extended help on the AJOIN command.
 **/
int do_help(User *u) {
	moduleNoticeLang(s_NickServ, u, LANG_AJOIN_SYNTAX_EXT);
	return MOD_CONT;
}
/**
 * Add the AJOIN option to the end of the SET options listing.
 **/
int add_ajoin_option(User *u) {
	moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_DESC);
	return MOD_CONT;
}
/**
 * Show the extended help on the SET AJOIN command.
 **/
int do_set_ajoin_help(User *u) {
	moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_SYNTAX_EXT);
	return MOD_CONT;
}
/* ------------------------------------------------------------------------------- */
int do_ajoin(User *u) {
	ChannelInfo *ci;
	char *buffer, *cmd, *chan, *key;
	buffer = moduleGetLastBuffer();
	cmd = myStrGetToken(buffer, ' ', 0);
	chan = myStrGetToken(buffer, ' ', 1);
	key = myStrGetToken(buffer, ' ', 2);
	if (!cmd)
		moduleNoticeLang(s_NickServ, u, LANG_AJOIN_SYNTAX);
	else if (stricmp(cmd, "ADD") == 0) {
		if (readonly)
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_DISABLED);
		else if (!u->na)
			notice_lang(s_NickServ, u, NICK_NOT_REGISTERED);
		else if (!nick_identified(u))
			notice_lang(s_NickServ, u, NICK_IDENTIFY_REQUIRED, s_NickServ);
		else if (!chan)
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_SYNTAX);
		else if (*chan == '&')
			moduleNoticeLang(s_NickServ, u, LANG_NO_LOCAL_CHAN);
		else if (*chan != '#')
			moduleNoticeLang(s_NickServ, u, LANG_CHAN_SYMB_REQUIRED);
		else if (!anope_valid_chan(chan))
			notice_lang(s_NickServ, u, CHAN_X_INVALID, chan);
		else if (!(ci = cs_findchan(chan)))
			notice_lang(s_NickServ, u, CHAN_X_NOT_REGISTERED, chan);
		else if (ci->flags & CI_VERBOTEN)
			notice_lang(s_NickServ, u, CHAN_X_FORBIDDEN, chan);
		else if (ci->flags & CI_SUSPENDED)
			notice_lang(s_NickServ, u, CHAN_X_FORBIDDEN, chan);
		else {
			AjoinEntry *ae = NULL;
			AjoinChan *ac = NULL;
			ae = getAjoinEntry(u->na->nc);
			ac = findAjoinChan(ae, chan);
			if (!ae)
				ae = createAjoinEntry(u->na->nc);
			if (ac) {
				/* update existing entry */
				if (!key && ac->key) {
					free(ac->key);
					ac->key = NULL;
				} else if (key) {
					free(ac->key);
					ac->key = sstrdup(key);
				}
				moduleNoticeLang(s_NickServ, u, LANG_CHAN_UPDATED);
			} else {
				if (ae->ajchannels == NSAjoinMax)
					moduleNoticeLang(s_NickServ, u, LANG_AJOIN_LIST_FULL);
				else {
					addAjoinChan(ae, chan, key);
					moduleNoticeLang(s_NickServ, u, LANG_CHAN_ADDED, chan);
				}
			}
		}
	} else if (stricmp(cmd, "DEL") == 0) {
		AjoinEntry *entry;
		if (readonly)
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_DISABLED);
		else if (!u->na)
			notice_lang(s_NickServ, u, NICK_NOT_REGISTERED);
		else if (!chan)
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_SYNTAX);
		else if (!(entry = getAjoinEntry(u->na->nc)))
			moduleNoticeLang(s_NickServ, u, LANG_NO_AJOINS);
		else {
			/* we r a valid chan, delete us if we r present... */
			if (entry->ajchannels == 0)
				moduleNoticeLang(s_NickServ, u, LANG_AJOIN_LIST_EMPTY);
			else {
				/* Check if the given target is a number/list.
				 * Only  search if it isn't. */
				if (isdigit(*chan) && strspn(chan, "1234567890,-") == strlen(chan)) {
					int deleted, count;
					deleted = process_numlist(chan, &count, ajoin_del_callback, u, entry);
					numberAjoinChans(entry);
					if (deleted == 1)
						moduleNoticeLang(s_NickServ, u, LANG_AJOIN_DELETED_NR_1, deleted);
					else
						moduleNoticeLang(s_NickServ, u, LANG_AJOIN_DELETED_NR, deleted);
				} else {
					if (*chan != '#')
						moduleNoticeLang(s_NickServ, u, LANG_CHAN_SYMB_REQUIRED);
					else {
						AjoinChan *ac = findAjoinChan(entry, chan);
						if (!ac)
							moduleNoticeLang(s_NickServ, u, LANG_NO_ENTRY, chan);
						else {
							deleteAjoinChan(entry, ac);
							numberAjoinChans(entry);
							moduleNoticeLang(s_NickServ, u, LANG_CHAN_DELETED, chan);
						}
					}
				}
			}
		}
	} else if (stricmp(cmd, "LIST") == 0) {
		int i = 0;
		AjoinEntry *ae = NULL;
		AjoinChan *ac = NULL;
		if (!u->na) {
			notice_lang(s_NickServ, u, NICK_NOT_REGISTERED);
		} else {
			ae = getAjoinEntry(u->na->nc);
			if (!ae)
				moduleNoticeLang(s_NickServ, u, LANG_NO_AJOINS);
			else {
				if (ae->ajchannels == 0)
					moduleNoticeLang(s_NickServ, u, LANG_AJOIN_LIST_EMPTY);
				else {
					/* Check if the given target is a number/list. */
					if (chan && isdigit(*chan) && strspn(chan, "1234567890,-") == strlen(chan)) {
						int count, found;
						found = process_numlist(chan, &count, ajoin_list_callback, u, ae);
						moduleNoticeLang(s_NickServ, u, LANG_AJOIN_ENTRIES, found);
					} else {
						for (ac = ae->channels; ac; ac = ac->next) {
							if (chan && !match_wild_nocase(chan, ac->channel))
								continue;
							moduleNoticeLang(s_NickServ, u, LANG_AJOIN_ENTRY, ac->nr, ac->channel,
									((ac->key) ? ac->key:""));
							i++;
						}
						moduleNoticeLang(s_NickServ, u, LANG_AJOIN_ENTRIES, i);
					}
				}
			}
		}
	} else if (stricmp(cmd, "CLEAR") == 0) {
		AjoinEntry *ae;
		if (readonly)
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_DISABLED);
		else if (!u->na)
			notice_lang(s_NickServ, u, NICK_NOT_REGISTERED);
		else if (!(ae = getAjoinEntry(u->na->nc)))
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_LIST_EMPTY);
		else {
			clearAjoinChans(ae);
			moduleNoticeLang(s_NickServ, u, LANG_AJOIN_LIST_CLEARED);
		}
	} else {
		moduleNoticeLang(s_NickServ, u, LANG_UNKWN_AJOIN_OPTION);
		moduleNoticeLang(s_NickServ, u, LANG_AJOIN_SYNTAX);
	}
	if (cmd)
		free(cmd);
	if (chan)
		free(chan);
	if (key)
		free(key);
	return MOD_CONT;
}
int do_identify(User *u) {
	int i = 0, ok, needinvite, counter_succ = 0, counter_f = 0;
	if (nick_identified(u) && u->na->nc) {
		AjoinEntry *ae = NULL;
		ae = getAjoinEntry(u->na->nc);
		if (ae && (ae->flags & AJOIN_ON) && (ae->ajchannels > 0)) {
			AjoinChan *ac = NULL;
			ac = ae->channels;
			for (i = 0; i < ae->ajchannels; i++) {
				Channel *c = NULL;
				ChannelInfo *ci;
				ok = 1, needinvite = 0;
				if (!ac)
					break;
				/* If the channel is no longer registered we won't complain..
				 * We are sure the name is valid because it was registered once.
				 * We do complain when the channel is forbidden.. no point in joining. */
				/* We do the same when it s suspended. We do NOT automatically remove channels though. */
				ci = cs_findchan(ac->channel);
				if (ci && (ci->flags & CI_VERBOTEN)) {
					notice_lang(s_NickServ, u, CHAN_X_FORBIDDEN, ac->channel);
					ok = 0;
				}
				if (ci && (ci->flags & CI_SUSPENDED)) {
					notice_lang(s_NickServ, u, CHAN_X_FORBIDDEN, ac->channel);
					ok = 0;
				}
				/* Check if channel already exists...
				 * We can't use c->ci here because we don't require the chan to still be registered */
				if (ok && ((c = findchan(ac->channel)))) {
					if (is_on_chan(c, u)) {
						ac = ac->next;
						continue;
					}
					else {
						if (c->key) {
							if (!ac->key || (stricmp(ac->key, c->key) != 0)) {
								if (ci && check_access(u, ci, CA_GETKEY)) {
									if (ac->key)
										free(ac->key);
									ac->key = sstrdup(c->key);
								} else
									needinvite = 1;
							}
							/* Any IRCd other then unreal needs invite cause they don't support
							 * SVSJOIN with a key.. */
							if (stricmp(IRCDModule,"unreal32"))
								needinvite = 1;
						}
						/* Check if user is banned and if so, remove is possible.
						 * If we cannot remove it, try to invite */
						if (ci && is_banned(ci, u)) {
							if (check_access(u, ci, CA_UNBAN))
								common_unban(ci, u->nick);
							else
								needinvite = 1;
						}
						if ((c->mode & anope_get_invite_mode()) || needinvite) {
							if (ci && check_access(u, ci, CA_INVITE))
								anope_cmd_invite(s_NickServ, ac->channel, u->nick);
							else
								ok = 0;
						}
					}
				}
				if (ok) {
					if (ae->flags & AJOIN_SILENT)
						counter_succ++;
					else
						moduleNoticeLang(s_NickServ, u, LANG_AJOINING, ac->channel);
					anope_cmd_svsjoin(s_NickServ, u->nick, ac->channel, ac->key);
				} else {
					if (ae->flags & AJOIN_SILENT)
						counter_f++;
					else
						moduleNoticeLang(s_NickServ, u, LANG_AJOINING_FAILED, ac->channel);
				}
				ac = ac->next;
			}
			if (ae->flags & AJOIN_SILENT) {
				if (counter_succ > 0)
					moduleNoticeLang(s_NickServ, u, LANG_AJOINING_SUM_SUCCESS, counter_succ);
				if (counter_f > 0)
					moduleNoticeLang(s_NickServ, u, LANG_AJOINING_SUM_FAILED, counter_f);
			}
		}
	}
	return MOD_CONT;
}
int ns_set(User *u) {
	char *args, *cmd, *option;
	/* If readonly, let the core ns_set handle it.. */
	if (readonly)
		return MOD_CONT;
	args = moduleGetLastBuffer();
	if (!args)
		return MOD_CONT;
	cmd = myStrGetToken(args, ' ', 0);
	option = myStrGetToken(args, ' ', 1);
	if (cmd && !stricmp(cmd, "AJOIN")) {
		if (!option)
			moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_SYNTAX);
		else if (!u->na)
			notice_lang(s_NickServ, u, NICK_NOT_REGISTERED);
		else if (!nick_identified(u))
			notice_lang(s_NickServ, u, NICK_IDENTIFY_REQUIRED, s_NickServ);
		else {
			AjoinEntry *ae = NULL;
			ae = getAjoinEntry(u->na->nc);
			if (!ae)
				ae = createAjoinEntry(u->na->nc);
			if (!stricmp(option, "ON")) {
				/* Turn off silent bit */
				ae->flags &= ~AJOIN_SILENT;
				/* Turn on use autojoin bit */
				ae->flags |= AJOIN_ON;
				moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_ON);
			} else if (!stricmp(option, "SILENT")) {
				/* Turn on silent bit */
				ae->flags |= AJOIN_SILENT;
				/* Turn on use autojoin bit */
				ae->flags |= AJOIN_ON;
				moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_SILENT);
			} else if (!stricmp(option, "OFF")) {
				/* Turn off silent bit */
				ae->flags &= ~AJOIN_SILENT;
				/* Turn off use autojoin bit */
				ae->flags &= ~AJOIN_ON;
				moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_OFF);
			} else
				moduleNoticeLang(s_NickServ, u, LANG_SET_AJOIN_SYNTAX);
		}
		free(cmd);
		if (option) free (option);
		return MOD_STOP;
	}
	if (cmd) free(cmd);
	if (option) free (option);
	return MOD_CONT;
}
int null_func(User *u) {
  return MOD_CONT;
}
int is_banned(ChannelInfo *ci, User *u) {
	int i, isbanned = 0;
	char *host = NULL;
	if (!ci->c)
		return 0;
	host = host_resolve(u->host);
	for (i = 0; i < ci->c->bancount; i++) {
		if (match_usermask(ci->c->bans[i], u) ||
				(host && match_userip(ci->c->bans[i], u, host))) {
			isbanned = 1;
			break;
		}
	}
	if (host)
		free(host);
	return isbanned;
}
/* ------------------------------------------------------------------------------- */
/**************************************************************************
 *         Generic DataBase Functions  (Taken from swhois by Trystan)
 **************************************************************************/
int new_open_db_read(DBFile *dbptr, char **key, char **value) {
	*key = malloc(MAXKEYLEN);
	*value = malloc(MAXVALLEN);
	if (!(dbptr->fptr = fopen(dbptr->filename, "rb"))) {
		if (debug) {
			alog("debug: Can't read %s database %s : errno(%d)", dbptr->service,
				dbptr->filename, errno);
		}
		free(*key);
		*key = NULL;
		free(*value);
		*value = NULL;
		return DB_READ_ERROR;
	}
	dbptr->db_version = fgetc(dbptr->fptr) << 24 | fgetc(dbptr->fptr) << 16
		| fgetc(dbptr->fptr) << 8 | fgetc(dbptr->fptr);
	if (ferror(dbptr->fptr)) {
		if (debug) {
			alog("debug: Error reading version number on %s", dbptr->filename);
		}
		free(*key);
		*key = NULL;
		free(*value);
		*value = NULL;
		return DB_READ_ERROR;
	} else if (feof(dbptr->fptr)) {
		if (debug) {
			alog("debug: Error reading version number on %s: End of file detected",
				dbptr->filename);
		}
		free(*key);
		*key = NULL;
		free(*value);
		*value = NULL;
		return DB_EOF_ERROR;
	} else if (dbptr->db_version < 1) {
		if (debug) {
			alog("debug: Invalid version number (%d) on %s", dbptr->db_version, dbptr->filename);
		}
		free(*key);
		*key = NULL;
		free(*value);
		*value = NULL;
		return DB_VERSION_ERROR;
	}
	return DB_READ_SUCCESS;
}
int new_open_db_write(DBFile *dbptr) {
	if (!(dbptr->fptr = fopen(dbptr->filename, "wb"))) {
		if (debug) {
			alog("debug: %s Can't open %s database for writing", dbptr->service, dbptr->filename);
		}
		return DB_WRITE_ERROR;
	}
	if (fputc(dbptr->core_db_version >> 24 & 0xFF, dbptr->fptr) < 0 ||
		fputc(dbptr->core_db_version >> 16 & 0xFF, dbptr->fptr) < 0 ||
		fputc(dbptr->core_db_version >> 8 & 0xFF, dbptr->fptr) < 0 ||
		fputc(dbptr->core_db_version & 0xFF, dbptr->fptr) < 0) {
		if (debug) {
			alog("debug: Error writing version number on %s", dbptr->filename);
		}
		return DB_WRITE_ERROR;
	}
	return DB_WRITE_SUCCESS;
}
void new_close_db(FILE *fptr, char **key, char **value) {
	if (key && *key) {
		free(*key);
		*key = NULL;
	}
	if (value && *value) {
		free(*value);
		*value = NULL;
	}
	if (fptr) {
		fclose(fptr);
	}
}
int new_read_db_entry(char **key, char **value, FILE *fptr) {
	char *string = *key;
	int character;
	int i = 0;
	**key = '\0';
	**value = '\0';
	while (1) {
		if ((character = fgetc(fptr)) == EOF) { /* a problem occurred reading the file */
			if (ferror(fptr)) {
				return DB_READ_ERROR;   /* error! */
			}
			return DB_EOF_ERROR;        /* end of file */
		} else if (character == BLOCKEND) {     /* END OF BLOCK */
			return DB_READ_BLOCKEND;
		} else if (character == VALUEEND) {     /* END OF VALUE */
			string[i] = '\0';   /* end of value */
			return DB_READ_SUCCESS;
		} else if (character == SEPARATOR) {    /* END OF KEY */
			string[i] = '\0';   /* end of key */
			string = *value;    /* beginning of value */
			i = 0;              /* start with the first character of our value */
		} else {
			if ((i == (MAXKEYLEN - 1)) && (string == *key)) {   /* max key length reached, continuing with value */
				string[i] = '\0';       /* end of key */
				string = *value;        /* beginning of value */
				i = 0;          /* start with the first character of our value */
			} else if ((i == (MAXVALLEN - 1)) && (string == *value)) {  /* max value length reached, returning */
				string[i] = '\0';
				return DB_READ_SUCCESS;
			} else {
				string[i] = character;  /* read string (key or value) */
				i++;
			}
		}
	}
}
int new_write_db_entry(const char *key, DBFile *dbptr, const char *fmt, ...) {
	char string[MAXKEYLEN + MAXVALLEN + 2], value[MAXVALLEN];   /* safety byte :P */
	va_list ap;
	unsigned int length;
	if (!dbptr) {
		return DB_WRITE_ERROR;
	}
	va_start(ap, fmt);
	vsnprintf(value, MAXVALLEN, fmt, ap);
	va_end(ap);
	if (!stricmp(value, "(null)")) {
		return DB_WRITE_NOVAL;
	}
	snprintf(string, MAXKEYLEN + MAXVALLEN + 1, "%s%c%s", key, SEPARATOR, value);
	length = strlen(string);
	string[length] = VALUEEND;
	length++;
	if (fwrite(string, 1, length, dbptr->fptr) < length) {
		if (debug) {
			alog("debug: Error writing to %s", dbptr->filename);
		}
		new_close_db(dbptr->fptr, NULL, NULL);
		if (debug) {
			alog("debug: Restoring backup.");
		}
		remove(dbptr->filename);
		rename(dbptr->temp_name, dbptr->filename);
		free(dbptr);
		dbptr = NULL;
		return DB_WRITE_ERROR;
	}
	return DB_WRITE_SUCCESS;
}
int new_write_db_endofblock(DBFile *dbptr) {
	if (!dbptr) {
		return DB_WRITE_ERROR;
	}
	if (fputc(BLOCKEND, dbptr->fptr) == EOF) {
		if (debug) {
			alog("debug: Error writing to %s", dbptr->filename);
		}
		new_close_db(dbptr->fptr, NULL, NULL);
		return DB_WRITE_ERROR;
	}
	return DB_WRITE_SUCCESS;
}
void fill_db_ptr(DBFile *dbptr, int version, int core_version,
				char service[256], char filename[256]) {
	dbptr->db_version = version;
	dbptr->core_db_version = core_version;
	if (!service)
		strcpy(dbptr->service, service);
	else
		strcpy(dbptr->service, "");
	strcpy(dbptr->filename, filename);
	snprintf(dbptr->temp_name, 261, "%s.temp", filename);
	return;
}
/* ------------------------------------------------------------------------------- */
static int ajoin_del_callback(User * u, int num, va_list args) {
	AjoinEntry *ae = va_arg(args, AjoinEntry *);
	AjoinChan *ac = findAjoinChanNr(ae, num);
	if (!ac)
		return 0;
	moduleNoticeLang(s_NickServ, u, LANG_CHAN_DELETED, ac->channel);
	deleteAjoinChan(ae, ac);
	return 1;
}
static int ajoin_list_callback(User * u, int num, va_list args) {
	AjoinEntry *ae = va_arg(args, AjoinEntry *);
	AjoinChan *ac = findAjoinChanNr(ae, num);
	if (!ac)
		return 0;
	moduleNoticeLang(s_NickServ, u, LANG_AJOIN_ENTRY, ac->nr, ac->channel,
			((ac->key) ? ac->key:""));
	return 1;
}
/* ------------------------------------------------------------------------------- */
/**
 * We add an AjoinChan to the list and update the nr of the channels.
 **/
static AjoinChan *addAjoinChan(AjoinEntry *ae, char *chan, char *key) {
	AjoinChan *ac = NULL, *current = NULL, *previous = NULL;
	if (!ae || !chan)
		return NULL;
	if ((ac = malloc(sizeof(AjoinChan))) == NULL) {
		fatal("Out Of Memory!");
	}
	for (current = ae->channels; current; current = current->next) {
		/* search for the position to insert the new channel */
		if (stricmp(chan, current->channel) > 0)
			previous = current;
		else
			break;
	}
	ac->next = current;
	ac->prev = previous;
	ac->channel = sstrdup(chan);
	if (key)
		ac->key = sstrdup(key);
	else
		ac->key = NULL;
	if (previous == NULL) {
		ae->channels = ac;
		ac->nr = 1;
	} else {
		previous->next = ac;
		ac->nr = previous->nr + 1;
	}
	if (current)
		current->prev = ac;
	while (current) {
		current->nr++;
		current = current->next;
	}
	ae->ajchannels++;
	return ac;
}
AjoinChan *findAjoinChan(AjoinEntry *ae, char *chan) {
	AjoinChan *current;
	if (!ae)
		return NULL;
	for (current = ae->channels; current; current = current->next) {
		if (!stricmp(chan, current->channel))
			return current;
	}
	return NULL;
}
AjoinChan *findAjoinChanNr(AjoinEntry *ae, int nr) {
	AjoinChan *current;
	if (!nr)
		return NULL;
	for (current = ae->channels; current; current = current->next) {
		if (current->nr == nr)
			return current;
	}
	return NULL;
}
/**
 * We delete an AjoinChan from the list, but we do not update the numbering.
 * This needs to be done later by calling numberAjoinChans() because numbers shouldn't
 * change while deleting ajoinchans by number.
 **/
static int deleteAjoinChan(AjoinEntry *ae, AjoinChan *ac) {
	if (!ae || !ac)
		return 0;
	if (ac->prev)
		ac->prev->next = ac->next;
	else
		ae->channels = ac->next;
	if (ac->next)
		ac->next->prev = ac->prev;
	if (ac->channel)
		free(ac->channel);
	if (ac->key)
		free(ac->key);
	free(ac);
	ae->ajchannels--;
	return 1;
}
static int clearAjoinChans(AjoinEntry *ae) {
	AjoinChan *ac = NULL, *next = NULL;
	if (!ae)
		return 0;
	for (ac = ae->channels; ac; ac = next) {
		if (ac->next)
			next = ac->next;
		else
			next = NULL;
		if (ac->channel)
			free(ac->channel);
		if (ac->key)
			free(ac->key);
		free(ac);
	}
	ae->channels = NULL;
	ae->ajchannels = 0;
	return 1;
}
static int numberAjoinChans(AjoinEntry *ae) {
	AjoinChan *current;
	int nr = 1;
	if (!ae)
		return 0;
	for (current = ae->channels; current; current = current->next) {
		current->nr  = nr++;
	}
	return 1;
}
/* ------------------------------------------------------------------------------- */
static AjoinEntry *createAjoinEntry(NickCore *nc) {
	AjoinEntry *ae = NULL;
	char buf[BUFSIZE];
	if (!nc)
		return NULL;
	if ((ae = malloc(sizeof(AjoinEntry))) == NULL) {
		fatal("Out Of Memory!");
	}
	ae->prev = NULL;
	ae->next = NULL;
	ae->row = counter;
	ae->ajchannels = 0;
	ae->in_use = 0;
	ae->flags = DefAjoinFlags;
	ae->channels = NULL;
	/* add it to the ajoinTable */
	if (ajoinTable[counter] == NULL) {
		ajoinTable[counter] = ae;
		ae->col = 0;
	} else {
		int i = 0, last = -1;
		AjoinEntry *t, *prev = NULL;
		for (t = ajoinTable[counter]; t; t = t->next) {
			if (t->col != last + 1)
				break;
			last = t->col;
			i++;
			prev = t;
		}
		ae->col = i;
		if (prev) {
			if (prev->next) {
				ae->next = prev->next;
				prev->next->prev = ae;
			}
			prev->next = ae;
		}
		ae->prev = prev;
	}
	/* add to the nc, so it can be addressed relative quickly */
	snprintf(buf, BUFSIZE, "%d;%d", ae->row, ae->col);
	moduleAddData(&nc->moduleData, ModDataKey, buf);
	if (counter < 1023)
		counter++;
	else
		counter = 0;
	return ae;
}
AjoinEntry *getAjoinEntry(NickCore *nc) {
	char *t, *x, *y;
	int row, col;
	AjoinEntry *ae = NULL;
	if (!nc)
		return NULL;
	t = moduleGetData(&nc->moduleData, ModDataKey);
	if (t == NULL) {
		return NULL;
	}
	x = myStrGetToken(t, ';', 0);
	y = myStrGetToken(t, ';', 1);
	row = atoi(x);
	col = atoi(y);
	for (ae = ajoinTable[row]; ae; ae = ae->next) {
		if (ae->col == col)
			break;
	}
	free(t);
	free(x);
	free(y);
	return ae;
}
static int deleteAjoinEntry(AjoinEntry *ae) {
	if (!ae)
		return 0;
	if (ajoinTable[ae->row] == ae)
		ajoinTable[ae->row] = ae->next;
	if (ae->prev)
		ae->prev->next = ae->next;
	if (ae->next)
		ae->next->prev = ae->prev;
	clearAjoinChans(ae);
	ae->prev = NULL;
	ae->next = NULL;
	free(ae);
	return 1;
}
/**
 * When we are called, we assume used entries have in_use set to 1 (done while the
 * DB is saved) and reset it to 0 for all other entries.
 * Entries with in_use to 0 are free()'d and removed from the AjoinTable since we
 * assume all other references to it have allready been removed.
 **/
void freeUnusedEntries() {
	AjoinEntry *next = NULL, *ae = NULL;
	int i;
	for (i = 0; i < 1024; i++) {
		for (ae = ajoinTable[i]; ae; ae = next) {
			if (ae->in_use == 1) {
				/* Entry is still being used... resetting */
				ae->in_use = 0;
				next = ae->next;
				continue;
			}
			/* Entry is no longer being used... free it */
			next = ae->next;
			deleteAjoinEntry(ae);
		}
	}
}
/* ------------------------------------------------------------------------------- */
/**
 * When anope saves her databases, we do the same.
 **/
int do_save(int argc, char **argv) {
	if ((argc >= 1) && (!stricmp(argv[0], EVENT_STOP)))
		save_ajoin_db();
	return MOD_CONT;
}
/**
 * When anope backs her databases up, we do the same.
 **/
int db_backup(int argc, char **argv) {
	if ((argc >= 1) && (!stricmp(argv[0], EVENT_STOP))) {
		alog("[ns_ajoin] Backing up AJOIN database...");
		ModuleDatabaseBackup(AJoinDB);
		/* When anope backs her databases up, we also clear unused entries from the AjoinTable */
		freeUnusedEntries();
	}
	return MOD_CONT;
}
/**************************************************************************
 *                DataBase Handling
 **************************************************************************/
void load_ajoin_db(void) {
	DBFile *dbptr = scalloc(1, sizeof(DBFile));
	AjoinEntry *ae = NULL;
	NickCore *nc = NULL;
	char *key, *value;
	int retval = 0;
	fill_db_ptr(dbptr, 0, AJOINDBVERSION, s_NickServ, AJoinDB);
	/* let's remove existing temp files here, because we only load dbs on startup */
	remove(dbptr->temp_name);
	/* Open the db, fill the rest of dbptr and allocate memory for key and value */
	if (new_open_db_read(dbptr, &key, &value)) {
		free(dbptr);
		return;                 /* Bang, an error occurred */
	}
	while (1) {
		/* read a new entry and fill key and value with it -Certus */
		retval = new_read_db_entry(&key, &value, dbptr->fptr);
		if (retval == DB_READ_ERROR) {
			new_close_db(dbptr->fptr, &key, &value);
			free(dbptr);
			return;
		} else if (retval == DB_EOF_ERROR) {
			new_close_db(dbptr->fptr, &key, &value);
			free(dbptr);
			return;
        } else if (retval == DB_READ_BLOCKEND) {        /* DB_READ_BLOCKEND */
        	/* prevent chans from one block ending up in another... */
            ae = NULL;
		} else {		 /* DB_READ_SUCCESS */
			if (!*key || !*value)
				continue;
			/* nick */
			if (!stricmp(key, "ncd")) {
				if ((nc = findcore(value))) {
					ae = createAjoinEntry(nc);
				} else
					ae = NULL;
			/* flags */
			} else if (!stricmp(key, "flgs") && ae) {
				ae->flags = (uint16) atoi(value);
			/* channel entry */
			} else if (!stricmp(key, "che") && ae) {
				char *chan = myStrGetToken(value, ' ', 0);
				char *chkey = myStrGetToken(value, ' ', 1);
				addAjoinChan(ae, chan, chkey);
				free(chan);
				if (chkey)
					free(chkey);
			} else if (!stricmp(key, "AJOIN DB VERSION")) {
				if ((int)atoi(value) != AJOINDBVERSION) {
					alog("[\002ns_ajoin\002] Database version does not match any database versions supported by this module.");
					alog("[\002ns_ajoin\002] Continuing with clean database...");
					break;
				}
			}
		} 					/* else */
	}					/* while */
	free(dbptr);
}
void save_ajoin_db(void) {
	DBFile *dbptr = scalloc(1, sizeof(DBFile));
	NickCore *nc;
	int i;
	fill_db_ptr(dbptr, 0, AJOINDBVERSION, s_NickServ, AJoinDB);
	/* time to backup the old db */
	rename(AJoinDB, dbptr->temp_name);
	if (new_open_db_write(dbptr)) {
		rename(dbptr->temp_name, AJoinDB);
		free(dbptr);
		return;                /* Bang, an error occurred */
	}
	/* Store the version of the DB in the DB as well...
	 * This will make stuff a lot easier if the database scheme needs to modified. */
	new_write_db_entry("AJOIN DB VERSION", dbptr, "%d", AJOINDBVERSION);
	new_write_db_endofblock(dbptr);
	/* Go over the entire NickCore list and check whether each one has an ajoin entry */
    for (i = 0; i < 1024; i++) {
        for (nc = nclists[i]; nc; nc = nc->next) {
			AjoinEntry *ae = NULL;
			AjoinChan *ac = NULL;
			ae = getAjoinEntry(nc);
			if (!ae)
				continue;
			/* If there are no entries in the list and the configuration is set to the default
			 * value, there is no need to save it to the db. */
			if (ae->ajchannels == 0 && ae->flags == DefAjoinFlags) {
				/* Delete the entry first since it s just eating up space... */
				moduleDelData(&nc->moduleData, ModDataKey);
				deleteAjoinEntry(ae);
				continue;
			}
			new_write_db_entry("ncd", dbptr, "%s", nc->display);
			new_write_db_entry("flgs", dbptr, "%d", ae->flags);
			if (ae->ajchannels > 0) {
				for (ac = ae->channels; ac; ac = ac->next) {
					new_write_db_entry("che", dbptr, "%s %s", ac->channel, (ac->key ? ac->key : ""));
				}
			}
			new_write_db_endofblock(dbptr);
			/* Marking the entry as in use... */
			ae->in_use = 1;
		}
	}
	if (dbptr) {
		new_close_db(dbptr->fptr, NULL, NULL);  /* close file */
		remove(dbptr->temp_name);       /* saved successfully, no need to keep the old one */
		free(dbptr);           /* free the db struct */
	}
}
/* ------------------------------------------------------------------------------- */
/**
 * We check if the ircd is supported
 **/
int valid_ircd(void) {
	if (!stricmp(IRCDModule, "unreal31"))
		return 1;
	if (!stricmp(IRCDModule, "unreal32"))
		return 1;
	if (!stricmp(IRCDModule, "viagra"))
		return 1;
	if (!stricmp(IRCDModule, "ptlink"))
		return 1;
	if (!stricmp(IRCDModule, "ultimate2"))
		return 1;
	if (!stricmp(IRCDModule, "plexus3"))
		return 1;
	if (!stricmp(IRCDModule, "inspircd10"))
		return 1;
	if (!stricmp(IRCDModule, "inspircd11"))
		return 1;
	return 0;
}
/* ------------------------------------------------------------------------------- */
/**
 * (Re)Load the configuration directives
 **/
void load_config(void) {
	int i;
	Directive confvalues[][1] = {
		{{"AJoinDB", {{PARAM_STRING, PARAM_RELOAD, &AJoinDB}}}},
	};
	if (AJoinDB)
		free(AJoinDB);
	AJoinDB = NULL;
	for (i = 0; i < 1; i++)
		moduleGetConfigDirective(confvalues[i]);
	if (!AJoinDB)
		AJoinDB = sstrdup(DefAJoinDB);
	if (debug)
		alog ("[ns_ajoin] debug: AJoinDB set to %s", AJoinDB);
}
/**
 * Upon /os reload call the routines for reloading the configuration directives
 **/
int reload_config(int argc, char **argv) {
	if (argc >= 1) {
		if (!stricmp(argv[0], EVENT_START)) {
			alog("[ns_ajoin]: Reloading configuration directives...");
			load_config();
		}
	}
	return MOD_CONT;
}
/* ------------------------------------------------------------------------------- */
/**
 * Add language strings to the module's language db.
 **/
void add_languages(void) {
	char *langtable_en_us[] = {
		/* LANG_AJOIN_DESC */
		" AJOIN       Manipulate the AutoJoin list.",
		/* LANG_AJOIN_SYNTAX */
		" Syntax: AJOIN { ADD | DEL | LIST | CLEAR } [\037channel\037 | \037entry-list\037] [\037key\037]",
		/* LANG_AJOIN_SYNTAX_EXT */
		" Syntax: \002AJOIN ADD \037channel\037 \037key\037\002 \n"
		"         \002AJOIN DEL {\037channel\037 | \037entry-nr\037 | \037list\037}\002\n"
		"         \002AJOIN LIST [\037mask\037 | \037list\037]\002 \n"
		"         \002AJOIN CLEAR\002 \n"
		" \n"
		" Maintains the \002AutoJoin list\002 for nick group. \n"
		" If a user identifies to his nickname, he will \n"
		" automatically join the listed channels. \n"
		" \n"
		" The \002AJOIN ADD\002 command adds the given channel\n"
		" to the AutoJoin list or updates the existing entry, \n"
		" if it is allready present on the AutoJoin list. \n"
		" If no key is given and an entry for the channel is already\n"
		" present, the key allready set (if any) will be unset. \n"
		" Only \002registered channels\002 are accepted! \n"
		" \n"
		" The \002AJOIN DEL\002 command removes the given channel \n"
		" from the AutoJoin list. If a list of entry numbers is given,\n"
		" those entries are deleted.\n"
		" \n"
		" The \002AJOIN LIST\002 command displays the AutoJoin list.\n"
		" If a wildcard mask is given, only those entries matching the\n"
		" mask are displayed. If a list of entry numbers is given, only\n"
		" those entries are shown.\n"
		" \n"
		" The \002AJOIN CLEAR\002 command clears all entries on the \n"
		" AutoJoin list.",
		/* LANG_AJOIN_DISABLED */
		" Sorry, AutoJoin list modification is temporarily disabled.",
		/* LANG_NO_LOCAL_CHAN */
		" Cannot add local channels to ajoin list.",
		/* LANG_CHAN_SYMB_REQUIRED */
		" The channel symbol '\002#\002' is required.",
		/* LANG_CHAN_UPDATED */
		" Ajoin entry updated.",
		/* LANG_AJOIN_LIST_FULL */
		" The AutoJoin list is full.",
		/* LANG_CHAN_ADDED */
		" \002%s\002 has been added to the ajoin list.",
		/* LANG_NO_AJOINS */
		" No AutoJoins have been set.",
		/* LANG_AJOIN_LIST_EMPTY */
		" The AutoJoins list is empty.",
		/* LANG_NO_ENTRY */
		" \002%s\002 is not in the AutoJoin list.",
		/* LANG_CHAN_DELETED */
		" \002%s\002 has been deleted from the AutoJoin list.",
		/* LANG_AJOIN_ENTRY */
		" %d - Channel: %s  -  Key: %s",
		/* LANG_AJOIN_ENTRIES */
		" Found %d entries",
		/* LANG_AJOIN_LIST_CLEARED */
		" AJOIN list cleared.",
		/* LANG_UNKWN_AJOIN_OPTION */
		" Unknown AJOIN option.",
		/* LANG_AJOINING */
		" Automatically joining %s .",
		/* LANG_AJOINING_FAILED */
		" Could not automatically join %s .",
		/* LANG_SET_AJOIN_DESC */
		" \n"
		" Commands Provided by module ns_ajoin: \n"
		" AJOIN      Change your autojoin settings.",
		/* LANG_SET_AJOIN_SYNTAX */
		" Syntax: SET AJOIN { ON | OFF | SILENT }",
		/* LANG_SET_AJOIN_SYNTAX_EXT */
		" Syntax: \002SET AJOIN { ON | OFF | SILENT }\002 \n"
		" \n"
		" Toggles whether or not your AutoJoin list is performed when \n"
		" you IDENTIFY or UPDATE.\n"
		" When set to ON, you will receive a notification for each channel\n"
		" you are automatically joined to. When set to SILENT, you will receive\n"
		" one message indicating you are automatically joining channels. \n"
		" When set to OFF, you will not join any channels upon identifying.\n"
		" \n"
		" The default setting is \002ON\002.",
		/* LANG_SET_AJOIN_ON */
		" AutoJoin is now \002ON\002.",
		/* LANG_SET_AJOIN_SILENT */
		" AutoJoin is now \002ON\002 and set to \002SILENT\002 mode.",
		/* LANG_SET_AJOIN_OFF */
		" AutoJoin is now \002OFF\002.",
		/* LANG_AJOINING_SUM_SUCCESS */
		" Successfully joined %d channel(s).",
		/* LANG_AJOINING_SUM_FAILED */
		" Failed joining %d channel(s).",
		/* LANG_AJOIN_DELETED_NR_1 */
		" Deleted %d entry from the AutoJoin list.",
		/* LANG_AJOIN_DELETED_NR */
		" Deleted %d entries from the AutoJoin list.",
	};
	char *langtable_nl[] = {
		/* LANG_AJOIN_DESC */
		" AJOIN       Beheer de autojoin lijst.",
		/* LANG_AJOIN_SYNTAX */
		" Gebruik: AJOIN {ADD | DEL | LIST | CLEAR} [\037kanaal\037 | \037lijst\037] [\037wachtwoord\037]",
		/* LANG_AJOIN_SYNTAX_EXT */
		" Gebruik: \002AJOIN ADD \037kanaal\037 \037wachtwoord\037\002 \n"
		"          \002AJOIN DEL {\037kanaal\037 | \037nummer\037 | \037lijst\037}\002 \n"
		"          \002AJOIN LIST [\037mask\037 | \037lijst\037]\002 \n"
		"          \002AJOIN CLEAR\002 \n"
		" \n"
		" Onderhoud de \002AutoJoin lijst\002 voor een nick groep. \n"
		" Als een gebruiker zich voor zijn nickname identificeerd,\n"
		" zal hij automatisch de kanalen in de lijst joinen.\n"
		" \n"
		" Het \002AJOIN ADD\002 commando voegt het gegeven kanaal toe \n"
		" aan de AutoJoin lijst of werkt de bestaande entry bij, \n"
		" indien het kanaal reeds aanwezig is op de AutoJoin lijst. \n"
		" Als er geen wachtwoord is opgegeven en het kanaal reeds \n"
		" aanwezig is op de lijst, zal het wachtwoord dat reeds is \n"
		" ingesteld (indien aanwezig) verwijderd worden. \n"
		" Enkel \002geregistreerde kanalen\002 worden aanvaard!. \n"
		" \n"
		" Het \002AJOIN DEL\002 commando verwiderd het gegeven kanaal \n"
		" van de AutoJoin lijst. Als een lijst met entry-nummers is\n"
		" gegeven, worden deze verwijderd.\n"
		" \n"
		" Het \002AJOIN LIST\002 commando geeft de AutoJoin lijst weer.\n"
		" Als een wildcard mask is opgegeven worden alleen de overeenkomsten\n"
		" weergegeven. Als een lijst van entry-nummers in gegeven wordt,\n"
		" worden alleen die weergegeven\n"
		" \n"
		" Het \002AJOIN CLEAR\002 commando maakt de AutoJoin lilst leeg.",
		/* LANG_AJOIN_DISABLED */
		" Sorry,het wijzigen van de AutoJoin lijst is tijdelijk uitgeschakeld.",
		/* LANG_NO_LOCAL_CHAN */
		" Kan geen lokale kanalen  toevoegen aan de ajoin list.",
		/* LANG_CHAN_SYMB_REQUIRED */
		" Het kanaal symbool '\002#\002' is vereist.",
		/* LANG_CHAN_UPDATED */
		" AutoJoin entry bijgewerkt.",
		/* LANG_AJOIN_LIST_FULL */
		" De AutoJoin lijst is vol.",
		/* LANG_CHAN_ADDED */
		" \002%s\002 is aan de autojoin lijst toegevoegd.",
		/* LANG_NO_AJOINS */
		" Er zijn geen AutoJoins ingesteld.",
		/* LANG_AJOIN_LIST_EMPTY */
		" De AutoJoin lijst is leeg.",
		/* LANG_NO_ENTRY */
		" \002%s\002 bestaat niet in de AutoJoin lijst.",
		/* LANG_CHAN_DELETED */
		" \002%s\002 is van de AutoJoin lijst verwijderd.",
		/* LANG_AJOIN_ENTRY */
		" %d - Kanaal: %s  -  Wachtwoord: %s",
		/* LANG_AJOIN_ENTRIES */
		" %d plaatsen gevonden.",
		/* LANG_AJOIN_LIST_CLEARED */
		" AutoJoin lijst is leeggemaakt.",
		/* LANG_UNKWN_AJOIN_OPTION */
		" Onbekende AJOIN parameter.",
		/* LANG_AJOINING */
		" Automatisch %s joinen",
		/* LANG_AJOINING_FAILED */
		" Kon %s niet automatisch joinen.",
		/* LANG_SET_AJOIN_DESC */
		" \n"
		" Commandos toegevoegd door module ns_ajoin: \n"
		" AJOIN      Wijzig de instellingen van je autojoin.",
		/* LANG_SET_AJOIN_SYNTAX */
		" Gebruik: SET AJOIN { ON | OFF | SILENT }",
		/* LANG_SET_AJOIN_SYNTAX_EXT */
		" Gebruik: \002SET AJOIN { ON | OFF | SILENT }\002 \n"
		" \n"
		" Configureert of je AutoJoin lijst al dan niet wordt uitgevoerd bij \n"
		" een nickserv IDENTIFY of UPDATE.\n"
		" Wanneer deze optie op ON staat zal je een melding krijgen voor elk kanaal\n"
		" dat je automatisch joint. Wanneer het op SILENT staat, krijg je één melding\n"
		" die aangeeft dat je automatisch kanalen joint. \n"
		" Wanneer deze optie op OFF staat zal je niet automatisch kanalen joinen.\n"
		" \n"
		" De standaard waarde is \002ON\002.",
		/* LANG_SET_AJOIN_ON */
		" AutoJoin staat nu \002OAAN\002.",
		/* LANG_SET_AJOIN_SILENT */
		" AutoJoin staat nu \002AAN\002 in \002STILLE\002 modus.",
		/* LANG_SET_AJOIN_OFF */
		" AutoJoin staat nu \002UIT\002.",
		/* LANG_AJOINING_SUM_SUCCESS */
		" Succesvol %d kanaal/kanalen gejoint.",
		/* LANG_AJOINING_SUM_FAILED */
		" Kon %d kanaal/kanalen niet joinen.",
		/* LANG_AJOIN_DELETED_NR_1 */
		" %d kanaal uit de AutoJoin lijst verwijderd.",
		/* LANG_AJOIN_DELETED_NR */
		" %d kanalen uit de AutoJoin lijst verwijderd.",
	};
	char *langtable_de[] = {
		/* LANG_AJOIN_DESC */
		" AJOIN       AutoJoin-Liste bearbeiten.",
		/* LANG_AJOIN_SYNTAX */
		" Syntax: AJOIN { ADD | DEL | LIST | CLEAR } [\037channel\037 | \037entry-list\037] [\037key\037]",
		/* LANG_AJOIN_SYNTAX_EXT */
		" Syntax: \002AJOIN ADD \037channel\037 \037key\037\002 \n"
		"         \002AJOIN DEL {\037channel\037 | \037entry-nr\037 | \037list\037}\002 \n"
		"         \002AJOIN LIST [\037mask\037 | \037list\037]\002 \n"
		"         \002AJOIN CLEAR\002 \n"
		" \n"
		" Verwaltet die \002AutoJoin Liste\002 für eine Nick-Gruppe. \n"
		" Wenn ein user sich mit seinem Nicknamen identifiziert, so \n"
		" betritt er automatisch die Räume aus der Liste. \n"
		" \n"
		" \002AJOIN ADD\002 fügt den angegebenen Raum \n"
		" der AutoJoin-Liste hinzu oder aktualisiert den \n"
		" Eintrag, wenn der angegebene Raum schon vorhanden ist. \n"
		" Wenn kein Raum-Schlüssel angegeben wird und es bereits \n"
		" einen Listeneintrag für diesen Raum gibt, wo ein Schlüssel \n"
		" angegeben wurde, dann wird dieser Schlüssel zurückgesetzt. \n"
		" Nur \002registrierte Räume\002 sind erlaubt! \n"
		" \n"
		" \002AJOIN DEL\002 entfernt den angegebenen Raum \n"
		" aus der AutoJoin-Liste. Wurden die Nummern der Einträge angegeben,\n"
		" werden die dazugehörigen Einträge gelöscht.\n"
		" \n"
		" \002AJOIN LIST\002 zeigt die AutoJoin-liste.\n"
		" If a wildcard mask is given, only those entries matching the\n"
		" mask are displayed. Wurden die Nummern der Listeneinträge angegeben,\n"
		" werden nur diese Einträge angezeigt.\n"
		" \n"
		" \002AJOIN CLEAR\002 löscht alle Einträge aus der \n"
		" AutoJoin-Liste.",
		/* LANG_AJOIN_DISABLED */
		" Sorry, das Bearbeiten der AutoJoin-Liste ist vorrübergehend deaktiviert.",
		/* LANG_NO_LOCAL_CHAN */
		" Kann keine lokalen Räume zur AutoJoin-Liste hinzufügen.",
		/* LANG_CHAN_SYMB_REQUIRED */
		" Das Raumsymbol '\002#\002' wird benötigt.",
		/* LANG_CHAN_UPDATED */
		" AutoJoin Eintrag aktualisiert.",
		/* LANG_AJOIN_LIST_FULL */
		" Die AutoJoin-Liste ist voll.",
		/* LANG_CHAN_ADDED */
		" \002%s\002 wurde zur AutoJoin-Liste hinzugefügt.",
		/* LANG_NO_AJOINS */
		" Es wurden noch keine AutoJoins festgelegt.",
		/* LANG_AJOIN_LIST_EMPTY */
		" Die AutoJoin-Liste ist leer.",
		/* LANG_NO_ENTRY */
		" \002%s\002 ist nicht in der AutoJoin-liste.",
		/* LANG_CHAN_DELETED */
		" \002%s\002 wurde aus der AutoJoin-Liste gelöscht.",
		/* LANG_AJOIN_ENTRY */
		" %d - Channel: %s  -  Key: %s",
		/* LANG_AJOIN_ENTRIES */
		" Habe %d Einträge gefunden ",
		/* LANG_AJOIN_LIST_CLEARED */
		" AutoJoin-Liste gesäubert.",
		/* LANG_UNKWN_AJOIN_OPTION */
		" Unbekannte AutoJoin-Option.",
		/* LANG_AJOINING */
		" AutoJoining %s .",
		/* LANG_AJOINING_FAILED */
		" Konnte %s nicht AutoJoinen .",
		/* LANG_SET_AJOIN_DESC */
		" \n"
		" Unterstützte Kommandos von Modul ns_ajoin: \n"
		" AJOIN      AutoJoin-Settings ändern.",
		/* LANG_SET_AJOIN_SYNTAX */
		" Syntax: SET AJOIN { ON | OFF | SILENT }",
		/* LANG_SET_AJOIN_SYNTAX_EXT */
		" Syntax: \002SET AJOIN { ON | OFF | SILENT }\002 \n"
		" \n"
		" Stellt ein, ob die AutoJoin-Liste abgearbeitet wird, wenn \n"
		" IDENTIFY oder UPDATE ausgeführt wird.\n"
		" Wenn ON eingestellt wird, bekommst du für jeden Raum, den du AutoJoinst \n"
		" eine Nachricht. Bei Einstellung SILENT, bekommst du \n"
		" eine Nachricht, welche dir mitteilt, dass du AutoJoinst. \n"
		" Die Einstellung OFF bewirkt, dass du keinen Raum beim IDENTIFY Befehl betrittst. \n"
		" \n"
		" Voreinstellung ist: \002ON\002.",
		/* LANG_SET_AJOIN_ON */
		" AutoJoin ist jetzt \002ON\002.",
		/* LANG_SET_AJOIN_SILENT */
		" AutoJoin ist jetzt \002ON\002 und auf \002SILENT\002 eingestellt.",
		/* LANG_SET_AJOIN_OFF */
		" AutoJoin ist jetzt \002OFF\002.",
		/* LANG_AJOINING_SUM_SUCCESS */
		" Erfolgreich %d Channel(s) automatisch betreten.",
		/* LANG_AJOINING_SUM_FAILED */
		" Fehler beim AutoJoinen von %d channel(s).",
		/* LANG_AJOIN_DELETED_NR_1 */
		" %d Eintrag wurde aus der AutoJoin-Liste entfernt.",
		/* LANG_AJOIN_DELETED_NR */
		" %d Einträge wurden aus der AutoJoin-Liste entfernt.",
	};
	char *langtable_tr[] = {
		/* LANG_AJOIN_DESC */
		" AJOIN       Otojoin listesinizi yönetebilirsiniz.",
		/* LANG_AJOIN_SYNTAX */
		" Kullanýmý: AJOIN { ADD | DEL | LIST | CLEAR } [\037#kanaladi\037 | \037entry-list\037] [\037anahtar\037]",
		/* LANG_AJOIN_SYNTAX_EXT */
		" Kullanýmý: \002AJOIN ADD \037#kanaladi\037 \037anahtar\037\002 \n"
		"            \002AJOIN DEL {\037#kanaladi\037 | \037entry-nr\037 | \037list\037}\002 \n"
		"            \002AJOIN LIST [\037mask\037 | \037list\037]\002 \n"
		"            \002AJOIN CLEAR\002 \n"
		" \n"
		" Nick grubunuzun \002otojoin list\002 düzenini görüntüler. \n"
		" Nickinizi identify ettikten sonra, otomatik \n"
		" olarak eklediðiniz kanallara giriþ yaparsýnýz. \n"
		" \n"
		" Otojoin listeniz hazýrda var ise \002AJOIN ADD\002 komutu\n"
		" ile kanal ekleyebilir veya otojoin  \n"
		" listesini düzenlersiniz. \n"
		" Eðer kanal eklerken anahtar belirtilmediyse ve kanalda \n"
		" anahtar var ise, kanala girmek için anahtar olmucak.\n"
		" Sadece \002kayýtlý kanal\002 kabul edilir! \n"
		" \n"
		" \002AJOIN DEL\002 komutu eklediðiniz kanalý otojoin \n"
		" listesinden çýkartýr. Eðerki listede ki giriþ numaralarý verilirse,\n"
		" bu giriþler silinir.\n"
		" \n"
		" \002AJOIN LIST\002 komutu otojoin listenizi görüntüler.\n"
		" Eðerki mask belirtilirse, sadece belirtrilen bu masklar\n"
		" görüntülenir. Eðerki sadece listede ki giriþ numaralarý belirtilirse, sadece\n"
		" bunlar gözükür.\n"
		" \n"
		" \002AJOIN CLEAR\002 komutu bütün ekli olan kanallarýnýzý \n"
		" otojoin listenizden siler.",
		/* LANG_AJOIN_DISABLED */
		" Üzgünüm, ajoin komutu kullaným dýþý býrakýlmýþtýr.",
		/* LANG_NO_LOCAL_CHAN */
		" Lokal kanal ekleyemezsiniz.",
		/* LANG_CHAN_SYMB_REQUIRED */
		" '\002#\002' iþaretini kulanmanýz gerekmektedir.",
		/* LANG_CHAN_UPDATED */
		" Otojoin giriþleriniz güncellendi.",
		/* LANG_AJOIN_LIST_FULL */
		" Otojoin listiniz dolmuþtur.",
		/* LANG_CHAN_ADDED */
		" \002%s\002 kanali otojoin listenize eklendi.",
		/* LANG_NO_AJOINS */
		" Herhangi bir otojoin kullanýlmadý.",
		/* LANG_AJOIN_LIST_EMPTY */
		" Otojoin listiniz boþ.",
		/* LANG_NO_ENTRY */
		" \002%s\002 kanali otojoin listenizde deðildir.",
		/* LANG_CHAN_DELETED */
		" \002%s\002 kanali otojoin listenizden silindi.",
		/* LANG_AJOIN_ENTRY */
		" %d - Kanaladi: %s  -  Anahtar: %s",
		/* LANG_AJOIN_ENTRIES */
		" %d giriþ bulundu",
		/* LANG_AJOIN_LIST_CLEARED */
		" Otojoin listeniz temizlendi.",
		/* LANG_UNKWN_AJOIN_OPTION */
		" Bilinmeyen otojoin komutu.",
		/* LANG_AJOINING */
		" %s kanalýna otomatik olarak giriþ yaptýnýz .",
		/* LANG_AJOINING_FAILED */
		" %s kanalýna otomatik giriþ yapamadýnýz .",
		/* LANG_SET_AJOIN_DESC */
		" \n"
		" ns_ajoin modulu hakkýnda: \n"
		" AJOIN      Otojoin ayarlarýnýzý düzenler.",
		/* LANG_SET_AJOIN_SYNTAX */
		" Kullanýmý: SET AJOIN { ON | OFF | SILENT }",
		/* LANG_SET_AJOIN_SYNTAX_EXT */
		" Kullanýmý: \002SET AJOIN { ON | OFF | SILENT }\002 \n"
		" \n"
		" Otojoin listiniz hakkýnda IDENTIFY ve UPDATE edildiginde \n"
		" geniþ bilgi sunar.\n"
		" ON olduðunda otojoin listesinizdeki kanallara giriþ\n"
		" yapmanýzý saðlar. Bu ayar SILENT olduðunda, sadece bir tane\n"
		" otomatik giriþ yaptýðýnýz kanallardan giriþ mesajýnýz alýrsýnýz. \n"
		" Bu ayar OFF olduðunda, hiçbir kanala otojoin yapamazsýnýz.\n"
		" \n"
		" Belirlenmiþ ayar \002ON\002.",
		/* LANG_SET_AJOIN_ON */
		" Þuan otojoin \002ON\002.",
		/* LANG_SET_AJOIN_SILENT */
		" Otojoin þuan \002ON\002 ve \002SILENT\002 modundadýr.",
		/* LANG_SET_AJOIN_OFF */
		" Otojoin þuan \002OFF\002.",
		/* LANG_AJOINING_SUM_SUCCESS */
		" %d kanallarýna baþarýlý þekilde giriþ yaptýnýz.",
		/* LANG_AJOINING_SUM_FAILED */
		" %d kanallarýna giriþ yapamadýnýz.",
		/* LANG_AJOIN_DELETED_NR_1 */
		" %d giriþ otojoin listesinden silindi.",
		/* LANG_AJOIN_DELETED_NR */
		" %d giriþler otojoin listesinden silindi.",
	};
	char *langtable_ru[] = {
		/* LANG_AJOIN_DESC */
		"    AJOIN      Óïðàâëÿåò AJOIN-ñïèñêîì (àâòîçàõîä).",
		/* LANG_AJOIN_SYNTAX */
		"Ñèíòàêñèñ: AJOIN { ADD | DEL | LIST | CLEAR } [\037#êàíàë\037 | \037ñïèñîê_çàïèñåé\037] [\037êëþ÷\037]",
		/* LANG_AJOIN_SYNTAX_EXT */
		"Ñèíòàêñèñ: \002AJOIN ADD \037#êàíàë\037 \037êëþ÷\037\002 \n"
		"           \002AJOIN DEL {\037#êàíàë\037 | \037íîìåð_çàïèñè\037 | \037ñïèñîê_çàïèñåé\037}\002\n"
		"           \002AJOIN LIST [\037ìàñêà\037 | \037ñïèñîê\037]\002 \n"
		"           \002AJOIN CLEAR\002 \n"
		" \n"
		"Ïîçâîëÿåò óïðàâëÿòü ñïèñêîì \002àâòîçàõîäà\002 íà êàíàëû. \n"
		"Êàê òîëüêî ïîëüçîâàòåëü èäåíòèôèöèðóåòñÿ ê ñâîåìó íèêó, îí áóäåò \n"
		"àâòîìàòè÷åñêè ïðèñîåäèíåí ê óêàçàííûì â ñïèñêå êàíàëàì. \n"
		" \n"
		"Êîìàíäà \002AJOIN ADD\002 ïîçâîëÿåò äîáàâèòü êàíàë â ñïèñîê \n"
		"àâòîçàõîäà èëè îáíîâèòü èíôîðìàöèþ äëÿ óæå ñóùåñòâóþùåãî â \n"
		"â ñïèñêå êàíàëà. \n"
		"Åñëè ïðè îáíîâëåíèè çàïèñè äëÿ êàêîãî ëèáî êàíàëà, âû íå óêàæåòå \n"
		"äîïîëíèòåëüíûé ïàðàìåòð 'êëþ÷' - òåêóùèé óñòàíîâëåííûé äëÿ êàíàëà \n"
		"êëþ÷ áóäåò óäàëåí (ïðè óñëîâèè ÷òî îí ñòîÿë, êîíå÷íî) \n"
		"Ñïèñîê àâòîçàõîäà ìîæåò ñîäåðæàòü òîëüêî \002çàðåãèñòðèðîâàííûå êàíàëû\002 \n"
		" \n"
		"Êîìàíäà \002AJOIN DEL\002 ïîçâîëÿåò óäàëèòü óêàçàííûé êàíàë èç \n"
		"ñïèñêà àâòîçàõîäà.  êà÷åñòâå ïàðàìåòðà âû ìîæåòå óêàçàòü ñïèñîê \n"
		"çàïèñåé èëè êîíêðåòíûé íîìåð çàïèñè ajoin-ñïèñêà \n"
		" \n"
		"Êîìàíäà \002AJOIN LIST\002 îòîáðàæàåò òåêóùèé ñïèñîê àâòîçàõîäà.\n"
		"Äîïóñòèìî èñïîëüçîâàíèå ïîäñòàíîâî÷íûõ ñèìâîëîâ (ãëîáàëüíûõ ìàñîê)\n"
		"â êà÷åñòâå äîïîëíèòåëüíîãî ïàðàìåòðà. Òàê æå, â êà÷åñòâå óëîâèÿ ëèñòèíãà\n"
		"êàíàëîâ, âû ìîæåòå óêàçàòü ñïèñîê çàïèñåé.\n"
		" \n"
		"Êîìàíäà \002AJOIN CLEAR\002 ïîëíîñòüþ óäàëÿåò \002âñå\002 çàïèñè èç \n"
		"ñïèñêà àâòîçàõîäà.",
		/* LANG_AJOIN_DISABLED */
		"Èçâèíèòå, íî ìîäèôèêàöèÿ ñïèñêà àâòîçàõîäà âðåìåííî íåäîñòóïíà.",
		/* LANG_NO_LOCAL_CHAN */
		"Âû íå ìîæåòå äîáàâèòü ëîêàëüíûå êàíàëû â ñïèñîê àâòîçàõîäà.",
		/* LANG_CHAN_SYMB_REQUIRED */
		"Âû äîëæíû óêàçàòü ñèìâîë êàíàëà: '\002#\002' .",
		/* LANG_CHAN_UPDATED */
		"Îáíîâëåíà îäíà çàïèñü â ñïèñêå àâòîçàõîäà.",
		/* LANG_AJOIN_LIST_FULL */
		"Ñïèñîê àâòîçàõîäà ïåðåïîëíåí.",
		/* LANG_CHAN_ADDED */
		"Êàíàë \002%s\002 óñïåøíî äîáàâëåí â ñïèñîê àâòîçàõîäà.",
		/* LANG_NO_AJOINS */
		"Ñïèñîê àâòîçàõîäà äëÿ âàøåãî íèêà íå íàñòðîåí.",
		/* LANG_AJOIN_LIST_EMPTY */
		"Ñïèñîê àâòîçàõîäà ïóñò.",
		/* LANG_NO_ENTRY */
		"Êàíàë \002%s\002 íå ïðèñóòñòâóåò â ñïèñêå àâòîçàõîäà.",
		/* LANG_CHAN_DELETED */
		"Êàíàë \002%s\002 óñïåøíî óäàëåí èç ñïèñêà àâòîçàõîäà.",
		/* LANG_AJOIN_ENTRY */
		" %d - Êàíàë: %s  -  Êëþ÷: %s",
		/* LANG_AJOIN_ENTRIES */
		"Âñåãî çàïèñåé: %d",
		/* LANG_AJOIN_LIST_CLEARED */
		"Ñïèñîê àâòîçàõîäà ïîëíîñòüþ î÷èùåí.",
		/* LANG_UNKWN_AJOIN_OPTION */
		"Íåèçâåñòíàÿ AJOIN-îïöèÿ.",
		/* LANG_AJOINING */
		"Óñïåøíûé àâòîçàõîä íà %s .",
		/* LANG_AJOINING_FAILED */
		"Íåóäà÷íûé àâòîçàõîä íà %s .",
		/* LANG_SET_AJOIN_DESC */
		" \n"
		"Êîìàíäû, ïðåäîñòàâëÿåìûå ìîäóëåì ns_ajoin: \n"
		"    AJOIN      Ïîçâîëÿåò íàñòðîèòü ðåæèì àâòîçàõîäà.",
		/* LANG_SET_AJOIN_SYNTAX */
		"Ñèíòàêñèñ: SET AJOIN { ON | OFF | SILENT }",
		/* LANG_SET_AJOIN_SYNTAX_EXT */
		"Ñèíòàêñèñ: \002SET AJOIN { ON | OFF | SILENT }\002 \n"
		" \n"
		"Ïîçâîëÿåò âêëþ÷èòü èëè âûêëþ÷èòü àâòîçàõîä íà êàíàëû êàê òîëüêî \n"
		"âû èäåíòèôèöèðóåòåñü ê íèêó èëè èñïîëüçóåòå êîìàíäó UPDATE. \n"
		"Óñòàíîâêà â ON àêòèâèðóåò àâòîìàòè÷åñêîå óâåäîìëåíèå îá àâòîçàõîäå íà \n"
		"êàæäûé êàíàë èç âàøåãî ñïèñêà àâòîçàõîäà. Óñòàíîâêà â SILENT àêòèâèðóåò \n"
		"ëèøü åäèíñòâåííîå óâåäîìëåíèå î òîì, ê êàêîìó êîëè÷åñòâó êàíàëîâ èç âàøåãî \n"
		"ñïèñêà àâòîçàõîäà âû óñïåøíî/íåóñïåøíî ïðèñîåäèíåíû. \n"
		"Óñòàíîâêà îïöèè â OFF îòêëþ÷èò âîçìîæíîñòü àâòîçàõîäà âîîáùå. \n"
		" \n"
		"Óñòàíîâêà ïî-óìîë÷àíèþ: \002ON\002.",
		/* LANG_SET_AJOIN_ON */
		"Ðåæèì àâòîçàõîäà \002ÂÊËÞ×ÅÍ\002.",
		/* LANG_SET_AJOIN_SILENT */
		"Ðåæèì àâòîçàõîäà \002ÂÊËÞ×ÅÍ\002, áóäåò èñïîëüçîâàí \002SILENT\002-ðåæèì (ìèíèìóì óâåäîìëåíèé).",
		/* LANG_SET_AJOIN_OFF */
		"Ðåæèì àâòîçàõîäà \002ÂÛÊËÞ×ÅÍ\002.",
		/* LANG_AJOINING_SUM_SUCCESS */
		"Óñïåøíûé àâòîçàõîä íà %d êàíàë(à/îâ).",
		/* LANG_AJOINING_SUM_FAILED */
		"Íåóäà÷íûé àâòîçàõîä íà %d êàíàë(à/îâ).",
		/* LANG_AJOIN_DELETED_NR_1 */
		"Çàïèñü ïîä íîìåðîì %d óäàëåíà èç ñïèñêà àâòîçàõîäà.",
		/* LANG_AJOIN_DELETED_NR */
		"Èç ñïèñêà àâòîçàõîäà óñïåøíî óäàëåíû %d çàïèñåé.",
	};
	moduleInsertLanguage(LANG_EN_US, LANG_NUM_STRINGS, langtable_en_us);
	moduleInsertLanguage(LANG_NL, LANG_NUM_STRINGS, langtable_nl);
	moduleInsertLanguage(LANG_DE, LANG_NUM_STRINGS, langtable_de);
	moduleInsertLanguage(LANG_TR, LANG_NUM_STRINGS, langtable_tr);
	moduleInsertLanguage(LANG_RU, LANG_NUM_STRINGS, langtable_ru);
}
/* EOF */
 
 
Tidak ada komentar:
Posting Komentar