575 lines
13 KiB
C
575 lines
13 KiB
C
|
/*****************************************************************************
|
||
|
*
|
||
|
* $Id: mbdiff.c,v 1.32 2008/11/26 22:12:28 mbse Exp $
|
||
|
* Purpose ...............: Nodelist diff processor
|
||
|
* Original ideas ........: Eugene G. Crosser.
|
||
|
*
|
||
|
*****************************************************************************
|
||
|
* Copyright (C) 1997-2008
|
||
|
*
|
||
|
* 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 "mbdiff.h"
|
||
|
|
||
|
|
||
|
|
||
|
#ifndef BLKSIZ
|
||
|
#define BLKSIZ 512
|
||
|
#endif
|
||
|
|
||
|
extern unsigned short crc16xmodemtab[];
|
||
|
#define updcrc(cp, crc) ( crc16xmodemtab[((crc >> 8) & 255) ^ cp] ^ (crc << 8))
|
||
|
|
||
|
extern int show_log;
|
||
|
extern int e_pid;
|
||
|
extern int do_quiet; /* Suppress screen output */
|
||
|
extern int show_log; /* Show logging */
|
||
|
time_t t_start; /* Start time */
|
||
|
time_t t_end; /* End time */
|
||
|
|
||
|
|
||
|
|
||
|
void ProgName(void)
|
||
|
{
|
||
|
if (do_quiet)
|
||
|
return;
|
||
|
|
||
|
mbse_colour(WHITE, BLACK);
|
||
|
printf("\nMBDIFF: MBSE BBS %s Nodelist Diff Processor\n", VERSION);
|
||
|
mbse_colour(YELLOW, BLACK);
|
||
|
printf(" %s\n", COPYRIGHT);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void die(int onsig)
|
||
|
{
|
||
|
/*
|
||
|
* First check if a child is running, if so, kill it.
|
||
|
*/
|
||
|
if (e_pid) {
|
||
|
if ((kill(e_pid, SIGTERM)) == 0)
|
||
|
Syslog('+', "SIGTERM to pid %d succeeded", e_pid);
|
||
|
else {
|
||
|
if ((kill(e_pid, SIGKILL)) == 0)
|
||
|
Syslog('+', "SIGKILL to pid %d succeded", e_pid);
|
||
|
else
|
||
|
WriteError("$Failed to kill pid %d", e_pid);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* In case the child had the tty in raw mode...
|
||
|
*/
|
||
|
execute_pth((char *)"stty", (char *)"sane", (char *)"/dev/null", (char *)"/dev/null", (char *)"/dev/null");
|
||
|
}
|
||
|
|
||
|
signal(onsig, SIG_IGN);
|
||
|
|
||
|
if (onsig) {
|
||
|
if (onsig <= NSIG)
|
||
|
WriteError("Terminated on signal %d (%s)", onsig, SigName[onsig]);
|
||
|
else
|
||
|
WriteError("Terminated with error %d", onsig);
|
||
|
}
|
||
|
|
||
|
t_end = time(NULL);
|
||
|
Syslog(' ', "MBDIFF finished in %s", t_elapsed(t_start, t_end));
|
||
|
|
||
|
if (!do_quiet) {
|
||
|
mbse_colour(LIGHTGRAY, BLACK);
|
||
|
printf("\n");
|
||
|
}
|
||
|
ExitClient(onsig);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
int i, Match, rc;
|
||
|
char *cmd, *nl = NULL, *nd = NULL, *nn, *p, *q, *arc, *wrk, *onl, *ond;
|
||
|
struct passwd *pw;
|
||
|
DIR *dp;
|
||
|
struct dirent *de;
|
||
|
|
||
|
InitConfig();
|
||
|
mbse_TermInit(1, 80, 25);
|
||
|
t_start = time(NULL);
|
||
|
umask(002);
|
||
|
|
||
|
/*
|
||
|
* Catch all signals we can, and ignore the rest.
|
||
|
*/
|
||
|
for (i = 0; i < NSIG; i++) {
|
||
|
if ((i == SIGHUP) || (i == SIGINT) || (i == SIGBUS) || (i == SIGILL) || (i == SIGSEGV) || (i == SIGIOT))
|
||
|
signal(i, (void (*))die);
|
||
|
else if (i == SIGCHLD)
|
||
|
signal(i, SIG_DFL);
|
||
|
else if ((i != SIGKILL) && (i != SIGSTOP))
|
||
|
signal(i, SIG_IGN);
|
||
|
}
|
||
|
|
||
|
if(argc < 3)
|
||
|
Help();
|
||
|
|
||
|
cmd = xstrcpy((char *)"Cmd: mbdiff");
|
||
|
|
||
|
for (i = 1; i < argc; i++) {
|
||
|
|
||
|
cmd = xstrcat(cmd, (char *)" ");
|
||
|
cmd = xstrcat(cmd, argv[i]);
|
||
|
|
||
|
if (i == 1)
|
||
|
if ((nl = argv[i]) == NULL)
|
||
|
Help();
|
||
|
if (i == 2)
|
||
|
if ((nd = argv[i]) == NULL)
|
||
|
Help();
|
||
|
if (!strncasecmp(argv[i], "-q", 2))
|
||
|
do_quiet = TRUE;
|
||
|
|
||
|
}
|
||
|
|
||
|
ProgName();
|
||
|
pw = getpwuid(getuid());
|
||
|
InitClient(pw->pw_name, (char *)"mbdiff", CFG.location, CFG.logfile,
|
||
|
CFG.util_loglevel, CFG.error_log, CFG.mgrlog, CFG.debuglog);
|
||
|
|
||
|
Syslog(' ', " ");
|
||
|
Syslog(' ', "MBDIFF v%s", VERSION);
|
||
|
Syslog(' ', cmd);
|
||
|
free(cmd);
|
||
|
|
||
|
if (!do_quiet) {
|
||
|
mbse_colour(LIGHTRED, BLACK);
|
||
|
printf("\n");
|
||
|
}
|
||
|
|
||
|
if (enoughspace(CFG.freespace) == 0)
|
||
|
die(MBERR_DISK_FULL);
|
||
|
|
||
|
/*
|
||
|
* Extract work directory from the first commandline parameter
|
||
|
* and set that directory as default.
|
||
|
*/
|
||
|
show_log = TRUE;
|
||
|
wrk = xstrcpy(nl);
|
||
|
if (strrchr(wrk, '/') == NULL) {
|
||
|
WriteError("No path in nodelist name");
|
||
|
free(wrk);
|
||
|
die(MBERR_COMMANDLINE);
|
||
|
}
|
||
|
if (strrchr(wrk, '.') != NULL) {
|
||
|
WriteError("Filename extension given for nodelist");
|
||
|
free(wrk);
|
||
|
die(MBERR_COMMANDLINE);
|
||
|
}
|
||
|
if (strrchr(nd, '/') == NULL) {
|
||
|
WriteError("No path in nodediff name");
|
||
|
free(wrk);
|
||
|
die(MBERR_COMMANDLINE);
|
||
|
}
|
||
|
show_log = FALSE;
|
||
|
|
||
|
while (wrk[strlen(wrk) -1] != '/')
|
||
|
wrk[strlen(wrk) -1] = '\0';
|
||
|
wrk[strlen(wrk) -1] = '\0';
|
||
|
|
||
|
show_log = TRUE;
|
||
|
if (access(wrk, R_OK|W_OK)) {
|
||
|
WriteError("$No R/W access in %s", wrk);
|
||
|
free(wrk);
|
||
|
die(MBERR_INIT_ERROR);
|
||
|
}
|
||
|
|
||
|
if (chdir(wrk)) {
|
||
|
WriteError("$Can't chdir to %s", wrk);
|
||
|
free(wrk);
|
||
|
die(MBERR_INIT_ERROR);
|
||
|
}
|
||
|
show_log = FALSE;
|
||
|
|
||
|
onl = xstrcpy(strrchr(nl, '/') + 1);
|
||
|
onl = xstrcat(onl, (char *)".???");
|
||
|
|
||
|
if ((dp = opendir(wrk)) == 0) {
|
||
|
show_log = TRUE;
|
||
|
free(wrk);
|
||
|
WriteError("$Error opening directory %s", wrk);
|
||
|
die(MBERR_INIT_ERROR);
|
||
|
}
|
||
|
|
||
|
Match = FALSE;
|
||
|
while ((de = readdir(dp))) {
|
||
|
if (strlen(de->d_name) == strlen(onl)) {
|
||
|
Match = TRUE;
|
||
|
for (i = 0; i < strlen(onl); i++) {
|
||
|
if ((onl[i] != '?') && (onl[i] != de->d_name[i]))
|
||
|
Match = FALSE;
|
||
|
}
|
||
|
if (Match) {
|
||
|
free(onl);
|
||
|
onl = xstrcpy(de->d_name);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
closedir(dp);
|
||
|
if (!Match) {
|
||
|
show_log = TRUE;
|
||
|
free(wrk);
|
||
|
free(onl);
|
||
|
WriteError("Old nodelist not found");
|
||
|
die(MBERR_INIT_ERROR);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Now try to get the diff file into the workdir.
|
||
|
*/
|
||
|
if ((arc = unpacker(nd)) == NULL) {
|
||
|
show_log = TRUE;
|
||
|
WriteError("Can't get filetype for %s", nd);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
die(MBERR_CONFIG_ERROR);
|
||
|
}
|
||
|
|
||
|
ond = xstrcpy(strrchr(nd, '/') + 1);
|
||
|
|
||
|
if (strncmp(arc, "ASC", 3)) {
|
||
|
if (!getarchiver(arc)) {
|
||
|
show_log = TRUE;
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
free(ond);
|
||
|
WriteError("Can't find unarchiver %s", arc);
|
||
|
die(MBERR_CONFIG_ERROR);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* We may both use the unarchive command for files and mail,
|
||
|
* unarchiving isn't recursive anyway.
|
||
|
*/
|
||
|
if (strlen(archiver.funarc))
|
||
|
cmd = xstrcpy(archiver.funarc);
|
||
|
else
|
||
|
cmd = xstrcpy(archiver.munarc);
|
||
|
|
||
|
if ((cmd == NULL) || (strlen(cmd) == 0)) {
|
||
|
show_log = TRUE;
|
||
|
free(cmd);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
free(ond);
|
||
|
WriteError("No unarc command available for %s", arc);
|
||
|
die(MBERR_CONFIG_ERROR);
|
||
|
}
|
||
|
|
||
|
if (execute_str(cmd, nd, (char *)NULL, (char *)"/dev/null", (char *)"/dev/null", (char *)"/dev/null")) {
|
||
|
show_log = TRUE;
|
||
|
free(cmd);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
free(ond);
|
||
|
WriteError("Fatal: unpack error");
|
||
|
die(MBERR_EXEC_FAILED);
|
||
|
}
|
||
|
free(cmd);
|
||
|
|
||
|
Match = FALSE;
|
||
|
if ((dp = opendir(wrk)) != NULL) {
|
||
|
while ((de = readdir(dp))) {
|
||
|
if (strlen(ond) == strlen(de->d_name)) {
|
||
|
Match = TRUE;
|
||
|
for (i = 0; i < (strlen(ond) -3); i++)
|
||
|
if (toupper(ond[i]) != toupper(de->d_name[i]))
|
||
|
Match = FALSE;
|
||
|
if (Match) {
|
||
|
free(ond);
|
||
|
ond = xstrcpy(de->d_name);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
closedir(dp);
|
||
|
}
|
||
|
if (!Match) {
|
||
|
show_log = TRUE;
|
||
|
free(ond);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
WriteError("Could not find extracted file");
|
||
|
die(MBERR_DIFF_ERROR);
|
||
|
}
|
||
|
} else {
|
||
|
if ((rc = file_cp(nd, ond))) {
|
||
|
show_log = TRUE;
|
||
|
free(ond);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
WriteError("Copy %s failed, %s", nd, strerror(rc));
|
||
|
die(MBERR_DIFF_ERROR);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (((p = strrchr(onl, '.'))) && ((q = strrchr(ond, '.'))) && (strlen(p) == strlen(q))) {
|
||
|
nn = xstrcpy(onl);
|
||
|
p = strrchr(nn, '.') + 1;
|
||
|
q++;
|
||
|
strcpy(p, q);
|
||
|
} else
|
||
|
nn = xstrcpy((char *)"newnodelist");
|
||
|
|
||
|
if (strcmp(onl, nn) == 0) {
|
||
|
show_log = TRUE;
|
||
|
WriteError("Attempt to update nodelist to the same version");
|
||
|
unlink(ond);
|
||
|
free(ond);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
free(nn);
|
||
|
die(MBERR_DIFF_ERROR);
|
||
|
}
|
||
|
|
||
|
Syslog('+', "Apply %s with %s to %s", onl, ond, nn);
|
||
|
if (!do_quiet) {
|
||
|
mbse_colour(CYAN, BLACK);
|
||
|
printf("Apply %s with %s to %s\n", onl, ond, nn);
|
||
|
}
|
||
|
rc = apply(onl, ond, nn);
|
||
|
|
||
|
unlink(ond);
|
||
|
if (rc) {
|
||
|
unlink(nn);
|
||
|
free(nn);
|
||
|
free(ond);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
die(MBERR_DIFF_ERROR);
|
||
|
} else {
|
||
|
unlink(onl);
|
||
|
cmd = xstrcpy(archiver.farc);
|
||
|
|
||
|
if ((cmd == NULL) || (!strlen(cmd))) {
|
||
|
free(cmd);
|
||
|
Syslog('+', "No archive command for %s, fallback to ZIP", arc);
|
||
|
if (!getarchiver((char *)"ZIP")) {
|
||
|
WriteError("No ZIP command available");
|
||
|
free(ond);
|
||
|
free(onl);
|
||
|
free(wrk);
|
||
|
free(nn);
|
||
|
die(MBERR_DIFF_ERROR);
|
||
|
} else {
|
||
|
cmd = xstrcpy(archiver.farc);
|
||
|
}
|
||
|
} else {
|
||
|
free(cmd);
|
||
|
cmd = xstrcpy(archiver.farc);
|
||
|
}
|
||
|
|
||
|
if ((cmd == NULL) || (!strlen(cmd))) {
|
||
|
WriteError("No archiver command available");
|
||
|
} else {
|
||
|
free(onl);
|
||
|
onl = xstrcpy(nn);
|
||
|
onl[strlen(onl) -3] = tolower(archiver.name[0]);
|
||
|
tl(onl);
|
||
|
p = xstrcpy(onl);
|
||
|
p = xstrcat(p, (char *)" ");
|
||
|
p = xstrcat(p, nn);
|
||
|
if (execute_str(cmd, p, (char *)NULL, (char *)"/dev/null", (char *)"/dev/null", (char *)"/dev/null"))
|
||
|
WriteError("Create %s failed", onl);
|
||
|
else {
|
||
|
CreateSema((char *)"mailin");
|
||
|
}
|
||
|
free(p);
|
||
|
free(cmd);
|
||
|
}
|
||
|
|
||
|
free(onl);
|
||
|
free(ond);
|
||
|
free(wrk);
|
||
|
free(nn);
|
||
|
die(MBERR_OK);
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void Help(void)
|
||
|
{
|
||
|
do_quiet = FALSE;
|
||
|
ProgName();
|
||
|
|
||
|
mbse_colour(LIGHTCYAN, BLACK);
|
||
|
printf("\nUsage: mbdiff [nodelist] [nodediff] <options>\n\n");
|
||
|
mbse_colour(CYAN, BLACK);
|
||
|
printf(" The nodelist must be the full path and filename\n");
|
||
|
printf(" without the dot and daynumber digits to the working\n");
|
||
|
printf(" directory of that nodelist.\n");
|
||
|
printf(" The nodediff must be the full path and filename\n");
|
||
|
printf(" to the (compressed) nodediff file in the download\n");
|
||
|
printf(" directory.\n");
|
||
|
mbse_colour(LIGHTBLUE, BLACK);
|
||
|
printf("\n Options are:\n\n");
|
||
|
mbse_colour(CYAN, BLACK);
|
||
|
printf(" -quiet Quiet mode\n");
|
||
|
mbse_colour(LIGHTGRAY, BLACK);
|
||
|
printf("\n");
|
||
|
die(MBERR_COMMANDLINE);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
int apply(char *nl, char *nd, char *nn)
|
||
|
{
|
||
|
FILE *fo, *fd, *fn;
|
||
|
unsigned char cmdbuf[BLKSIZ], lnbuf[BLKSIZ];
|
||
|
char *p;
|
||
|
int i, count, ac = 0, cc = 0, dc = 0, rc = 0, firstline = 1;
|
||
|
unsigned short theircrc = 0, mycrc = 0;
|
||
|
|
||
|
if ((fo = fopen(nl, "r")) == NULL) {
|
||
|
WriteError("$Can't open %s", nl);
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
if ((fd = fopen(nd, "r")) == NULL) {
|
||
|
WriteError("$Can't open %s", nd);
|
||
|
fclose(fo);
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
if ((fn = fopen(nn, "w")) == NULL) {
|
||
|
WriteError("$Can't open %s", nn);
|
||
|
fclose(fo);
|
||
|
fclose(fd);
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
if ((fgets((char *)cmdbuf, sizeof(cmdbuf)-1, fd) == NULL) ||
|
||
|
(fgets((char *)lnbuf, sizeof(cmdbuf)-1, fo) == NULL) ||
|
||
|
(strcmp((char *)cmdbuf, (char *)lnbuf) != 0)) {
|
||
|
rc = 6;
|
||
|
} else {
|
||
|
rewind(fo);
|
||
|
rewind(fd);
|
||
|
|
||
|
while ((rc == 0) && fgets((char *)cmdbuf, sizeof(cmdbuf)-1, fd)) {
|
||
|
switch (cmdbuf[0]) {
|
||
|
case '\032': break;
|
||
|
case ';': Striplf((char *)cmdbuf);
|
||
|
break;
|
||
|
case 'A': count = atoi((char *)cmdbuf+1);
|
||
|
ac += count;
|
||
|
Striplf((char *)cmdbuf);
|
||
|
for (i = 0;(i < count) && (rc == 0); i++)
|
||
|
if (fgets((char *)lnbuf, sizeof(lnbuf)-1, fd)) {
|
||
|
if (firstline) {
|
||
|
firstline = 0;
|
||
|
if ((p = strrchr((char *)lnbuf, ':'))) {
|
||
|
theircrc = atoi((char *)p+1);
|
||
|
}
|
||
|
} else {
|
||
|
for (p = (char *)lnbuf; *p; p++)
|
||
|
mycrc = updcrc(*p, mycrc);
|
||
|
}
|
||
|
fputs((char *)lnbuf, fn);
|
||
|
} else
|
||
|
rc = 3;
|
||
|
break;
|
||
|
case 'D': count = atoi((char *)cmdbuf + 1);
|
||
|
dc += count;
|
||
|
Striplf((char *)cmdbuf);
|
||
|
for (i = 0;(i < count) && (rc == 0); i++)
|
||
|
if (fgets((char *)lnbuf, sizeof(lnbuf)-1, fo) == NULL)
|
||
|
rc = 3;
|
||
|
break;
|
||
|
case 'C': count = atoi((char *)cmdbuf+1);
|
||
|
cc += count;
|
||
|
Striplf((char *)cmdbuf);
|
||
|
for (i = 0; (i < count) && (rc == 0); i++)
|
||
|
if (fgets((char *)lnbuf, sizeof(lnbuf) - 1, fo)) {
|
||
|
/*
|
||
|
* Don't use EOF character for CRC test.
|
||
|
*/
|
||
|
if (lnbuf[0] != '\032') {
|
||
|
for (p = (char *)lnbuf; *p; p++)
|
||
|
mycrc = updcrc(*p, mycrc);
|
||
|
fputs((char *)lnbuf, fn);
|
||
|
}
|
||
|
} else
|
||
|
rc = 3;
|
||
|
break;
|
||
|
default: rc = 5;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fclose(fo);
|
||
|
fclose(fd);
|
||
|
fputc('\032', fn);
|
||
|
fclose(fn);
|
||
|
|
||
|
if ((rc != 0) && !do_quiet) {
|
||
|
show_log = TRUE;
|
||
|
mbse_colour(LIGHTRED, BLACK);
|
||
|
}
|
||
|
|
||
|
if ((rc == 0) && (mycrc != theircrc))
|
||
|
rc = 4;
|
||
|
|
||
|
if (rc == 3) {
|
||
|
WriteError("Could not read some of the files");
|
||
|
if (!do_quiet)
|
||
|
printf("Could not read some of the files\n");
|
||
|
} else if (rc == 4) {
|
||
|
WriteError("CRC is %hu, should be %hu", mycrc, theircrc);
|
||
|
if (!do_quiet)
|
||
|
printf("CRC is %hu, should be %hu\n", mycrc, theircrc);
|
||
|
} else if (rc == 5) {
|
||
|
WriteError("Unknown input line: \"%s\"", cmdbuf);
|
||
|
if (!do_quiet)
|
||
|
printf("Unknown input line: \"%s\"\n", cmdbuf);
|
||
|
} else if (rc == 6) {
|
||
|
WriteError("Diff does not match old list");
|
||
|
if (!do_quiet)
|
||
|
printf("Diff does not match old list\n");
|
||
|
} else {
|
||
|
Syslog('+', "Copied %d, added %d, deleted %d, difference %d", cc, ac, dc, ac-dc);
|
||
|
if (!do_quiet)
|
||
|
printf("Created new nodelist\n");
|
||
|
}
|
||
|
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
|