Skip to content
Snippets Groups Projects
Select Git revision
  • dailybuild_linux-x64
  • dailybuild_win32
  • master default protected
  • sqlite
  • rip_abstraction
  • dailybuild_macos-armv8
  • dd_file_lister_filanem_in_desc_color
  • mode7
  • dd_msg_reader_are_you_there_warning_improvement
  • c23-playing
  • syncterm-1.3
  • syncterm-1.2
  • test-build
  • hide_remote_connection_with_telgate
  • 638-can-t-control-c-during-a-file-search
  • add_body_to_pager_email
  • mingw32-build
  • cryptlib-3.4.7
  • ree/mastermind
  • new_user_dat
  • sbbs320d
  • syncterm-1.6
  • syncterm-1.5
  • syncterm-1.4
  • sbbs320b
  • syncterm-1.3
  • syncterm-1.2
  • syncterm-1.2rc6
  • syncterm-1.2rc5
  • push
  • syncterm-1.2rc4
  • syncterm-1.2rc2
  • syncterm-1.2rc1
  • sbbs319b
  • sbbs318b
  • goodbuild_linux-x64_Sep-01-2020
  • goodbuild_win32_Sep-01-2020
  • goodbuild_linux-x64_Aug-31-2020
  • goodbuild_win32_Aug-31-2020
  • goodbuild_win32_Aug-30-2020
40 results

qwk.cpp

