Select Git revision
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));
}