605 lines
14 KiB
C
605 lines
14 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 ();
|
|
|
|
/* Data structures */
|
|
struct dg_config **myconfig = NULL;
|
|
struct dg_config defconfig = {
|
|
/* chroot = */ /*"/var/lib/dgamelaunch/",*/
|
|
/* game_path = */ "/bin/nethack",
|
|
/* game_name = */ "NetHack",
|
|
/* shortname = */ "NH",
|
|
/* chdir = */ /*NULL,*/
|
|
/* mkdir = */ /*NULL,*/
|
|
/* dglroot = *//* "/dgldir/",*/
|
|
/* lockfile = */ /*"/dgl-lock",*/
|
|
/* passwd = */ /*"/dgl-login",*/
|
|
/* banner = */ /*"/dgl-banner",*/
|
|
/* rcfile = */ NULL, /*"/dgl-default-rcfile",*/
|
|
/* spool = */ "/var/mail/",
|
|
/* shed_user = */ /*"games",*/
|
|
/* shed_group = */ /*"games",*/
|
|
/* shed_uid = *//* 5,*/
|
|
/* shed_gid = */ /*60,*/ /* games:games in Debian */
|
|
/* max = */ /*64000,*/
|
|
/* savefilefmt = */ /*"",*/ /* don't do this by default */
|
|
/* inprogressdir = */ "inprogress/",
|
|
/* num_args = */ 0,
|
|
/* bin_args = */ NULL,
|
|
/* rc_fmt = */ "%rrcfiles/%n.nethackrc", /* [dglroot]rcfiles/[username].nethackrc */
|
|
/* cmdqueue = */ NULL
|
|
};
|
|
|
|
char* config = NULL;
|
|
int silent = 0;
|
|
int loggedin = 0;
|
|
char *chosen_name;
|
|
int num_games = 0;
|
|
|
|
int selected_game = 0;
|
|
int return_from_submenu = 0;
|
|
|
|
mode_t default_fmode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH;
|
|
|
|
struct dg_globalconfig globalconfig;
|
|
|
|
|
|
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)
|
|
* %r == chroot (string) (aka "dglroot" config var)
|
|
* %g == game name
|
|
*/
|
|
char *
|
|
dgl_format_str(int game, struct dg_user *me, char *str)
|
|
{
|
|
static char buf[1024];
|
|
char *f, *p, *end;
|
|
int ispercent = 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) snprintf (p, end + 1 - p, "%s", me->username);
|
|
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);
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
case 'r':
|
|
snprintf (p, end + 1 - p, "%s", globalconfig.dglroot);
|
|
while (*p != '\0')
|
|
p++;
|
|
break;
|
|
default:
|
|
*p = *f;
|
|
if (p < end)
|
|
p++;
|
|
}
|
|
ispercent = 0;
|
|
} else {
|
|
if (*f == '%')
|
|
ispercent = 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;
|
|
|
|
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));
|
|
if (tmp->param2) strcpy(p2, dgl_format_str(game, me, tmp->param2));
|
|
|
|
switch (tmp->cmd) {
|
|
default: 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) chdir(p1);
|
|
break;
|
|
case DGLCMD_IF_NX_CP:
|
|
if (p1 && p2) {
|
|
FILE *tmpfile;
|
|
tmpfile = fopen(p2, "r");
|
|
if (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;
|
|
|
|
endwin();
|
|
child = fork();
|
|
if (child == -1) {
|
|
perror("fork");
|
|
graceful_exit(114);
|
|
} else if (child == 0) {
|
|
execvp(p1, myargv);
|
|
exit(0);
|
|
} else
|
|
waitpid(child, NULL, 0);
|
|
refresh();
|
|
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(game);
|
|
break;
|
|
case DGLCMD_LOGIN:
|
|
if (!loggedin) loginprompt(0);
|
|
if (loggedin) runmenuloop(dgl_find_menu("mainmenu_user"));
|
|
break;
|
|
case DGLCMD_REGISTER:
|
|
if (!loggedin && globalconfig.allow_registration) newuser();
|
|
break;
|
|
case DGLCMD_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_EDITOPTIONS:
|
|
if (loggedin && p1) {
|
|
int i;
|
|
for (i = 0; i < num_games; i++) {
|
|
if ((!strcmp(myconfig[i]->game_name, p1) || !strcmp(myconfig[i]->shortname, p1)) && myconfig[i]->rcfile) {
|
|
editoptions(i);
|
|
check_retard(1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case DGLCMD_PLAYGAME:
|
|
if (loggedin && me && p1) {
|
|
int userchoice, i;
|
|
char *tmpstr;
|
|
for (userchoice = 0; userchoice < num_games; userchoice++) {
|
|
if (!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), R_OK) == -1)
|
|
write_canned_rcfile (userchoice, dgl_format_str(userchoice, me, myconfig[userchoice]->rc_fmt));
|
|
}
|
|
|
|
setproctitle("%s [playing %s]", me->username, myconfig[userchoice]->shortname);
|
|
|
|
endwin ();
|
|
signal(SIGWINCH, SIG_DFL);
|
|
|
|
/* 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]));
|
|
free(myconfig[userchoice]->bin_args[i]);
|
|
myconfig[userchoice]->bin_args[i] = tmpstr;
|
|
}
|
|
|
|
/* launch program */
|
|
ttyrec_main (userchoice, me->username, gen_ttyrec_filename());
|
|
check_retard(1); /* reset retard counter */
|
|
|
|
setproctitle ("%s", me->username);
|
|
|
|
initcurses ();
|
|
|
|
check_retard(1);
|
|
}
|
|
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);
|
|
}
|
|
|
|
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 (game2->idle_time != game1->idle_time)
|
|
return difftime(game2->idle_time, game1->idle_time);
|
|
else
|
|
return strcasecmp(game1->name, game2->name);
|
|
}
|
|
|
|
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_IDLETIME: qsort(games, len, sizeof(struct dg_game *), sort_game_idletime); break;
|
|
default: ;
|
|
}
|
|
return games;
|
|
}
|
|
|
|
struct dg_game **
|
|
populate_games (int xgame, int *l)
|
|
{
|
|
int fd, len, n, is_nhext, pid;
|
|
DIR *pdir;
|
|
struct dirent *pdirent;
|
|
struct stat pstat;
|
|
char fullname[130], ttyrecname[130], pidws[80];
|
|
char *replacestr, *dir, *p;
|
|
struct dg_game **games = NULL;
|
|
struct flock fl = { 0 };
|
|
size_t slen;
|
|
|
|
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); game++) {
|
|
|
|
slen = strlen(globalconfig.dglroot) + strlen(myconfig[game]->inprogressdir) + 1;
|
|
dir = malloc(slen);
|
|
snprintf(dir, slen, "%s%s", globalconfig.dglroot, myconfig[game]->inprogressdir);
|
|
|
|
if (!(pdir = opendir (dir)))
|
|
graceful_exit (140);
|
|
|
|
while ((pdirent = readdir (pdir)))
|
|
{
|
|
if (!strcmp (pdirent->d_name, ".") || !strcmp (pdirent->d_name, ".."))
|
|
continue;
|
|
|
|
is_nhext = !strcmp (pdirent->d_name + strlen (pdirent->d_name) - 6, ".nhext");
|
|
|
|
snprintf (fullname, 130, "%s%s%s", globalconfig.dglroot, myconfig[game]->inprogressdir, 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 && (is_nhext || fcntl (fd, F_SETLK, &fl) == -1))
|
|
{
|
|
|
|
/* stat to check idle status */
|
|
if (!is_nhext)
|
|
{
|
|
snprintf (ttyrecname, 130, "%sttyrec/%s", globalconfig.dglroot, pdirent->d_name);
|
|
replacestr = strchr (ttyrecname, ':');
|
|
if (!replacestr)
|
|
graceful_exit (145);
|
|
replacestr[0] = '/';
|
|
}
|
|
if (is_nhext || !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 (pdirent->d_name);
|
|
|
|
if (!(replacestr = strchr (pdirent->d_name, ':')))
|
|
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;
|
|
|
|
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 (is_nhext)
|
|
{
|
|
if (kill (pid, 0) != 0)
|
|
{
|
|
/* Dead game */
|
|
free (games[len]->ttyrec_fn);
|
|
free (games[len]->name);
|
|
free (games[len]->date);
|
|
free (games[len]->time);
|
|
free (games[len]);
|
|
unlink (fullname);
|
|
}
|
|
else
|
|
len++;
|
|
}
|
|
else
|
|
{
|
|
if (games[len]->ws_row < 4 || games[len]->ws_col < 4)
|
|
{
|
|
games[len]->ws_row = 24;
|
|
games[len]->ws_col = 80;
|
|
}
|
|
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.sortmode = SORTMODE_NONE;
|
|
|
|
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);
|
|
graceful_exit(104);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#ifdef DEFCONFIG
|
|
/* fprintf(stderr, "DEFCONFIG: %s\n", DEFCONFIG);*/
|
|
config = DEFCONFIG;
|
|
if ((config_file = fopen(DEFCONFIG, "r")) != NULL)
|
|
{
|
|
yyin = config_file;
|
|
/* fprintf(stderr, "PARSING\n");*/
|
|
yyparse();
|
|
/* fprintf(stderr, "PARSED\n");*/
|
|
fclose(config_file);
|
|
}
|
|
#else
|
|
/* fprintf(stderr, "NO DEFCONFIG\n");*/
|
|
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");
|
|
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 = 20;
|
|
if (!globalconfig.dglroot) globalconfig.dglroot = "/dgldir/";
|
|
if (!globalconfig.banner) globalconfig.banner = "/dgl-banner";
|
|
|
|
if (!globalconfig.passwd) globalconfig.passwd = "/dgl-login";
|
|
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 */
|
|
}
|
|
|
|
}
|