Blame
  • qwk.cpp 27.32 KiB
    /* qwk.cpp */
    
    /* Synchronet QWK packet-related functions */
    
    /* $Id$ */
    
    /****************************************************************************
     * @format.tab-size 4		(Plain Text/Source Code File Header)			*
     * @format.use-tabs true	(see http://www.synchro.net/ptsc_hdr.html)		*
     *																			*
     * Copyright 2000 Rob Swindell - http://www.synchro.net/copyright.html		*
     *																			*
     * 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.					*
     * See the GNU General Public License for more details: gpl.txt or			*
     * http://www.fsf.org/copyleft/gpl.html										*
     *																			*
     * Anonymous FTP access to the most recent released source is available at	*
     * ftp://vert.synchro.net, ftp://cvs.synchro.net and ftp://ftp.synchro.net	*
     *																			*
     * Anonymous CVS access to the development source and modification history	*
     * is available at cvs.synchro.net:/cvsroot/sbbs, example:					*
     * cvs -d :pserver:anonymous@cvs.synchro.net:/cvsroot/sbbs login			*
     *     (just hit return, no password is necessary)							*
     * cvs -d :pserver:anonymous@cvs.synchro.net:/cvsroot/sbbs checkout src		*
     *																			*
     * For Synchronet coding style and modification guidelines, see				*
     * http://www.synchro.net/source.html										*
     *																			*
     * You are encouraged to submit any modifications (preferably in Unix diff	*
     * format) via e-mail to mods@synchro.net									*
     *																			*
     * Note: If this box doesn't appear square, then you need to fix your tabs.	*
     ****************************************************************************/
    
    #include "sbbs.h"
    #include "post.h"
    #include "qwk.h"
    
    /****************************************************************************/
    /* Converts a long to an msbin real number. required for QWK NDX file		*/
    /****************************************************************************/
    float ltomsbin(long val)
    {
    	converter t;
    	int   sign, exp;	/* sign and exponent */
    
    	t.f[0]=(float)val;
    	sign=t.uc[3]/0x80;
    	exp=((t.ui[1]>>7)-0x7f+0x81)&0xff;
    	t.ui[1]=(t.ui[1]&0x7f)|(sign<<7)|(exp<<8);
    	return(t.f[0]);
    }
    
    bool route_circ(char *via, char *id)
    {
    	char str[256],*p,*sp;
    
    	strcpy(str,via);
    	p=str;
    	while(*p && *p<=SP)
    		p++;
    	while(*p) {
    		sp=strchr(p,'/');
    		if(sp) *sp=0;
    		if(!stricmp(p,id))
    			return(true);
    		if(!sp)
    			break;
    		p=sp+1; }
    	return(false);
    }
    
    int sbbs_t::qwk_route(char *inaddr, char *fulladdr)
    {
    	char node[10],str[256],*p;
    	int file,i;
    	FILE *stream;
    
    	fulladdr[0]=0;
    	sprintf(str,"%.255s",inaddr);
    	p=strrchr(str,'/');
    	if(p) p++;
    	else p=str;
    	sprintf(node,"%.8s",p);                 /* node = destination node */
    	truncsp(node);
    
    	for(i=0;i<cfg.total_qhubs;i++)				/* Check if destination is our hub */
    		if(!stricmp(cfg.qhub[i]->id,node))
    			break;
    	if(i<cfg.total_qhubs) {
    		strcpy(fulladdr,node);
    		return(0); }
    
    	i=matchuser(&cfg,node);						/* Check if destination is a node */
    	if(i) {
    		getuserrec(&cfg,i,U_REST,8,str);
    		if(ahtoul(str)&FLAG('Q')) {
    			strcpy(fulladdr,node);
    			return(i); } }
    
    	sprintf(node,"%.8s",inaddr);            /* node = next hop */
    	p=strchr(node,'/');
    	if(p) *p=0;
    	truncsp(node);							
    
    	if(strchr(inaddr,'/')) {                /* Multiple hops */
    
    		for(i=0;i<cfg.total_qhubs;i++)			/* Check if next hop is our hub */
    			if(!stricmp(cfg.qhub[i]->id,node))
    				break;
    		if(i<cfg.total_qhubs) {
    			strcpy(fulladdr,inaddr);
    			return(0); }
    
    		i=matchuser(&cfg,node);					/* Check if next hop is a node */
    		if(i) {
    			getuserrec(&cfg,i,U_REST,8,str);
    			if(ahtoul(str)&FLAG('Q')) {
    				strcpy(fulladdr,inaddr);
    				return(i); } } }
    
    	p=strchr(node,SP);
    	if(p) *p=0;
    
    	sprintf(str,"%sqnet/route.dat",cfg.data_dir);
    	if((stream=fnopen(&file,str,O_RDONLY))==NULL)
    		return(0);
    
    	strcat(node,":");
    	fulladdr[0]=0;
    	while(!feof(stream)) {
    		if(!fgets(str,256,stream))
    			break;
    		if(!strnicmp(str+9,node,strlen(node))) {
    			fclose(stream);
    			truncsp(str);
    			sprintf(fulladdr,"%s/%s",str+9+strlen(node),inaddr);
    			break; } }
    
    	fclose(stream);
    	if(!fulladdr[0])			/* First hop not found in ROUTE.DAT */
    		return(0);
    
    	sprintf(node,"%.8s",fulladdr);
    	p=strchr(node,'/');
    	if(p) *p=0;
    	truncsp(node);
    
    	for(i=0;i<cfg.total_qhubs;i++)				/* Check if first hop is our hub */
    		if(!stricmp(cfg.qhub[i]->id,node))
    			break;
    	if(i<cfg.total_qhubs)
    		return(0);
    
    	i=matchuser(&cfg,node);						/* Check if first hop is a node */
    	if(i) {
    		getuserrec(&cfg,i,U_REST,8,str);
    		if(ahtoul(str)&FLAG('Q'))
    			return(i); }
    	fulladdr[0]=0;
    	return(0);
    }
    
    
    /* Via is in format: NODE/NODE/... */
    void sbbs_t::update_qwkroute(char *via)
    {
    	static uint total_nodes;
    	static char **qwk_node;
    	static char **qwk_path;
    	static time_t *qwk_time;
    	char str[256],*p,*tp,node[9];
    	uint	i;
    	int		file;
    	time_t	t;
    	FILE *	stream;
    
    	if(via==NULL) {
    		if(!total_nodes)
    			return;
    		sprintf(str,"%sqnet/route.dat",cfg.data_dir);
    		if((stream=fnopen(&file,str,O_WRONLY|O_CREAT|O_TRUNC))!=NULL) {
    			t=time(NULL);
    			t-=(90L*24L*60L*60L);
    			for(i=0;i<total_nodes;i++)
    				if(qwk_time[i]>t)
    					fprintf(stream,"%s %s:%s\r\n"
    						,unixtodstr(&cfg,qwk_time[i],str),qwk_node[i],qwk_path[i]);
    			fclose(stream); }
    		else
    			errormsg(WHERE,ERR_OPEN,str,O_WRONLY|O_CREAT|O_TRUNC);
    		for(i=0;i<total_nodes;i++) {
    			FREE(qwk_node[i]);
    			FREE(qwk_path[i]); }
    		if(qwk_node) {
    			FREE(qwk_node);
    			qwk_node=NULL; }
    		if(qwk_path) {
    			FREE(qwk_path);
    			qwk_path=NULL; }
    		if(qwk_time) {
    			FREE(qwk_time);
    			qwk_time=NULL; }
    		total_nodes=0;
    		return; }
    
    	if(!total_nodes) {
    		sprintf(str,"%sqnet/route.dat",cfg.data_dir);
    		if((stream=fnopen(&file,str,O_RDONLY))!=NULL) {
    			while(!feof(stream)) {
    				if(!fgets(str,255,stream))
    					break;
    				truncsp(str);
    				t=dstrtounix(&cfg,str);
    				p=strchr(str,':');
    				if(!p) continue;
    				*p=0;
    				sprintf(node,"%.8s",str+9);
    				tp=strchr(node,SP); 		/* change "node bbs:" to "node:" */
    				if(tp) *tp=0;
    				for(i=0;i<total_nodes;i++)
    					if(!stricmp(qwk_node[i],node))
    						break;
    				if(i<total_nodes && qwk_time[i]>t)
    					continue;
    				if(i==total_nodes) {
    					if((qwk_node=(char **)REALLOC(qwk_node,sizeof(char *)*(i+1)))==NULL) {
    						errormsg(WHERE,ERR_ALLOC,str,9*(i+1));
    						break; }
    					if((qwk_path=(char **)REALLOC(qwk_path,sizeof(char *)*(i+1)))==NULL) {
    						errormsg(WHERE,ERR_ALLOC,str,128*(i+1));
    						break; }
    					if((qwk_time=(time_t *)REALLOC(qwk_time,sizeof(time_t)*(i+1)))==NULL) {
    						errormsg(WHERE,ERR_ALLOC,str,sizeof(time_t)*(i+1));
    						break; }
    					if((qwk_node[i]=(char *)MALLOC(9))==NULL) {
    						errormsg(WHERE,ERR_ALLOC,str,9);
    						break; }
    					if((qwk_path[i]=(char *)MALLOC(128))==NULL) {
    						errormsg(WHERE,ERR_ALLOC,str,128);
    						break; }
    					total_nodes++; }
    				strcpy(qwk_node[i],node);
    				p++;
    				while(*p && *p<=SP) p++;
    				sprintf(qwk_path[i],"%.127s",p);
    				qwk_time[i]=t; }
    			fclose(stream); } }
    
    	strupr(via);
    	p=strchr(via,'/');   /* Skip uplink */
    
    	while(p && *p) {
    		p++;
    		sprintf(node,"%.8s",p);
    		tp=strchr(node,'/');
    		if(tp) *tp=0;
    		tp=strchr(node,SP); 		/* no spaces allowed */
    		if(tp) *tp=0;
    		truncsp(node);
    		for(i=0;i<total_nodes;i++)
    			if(!stricmp(qwk_node[i],node))
    				break;
    		if(i==total_nodes) {		/* Not in list */
    			if((qwk_node=(char **)REALLOC(qwk_node,sizeof(char *)*(total_nodes+1)))==NULL) {
    				errormsg(WHERE,ERR_ALLOC,str,9*(total_nodes+1));
    				break; }
    			if((qwk_path=(char **)REALLOC(qwk_path,sizeof(char *)*(total_nodes+1)))==NULL) {
    				errormsg(WHERE,ERR_ALLOC,str,128*(total_nodes+1));
    				break; }
    			if((qwk_time=(time_t *)REALLOC(qwk_time,sizeof(time_t)*(total_nodes+1)))
    				==NULL) {
    				errormsg(WHERE,ERR_ALLOC,str,sizeof(time_t)*(total_nodes+1));
    				break; }
    			if((qwk_node[total_nodes]=(char *)MALLOC(9))==NULL) {
    				errormsg(WHERE,ERR_ALLOC,str,9);
    				break; }
    			if((qwk_path[total_nodes]=(char *)MALLOC(128))==NULL) {
    				errormsg(WHERE,ERR_ALLOC,str,128);
    				break; }
    			total_nodes++; }
    		sprintf(qwk_node[i],"%.8s",node);
    		sprintf(qwk_path[i],"%.*s",(uint)((p-1)-via),via);
    		qwk_time[i]=time(NULL);
    		p=strchr(p,'/'); }
    	}
    
    /****************************************************************************/
    /* Successful download of QWK packet										*/
    /****************************************************************************/
    void sbbs_t::qwk_success(ulong msgcnt, char bi, char prepack)
    {
    	char	str[128];
    	int 	i;
    	ulong	l,msgs,deleted=0;
    	mail_t	*mail;
    	smbmsg_t msg;
    
    	if(useron.rest&FLAG('Q')) {	// Was if(!prepack) only
    		sprintf(str,"%sqnet/%.8s.out/",cfg.data_dir,useron.alias);
    		strlwr(str);
    		delfiles(str,ALLFILES); 
    	}
    
    	if(!prepack) {
    		sprintf(str,"%s downloaded QWK packet",useron.alias);
    		logline("D-",str);
    		posts_read+=msgcnt;
    
    		sprintf(str,"%sfile/%04u.qwk",cfg.data_dir,useron.number);
    		remove(str);
    
    		if(!bi) {
    			batch_download(-1);
    			delfiles(cfg.temp_dir,ALLFILES); } }
    
    	if(useron.rest&FLAG('Q'))
    		useron.qwk|=(QWK_EMAIL|QWK_ALLMAIL|QWK_DELMAIL);
    	if(useron.qwk&(QWK_EMAIL|QWK_ALLMAIL)) {
    		sprintf(smb.file,"%smail",cfg.data_dir);
    		smb.retry_time=cfg.smb_retry_time;
    		if((i=smb_open(&smb))!=0) {
    			errormsg(WHERE,ERR_OPEN,smb.file,i,smb.last_error);
    			return; }
    
    		mail=loadmail(&smb,&msgs,useron.number,0
    			,useron.qwk&QWK_ALLMAIL ? 0 : LM_UNREAD);
    
    		if((i=smb_locksmbhdr(&smb))!=0) {			  /* Lock the base, so nobody */
    			if(msgs)
    				FREE(mail);
    			smb_close(&smb);
    			errormsg(WHERE,ERR_LOCK,smb.file,i);	/* messes with the index */
    			return; }
    
    		if((i=smb_getstatus(&smb))!=0) {
    			if(msgs)
    				FREE(mail);
    			smb_close(&smb);
    			errormsg(WHERE,ERR_READ,smb.file,i);
    			return; }
    
    		/* Mark as READ and DELETE */
    		for(l=0;l<msgs;l++) {
    			if(mail[l].time>qwkmail_time)
    				continue;
    			msg.idx.offset=0;
    			if(!loadmsg(&msg,mail[l].number))
    				continue;
    			if(!(msg.hdr.attr&MSG_READ)) {
    				if(thisnode.status==NODE_INUSE)
    					telluser(&msg);
    				msg.hdr.attr|=MSG_READ;
    				msg.idx.attr=msg.hdr.attr;
    				smb_putmsg(&smb,&msg); }
    			if(!(msg.hdr.attr&MSG_PERMANENT)
    				&& ((msg.hdr.attr&MSG_KILLREAD && msg.hdr.attr&MSG_READ)
    				|| (useron.qwk&QWK_DELMAIL))) {
    				msg.hdr.attr|=MSG_DELETE;
    				msg.idx.attr=msg.hdr.attr;
    				if((i=smb_putmsg(&smb,&msg))!=0)
    					errormsg(WHERE,ERR_WRITE,smb.file,i);
    				else
    					deleted++; }
    			smb_freemsgmem(&msg);
    			smb_unlockmsghdr(&smb,&msg); }
    
    		if(deleted && cfg.sys_misc&SM_DELEMAIL)
    			delmail(useron.number,MAIL_YOUR);
    		smb_close(&smb);
    		if(msgs)
    			FREE(mail); }
    
    }
    
    /****************************************************************************/
    /* QWK mail packet section													*/
    /****************************************************************************/
    void sbbs_t::qwk_sec()
    {
    	char	str[256],tmp2[256],ch,bi=0
    			,*AttemptedToDownloadQWKpacket="Attempted to download QWK packet";
    	char 	tmp[512];
    	int		error;
    	int 	s;
    	uint	i,k;
    	ulong	msgcnt;
    	time_t	*sav_ptr;
    	file_t	fd;
    
    	getusrdirs();
    	fd.dir=cfg.total_dirs;
    	if((sav_ptr=(time_t *)MALLOC(sizeof(time_t)*cfg.total_subs))==NULL) {
    		errormsg(WHERE,ERR_ALLOC,nulstr,sizeof(time_t)*cfg.total_subs);
    		return; }
    	for(i=0;i<cfg.total_subs;i++)
    		sav_ptr[i]=sub_ptr[i];
    	for(i=0;i<cfg.total_prots;i++)
    		if(cfg.prot[i]->bicmd[0] && chk_ar(cfg.prot[i]->ar,&useron))
    			bi++;				/* number of bidirectional protocols configured */
    	if(useron.rest&FLAG('Q'))
    		getusrsubs();
    	delfiles(cfg.temp_dir,ALLFILES);
    	while(online) {
    		if((useron.misc&(WIP|RIP) || !(useron.misc&EXPERT))
    			&& (useron.logons<2 || !(useron.rest&FLAG('Q'))))
    			menu("qwk");
    		action=NODE_TQWK;
    		ASYNC;
    		bputs(text[QWKPrompt]);
    		strcpy(str,"?UDCSPQ\r");
    		if(bi)
    			strcat(str,"B");
    		ch=(char)getkeys(str,0);
    		if(ch>SP)
    			logch(ch,0);
    		if(sys_status&SS_ABORT || ch=='Q' || ch==CR)
    			break;
    		if(ch=='?') {
    			if((useron.misc&(WIP|RIP) || !(useron.misc&EXPERT))
    				&& !(useron.rest&FLAG('Q')))
    				continue;
    			menu("qwk");
    			continue; }
    		if(ch=='S') {
    			new_scan_cfg(SUB_CFG_NSCAN);
    			delfiles(cfg.temp_dir,ALLFILES);
    			continue; }
    		if(ch=='P') {
    			new_scan_ptr_cfg();
    			for(i=0;i<cfg.total_subs;i++)
    				sav_ptr[i]=sub_ptr[i];
    			delfiles(cfg.temp_dir,ALLFILES);
    			continue; }
    		if(ch=='C') {
    			while(online) {
    				CLS;
    				bputs("\1n\1gQWK Settings:\1n\r\n\r\n");
    				bprintf("\1hA\1n) %-30s: \1h%s\r\n"
    					,"Ctrl-A Color Codes"
    					,useron.qwk&QWK_EXPCTLA
    					? "Expand to ANSI" : useron.qwk&QWK_RETCTLA ? "Leave in"
    					: "Strip");
    				bprintf("\1hT\1n) %-30s: \1h%s\r\n"
    					,"Archive Type"
    					,useron.tmpext);
    				bprintf("\1hE\1n) %-30s: \1h%s\r\n"
    					,"Include E-mail Messages"
    					,useron.qwk&QWK_EMAIL ? "Un-read Only"
    					: useron.qwk&QWK_ALLMAIL ? text[Yes] : text[No]);
    				bprintf("\1hI\1n) %-30s: \1h%s\r\n"
    					,"Include File Attachments"
    					,useron.qwk&QWK_ATTACH ? text[Yes] : text[No]);
    				bprintf("\1hD\1n) %-30s: \1h%s\r\n"
    					,"Delete E-mail Automatically"
    					,useron.qwk&QWK_DELMAIL ? text[Yes]:text[No]);
    				bprintf("\1hF\1n) %-30s: \1h%s\r\n"
    					,"Include New Files List"
    					,useron.qwk&QWK_FILES ? text[Yes]:text[No]);
    				bprintf("\1hN\1n) %-30s: \1h%s\r\n"
    					,"Include Index Files"
    					,useron.qwk&QWK_NOINDEX ? text[No]:text[Yes]);
    				bprintf("\1hC\1n) %-30s: \1h%s\r\n"
    					,"Include Control Files"
    					,useron.qwk&QWK_NOCTRL ? text[No]:text[Yes]);
    				bprintf("\1hY\1n) %-30s: \1h%s\r\n"
    					,"Include Messages from You"
    					,useron.qwk&QWK_BYSELF ? text[Yes]:text[No]);
    				bprintf("\1hZ\1n) %-30s: \1h%s\r\n"
    					,"Include Time Zone (TZ)"
    					,useron.qwk&QWK_TZ ? text[Yes]:text[No]);
    				bprintf("\1hV\1n) %-30s: \1h%s\1n\r\n"
    					,"Include Message Path (VIA)"
    					,useron.qwk&QWK_VIA ? text[Yes]:text[No]);
    				bputs(text[UserDefaultsWhich]);
    				ch=(char)getkeys("AQEDFIOQTYNCZV",0);
    				if(sys_status&SS_ABORT || !ch || ch=='Q')
    					break;
    				switch(ch) {
    					case 'A':
    						if(!(useron.qwk&(QWK_EXPCTLA|QWK_RETCTLA)))
    							useron.qwk|=QWK_EXPCTLA;
    						else if(useron.qwk&QWK_EXPCTLA) {
    							useron.qwk&=~QWK_EXPCTLA;
    							useron.qwk|=QWK_RETCTLA; }
    						else
    							useron.qwk&=~(QWK_EXPCTLA|QWK_RETCTLA);
    						break;
    					case 'T':
    						for(i=0;i<cfg.total_fcomps;i++)
    							uselect(1,i,"Archive Types",cfg.fcomp[i]->ext,cfg.fcomp[i]->ar);
    						s=uselect(0,0,0,0,0);
    						if(s>=0) {
    							strcpy(useron.tmpext,cfg.fcomp[s]->ext);
    							putuserrec(&cfg,useron.number,U_TMPEXT,3,useron.tmpext); }
    						break;
    					case 'E':
    						if(!(useron.qwk&(QWK_EMAIL|QWK_ALLMAIL)))
    							useron.qwk|=QWK_EMAIL;
    						else if(useron.qwk&QWK_EMAIL) {
    							useron.qwk&=~QWK_EMAIL;
    							useron.qwk|=QWK_ALLMAIL; }
    						else
    							useron.qwk&=~(QWK_EMAIL|QWK_ALLMAIL);
    						break;
    					case 'I':
    						useron.qwk^=QWK_ATTACH;
    						break;
    					case 'D':
    						useron.qwk^=QWK_DELMAIL;
    						break;
    					case 'F':
    						useron.qwk^=QWK_FILES;
    						break;
    					case 'N':   /* NO IDX files */
    						useron.qwk^=QWK_NOINDEX;
    						break;
    					case 'C':
    						useron.qwk^=QWK_NOCTRL;
    						break;
    					case 'Z':
    						useron.qwk^=QWK_TZ;
    						break;
    					case 'V':
    						useron.qwk^=QWK_VIA;
    						break;
    					case 'Y':   /* Yourself */
    						useron.qwk^=QWK_BYSELF;
    						break; }
    				putuserrec(&cfg,useron.number,U_QWK,8,ultoa(useron.qwk,str,16)); }
    			delfiles(cfg.temp_dir,ALLFILES);
    			continue; }
    
    
    		if(ch=='B') {   /* Bidirectional QWK and REP packet transfer */
    			sprintf(str,"%s%s.qwk",cfg.temp_dir,cfg.sys_id);
    			if(!fexist(str) && !pack_qwk(str,&msgcnt,0)) {
    				for(i=0;i<cfg.total_subs;i++)
    					sub_ptr[i]=sav_ptr[i];
    				remove(str);
    				last_ns_time=ns_time;
    				continue; }
    			bprintf(text[UploadingREP],cfg.sys_id);
    			menu("biprot");
    			mnemonics(text[ProtocolOrQuit]);
    			strcpy(tmp2,"Q");
    			for(i=0;i<cfg.total_prots;i++)
    				if(cfg.prot[i]->bicmd[0] && chk_ar(cfg.prot[i]->ar,&useron)) {
    					sprintf(tmp,"%c",cfg.prot[i]->mnemonic);
    					strcat(tmp2,tmp); }
    			ch=(char)getkeys(tmp2,0);
    			if(ch=='Q' || sys_status&SS_ABORT) {
    				for(i=0;i<cfg.total_subs;i++)
    					sub_ptr[i]=sav_ptr[i];   /* re-load saved pointers */
    				last_ns_time=ns_time;
    				continue; }
    			for(i=0;i<cfg.total_prots;i++)
    				if(cfg.prot[i]->bicmd[0] && cfg.prot[i]->mnemonic==ch
    					&& chk_ar(cfg.prot[i]->ar,&useron))
    					break;
    			if(i<cfg.total_prots) {
    				batup_total=1;
    				batup_dir[0]=cfg.total_dirs;
    				sprintf(batup_name[0],"%s.rep",cfg.sys_id);
    				batdn_total=1;
    				batdn_dir[0]=cfg.total_dirs;
    				sprintf(batdn_name[0],"%s.qwk",cfg.sys_id);
    				if(!create_batchdn_lst() || !create_batchup_lst()
    					|| !create_bimodem_pth()) {
    					batup_total=batdn_total=0;
    					continue; }
    				sprintf(str,"%s%s.qwk",cfg.temp_dir,cfg.sys_id);
    				sprintf(tmp2,"%s.qwk",cfg.sys_id);
    				padfname(tmp2,fd.name);
    				sprintf(str,"%sBATCHDN.LST",cfg.node_dir);
    				sprintf(tmp2,"%sBATCHUP.LST",cfg.node_dir);
    				error=protocol(cmdstr(cfg.prot[i]->bicmd,str,tmp2,NULL),true);
    				batdn_total=batup_total=0;
    				if(cfg.prot[i]->misc&PROT_DSZLOG) {
    					if(!checkprotlog(&fd)) {
    						logline("D!",AttemptedToDownloadQWKpacket);
    						last_ns_time=ns_time;
    						for(i=0;i<cfg.total_subs;i++)
    							sub_ptr[i]=sav_ptr[i]; } /* re-load saved pointers */
    					else {
    						qwk_success(msgcnt,1,0);
    						for(i=0;i<cfg.total_subs;i++)
    							sav_ptr[i]=sub_ptr[i]; } }
    				else if(error) {
    					logline("D!",AttemptedToDownloadQWKpacket);
    					last_ns_time=ns_time;
    					for(i=0;i<cfg.total_subs;i++)
    						sub_ptr[i]=sav_ptr[i]; }
    				else {
    					qwk_success(msgcnt,1,0);
    					for(i=0;i<cfg.total_subs;i++)
    						sav_ptr[i]=sub_ptr[i]; }
    				sprintf(str,"%s%s.qwk",cfg.temp_dir,cfg.sys_id);
    				remove(str);
    				unpack_rep();
    				delfiles(cfg.temp_dir,ALLFILES);
    				//autohangup();
    				}
    			else {
    				last_ns_time=ns_time;
    				for(i=0;i<cfg.total_subs;i++)
    					sub_ptr[i]=sav_ptr[i]; } }
    
    		else if(ch=='D') {   /* Download QWK Packet of new messages */
    			sprintf(str,"%s%s.qwk",cfg.temp_dir,cfg.sys_id);
    			if(!fexist(str) && !pack_qwk(str,&msgcnt,0)) {
    				for(i=0;i<cfg.total_subs;i++)
    					sub_ptr[i]=sav_ptr[i];
    				last_ns_time=ns_time;
    				remove(str);
    				continue; }
    			if(online==ON_LOCAL) {			/* Local QWK packet creation */
    				bputs(text[EnterPath]);
    				if(!getstr(str,60,K_LINE|K_UPPER)) {
    					for(i=0;i<cfg.total_subs;i++)
    						sub_ptr[i]=sav_ptr[i];   /* re-load saved pointers */
    					last_ns_time=ns_time;
    					continue; }
    				backslashcolon(str);
    				sprintf(tmp2,"%s%s.qwk",str,cfg.sys_id);
    				if(fexist(tmp2)) {
    					for(i=0;i<10;i++) {
    						sprintf(tmp2,"%s%s.QW%d",str,cfg.sys_id,i);
    						if(!fexist(tmp2))
    							break; }
    					if(i==10) {
    						bputs(text[FileAlreadyThere]);
    						last_ns_time=ns_time;
    						for(i=0;i<cfg.total_subs;i++)
    							sub_ptr[i]=sav_ptr[i];
    						continue; } }
    				sprintf(tmp,"%s%s.qwk",cfg.temp_dir,cfg.sys_id);
    				if(mv(tmp,tmp2,0)) { /* unsuccessful */
    					for(i=0;i<cfg.total_subs;i++)
    						sub_ptr[i]=sav_ptr[i];
    					last_ns_time=ns_time; }
    				else {
    					bprintf(text[FileNBytesSent],tmp2,ultoac(flength(tmp2),tmp));
    					qwk_success(msgcnt,0,0);
    					for(i=0;i<cfg.total_subs;i++)
    						sav_ptr[i]=sub_ptr[i]; }
    				continue; }
    
    			/***************/
    			/* Send Packet */
    			/***************/
    			menu("dlprot");
    			mnemonics(text[ProtocolOrQuit]);
    			strcpy(tmp2,"Q");
    			for(i=0;i<cfg.total_prots;i++)
    				if(cfg.prot[i]->dlcmd[0] && chk_ar(cfg.prot[i]->ar,&useron)) {
    					sprintf(tmp,"%c",cfg.prot[i]->mnemonic);
    					strcat(tmp2,tmp); }
    			ungetkey(useron.prot);
    			ch=(char)getkeys(tmp2,0);
    			if(ch=='Q' || sys_status&SS_ABORT) {
    				for(i=0;i<cfg.total_subs;i++)
    					sub_ptr[i]=sav_ptr[i];   /* re-load saved pointers */
    				last_ns_time=ns_time;
    				continue; }
    			for(i=0;i<cfg.total_prots;i++)
    				if(cfg.prot[i]->dlcmd[0] && cfg.prot[i]->mnemonic==ch
    					&& chk_ar(cfg.prot[i]->ar,&useron))
    					break;
    			if(i<cfg.total_prots) {
    				sprintf(str,"%s%s.qwk",cfg.temp_dir,cfg.sys_id);
    				sprintf(tmp2,"%s.qwk",cfg.sys_id);
    				padfname(tmp2,fd.name);
    				error=protocol(cmdstr(cfg.prot[i]->dlcmd,str,nulstr,NULL),false);
    				if(cfg.prot[i]->misc&PROT_DSZLOG) {
    					if(!checkprotlog(&fd)) {
    						last_ns_time=ns_time;
    						for(i=0;i<cfg.total_subs;i++)
    							sub_ptr[i]=sav_ptr[i]; } /* re-load saved pointers */
    					else {
    						qwk_success(msgcnt,0,0);
    						for(i=0;i<cfg.total_subs;i++)
    							sav_ptr[i]=sub_ptr[i]; } }
    				else if(error) {
    					logline("D!",AttemptedToDownloadQWKpacket);
    					last_ns_time=ns_time;
    					for(i=0;i<cfg.total_subs;i++)
    						sub_ptr[i]=sav_ptr[i]; }
    				else {
    					qwk_success(msgcnt,0,0);
    					for(i=0;i<cfg.total_subs;i++)
    						sav_ptr[i]=sub_ptr[i]; }
    				autohangup(); }
    			else {	 /* if not valid protocol (hungup?) */
    				for(i=0;i<cfg.total_subs;i++)
    					sub_ptr[i]=sav_ptr[i];
    				last_ns_time=ns_time; } }
    
    		else if(ch=='U') { /* Upload REP Packet */
    	/*
    			if(useron.rest&FLAG('Q') && useron.rest&FLAG('P')) {
    				bputs(text[R_Post]);
    				continue; }
    	*/
    
    			delfiles(cfg.temp_dir,ALLFILES);
    			bprintf(text[UploadingREP],cfg.sys_id);
    			for(k=0;k<cfg.total_fextrs;k++)
    				if(!stricmp(cfg.fextr[k]->ext,useron.tmpext)
    					&& chk_ar(cfg.fextr[k]->ar,&useron))
    					break;
    			if(k>=cfg.total_fextrs) {
    				bputs(text[QWKExtractionFailed]);
    				errorlog("Couldn't extract REP packet - configuration error");
    				continue; }
    
    			if(online==ON_LOCAL) {		/* Local upload of rep packet */
    				bputs(text[EnterPath]);
    				if(!getstr(str,60,K_LINE|K_UPPER))
    					continue;
    				backslashcolon(str);
    				sprintf(tmp,"%s.rep",cfg.sys_id);
    				strcat(str,tmp);
    				sprintf(tmp,"%s%s.rep",cfg.temp_dir,cfg.sys_id);
    				if(!mv(str,tmp,0))
    					unpack_rep();
    				delfiles(cfg.temp_dir,ALLFILES);
    				continue; }
    
    			/******************/
    			/* Receive Packet */
    			/******************/
    			menu("ulprot");
    			mnemonics(text[ProtocolOrQuit]);
    			strcpy(tmp2,"Q");
    			for(i=0;i<cfg.total_prots;i++)
    				if(cfg.prot[i]->ulcmd[0] && chk_ar(cfg.prot[i]->ar,&useron)) {
    					sprintf(tmp,"%c",cfg.prot[i]->mnemonic);
    					strcat(tmp2,tmp); }
    			ch=(char)getkeys(tmp2,0);
    			if(ch=='Q' || sys_status&SS_ABORT)
    				continue;
    			for(i=0;i<cfg.total_prots;i++)
    				if(cfg.prot[i]->ulcmd[0] && cfg.prot[i]->mnemonic==ch
    					&& chk_ar(cfg.prot[i]->ar,&useron))
    					break;
    			if(i>=cfg.total_prots)	/* This shouldn't happen */
    				continue;
    			sprintf(str,"%s%s.rep",cfg.temp_dir,cfg.sys_id);
    			protocol(cmdstr(cfg.prot[i]->ulcmd,str,nulstr,NULL),true);
    			unpack_rep();
    			delfiles(cfg.temp_dir,ALLFILES);
    			//autohangup();
    			} }
    	delfiles(cfg.temp_dir,ALLFILES);
    	FREE(sav_ptr);
    }
    
    void sbbs_t::qwksetptr(uint subnum, char *buf, int reset)
    {
    	long	l;
    	ulong	last;
    
    	if(buf[2]=='/' && buf[5]=='/') {    /* date specified */
    		l=dstrtounix(&cfg,buf);
    		sub_ptr[subnum]=getmsgnum(subnum,l);
    		return; }
    	l=atol(buf);
    	if(l>=0)							  /* ptr specified */
    		sub_ptr[subnum]=l;
    	else if(l) {						  /* relative ptr specified */
    		getlastmsg(subnum,&last,0);
    		if(-l>(long)last)
    			sub_ptr[subnum]=0;
    		else
    			sub_ptr[subnum]=last+l; }
    	else if(reset)
    		getlastmsg(subnum,&sub_ptr[subnum],0);
    }
    
    
    /****************************************************************************/
    /* Process a QWK Config line												*/
    /****************************************************************************/
    void sbbs_t::qwkcfgline(char *buf,uint subnum)
    {
    	char	str[128];
    	char 	tmp[512];
    	uint 	x,y;
    	long	l;
    	ulong	qwk=useron.qwk;
    	file_t	f;
    
    	sprintf(str,"%.25s",buf);
    	strupr(str);
    	bprintf("\1n\r\n\1b\1hQWK Control [\1c%s\1b]: \1g%s\r\n"
    		,subnum==INVALID_SUB ? "Mail":cfg.sub[subnum]->qwkname,str);
    
    	if(subnum!=INVALID_SUB) {					/* Only valid in sub-boards */
    
    		if(!strncmp(str,"DROP ",5)) {              /* Drop from new-scan */
    			l=atol(str+5);
    			if(!l)
    				sub_cfg[subnum]&=~SUB_CFG_NSCAN;
    			else {
    				x=l/1000;
    				y=l-(x*1000);
    				if(x>=usrgrps || y>=usrsubs[x]) {
    					bprintf(text[QWKInvalidConferenceN],l);
    					sprintf(str,"Invalid conference number %lu",l);
    					logline("Q!",str); }
    				else
    					sub_cfg[usrsub[x][y]]&=~SUB_CFG_NSCAN; }
    			return; }
    
    		if(!strncmp(str,"ADD YOURS ",10)) {               /* Add to new-scan */
    			sub_cfg[subnum]|=(SUB_CFG_NSCAN|SUB_CFG_YSCAN);
    			qwksetptr(subnum,str+10,0);
    			return; }
    
    		else if(!strncmp(str,"YOURS ",6)) {
    			sub_cfg[subnum]|=(SUB_CFG_NSCAN|SUB_CFG_YSCAN);
    			qwksetptr(subnum,str+6,0);
    			return; }
    
    		else if(!strncmp(str,"ADD ",4)) {               /* Add to new-scan */
    			sub_cfg[subnum]|=SUB_CFG_NSCAN;
    			sub_cfg[subnum]&=~SUB_CFG_YSCAN;
    			qwksetptr(subnum,str+4,0);
    			return; }
    
    		if(!strncmp(str,"RESET ",6)) {             /* set msgptr */
    			qwksetptr(subnum,str+6,1);
    			return; }
    
    		if(!strncmp(str,"SUBPTR ",7)) {
    			qwksetptr(subnum,str+7,1);
    			return; }
    		}
    
    	if(!strncmp(str,"RESETALL ",9)) {              /* set all ptrs */
    		for(x=0;x<usrgrps;x++)
    			for(y=0;y<usrsubs[x];y++)
    				if(sub_cfg[usrsub[x][y]]&SUB_CFG_NSCAN)
    					qwksetptr(usrsub[x][y],str+9,1); }
    
    	else if(!strncmp(str,"ALLPTR ",7)) {              /* set all ptrs */
    		for(x=0;x<usrgrps;x++)
    			for(y=0;y<usrsubs[x];y++)
    				if(sub_cfg[usrsub[x][y]]&SUB_CFG_NSCAN)
    					qwksetptr(usrsub[x][y],str+7,1); }
    
    	else if(!strncmp(str,"FILES ",6)) {                 /* files list */
    		if(!strncmp(str+6,"ON ",3))
    			useron.qwk|=QWK_FILES;
    		else if(str[8]=='/' && str[11]=='/') {      /* set scan date */
    			useron.qwk|=QWK_FILES;
    			ns_time=dstrtounix(&cfg,str+6); }
    		else
    			useron.qwk&=~QWK_FILES; }
    
    	else if(!strncmp(str,"OWN ",4)) {                   /* message from you */
    		if(!strncmp(str+4,"ON ",3))
    			useron.qwk|=QWK_BYSELF;
    		else
    			useron.qwk&=~QWK_BYSELF;
    		return; }
    
    	else if(!strncmp(str,"NDX ",4)) {                   /* include indexes */
    		if(!strncmp(str+4,"OFF ",4))
    			useron.qwk|=QWK_NOINDEX;
    		else
    			useron.qwk&=~QWK_NOINDEX; }
    
    	else if(!strncmp(str,"CONTROL ",8)) {               /* exclude ctrl files */
    		if(!strncmp(str+8,"OFF ",4))
    			useron.qwk|=QWK_NOCTRL;
    		else
    			useron.qwk&=~QWK_NOCTRL; }
    
    	else if(!strncmp(str,"VIA ",4)) {                   /* include @VIA: */
    		if(!strncmp(str+4,"ON  ",3))
    			useron.qwk|=QWK_VIA;
    		else
    			useron.qwk&=~QWK_VIA; }
    
    	else if(!strncmp(str,"TZ ",3)) {                    /* include @TZ: */
    		if(!strncmp(str+3,"ON ",3))
    			useron.qwk|=QWK_TZ;
    		else
    			useron.qwk&=~QWK_TZ; }
    
    	else if(!strncmp(str,"ATTACH ",7)) {                /* file attachments */
    		if(!strncmp(str+7,"ON ",3))
    			useron.qwk|=QWK_ATTACH;
    		else
    			useron.qwk&=~QWK_ATTACH; }
    
    	else if(!strncmp(str,"DELMAIL ",8)) {               /* delete mail */
    		if(!strncmp(str+8,"ON ",3))
    			useron.qwk|=QWK_DELMAIL;
    		else
    			useron.qwk&=~QWK_DELMAIL; }
    
    	else if(!strncmp(str,"CTRL-A ",7)) {                /* Ctrl-a codes  */
    		if(!strncmp(str+7,"KEEP ",5)) {
    			useron.qwk|=QWK_RETCTLA;
    			useron.qwk&=~QWK_EXPCTLA; }
    		else if(!strncmp(str+7,"EXPAND ",7)) {
    			useron.qwk|=QWK_EXPCTLA;
    			useron.qwk&=~QWK_RETCTLA; }
    		else
    			useron.qwk&=~(QWK_EXPCTLA|QWK_RETCTLA); }
    
    	else if(!strncmp(str,"MAIL ",5)) {                  /* include e-mail */
    		if(!strncmp(str+5,"ALL ",4)) {
    			useron.qwk|=QWK_ALLMAIL;
    			useron.qwk&=~QWK_EMAIL; }
    		else if(!strncmp(str+5,"ON ",3)) {
    			useron.qwk|=QWK_EMAIL;
    			useron.qwk&=~QWK_ALLMAIL; }
    		else
    			useron.qwk&=~(QWK_ALLMAIL|QWK_EMAIL); }
    
    	else if(!strncmp(str,"FREQ ",5)) {                  /* file request */
    		padfname(str+5,f.name);
    		strupr(f.name);
    		for(x=0;x<usrlibs;x++) {
    			for(y=0;y<usrdirs[x];y++)
    				if(findfile(&cfg,usrdir[x][y],f.name))
    					break;
    			if(y<usrdirs[x])
    				break; }
    		if(x>=usrlibs) {
    			bprintf("\r\n%s",f.name);
    			bputs(text[FileNotFound]); }
    		else {
    			f.dir=usrdir[x][y];
    			getfileixb(&cfg,&f);
    			f.size=0;
    			getfiledat(&cfg,&f);
    			if(f.size==-1L)
    				bprintf(text[FileIsNotOnline],f.name);
    			else
    				addtobatdl(&f); } }
    
    	else {
    		attr(cfg.color[clr_err]);
    		bputs("Unrecognized Control Command!\1n\r\n");
    	}
    
    	if(qwk!=useron.qwk)
    		putuserrec(&cfg,useron.number,U_QWK,8,ultoa(useron.qwk,tmp,16));
    }