1705 lines
47 KiB
C
1705 lines
47 KiB
C
/*****************************************************************************
|
|
*
|
|
* $Id$
|
|
* Purpose ...............: Message Areas Setup
|
|
*
|
|
*****************************************************************************
|
|
* Copyright (C) 1997-2002
|
|
*
|
|
* 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.
|
|
*
|
|
* MB 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 MB BBS; see the file COPYING. If not, write to the Free
|
|
* Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*****************************************************************************/
|
|
|
|
#include "../lib/libs.h"
|
|
#include "../lib/structs.h"
|
|
#include "../lib/users.h"
|
|
#include "../lib/records.h"
|
|
#include "../lib/common.h"
|
|
#include "../lib/clcomm.h"
|
|
#include "../lib/msg.h"
|
|
#include "screen.h"
|
|
#include "mutil.h"
|
|
#include "ledit.h"
|
|
#include "grlist.h"
|
|
#include "m_global.h"
|
|
#include "m_node.h"
|
|
#include "m_mgroup.h"
|
|
#include "m_marea.h"
|
|
|
|
|
|
int MsgUpdated = 0;
|
|
unsigned long MsgCrc;
|
|
FILE *tfil = NULL;
|
|
extern int exp_golded;
|
|
int MailForced = FALSE;
|
|
|
|
|
|
/*
|
|
* Count nr of msgs records in the database.
|
|
* Creates the database if it doesn't exist.
|
|
*/
|
|
int CountMsgarea(void)
|
|
{
|
|
FILE *fil;
|
|
char ffile[PATH_MAX];
|
|
int count, i;
|
|
struct _sysconnect syscon;
|
|
|
|
sprintf(ffile, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((fil = fopen(ffile, "r")) == NULL) {
|
|
if ((fil = fopen(ffile, "a+")) != NULL) {
|
|
Syslog('+', "Created new %s", ffile);
|
|
msgshdr.hdrsize = sizeof(msgshdr);
|
|
msgshdr.recsize = sizeof(msgs);
|
|
msgshdr.syssize = CFG.toss_systems * sizeof(sysconnect);
|
|
msgshdr.lastupd = time(NULL);
|
|
fwrite(&msgshdr, sizeof(msgshdr), 1, fil);
|
|
/*
|
|
* Default first message area
|
|
*/
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
sprintf(msgs.Name, "Local users chat");
|
|
sprintf(msgs.Base, "%s/var/mail/local/users", getenv("MBSE_ROOT"));
|
|
sprintf(msgs.QWKname, "LOC_USERS");
|
|
sprintf(msgs.Group, "LOCAL");
|
|
msgs.Active = TRUE;
|
|
msgs.Type = LOCALMAIL;
|
|
msgs.MsgKinds = PUBLIC;
|
|
msgs.SYSec.level = 32000;
|
|
msgs.UsrDelete = TRUE;
|
|
msgs.Aliases = TRUE;
|
|
msgs.Quotes = TRUE;
|
|
msgs.DaysOld = CFG.defdays;
|
|
msgs.MaxMsgs = CFG.defmsgs;
|
|
msgs.Rfccode = CHRS_DEFAULT_RFC;
|
|
msgs.Ftncode = CHRS_DEFAULT_FTN;
|
|
strcpy(msgs.Origin, CFG.origin);
|
|
fwrite(&msgs, sizeof(msgs), 1, fil);
|
|
mkdirs(msgs.Base, 0770);
|
|
if (Msg_Open(msgs.Base))
|
|
Msg_Close();
|
|
memset(&syscon, 0, sizeof(syscon));
|
|
for (i = 1; i <= CFG.toss_systems; i++)
|
|
fwrite(&syscon, sizeof(syscon), 1, fil);
|
|
/*
|
|
* Default message area for badmail
|
|
*/
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
sprintf(msgs.Name, "Bad mail");
|
|
sprintf(msgs.Base, "/opt/mbse/var/mail/badmail");
|
|
sprintf(msgs.QWKname, "BADMAIL");
|
|
sprintf(msgs.Group, "LOCAL");
|
|
msgs.Active = TRUE;
|
|
msgs.Type = LOCALMAIL;
|
|
msgs.MsgKinds = PUBLIC;
|
|
msgs.RDSec.level = 32000;
|
|
msgs.WRSec.level = 32000;
|
|
msgs.SYSec.level = 32000;
|
|
msgs.DaysOld = CFG.defdays;
|
|
msgs.MaxMsgs = CFG.defmsgs;
|
|
msgs.Rfccode = CHRS_DEFAULT_RFC;
|
|
msgs.Ftncode = CHRS_DEFAULT_FTN;
|
|
fwrite(&msgs, sizeof(msgs), 1, fil);
|
|
mkdirs(msgs.Base, 0770);
|
|
if (Msg_Open(msgs.Base))
|
|
Msg_Close();
|
|
for (i = 1; i <= CFG.toss_systems; i++)
|
|
fwrite(&syscon, sizeof(syscon), 1, fil);
|
|
/*
|
|
* Default dupemail message area
|
|
*/
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
sprintf(msgs.Name, "Dupe mail");
|
|
sprintf(msgs.Base, "/opt/mbse/var/mail/dupemail");
|
|
sprintf(msgs.QWKname, "DUPEMAIL");
|
|
sprintf(msgs.Group, "LOCAL");
|
|
msgs.Active = TRUE;
|
|
msgs.Type = LOCALMAIL;
|
|
msgs.MsgKinds = PUBLIC;
|
|
msgs.RDSec.level = 32000;
|
|
msgs.WRSec.level = 32000;
|
|
msgs.SYSec.level = 32000;
|
|
msgs.DaysOld = CFG.defdays;
|
|
msgs.MaxMsgs = CFG.defmsgs;
|
|
msgs.Rfccode = CHRS_DEFAULT_RFC;
|
|
msgs.Ftncode = CHRS_DEFAULT_FTN;
|
|
fwrite(&msgs, sizeof(msgs), 1, fil);
|
|
mkdirs(msgs.Base, 0770);
|
|
if (Msg_Open(msgs.Base))
|
|
for (i = 1; i <= CFG.toss_systems; i++)
|
|
fwrite(&syscon, sizeof(syscon), 1, fil);
|
|
|
|
fclose(fil);
|
|
exp_golded = TRUE;
|
|
chmod(ffile, 0660);
|
|
return 3;
|
|
} else
|
|
return -1;
|
|
}
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fil);
|
|
fseek(fil, 0, SEEK_SET);
|
|
fread(&msgshdr, msgshdr.hdrsize, 1, fil);
|
|
fseek(fil, 0, SEEK_END);
|
|
count = (ftell(fil) - msgshdr.hdrsize) / (msgshdr.recsize + msgshdr.syssize);
|
|
fclose(fil);
|
|
|
|
return count;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Open database for editing. The datafile is copied, if the format
|
|
* is changed it will be converted on the fly. All editing must be
|
|
* done on the copied file.
|
|
*/
|
|
int OpenMsgarea(void);
|
|
int OpenMsgarea(void)
|
|
{
|
|
FILE *fin, *fout;
|
|
char fnin[PATH_MAX], fnout[PATH_MAX];
|
|
long oldsize, oldsys;
|
|
struct _sysconnect syscon;
|
|
int i, oldsystems;
|
|
|
|
sprintf(fnin, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
sprintf(fnout, "%s/etc/mareas.temp", getenv("MBSE_ROOT"));
|
|
if ((fin = fopen(fnin, "r")) != NULL) {
|
|
if ((fout = fopen(fnout, "w")) != NULL) {
|
|
MsgUpdated = 0;
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fin);
|
|
fseek(fin, 0, SEEK_SET);
|
|
fread(&msgshdr, msgshdr.hdrsize, 1, fin);
|
|
if (msgshdr.hdrsize != sizeof(msgshdr)) {
|
|
msgshdr.hdrsize = sizeof(msgshdr);
|
|
msgshdr.lastupd = time(NULL);
|
|
MsgUpdated = 1;
|
|
Syslog('+', "Updated %s, format changed", fnin);
|
|
}
|
|
|
|
/*
|
|
* In case we are automatic upgrading the data format
|
|
* we save the old format. If it is changed, the
|
|
* database must always be updated.
|
|
*/
|
|
oldsize = msgshdr.recsize;
|
|
oldsys = msgshdr.syssize;
|
|
oldsystems = oldsys / sizeof(syscon);
|
|
if ((oldsize != sizeof(msgs)) || (CFG.toss_systems != oldsystems)) {
|
|
MsgUpdated = 1;
|
|
Syslog('+', "Updated %s, nr of systems is changed to %d", fnin, CFG.toss_systems);
|
|
}
|
|
msgshdr.hdrsize = sizeof(msgshdr);
|
|
msgshdr.recsize = sizeof(msgs);
|
|
msgshdr.syssize = sizeof(syscon) * CFG.toss_systems;
|
|
fwrite(&msgshdr, sizeof(msgshdr), 1, fout);
|
|
|
|
/*
|
|
* The datarecord is filled with zero's before each
|
|
* read, so if the format changed, the new fields
|
|
* will be empty.
|
|
*/
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
while (fread(&msgs, oldsize, 1, fin) == 1) {
|
|
if ((oldsize != sizeof(msgs)) && !msgs.Rfccode) {
|
|
msgs.Rfccode = CHRS_DEFAULT_RFC;
|
|
msgs.Ftncode = CHRS_DEFAULT_FTN;
|
|
}
|
|
fwrite(&msgs, sizeof(msgs), 1, fout);
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
/*
|
|
* Copy the existing connections
|
|
*/
|
|
for (i = 1; i <= oldsystems; i++) {
|
|
fread(&syscon, sizeof(syscon), 1, fin);
|
|
/*
|
|
* Write only valid records
|
|
*/
|
|
if (i <= CFG.toss_systems)
|
|
fwrite(&syscon, sizeof(syscon), 1, fout);
|
|
}
|
|
if (oldsystems < CFG.toss_systems) {
|
|
/*
|
|
* The size is increased, fill with
|
|
* blank records.
|
|
*/
|
|
memset(&syscon, 0, sizeof(syscon));
|
|
for (i = (oldsystems + 1); i <= CFG.toss_systems; i++)
|
|
fwrite(&syscon, sizeof(syscon), 1, fout);
|
|
}
|
|
}
|
|
|
|
fclose(fin);
|
|
fclose(fout);
|
|
return 0;
|
|
} else
|
|
return -1;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
|
|
|
|
void CloseMsgarea(int Force)
|
|
{
|
|
char fin[PATH_MAX], fout[PATH_MAX];
|
|
|
|
sprintf(fin, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
sprintf(fout,"%s/etc/mareas.temp", getenv("MBSE_ROOT"));
|
|
|
|
if (MsgUpdated == 1) {
|
|
if (Force || (yes_no((char *)"Messages database is changed, save changes") == 1)) {
|
|
working(1, 0, 0);
|
|
exp_golded = TRUE;
|
|
if ((rename(fout, fin)) == 0)
|
|
unlink(fout);
|
|
chmod(fin, 0660);
|
|
Syslog('+', "Updated \"mareas.data\"");
|
|
return;
|
|
}
|
|
}
|
|
chmod(fin, 0660);
|
|
working(1, 0, 0);
|
|
unlink(fout);
|
|
}
|
|
|
|
|
|
|
|
void InitMsgRec(void);
|
|
void InitMsgRec(void)
|
|
{
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
/*
|
|
* Fill in default values
|
|
*/
|
|
msgs.DaysOld = CFG.defdays;
|
|
msgs.MaxMsgs = CFG.defmsgs;
|
|
msgs.Type = ECHOMAIL;
|
|
msgs.MsgKinds = PUBLIC;
|
|
msgs.UsrDelete = TRUE;
|
|
msgs.Rfccode = CHRS_DEFAULT_RFC;
|
|
msgs.Ftncode = CHRS_DEFAULT_FTN;
|
|
msgs.MaxArticles = CFG.maxarticles;
|
|
strcpy(msgs.Origin, CFG.origin);
|
|
}
|
|
|
|
|
|
|
|
void DeleteJAM(char *);
|
|
void DeleteJAM(char *Base)
|
|
{
|
|
char *temp;
|
|
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
sprintf(temp, "%s.jdt", Base);
|
|
unlink(temp);
|
|
sprintf(temp, "%s.jdx", Base);
|
|
unlink(temp);
|
|
sprintf(temp, "%s.jhr", Base);
|
|
unlink(temp);
|
|
sprintf(temp, "%s.jlr", Base);
|
|
unlink(temp);
|
|
free(temp);
|
|
}
|
|
|
|
|
|
|
|
int AppendMsgarea(void);
|
|
int AppendMsgarea()
|
|
{
|
|
FILE *fil;
|
|
char ffile[PATH_MAX];
|
|
struct _sysconnect syscon;
|
|
int i;
|
|
|
|
sprintf(ffile, "%s/etc/mareas.temp", getenv("MBSE_ROOT"));
|
|
if ((fil = fopen(ffile, "a")) != NULL) {
|
|
InitMsgRec();
|
|
fwrite(&msgs, sizeof(msgs), 1, fil);
|
|
memset(&syscon, 0, sizeof(syscon));
|
|
for (i = 1; i <= CFG.toss_systems; i++)
|
|
fwrite(&syscon, sizeof(syscon), 1, fil);
|
|
fclose(fil);
|
|
MsgUpdated = 1;
|
|
return 0;
|
|
} else
|
|
return -1;
|
|
}
|
|
|
|
|
|
|
|
void EditSystem(sysconnect *);
|
|
void EditSystem(sysconnect *Sys)
|
|
{
|
|
sysconnect S;
|
|
unsigned short zone = 0;
|
|
int refresh = TRUE;
|
|
|
|
S = (* Sys);
|
|
for (;;) {
|
|
if (refresh) {
|
|
clr_index();
|
|
set_color(WHITE, BLACK);
|
|
mvprintw( 5,6, "9.2.26 EDIT CONNECTION");
|
|
set_color(CYAN, BLACK);
|
|
mvprintw( 7,6, "1. Aka");
|
|
mvprintw( 8,6, "2. Send to");
|
|
mvprintw( 9,6, "3. Recv from");
|
|
mvprintw(10,6, "4. Pause");
|
|
mvprintw(11,6, "5. Excluded");
|
|
mvprintw(12,6, "6. Delete");
|
|
refresh = FALSE;
|
|
}
|
|
set_color(WHITE, BLACK);
|
|
show_str( 7,23,23, aka2str(S.aka));
|
|
show_bool( 8,23, S.sendto);
|
|
show_bool( 9,23, S.receivefrom);
|
|
show_bool(10,23, S.pause);
|
|
show_bool(11,23, S.cutoff);
|
|
zone = S.aka.zone;
|
|
|
|
switch(select_menu(6)) {
|
|
case 0: (* Sys) = S;
|
|
return;
|
|
case 1: S.aka = PullUplink((char *)"9.2.29");
|
|
refresh = TRUE;
|
|
break;
|
|
case 2: E_BOOL( 8,23, S.sendto, "^Send^ mail ^to^ this node")
|
|
case 3: E_BOOL( 9,23, S.receivefrom, "^Receive^ mail ^from^ this node")
|
|
case 4: E_BOOL(10,23, S.pause, "Is this node ^paused^")
|
|
case 5: E_BOOL(11,23, S.cutoff, "Is this node ^excluded (cutoff)^ by a moderator")
|
|
case 6: if (yes_no((char *)"Delete this entry")) {
|
|
memset(&S, 0, sizeof(sysconnect));
|
|
(* Sys) = S;
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* Set sendto and receivefrom to on when a new
|
|
* zone is entered.
|
|
*/
|
|
if ((S.aka.zone) && (!zone)) {
|
|
S.sendto = 1;
|
|
S.receivefrom = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int EditConnections(FILE *);
|
|
int EditConnections(FILE *fil)
|
|
{
|
|
int systems, o = 0, i, y, x;
|
|
long offset;
|
|
char pick[12];
|
|
sysconnect System;
|
|
char status[5];
|
|
char temp[41];
|
|
|
|
systems = msgshdr.syssize / sizeof(sysconnect);
|
|
for (;;) {
|
|
clr_index();
|
|
set_color(WHITE, BLACK);
|
|
mvprintw( 5, 5, "9.2.29 MESSAGE AREA CONNECTIONS");
|
|
set_color(CYAN, BLACK);
|
|
y = 7;
|
|
x = 2;
|
|
for (i = 1; i <= 20; i++) {
|
|
if ((o+i-1) < systems) {
|
|
if (i == 11) {
|
|
y = 7;
|
|
x = 42;
|
|
}
|
|
offset = (o+i-1) * sizeof(sysconnect);
|
|
if ((fseek(fil, offset, 0)) != 0) {
|
|
working(2, 0, 0);
|
|
return FALSE;
|
|
}
|
|
fread(&System, sizeof(sysconnect), 1, fil);
|
|
memset(&status, 0, 5);
|
|
memset(&status, '-', 4);
|
|
if (System.sendto)
|
|
status[0] = 'S';
|
|
if (System.receivefrom)
|
|
status[1] = 'R';
|
|
if (System.pause)
|
|
status[2] = 'P';
|
|
if (System.cutoff)
|
|
status[3] = 'C';
|
|
|
|
if (System.aka.zone) {
|
|
set_color(CYAN,BLACK);
|
|
sprintf(temp, "%3d. %s %d:%d/%d.%d@%s", o+i, status,
|
|
System.aka.zone, System.aka.net, System.aka.node,
|
|
System.aka.point, System.aka.domain);
|
|
} else {
|
|
set_color(LIGHTBLUE, BLACK);
|
|
sprintf(temp, "%3d.", o+i);
|
|
}
|
|
mvprintw(y, x, temp);
|
|
y++;
|
|
}
|
|
}
|
|
strcpy(pick, select_pick(systems, 20));
|
|
|
|
if (strncmp(pick, "-", 1) == 0) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (strncmp(pick, "N", 1) == 0)
|
|
if ((o + 20) < systems)
|
|
o = o + 20;
|
|
|
|
if (strncmp(pick, "P", 1) == 0)
|
|
if ((o - 20) >= 0)
|
|
o = o - 20;
|
|
|
|
if (((atoi(pick) > 0) && (atoi(pick) <= systems))) {
|
|
offset = (atoi(pick) -1) * sizeof(sysconnect);
|
|
fseek(fil, offset, 0);
|
|
fread(&System, sizeof(sysconnect), 1, fil);
|
|
EditSystem(&System);
|
|
fseek(fil, offset, 0);
|
|
fwrite(&System, sizeof(sysconnect), 1, fil);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void SetScreen(void);
|
|
void SetScreen()
|
|
{
|
|
clr_index();
|
|
set_color(WHITE, BLACK);
|
|
mvprintw( 4, 2, "9.2 EDIT MESSAGE AREA");
|
|
set_color(CYAN, BLACK);
|
|
mvprintw( 6, 2, "1. Area Name");
|
|
mvprintw( 7, 2, "2. FTN area");
|
|
mvprintw( 8, 2, "3. Group");
|
|
mvprintw( 9, 2, "4. Newsgroup");
|
|
mvprintw(10, 2, "5. JAM base");
|
|
mvprintw(11, 2, "6. Origin");
|
|
mvprintw(12, 2, "7. Fido Aka");
|
|
mvprintw(13, 2, "8. QWK name");
|
|
mvprintw(14, 2, "9. Distrib.");
|
|
mvprintw(15, 2, "10. Area Type");
|
|
mvprintw(16, 2, "11. Msg Kinds");
|
|
mvprintw(17, 2, "12. FTN chars");
|
|
mvprintw(18, 2, "13. RFC chars");
|
|
mvprintw(19, 2, "14. Active");
|
|
|
|
mvprintw(13,36, "15. Days Old");
|
|
mvprintw(14,36, "16. Max. Msgs");
|
|
switch (msgs.Type) {
|
|
case ECHOMAIL: mvprintw(15,36, "17. Netreply");
|
|
break;
|
|
case NEWS: mvprintw(15,36, "17. Articles");
|
|
break;
|
|
default: mvprintw(15,36, "17. N/A");
|
|
break;
|
|
}
|
|
mvprintw(16,36, "18. Read Sec.");
|
|
mvprintw(17,36, "19. Write Sec.");
|
|
mvprintw(18,36, "20. Sysop Sec.");
|
|
mvprintw(19,36, "21. User Del.");
|
|
|
|
mvprintw(13,60, "22. Aliases");
|
|
mvprintw(14,60, "23. Quotes");
|
|
mvprintw(15,60, "24. Mandatory");
|
|
mvprintw(16,60, "25. UnSecure");
|
|
mvprintw(17,60, "26. OLR Default");
|
|
mvprintw(18,60, "27. OLR Forced");
|
|
switch (msgs.Type) {
|
|
case ECHOMAIL:
|
|
case NEWS:
|
|
case LIST: mvprintw(19,60, "28. Connections");
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
long LoadMsgRec(int, int);
|
|
long LoadMsgRec(int Area, int work)
|
|
{
|
|
FILE *fil;
|
|
char mfile[PATH_MAX];
|
|
long offset;
|
|
sysconnect System;
|
|
int i;
|
|
|
|
if (work)
|
|
working(1, 0, 0);
|
|
|
|
sprintf(mfile, "%s/etc/mareas.temp", getenv("MBSE_ROOT"));
|
|
if ((fil = fopen(mfile, "r")) == NULL) {
|
|
working(2, 0, 0);
|
|
return -1;
|
|
}
|
|
|
|
if ((tfil = tmpfile()) == NULL) {
|
|
working(2, 0, 0);
|
|
return -1;
|
|
}
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fil);
|
|
offset = msgshdr.hdrsize + (((Area -1) * (msgshdr.recsize + msgshdr.syssize)));
|
|
if (fseek(fil, offset, SEEK_SET) != 0) {
|
|
fclose(tfil);
|
|
tfil = NULL;
|
|
working(2, 0, 0);
|
|
return -1;
|
|
}
|
|
|
|
fread(&msgs, msgshdr.recsize, 1, fil);
|
|
MsgCrc = 0xffffffff;
|
|
MsgCrc = upd_crc32((char *)&msgs, MsgCrc, msgshdr.recsize);
|
|
for (i = 0; i < (msgshdr.syssize / sizeof(sysconnect)); i++) {
|
|
fread(&System, sizeof(sysconnect), 1, fil);
|
|
fwrite(&System, sizeof(sysconnect), 1, tfil);
|
|
MsgCrc = upd_crc32((char *)&System, MsgCrc, sizeof(sysconnect));
|
|
}
|
|
fclose(fil);
|
|
if (work)
|
|
working(0, 0, 0);
|
|
|
|
return offset;
|
|
}
|
|
|
|
|
|
|
|
int SaveMsgRec(int, int);
|
|
int SaveMsgRec(int Area, int work)
|
|
{
|
|
int i;
|
|
FILE *fil;
|
|
long offset;
|
|
char mfile[PATH_MAX];
|
|
sysconnect System;
|
|
|
|
if (work)
|
|
working(1, 0, 0);
|
|
sprintf(mfile, "%s/etc/mareas.temp", getenv("MBSE_ROOT"));
|
|
if ((fil = fopen(mfile, "r+")) == 0) {
|
|
working(2, 0, 0);
|
|
return -1;
|
|
}
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fil);
|
|
offset = msgshdr.hdrsize + (((Area -1) * (msgshdr.recsize + msgshdr.syssize)));
|
|
if (fseek(fil, offset, SEEK_SET)) {
|
|
fclose(fil);
|
|
working(2, 0, 0);
|
|
return -1;
|
|
}
|
|
|
|
fwrite(&msgs, msgshdr.recsize, 1, fil);
|
|
fseek(tfil, 0, SEEK_SET);
|
|
for (i = 0; i < (msgshdr.syssize / sizeof(sysconnect)); i++) {
|
|
fread(&System, sizeof(sysconnect), 1, tfil);
|
|
fwrite(&System, sizeof(sysconnect), 1, fil);
|
|
}
|
|
fclose(fil);
|
|
fclose(tfil);
|
|
tfil = NULL;
|
|
if (work)
|
|
working(0, 0, 0);
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
void ShowStatus(sysconnect);
|
|
void ShowStatus(sysconnect S)
|
|
{
|
|
clr_index();
|
|
set_color(CYAN, BLACK);
|
|
mvprintw( 7, 6, "Aka");
|
|
mvprintw( 8, 6, "Send to");
|
|
mvprintw( 9, 6, "Recv from");
|
|
mvprintw(10, 6, "Pause");
|
|
mvprintw(11, 6, "Excluded");
|
|
set_color(WHITE, BLACK);
|
|
show_str( 7,16,23, aka2str(S.aka));
|
|
show_bool( 8,16, S.sendto);
|
|
show_bool( 9,16, S.receivefrom);
|
|
show_bool(10,16, S.pause);
|
|
show_bool(11,16, S.cutoff);
|
|
}
|
|
|
|
|
|
|
|
void MsgGlobal(void);
|
|
void MsgGlobal(void)
|
|
{
|
|
gr_list *mgr = NULL, *tmp;
|
|
char *p, mfile[PATH_MAX];
|
|
FILE *fil;
|
|
fidoaddr a1, a2;
|
|
int menu = 0, marea, Areas, akan = 0, Found;
|
|
int Total, Done, netbrd, daysold, maxmsgs, maxarticles;
|
|
long offset;
|
|
securityrec rs, ws, ss;
|
|
sysconnect S, Sc;
|
|
|
|
/*
|
|
* Build the groups select array
|
|
*/
|
|
working(1, 0, 0);
|
|
sprintf(mfile, "%s/etc/mgroups.data", getenv("MBSE_ROOT"));
|
|
if ((fil = fopen(mfile, "r")) != NULL) {
|
|
fread(&mgrouphdr, sizeof(mgrouphdr), 1, fil);
|
|
|
|
while (fread(&mgroup, mgrouphdr.recsize, 1, fil) == 1)
|
|
fill_grlist(&mgr, mgroup.Name);
|
|
|
|
fclose(fil);
|
|
sort_grlist(&mgr);
|
|
}
|
|
working(0, 0, 0);
|
|
|
|
/*
|
|
* Initialize some variables
|
|
*/
|
|
memset(&rs, 0, sizeof(securityrec));
|
|
memset(&ws, 0, sizeof(securityrec));
|
|
memset(&ss, 0, sizeof(securityrec));
|
|
memset(&S, 0, sizeof(sysconnect));
|
|
S.sendto = TRUE;
|
|
S.receivefrom = TRUE;
|
|
memset(&mfile, 0, sizeof(mfile));
|
|
sprintf(mfile, "%s", CFG.origin);
|
|
daysold = CFG.defdays;
|
|
maxmsgs = CFG.defmsgs;
|
|
maxarticles = CFG.maxarticles;
|
|
netbrd = 1;
|
|
|
|
for (;;) {
|
|
clr_index();
|
|
set_color(WHITE, BLACK);
|
|
mvprintw( 5, 6, "9.2 GLOBAL EDIT MESSAGE AREAS");
|
|
set_color(CYAN, BLACK);
|
|
mvprintw( 7, 6, "1. Delete connection");
|
|
mvprintw( 8, 6, "2. Add new connection");
|
|
mvprintw( 9, 6, "3. Replace connection");
|
|
mvprintw(10, 6, "4. Change connection status");
|
|
mvprintw(11, 6, "5. Change days old");
|
|
mvprintw(12, 6, "6. Change max. messages");
|
|
mvprintw(13, 6, "7. Change max. articles");
|
|
mvprintw(14, 6, "8. Change security");
|
|
mvprintw(15, 6, "9. Change aka to use");
|
|
mvprintw(16, 6, "10. Change origin line");
|
|
mvprintw(17, 6, "11. Change netmail reply");
|
|
mvprintw(18, 6, "12. Delete message area");
|
|
|
|
memset(&a1, 0, sizeof(fidoaddr));
|
|
memset(&a2, 0, sizeof(fidoaddr));
|
|
|
|
menu = select_menu(12);
|
|
switch (menu) {
|
|
case 0: return;
|
|
case 1: a1 = PullUplink((char *)"AKA TO DELETE");
|
|
break;
|
|
case 2: a2 = PullUplink((char *)"AKA TO ADD");
|
|
break;
|
|
case 3: a1 = PullUplink((char *)"AKA TO REPLACE");
|
|
a2 = PullUplink((char *)"NEW AKA");
|
|
break;
|
|
case 4: S.aka = PullUplink((char *)"AKA TO CHANGE STATUS");
|
|
ShowStatus(S);
|
|
S.sendto = edit_bool(8,16,S.sendto, (char *)"^Send^ mail to this node");
|
|
S.receivefrom = edit_bool(9,16,S.receivefrom, (char *)"^Receive^ mail from this node");
|
|
S.pause = edit_bool(10,16,S.pause, (char *)"Is this node ^paused^");
|
|
S.cutoff = edit_bool(11,16,S.cutoff, (char *)"Is this node ^excluded^");
|
|
break;
|
|
case 5: mvprintw(LINES -3, 5, "Days old");
|
|
E_INT(LINES -3, 14, daysold, (char *)"Enter new number of ^days old^")
|
|
case 6: mvprintw(LINES -3, 5, "Max. messages");
|
|
E_INT(LINES -3, 19, maxmsgs, (char *)"Enter ^maximum messages^")
|
|
case 7: mvprintw(LINES -3, 6, "Max. articles");
|
|
E_INT(LINES -3, 19, maxarticles, (char *)"Enter ^maximum news articles^ to fetch")
|
|
case 8: rs = edit_sec(6, 5, rs, (char *)"9.2.7 READ SECURITY");
|
|
ws = edit_sec(7, 5, ws, (char *)"9.2.7 WRITE SECURITY");
|
|
ss = edit_sec(8, 5, ss, (char *)"9.2.7 SYSOP SECURITY");
|
|
break;
|
|
case 9: akan = PickAka((char *)"9.2.8", TRUE);
|
|
break;
|
|
case 10:E_STR(LINES -3, 5, 64, mfile, "Enter new ^origin^ line");
|
|
case 11:mvprintw(LINES -3, 5, "Netmail reply board");
|
|
E_INT(LINES -3, 25, netbrd, (char *)"The ^netmail reply^ board number")
|
|
}
|
|
|
|
E_Group(&mgr, (char *)"SELECT MESSAGE GROUPS TO CHANGE");
|
|
|
|
/*
|
|
* Show settings before proceeding
|
|
*/
|
|
switch (menu) {
|
|
case 1: mvprintw(7, 6, "Delete aka %s", aka2str(a1));
|
|
break;
|
|
case 2: mvprintw(7, 6, "Add aka %s", aka2str(a2));
|
|
break;
|
|
case 3: p = xstrcpy(aka2str(a1));
|
|
mvprintw(7, 6, "Replace aka %s with %s", p, aka2str(a2));
|
|
free(p);
|
|
break;
|
|
case 4: ShowStatus(S);
|
|
mvprintw(14, 6, "Change the link status");
|
|
break;
|
|
case 5: mvprintw(7, 6, "Change days old to %d", daysold);
|
|
break;
|
|
case 6: mvprintw(7, 6, "Change maximum messages to %d", maxmsgs);
|
|
break;
|
|
case 7: mvprintw(7, 6, "Change maximum news articles to fetch to %d", maxarticles);
|
|
break;
|
|
case 8: set_color(CYAN, BLACK);
|
|
mvprintw(7, 6, "Read security");
|
|
mvprintw(8, 6, "Write security");
|
|
mvprintw(9, 6, "Sysop security");
|
|
set_color(WHITE, BLACK);
|
|
show_sec(7, 21, rs);
|
|
show_sec(8, 21, ws);
|
|
show_sec(9, 21, ss);
|
|
break;
|
|
case 9: if (akan != -1)
|
|
mvprintw( 7, 6, "Set %s as new aka to use", aka2str(CFG.aka[akan]));
|
|
break;
|
|
case 10:mvprintw(7, 6, "Origin: %s", mfile);
|
|
break;
|
|
case 11:mvprintw(7, 6, "New netmail reply board %d", netbrd);
|
|
break;
|
|
case 12:mvprintw(7, 6, "Delete message areas");
|
|
break;
|
|
}
|
|
|
|
if (yes_no((char *)"Perform changes")) {
|
|
working(1, 0, 0);
|
|
Areas = CountMsgarea();
|
|
Total = Done = 0;
|
|
|
|
for (marea = 1; marea <= Areas; marea++) {
|
|
offset = LoadMsgRec(marea, FALSE);
|
|
if (msgs.Active && strlen(msgs.Group)) {
|
|
for (tmp = mgr; tmp; tmp = tmp->next) {
|
|
if (tmp->tagged && (strcmp(tmp->group, msgs.Group) == 0)) {
|
|
Total++;
|
|
switch (menu) {
|
|
case 1: fseek(tfil, 0, SEEK_SET);
|
|
while (fread(&Sc, sizeof(sysconnect), 1, tfil) == 1) {
|
|
if ((Sc.aka.zone == a1.zone) &&
|
|
(Sc.aka.net == a1.net) &&
|
|
(Sc.aka.node == a1.node) &&
|
|
(Sc.aka.point == a1.point)) {
|
|
fseek(tfil, - sizeof(sysconnect), SEEK_CUR);
|
|
memset(&Sc, 0, sizeof(sysconnect));
|
|
fwrite(&Sc, sizeof(sysconnect), 1, tfil);
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Deleted %s from %s", aka2str(a1), msgs.Tag);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 2: fseek(tfil, 0, SEEK_SET);
|
|
Found = FALSE;
|
|
while (fread(&Sc, sizeof(sysconnect), 1, tfil) == 1)
|
|
if ((Sc.aka.zone == a2.zone) &&
|
|
(Sc.aka.net == a2.net) &&
|
|
(Sc.aka.node == a2.node) &&
|
|
(Sc.aka.point == a2.point)) {
|
|
Found = TRUE;
|
|
break;
|
|
}
|
|
if (Found)
|
|
break;
|
|
fseek(tfil, 0, SEEK_SET);
|
|
while (fread(&Sc, sizeof(sysconnect), 1, tfil) == 1) {
|
|
if (Sc.aka.zone == 0) {
|
|
fseek(tfil, - sizeof(sysconnect), SEEK_CUR);
|
|
memset(&Sc, 0, sizeof(sysconnect));
|
|
Sc.aka.zone = a2.zone;
|
|
Sc.aka.net = a2.net;
|
|
Sc.aka.node = a2.node;
|
|
Sc.aka.point = a2.point;
|
|
Sc.sendto = TRUE;
|
|
Sc.receivefrom = TRUE;
|
|
sprintf(Sc.aka.domain, "%s", a2.domain);
|
|
fwrite(&Sc, sizeof(sysconnect), 1, tfil);
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Added %s to area %s", aka2str(a2), msgs.Tag);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 3: fseek(tfil, 0, SEEK_SET);
|
|
while (fread(&Sc, sizeof(sysconnect), 1, tfil) == 1) {
|
|
if ((Sc.aka.zone == a1.zone) &&
|
|
(Sc.aka.net == a1.net) &&
|
|
(Sc.aka.node == a1.node) &&
|
|
(Sc.aka.point == a1.point)) {
|
|
Sc.aka.zone = a2.zone;
|
|
Sc.aka.net = a2.net;
|
|
Sc.aka.node = a2.node;
|
|
Sc.aka.point = a2.point;
|
|
sprintf(Sc.aka.domain, "%s", a2.domain);
|
|
fseek(tfil, - sizeof(sysconnect), SEEK_CUR);
|
|
fwrite(&Sc, sizeof(sysconnect), 1, tfil);
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
p = xstrcpy(aka2str(a1));
|
|
Syslog('+', "Changed %s into %s in area %s", p, aka2str(a2), msgs.Tag);
|
|
free(p);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 4: fseek(tfil, 0, SEEK_SET);
|
|
while (fread(&Sc, sizeof(sysconnect), 1, tfil) == 1) {
|
|
if ((Sc.aka.zone == S.aka.zone) &&
|
|
(Sc.aka.net == S.aka.net) &&
|
|
(Sc.aka.node == S.aka.node) &&
|
|
(Sc.aka.point == S.aka.point)) {
|
|
Sc.sendto = S.sendto;
|
|
Sc.receivefrom = S.receivefrom;
|
|
Sc.pause = S.pause;
|
|
Sc.cutoff = S.cutoff;
|
|
fseek(tfil, - sizeof(sysconnect), SEEK_CUR);
|
|
fwrite(&Sc, sizeof(sysconnect), 1, tfil);
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Changed status of %s in area %s", aka2str(S.aka), msgs.Tag);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 5: if (daysold != msgs.DaysOld) {
|
|
msgs.DaysOld = daysold;
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Changed days old to %d in area %s", daysold, msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
case 6: if (maxmsgs != msgs.MaxMsgs) {
|
|
msgs.MaxMsgs = maxmsgs;
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Changed max. messages to %d in area %s", maxmsgs, msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
case 7: if (maxarticles != msgs.MaxArticles) {
|
|
msgs.MaxArticles = maxarticles;
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Changed max news articles to fetch to %d in area %s", maxarticles, msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
case 8: if ((msgs.RDSec.level != rs.level) ||
|
|
(msgs.RDSec.flags != rs.flags) ||
|
|
(msgs.RDSec.notflags != rs.notflags) ||
|
|
(msgs.WRSec.level != ws.level) ||
|
|
(msgs.WRSec.flags != ws.flags) ||
|
|
(msgs.WRSec.notflags != ws.notflags) ||
|
|
(msgs.SYSec.level != ss.level) ||
|
|
(msgs.SYSec.flags != ss.flags) ||
|
|
(msgs.SYSec.notflags != ss.notflags)) {
|
|
memcpy(&msgs.RDSec, &rs, sizeof(securityrec));
|
|
memcpy(&msgs.WRSec, &ws, sizeof(securityrec));
|
|
memcpy(&msgs.SYSec, &ss, sizeof(securityrec));
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Updated security levels in area %s", msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
case 9: if (akan != -1) {
|
|
if ((msgs.Aka.zone != CFG.aka[akan].zone) ||
|
|
(msgs.Aka.net != CFG.aka[akan].net) ||
|
|
(msgs.Aka.node != CFG.aka[akan].node) ||
|
|
(msgs.Aka.point != CFG.aka[akan].point)) {
|
|
msgs.Aka.zone = CFG.aka[akan].zone;
|
|
msgs.Aka.net = CFG.aka[akan].net;
|
|
msgs.Aka.node = CFG.aka[akan].node;
|
|
msgs.Aka.point = CFG.aka[akan].point;
|
|
sprintf(msgs.Aka.domain, "%s", CFG.aka[akan].domain);
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Area %s now uses aka %s", msgs.Tag, aka2str(msgs.Aka));
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 10:if (strcmp(msgs.Origin, mfile)) {
|
|
sprintf(msgs.Origin, "%s", mfile);
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Changed origin line in area %s", msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
case 11:if (netbrd != msgs.NetReply) {
|
|
msgs.NetReply = netbrd;
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Changed netmail board to %d in area %s", netbrd, msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
case 12:if (msgs.Active) {
|
|
msgs.Active = FALSE;
|
|
DeleteJAM(msgs.Base);
|
|
memset(&msgs, 0, sizeof(msgs));
|
|
if (SaveMsgRec(marea, FALSE) == 0) {
|
|
Done++;
|
|
Syslog('+', "Deleted message area %s", msgs.Tag);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (tfil != NULL)
|
|
fclose(tfil);
|
|
}
|
|
|
|
working(0, 0, 0);
|
|
mvprintw(LINES -3, 6,"Made %d changes in %d possible areas", Done, Total);
|
|
(void)readkey(LINES -3, 50, LIGHTGRAY, BLACK);
|
|
if (Done)
|
|
MsgUpdated = TRUE;
|
|
}
|
|
}
|
|
|
|
tidy_grlist(&mgr);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Edit one record, return -1 if record doesn't exist, 0 if ok.
|
|
*/
|
|
int EditMsgRec(int);
|
|
int EditMsgRec(int Area)
|
|
{
|
|
unsigned long crc1;
|
|
int tmp, i, connections = 0, changed = FALSE, Active, Forced = FALSE;
|
|
sysconnect System;
|
|
char *temp;
|
|
|
|
clr_index();
|
|
IsDoing("Edit Msg Area");
|
|
|
|
if (LoadMsgRec(Area, TRUE) == -1)
|
|
return -1;
|
|
|
|
SetScreen();
|
|
|
|
for (;;) {
|
|
set_color(WHITE, BLACK);
|
|
show_str( 6,16,40, msgs.Name);
|
|
show_str( 7,16,50, msgs.Tag);
|
|
show_str( 8,16,12, msgs.Group);
|
|
show_str( 9,16,64, msgs.Newsgroup);
|
|
show_str(10,16,64, msgs.Base);
|
|
show_str(11,16,64, msgs.Origin);
|
|
show_aka(12,16, msgs.Aka);
|
|
show_str(13,16,13, msgs.QWKname);
|
|
show_str(14,16,16, msgs.Distribution);
|
|
show_msgtype(15,16, msgs.Type);
|
|
show_msgkinds(16,16, msgs.MsgKinds);
|
|
show_str(17,16,16, printable(getchrs(msgs.Ftncode), 0));
|
|
show_str(18,16,16, printable(getchrs(msgs.Rfccode), 0));
|
|
show_bool(19,16, msgs.Active);
|
|
|
|
show_int( 13,52, msgs.DaysOld);
|
|
show_int( 14,52, msgs.MaxMsgs);
|
|
switch (msgs.Type) {
|
|
case ECHOMAIL: show_int( 15,52, msgs.NetReply);
|
|
break;
|
|
case NEWS: show_int( 15,52, msgs.MaxArticles);
|
|
break;
|
|
}
|
|
show_int( 16,52, msgs.RDSec.level);
|
|
show_int( 17,52, msgs.WRSec.level);
|
|
show_int( 18,52, msgs.SYSec.level);
|
|
show_bool(19,52, msgs.UsrDelete);
|
|
|
|
show_bool(13,76, msgs.Aliases);
|
|
show_bool(14,76, msgs.Quotes);
|
|
show_bool(15,76, msgs.Mandatory);
|
|
show_bool(16,76, msgs.UnSecure);
|
|
show_bool(17,76, msgs.OLR_Default);
|
|
show_bool(18,76, msgs.OLR_Forced);
|
|
connections = 0;
|
|
switch (msgs.Type) {
|
|
case ECHOMAIL:
|
|
case NEWS:
|
|
case LIST: fseek(tfil, 0, SEEK_SET);
|
|
while (fread(&System, sizeof(System), 1, tfil) == 1)
|
|
if (System.aka.zone)
|
|
connections++;
|
|
show_int(19,76, connections);
|
|
break;
|
|
}
|
|
|
|
switch (select_menu(28)) {
|
|
case 0:
|
|
crc1 = 0xffffffff;
|
|
crc1 = upd_crc32((char *)&msgs, crc1, msgshdr.recsize);
|
|
fseek(tfil, 0, 0);
|
|
for (i = 0; i < (msgshdr.syssize / sizeof(sysconnect)); i++) {
|
|
fread(&System, sizeof(sysconnect), 1, tfil);
|
|
crc1 = upd_crc32((char *)&System, crc1, sizeof(sysconnect));
|
|
}
|
|
if (msgs.Active)
|
|
test_jam(msgs.Base);
|
|
if ((MsgCrc != crc1) || (changed)) {
|
|
if (msgs.Active && !strlen(msgs.Base)) {
|
|
errmsg((char *)"JAM message base is not set");
|
|
break;
|
|
} else if (msgs.Active && !strlen(msgs.Group) &&
|
|
(msgs.Type == ECHOMAIL || msgs.Type == NEWS || msgs.Type == LIST)) {
|
|
errmsg((char *)"Message area has no group assigned");
|
|
break;
|
|
} else if (Forced || yes_no((char *)"Record is changed, save") == 1) {
|
|
if (SaveMsgRec(Area, TRUE) == -1)
|
|
return -1;
|
|
MsgUpdated = 1;
|
|
Syslog('+', "Saved message area record %d", Area);
|
|
}
|
|
}
|
|
IsDoing("Browsing Menu");
|
|
return 0;
|
|
case 1: E_STR( 6,16,40,msgs.Name, "The ^Name^ of this area")
|
|
case 2: strcpy(msgs.Tag, edit_ups(7,16,50, msgs.Tag, (char *)"The ^Area Tag^ for Echomail"));
|
|
if (!strlen(msgs.QWKname)) {
|
|
memset(&msgs.QWKname, '\0', strlen(msgs.QWKname));
|
|
strncpy(msgs.QWKname, msgs.Tag, 13);
|
|
}
|
|
break;
|
|
case 3: strcpy(msgs.Group, PickMGroup((char *)"9.2.3"));
|
|
if (strlen(msgs.Group)) {
|
|
msgs.Aka = mgroup.UseAka;
|
|
msgs.Active = TRUE;
|
|
/*
|
|
* If there is an uplink defined in the group,
|
|
* and the first connected system is empty,
|
|
* copy the uplink as default connection.
|
|
*/
|
|
if (mgroup.UpLink.zone) {
|
|
fseek(tfil, 0, SEEK_SET);
|
|
fread(&System, sizeof(sysconnect), 1, tfil);
|
|
if (!System.aka.zone) {
|
|
memset(&System, 0, sizeof(sysconnect));
|
|
System.aka = mgroup.UpLink;
|
|
System.sendto = TRUE;
|
|
System.receivefrom = TRUE;
|
|
fseek(tfil, 0, SEEK_SET);
|
|
fwrite(&System, sizeof(sysconnect), 1, tfil);
|
|
}
|
|
}
|
|
if (!strlen(msgs.Base) && strlen(msgs.Tag)) {
|
|
/*
|
|
* Invent the place for the JAM message base.
|
|
*/
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
sprintf(temp, "%s/%s", msgs.Group, msgs.Tag);
|
|
for (i = 0; i < strlen(temp); i++) {
|
|
if (isupper(temp[i]))
|
|
temp[i] = tolower(temp[i]);
|
|
/*
|
|
* If dots in the group or area tag, replace them
|
|
* with directory slashes to create a group tree.
|
|
*/
|
|
if (temp[i] == '.')
|
|
temp[i] = '/';
|
|
}
|
|
sprintf(msgs.Base, "%s/var/mail/%s", getenv("MBSE_ROOT"), temp);
|
|
free(temp);
|
|
}
|
|
}
|
|
SetScreen();
|
|
break;
|
|
case 4: E_STR( 9,16,64,msgs.Newsgroup, "The ^Newsgroup^ name of this area")
|
|
case 5: strcpy(msgs.Base, edit_jam(10,16,64,msgs.Base ,(char *)"The path to the ^JAM Message Base^"));
|
|
Forced = TRUE;
|
|
MailForced = TRUE;
|
|
break;
|
|
case 6: E_STR( 11,16,64,msgs.Origin, "The ^Origin line^ to append to Echomail messages")
|
|
case 7: tmp = PickAka((char *)"9.2.7", TRUE);
|
|
if (tmp != -1)
|
|
msgs.Aka = CFG.aka[tmp];
|
|
SetScreen(); break;
|
|
case 8: E_UPS( 13,16,13,msgs.QWKname, "The name for ^QWK or Bluewave^ message packets")
|
|
case 9: E_STR( 14,16,16,msgs.Distribution, "The ^Distribution^ name if this is a newsgroup")
|
|
case 10:msgs.Type = edit_msgtype(15,16, msgs.Type);
|
|
SetScreen();
|
|
break;
|
|
case 11:msgs.MsgKinds = edit_msgkinds(16,16, msgs.MsgKinds); break;
|
|
case 14:Active = edit_bool(19,16, msgs.Active, (char *)"Is this area ^Active^");
|
|
if (msgs.Active && !Active) {
|
|
/*
|
|
* Attempt to deactivate area, do some checks.
|
|
*/
|
|
if (connections) {
|
|
if (yes_no((char *)"There are nodes connected, disconnect them") == 0)
|
|
Active = TRUE;
|
|
}
|
|
if (!Active) {
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
sprintf(temp, "%s.jhr", msgs.Base);
|
|
if (strlen(msgs.Base) && (file_size(temp) != 1024)) {
|
|
if (yes_no((char *)"There are messages in this area, delete them") == 0)
|
|
Active = TRUE;
|
|
}
|
|
free(temp);
|
|
}
|
|
if (!Active) {
|
|
/*
|
|
* Make it so
|
|
*/
|
|
DeleteJAM(msgs.Base);
|
|
memset(&System, 0, sizeof(System));
|
|
fseek(tfil, 0, SEEK_SET);
|
|
for (i = 0; i < (msgshdr.syssize / sizeof(sysconnect)); i++)
|
|
fwrite(&System, sizeof(System), 1, tfil);
|
|
InitMsgRec();
|
|
Syslog('+', "Deleted message area %d", Area);
|
|
changed = TRUE;
|
|
}
|
|
}
|
|
if (!msgs.Active && Active)
|
|
msgs.Active = TRUE;
|
|
SetScreen();
|
|
break;
|
|
case 15:E_INT( 13,52, msgs.DaysOld, "Maximum ^days^ to keep mail in this area")
|
|
case 16:E_INT( 14,52, msgs.MaxMsgs, "The ^maximum^ amount of messages in this area")
|
|
case 17:switch (msgs.Type) {
|
|
case ECHOMAIL: msgs.NetReply = edit_int(15,52,msgs.NetReply,
|
|
(char *)"The ^Area Number^ for netmail replies");
|
|
break;
|
|
case NEWS: msgs.MaxArticles = edit_int(15,52,msgs.MaxArticles,
|
|
(char *)"The ^maximum news articles^ to fetch");
|
|
break;
|
|
}
|
|
break;
|
|
case 18:E_SEC( 16,52, msgs.RDSec, "9.2 EDIT READ SECURITY", SetScreen)
|
|
case 19:E_SEC( 17,52, msgs.WRSec, "9.2 EDIT WRITE SECURITY", SetScreen)
|
|
case 20:E_SEC( 18,52, msgs.SYSec, "9.2 EDIT SYSOP SECURITY", SetScreen)
|
|
case 21:E_BOOL(19,52, msgs.UsrDelete, "Allow users to ^Delete^ their messages")
|
|
|
|
case 22:E_BOOL(13,76, msgs.Aliases, "Allow ^aliases^ or real names only")
|
|
case 23:E_BOOL(14,76, msgs.Quotes, "Add random ^quotes^ to new messages")
|
|
case 24:E_BOOL(15,76, msgs.Mandatory, "Is this area ^mandatory^ for nodes")
|
|
case 25:E_BOOL(16,76, msgs.UnSecure, "Toss messages ^UnSecure^, ie: no originating check")
|
|
case 26:E_BOOL(17,76, msgs.OLR_Default, "Area is ^default^ for ^offline^ users.")
|
|
case 27:E_BOOL(18,76, msgs.OLR_Forced, "Area is ^always on^ for ^offline^ users.")
|
|
case 28:switch (msgs.Type) {
|
|
case ECHOMAIL:
|
|
case NEWS:
|
|
case LIST: if (EditConnections(tfil))
|
|
changed = TRUE;
|
|
SetScreen();
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void EditMsgarea(void)
|
|
{
|
|
int records, i, o, y;
|
|
char pick[12];
|
|
FILE *fil;
|
|
char temp[PATH_MAX];
|
|
long offset;
|
|
int from, too;
|
|
sysconnect System;
|
|
|
|
clr_index();
|
|
working(1, 0, 0);
|
|
IsDoing("Browsing Menu");
|
|
if (config_read() == -1) {
|
|
working(2, 0, 0);
|
|
return;
|
|
}
|
|
|
|
records = CountMsgarea();
|
|
if (records == -1) {
|
|
working(2, 0, 0);
|
|
return;
|
|
}
|
|
|
|
if (OpenMsgarea() == -1) {
|
|
working(2, 0, 0);
|
|
return;
|
|
}
|
|
working(0, 0, 0);
|
|
o = 0;
|
|
|
|
for (;;) {
|
|
clr_index();
|
|
set_color(WHITE, BLACK);
|
|
mvprintw( 5, 3, "9.2 MESSAGE AREA SETUP");
|
|
set_color(CYAN, BLACK);
|
|
if (records != 0) {
|
|
sprintf(temp, "%s/etc/mareas.temp", getenv("MBSE_ROOT"));
|
|
working(1, 0, 0);
|
|
if ((fil = fopen(temp, "r")) != NULL) {
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fil);
|
|
y = 7;
|
|
set_color(CYAN, BLACK);
|
|
for (i = 1; i <= 10; i++) {
|
|
if ((o + i) <= records) {
|
|
offset = sizeof(msgshdr) + (((o + i) - 1) * (msgshdr.recsize + msgshdr.syssize));
|
|
fseek(fil, offset, 0);
|
|
fread(&msgs, msgshdr.recsize, 1, fil);
|
|
if (msgs.Active) {
|
|
set_color(CYAN, BLACK);
|
|
sprintf(temp, "%3d. %-8s %-25s %-40s", o + i, getmsgtype(msgs.Type), msgs.Tag, msgs.Name);
|
|
} else {
|
|
set_color(LIGHTBLUE, BLACK);
|
|
sprintf(temp, "%3d.", o+i);
|
|
}
|
|
mvprintw(y, 2, temp);
|
|
y++;
|
|
}
|
|
}
|
|
fclose(fil);
|
|
}
|
|
}
|
|
working(0, 0, 0);
|
|
strcpy(pick, select_area(records, 10));
|
|
|
|
if (strncmp(pick, "-", 1) == 0) {
|
|
CloseMsgarea(MailForced);
|
|
return;
|
|
}
|
|
|
|
if (strncmp(pick, "A", 1) == 0) {
|
|
working(1, 0, 0);
|
|
if (AppendMsgarea() == 0) {
|
|
records++;
|
|
working(1, 0, 0);
|
|
} else
|
|
working(2, 0, 0);
|
|
working(0, 0, 0);
|
|
}
|
|
|
|
if (strncmp(pick, "G", 1) == 0) {
|
|
MsgGlobal();
|
|
}
|
|
|
|
if (strncmp(pick, "N", 1) == 0)
|
|
if ((o + 10) < records)
|
|
o = o + 10;
|
|
|
|
if (strncmp(pick, "P", 1) == 0)
|
|
if ((o - 10) >= 0)
|
|
o = o - 10;
|
|
|
|
if (strncmp(pick, "M", 1) == 0) {
|
|
from = too = 0;
|
|
mvprintw(LINES -3, 5, "From");
|
|
from = edit_int(LINES -3, 10, from, (char *)"Wich ^area^ you want to move");
|
|
mvprintw(LINES -3,15, "To");
|
|
too = edit_int(LINES -3, 18, too, (char *)"Too which ^area^ to move");
|
|
if ((LoadMsgRec(from, TRUE) == -1) || (!msgs.Active)) {
|
|
errmsg((char *)"The originating area is invalid");
|
|
fclose(tfil);
|
|
} else {
|
|
fclose(tfil);
|
|
if ((LoadMsgRec(too, TRUE) == -1) || (msgs.Active)) {
|
|
errmsg((char *)"The destination area is invalid");
|
|
fclose(tfil);
|
|
} else {
|
|
fclose(tfil);
|
|
LoadMsgRec(from, TRUE);
|
|
SaveMsgRec(too, TRUE);
|
|
LoadMsgRec(from, TRUE);
|
|
InitMsgRec();
|
|
fseek(tfil, 0, SEEK_SET);
|
|
memset(&System, 0, sizeof(sysconnect));
|
|
for (i = 0; i < (msgshdr.syssize / sizeof(sysconnect)); i++) {
|
|
fwrite(&System, sizeof(sysconnect), 1, tfil);
|
|
}
|
|
SaveMsgRec(from, TRUE);
|
|
MsgUpdated = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((atoi(pick) >= 1) && (atoi(pick) <= records)) {
|
|
EditMsgRec(atoi(pick));
|
|
o = ((atoi(pick) - 1) / 10) * 10;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void InitMsgarea(void)
|
|
{
|
|
CountMsgarea();
|
|
OpenMsgarea();
|
|
CloseMsgarea(TRUE);
|
|
}
|
|
|
|
|
|
|
|
char *PickMsgarea(char *shdr)
|
|
{
|
|
int records, i, o = 0, x, y;
|
|
char pick[12];
|
|
FILE *fil;
|
|
char temp[PATH_MAX];
|
|
long offset;
|
|
static char Buf[81];
|
|
|
|
clr_index();
|
|
working(1, 0, 0);
|
|
if (config_read() == -1) {
|
|
working(2, 0, 0);
|
|
return '\0';
|
|
}
|
|
|
|
records = CountMsgarea();
|
|
if (records == -1) {
|
|
working(2, 0, 0);
|
|
return '\0';
|
|
}
|
|
|
|
working(0, 0, 0);
|
|
|
|
for (;;) {
|
|
clr_index();
|
|
set_color(WHITE, BLACK);
|
|
sprintf(temp, "%s. MESSAGE AREA SELECT", shdr);
|
|
mvprintw(5, 3, temp);
|
|
set_color(CYAN, BLACK);
|
|
|
|
if (records) {
|
|
sprintf(temp, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
working(1, 0, 0);
|
|
if ((fil = fopen(temp, "r")) != NULL) {
|
|
fread(&msgshdr, sizeof(msgshdr), 1, fil);
|
|
x = 2;
|
|
y = 7;
|
|
set_color(CYAN, BLACK);
|
|
for (i = 1; i <= 20; i++) {
|
|
if (i == 11) {
|
|
x = 42;
|
|
y = 7;
|
|
}
|
|
if ((o + i) <= records) {
|
|
offset = sizeof(msgshdr) + (((o + i) - 1) * (msgshdr.recsize + msgshdr.syssize));
|
|
fseek(fil, offset, SEEK_SET);
|
|
fread(&msgs, msgshdr.recsize, 1, fil);
|
|
if (msgs.Active)
|
|
set_color(CYAN, BLACK);
|
|
else
|
|
set_color(LIGHTBLUE, BLACK);
|
|
sprintf(temp, "%3d. %-31s", o + i, msgs.Name);
|
|
temp[38] = '\0';
|
|
mvprintw(y, x, temp);
|
|
y++;
|
|
}
|
|
}
|
|
fclose(fil);
|
|
}
|
|
}
|
|
working(0, 0, 0);
|
|
strcpy(pick, select_pick(records, 20));
|
|
|
|
if (strncmp(pick, "-", 1) == 0)
|
|
return '\0';
|
|
|
|
if (strncmp(pick, "N", 1) == 0)
|
|
if ((o + 20) < records)
|
|
o += 20;
|
|
|
|
if (strncmp(pick, "P", 1) == 0)
|
|
if ((o - 20) >= 0)
|
|
o -= 20;
|
|
|
|
if ((atoi(pick) >= 1) && (atoi(pick) <= records)) {
|
|
sprintf(temp, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((fil = fopen(temp, "r")) != NULL) {
|
|
offset = msgshdr.hdrsize + ((atoi(pick) - 1) * (msgshdr.recsize + msgshdr.syssize));
|
|
fseek(fil, offset, SEEK_SET);
|
|
fread(&msgs, msgshdr.recsize, 1, fil);
|
|
fclose(fil);
|
|
if (msgs.Active) {
|
|
memset(&Buf, 0, sizeof(Buf));
|
|
sprintf(Buf, "%s", msgs.Base);
|
|
return Buf;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int GroupInMarea(char *Group)
|
|
{
|
|
int Area = 0, RetVal = 0, systems;
|
|
FILE *no;
|
|
char temp[PATH_MAX];
|
|
|
|
sprintf(temp, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((no = fopen(temp, "r")) == NULL)
|
|
return 0;
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, no);
|
|
fseek(no, 0, SEEK_SET);
|
|
fread(&msgshdr, msgshdr.hdrsize, 1, no);
|
|
systems = msgshdr.syssize / sizeof(sysconnect);
|
|
|
|
while (fread(&msgs, msgshdr.recsize, 1, no) == 1) {
|
|
Area++;
|
|
|
|
if (msgs.Active && !strcmp(msgs.Group, Group) && strlen(msgs.Group)) {
|
|
RetVal++;
|
|
Syslog('-', "Group %s in msg area %d: %s", Group, Area, msgs.Tag);
|
|
}
|
|
|
|
fseek(no, msgshdr.syssize, SEEK_CUR);
|
|
}
|
|
|
|
fclose(no);
|
|
return RetVal;
|
|
}
|
|
|
|
|
|
|
|
int NodeInMarea(fidoaddr A)
|
|
{
|
|
int i, Area = 0, RetVal = 0, systems;
|
|
FILE *no;
|
|
char temp[PATH_MAX];
|
|
sysconnect S;
|
|
|
|
sprintf(temp, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((no = fopen(temp, "r")) == NULL)
|
|
return 0;
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, no);
|
|
fseek(no, 0, SEEK_SET);
|
|
fread(&msgshdr, msgshdr.hdrsize, 1, no);
|
|
systems = msgshdr.syssize / sizeof(sysconnect);
|
|
|
|
while (fread(&msgs, msgshdr.recsize, 1, no) == 1) {
|
|
Area++;
|
|
for (i = 0; i < systems; i++) {
|
|
fread(&S, sizeof(sysconnect), 1, no);
|
|
if (S.aka.zone && (S.aka.zone == A.zone) && (S.aka.net == A.net) &&
|
|
(S.aka.node == A.node) && (S.aka.point == A.point) && msgs.Active) {
|
|
RetVal++;
|
|
Syslog('-', "Node %s connected to msg area %d: %s", aka2str(A), Area, msgs.Tag);
|
|
}
|
|
}
|
|
}
|
|
|
|
fclose(no);
|
|
return RetVal;
|
|
}
|
|
|
|
|
|
|
|
void gold_areas(FILE *fp)
|
|
{
|
|
char *temp, *aka;
|
|
FILE *no;
|
|
int i = 0;
|
|
|
|
temp = calloc(PATH_MAX, sizeof(char));
|
|
sprintf(temp, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((no = fopen(temp, "r")) == NULL)
|
|
return;
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, no);
|
|
fseek(no, 0, SEEK_SET);
|
|
fread(&msgshdr, msgshdr.hdrsize, 1, no);
|
|
|
|
fprintf(fp, "; Message Areas\n;\n");
|
|
fprintf(fp, "AREASCAN *\n");
|
|
fprintf(fp, "AREATYPEORDER Net Email Echo News Local\n");
|
|
|
|
while (fread(&msgs, msgshdr.recsize, 1, no) == 1) {
|
|
|
|
i++;
|
|
if (msgs.Active) {
|
|
fprintf(fp, "AREADEF ");
|
|
if (strlen(msgs.Tag))
|
|
fprintf(fp, "%s", msgs.Tag);
|
|
else
|
|
fprintf(fp, "AREA%d", i);
|
|
fprintf(fp, " \"%s\" ", msgs.Name);
|
|
switch (msgs.Type) {
|
|
case LOCALMAIL : fprintf(fp, "0 Local"); break;
|
|
case NETMAIL : fprintf(fp, "N Net"); break;
|
|
case ECHOMAIL : fprintf(fp, "C Echo"); break;
|
|
case NEWS : fprintf(fp, "I News"); break;
|
|
}
|
|
aka = xstrcpy(strtok(aka2str(msgs.Aka), "@"));
|
|
fprintf(fp, " JAM %s %s ", msgs.Base, aka);
|
|
free(aka);
|
|
if (msgs.Type == NETMAIL)
|
|
fprintf(fp, "(Loc Pvt)");
|
|
else
|
|
fprintf(fp, "(Loc)");
|
|
fprintf(fp, " \"%s\"\n", msgs.Origin);
|
|
}
|
|
fseek(no, msgshdr.syssize, SEEK_CUR);
|
|
}
|
|
|
|
fclose(no);
|
|
free(temp);
|
|
fprintf(fp, "\n");
|
|
}
|
|
|
|
|
|
|
|
int mail_area_doc(FILE *fp, FILE *toc, int page)
|
|
{
|
|
char temp[PATH_MAX], status[5];
|
|
FILE *no;
|
|
int i = 0, j, systems, First = TRUE;
|
|
sysconnect System;
|
|
int LMiy;
|
|
struct tm *t;
|
|
time_t Now;
|
|
|
|
Now = time(NULL);
|
|
t = localtime(&Now);
|
|
|
|
Diw = t->tm_wday;
|
|
Miy = t->tm_mon;
|
|
if (Miy == 0)
|
|
LMiy = 11;
|
|
else
|
|
LMiy = Miy - 1;
|
|
|
|
sprintf(temp, "%s/etc/mareas.data", getenv("MBSE_ROOT"));
|
|
if ((no = fopen(temp, "r")) == NULL)
|
|
return page;
|
|
|
|
fread(&msgshdr, sizeof(msgshdr), 1, no);
|
|
fseek(no, 0, SEEK_SET);
|
|
fread(&msgshdr, msgshdr.hdrsize, 1, no);
|
|
systems = msgshdr.syssize / sizeof(sysconnect);
|
|
|
|
while (fread(&msgs, msgshdr.recsize, 1, no) == 1) {
|
|
|
|
i++;
|
|
if (msgs.Active) {
|
|
page = newpage(fp, page);
|
|
|
|
if (First) {
|
|
addtoc(fp, toc, 9, 2, page, (char *)"Mail areas");
|
|
First = FALSE;
|
|
fprintf(fp, "\n");
|
|
} else
|
|
fprintf(fp, "\n\n");
|
|
|
|
fprintf(fp, " Area number %d\n", i);
|
|
fprintf(fp, " Area name %s\n", msgs.Name);
|
|
fprintf(fp, " Area tag %s\n", msgs.Tag);
|
|
fprintf(fp, " Newsgroup name %s\n", msgs.Newsgroup);
|
|
fprintf(fp, " Distribution %s\n", msgs.Distribution);
|
|
fprintf(fp, " JAM message base %s\n", msgs.Base);
|
|
fprintf(fp, " Offline name %s\n", msgs.QWKname);
|
|
fprintf(fp, " Area type %s\n", getmsgtype(msgs.Type));
|
|
fprintf(fp, " Messages type %s\n", getmsgkinds(msgs.MsgKinds));
|
|
fprintf(fp, " FTN charset %s\n", printable(getchrs(msgs.Ftncode), 0));
|
|
fprintf(fp, " RFC charset %s\n", printable(getchrs(msgs.Rfccode), 0));
|
|
fprintf(fp, " Days old msgs. %d\n", msgs.DaysOld);
|
|
fprintf(fp, " Maximum msgs. %d\n", msgs.MaxMsgs);
|
|
fprintf(fp, " Max articles %d\n", msgs.MaxArticles);
|
|
fprintf(fp, " Users delete %s\n", getboolean(msgs.UsrDelete));
|
|
fprintf(fp, " Read security %s\n", get_secstr(msgs.RDSec));
|
|
fprintf(fp, " Write security %s\n", get_secstr(msgs.WRSec));
|
|
fprintf(fp, " Sysop security %s\n", get_secstr(msgs.SYSec));
|
|
fprintf(fp, " Minimum age %d\n", msgs.Age);
|
|
fprintf(fp, " Password %s\n", msgs.Password);
|
|
fprintf(fp, " Group %s\n", msgs.Group);
|
|
fprintf(fp, " Fido address %s\n", aka2str(msgs.Aka));
|
|
fprintf(fp, " Netmail board %d\n", msgs.NetReply);
|
|
fprintf(fp, " Origin line %s\n", msgs.Origin);
|
|
fprintf(fp, " Allow aliases %s\n", getboolean(msgs.Aliases));
|
|
fprintf(fp, " OLR mandatory %s\n", getboolean(msgs.OLR_Forced));
|
|
fprintf(fp, " OLR default on %s\n", getboolean(msgs.OLR_Default));
|
|
fprintf(fp, " Append quotes %s\n", getboolean(msgs.Quotes));
|
|
fprintf(fp, " Nodes mandatory %s\n", getboolean(msgs.Mandatory));
|
|
fprintf(fp, " UnSecure toss %s\n", getboolean(msgs.UnSecure));
|
|
fprintf(fp, " Last msg rcvd. %s", ctime(&msgs.LastRcvd));
|
|
fprintf(fp, " Last msg posted %s", ctime(&msgs.LastPosted));
|
|
|
|
for (j = 0; j < systems; j++) {
|
|
fread(&System, sizeof(sysconnect), 1, no);
|
|
if (System.aka.zone) {
|
|
memset(&status, 0, 5);
|
|
memset(&status, '-', 4);
|
|
if (System.sendto)
|
|
status[0] = 'S';
|
|
if (System.receivefrom)
|
|
status[1] = 'R';
|
|
if (System.pause)
|
|
status[2] = 'P';
|
|
if (System.cutoff)
|
|
status[3] = 'C';
|
|
|
|
fprintf(fp, " Link %2d %s %s\n", j+1, status, aka2str(System.aka));
|
|
}
|
|
}
|
|
fprintf(fp, "\n");
|
|
fprintf(fp, " Total This Month Last Month\n");
|
|
fprintf(fp, " ---------- ---------- ----------\n");
|
|
fprintf(fp, " Msgs received %-10ld %-10ld %-10ld\n", msgs.Received.total, msgs.Received.month[Miy], msgs.Received.month[LMiy]);
|
|
fprintf(fp, " Msgs posted %-10ld %-10ld %-10ld\n", msgs.Posted.total, msgs.Posted.month[Miy], msgs.Posted.month[LMiy]);
|
|
|
|
} else
|
|
fseek(no, msgshdr.syssize, SEEK_CUR);
|
|
}
|
|
|
|
fclose(no);
|
|
return page;
|
|
}
|
|
|
|
|