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/bbs.c

711 lines
14 KiB
C
Raw Normal View History

2016-03-22 11:48:59 +10:00
#include <stdio.h>
#include <netdb.h>
#include <netinet/in.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/fcntl.h>
#include <signal.h>
2016-03-23 10:14:31 +10:00
#include <sys/utsname.h>
#include <sys/time.h>
2016-04-10 14:10:18 +10:00
#include <stdarg.h>
#include "bbs.h"
#include "lua/lua.h"
#include "lua/lualib.h"
#include "lua/lauxlib.h"
2016-03-22 11:48:59 +10:00
int mynode;
struct bbs_config conf;
2016-03-22 11:48:59 +10:00
struct user_record *gUser;
int gSocket;
int sshBBS;
2016-03-22 11:48:59 +10:00
int usertimeout;
int timeoutpaused;
2016-03-22 11:48:59 +10:00
2016-04-13 17:42:22 +10:00
char *ipaddress;
2016-04-10 14:10:18 +10:00
void dolog(char *fmt, ...) {
char buffer[512];
struct tm time_now;
time_t timen;
FILE *logfptr;
2016-04-10 14:10:18 +10:00
if (conf.log_path == NULL) return;
2016-04-10 14:10:18 +10:00
timen = time(NULL);
2016-04-10 14:10:18 +10:00
localtime_r(&timen, &time_now);
2016-04-10 14:10:18 +10:00
snprintf(buffer, 512, "%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) {
dolog("Error opening log file!");
2016-04-10 14:10:18 +10:00
return;
}
2016-08-06 14:36:36 +10:00
va_list ap;
va_start(ap, fmt);
vsnprintf(buffer, 512, fmt, ap);
va_end(ap);
2016-08-06 14:36:36 +10:00
fprintf(logfptr, "%02d:%02d:%02d [%s] %s\n", time_now.tm_hour, time_now.tm_min, time_now.tm_sec, ipaddress, buffer);
2016-04-10 14:10:18 +10:00
fclose(logfptr);
}
2016-03-24 17:23:42 +10:00
struct fido_addr *parse_fido_addr(const char *str) {
struct fido_addr *ret = (struct fido_addr *)malloc(sizeof(struct fido_addr));
int c;
int state = 0;
2016-03-24 17:23:42 +10:00
ret->zone = 0;
ret->net = 0;
ret->node = 0;
ret->point = 0;
2016-03-24 17:23:42 +10: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-25 08:27:06 +10:00
ret->zone = ret->zone * 10 + (str[c] - '0');
2016-03-24 17:23:42 +10:00
break;
case 1:
2016-03-25 08:27:06 +10:00
ret->net = ret->net * 10 + (str[c] - '0');
2016-03-24 17:23:42 +10:00
break;
case 2:
2016-03-25 08:27:06 +10:00
ret->node = ret->node * 10 + (str[c] - '0');
2016-03-24 17:23:42 +10:00
break;
case 3:
2016-03-25 08:27:06 +10:00
ret->point = ret->point * 10 + (str[c] - '0');
2016-03-24 17:23:42 +10:00
break;
}
}
break;
default:
free(ret);
return NULL;
}
}
return ret;
}
2016-03-22 11:48:59 +10:00
void timer_handler(int signum) {
if (signum == SIGALRM) {
if (gUser != NULL) {
gUser->timeleft--;
2016-03-22 11:48:59 +10:00
if (gUser->timeleft <= 0) {
2016-08-06 14:36:36 +10:00
s_printf("\r\n\r\nSorry, you're out of time today..\r\n");
disconnect("Out of Time");
2016-03-22 11:48:59 +10:00
}
2016-03-22 11:48:59 +10:00
}
if (timeoutpaused == 0) {
usertimeout--;
}
2016-03-22 11:48:59 +10:00
if (usertimeout <= 0) {
2016-08-06 14:36:36 +10:00
s_printf("\r\n\r\nTimeout waiting for input..\r\n");
disconnect("Timeout");
}
2016-03-22 11:48:59 +10:00
}
}
2016-08-06 14:36:36 +10:00
void s_printf(char *fmt, ...) {
char buffer[512];
va_list ap;
va_start(ap, fmt);
vsnprintf(buffer, 512, fmt, ap);
va_end(ap);
s_putstring(buffer);
}
void s_putchar(char c) {
if (sshBBS) {
putchar(c);
} else {
write(gSocket, &c, 1);
}
2016-03-22 11:48:59 +10:00
}
2016-08-06 14:36:36 +10:00
void s_putstring(char *c) {
if (sshBBS) {
printf("%s", c);
} else {
write(gSocket, c, strlen(c));
}
2016-03-22 11:48:59 +10:00
}
2016-08-06 14:36:36 +10:00
void s_displayansi_p(char *file) {
2016-03-27 10:33:43 +10:00
FILE *fptr;
char c;
2016-03-27 10:33:43 +10:00
fptr = fopen(file, "r");
if (!fptr) {
return;
}
c = fgetc(fptr);
while (!feof(fptr) && c != 0x1a) {
2016-08-06 14:36:36 +10:00
s_putchar(c);
2016-03-27 10:33:43 +10:00
c = fgetc(fptr);
}
fclose(fptr);
}
2016-08-06 14:36:36 +10:00
void s_displayansi(char *file) {
2016-03-22 11:48:59 +10:00
FILE *fptr;
char c;
2016-03-22 11:48:59 +10:00
char buffer[256];
2016-03-22 11:48:59 +10:00
sprintf(buffer, "%s/%s.ans", conf.ansi_path, file);
2016-03-22 11:48:59 +10:00
fptr = fopen(buffer, "r");
if (!fptr) {
return;
}
c = fgetc(fptr);
while (!feof(fptr) && c != 0x1a) {
2016-08-06 14:36:36 +10:00
s_putchar(c);
2016-03-22 11:48:59 +10:00
c = fgetc(fptr);
}
fclose(fptr);
}
2016-08-06 14:36:36 +10:00
char s_getchar() {
2016-03-22 11:48:59 +10:00
unsigned char c;
int len;
2016-03-24 10:46:01 +10:00
do {
if (sshBBS) {
len = read(stdin, &c, 1);
} else {
len = read(gSocket, &c, 1);
}
2016-03-22 11:48:59 +10:00
if (len == 0) {
2016-08-06 14:36:36 +10:00
disconnect("Socket Closed");
2016-03-22 11:48:59 +10:00
}
if (!sshBBS) {
while (c == 255) {
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
} else if (c == 255) {
usertimeout = 10;
return c;
}
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
}
len = read(gSocket, &c, 1);
if (len == 0) {
disconnect("Socket Closed");
}
}
}
2016-03-24 10:46:01 +10:00
2016-03-24 10:46:01 +10:00
if (c == '\r') {
if (len == 0) {
2016-08-06 14:36:36 +10:00
disconnect("Socket Closed");
2016-03-24 10:46:01 +10:00
}
2016-03-23 10:14:31 +10:00
}
2016-03-24 10:46:01 +10:00
} while (c == '\n');
2016-03-22 11:48:59 +10:00
usertimeout = 10;
return (char)c;
}
2016-08-06 14:36:36 +10:00
char s_getc() {
char c = s_getchar();
2016-03-22 11:48:59 +10:00
2016-08-06 14:36:36 +10:00
s_putchar(c);
2016-03-22 11:48:59 +10:00
return (char)c;
}
2016-08-06 14:36:36 +10:00
void s_readstring(char *buffer, int max) {
2016-03-22 11:48:59 +10:00
int i;
char c;
2016-03-23 10:14:31 +10:00
memset(buffer, 0, max);
2016-03-22 11:48:59 +10:00
for (i=0;i<max;i++) {
2016-08-06 14:36:36 +10:00
c = s_getchar();
2016-03-22 11:48:59 +10:00
if ((c == '\b' || c == 127) && i > 0) {
buffer[i-1] = '\0';
i -= 2;
2016-08-06 14:36:36 +10:00
s_printf("\e[D \e[D");
2016-03-22 11:48:59 +10:00
continue;
2016-08-04 19:36:29 +10:00
} else if (c == '\b' || c == 127) {
2016-08-04 19:49:56 +10:00
i -= 1;
2016-08-04 19:36:29 +10:00
continue;
}
2016-03-22 11:48:59 +10:00
if (c == '\n' || c == '\r') {
return;
}
2016-08-06 14:36:36 +10:00
s_putchar(c);
2016-03-22 11:48:59 +10:00
buffer[i] = c;
buffer[i+1] = '\0';
}
}
2016-08-06 14:36:36 +10:00
void s_readpass(char *buffer, int max) {
2016-03-22 11:48:59 +10:00
int i;
char c;
2016-03-22 11:48:59 +10:00
for (i=0;i<max;i++) {
2016-08-06 14:36:36 +10:00
c = s_getchar();
2016-03-22 11:48:59 +10:00
if ((c == '\b' || c == 127) && i > 0) {
buffer[i-1] = '\0';
i-=2;
2016-08-06 14:36:36 +10:00
s_printf("\e[D \e[D");
2016-03-22 11:48:59 +10:00
continue;
2016-08-04 19:36:29 +10:00
} else if (c == '\b' || c == 127) {
2016-08-04 19:49:56 +10:00
i -= 1;
2016-08-04 19:36:29 +10:00
continue;
}
2016-03-22 11:48:59 +10:00
if (c == '\n' || c == '\r') {
return;
}
2016-08-06 14:36:36 +10:00
s_putchar('*');
2016-03-22 11:48:59 +10:00
buffer[i] = c;
buffer[i+1] = '\0';
}
}
2016-08-06 14:36:36 +10:00
void disconnect(char *calledby) {
2016-03-22 11:48:59 +10:00
char buffer[256];
if (gUser != NULL) {
save_user(gUser);
}
dolog("Node %d disconnected (%s)", mynode, calledby);
2016-03-22 11:48:59 +10:00
sprintf(buffer, "%s/nodeinuse.%d", conf.bbs_path, mynode);
remove(buffer);
if (!sshBBS) {
close(gSocket);
}
2016-03-22 11:48:59 +10:00
exit(0);
}
void record_last10_callers(struct user_record *user) {
2016-03-23 15:21:53 +10:00
struct last10_callers new_entry;
struct last10_callers callers[10];
int i,j;
2016-08-06 14:55:18 +10:00
FILE *fptr = fopen("last10.dat", "rb");
2016-03-23 15:21:53 +10: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;
}
if (strcasecmp(conf.sysop_name, user->loginname) != 0 ) {
2016-03-23 15:21:53 +10: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-03-23 15:21:53 +10: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-06 14:36:36 +10:00
void display_last10_callers(struct user_record *user) {
struct last10_callers callers[10];
2016-08-06 14:55:18 +10:00
int i,z;
2016-08-06 14:36:36 +10:00
struct tm l10_time;
FILE *fptr = fopen("last10.dat", "rb");
2016-08-06 14:36:36 +10:00
s_printf("\r\n\e[1;37mLast 10 callers:\r\n");
s_printf("\e[1;30m-------------------------------------------------------------------------------\r\n");
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;
}
for (z=0;z<i;z++) {
localtime_r(&callers[z].time, &l10_time);
2016-08-06 14:36:36 +10:00
s_printf("\e[1;37m%-16s \e[1;36m%-32s \e[1;32m%02d:%02d %02d-%02d-%02d\e[0m\r\n", 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);
}
2016-08-06 14:36:36 +10:00
s_printf("\e[1;30m-------------------------------------------------------------------------------\e[0m\r\n");
s_printf("Press any key to continue...\r\n");
s_getc();
2016-03-23 15:21:53 +10:00
}
2016-08-06 14:36:36 +10:00
void display_info() {
struct utsname name;
2016-04-01 14:33:37 +10:00
2016-03-23 10:14:31 +10:00
uname(&name);
2016-08-06 14:36:36 +10:00
s_printf("\r\n\r\n\e[1;37mSystem Information\r\n");
s_printf("\e[1;30m----------------------------------------------\r\n");
s_printf("\e[1;32mBBS Name : \e[1;37m%s\r\n", conf.bbs_name);
s_printf("\e[1;32mSysOp Name : \e[1;37m%s\r\n", conf.sysop_name);
s_printf("\e[1;32mNode : \e[1;37m%d\r\n", mynode);
s_printf("\e[1;32mBBS Version : \e[1;37mMagicka %d.%d (%s)\r\n", VERSION_MAJOR, VERSION_MINOR, VERSION_STR);
s_printf("\e[1;32mSystem : \e[1;37m%s (%s)\r\n", name.sysname, name.machine);
s_printf("\e[1;30m----------------------------------------------\e[0m\r\n");
s_printf("Press any key to continue...\r\n");
s_getc();
2016-03-23 10:14:31 +10:00
}
2016-08-06 14:36:36 +10:00
void automessage_write(struct user_record *user) {
2016-04-18 20:34:07 +10:00
FILE *fptr;
char automsg[450];
2016-08-06 14:36:36 +10:00
char buffer[76];
2016-04-18 20:34:07 +10:00
int i;
struct tm timenow;
time_t timen;
2016-04-18 20:34:07 +10:00
memset(automsg, 0, 450);
2016-08-06 14:36:36 +10:00
memset(buffer, 0, 76);
2016-04-18 20:34:07 +10:00
if (user->sec_level >= conf.automsgwritelvl) {
timen = time(NULL);
localtime_r(&timen, &timenow);
sprintf(automsg, "Automessage Posted by %s @ %s", user->loginname, asctime(&timenow));
2016-04-18 20:34:07 +10:00
automsg[strlen(automsg) - 1] = '\r';
automsg[strlen(automsg)] = '\n';
2016-08-06 14:36:36 +10:00
s_printf("\r\nEnter your message (4 lines):\r\n");
2016-04-18 20:34:07 +10:00
for (i=0;i<4;i++) {
2016-08-06 14:36:36 +10:00
s_printf("\r\n%d: ", i);
s_readstring(buffer, 75);
2016-04-18 20:34:07 +10:00
strcat(automsg, buffer);
strcat(automsg, "\r\n");
}
2016-04-18 20:34:07 +10:00
fptr = fopen("automessage.txt", "w");
if (fptr) {
fwrite(automsg, strlen(automsg), 1, fptr);
fclose(fptr);
} else {
dolog("Unable to open automessage.txt for writing");
2016-04-18 20:34:07 +10:00
}
}
2016-04-18 20:34:07 +10:00
}
2016-08-06 14:36:36 +10:00
void automessage_display() {
2016-04-18 20:34:07 +10:00
struct stat s;
FILE *fptr;
char buffer[90];
int i;
2016-08-06 14:36:36 +10:00
s_printf("\r\n\r\n");
2016-04-18 20:34:07 +10: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-06 14:36:36 +10:00
s_printf(buffer);
2016-04-18 20:34:07 +10:00
}
fclose(fptr);
} else {
dolog("Error opening automessage.txt");
}
2016-04-18 20:34:07 +10:00
} else {
2016-08-06 14:36:36 +10:00
s_printf("No automessage!\r\n");
2016-04-18 20:34:07 +10:00
}
2016-08-06 14:36:36 +10:00
s_printf("\e[0mPress any key to continue...\r\n");
s_getc();
2016-04-18 20:34:07 +10:00
}
void runbbs_real(int socket, char *ip, int ssh) {
2016-03-22 11:48:59 +10:00
char buffer[256];
char password[17];
struct stat s;
FILE *nodefile;
int i;
char iac_echo[] = {255, 251, 1, '\0'};
char iac_sga[] = {255, 251, 3, '\0'};
struct user_record *user;
struct tm thetime;
struct tm oldtime;
time_t now;
struct itimerval itime;
struct sigaction sa;
lua_State *L;
int do_internal_login = 0;
2016-04-13 17:42:22 +10:00
ipaddress = ip;
if (!ssh) {
write(socket, iac_echo, 3);
write(socket, iac_sga, 3);
gUser = NULL;
sshBBS = 0;
} else {
sshBBS = 1;
}
2016-08-06 14:36:36 +10:00
gSocket = socket;
2016-08-06 14:36:36 +10:00
s_printf("Magicka BBS v%d.%d (%s), Loading...\r\n", VERSION_MAJOR, VERSION_MINOR, VERSION_STR);
// find out which node we are
2016-03-22 11:48:59 +10:00
mynode = 0;
for (i=1;i<=conf.nodes;i++) {
sprintf(buffer, "%s/nodeinuse.%d", conf.bbs_path, i);
if (stat(buffer, &s) != 0) {
mynode = i;
nodefile = fopen(buffer, "w");
if (!nodefile) {
dolog("Error opening nodefile!");
2016-03-22 11:48:59 +10:00
close(socket);
exit(1);
}
2016-03-22 11:48:59 +10:00
fputs("UNKNOWN", nodefile);
fclose(nodefile);
2016-03-22 11:48:59 +10:00
break;
}
}
2016-03-22 11:48:59 +10:00
if (mynode == 0) {
2016-08-06 14:36:36 +10:00
s_printf("Sorry, all nodes are in use. Please try later\r\n");
if (!ssh) {
close(socket);
}
2016-03-22 11:48:59 +10:00
exit(1);
}
2016-08-06 14:36:36 +10:00
2016-03-22 11:48:59 +10:00
usertimeout = 10;
timeoutpaused = 0;
2016-03-22 11:48:59 +10:00
memset (&sa, 0, sizeof (sa));
sa.sa_handler = &timer_handler;
sa.sa_flags = SA_RESTART;
sigaction (SIGALRM, &sa, 0);
2016-03-22 11:48:59 +10: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-03-22 11:48:59 +10:00
setitimer (ITIMER_REAL, &itime, 0);
2016-08-06 14:36:36 +10:00
s_displayansi("issue");
if (!ssh) {
s_printf("\e[0mEnter your Login Name or NEW to create an account\r\n");
s_printf("Login:> ");
s_readstring(buffer, 25);
if (strcasecmp(buffer, "new") == 0) {
user = new_user();
} else {
s_printf("\r\nPassword:> ");
s_readpass(password, 16);
user = check_user_pass(buffer, password);
if (user == NULL) {
s_printf("\r\nIncorrect Login.\r\n");
disconnect("Incorrect Login");
}
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("\r\nYou are already logged in.\r\n");
disconnect("Already Logged in");
}
fclose(nodefile);
}
}
}
} else {
if (gUser != NULL) {
user = gUser;
s_printf("\e[0mWelcome back %s. Press enter to log in...\r\n", gUser->loginname);
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("\r\nYou are already logged in.\r\n");
disconnect("Already Logged in");
}
fclose(nodefile);
}
}
} else {
gUser = new_user();
user = gUser;
}
}
2016-03-22 11:48:59 +10:00
sprintf(buffer, "%s/nodeinuse.%d", conf.bbs_path, mynode);
nodefile = fopen(buffer, "w");
if (!nodefile) {
dolog("Error opening nodefile!");
2016-03-22 11:48:59 +10:00
close(socket);
exit(1);
}
2016-03-22 11:48:59 +10:00
fputs(user->loginname, nodefile);
fclose(nodefile);
2016-03-22 11:48:59 +10:00
// do post-login
2016-04-10 14:10:18 +10:00
dolog("%s logged in, on node %d", user->loginname, mynode);
2016-03-22 11:48:59 +10:00
// check time left
now = time(NULL);
localtime_r(&now, &thetime);
localtime_r(&user->laston, &oldtime);
2016-03-22 11:48:59 +10: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-03-22 11:48:59 +10:00
gUser = user;
2016-03-23 16:31:00 +10:00
user->timeson++;
if (conf.script_path != NULL) {
sprintf(buffer, "%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 22:46:35 +10:00
} else {
do_internal_login = 1;
2016-03-25 22:46:35 +10:00
}
if (do_internal_login == 1) {
// bulletins
i = 0;
sprintf(buffer, "%s/bulletin%d.ans", conf.ansi_path, i);
while (stat(buffer, &s) == 0) {
sprintf(buffer, "bulletin%d", i);
2016-08-06 14:36:36 +10:00
s_displayansi(buffer);
s_printf("\e[0mPress any key to continue...\r\n");
s_getc();
i++;
sprintf(buffer, "%s/bulletin%d.ans", conf.ansi_path, i);
}
// external login cmd
// display info
2016-08-06 14:36:36 +10:00
display_info();
2016-08-06 14:36:36 +10:00
display_last10_callers(user);
// check email
i = mail_getemailcount(user);
if (i > 0) {
2016-08-06 14:36:36 +10:00
s_printf("\r\nYou have %d e-mail(s) in your inbox.\r\n", i);
} else {
2016-08-06 14:36:36 +10:00
s_printf("\r\nYou have no e-mail.\r\n");
}
2016-08-06 14:36:36 +10:00
mail_scan(user);
2016-08-06 14:36:36 +10:00
automessage_display();
}
record_last10_callers(user);
2016-03-22 11:48:59 +10:00
// main menu
2016-08-06 14:36:36 +10:00
main_menu(user);
2016-08-06 14:36:36 +10:00
s_displayansi("goodbye");
2016-04-10 14:10:18 +10:00
dolog("%s is logging out, on node %d", user->loginname, mynode);
2016-08-06 14:36:36 +10:00
disconnect("Log out");
2016-03-22 11:48:59 +10:00
}
void runbbs(int socket, char *ip) {
runbbs_real(socket, ip, 0);
}
void runbbs_ssh(char *ip) {
runbbs_real(-1, ip, 1);
}