This repository has been archived on 2024-04-08. You can view files and clone it, but cannot push or open issues or pull requests.
magicka/src/bbs.c

1235 lines
25 KiB
C
Raw Normal View History

2016-03-22 01:48:59 +00:00
#include <stdio.h>
#include <netdb.h>
#include <netinet/in.h>
2016-12-08 12:27:06 +00:00
#include <arpa/inet.h>
2016-03-22 01:48:59 +00:00
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/fcntl.h>
#include <signal.h>
2016-03-23 00:14:31 +00:00
#include <sys/utsname.h>
#include <sys/time.h>
2016-04-10 04:10:18 +00:00
#include <stdarg.h>
#include <fts.h>
#include <errno.h>
#include <sys/socket.h>
2017-04-23 00:51:40 +00:00
#include <iconv.h>
2016-08-10 01:22:55 +00:00
#include "bbs.h"
#include "lua/lua.h"
#include "lua/lualib.h"
#include "lua/lauxlib.h"
2016-03-22 01:48:59 +00:00
int telnet_bin_mode = 0;
2016-08-10 01:22:55 +00:00
int mynode = 0;
struct bbs_config conf;
2016-03-22 01:48:59 +00:00
2016-03-22 01:48:59 +00:00
struct user_record *gUser;
int gSocket;
2016-08-10 01:22:55 +00:00
int sshBBS;
2016-03-22 01:48:59 +00:00
int usertimeout;
int timeoutpaused;
2017-09-28 09:11:00 +00:00
time_t userlaston;
2016-03-22 01:48:59 +00:00
2016-08-17 09:26:01 +00:00
char *ipaddress = NULL;
2016-08-10 01:22:55 +00:00
void sigterm_handler2(int s)
{
if (mynode != 0) {
disconnect("Terminated.");
}
dolog("Terminated...");
exit(0);
}
void sigint_handler(int s)
{
// do nothing...
}
void broadcast(char *mess, ...) {
2017-03-19 03:39:03 +00:00
char json[1024];
char buffer[512];
struct sockaddr_in s;
int bcast_sock;
int broadcastEnable=1;
int ret;
if (conf.broadcast_enable && conf.broadcast_port > 1024 && conf.broadcast_port < 65536 && conf.broadcast_address != NULL) {
bcast_sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
ret=setsockopt(bcast_sock, SOL_SOCKET, SO_BROADCAST, &broadcastEnable, sizeof(broadcastEnable));
if (ret) {
dolog("broadcast: Couldn't set socket to broadcast mode");
close(bcast_sock);
return;
}
memset(&s, 0, sizeof(struct sockaddr_in));
s.sin_family=AF_INET;
2016-12-08 12:27:06 +00:00
s.sin_addr.s_addr = inet_addr(conf.broadcast_address);
s.sin_port = htons((unsigned short)conf.broadcast_port);
bind(bcast_sock, (struct sockaddr *)&s, sizeof(struct sockaddr_in));
va_list ap;
va_start(ap, mess);
vsnprintf(buffer, 512, mess, ap);
va_end(ap);
2017-03-19 03:42:36 +00:00
snprintf(json, 1024, "{\"System\": \"%s\", \"Program\": \"MagickaBBS\", \"Message\": \"%s\"}", conf.bbs_name, buffer);
2017-03-19 03:39:03 +00:00
ret = sendto(bcast_sock, json, strlen(json) + 1, 0, (struct sockaddr *)&s, sizeof(struct sockaddr_in));
if (ret < 0) {
dolog("broadcast: Couldn't send broadcast");
}
close(bcast_sock);
}
}
2016-08-10 01:22:55 +00:00
2018-01-21 09:05:01 +00:00
void dolog_www(char *ipaddr, char *fmt, ...) {
char buffer[PATH_MAX];
struct tm time_now;
time_t timen;
FILE *logfptr;
int mypid = getpid();
if (conf.log_path == NULL) return;
timen = time(NULL);
localtime_r(&timen, &time_now);
snprintf(buffer, PATH_MAX, "%s/%04d%02d%02d.log", conf.log_path, time_now.tm_year + 1900, time_now.tm_mon + 1, time_now.tm_mday);
logfptr = fopen(buffer, "a");
if (!logfptr) {
return;
}
va_list ap;
va_start(ap, fmt);
vsnprintf(buffer, PATH_MAX, fmt, ap);
va_end(ap);
fprintf(logfptr, "%02d:%02d:%02d [%d][%s] %s\n", time_now.tm_hour, time_now.tm_min, time_now.tm_sec, mypid, ipaddr, buffer);
fclose(logfptr);
}
2016-04-10 04:10:18 +00:00
void dolog(char *fmt, ...) {
2018-01-21 09:05:01 +00:00
char buffer[PATH_MAX];
2016-04-10 04:10:18 +00:00
struct tm time_now;
time_t timen;
FILE *logfptr;
2016-08-10 01:22:55 +00:00
int mypid = getpid();
2016-04-10 04:10:18 +00:00
if (conf.log_path == NULL) return;
2016-08-10 01:22:55 +00:00
2016-04-10 04:10:18 +00:00
timen = time(NULL);
2016-08-10 01:22:55 +00:00
2016-04-10 04:10:18 +00:00
localtime_r(&timen, &time_now);
2016-08-10 01:22:55 +00:00
2018-01-21 09:05:01 +00:00
snprintf(buffer, PATH_MAX, "%s/%04d%02d%02d.log", conf.log_path, time_now.tm_year + 1900, time_now.tm_mon + 1, time_now.tm_mday);
2016-04-10 04:10:18 +00:00
logfptr = fopen(buffer, "a");
if (!logfptr) {
return;
}
va_list ap;
va_start(ap, fmt);
2018-01-21 09:05:01 +00:00
vsnprintf(buffer, PATH_MAX, fmt, ap);
va_end(ap);
2016-08-10 01:22:55 +00:00
fprintf(logfptr, "%02d:%02d:%02d [%d][%s] %s\n", time_now.tm_hour, time_now.tm_min, time_now.tm_sec, mypid, ipaddress, buffer);
2016-08-10 01:22:55 +00:00
2016-04-10 04:10:18 +00:00
fclose(logfptr);
}
2016-03-24 07:23:42 +00:00
struct fido_addr *parse_fido_addr(const char *str) {
2016-12-06 07:06:28 +00:00
if (str == NULL) {
return NULL;
}
2016-03-24 07:23:42 +00:00
struct fido_addr *ret = (struct fido_addr *)malloc(sizeof(struct fido_addr));
int c;
int state = 0;
2016-08-10 01:22:55 +00:00
2016-03-24 07:23:42 +00:00
ret->zone = 0;
ret->net = 0;
ret->node = 0;
ret->point = 0;
2016-08-10 01:22:55 +00:00
2016-03-24 07:23:42 +00:00
for (c=0;c<strlen(str);c++) {
switch(str[c]) {
case ':':
state = 1;
break;
case '/':
state = 2;
break;
case '.':
state = 3;
break;
case '0':
case '1':
case '2':
case '3':
case '4':
case '5':
case '6':
case '7':
case '8':
case '9':
{
switch (state) {
case 0:
2016-03-24 22:27:06 +00:00
ret->zone = ret->zone * 10 + (str[c] - '0');
2016-03-24 07:23:42 +00:00
break;
case 1:
2016-03-24 22:27:06 +00:00
ret->net = ret->net * 10 + (str[c] - '0');
2016-03-24 07:23:42 +00:00
break;
case 2:
2016-03-24 22:27:06 +00:00
ret->node = ret->node * 10 + (str[c] - '0');
2016-03-24 07:23:42 +00:00
break;
case 3:
2016-03-24 22:27:06 +00:00
ret->point = ret->point * 10 + (str[c] - '0');
2016-03-24 07:23:42 +00:00
break;
}
}
break;
default:
free(ret);
return NULL;
}
}
return ret;
}
2016-03-22 01:48:59 +00:00
void timer_handler(int signum) {
if (signum == SIGALRM) {
if (gUser != NULL) {
gUser->timeleft--;
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
if (gUser->timeleft <= 0) {
s_printf(get_string(0));
2016-08-10 01:22:55 +00:00
disconnect("Out of Time");
2016-03-22 01:48:59 +00:00
}
2016-08-10 01:22:55 +00:00
}
if (timeoutpaused == 0) {
usertimeout--;
}
2016-03-22 01:48:59 +00:00
if (usertimeout <= 0) {
s_printf(get_string(1));
2016-08-10 01:22:55 +00:00
disconnect("Timeout");
}
2016-03-22 01:48:59 +00:00
}
}
2016-08-10 01:22:55 +00:00
void s_printf(char *fmt, ...) {
char buffer[512];
2017-04-23 00:51:40 +00:00
va_list ap;
va_start(ap, fmt);
vsnprintf(buffer, 512, fmt, ap);
va_end(ap);
2016-08-10 01:22:55 +00:00
s_putstring(buffer);
2016-03-22 01:48:59 +00:00
}
2017-04-23 00:51:40 +00:00
int should_convert_utf8() {
2017-04-23 02:28:01 +00:00
if (gUser != NULL) {
return gUser->codepage;
}
2017-04-23 00:51:40 +00:00
return conf.codepage;
}
2016-08-10 01:22:55 +00:00
void s_putchar(char c) {
2017-04-23 00:51:40 +00:00
iconv_t ic;
char *inbuf;
char *outbuf;
char *ptr1;
char *ptr2;
size_t inc;
size_t ouc;
size_t sz;
if (!should_convert_utf8()) {
if (sshBBS) {
putchar(c);
} else {
write(gSocket, &c, 1);
}
2016-08-10 01:22:55 +00:00
} else {
2017-04-23 00:51:40 +00:00
ic = iconv_open("UTF-8", "CP437");
2017-04-23 03:36:54 +00:00
inbuf = (char *)malloc(4);
outbuf = (char *)malloc(4);
memset(outbuf, 0, 4);
2017-04-23 00:51:40 +00:00
sprintf(inbuf, "%c", c);
inc = 1;
2017-04-23 03:36:54 +00:00
ouc = 4;
2017-04-23 00:51:40 +00:00
ptr1 = outbuf;
ptr2 = inbuf;
sz = iconv(ic, &inbuf, &inc, &outbuf, &ouc);
if (sshBBS) {
fprintf(stdout, "%s", ptr1);
} else {
write(gSocket, ptr1, outbuf - ptr1);
2017-04-23 00:51:40 +00:00
}
iconv_close(ic);
free(ptr1);
free(ptr2);
2016-08-10 01:22:55 +00:00
}
2016-03-22 01:48:59 +00:00
}
2016-08-10 01:22:55 +00:00
void s_putstring(char *c) {
2017-04-23 00:51:40 +00:00
iconv_t ic;
char *inbuf;
char *outbuf;
size_t inc;
size_t ouc;
size_t sz;
char *ptr1;
char *ptr2;
if (!should_convert_utf8()) {
if (sshBBS) {
2017-09-15 08:20:15 +00:00
fprintf(stdout, "%s", c);
2017-04-23 00:51:40 +00:00
} else {
write(gSocket, c, strlen(c));
}
2016-08-10 01:22:55 +00:00
} else {
2017-04-23 00:51:40 +00:00
ic = iconv_open("UTF-8", "CP437");
inc = strlen(c);
inbuf = strdup(c);
2017-04-23 03:36:54 +00:00
outbuf = (char *)malloc(inc * 4);
memset(outbuf, 0, inc * 4);
2017-04-23 00:51:40 +00:00
ptr1 = outbuf;
ptr2 = inbuf;
2017-04-23 03:36:54 +00:00
ouc = inc * 4;
2017-04-23 00:51:40 +00:00
sz = iconv(ic, &inbuf, &inc, &outbuf, &ouc);
if (sshBBS) {
fprintf(stdout, "%s", ptr1);
} else {
write(gSocket, ptr1, outbuf - ptr1);
2017-04-23 00:51:40 +00:00
}
iconv_close(ic);
free(ptr1);
free(ptr2);
2016-08-10 01:22:55 +00:00
}
}
2017-09-09 03:28:49 +00:00
void s_displayansi_pause(char *file, int pause) {
2016-03-27 00:33:43 +00:00
FILE *fptr;
char c;
2017-09-19 20:04:47 +00:00
char ch;
2017-09-09 03:28:49 +00:00
int lines = 0;
char lastch = 0;
2016-03-27 00:33:43 +00:00
fptr = fopen(file, "r");
if (!fptr) {
return;
}
c = fgetc(fptr);
2016-08-10 01:22:55 +00:00
while (!feof(fptr) && c != 0x1a) {
if (c == '\n' && lastch != '\r') {
s_putchar('\r');
}
2016-08-10 01:22:55 +00:00
s_putchar(c);
lastch = c;
2017-09-09 03:28:49 +00:00
if (pause) {
if (c == '\n') {
lines++;
if (lines == 24) {
2017-09-19 20:04:47 +00:00
s_printf(get_string(223));
ch = s_getchar();
s_printf("\r\n");
switch(tolower(ch)) {
case 'c':
pause = 0;
break;
case 'n':
fclose(fptr);
return;
default:
break;
}
2017-09-09 03:28:49 +00:00
lines = 0;
}
}
}
2016-03-27 00:33:43 +00:00
c = fgetc(fptr);
}
fclose(fptr);
}
2017-09-09 03:28:49 +00:00
void s_displayansi_p(char *file) {
s_displayansi_pause(file, 0);
}
2016-03-27 00:33:43 +00:00
2016-08-10 01:22:55 +00:00
void s_displayansi(char *file) {
2016-03-22 01:48:59 +00:00
FILE *fptr;
char c;
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
char buffer[256];
2016-08-10 01:22:55 +00:00
2017-10-11 06:24:59 +00:00
if (strchr(file, '/') == NULL) {
sprintf(buffer, "%s/%s.ans", conf.ansi_path, file);
s_displayansi_pause(buffer, 0);
} else {
s_displayansi_pause(file, 0);
}
2016-03-22 01:48:59 +00:00
}
2016-08-10 01:22:55 +00:00
char s_getchar() {
2016-03-22 01:48:59 +00:00
unsigned char c;
unsigned char d;
2016-03-22 01:48:59 +00:00
int len;
char iac_binary_will[] = {IAC, IAC_WILL, IAC_TRANSMIT_BINARY, '\0'};
char iac_binary_do[] = {IAC, IAC_DO, IAC_TRANSMIT_BINARY, '\0'};
char iac_binary_wont[] = {IAC, IAC_WONT, IAC_TRANSMIT_BINARY, '\0'};
char iac_binary_dont[] = {IAC, IAC_DONT, IAC_TRANSMIT_BINARY, '\0'};
2016-03-22 01:48:59 +00:00
2016-03-24 00:46:01 +00:00
do {
2016-08-10 01:22:55 +00:00
if (sshBBS) {
c = getchar();
} else {
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
}
}
if (!sshBBS) {
while (c == IAC) {
2016-08-10 01:22:55 +00:00
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
} else if (c == IAC) {
2016-08-10 01:22:55 +00:00
usertimeout = 10;
return c;
}
if (c == IAC_WILL || c == IAC_WONT || c == IAC_DO || c == IAC_DONT) {
len = read(gSocket, &d, 1);
2016-12-12 01:32:57 +00:00
if (len == 0) {
disconnect("Socket Closed");
}
switch (c) {
case IAC_WILL:
if (d == 0) {
if (telnet_bin_mode != 1) {
telnet_bin_mode = 1;
write(gSocket, iac_binary_do, 3);
}
}
break;
case IAC_WONT:
if (d == 0) {
if (telnet_bin_mode != 0) {
telnet_bin_mode = 0;
write(gSocket, iac_binary_dont, 3);
}
}
break;
case IAC_DO:
if (d == 0) {
if (telnet_bin_mode != 1) {
telnet_bin_mode = 1;
write(gSocket, iac_binary_will, 3);
}
}
break;
case IAC_DONT:
if (d == 0) {
if (telnet_bin_mode != 0) {
telnet_bin_mode = 0;
write(gSocket, iac_binary_wont, 3);
}
}
break;
}
2016-12-12 01:32:57 +00:00
} else if (c == 250) {
do {
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
}
} while(c != 240);
}
2016-08-10 01:22:55 +00:00
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
}
}
}
2016-12-09 23:44:54 +00:00
} while (c == '\n' || c == '\0');
2016-08-10 01:22:55 +00:00
usertimeout = 10;
2016-03-22 01:48:59 +00:00
return (char)c;
}
2016-08-10 01:22:55 +00:00
char s_getc() {
char c = s_getchar();
2016-03-22 01:48:59 +00:00
2016-08-10 01:22:55 +00:00
s_putchar(c);
2016-03-22 01:48:59 +00:00
return (char)c;
}
2016-08-10 01:22:55 +00:00
void s_readstring(char *buffer, int max) {
2016-03-22 01:48:59 +00:00
int i;
char c;
2016-08-10 01:22:55 +00:00
2016-03-23 00:14:31 +00:00
memset(buffer, 0, max);
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
for (i=0;i<max;i++) {
2016-08-10 01:22:55 +00:00
c = s_getchar();
2016-12-09 23:44:54 +00:00
2016-03-22 01:48:59 +00:00
if ((c == '\b' || c == 127) && i > 0) {
buffer[i-1] = '\0';
i -= 2;
2016-08-10 01:22:55 +00:00
s_printf("\e[D \e[D");
continue;
} else if (c == '\b' || c == 127) {
i -= 1;
2016-03-22 01:48:59 +00:00
continue;
} else if (c == 27) {
c = s_getchar();
if (c == 91) {
c = s_getchar();
}
2016-12-04 03:26:29 +00:00
i -= 1;
continue;
2016-08-10 01:22:55 +00:00
}
2016-03-22 01:48:59 +00:00
if (c == '\n' || c == '\r') {
return;
}
2016-08-10 01:22:55 +00:00
s_putchar(c);
2016-03-22 01:48:59 +00:00
buffer[i] = c;
buffer[i+1] = '\0';
}
}
2016-08-10 01:22:55 +00:00
void s_readpass(char *buffer, int max) {
2016-03-22 01:48:59 +00:00
int i;
char c;
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
for (i=0;i<max;i++) {
2016-08-10 01:22:55 +00:00
c = s_getchar();
2016-03-22 01:48:59 +00:00
if ((c == '\b' || c == 127) && i > 0) {
buffer[i-1] = '\0';
i-=2;
2016-08-10 01:22:55 +00:00
s_printf("\e[D \e[D");
continue;
} else if (c == '\b' || c == 127) {
i -= 1;
2016-03-22 01:48:59 +00:00
continue;
2016-12-11 23:54:00 +00:00
} else if (c == 27) {
c = s_getchar();
if (c == 91) {
c = s_getchar();
}
i -= 1;
continue;
2016-08-10 01:22:55 +00:00
}
2016-03-22 01:48:59 +00:00
if (c == '\n' || c == '\r') {
return;
}
2016-08-10 01:22:55 +00:00
s_putchar('*');
2016-03-22 01:48:59 +00:00
buffer[i] = c;
buffer[i+1] = '\0';
}
}
2017-09-15 12:06:02 +00:00
void exit_bbs() {
char buffer[1024];
snprintf(buffer, 1024, "%s/nodeinuse.%d", conf.bbs_path, mynode);
remove(buffer);
}
2016-08-10 01:22:55 +00:00
void disconnect(char *calledby) {
2017-09-15 12:06:02 +00:00
2016-03-22 01:48:59 +00:00
if (gUser != NULL) {
save_user(gUser);
}
dolog("Node %d disconnected (%s)", mynode, calledby);
2017-09-15 12:06:02 +00:00
2016-08-10 01:22:55 +00:00
if (!sshBBS) {
close(gSocket);
2017-09-15 11:32:20 +00:00
}
2016-03-22 01:48:59 +00:00
exit(0);
}
void record_last10_callers(struct user_record *user) {
2016-03-23 05:21:53 +00:00
struct last10_callers new_entry;
struct last10_callers callers[10];
int i,j;
2016-08-10 01:22:55 +00:00
FILE *fptr = fopen("last10.dat", "rb");
2016-03-23 05:21:53 +00:00
if (fptr != NULL) {
for (i=0;i<10;i++) {
if (fread(&callers[i], sizeof(struct last10_callers), 1, fptr) < 1) {
break;
}
}
fclose(fptr);
} else {
i = 0;
}
2016-08-10 01:22:55 +00:00
if (strcasecmp(conf.sysop_name, user->loginname) != 0 ) {
2016-03-23 05:21:53 +00:00
memset(&new_entry, 0, sizeof(struct last10_callers));
strcpy(new_entry.name, user->loginname);
strcpy(new_entry.location, user->location);
new_entry.time = time(NULL);
2016-08-10 01:22:55 +00:00
2016-03-23 05:21:53 +00:00
if (i == 10) {
j = 1;
} else {
j = 0;
}
fptr = fopen("last10.dat", "wb");
for (;j<i;j++) {
fwrite(&callers[j], sizeof(struct last10_callers), 1, fptr);
}
fwrite(&new_entry, sizeof(struct last10_callers), 1, fptr);
fclose(fptr);
}
}
2016-08-10 01:22:55 +00:00
void display_last10_callers(struct user_record *user) {
struct last10_callers callers[10];
2016-08-10 01:22:55 +00:00
int i,z;
struct tm l10_time;
FILE *fptr = fopen("last10.dat", "rb");
time_t l10_timet;
2016-08-10 01:22:55 +00:00
s_printf(get_string(2));
s_printf(get_string(3));
2016-08-10 01:22:55 +00:00
if (fptr != NULL) {
2016-08-10 01:22:55 +00:00
for (i=0;i<10;i++) {
if (fread(&callers[i], sizeof(struct last10_callers), 1, fptr) < 1) {
break;
}
}
2016-08-10 01:22:55 +00:00
fclose(fptr);
} else {
i = 0;
}
2016-08-10 01:22:55 +00:00
for (z=0;z<i;z++) {
l10_timet = callers[z].time;
localtime_r(&l10_timet, &l10_time);
2017-10-03 02:09:50 +00:00
if (conf.date_style == 1) {
s_printf(get_string(4), callers[z].name, callers[z].location, l10_time.tm_hour, l10_time.tm_min, l10_time.tm_mon + 1, l10_time.tm_mday, l10_time.tm_year - 100);
} else {
s_printf(get_string(4), callers[z].name, callers[z].location, l10_time.tm_hour, l10_time.tm_min, l10_time.tm_mday, l10_time.tm_mon + 1, l10_time.tm_year - 100);
}
}
s_printf(get_string(5));
s_printf(get_string(6));
2016-08-10 01:22:55 +00:00
s_getc();
2016-03-23 05:21:53 +00:00
}
2016-08-10 01:22:55 +00:00
void display_info() {
struct utsname name;
2016-04-01 04:33:37 +00:00
2016-03-23 00:14:31 +00:00
uname(&name);
2016-08-10 01:22:55 +00:00
s_printf(get_string(7));
s_printf(get_string(8));
s_printf(get_string(9), conf.bbs_name);
s_printf(get_string(10), conf.sysop_name);
s_printf(get_string(11), mynode);
s_printf(get_string(12), VERSION_MAJOR, VERSION_MINOR, VERSION_STR);
s_printf(get_string(13), name.sysname, name.machine);
s_printf(get_string(14));
2016-08-10 01:22:55 +00:00
s_printf(get_string(6));
2016-08-10 01:22:55 +00:00
s_getc();
2016-03-23 00:14:31 +00:00
}
2016-08-10 01:22:55 +00:00
void automessage_write(struct user_record *user) {
2016-04-18 10:34:07 +00:00
FILE *fptr;
char automsg[450];
2016-08-10 01:22:55 +00:00
char buffer[76];
2016-04-18 10:34:07 +00:00
int i;
struct tm timenow;
time_t timen;
2016-08-10 01:22:55 +00:00
2016-04-18 10:34:07 +00:00
memset(automsg, 0, 450);
2016-08-10 01:22:55 +00:00
memset(buffer, 0, 76);
2016-04-18 10:34:07 +00:00
if (user->sec_level >= conf.automsgwritelvl) {
timen = time(NULL);
localtime_r(&timen, &timenow);
2016-08-10 01:22:55 +00:00
sprintf(automsg, get_string(15), user->loginname, asctime(&timenow));
2016-08-10 01:22:55 +00:00
2016-04-18 10:34:07 +00:00
automsg[strlen(automsg) - 1] = '\r';
automsg[strlen(automsg)] = '\n';
s_printf(get_string(16));
2016-04-18 10:34:07 +00:00
for (i=0;i<4;i++) {
2016-08-10 01:22:55 +00:00
s_printf("\r\n%d: ", i);
s_readstring(buffer, 75);
2016-04-18 10:34:07 +00:00
strcat(automsg, buffer);
strcat(automsg, "\r\n");
}
2016-08-10 01:22:55 +00:00
2016-04-18 10:34:07 +00:00
fptr = fopen("automessage.txt", "w");
if (fptr) {
fwrite(automsg, strlen(automsg), 1, fptr);
fclose(fptr);
} else {
2016-08-10 01:22:55 +00:00
dolog("Unable to open automessage.txt for writing");
2016-04-18 10:34:07 +00:00
}
2016-08-10 01:22:55 +00:00
}
2016-04-18 10:34:07 +00:00
}
2016-08-10 01:22:55 +00:00
void automessage_display() {
2016-04-18 10:34:07 +00:00
struct stat s;
FILE *fptr;
char buffer[90];
int i;
2016-08-10 01:22:55 +00:00
s_printf("\r\n\r\n");
2016-04-18 10:34:07 +00:00
if (stat("automessage.txt", &s) == 0) {
fptr = fopen("automessage.txt", "r");
if (fptr) {
for (i=0;i<5;i++) {
memset(buffer, 0, 90);
fgets(buffer, 88, fptr);
buffer[strlen(buffer) - 1] = '\r';
buffer[strlen(buffer)] = '\n';
2016-08-10 01:22:55 +00:00
s_printf(buffer);
2016-04-18 10:34:07 +00:00
}
fclose(fptr);
} else {
2016-08-10 01:22:55 +00:00
dolog("Error opening automessage.txt");
}
2016-04-18 10:34:07 +00:00
} else {
s_printf(get_string(17));
2016-04-18 10:34:07 +00:00
}
s_printf(get_string(6));
2016-08-10 01:22:55 +00:00
s_getc();
2016-04-18 10:34:07 +00:00
}
2016-08-10 01:22:55 +00:00
void runbbs_real(int socket, char *ip, int ssh) {
2018-01-30 07:33:57 +00:00
char buffer[PATH_MAX];
2016-03-22 01:48:59 +00:00
char password[17];
struct stat s;
FILE *nodefile;
int i;
char iac_echo[] = {IAC, IAC_WILL, IAC_ECHO, '\0'};
char iac_sga[] = {IAC, IAC_WILL, IAC_SUPPRESS_GO_AHEAD, '\0'};
2016-03-22 01:48:59 +00:00
struct user_record *user;
struct tm thetime;
struct tm oldtime;
time_t now;
struct itimerval itime;
2016-08-10 01:22:55 +00:00
struct sigaction sa;
struct sigaction st;
lua_State *L;
int do_internal_login = 0;
2016-12-07 06:57:57 +00:00
int usernotfound;
int tries;
2017-09-15 12:06:02 +00:00
atexit(exit_bbs);
2016-04-13 07:42:22 +00:00
ipaddress = ip;
2016-08-10 01:22:55 +00:00
if (!ssh) {
gUser = NULL;
sshBBS = 0;
2016-12-12 01:32:57 +00:00
if (write(socket, iac_echo, 3) != 3) {
dolog("Failed to send iac_echo");
exit(0);
}
if (write(socket, iac_sga, 3) != 3) {
dolog("Failed to send iac_sga");
exit(0);
}
2016-08-10 01:22:55 +00:00
} else {
sshBBS = 1;
}
st.sa_handler = sigterm_handler2;
sigemptyset(&st.sa_mask);
2016-08-17 09:26:01 +00:00
st.sa_flags = SA_SIGINFO;
2016-08-10 01:22:55 +00:00
if (sigaction(SIGTERM, &st, NULL) == -1) {
dolog("Failed to setup sigterm handler.");
exit(1);
}
gSocket = socket;
s_printf("Magicka BBS v%d.%d (%s), Loading...\r\n", VERSION_MAJOR, VERSION_MINOR, VERSION_STR);
// find out which node we are
for (i=1;i<=conf.nodes;i++) {
2016-03-22 01:48:59 +00:00
sprintf(buffer, "%s/nodeinuse.%d", conf.bbs_path, i);
if (stat(buffer, &s) != 0) {
mynode = i;
nodefile = fopen(buffer, "w");
if (!nodefile) {
2016-08-10 01:22:55 +00:00
dolog("Error opening nodefile!");
2016-03-22 01:48:59 +00:00
close(socket);
exit(1);
}
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
fputs("UNKNOWN", nodefile);
fclose(nodefile);
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
break;
}
}
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
if (mynode == 0) {
s_printf(get_string(18));
2016-08-10 01:22:55 +00:00
if (!ssh) {
close(socket);
}
2016-03-22 01:48:59 +00:00
exit(1);
}
2016-08-10 01:22:55 +00:00
2016-12-11 23:54:00 +00:00
dolog("Incoming connection on node %d", mynode);
2016-03-22 01:48:59 +00:00
usertimeout = 10;
timeoutpaused = 0;
2016-12-07 06:57:57 +00:00
tries = 0;
2016-03-22 01:48:59 +00:00
memset (&sa, 0, sizeof (sa));
sa.sa_handler = &timer_handler;
sa.sa_flags = SA_RESTART;
sigaction (SIGALRM, &sa, 0);
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
itime.it_interval.tv_sec = 60;
itime.it_interval.tv_usec = 0;
itime.it_value.tv_sec = 60;
itime.it_value.tv_usec = 0;
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
setitimer (ITIMER_REAL, &itime, 0);
2016-08-10 01:22:55 +00:00
s_displayansi("issue");
if (!ssh) {
2016-12-07 06:57:57 +00:00
tryagain:
s_printf(get_string(19));
s_printf(get_string(20));
2016-08-10 01:22:55 +00:00
s_readstring(buffer, 25);
2016-12-07 06:57:57 +00:00
usernotfound = 0;
2016-08-10 01:22:55 +00:00
if (strcasecmp(buffer, "new") == 0) {
2016-12-07 06:57:57 +00:00
usernotfound = 1;
} else if (check_user(buffer)) {
usernotfound = 1;
s_printf(get_string(203));
}
if (usernotfound) {
2016-12-11 23:54:00 +00:00
dolog("New user on node %d", mynode);
2016-08-10 01:22:55 +00:00
user = new_user();
gUser = user;
} else {
s_printf(get_string(21));
2016-08-10 01:22:55 +00:00
s_readpass(password, 16);
user = check_user_pass(buffer, password);
if (user == NULL) {
2016-12-07 06:57:57 +00:00
if (tries == 3) {
s_printf(get_string(22));
disconnect("Incorrect Login");
} else {
tries++;
s_printf(get_string(22));
goto tryagain;
}
2016-08-10 01:22:55 +00:00
}
gUser = user;
for (i=1;i<=conf.nodes;i++) {
sprintf(buffer, "%s/nodeinuse.%d", conf.bbs_path, i);
if (stat(buffer, &s) == 0) {
nodefile = fopen(buffer, "r");
if (!nodefile) {
dolog("Error opening nodefile!");
disconnect("Error opening nodefile!");
}
fgets(buffer, 256, nodefile);
if (strcasecmp(user->loginname, buffer) == 0) {
fclose(nodefile);
s_printf(get_string(23));
2016-08-10 01:22:55 +00:00
disconnect("Already Logged in");
}
fclose(nodefile);
}
}
}
} else {
if (gUser != NULL) {
user = gUser;
s_printf(get_string(24), gUser->loginname);
2016-08-10 01:22:55 +00:00
s_getc();
for (i=1;i<=conf.nodes;i++) {
sprintf(buffer, "%s/nodeinuse.%d", conf.bbs_path, i);
if (stat(buffer, &s) == 0) {
nodefile = fopen(buffer, "r");
if (!nodefile) {
dolog("Error opening nodefile!");
disconnect("Error opening nodefile!");
}
fgets(buffer, 256, nodefile);
if (strcasecmp(user->loginname, buffer) == 0) {
fclose(nodefile);
s_printf(get_string(23));
2016-08-10 01:22:55 +00:00
disconnect("Already Logged in");
}
fclose(nodefile);
}
}
} else {
s_printf(get_string(25), conf.bbs_name);
2016-08-10 01:22:55 +00:00
s_getc();
gUser = new_user();
user = gUser;
}
}
2018-01-30 07:33:57 +00:00
snprintf(buffer, PATH_MAX, "%s/nodeinuse.%d", conf.bbs_path, mynode);
2016-03-22 01:48:59 +00:00
nodefile = fopen(buffer, "w");
if (!nodefile) {
2016-08-10 01:22:55 +00:00
dolog("Error opening nodefile!");
2016-03-22 01:48:59 +00:00
close(socket);
exit(1);
}
2016-08-10 01:22:55 +00:00
2016-03-22 01:48:59 +00:00
fputs(user->loginname, nodefile);
2016-08-10 01:22:55 +00:00
fclose(nodefile);
2018-01-30 07:33:57 +00:00
snprintf(buffer, PATH_MAX, "%s/node%d/nodemsg.txt", conf.bbs_path, mynode);
2016-08-10 01:22:55 +00:00
2017-04-19 05:35:46 +00:00
if (stat(buffer, &s) == 0) {
unlink(buffer);
}
2016-08-10 01:22:55 +00:00
2018-01-30 07:33:57 +00:00
snprintf(buffer, PATH_MAX, "%s/node%d/lua/", conf.bbs_path, mynode);
if (stat(buffer, &s) == 0) {
recursive_delete(buffer);
}
2018-01-23 10:57:58 +00:00
#if defined(ENABLE_WWW)
www_expire_old_links();
#endif
2018-01-30 07:33:57 +00:00
2016-03-22 01:48:59 +00:00
// do post-login
2016-04-10 04:10:18 +00:00
dolog("%s logged in, on node %d", user->loginname, mynode);
2017-03-19 03:48:50 +00:00
broadcast("%s logged in, on node %d", user->loginname, mynode);
2016-03-22 01:48:59 +00:00
// check time left
now = time(NULL);
localtime_r(&now, &thetime);
localtime_r(&user->laston, &oldtime);
2016-08-10 01:22:55 +00:00
2017-09-28 09:11:00 +00:00
userlaston = user->laston;
2016-03-22 01:48:59 +00:00
if (thetime.tm_mday != oldtime.tm_mday || thetime.tm_mon != oldtime.tm_mon || thetime.tm_year != oldtime.tm_year) {
user->timeleft = user->sec_info->timeperday;
user->laston = now;
save_user(user);
2016-08-10 01:22:55 +00:00
}
2016-03-23 06:31:00 +00:00
user->timeson++;
if (conf.script_path != NULL) {
2018-01-30 07:33:57 +00:00
snprintf(buffer, PATH_MAX, "%s/login_stanza.lua", conf.script_path);
if (stat(buffer, &s) == 0) {
L = luaL_newstate();
luaL_openlibs(L);
lua_push_cfunctions(L);
luaL_dofile(L, buffer);
lua_close(L);
do_internal_login = 0;
} else {
do_internal_login = 1;
}
2016-03-25 12:46:35 +00:00
} else {
do_internal_login = 1;
2016-03-25 12:46:35 +00:00
}
if (do_internal_login == 1) {
// bulletins
display_bulletins();
2016-08-10 01:22:55 +00:00
// external login cmd
2016-08-10 01:22:55 +00:00
// display info
2016-08-10 01:22:55 +00:00
display_info();
display_last10_callers(user);
// check email
i = mail_getemailcount(user);
if (i > 0) {
s_printf(get_string(26), i);
} else {
s_printf(get_string(27));
}
2016-08-10 01:22:55 +00:00
mail_scan(user);
2017-09-28 09:11:00 +00:00
file_scan();
2016-08-10 01:22:55 +00:00
automessage_display();
}
record_last10_callers(user);
2016-03-22 01:48:59 +00:00
// main menu
2016-08-10 01:22:55 +00:00
2017-04-12 12:34:08 +00:00
menu_system(conf.root_menu);
do_logout();
2016-04-10 04:10:18 +00:00
dolog("%s is logging out, on node %d", user->loginname, mynode);
2017-03-19 03:48:50 +00:00
broadcast("%s is logging out, on node %d", user->loginname, mynode);
2016-08-10 01:22:55 +00:00
disconnect("Log out");
}
2017-04-12 12:34:08 +00:00
void do_logout() {
2016-09-05 04:05:17 +00:00
char buffer[256];
struct stat s;
lua_State *L;
int ret = 0;
char c;
int result;
int do_internal_logout = 1;
if (conf.script_path != NULL) {
2018-01-30 07:33:57 +00:00
snprintf(buffer, PATH_MAX, "%s/logout_stanza.lua", conf.script_path);
2016-09-05 04:05:17 +00:00
if (stat(buffer, &s) == 0) {
L = luaL_newstate();
luaL_openlibs(L);
lua_push_cfunctions(L);
luaL_loadfile(L, buffer);
result = lua_pcall(L, 0, 1, 0);
if (result) {
dolog("Failed to run script: %s", lua_tostring(L, -1));
do_internal_logout = 1;
lua_close(L);
} else {
do_internal_logout = 0;
}
}
}
if (do_internal_logout == 1) {
2017-04-12 12:34:08 +00:00
s_displayansi("goodbye");
2016-09-05 04:05:17 +00:00
} else {
lua_getglobal(L, "logout");
2017-04-12 12:34:08 +00:00
result = lua_pcall(L, 0, 0, 0);
2016-09-05 04:05:17 +00:00
if (result) {
dolog("Failed to run script: %s", lua_tostring(L, -1));
}
lua_close(L);
}
}
2016-08-10 01:22:55 +00:00
void runbbs(int socket, char *ip) {
runbbs_real(socket, ip, 0);
}
void runbbs_ssh(char *ip) {
struct sigaction si;
setbuf(stdin, NULL);
setbuf(stdout, NULL);
si.sa_handler = sigint_handler;
sigemptyset(&si.sa_mask);
if (sigaction(SIGINT, &si, NULL) == -1) {
dolog("Failed to setup sigint handler.");
exit(1);
}
runbbs_real(-1, ip, 1);
2016-03-22 01:48:59 +00:00
}
int recursive_delete(const char *dir) {
int ret = 0;
FTS *ftsp = NULL;
FTSENT *curr;
char *files[] = { (char *) dir, NULL };
ftsp = fts_open(files, FTS_NOCHDIR | FTS_PHYSICAL | FTS_XDEV, NULL);
if (!ftsp) {
dolog("%s: fts_open failed: %s", dir, strerror(errno));
ret = -1;
goto finish;
}
while ((curr = fts_read(ftsp))) {
switch (curr->fts_info) {
case FTS_NS:
case FTS_DNR:
case FTS_ERR:
dolog("%s: fts_read error: %s", curr->fts_accpath, strerror(curr->fts_errno));
break;
case FTS_DC:
case FTS_DOT:
case FTS_NSOK:
break;
case FTS_D:
break;
case FTS_DP:
case FTS_F:
case FTS_SL:
case FTS_SLNONE:
case FTS_DEFAULT:
if (remove(curr->fts_accpath) < 0) {
dolog("%s: Failed to remove: %s", curr->fts_path, strerror(errno));
ret = -1;
}
break;
}
}
finish:
if (ftsp) {
fts_close(ftsp);
}
return ret;
}
int copy_file(char *src, char *dest) {
FILE *src_file;
FILE *dest_file;
char c;
src_file = fopen(src, "rb");
if (!src_file) {
return -1;
}
dest_file = fopen(dest, "wb");
if (!dest_file) {
fclose(src_file);
return -1;
}
while(1) {
c = fgetc(src_file);
if (!feof(src_file)) {
fputc(c, dest_file);
} else {
break;
}
}
fclose(src_file);
fclose(dest_file);
return 0;
}
2018-01-21 05:02:21 +00:00
2018-01-21 09:05:01 +00:00
char *str_replace(const char *str, const char *from, const char *to) {
/* Adjust each of the below values to suit your needs. */
/* Increment positions cache size initially by this number. */
size_t cache_sz_inc = 16;
/* Thereafter, each time capacity needs to be increased,
* multiply the increment by this factor. */
const size_t cache_sz_inc_factor = 3;
/* But never increment capacity by more than this number. */
const size_t cache_sz_inc_max = 1048576;
char *pret, *ret = NULL;
const char *pstr2, *pstr = str;
size_t i, count = 0;
uintptr_t *pos_cache_tmp, *pos_cache = NULL;
size_t cache_sz = 0;
size_t cpylen, orglen, retlen, tolen, fromlen = strlen(from);
/* Find all matches and cache their positions. */
while ((pstr2 = strstr(pstr, from)) != NULL) {
count++;
/* Increase the cache size when necessary. */
if (cache_sz < count) {
cache_sz += cache_sz_inc;
pos_cache_tmp = realloc(pos_cache, sizeof(*pos_cache) * cache_sz);
if (pos_cache_tmp == NULL) {
goto end_repl_str;
} else pos_cache = pos_cache_tmp;
cache_sz_inc *= cache_sz_inc_factor;
if (cache_sz_inc > cache_sz_inc_max) {
cache_sz_inc = cache_sz_inc_max;
}
}
2018-01-21 05:02:21 +00:00
2018-01-21 09:05:01 +00:00
pos_cache[count-1] = pstr2 - str;
pstr = pstr2 + fromlen;
}
orglen = pstr - str + strlen(pstr);
/* Allocate memory for the post-replacement string. */
if (count > 0) {
tolen = strlen(to);
retlen = orglen + (tolen - fromlen) * count;
} else retlen = orglen;
ret = malloc(retlen + 1);
if (ret == NULL) {
goto end_repl_str;
}
if (count == 0) {
/* If no matches, then just duplicate the string. */
strcpy(ret, str);
} else {
/* Otherwise, duplicate the string whilst performing
* the replacements using the position cache. */
pret = ret;
memcpy(pret, str, pos_cache[0]);
pret += pos_cache[0];
for (i = 0; i < count; i++) {
memcpy(pret, to, tolen);
pret += tolen;
pstr = str + pos_cache[i] + fromlen;
cpylen = (i == count-1 ? orglen : pos_cache[i+1]) - pos_cache[i] - fromlen;
memcpy(pret, pstr, cpylen);
pret += cpylen;
}
ret[retlen] = '\0';
}
end_repl_str:
/* Free the cache and return the post-replacement string,
* which will be NULL in the event of an error. */
free(pos_cache);
return ret;
}