1009 lines
27 KiB
C
1009 lines
27 KiB
C
/*****************************************************************************
|
|
*
|
|
* $Id: mgrutil.c,v 1.42 2005/10/11 20:49:47 mbse Exp $
|
|
* Purpose ...............: AreaMgr and FileMgr utilities.
|
|
*
|
|
*****************************************************************************
|
|
* Copyright (C) 1997-2005
|
|
*
|
|
* Michiel Broek FIDO: 2:280/2802
|
|
* Beekmansbos 10
|
|
* 1971 BV IJmuiden
|
|
* the Netherlands
|
|
*
|
|
* This file is part of MBSE BBS.
|
|
*
|
|
* This BBS 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, or (at your option) any
|
|
* later version.
|
|
*
|
|
* MBSE BBS is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with MBSE BBS; see the file COPYING. If not, write to the Free
|
|
* Software Foundation, 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
|
*****************************************************************************/
|
|
|
|
#include "../config.h"
|
|
#include "../lib/mbselib.h"
|
|
#include "../lib/users.h"
|
|
#include "../lib/mbsedb.h"
|
|
#include "../lib/diesel.h"
|
|
#include "sendmail.h"
|
|
#include "rollover.h"
|
|
#include "addpkt.h"
|
|
#include "createm.h"
|
|
#include "createf.h"
|
|
#include "mgrutil.h"
|
|
|
|
|
|
extern int net_out;
|
|
extern int do_quiet;
|
|
extern int do_flush;
|
|
|
|
|
|
void tidy_arealist(AreaList **);
|
|
void fill_arealist(AreaList **, char *, int);
|
|
|
|
|
|
|
|
void MacroRead(FILE *fi, FILE *fp)
|
|
{
|
|
char *line, *temp;
|
|
int res;
|
|
|
|
line = calloc(256, sizeof(char));
|
|
temp = calloc(256, sizeof(char));
|
|
|
|
while ((fgets(line, 254, fi) != NULL) && ((line[0]!='@') || (line[1]!='|'))) {
|
|
/*
|
|
* Skip comment lines
|
|
*/
|
|
if (line[0] != '#') {
|
|
Striplf(line);
|
|
if (strlen(line) == 0) {
|
|
/*
|
|
* Empty lines are just written
|
|
*/
|
|
fprintf(fp, "\r");
|
|
} else {
|
|
strncpy(temp, ParseMacro(line,&res), 254);
|
|
if (res)
|
|
Syslog('!', "Macro error line: \"%s\"", line);
|
|
/*
|
|
* Only output if something was evaluated
|
|
*/
|
|
if (strlen(temp))
|
|
fprintf(fp, "%s\r", temp);
|
|
}
|
|
}
|
|
}
|
|
free(line);
|
|
free(temp);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Write Echomail groups list to tempfile
|
|
*/
|
|
void WriteMailGroups(FILE *fp, faddr *f)
|
|
{
|
|
int Count = 0, First = TRUE;
|
|
char *Group, *temp;
|
|
FILE *gp,*fi;
|
|
faddr *g, *Temp;
|
|
fpos_t fileptr;
|
|
|
|
if ((fi = OpenMacro("areamgr.group", nodes.Language, FALSE)) == NULL)
|
|
return;
|
|
|
|
MacroRead(fi, fp);
|
|
fgetpos(fi,&fileptr);
|
|
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
snprintf(temp, PATH_MAX, "%s/etc/mgroups.data", getenv("MBSE_ROOT"));
|
|
|
|
if ((gp = fopen(temp, "r")) == NULL) {
|
|
WriteError("$Can't open %s", temp);
|
|
free(temp);
|
|
fclose(fi);
|
|
return;
|
|
}
|
|
fread(&mgrouphdr, sizeof(mgrouphdr), 1, gp);
|
|
|
|
while (TRUE) {
|
|
Group = GetNodeMailGrp(First);
|
|
if (Group == NULL)
|
|
break;
|
|
First = FALSE;
|
|
|
|
fseek(gp, mgrouphdr.hdrsize, SEEK_SET);
|
|
while (fread(&mgroup, mgrouphdr.recsize, 1, gp) == 1) {
|
|
Temp = fido2faddr(mgroup.UseAka);
|
|
g = bestaka_s(Temp);
|
|
if ((!strcmp(mgroup.Name, Group)) &&
|
|
(g->zone == f->zone) && (g->net == f->net) && (g->node == f->node) && (g->point == f->point)) {
|
|
MacroVars("gh", "ss", mgroup.Name, mgroup.Comment);
|
|
fsetpos(fi, &fileptr);
|
|
MacroRead(fi, fp);
|
|
Count++;
|
|
break;
|
|
}
|
|
tidy_faddr(Temp);
|
|
}
|
|
}
|
|
|
|
MacroVars("b", "d", Count);
|
|
MacroRead(fi, fp);
|
|
fclose(fi);
|
|
fclose(gp);
|
|
free(temp);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Write ticarea groups to tempfile
|
|
*/
|
|
void WriteFileGroups(FILE *fp, faddr *f)
|
|
{
|
|
int Count = 0, First = TRUE;
|
|
char *Group, *temp;
|
|
FILE *gp, *fi;
|
|
faddr *g, *Temp;
|
|
fpos_t fileptr;
|
|
|
|
if ((fi = OpenMacro("filemgr.group", nodes.Language, FALSE)) == NULL)
|
|
return;
|
|
|
|
MacroRead(fi, fp);
|
|
fgetpos(fi,&fileptr);
|
|
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
snprintf(temp, PATH_MAX, "%s/etc/fgroups.data", getenv("MBSE_ROOT"));
|
|
|
|
if ((gp = fopen(temp, "r")) == NULL) {
|
|
WriteError("$Can't open %s", temp);
|
|
free(temp);
|
|
fclose(fi);
|
|
return;
|
|
}
|
|
fread(&fgrouphdr, sizeof(fgrouphdr), 1, gp);
|
|
|
|
while (TRUE) {
|
|
Group = GetNodeFileGrp(First);
|
|
if (Group == NULL)
|
|
break;
|
|
First = FALSE;
|
|
|
|
fseek(gp, fgrouphdr.hdrsize, SEEK_SET);
|
|
while (fread(&fgroup, fgrouphdr.recsize, 1, gp) == 1) {
|
|
Temp = fido2faddr(fgroup.UseAka);
|
|
g = bestaka_s(Temp);
|
|
if ((!strcmp(fgroup.Name, Group)) &&
|
|
(g->zone == f->zone) && (g->net == f->net) && (g->node == f->node) && (g->point == f->point)) {
|
|
MacroVars("gh", "ss", fgroup.Name, fgroup.Comment );
|
|
fsetpos(fi,&fileptr);
|
|
MacroRead(fi, fp);
|
|
Count++;
|
|
break;
|
|
}
|
|
tidy_faddr(Temp);
|
|
}
|
|
}
|
|
|
|
MacroVars("b", "d", Count );
|
|
MacroRead(fi, fp);
|
|
fclose(fi);
|
|
fclose(gp);
|
|
free(temp);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Shift all characters in Buf Cnt places to left
|
|
*/
|
|
void ShiftBuf(char *Buf, int Cnt)
|
|
{
|
|
int i;
|
|
|
|
for (i = Cnt; i < strlen(Buf); i++)
|
|
Buf[i - Cnt] = Buf[i];
|
|
Buf[i - Cnt] = '\0';
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Remove spaces and = characters from begin of line
|
|
*/
|
|
void CleanBuf(char *Buf)
|
|
{
|
|
while (strlen(Buf) && ((Buf[0] == ' ') || (Buf[0] == '=')))
|
|
ShiftBuf(Buf, 1);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Change AreaMgr and FileMgr password for a node
|
|
*/
|
|
void MgrPasswd(faddr *t, char *Buf, FILE *tmp, int Len, int mgr)
|
|
{
|
|
ShiftBuf(Buf, Len);
|
|
CleanBuf(Buf);
|
|
MacroVars("SsP", "sss", CFG.sysop_name, nodes.Sysop, mgr?(char *)"Filemgr":(char *)"Areamgr");
|
|
if ((strlen(Buf) < 3) || (strlen(Buf) > 15)) {
|
|
MacroVars("RABCDE", "ssssss",(char *)"ERR_PASS_LEN",(char *)"",(char *)"",(char *)"",(char *)"",(char *)"");
|
|
MsgResult(mgr?"filemgr.responses":"areamgr.responses",tmp,'\n');
|
|
Mgrlog("%s: Password length %d, not changed", mgr?(char *)"Filemgr":(char *)"Areamgr", strlen(Buf));
|
|
return;
|
|
}
|
|
|
|
memset(&nodes.Apasswd, 0, sizeof(nodes.Apasswd));
|
|
strncpy(nodes.Apasswd, tu(Buf), 15);
|
|
MacroVars("RABCDE", "ssssss",(char *)"OK_PASS",nodes.Apasswd,(char *)"",(char *)"",(char *)"",(char *)"");
|
|
MsgResult(mgr?"filemgr.responses":"areamgr.responses",tmp,'\n');
|
|
Mgrlog("%s: Password \"%s\" for node %s", mgr?(char *)"Filemgr":(char *)"Areamgr", nodes.Apasswd, ascfnode(t, 0x1f));
|
|
MacroClear();
|
|
UpdateNode();
|
|
SearchNodeFaddr(t);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Change AreaMgr/FileMgr nodify flag for node
|
|
*/
|
|
void MgrNotify(faddr *t, char *Buf, FILE *tmp, int mgr)
|
|
{
|
|
/*
|
|
* First strip leading garbage
|
|
*/
|
|
ShiftBuf(Buf, 7);
|
|
CleanBuf(Buf);
|
|
|
|
if (!strncasecmp(Buf, "on", 2))
|
|
nodes.Notify = TRUE;
|
|
else if (!strncasecmp(Buf, "off", 3))
|
|
nodes.Notify = FALSE;
|
|
else
|
|
return;
|
|
|
|
UpdateNode();
|
|
SearchNodeFaddr(t);
|
|
Mgrlog("%s: Notify %s", mgr?(char *)"Filemgr":(char *)"Areamgr", nodes.Notify?"Yes":"No");
|
|
MacroVars("SsP", "sss", CFG.sysop_name, nodes.Sysop,mgr?(char *)"Filemgr":(char *)"Areamgr");
|
|
MacroVars("RABCDE", "sdssss",(char *)"NOTIFY",nodes.Notify,(char *)"",(char *)"",(char *)"",(char *)"");
|
|
MsgResult(mgr?"filemgr.responses":"areamgr.responses",tmp,'\n');
|
|
MacroClear();
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Create uplink areamgr request. One netmail per request.
|
|
* More is possible, cmd is then: "+area1\r+area2\r-area3"
|
|
* Return values:
|
|
* 0 - Ok
|
|
* 1 - Node not in setup
|
|
* 2 - No Uplink mgr program in setup
|
|
* 3 - No uplink password in setup
|
|
* 4 - Can't add mail to outbound
|
|
*/
|
|
int UplinkRequest(faddr *t, faddr *From, int FileMgr, char *cmd)
|
|
{
|
|
FILE *qp;
|
|
time_t Now;
|
|
struct tm *tm;
|
|
fidoaddr Orig, Dest;
|
|
unsigned flags = M_PVT;
|
|
char ext[4], *mgrname, *subj, cmdline[81];
|
|
int i, j;
|
|
|
|
memset(&Orig, 0, sizeof(Orig));
|
|
Orig.zone = From->zone;
|
|
Orig.net = From->net;
|
|
Orig.node = From->node;
|
|
Orig.point = From->point;
|
|
snprintf(Orig.domain, 13, "%s", From->domain);
|
|
|
|
memset(&Dest, 0, sizeof(Dest));
|
|
Dest.zone = t->zone;
|
|
Dest.net = t->net;
|
|
Dest.node = t->node;
|
|
Dest.point = t->point;
|
|
snprintf(Dest.domain, 13, "%s", t->domain);
|
|
|
|
if (!SearchNode(Dest)) {
|
|
Syslog('+', "Can't find node %s in setup", aka2str(Dest));
|
|
return 1;
|
|
}
|
|
|
|
if (FileMgr) {
|
|
if (strlen(nodes.UplFmgrPgm) == 0) {
|
|
Syslog('!', "FileMgr program not defined in setup of node %s", aka2str(Dest));
|
|
return 2;
|
|
}
|
|
mgrname = xstrcpy(nodes.UplFmgrPgm);
|
|
if (strlen(nodes.UplFmgrPass) == 0) {
|
|
Syslog('!', "No FileMgr password set for node %s", aka2str(Dest));
|
|
return 3;
|
|
}
|
|
subj = xstrcpy(nodes.UplFmgrPass);
|
|
} else {
|
|
if (strlen(nodes.UplAmgrPgm) == 0) {
|
|
Syslog('!', "AreaMgr program not defined in setup of node %s", aka2str(Dest));
|
|
return 2;
|
|
}
|
|
mgrname = xstrcpy(nodes.UplAmgrPgm);
|
|
if (strlen(nodes.UplAmgrPass) == 0) {
|
|
Syslog('!', "No AreaMgr password set for node %s", aka2str(Dest));
|
|
return 3;
|
|
}
|
|
subj = xstrcpy(nodes.UplAmgrPass);
|
|
}
|
|
|
|
Mgrlog("Sending uplink request from %s to \"%s\" at %s", aka2str(Orig), mgrname, ascfnode(t, 0x1f));
|
|
|
|
Now = time(NULL) - (gmt_offset((time_t)0) * 60);
|
|
flags |= (nodes.Crash) ? M_CRASH : 0;
|
|
flags |= (nodes.Hold) ? M_HOLD : 0;
|
|
|
|
/*
|
|
* Increase counters, update record and reload.
|
|
*/
|
|
StatAdd(&nodes.MailSent, 1L);
|
|
UpdateNode();
|
|
SearchNode(Dest);
|
|
|
|
memset(&ext, 0, sizeof(ext));
|
|
if (nodes.PackNetmail)
|
|
snprintf(ext, 4, (char *)"qqq");
|
|
else if (nodes.Crash)
|
|
snprintf(ext, 4, (char *)"ccc");
|
|
else if (nodes.Hold)
|
|
snprintf(ext, 4, (char *)"hhh");
|
|
else
|
|
snprintf(ext, 4, (char *)"nnn");
|
|
|
|
if ((qp = OpenPkt(Orig, Dest, (char *)ext)) == NULL)
|
|
return 4;
|
|
|
|
if (AddMsgHdr(qp, From, t, flags, 0, Now, mgrname, CFG.sysop_name, subj)) {
|
|
fclose(qp);
|
|
return 4;
|
|
}
|
|
|
|
if (Dest.point)
|
|
fprintf(qp, "\001TOPT %d\r", Dest.point);
|
|
if (Orig.point)
|
|
fprintf(qp, "\001FMPT %d\r", Orig.point);
|
|
|
|
fprintf(qp, "\001INTL %d:%d/%d %d:%d/%d\r", Dest.zone, Dest.net, Dest.node, Orig.zone, Orig.net, Orig.node);
|
|
|
|
/*
|
|
* Add MSGID, REPLY and PID
|
|
*/
|
|
fprintf(qp, "\001MSGID: %s %08x\r", aka2str(Orig), sequencer());
|
|
fprintf(qp, "\001PID: MBSE-FIDO %s (%s-%s)\r", VERSION, OsName(), OsCPU());
|
|
fprintf(qp, "\001TZUTC: %s\r", gmtoffset(Now));
|
|
|
|
/*
|
|
* Send command, may be format *AREA1\r+AREA2\r=AREA3
|
|
*/
|
|
j = 0;
|
|
for (i = 0; i < strlen(cmd); i++) {
|
|
putc(cmd[i], qp);
|
|
if (cmd[i] == '\r') {
|
|
cmdline[j] = '\0';
|
|
Mgrlog("%s", cmdline);
|
|
j = 0;
|
|
} else {
|
|
cmdline[j] = cmd[i];
|
|
j++;
|
|
}
|
|
}
|
|
putc('\r', qp); /* There is no return after the last one. */
|
|
cmdline[j] = '\0';
|
|
Mgrlog("%s", cmdline);
|
|
Mgrlog("---");
|
|
fprintf(qp, TearLine());
|
|
|
|
/*
|
|
* Add a warning after the tearline.
|
|
*/
|
|
fprintf(qp, "Please note, this is an automatic created message\r");
|
|
|
|
tm = gmtime(&Now);
|
|
fprintf(qp, "\001Via %s @%d%02d%02d.%02d%02d%02d.UTC mbfido %s\r",
|
|
ascfnode(bestaka_s(t), 0x1f), tm->tm_year+1900, tm->tm_mon+1, tm->tm_mday,
|
|
tm->tm_hour, tm->tm_min, tm->tm_sec, VERSION);
|
|
|
|
putc(0, qp);
|
|
fclose(qp);
|
|
|
|
free(mgrname);
|
|
free(subj);
|
|
net_out++;
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
void GetRpSubject(const char *report, char* subject, size_t size)
|
|
{
|
|
FILE *fi;
|
|
char *temp;
|
|
int res;
|
|
|
|
temp = calloc(256,sizeof(char));
|
|
if ((fi=OpenMacro(report, nodes.Language, FALSE))!=NULL){
|
|
while ( fgets(temp, 254, fi) != NULL )
|
|
if (temp[0] != '#')
|
|
ParseMacro(temp,&res);
|
|
fclose(fi);
|
|
}
|
|
|
|
res=diesel((char *)"@(getvar,subject)",temp);
|
|
|
|
if(res==0)
|
|
snprintf(subject,size,"%s",temp);
|
|
free(temp);
|
|
}
|
|
|
|
|
|
|
|
int MsgResult(const char * report, FILE *fo, char eol)
|
|
{
|
|
FILE *fi;
|
|
char *temp, *resp;
|
|
int res;
|
|
|
|
temp = calloc(256,sizeof(char));
|
|
resp = calloc(256,sizeof(char));
|
|
|
|
if ((fi = OpenMacro(report, nodes.Language, FALSE)) != NULL){
|
|
while ( fgets(temp, 254, fi) != NULL ){
|
|
if (temp[0] != '#') {
|
|
strncpy(resp, ParseMacro(temp, &res), 80);
|
|
if ((res == 0) && strlen(resp))
|
|
fprintf(fo,"%s%c",ParseMacro(temp,&res), eol);
|
|
}
|
|
}
|
|
fclose(fi);
|
|
res=1;
|
|
} else {
|
|
res = 0;
|
|
}
|
|
|
|
free(resp);
|
|
free(temp);
|
|
return res;
|
|
}
|
|
|
|
|
|
|
|
void tidy_arealist(AreaList **fdp)
|
|
{
|
|
AreaList *tmp, *old;
|
|
|
|
for (tmp = *fdp; tmp; tmp = old) {
|
|
old = tmp->next;
|
|
free(tmp);
|
|
}
|
|
*fdp = NULL;
|
|
}
|
|
|
|
|
|
|
|
void fill_arealist(AreaList **fdp, char *tag, int DoDelete)
|
|
{
|
|
AreaList **tmp;
|
|
|
|
for (tmp = fdp; *tmp; tmp = &((*tmp)->next));
|
|
*tmp = (AreaList *)malloc(sizeof(AreaList));
|
|
(*tmp)->next = NULL;
|
|
strcpy((*tmp)->Name, tag);
|
|
(*tmp)->IsPresent = FALSE;
|
|
(*tmp)->DoDelete = DoDelete;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Process areas commandline. Check for missing records in message and
|
|
* file groups or areas dropped from the area taglists. Supported taglists
|
|
* are plain areaname description lists and for file areas the filegate.zxx
|
|
* formatted lists.
|
|
*/
|
|
int Areas(void)
|
|
{
|
|
FILE *gp, *ap, *fp;
|
|
char *temp, *buf, *tag, *desc, *p, *cmd = NULL;
|
|
AreaList *alist = NULL, *tmp;
|
|
int i, count = 0, Found;
|
|
sysconnect System;
|
|
faddr *From, *To;
|
|
|
|
Mgrlog("Process areas taglists");
|
|
|
|
if (!do_quiet) {
|
|
mbse_colour(CYAN, BLACK);
|
|
printf("Processing areas taglists...\n");
|
|
}
|
|
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
buf = calloc(4097, sizeof(char));
|
|
|
|
snprintf(temp, PATH_MAX, "%s/etc/mgroups.data", getenv("MBSE_ROOT"));
|
|
if ((gp = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
} else {
|
|
fread(&mgrouphdr, sizeof(mgrouphdr), 1, gp);
|
|
fseek(gp, mgrouphdr.hdrsize, SEEK_SET);
|
|
|
|
while ((fread(&mgroup, mgrouphdr.recsize, 1, gp)) == 1) {
|
|
if (mgroup.Active && mgroup.AutoChange && strlen(mgroup.AreaFile) && mgroup.UpLink.zone && SearchNode(mgroup.UpLink)) {
|
|
if (!do_quiet) {
|
|
mbse_colour(CYAN, BLACK);
|
|
printf("\rEcho group %-12s ", mgroup.Name);
|
|
fflush(stdout);
|
|
}
|
|
Syslog('+', "Checking mail group %s, file %s", mgroup.Name, mgroup.AreaFile);
|
|
snprintf(temp, PATH_MAX, "%s/%s", CFG.alists_path, mgroup.AreaFile);
|
|
if ((ap = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
} else {
|
|
while (fgets(buf, 4096, ap)) {
|
|
if (strlen(buf) && isalnum(buf[0])) {
|
|
tag = strtok(buf, "\t \r\n\0");
|
|
fill_arealist(&alist, tag, FALSE);
|
|
}
|
|
}
|
|
fclose(ap);
|
|
|
|
/*
|
|
* Mark areas already present in the taglist.
|
|
*/
|
|
if (!do_quiet) {
|
|
mbse_colour(LIGHTRED, BLACK);
|
|
printf("(check missing areas)\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
snprintf(temp, PATH_MAX, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((fp = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
tidy_arealist(&alist);
|
|
free(buf);
|
|
free(temp);
|
|
return FALSE;
|
|
}
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fp);
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
fseek(fp, msgshdr.hdrsize, SEEK_SET);
|
|
if (CFG.slow_util && do_quiet)
|
|
msleep(1);
|
|
while (fread(&msgs, msgshdr.recsize, 1, fp) == 1) {
|
|
if (msgs.Active && !strcmp(msgs.Group, mgroup.Name) && !strcmp(msgs.Tag, tmp->Name))
|
|
tmp->IsPresent = TRUE;
|
|
fseek(fp, msgshdr.syssize, SEEK_CUR);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Add areas to AreaList not in the taglist, they must be deleted.
|
|
*/
|
|
if (!do_quiet) {
|
|
printf("(check deleted areas)\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
fseek(fp, msgshdr.hdrsize, SEEK_SET);
|
|
while (fread(&msgs, msgshdr.recsize, 1, fp) == 1) {
|
|
if (msgs.Active && !strcmp(msgs.Group, mgroup.Name) && (msgs.Type == ECHOMAIL)) {
|
|
Found = FALSE;
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
if (!strcmp(msgs.Tag, tmp->Name))
|
|
Found = TRUE;
|
|
}
|
|
if (!Found)
|
|
fill_arealist(&alist, msgs.Tag, TRUE);
|
|
}
|
|
fseek(fp, msgshdr.syssize, SEEK_CUR);
|
|
}
|
|
fclose(fp);
|
|
if (!do_quiet) {
|
|
printf("(update database) \b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
|
|
/*
|
|
* Make modification, first add missing areas
|
|
*/
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
if (!tmp->IsPresent && !tmp->DoDelete) {
|
|
/*
|
|
* Autocraete group, don't sent uplink request yet.
|
|
*/
|
|
CheckEchoGroup(tmp->Name, FALSE, NULL);
|
|
if (cmd == NULL) {
|
|
cmd = xstrcpy((char *)"");
|
|
} else {
|
|
cmd = xstrcat(cmd, (char *)"\r");
|
|
}
|
|
if (nodes.UplAmgrBbbs)
|
|
cmd = xstrcat(cmd, (char *)"echo +");
|
|
else
|
|
cmd = xstrcat(cmd, (char *)"+");
|
|
cmd = xstrcat(cmd, tmp->Name);
|
|
if (CFG.slow_util && do_quiet)
|
|
msleep(1);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Now remove deleted areas. If there are messages in the area,
|
|
* the area is set to read-only and all links are disconnected.
|
|
* If the area is empty, it is removed from the setup.
|
|
*/
|
|
snprintf(temp, PATH_MAX, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((fp = fopen(temp, "r+")) == NULL) {
|
|
WriteError("Can't open %s for r/w");
|
|
} else {
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fp);
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
if (!tmp->IsPresent && tmp->DoDelete) {
|
|
fseek(fp, msgshdr.hdrsize, SEEK_SET);
|
|
Syslog('m', "Delete %s", tmp->Name);
|
|
if (CFG.slow_util && do_quiet)
|
|
msleep(1);
|
|
while (fread(&msgs, msgshdr.recsize, 1, fp) == 1) {
|
|
if (msgs.Active && !strcmp(msgs.Group, mgroup.Name) && !strcmp(msgs.Tag, tmp->Name)) {
|
|
fseek(fp, - msgshdr.recsize, SEEK_CUR);
|
|
snprintf(temp, PATH_MAX, "%s.jhr", msgs.Base);
|
|
if (strlen(msgs.Base) && (file_size(temp) != 1024)) {
|
|
Mgrlog("Marking echo %s, group %s, area %d read-only", msgs.Tag, mgroup.Name,
|
|
((ftell(fp) - msgshdr.hdrsize) / (msgshdr.recsize + msgshdr.syssize)) + 1);
|
|
msgs.MsgKinds = RONLY; // Area read-only
|
|
snprintf(msgs.Group, 13, "DELETED"); // Make groupname invalid
|
|
} else {
|
|
Mgrlog("Removing empty echo %s, group %s, area %d", msgs.Tag, mgroup.Name,
|
|
((ftell(fp) - msgshdr.hdrsize) / (msgshdr.recsize + msgshdr.syssize)) + 1);
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
msgs.DaysOld = CFG.defdays;
|
|
msgs.MaxMsgs = CFG.defmsgs;
|
|
msgs.Type = ECHOMAIL;
|
|
msgs.MsgKinds = PUBLIC;
|
|
msgs.UsrDelete = TRUE;
|
|
msgs.Charset = FTNC_NONE;
|
|
msgs.MaxArticles = CFG.maxarticles;
|
|
strcpy(msgs.Origin, CFG.origin);
|
|
}
|
|
fwrite(&msgs, msgshdr.recsize, 1, fp);
|
|
/*
|
|
* Always clear all connections, the area doesn't exist anymore.
|
|
*/
|
|
memset(&System, 0, sizeof(System));
|
|
for (i = 0; i < (msgshdr.syssize / sizeof(sysconnect)); i++)
|
|
fwrite(&System, sizeof(system), 1, fp);
|
|
/*
|
|
* Prepare uplink command
|
|
*/
|
|
if (cmd == NULL)
|
|
cmd = xstrcpy((char *)"");
|
|
else
|
|
cmd = xstrcat(cmd, (char *)"\r");
|
|
if (nodes.UplAmgrBbbs)
|
|
cmd = xstrcat(cmd, (char *)"echo -");
|
|
else
|
|
cmd = xstrcat(cmd, (char *)"-");
|
|
cmd = xstrcat(cmd, tmp->Name);
|
|
break;
|
|
} else {
|
|
fseek(fp, msgshdr.syssize, SEEK_CUR);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
fclose(fp);
|
|
}
|
|
tidy_arealist(&alist);
|
|
if (cmd != NULL) {
|
|
/*
|
|
* Sent one uplink command with additions and deletions
|
|
*/
|
|
From = fido2faddr(mgroup.UseAka);
|
|
To = fido2faddr(mgroup.UpLink);
|
|
if (UplinkRequest(To, From, FALSE, cmd)) {
|
|
WriteError("Uplink request failed");
|
|
} else {
|
|
Mgrlog("AreaMgr request sent to %s", aka2str(mgroup.UpLink));
|
|
}
|
|
tidy_faddr(From);
|
|
tidy_faddr(To);
|
|
free(cmd);
|
|
cmd = NULL;
|
|
}
|
|
if (!do_quiet) {
|
|
printf(" \b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
fclose(gp);
|
|
}
|
|
|
|
snprintf(temp, PATH_MAX, "%s/etc/fgroups.data", getenv("MBSE_ROOT"));
|
|
if ((gp = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
} else {
|
|
fread(&fgrouphdr, sizeof(fgrouphdr), 1, gp);
|
|
fseek(gp, fgrouphdr.hdrsize, SEEK_SET);
|
|
|
|
while ((fread(&fgroup, fgrouphdr.recsize, 1, gp)) == 1) {
|
|
if (fgroup.Active && fgroup.AutoChange && strlen(fgroup.AreaFile) && fgroup.UpLink.zone && SearchNode(fgroup.UpLink)) {
|
|
if (!do_quiet) {
|
|
mbse_colour(CYAN, BLACK);
|
|
printf("\r TIC group %-12s ", fgroup.Name);
|
|
fflush(stdout);
|
|
}
|
|
Syslog('+', "Checking tic group %s, file %s", fgroup.Name, fgroup.AreaFile);
|
|
snprintf(temp, PATH_MAX, "%s/%s", CFG.alists_path, fgroup.AreaFile);
|
|
if ((ap = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
} else {
|
|
if (fgroup.FileGate) {
|
|
/*
|
|
* filegate.zxx format
|
|
*/
|
|
Found = FALSE;
|
|
while (fgets(buf, 4096, ap)) {
|
|
/*
|
|
* Each group starts with % FDN: FileGroup Descrition
|
|
*/
|
|
if (strlen(buf) && !strncmp(buf, "% FDN:", 6)) {
|
|
tag = strtok(buf, "\t \r\n\0");
|
|
p = strtok(NULL, "\t \r\n\0");
|
|
p = strtok(NULL, "\r\n\0");
|
|
desc = p;
|
|
while ((*desc == ' ') || (*desc == '\t'))
|
|
desc++;
|
|
if (!strcmp(desc, fgroup.Comment)) {
|
|
Syslog('f', "Start of group \"%s\" found", desc);
|
|
while (fgets(buf, 4096, ap)) {
|
|
if (!strncasecmp(buf, "Area ", 5)) {
|
|
Syslog('f', "Area: %s", buf);
|
|
tag = strtok(buf, "\t \r\n\0");
|
|
tag = strtok(NULL, "\t \r\n\0");
|
|
Found = TRUE;
|
|
fill_arealist(&alist, tag, FALSE);
|
|
}
|
|
if (strlen(buf) && !strncmp(buf, "% FDN:", 6)) {
|
|
/*
|
|
* All entries in group are seen, the area wasn't there.
|
|
*/
|
|
Syslogp('f', buf);
|
|
break;
|
|
}
|
|
}
|
|
if (Found)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/*
|
|
* Normal taglist format
|
|
*/
|
|
while (fgets(buf, 4096, ap)) {
|
|
if (strlen(buf) && isalnum(buf[0])) {
|
|
tag = strtok(buf, "\t \r\n\0");
|
|
fill_arealist(&alist, tag, FALSE);
|
|
}
|
|
}
|
|
}
|
|
fclose(ap);
|
|
|
|
/*
|
|
* Mark areas already present in the taglist.
|
|
*/
|
|
if (!do_quiet) {
|
|
mbse_colour(LIGHTRED, BLACK);
|
|
printf("(check missing areas)\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
snprintf(temp, PATH_MAX, "%s/etc/tic.data", getenv("MBSE_ROOT"));
|
|
if ((fp = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
tidy_arealist(&alist);
|
|
free(buf);
|
|
free(temp);
|
|
return FALSE;
|
|
}
|
|
fread(&tichdr, sizeof(tichdr), 1, fp);
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
fseek(fp, tichdr.hdrsize, SEEK_SET);
|
|
if (CFG.slow_util && do_quiet)
|
|
msleep(1);
|
|
while (fread(&tic, tichdr.recsize, 1, fp) == 1) {
|
|
if (tic.Active && !strcmp(tic.Group, fgroup.Name) && !strcmp(tic.Name, tmp->Name))
|
|
tmp->IsPresent = TRUE;
|
|
fseek(fp, tichdr.syssize, SEEK_CUR);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Add areas to AreaList not in the taglist, they must be deleted.
|
|
*/
|
|
if (!do_quiet) {
|
|
printf("(check deleted areas)\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
fseek(fp, tichdr.hdrsize, SEEK_SET);
|
|
while (fread(&tic, tichdr.recsize, 1, fp) == 1) {
|
|
if (tic.Active && !strcmp(tic.Group, fgroup.Name)) {
|
|
Found = FALSE;
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
if (!strcmp(tic.Name, tmp->Name))
|
|
Found = TRUE;
|
|
}
|
|
if (!Found)
|
|
fill_arealist(&alist, tic.Name, TRUE);
|
|
}
|
|
fseek(fp, tichdr.syssize, SEEK_CUR);
|
|
}
|
|
fclose(fp);
|
|
if (!do_quiet) {
|
|
printf("(update database) \b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
|
|
/*
|
|
* Make modification, first add missing areas
|
|
*/
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
if (!tmp->IsPresent && !tmp->DoDelete) {
|
|
/*
|
|
* Autocraete group, don't sent uplink request yet.
|
|
*/
|
|
CheckTicGroup(tmp->Name, FALSE, NULL);
|
|
if (cmd == NULL) {
|
|
cmd = xstrcpy((char *)"");
|
|
} else {
|
|
cmd = xstrcat(cmd, (char *)"\r");
|
|
}
|
|
if (nodes.UplFmgrBbbs)
|
|
cmd = xstrcat(cmd, (char *)"file +");
|
|
else
|
|
cmd = xstrcat(cmd, (char *)"+");
|
|
cmd = xstrcat(cmd, tmp->Name);
|
|
if (CFG.slow_util && do_quiet)
|
|
msleep(1);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Mark TIC areas for deletion. The original file areas
|
|
* are not deleted. They probably contain files and we
|
|
* may want to keep these. If the area was empty we are
|
|
* still warned about that by the "mbfile check" command.
|
|
*/
|
|
Found = FALSE;
|
|
snprintf(temp, PATH_MAX, "%s/etc/tic.data", getenv("MBSE_ROOT"));
|
|
if ((fp = fopen(temp, "r+")) == NULL) {
|
|
WriteError("Can't open %s for r/w");
|
|
} else {
|
|
fread(&tichdr, sizeof(tichdr), 1, fp);
|
|
for (tmp = alist; tmp; tmp = tmp->next) {
|
|
if (!tmp->IsPresent && tmp->DoDelete) {
|
|
fseek(fp, tichdr.hdrsize, SEEK_SET);
|
|
Syslog('f', "Delete %s", tmp->Name);
|
|
if (CFG.slow_util && do_quiet)
|
|
msleep(1);
|
|
while (fread(&tic, tichdr.recsize, 1, fp) == 1) {
|
|
if (tic.Active && !strcmp(tic.Group, fgroup.Name) && !strcmp(tic.Name, tmp->Name)) {
|
|
fseek(fp, - tichdr.recsize, SEEK_CUR);
|
|
Mgrlog("Marked TIC area %s, group %s for deletion", tmp->Name, fgroup.Name);
|
|
tic.Deleted = TRUE;
|
|
tic.Active = FALSE;
|
|
fwrite(&tic, tichdr.recsize, 1, fp);
|
|
Found = TRUE;
|
|
/*
|
|
* Prepare uplink command
|
|
*/
|
|
if (cmd == NULL)
|
|
cmd = xstrcpy((char *)"");
|
|
else
|
|
cmd = xstrcat(cmd, (char *)"\r");
|
|
if (nodes.UplFmgrBbbs)
|
|
cmd = xstrcat(cmd, (char *)"file -");
|
|
else
|
|
cmd = xstrcat(cmd, (char *)"-");
|
|
cmd = xstrcat(cmd, tmp->Name);
|
|
}
|
|
fseek(fp, tichdr.syssize, SEEK_CUR);
|
|
}
|
|
}
|
|
}
|
|
fclose(fp);
|
|
}
|
|
if (Found) {
|
|
/*
|
|
* Purge marked records
|
|
*/
|
|
snprintf(buf, 4096, "%s/etc/tic.temp", getenv("MBSE_ROOT"));
|
|
if ((fp = fopen(temp, "r")) == NULL) {
|
|
WriteError("Can't open %s", temp);
|
|
} else if ((ap = fopen(buf, "w")) == NULL) {
|
|
WriteError("Can't create %s", buf);
|
|
fclose(fp);
|
|
} else {
|
|
fread(&tichdr, tichdr.hdrsize, 1, fp);
|
|
fwrite(&tichdr, tichdr.hdrsize, 1, ap);
|
|
while (fread(&tic, tichdr.recsize, 1, fp) == 1) {
|
|
if (tic.Deleted && !tic.Active) {
|
|
fseek(fp, tichdr.syssize, SEEK_CUR);
|
|
count++;
|
|
} else {
|
|
fwrite(&tic, tichdr.recsize, 1, ap);
|
|
for (i = 0; i < (tichdr.syssize / sizeof(System)); i++) {
|
|
fread(&System, sizeof(System), 1, fp);
|
|
fwrite(&System, sizeof(System), 1, ap);
|
|
}
|
|
}
|
|
}
|
|
fclose(fp);
|
|
fclose(ap);
|
|
unlink(temp);
|
|
rename(buf, temp);
|
|
Mgrlog("Purged %d TIC areas", count);
|
|
}
|
|
}
|
|
|
|
tidy_arealist(&alist);
|
|
if (cmd != NULL) {
|
|
/*
|
|
* Sent one uplink command with additions and deletions
|
|
*/
|
|
From = fido2faddr(fgroup.UseAka);
|
|
To = fido2faddr(fgroup.UpLink);
|
|
if (UplinkRequest(To, From, TRUE, cmd)) {
|
|
WriteError("Uplink request failed");
|
|
} else {
|
|
Mgrlog("FileMgr request sent to %s", aka2str(fgroup.UpLink));
|
|
}
|
|
tidy_faddr(From);
|
|
tidy_faddr(To);
|
|
free(cmd);
|
|
cmd = NULL;
|
|
}
|
|
if (!do_quiet) {
|
|
printf(" \b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
|
|
fflush(stdout);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
fclose(gp);
|
|
}
|
|
|
|
if (!do_quiet)
|
|
printf("\r \r");
|
|
|
|
free(buf);
|
|
free(temp);
|
|
if (net_out)
|
|
do_flush = TRUE;
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|