873 lines
22 KiB
C
873 lines
22 KiB
C
/* Functions common to both dgamelaunch itself and dgl-wall. */
|
|
|
|
#include "dgamelaunch.h"
|
|
#include "ttyrec.h"
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include <dirent.h>
|
|
#include <stdio.h>
|
|
#include <fcntl.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <signal.h>
|
|
#include <unistd.h>
|
|
#include <pwd.h>
|
|
#include <grp.h>
|
|
#include <curses.h>
|
|
|
|
extern FILE* yyin;
|
|
extern int yyparse ();
|
|
extern void (*g_chain_winch)(int);
|
|
|
|
/* Data structures */
|
|
struct dg_config **myconfig = NULL;
|
|
struct dg_config defconfig = {
|
|
/* game_path = */ "/bin/nethack",
|
|
/* game_name = */ "NetHack",
|
|
/* game_id = */ NULL,
|
|
/* shortname = */ "NH",
|
|
/* rcfile = */ NULL, /*"/dgl-default-rcfile",*/
|
|
/* ttyrecdir =*/ "%ruserdata/%n/ttyrec/",
|
|
/* spool = */ "/var/mail/",
|
|
/* inprogressdir = */ "%rinprogress/",
|
|
/* num_args = */ 0,
|
|
/* bin_args = */ NULL,
|
|
/* rc_fmt = */ "%rrcfiles/%n.nethackrc", /* [dglroot]rcfiles/[username].nethackrc */
|
|
/* cmdqueue = */ NULL,
|
|
/* postcmdqueue = */ NULL,
|
|
/* max_idle_time = */ 0,
|
|
/* extra_info_file = */ NULL,
|
|
/* encoding */ 0
|
|
};
|
|
|
|
char* config = NULL;
|
|
int silent = 0;
|
|
int loggedin = 0;
|
|
char *chosen_name;
|
|
int num_games = 0;
|
|
|
|
int shm_n_games = 200;
|
|
|
|
int dgl_local_COLS = -1, dgl_local_LINES = -1;
|
|
int curses_resize = 0;
|
|
|
|
int selected_game = 0;
|
|
int return_from_submenu = 0;
|
|
|
|
mode_t default_fmode = S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH;
|
|
|
|
struct dg_globalconfig globalconfig;
|
|
|
|
void
|
|
sigwinch_func(int sig)
|
|
{
|
|
signal(SIGWINCH, sigwinch_func);
|
|
curses_resize = 1;
|
|
g_chain_winch(sig);
|
|
}
|
|
|
|
void
|
|
term_resize_check()
|
|
{
|
|
if ((COLS == dgl_local_COLS) && (LINES == dgl_local_LINES) && !curses_resize) return;
|
|
|
|
signal(SIGWINCH, SIG_IGN);
|
|
endwin();
|
|
initcurses();
|
|
dgl_local_COLS = COLS;
|
|
dgl_local_LINES = LINES;
|
|
curses_resize = 0;
|
|
signal(SIGWINCH, sigwinch_func);
|
|
}
|
|
|
|
int
|
|
check_retard(int reset)
|
|
{
|
|
static int retardation = 0; /* counter for retarded clients & flooding */
|
|
if (reset) retardation = 0;
|
|
else retardation++;
|
|
return ((retardation > 20) ? 1 : 0);
|
|
}
|
|
|
|
|
|
struct dg_menu *
|
|
dgl_find_menu(char *menuname)
|
|
{
|
|
struct dg_menulist *tmp = globalconfig.menulist;
|
|
|
|
while (tmp) {
|
|
if (!strcmp(tmp->menuname, menuname)) return tmp->menu;
|
|
tmp = tmp->next;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* replace following codes with variables:
|
|
* %u == shed_uid (number)
|
|
* %n == user name (string; gotten from 'me', or from 'plrname' if 'me' is null)
|
|
* %r == chroot (string) (aka "dglroot" config var)
|
|
* %g == game name
|
|
* %s == short game name
|
|
* %t == ttyrec file (full path&name) of the last game played.
|
|
*/
|
|
char *
|
|
dgl_format_str(int game, struct dg_user *me, char *str, char *plrname)
|
|
{
|
|
static char buf[1024];
|
|
char *f, *p, *end;
|
|
int ispercent = 0;
|
|
int isbackslash = 0;
|
|
|
|
if (!str) return NULL;
|
|
|
|
f = str;
|
|
p = buf;
|
|
end = buf + sizeof(buf) - 10;
|
|
|
|
while (*f) {
|
|
if (ispercent) {
|
|
switch (*f) {
|
|
case 'u':
|
|
snprintf (p, end + 1 - p, "%d", globalconfig.shed_uid);
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
case 'N':
|
|
if (me) *p = me->username[0];
|
|
else if (plrname) *p = plrname[0];
|
|
else return NULL;
|
|
p++;
|
|
*p = '\0';
|
|
break;
|
|
case 'n':
|
|
if (me) snprintf (p, end + 1 - p, "%s", me->username);
|
|
else if (plrname) snprintf(p, end + 1 - p, "%s", plrname);
|
|
else return NULL;
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
case 'g':
|
|
if (game >= 0 && game < num_games && myconfig[game]) snprintf (p, end + 1 - p, "%s", myconfig[game]->game_name);
|
|
else return NULL;
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
case 's':
|
|
if (game >= 0 && game < num_games && myconfig[game]) snprintf (p, end + 1 - p, "%s", myconfig[game]->shortname);
|
|
else return NULL;
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
case 'r':
|
|
snprintf (p, end + 1 - p, "%s", globalconfig.dglroot);
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
case 't':
|
|
snprintf (p, end + 1 - p, "%s", last_ttyrec);
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
default:
|
|
*p = *f;
|
|
if (p < end)
|
|
p++;
|
|
}
|
|
ispercent = 0;
|
|
} else if (isbackslash) {
|
|
switch (*f) {
|
|
case 'a': *p = '\007'; break;
|
|
case 'b': *p = '\010'; break;
|
|
case 't': *p = '\011'; break;
|
|
case 'n': *p = '\012'; break;
|
|
case 'v': *p = '\013'; break;
|
|
case 'f': *p = '\014'; break;
|
|
case 'r': *p = '\015'; break;
|
|
case 'e': *p = '\033'; break;
|
|
default: *p = *f;
|
|
}
|
|
if (p < end)
|
|
p++;
|
|
isbackslash = 0;
|
|
} else {
|
|
if (*f == '%') {
|
|
ispercent = 1;
|
|
} else if (*f == '\\') {
|
|
isbackslash = 1;
|
|
} else {
|
|
*p = *f;
|
|
if (p < end)
|
|
p++;
|
|
}
|
|
}
|
|
f++;
|
|
}
|
|
*p = '\0';
|
|
|
|
return buf;
|
|
}
|
|
|
|
int
|
|
dgl_exec_cmdqueue(struct dg_cmdpart *queue, int game, struct dg_user *me)
|
|
{
|
|
int i;
|
|
struct dg_cmdpart *tmp = queue;
|
|
char *p1;
|
|
char *p2;
|
|
int played = 0;
|
|
|
|
if (!queue) return 1;
|
|
|
|
p1 = (char *)malloc(1024);
|
|
p2 = (char *)malloc(1024);
|
|
|
|
if (!p1 || !p2) return 1;
|
|
|
|
return_from_submenu = 0;
|
|
|
|
while (tmp && !return_from_submenu) {
|
|
if (tmp->param1) strcpy(p1, dgl_format_str(game, me, tmp->param1, NULL));
|
|
if (tmp->param2) strcpy(p2, dgl_format_str(game, me, tmp->param2, NULL));
|
|
|
|
switch (tmp->cmd) {
|
|
default: break;
|
|
case DGLCMD_RAWPRINT:
|
|
if (p1) fprintf(stdout, "%s", p1);
|
|
break;
|
|
case DGLCMD_MKDIR:
|
|
if (p1 && (access(p1, F_OK) != 0)) mkdir(p1, 0755);
|
|
break;
|
|
case DGLCMD_UNLINK:
|
|
if (p1 && (access(p1, F_OK) == 0)) unlink(p1);
|
|
break;
|
|
case DGLCMD_CHDIR:
|
|
if (p1) {
|
|
if (chdir(p1) == -1) {
|
|
debug_write("chdir-command failed");
|
|
graceful_exit(123);
|
|
}
|
|
}
|
|
break;
|
|
case DGLCMD_IF_NX_CP:
|
|
if (p1 && p2) {
|
|
FILE *tmpfile;
|
|
tmpfile = fopen(p2, "r");
|
|
if (tmpfile) {
|
|
fclose(tmpfile);
|
|
break;
|
|
}
|
|
}
|
|
/* else fall through to cp */
|
|
case DGLCMD_CP:
|
|
if (p1 && p2) {
|
|
FILE *cannedf, *newfile;
|
|
char buf[1024];
|
|
size_t bytes;
|
|
/* FIXME: use nethack-themed error messages here, as per write_canned_rcfile() */
|
|
if (!(cannedf = fopen (p1, "r"))) break;
|
|
if (!(newfile = fopen (p2, "w"))) break;
|
|
while ((bytes = fread (buf, 1, 1024, cannedf)) > 0) {
|
|
if (fwrite (buf, 1, bytes, newfile) != bytes) {
|
|
if (ferror (newfile)) {
|
|
fclose (cannedf);
|
|
fclose (newfile);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
fclose (cannedf);
|
|
fclose (newfile);
|
|
chmod (p2, default_fmode);
|
|
}
|
|
break;
|
|
case DGLCMD_EXEC:
|
|
if (p1 && p2) {
|
|
pid_t child;
|
|
char *myargv[3];
|
|
|
|
myargv[0] = p1;
|
|
myargv[1] = p2;
|
|
myargv[2] = 0;
|
|
|
|
clear();
|
|
refresh();
|
|
endwin();
|
|
idle_alarm_set_enabled(0);
|
|
child = fork();
|
|
if (child == -1) {
|
|
perror("fork");
|
|
debug_write("exec-command fork failed");
|
|
graceful_exit(114);
|
|
} else if (child == 0) {
|
|
execvp(p1, myargv);
|
|
exit(0);
|
|
} else
|
|
waitpid(child, NULL, 0);
|
|
idle_alarm_set_enabled(1);
|
|
initcurses();
|
|
check_retard(1);
|
|
}
|
|
break;
|
|
case DGLCMD_SETENV:
|
|
if (p1 && p2) mysetenv(p1, p2, 1);
|
|
break;
|
|
case DGLCMD_CHPASSWD:
|
|
if (loggedin) changepw(1);
|
|
break;
|
|
case DGLCMD_CHMAIL:
|
|
if (loggedin) change_email();
|
|
break;
|
|
case DGLCMD_WATCH_MENU:
|
|
inprogressmenu(-1);
|
|
break;
|
|
case DGLCMD_LOGIN:
|
|
if (!loggedin) loginprompt(0);
|
|
if (loggedin) runmenuloop(dgl_find_menu(get_mainmenu_name()));
|
|
break;
|
|
case DGLCMD_REGISTER:
|
|
if (!loggedin && globalconfig.allow_registration) newuser();
|
|
break;
|
|
case DGLCMD_QUIT:
|
|
debug_write("command: quit");
|
|
graceful_exit(0);
|
|
/* break; */
|
|
case DGLCMD_SUBMENU:
|
|
if (p1)
|
|
runmenuloop(dgl_find_menu(p1));
|
|
break;
|
|
case DGLCMD_RETURN:
|
|
return_from_submenu = 1;
|
|
break;
|
|
case DGLCMD_PLAY_IF_EXIST:
|
|
if (!(loggedin && me && p1 && p2)) break;
|
|
{
|
|
FILE *tmpfile;
|
|
tmpfile = fopen(p2, "r");
|
|
if (tmpfile) {
|
|
fclose(tmpfile);
|
|
} else break;
|
|
}
|
|
/* else fall through to playgame */
|
|
case DGLCMD_PLAYGAME:
|
|
if (loggedin && me && p1 && !played) {
|
|
int userchoice, i;
|
|
char *tmpstr;
|
|
for (userchoice = 0; userchoice < num_games; userchoice++) {
|
|
if (!strcmp(myconfig[userchoice]->game_id, p1) || !strcmp(myconfig[userchoice]->game_name, p1) || !strcmp(myconfig[userchoice]->shortname, p1)) {
|
|
if (purge_stale_locks(userchoice)) {
|
|
if (myconfig[userchoice]->rcfile) {
|
|
if (access (dgl_format_str(userchoice, me, myconfig[userchoice]->rc_fmt, NULL), R_OK) == -1)
|
|
write_canned_rcfile (userchoice, dgl_format_str(userchoice, me, myconfig[userchoice]->rc_fmt, NULL));
|
|
}
|
|
|
|
setproctitle("%s [playing %s]", me->username, myconfig[userchoice]->shortname);
|
|
|
|
clear();
|
|
refresh();
|
|
endwin ();
|
|
|
|
/* first run the generic "do these when a game is started" commands */
|
|
dgl_exec_cmdqueue(globalconfig.cmdqueue[DGLTIME_GAMESTART], userchoice, me);
|
|
/* then run the game-specific commands */
|
|
dgl_exec_cmdqueue(myconfig[userchoice]->cmdqueue, userchoice, me);
|
|
|
|
/* fix the variables in the arguments */
|
|
for (i = 0; i < myconfig[userchoice]->num_args; i++) {
|
|
tmpstr = strdup(dgl_format_str(userchoice, me, myconfig[userchoice]->bin_args[i], NULL));
|
|
free(myconfig[userchoice]->bin_args[i]);
|
|
myconfig[userchoice]->bin_args[i] = tmpstr;
|
|
}
|
|
|
|
signal(SIGWINCH, SIG_DFL);
|
|
signal(SIGINT, SIG_DFL);
|
|
signal(SIGQUIT, SIG_DFL);
|
|
signal(SIGTERM, SIG_DFL);
|
|
idle_alarm_set_enabled(0);
|
|
/* launch program */
|
|
ttyrec_main (userchoice, me->username,
|
|
dgl_format_str(userchoice, me, myconfig[userchoice]->ttyrecdir, NULL),
|
|
gen_ttyrec_filename());
|
|
idle_alarm_set_enabled(1);
|
|
played = 1;
|
|
/* lastly, run the generic "do these when a game is left" commands */
|
|
signal (SIGHUP, catch_sighup);
|
|
signal (SIGINT, catch_sighup);
|
|
signal (SIGQUIT, catch_sighup);
|
|
signal (SIGTERM, catch_sighup);
|
|
signal(SIGWINCH, sigwinch_func);
|
|
|
|
dgl_exec_cmdqueue(myconfig[userchoice]->postcmdqueue, userchoice, me);
|
|
|
|
dgl_exec_cmdqueue(globalconfig.cmdqueue[DGLTIME_GAMEEND], userchoice, me);
|
|
|
|
setproctitle ("%s", me->username);
|
|
initcurses ();
|
|
check_retard(1); /* reset retard counter */
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
tmp = tmp->next;
|
|
}
|
|
|
|
free(p1);
|
|
free(p2);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
sort_game_username(const void *g1, const void *g2)
|
|
{
|
|
const struct dg_game *game1 = *(const struct dg_game **)g1;
|
|
const struct dg_game *game2 = *(const struct dg_game **)g2;
|
|
return strcasecmp(game1->name, game2->name);
|
|
}
|
|
|
|
time_t sort_ctime;
|
|
|
|
static int
|
|
sort_game_idletime(const void *g1, const void *g2)
|
|
{
|
|
const struct dg_game *game1 = *(const struct dg_game **)g1;
|
|
const struct dg_game *game2 = *(const struct dg_game **)g2;
|
|
if ((sort_ctime - game1->idle_time < 5) && (sort_ctime - game2->idle_time < 5))
|
|
return strcasecmp(game1->name, game2->name);
|
|
if (game2->idle_time != game1->idle_time)
|
|
return difftime(game2->idle_time, game1->idle_time);
|
|
else
|
|
return strcasecmp(game1->name, game2->name);
|
|
}
|
|
|
|
static int
|
|
sort_game_extrainfo(const void *g1, const void *g2)
|
|
{
|
|
const int extra_weight1 =
|
|
(*(const struct dg_game **) g1)->extra_info_weight;
|
|
const int extra_weight2 =
|
|
(*(const struct dg_game **) g2)->extra_info_weight;
|
|
return dglsign(extra_weight2 - extra_weight1);
|
|
}
|
|
|
|
static int
|
|
sort_game_gamenum(const void *g1, const void *g2)
|
|
{
|
|
const struct dg_game *game1 = *(const struct dg_game **)g1;
|
|
const struct dg_game *game2 = *(const struct dg_game **)g2;
|
|
if (game2->gamenum != game1->gamenum)
|
|
return dglsign(game2->gamenum - game1->gamenum);
|
|
else
|
|
return strcasecmp(game1->name, game2->name);
|
|
}
|
|
|
|
static int
|
|
sort_game_windowsize(const void *g1, const void *g2)
|
|
{
|
|
const struct dg_game *game1 = *(const struct dg_game **)g1;
|
|
const struct dg_game *game2 = *(const struct dg_game **)g2;
|
|
if (game2->ws_col != game1->ws_col)
|
|
return dglsign(game1->ws_col - game2->ws_col);
|
|
if (game2->ws_row != game1->ws_row)
|
|
return dglsign(game1->ws_row - game2->ws_row);
|
|
return strcasecmp(game1->name, game2->name);
|
|
}
|
|
|
|
static int
|
|
sort_game_starttime(const void *g1, const void *g2)
|
|
{
|
|
const struct dg_game *game1 = *(const struct dg_game **)g1;
|
|
const struct dg_game *game2 = *(const struct dg_game **)g2;
|
|
int i = strcmp(game1->date, game2->date);
|
|
if (!i)
|
|
i = strcmp(game1->time, game2->time);
|
|
if (!i)
|
|
return strcasecmp(game1->name, game2->name);
|
|
return i;
|
|
}
|
|
|
|
static int
|
|
sort_game_watchers(const void *g1, const void *g2)
|
|
{
|
|
const struct dg_game *game1 = *(const struct dg_game **)g1;
|
|
const struct dg_game *game2 = *(const struct dg_game **)g2;
|
|
int i = dglsign(game2->nwatchers - game1->nwatchers);
|
|
if (!i && (sort_ctime - game1->idle_time < 5) && (sort_ctime - game2->idle_time < 5))
|
|
return strcasecmp(game1->name, game2->name);
|
|
if (!i)
|
|
i = dglsign(game2->idle_time - game1->idle_time);
|
|
if (!i)
|
|
return strcasecmp(game1->name, game2->name);
|
|
return i;
|
|
}
|
|
|
|
struct dg_game **
|
|
sort_games (struct dg_game **games, int len, dg_sortmode sortmode)
|
|
{
|
|
switch (sortmode) {
|
|
case SORTMODE_USERNAME: qsort(games, len, sizeof(struct dg_game *), sort_game_username); break;
|
|
case SORTMODE_GAMENUM: qsort(games, len, sizeof(struct dg_game *), sort_game_gamenum); break;
|
|
case SORTMODE_WINDOWSIZE: qsort(games, len, sizeof(struct dg_game *), sort_game_windowsize); break;
|
|
case SORTMODE_IDLETIME:
|
|
(void) time(&sort_ctime);
|
|
qsort(games, len, sizeof(struct dg_game *), sort_game_idletime);
|
|
break;
|
|
case SORTMODE_DURATION:
|
|
case SORTMODE_STARTTIME: qsort(games, len, sizeof(struct dg_game *), sort_game_starttime); break;
|
|
|
|
case SORTMODE_EXTRA_INFO:
|
|
qsort(games, len, sizeof(struct dg_game *),
|
|
sort_game_extrainfo);
|
|
break;
|
|
|
|
#ifdef USE_SHMEM
|
|
case SORTMODE_WATCHERS:
|
|
(void) time(&sort_ctime);
|
|
qsort(games, len, sizeof(struct dg_game *), sort_game_watchers);
|
|
break;
|
|
#endif
|
|
default: ;
|
|
}
|
|
return games;
|
|
}
|
|
|
|
#ifdef USE_DEBUGFILE
|
|
void
|
|
debug_write(char *str)
|
|
{
|
|
FILE *fp;
|
|
fp = fopen("/dgldebug.log", "a");
|
|
if (!fp) return;
|
|
fprintf(fp, "%s\n", str);
|
|
fclose(fp);
|
|
|
|
}
|
|
#endif /* USE_DEBUGFILE */
|
|
|
|
void
|
|
free_populated_games(struct dg_game **games, int len)
|
|
{
|
|
int i;
|
|
if (!games || (len < 1)) return;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
if (games[i]->ttyrec_fn) free(games[i]->ttyrec_fn);
|
|
if (games[i]->name) free(games[i]->name);
|
|
if (games[i]->date) free(games[i]->date);
|
|
if (games[i]->time) free(games[i]->time);
|
|
if (games[i]->extra_info) free(games[i]->extra_info);
|
|
free(games[i]);
|
|
}
|
|
free(games);
|
|
}
|
|
|
|
static
|
|
void
|
|
game_read_extra_info(struct dg_game *game, const char *extra_info_file)
|
|
{
|
|
FILE *ei = NULL;
|
|
char *sep = NULL;
|
|
char buffer[120];
|
|
int buflen;
|
|
|
|
if (game->extra_info) {
|
|
free(game->extra_info);
|
|
game->extra_info = NULL;
|
|
}
|
|
game->extra_info_weight = 0;
|
|
|
|
if (!extra_info_file)
|
|
return;
|
|
|
|
if (!(ei = fopen(extra_info_file, "r")))
|
|
return;
|
|
*buffer = 0;
|
|
fgets(buffer, sizeof buffer, ei);
|
|
fclose(ei);
|
|
|
|
buflen = strlen(buffer);
|
|
if (buflen && buffer[buflen - 1] == '\n')
|
|
buffer[buflen - 1] = 0;
|
|
|
|
/* The extra info file format is <sort-weight>|<info> */
|
|
sep = strchr(buffer, '|');
|
|
game->extra_info = strdup(sep? sep + 1 : buffer);
|
|
|
|
if (sep) {
|
|
*sep = 0;
|
|
game->extra_info_weight = atoi(buffer);
|
|
}
|
|
}
|
|
|
|
struct dg_game **
|
|
populate_games (int xgame, int *l, struct dg_user *me)
|
|
{
|
|
int fd, len, n, pid;
|
|
DIR *pdir;
|
|
struct dirent *pdirent;
|
|
struct stat pstat;
|
|
char fullname[130], ttyrecname[130], pidws[80], playername[DGL_PLAYERNAMELEN+1];
|
|
char *replacestr, *dir, *p;
|
|
struct dg_game **games = NULL;
|
|
struct flock fl = { 0 };
|
|
|
|
int game;
|
|
|
|
fl.l_type = F_WRLCK;
|
|
fl.l_whence = SEEK_SET;
|
|
fl.l_start = 0;
|
|
fl.l_len = 0;
|
|
|
|
len = 0;
|
|
|
|
for (game = ((xgame < 0) ? 0 : xgame); game < ((xgame <= 0) ? num_games : (xgame+1)); game++) {
|
|
|
|
dir = strdup(dgl_format_str(game, me, myconfig[game]->inprogressdir, NULL));
|
|
if (!dir) continue;
|
|
|
|
if (!(pdir = opendir (dir))) {
|
|
debug_write("cannot open inprogress-dir");
|
|
graceful_exit (140);
|
|
}
|
|
|
|
while ((pdirent = readdir (pdir)))
|
|
{
|
|
char *inprog = NULL;
|
|
if (!strcmp (pdirent->d_name, ".") || !strcmp (pdirent->d_name, ".."))
|
|
continue;
|
|
|
|
inprog = dgl_format_str(game, me, myconfig[game]->inprogressdir, NULL);
|
|
|
|
if (!inprog) continue;
|
|
|
|
snprintf (fullname, 130, "%s%s", inprog, pdirent->d_name);
|
|
|
|
fd = 0;
|
|
/* O_RDWR here should be O_RDONLY, but we need to test for
|
|
* an exclusive lock */
|
|
fd = open (fullname, O_RDWR);
|
|
if (fd >= 0 && (fcntl (fd, F_SETLK, &fl) == -1))
|
|
{
|
|
char *ttrecdir = NULL;
|
|
strncpy(playername, pdirent->d_name, DGL_PLAYERNAMELEN);
|
|
playername[DGL_PLAYERNAMELEN] = '\0';
|
|
if ((replacestr = strchr(playername, ':')))
|
|
*replacestr = '\0';
|
|
|
|
replacestr = strchr(pdirent->d_name, ':');
|
|
if (!replacestr) {
|
|
debug_write("inprogress-filename does not have ':'");
|
|
graceful_exit(145);
|
|
}
|
|
replacestr++;
|
|
|
|
ttrecdir = dgl_format_str(game, me, myconfig[game]->ttyrecdir, playername);
|
|
if (!ttrecdir) continue;
|
|
snprintf (ttyrecname, 130, "%s%s", ttrecdir, replacestr);
|
|
|
|
if (!stat (ttyrecname, &pstat))
|
|
{
|
|
/* now it's a valid game for sure */
|
|
games = realloc (games, sizeof (struct dg_game) * (len + 1));
|
|
games[len] = malloc (sizeof (struct dg_game));
|
|
games[len]->ttyrec_fn = strdup (ttyrecname);
|
|
|
|
if (!(replacestr = strchr (pdirent->d_name, ':'))) {
|
|
debug_write("inprogress-filename does not have ':', pt. 2");
|
|
graceful_exit (146);
|
|
} else
|
|
*replacestr = '\0';
|
|
|
|
games[len]->name = malloc (strlen (pdirent->d_name) + 1);
|
|
strlcpy (games[len]->name, pdirent->d_name,
|
|
strlen (pdirent->d_name) + 1);
|
|
|
|
games[len]->date = malloc (11);
|
|
strlcpy (games[len]->date, replacestr + 1, 11);
|
|
|
|
games[len]->time = malloc (9);
|
|
strlcpy (games[len]->time, replacestr + 12, 9);
|
|
|
|
games[len]->idle_time = pstat.st_mtime;
|
|
|
|
games[len]->gamenum = game;
|
|
games[len]->is_in_shm = 0;
|
|
games[len]->nwatchers = 0;
|
|
games[len]->shm_idx = -1;
|
|
|
|
n = read(fd, pidws, sizeof(pidws) - 1);
|
|
if (n > 0)
|
|
{
|
|
pidws[n] = '\0';
|
|
p = pidws;
|
|
}
|
|
else
|
|
p = "";
|
|
pid = atoi(p);
|
|
while (*p != '\0' && *p != '\n')
|
|
p++;
|
|
if (*p != '\0')
|
|
p++;
|
|
games[len]->ws_row = atoi(p);
|
|
while (*p != '\0' && *p != '\n')
|
|
p++;
|
|
if (*p != '\0')
|
|
p++;
|
|
games[len]->ws_col = atoi(p);
|
|
|
|
if (games[len]->ws_row < 4 || games[len]->ws_col < 4) {
|
|
games[len]->ws_row = 24;
|
|
games[len]->ws_col = 80;
|
|
}
|
|
|
|
games[len]->extra_info = NULL;
|
|
games[len]->extra_info_weight = 0;
|
|
if (myconfig[game]->extra_info_file) {
|
|
char *extra_info_file =
|
|
dgl_format_str(game, NULL,
|
|
myconfig[game]->extra_info_file,
|
|
games[len]->name);
|
|
game_read_extra_info(games[len], extra_info_file);
|
|
}
|
|
|
|
len++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* clean dead ones */
|
|
unlink (fullname);
|
|
}
|
|
close (fd);
|
|
|
|
fl.l_type = F_WRLCK;
|
|
}
|
|
|
|
closedir (pdir);
|
|
}
|
|
*l = len;
|
|
return games;
|
|
}
|
|
|
|
void
|
|
graceful_exit (int status)
|
|
{
|
|
/*FILE *fp;
|
|
if (status != 1)
|
|
{
|
|
fp = fopen ("/crash.log", "a");
|
|
char buf[100];
|
|
sprintf (buf, "graceful_exit called with status %d", status);
|
|
fputs (buf, fp);
|
|
}
|
|
This doesn't work. Ever.
|
|
*/
|
|
endwin();
|
|
exit (status);
|
|
}
|
|
|
|
void
|
|
create_config ()
|
|
{
|
|
FILE *config_file = NULL;
|
|
int tmp;
|
|
|
|
if (!globalconfig.allow_registration) globalconfig.allow_registration = 1;
|
|
globalconfig.menulist = NULL;
|
|
globalconfig.banner_var_list = NULL;
|
|
globalconfig.locale = NULL;
|
|
globalconfig.defterm = NULL;
|
|
|
|
globalconfig.shed_uid = (uid_t)-1;
|
|
globalconfig.shed_gid = (gid_t)-1;
|
|
|
|
globalconfig.sortmode = SORTMODE_USERNAME;
|
|
globalconfig.utf8esc = 0;
|
|
globalconfig.flowctrl = -1; /* undefined, don't touch it */
|
|
|
|
if (config)
|
|
{
|
|
if ((config_file = fopen(config, "r")) != NULL)
|
|
{
|
|
yyin = config_file;
|
|
yyparse();
|
|
fclose(config_file);
|
|
free (config);
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "ERROR: can't find or open %s for reading\n", config);
|
|
debug_write("cannot read config file");
|
|
graceful_exit(104);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#ifdef DEFCONFIG
|
|
config = DEFCONFIG;
|
|
if ((config_file = fopen(DEFCONFIG, "r")) != NULL)
|
|
{
|
|
yyin = config_file;
|
|
yyparse();
|
|
fclose(config_file);
|
|
} else {
|
|
fprintf(stderr, "ERROR: can't find or open %s for reading\n", config);
|
|
debug_write("cannot read default config file");
|
|
graceful_exit(105);
|
|
return;
|
|
}
|
|
#else
|
|
num_games = 0;
|
|
myconfig = calloc(1, sizeof(myconfig[0]));
|
|
myconfig[0] = &defconfig;
|
|
return;
|
|
#endif
|
|
}
|
|
|
|
if (!myconfig) /* a parse error occurred */
|
|
{
|
|
fprintf(stderr, "ERROR: configuration parsing failed\n");
|
|
debug_write("config file parsing failed");
|
|
graceful_exit(113);
|
|
}
|
|
|
|
if (!globalconfig.chroot) globalconfig.chroot = "/var/lib/dgamelaunch/";
|
|
|
|
if (globalconfig.max == 0) globalconfig.max = 64000;
|
|
if (globalconfig.max_newnick_len == 0) globalconfig.max_newnick_len = DGL_PLAYERNAMELEN;
|
|
if (!globalconfig.dglroot) globalconfig.dglroot = "/dgldir/";
|
|
if (!globalconfig.banner) globalconfig.banner = "/dgl-banner";
|
|
|
|
#ifndef USE_SQLITE3
|
|
if (!globalconfig.passwd) globalconfig.passwd = "/dgl-login";
|
|
#else
|
|
if (!globalconfig.passwd) globalconfig.passwd = USE_SQLITE_DB;
|
|
#endif
|
|
if (!globalconfig.lockfile) globalconfig.lockfile = "/dgl-lock";
|
|
if (!globalconfig.shed_user && globalconfig.shed_uid == (uid_t)-1)
|
|
{
|
|
struct passwd *pw;
|
|
if ((pw = getpwnam("games")))
|
|
globalconfig.shed_uid = pw->pw_uid;
|
|
else
|
|
globalconfig.shed_uid = 5; /* games uid in debian */
|
|
}
|
|
|
|
if (!globalconfig.shed_group && globalconfig.shed_gid == (gid_t)-1)
|
|
{
|
|
struct group *gr;
|
|
if ((gr = getgrnam("games")))
|
|
globalconfig.shed_gid = gr->gr_gid;
|
|
else
|
|
globalconfig.shed_gid = 60; /* games gid in debian */
|
|
}
|
|
|
|
}
|