Two new things:
1) Allow admin to define some commands that are run when something happens. format is commands [ <time> ] = <command> ... <time> can be one of: login = when user logs in dglstart = when someone telnets in register = when a new user is registered gamestart = when a game is started <command> is: mkdir "foo" = creates a directory "foo" chdir "foo" = changes current work dir to "foo" cp "foo" "bar" = copies file "foo" to "bar" unlink "foo" = deletes file "foo" setenv "foo "bar" = sets environment variable "foo" to "bar" For example commands[login] = mkdir "foo", unlink "bar", setenv "Z" "foo" You can also define per-game commands that are executed when the game is started. 2) Defining game info in the config file has changed slightly. Put "DEFINE { ... }" around the game-specific stuff. See the example config file. git-svn-id: svn://katsu.triplehelix.org/dgamelaunch/trunk@391 db0b04b0-f4d1-0310-9a6d-de3e77497b0e
This commit is contained in:
parent
41d038ef3e
commit
24cb396d16
1
TODO
1
TODO
|
@ -1,3 +1,4 @@
|
||||||
|
-allow setting the default watch-screen sortmode in config file.
|
||||||
-allow admin to define series of actions dgl makes when user does
|
-allow admin to define series of actions dgl makes when user does
|
||||||
something. for example, do "mkdir dgldir/rcfiles/$nick" and
|
something. for example, do "mkdir dgldir/rcfiles/$nick" and
|
||||||
"cp defaultrc dgldir/rcfiles/$nick/" when a new nick is registered.
|
"cp defaultrc dgldir/rcfiles/$nick/" when a new nick is registered.
|
||||||
|
|
20
config.l
20
config.l
|
@ -17,7 +17,7 @@ unsigned int line = 1, col = 0;
|
||||||
|
|
||||||
NEGNUMBER -[0-9]+
|
NEGNUMBER -[0-9]+
|
||||||
NUMBER [0-9]+
|
NUMBER [0-9]+
|
||||||
VALUE \".*\"
|
VALUE \"[^"]*\"
|
||||||
MALSTRING \"[^\"\n]*\n
|
MALSTRING \"[^\"\n]*\n
|
||||||
WHITE [\t ]*
|
WHITE [\t ]*
|
||||||
COMMENT ^#.*
|
COMMENT ^#.*
|
||||||
|
@ -51,6 +51,11 @@ COMMENT ^#.*
|
||||||
{COMMENT} { }
|
{COMMENT} { }
|
||||||
|
|
||||||
"=" { return '='; }
|
"=" { return '='; }
|
||||||
|
"," { return ','; }
|
||||||
|
"[" { return '['; }
|
||||||
|
"]" { return ']'; }
|
||||||
|
"{" { return '{'; }
|
||||||
|
"}" { return '}'; }
|
||||||
"shed_user" { return TYPE_SUSER; }
|
"shed_user" { return TYPE_SUSER; }
|
||||||
"shed_group" { return TYPE_SGROUP; }
|
"shed_group" { return TYPE_SGROUP; }
|
||||||
"shed_uid" { return TYPE_SUID; }
|
"shed_uid" { return TYPE_SUID; }
|
||||||
|
@ -59,7 +64,6 @@ COMMENT ^#.*
|
||||||
"maxnicklen" { return TYPE_MAXNICKLEN; }
|
"maxnicklen" { return TYPE_MAXNICKLEN; }
|
||||||
"allow_new_nicks" { return TYPE_ALLOW_REGISTRATION; }
|
"allow_new_nicks" { return TYPE_ALLOW_REGISTRATION; }
|
||||||
|
|
||||||
"game_num" { return TYPE_GAMENUM; }
|
|
||||||
|
|
||||||
"chroot_path" { return TYPE_PATH_CHROOT; }
|
"chroot_path" { return TYPE_PATH_CHROOT; }
|
||||||
"game_name" { return TYPE_NAME_GAME; }
|
"game_name" { return TYPE_NAME_GAME; }
|
||||||
|
@ -77,6 +81,18 @@ COMMENT ^#.*
|
||||||
"inprogressdir" { return TYPE_PATH_INPROGRESS; }
|
"inprogressdir" { return TYPE_PATH_INPROGRESS; }
|
||||||
"game_args" { return TYPE_GAME_ARGS; }
|
"game_args" { return TYPE_GAME_ARGS; }
|
||||||
"rc_fmt" { return TYPE_RC_FMT; }
|
"rc_fmt" { return TYPE_RC_FMT; }
|
||||||
|
commands { return TYPE_CMDQUEUE; }
|
||||||
|
dglstart { yylval.i = DGLTIME_DGLSTART; return TYPE_CMDQUEUENAME; }
|
||||||
|
login { yylval.i = DGLTIME_LOGIN; return TYPE_CMDQUEUENAME; }
|
||||||
|
register { yylval.i = DGLTIME_REGISTER; return TYPE_CMDQUEUENAME; }
|
||||||
|
game_start { yylval.i = DGLTIME_GAMESTART; return TYPE_CMDQUEUENAME; }
|
||||||
|
cmdmkdir { yylval.i = DGLCMD_MKDIR; return TYPE_DGLCMD1; }
|
||||||
|
cmdchdir { yylval.i = DGLCMD_CHDIR; return TYPE_DGLCMD1; }
|
||||||
|
cmdcp { yylval.i = DGLCMD_CP; return TYPE_DGLCMD1; }
|
||||||
|
cmdunlink { yylval.i = DGLCMD_UNLINK; return TYPE_DGLCMD1; }
|
||||||
|
cmdsetenv { yylval.i = DGLCMD_SETENV; return TYPE_DGLCMD2; }
|
||||||
|
DEFINE { return TYPE_DEFINE_GAME; }
|
||||||
|
|
||||||
|
|
||||||
\n { line++; col = 0; }
|
\n { line++; col = 0; }
|
||||||
|
|
||||||
|
|
307
config.y
307
config.y
|
@ -19,6 +19,8 @@ extern unsigned int line, col;
|
||||||
|
|
||||||
extern int num_games;
|
extern int num_games;
|
||||||
int ncnf = 0;
|
int ncnf = 0;
|
||||||
|
struct dg_cmdpart *curr_cmdqueue;
|
||||||
|
int cmdqueue_num = -1;
|
||||||
|
|
||||||
static const char* lookup_token (int t);
|
static const char* lookup_token (int t);
|
||||||
|
|
||||||
|
@ -34,12 +36,15 @@ static const char* lookup_token (int t);
|
||||||
%token TYPE_PATH_CHDIR TYPE_PATH_MKDIR TYPE_GAME_SHORT_NAME
|
%token TYPE_PATH_CHDIR TYPE_PATH_MKDIR TYPE_GAME_SHORT_NAME
|
||||||
%token TYPE_ALLOW_REGISTRATION
|
%token TYPE_ALLOW_REGISTRATION
|
||||||
%token TYPE_PATH_GAME TYPE_NAME_GAME TYPE_PATH_DGLDIR TYPE_PATH_SPOOL
|
%token TYPE_PATH_GAME TYPE_NAME_GAME TYPE_PATH_DGLDIR TYPE_PATH_SPOOL
|
||||||
%token TYPE_PATH_BANNER TYPE_PATH_CANNED TYPE_PATH_CHROOT TYPE_GAMENUM
|
%token TYPE_PATH_BANNER TYPE_PATH_CANNED TYPE_PATH_CHROOT
|
||||||
%token TYPE_PATH_PASSWD TYPE_PATH_LOCKFILE TYPE_PATH_SAVEFILEFMT
|
%token TYPE_PATH_PASSWD TYPE_PATH_LOCKFILE TYPE_PATH_SAVEFILEFMT
|
||||||
%token TYPE_MALSTRING TYPE_PATH_INPROGRESS TYPE_GAME_ARGS TYPE_RC_FMT
|
%token TYPE_MALSTRING TYPE_PATH_INPROGRESS TYPE_GAME_ARGS TYPE_RC_FMT
|
||||||
|
%token TYPE_CMDQUEUE
|
||||||
%token <s> TYPE_VALUE
|
%token <s> TYPE_VALUE
|
||||||
%token <i> TYPE_NUMBER
|
%token <i> TYPE_NUMBER TYPE_CMDQUEUENAME
|
||||||
%type <kt> KeyType
|
%type <kt> KeyType
|
||||||
|
%token <i> TYPE_DGLCMD1 TYPE_DGLCMD2
|
||||||
|
%token TYPE_DEFINE_GAME
|
||||||
|
|
||||||
%%
|
%%
|
||||||
|
|
||||||
|
@ -51,26 +56,39 @@ KeyPairs: KeyPairs KeyPair
|
||||||
| KeyPair
|
| KeyPair
|
||||||
;
|
;
|
||||||
|
|
||||||
KeyPair: KeyType '=' TYPE_VALUE {
|
KeyPair: TYPE_CMDQUEUE '[' TYPE_CMDQUEUENAME ']'
|
||||||
|
{
|
||||||
|
int qnum = $<i>3;
|
||||||
|
|
||||||
|
if (globalconfig.cmdqueue[qnum]) {
|
||||||
|
fprintf(stderr, "%s:%d: command queue defined twice, bailing out\n",
|
||||||
|
config, line);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
cmdqueue_num = qnum;
|
||||||
|
}
|
||||||
|
'=' cmdlist
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
struct dg_cmdpart *tmp = curr_cmdqueue;
|
||||||
|
while (tmp) {
|
||||||
|
fprintf(stderr, "cmd=%i, p1=%s, p2=%s\n", tmp->cmd, tmp->param1, tmp->param2);
|
||||||
|
tmp = tmp->next;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
globalconfig.cmdqueue[cmdqueue_num] = curr_cmdqueue;
|
||||||
|
}
|
||||||
|
| definegame
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
| KeyType '=' TYPE_VALUE {
|
||||||
struct group* gr;
|
struct group* gr;
|
||||||
struct passwd* usr;
|
struct passwd* usr;
|
||||||
|
|
||||||
if (ncnf < 0 || ncnf >= DIFF_GAMES) {
|
|
||||||
fprintf(stderr, "%s:%d: illegal game number, bailing out\n",
|
|
||||||
config, line);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!myconfig)
|
globalconfig.shed_uid = (uid_t)-1;
|
||||||
{
|
globalconfig.shed_gid = (gid_t)-1;
|
||||||
int tmp;
|
|
||||||
myconfig = calloc(DIFF_GAMES, sizeof(myconfig[0]));
|
|
||||||
for (tmp = 0; tmp < DIFF_GAMES; tmp++) {
|
|
||||||
myconfig[tmp] = calloc(1, sizeof(struct dg_config));
|
|
||||||
}
|
|
||||||
globalconfig.shed_uid = (uid_t)-1;
|
|
||||||
globalconfig.shed_gid = (gid_t)-1;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch ($1)
|
switch ($1)
|
||||||
{
|
{
|
||||||
|
@ -130,30 +148,7 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
globalconfig.chroot = strdup ($3);
|
globalconfig.chroot = strdup ($3);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_PATH_GAME:
|
|
||||||
if (myconfig[ncnf]->game_path) free(myconfig[ncnf]->game_path);
|
|
||||||
myconfig[ncnf]->game_path = strdup ($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_NAME_GAME:
|
|
||||||
if (myconfig[ncnf]->game_name) free (myconfig[ncnf]->game_name);
|
|
||||||
myconfig[ncnf]->game_name = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_GAME_SHORT_NAME:
|
|
||||||
if (myconfig[ncnf]->shortname) free (myconfig[ncnf]->shortname);
|
|
||||||
myconfig[ncnf]->shortname = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_PATH_CHDIR:
|
|
||||||
if (myconfig[ncnf]->chdir) free(myconfig[ncnf]->chdir);
|
|
||||||
myconfig[ncnf]->chdir = strdup ($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_PATH_MKDIR:
|
|
||||||
if (myconfig[ncnf]->mkdir) free(myconfig[ncnf]->mkdir);
|
|
||||||
myconfig[ncnf]->mkdir = strdup ($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_PATH_DGLDIR:
|
case TYPE_PATH_DGLDIR:
|
||||||
if (globalconfig.dglroot) free(globalconfig.dglroot);
|
if (globalconfig.dglroot) free(globalconfig.dglroot);
|
||||||
|
@ -165,16 +160,6 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
globalconfig.banner = strdup($3);
|
globalconfig.banner = strdup($3);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_PATH_CANNED:
|
|
||||||
if (myconfig[ncnf]->rcfile) free(myconfig[ncnf]->rcfile);
|
|
||||||
myconfig[ncnf]->rcfile = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_PATH_SPOOL:
|
|
||||||
if (myconfig[ncnf]->spool) free (myconfig[ncnf]->spool);
|
|
||||||
myconfig[ncnf]->spool = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_PATH_LOCKFILE:
|
case TYPE_PATH_LOCKFILE:
|
||||||
if (globalconfig.lockfile) free (globalconfig.lockfile);
|
if (globalconfig.lockfile) free (globalconfig.lockfile);
|
||||||
globalconfig.lockfile = strdup($3);
|
globalconfig.lockfile = strdup($3);
|
||||||
|
@ -185,39 +170,6 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
globalconfig.passwd = strdup($3);
|
globalconfig.passwd = strdup($3);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_PATH_SAVEFILEFMT:
|
|
||||||
if (myconfig[ncnf]->savefilefmt) free(myconfig[ncnf]->savefilefmt);
|
|
||||||
myconfig[ncnf]->savefilefmt = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_RC_FMT:
|
|
||||||
if (myconfig[ncnf]->rc_fmt) free(myconfig[ncnf]->rc_fmt);
|
|
||||||
myconfig[ncnf]->rc_fmt = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_PATH_INPROGRESS:
|
|
||||||
if (myconfig[ncnf]->inprogressdir) free(myconfig[ncnf]->inprogressdir);
|
|
||||||
myconfig[ncnf]->inprogressdir = strdup($3);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_GAME_ARGS:
|
|
||||||
{
|
|
||||||
char **tmpargs;
|
|
||||||
if (myconfig[ncnf]->bin_args) {
|
|
||||||
myconfig[ncnf]->num_args++;
|
|
||||||
tmpargs = calloc((myconfig[ncnf]->num_args+1), sizeof(char *));
|
|
||||||
memcpy(tmpargs, myconfig[ncnf]->bin_args, (myconfig[ncnf]->num_args * sizeof(char *)));
|
|
||||||
free(myconfig[ncnf]->bin_args);
|
|
||||||
myconfig[ncnf]->bin_args = tmpargs;
|
|
||||||
} else {
|
|
||||||
myconfig[ncnf]->num_args = 1;
|
|
||||||
myconfig[ncnf]->bin_args = calloc(2, sizeof(char *));
|
|
||||||
}
|
|
||||||
myconfig[ncnf]->bin_args[(myconfig[ncnf]->num_args)-1] = strdup($3);
|
|
||||||
myconfig[ncnf]->bin_args[(myconfig[ncnf]->num_args)] = 0;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
fprintf(stderr, "%s:%d: token %s does not take a string, bailing out\n",
|
fprintf(stderr, "%s:%d: token %s does not take a string, bailing out\n",
|
||||||
config, line, lookup_token($1));
|
config, line, lookup_token($1));
|
||||||
|
@ -230,22 +182,8 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
| KeyType '=' TYPE_MALSTRING {}
|
| KeyType '=' TYPE_MALSTRING {}
|
||||||
| KeyType '=' TYPE_NUMBER {
|
| KeyType '=' TYPE_NUMBER {
|
||||||
|
|
||||||
if (ncnf < 0 || ncnf >= DIFF_GAMES) {
|
|
||||||
fprintf(stderr, "%s:%d: illegal game number, bailing out\n",
|
|
||||||
config, line);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!myconfig)
|
|
||||||
{
|
|
||||||
int tmp;
|
|
||||||
myconfig = calloc(DIFF_GAMES, sizeof(myconfig[0]));
|
|
||||||
for (tmp = 0; tmp < DIFF_GAMES; tmp++) {
|
|
||||||
myconfig[tmp] = calloc(1, sizeof(struct dg_config));
|
|
||||||
}
|
|
||||||
globalconfig.shed_uid = (uid_t)-1;
|
globalconfig.shed_uid = (uid_t)-1;
|
||||||
globalconfig.shed_gid = (gid_t)-1;
|
globalconfig.shed_gid = (gid_t)-1;
|
||||||
}
|
|
||||||
|
|
||||||
switch ($1)
|
switch ($1)
|
||||||
{
|
{
|
||||||
|
@ -260,7 +198,7 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
fprintf(stderr, "%s:%d: I refuse to run as uid 0 (root)! Aborting.\n", config, line);
|
fprintf(stderr, "%s:%d: I refuse to run as uid 0 (root)! Aborting.\n", config, line);
|
||||||
graceful_exit(1);
|
graceful_exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
globalconfig.shed_uid = $3;
|
globalconfig.shed_uid = $3;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -268,19 +206,8 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
if (!silent && globalconfig.shed_gid != (gid_t)-1 && globalconfig.shed_gid != $3)
|
if (!silent && globalconfig.shed_gid != (gid_t)-1 && globalconfig.shed_gid != $3)
|
||||||
fprintf(stderr, "%s:%d: 'shed_gid = %lu' entry overrides old setting %d\n",
|
fprintf(stderr, "%s:%d: 'shed_gid = %lu' entry overrides old setting %d\n",
|
||||||
config, line, $3, globalconfig.shed_gid);
|
config, line, $3, globalconfig.shed_gid);
|
||||||
|
|
||||||
globalconfig.shed_gid = $3;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TYPE_GAMENUM:
|
globalconfig.shed_gid = $3;
|
||||||
if (($3 < 0) || ($3 >= DIFF_GAMES)) {
|
|
||||||
fprintf(stderr, "%s:%d: illegal game number, bailing out\n",
|
|
||||||
config, line);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
ncnf = $3;
|
|
||||||
if (ncnf > num_games)
|
|
||||||
num_games = ncnf;
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_ALLOW_REGISTRATION:
|
case TYPE_ALLOW_REGISTRATION:
|
||||||
|
@ -302,13 +229,168 @@ KeyPair: KeyType '=' TYPE_VALUE {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
game_definition : TYPE_CMDQUEUE
|
||||||
|
{
|
||||||
|
if (myconfig[ncnf]->cmdqueue) {
|
||||||
|
fprintf(stderr, "%s:%d: command queue defined twice, bailing out\n",
|
||||||
|
config, line);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
'=' cmdlist
|
||||||
|
{
|
||||||
|
myconfig[ncnf]->cmdqueue = curr_cmdqueue;
|
||||||
|
}
|
||||||
|
| KeyType '=' TYPE_VALUE
|
||||||
|
{
|
||||||
|
switch ( $1 ) {
|
||||||
|
case TYPE_PATH_CANNED:
|
||||||
|
if (myconfig[ncnf]->rcfile) free(myconfig[ncnf]->rcfile);
|
||||||
|
myconfig[ncnf]->rcfile = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_PATH_SPOOL:
|
||||||
|
if (myconfig[ncnf]->spool) free (myconfig[ncnf]->spool);
|
||||||
|
myconfig[ncnf]->spool = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_PATH_GAME:
|
||||||
|
if (myconfig[ncnf]->game_path) free(myconfig[ncnf]->game_path);
|
||||||
|
myconfig[ncnf]->game_path = strdup ($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_NAME_GAME:
|
||||||
|
if (myconfig[ncnf]->game_name) free (myconfig[ncnf]->game_name);
|
||||||
|
myconfig[ncnf]->game_name = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_GAME_SHORT_NAME:
|
||||||
|
if (myconfig[ncnf]->shortname) free (myconfig[ncnf]->shortname);
|
||||||
|
myconfig[ncnf]->shortname = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_PATH_CHDIR:
|
||||||
|
if (myconfig[ncnf]->chdir) free(myconfig[ncnf]->chdir);
|
||||||
|
myconfig[ncnf]->chdir = strdup ($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_PATH_MKDIR:
|
||||||
|
if (myconfig[ncnf]->mkdir) free(myconfig[ncnf]->mkdir);
|
||||||
|
myconfig[ncnf]->mkdir = strdup ($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_PATH_SAVEFILEFMT:
|
||||||
|
if (myconfig[ncnf]->savefilefmt) free(myconfig[ncnf]->savefilefmt);
|
||||||
|
myconfig[ncnf]->savefilefmt = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_RC_FMT:
|
||||||
|
if (myconfig[ncnf]->rc_fmt) free(myconfig[ncnf]->rc_fmt);
|
||||||
|
myconfig[ncnf]->rc_fmt = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_PATH_INPROGRESS:
|
||||||
|
if (myconfig[ncnf]->inprogressdir) free(myconfig[ncnf]->inprogressdir);
|
||||||
|
myconfig[ncnf]->inprogressdir = strdup($3);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TYPE_GAME_ARGS:
|
||||||
|
{
|
||||||
|
char **tmpargs;
|
||||||
|
if (myconfig[ncnf]->bin_args) {
|
||||||
|
myconfig[ncnf]->num_args++;
|
||||||
|
tmpargs = calloc((myconfig[ncnf]->num_args+1), sizeof(char *));
|
||||||
|
memcpy(tmpargs, myconfig[ncnf]->bin_args, (myconfig[ncnf]->num_args * sizeof(char *)));
|
||||||
|
free(myconfig[ncnf]->bin_args);
|
||||||
|
myconfig[ncnf]->bin_args = tmpargs;
|
||||||
|
} else {
|
||||||
|
myconfig[ncnf]->num_args = 1;
|
||||||
|
myconfig[ncnf]->bin_args = calloc(2, sizeof(char *));
|
||||||
|
}
|
||||||
|
myconfig[ncnf]->bin_args[(myconfig[ncnf]->num_args)-1] = strdup($3);
|
||||||
|
myconfig[ncnf]->bin_args[(myconfig[ncnf]->num_args)] = 0;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "%s:%d: token does not belong into game definition, bailing out\n",
|
||||||
|
config, line);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
;
|
||||||
|
|
||||||
|
game_definitions : game_definitions game_definition
|
||||||
|
| game_definition
|
||||||
|
;
|
||||||
|
|
||||||
|
definegame : TYPE_DEFINE_GAME '{'
|
||||||
|
{
|
||||||
|
if ((ncnf < 0) || (ncnf >= DIFF_GAMES)) {
|
||||||
|
fprintf(stderr, "%s:%d: too many games defined, bailing out\n",
|
||||||
|
config, line);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!myconfig) {
|
||||||
|
int tmp;
|
||||||
|
myconfig = calloc(DIFF_GAMES, sizeof(myconfig[0]));
|
||||||
|
for (tmp = 0; tmp < DIFF_GAMES; tmp++) {
|
||||||
|
myconfig[tmp] = calloc(1, sizeof(struct dg_config));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
game_definitions '}'
|
||||||
|
{
|
||||||
|
ncnf++;
|
||||||
|
num_games = ncnf;
|
||||||
|
}
|
||||||
|
;
|
||||||
|
|
||||||
|
cmdlist : cmdlist ',' dglcmd
|
||||||
|
| dglcmd
|
||||||
|
;
|
||||||
|
|
||||||
|
dglcmd : TYPE_DGLCMD1 TYPE_VALUE
|
||||||
|
{
|
||||||
|
struct dg_cmdpart *tmp = malloc(sizeof(struct dg_cmdpart));
|
||||||
|
if (tmp) {
|
||||||
|
struct dg_cmdpart *foo = curr_cmdqueue;
|
||||||
|
if (foo) {
|
||||||
|
while (foo->next) foo = foo->next;
|
||||||
|
foo->next = tmp;
|
||||||
|
} else curr_cmdqueue = tmp;
|
||||||
|
tmp->next = NULL;
|
||||||
|
tmp->param1 = strdup( $2 );
|
||||||
|
tmp->param2 = NULL;
|
||||||
|
tmp->cmd = $<i>1;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
| TYPE_DGLCMD2 TYPE_VALUE TYPE_VALUE
|
||||||
|
{
|
||||||
|
struct dg_cmdpart *tmp = malloc(sizeof(struct dg_cmdpart));
|
||||||
|
if (tmp) {
|
||||||
|
struct dg_cmdpart *foo = curr_cmdqueue;
|
||||||
|
if (foo) {
|
||||||
|
while (foo->next) foo = foo->next;
|
||||||
|
foo->next = tmp;
|
||||||
|
} else curr_cmdqueue = tmp;
|
||||||
|
tmp->next = NULL;
|
||||||
|
tmp->param1 = strdup( $2 );
|
||||||
|
tmp->param2 = strdup( $3 );
|
||||||
|
tmp->cmd = $<i>1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
;
|
||||||
|
|
||||||
KeyType : TYPE_SUSER { $$ = TYPE_SUSER; }
|
KeyType : TYPE_SUSER { $$ = TYPE_SUSER; }
|
||||||
| TYPE_SGROUP { $$ = TYPE_SGROUP; }
|
| TYPE_SGROUP { $$ = TYPE_SGROUP; }
|
||||||
| TYPE_SUID { $$ = TYPE_SUID; }
|
| TYPE_SUID { $$ = TYPE_SUID; }
|
||||||
| TYPE_SGID { $$ = TYPE_SGID; }
|
| TYPE_SGID { $$ = TYPE_SGID; }
|
||||||
| TYPE_MAX { $$ = TYPE_MAX; }
|
| TYPE_MAX { $$ = TYPE_MAX; }
|
||||||
| TYPE_MAXNICKLEN { $$ = TYPE_MAXNICKLEN; }
|
| TYPE_MAXNICKLEN { $$ = TYPE_MAXNICKLEN; }
|
||||||
| TYPE_GAMENUM { $$ = TYPE_GAMENUM; }
|
|
||||||
| TYPE_PATH_CHROOT { $$ = TYPE_PATH_CHROOT; }
|
| TYPE_PATH_CHROOT { $$ = TYPE_PATH_CHROOT; }
|
||||||
| TYPE_ALLOW_REGISTRATION { $$ = TYPE_ALLOW_REGISTRATION; }
|
| TYPE_ALLOW_REGISTRATION { $$ = TYPE_ALLOW_REGISTRATION; }
|
||||||
| TYPE_PATH_GAME { $$ = TYPE_PATH_GAME; }
|
| TYPE_PATH_GAME { $$ = TYPE_PATH_GAME; }
|
||||||
|
@ -340,7 +422,6 @@ const char* lookup_token (int t)
|
||||||
case TYPE_SGID: return "shed_gid";
|
case TYPE_SGID: return "shed_gid";
|
||||||
case TYPE_MAX: return "maxusers";
|
case TYPE_MAX: return "maxusers";
|
||||||
case TYPE_MAXNICKLEN: return "maxnicklen";
|
case TYPE_MAXNICKLEN: return "maxnicklen";
|
||||||
case TYPE_GAMENUM: return "game_num";
|
|
||||||
case TYPE_PATH_CHROOT: return "chroot_path";
|
case TYPE_PATH_CHROOT: return "chroot_path";
|
||||||
case TYPE_PATH_CHDIR: return "chdir";
|
case TYPE_PATH_CHDIR: return "chdir";
|
||||||
case TYPE_PATH_MKDIR: return "mkdir";
|
case TYPE_PATH_MKDIR: return "mkdir";
|
||||||
|
|
|
@ -884,6 +884,7 @@ autologin (char* user, char *pass)
|
||||||
if (passwordgood(pass)) {
|
if (passwordgood(pass)) {
|
||||||
loggedin = 1;
|
loggedin = 1;
|
||||||
setproctitle ("%s", me->username);
|
setproctitle ("%s", me->username);
|
||||||
|
dgl_exec_cmdqueue(globalconfig.cmdqueue[DGLTIME_LOGIN], 0, me);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -949,6 +950,7 @@ loginprompt (int from_ttyplay)
|
||||||
{
|
{
|
||||||
loggedin = 1;
|
loggedin = 1;
|
||||||
setproctitle ("%s", me->username);
|
setproctitle ("%s", me->username);
|
||||||
|
dgl_exec_cmdqueue(globalconfig.cmdqueue[DGLTIME_LOGIN], 0, me);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -1109,11 +1111,16 @@ newuser ()
|
||||||
|
|
||||||
setproctitle ("%s", me->username);
|
setproctitle ("%s", me->username);
|
||||||
|
|
||||||
|
dgl_exec_cmdqueue(globalconfig.cmdqueue[DGLTIME_REGISTER], 0, me);
|
||||||
|
|
||||||
/* create their ttyrec dir */
|
/* create their ttyrec dir */
|
||||||
|
/*
|
||||||
snprintf (dirname, 100, "%sttyrec/%s", globalconfig.dglroot, me->username);
|
snprintf (dirname, 100, "%sttyrec/%s", globalconfig.dglroot, me->username);
|
||||||
|
|
||||||
if (access (dirname, F_OK) != 0)
|
if (access (dirname, F_OK) != 0)
|
||||||
mkdir (dirname, 0755);
|
mkdir (dirname, 0755);
|
||||||
|
*/
|
||||||
|
|
||||||
writefile (1);
|
writefile (1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2051,6 +2058,8 @@ main (int argc, char** argv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dgl_exec_cmdqueue(globalconfig.cmdqueue[DGLTIME_DGLSTART], 0, NULL);
|
||||||
|
|
||||||
if (nhext)
|
if (nhext)
|
||||||
{
|
{
|
||||||
char *myargv[3];
|
char *myargv[3];
|
||||||
|
@ -2116,6 +2125,11 @@ main (int argc, char** argv)
|
||||||
endwin ();
|
endwin ();
|
||||||
signal(SIGWINCH, SIG_DFL);
|
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);
|
||||||
|
|
||||||
if (!backup_savefile (userchoice))
|
if (!backup_savefile (userchoice))
|
||||||
graceful_exit (5);
|
graceful_exit (5);
|
||||||
|
|
||||||
|
@ -2183,6 +2197,8 @@ main (int argc, char** argv)
|
||||||
if (me)
|
if (me)
|
||||||
free (me);
|
free (me);
|
||||||
|
|
||||||
|
/* FIXME: free data in globalconfig */
|
||||||
|
|
||||||
graceful_exit (1);
|
graceful_exit (1);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -16,6 +16,23 @@
|
||||||
/* max # of different games playable from within this dgl */
|
/* max # of different games playable from within this dgl */
|
||||||
#define DIFF_GAMES 4
|
#define DIFF_GAMES 4
|
||||||
|
|
||||||
|
typedef enum
|
||||||
|
{
|
||||||
|
DGLTIME_DGLSTART = 0, /* when someone telnets in */
|
||||||
|
DGLTIME_LOGIN, /* right after user login */
|
||||||
|
DGLTIME_REGISTER, /* right after new nick is registered */
|
||||||
|
DGLTIME_GAMESTART, /* right before a game is started */
|
||||||
|
NUM_DGLTIMES
|
||||||
|
} dglcmd_times;
|
||||||
|
|
||||||
|
struct dg_cmdpart
|
||||||
|
{
|
||||||
|
dglcmd_times cmd;
|
||||||
|
char *param1;
|
||||||
|
char *param2;
|
||||||
|
struct dg_cmdpart *next;
|
||||||
|
};
|
||||||
|
|
||||||
struct dg_user
|
struct dg_user
|
||||||
{
|
{
|
||||||
#ifdef USE_SQLITE3
|
#ifdef USE_SQLITE3
|
||||||
|
@ -59,6 +76,7 @@ struct dg_config
|
||||||
int num_args; /* # of bin_args */
|
int num_args; /* # of bin_args */
|
||||||
char **bin_args; /* args for game binary */
|
char **bin_args; /* args for game binary */
|
||||||
char *rc_fmt;
|
char *rc_fmt;
|
||||||
|
struct dg_cmdpart *cmdqueue;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct dg_globalconfig
|
struct dg_globalconfig
|
||||||
|
@ -75,8 +93,19 @@ struct dg_globalconfig
|
||||||
char* passwd;
|
char* passwd;
|
||||||
char* lockfile;
|
char* lockfile;
|
||||||
int allow_registration; /* allow registering new nicks */
|
int allow_registration; /* allow registering new nicks */
|
||||||
|
|
||||||
|
struct dg_cmdpart *cmdqueue[NUM_DGLTIMES];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
typedef enum
|
||||||
|
{
|
||||||
|
DGLCMD_NONE = 0,
|
||||||
|
DGLCMD_MKDIR, /* mkdir foo */
|
||||||
|
DGLCMD_CHDIR, /* chdir foo */
|
||||||
|
DGLCMD_CP, /* cp foo bar */
|
||||||
|
DGLCMD_UNLINK, /* unlink foo */
|
||||||
|
DGLCMD_SETENV /* setenv foo bar */
|
||||||
|
} dglcmd_actions;
|
||||||
|
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
|
@ -115,6 +144,9 @@ extern void catch_sighup(int signum);
|
||||||
extern void loadbanner(int game, struct dg_banner *ban);
|
extern void loadbanner(int game, struct dg_banner *ban);
|
||||||
extern void drawbanner(unsigned int start_line, unsigned int howmany);
|
extern void drawbanner(unsigned int start_line, unsigned int howmany);
|
||||||
extern char *dgl_format_str(int game, struct dg_user *me, char *str);
|
extern char *dgl_format_str(int game, struct dg_user *me, char *str);
|
||||||
|
|
||||||
|
extern int dgl_exec_cmdqueue(struct dg_cmdpart *queue, int game, struct dg_user *me);
|
||||||
|
|
||||||
extern struct dg_game **populate_games(int game, int *l);
|
extern struct dg_game **populate_games(int game, int *l);
|
||||||
|
|
||||||
extern struct dg_game **sort_games(struct dg_game **games, int len, dg_sortmode sortmode);
|
extern struct dg_game **sort_games(struct dg_game **games, int len, dg_sortmode sortmode);
|
||||||
|
|
76
dgl-common.c
76
dgl-common.c
|
@ -39,7 +39,8 @@ struct dg_config defconfig = {
|
||||||
/* inprogressdir = */ "inprogress/",
|
/* inprogressdir = */ "inprogress/",
|
||||||
/* num_args = */ 0,
|
/* num_args = */ 0,
|
||||||
/* bin_args = */ NULL,
|
/* bin_args = */ NULL,
|
||||||
/* rc_fmt = */ "%rrcfiles/%n.nethackrc" /* [dglroot]rcfiles/[username].nethackrc */
|
/* rc_fmt = */ "%rrcfiles/%n.nethackrc", /* [dglroot]rcfiles/[username].nethackrc */
|
||||||
|
/* cmdqueue = */ NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
char* config = NULL;
|
char* config = NULL;
|
||||||
|
@ -55,7 +56,8 @@ struct dg_globalconfig globalconfig;
|
||||||
* replace following codes with variables:
|
* replace following codes with variables:
|
||||||
* %u == shed_uid (number)
|
* %u == shed_uid (number)
|
||||||
* %n == user name (string)
|
* %n == user name (string)
|
||||||
* %r == chroot (string)
|
* %r == chroot (string) (aka "dglroot" config var)
|
||||||
|
* %g == game name
|
||||||
*/
|
*/
|
||||||
char *
|
char *
|
||||||
dgl_format_str(int game, struct dg_user *me, char *str)
|
dgl_format_str(int game, struct dg_user *me, char *str)
|
||||||
|
@ -79,7 +81,12 @@ dgl_format_str(int game, struct dg_user *me, char *str)
|
||||||
p++;
|
p++;
|
||||||
break;
|
break;
|
||||||
case 'n':
|
case 'n':
|
||||||
snprintf (p, end + 1 - p, "%s", me->username);
|
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')
|
while (*p != '\0')
|
||||||
p++;
|
p++;
|
||||||
break;
|
break;
|
||||||
|
@ -110,6 +117,65 @@ dgl_format_str(int game, struct dg_user *me, char *str)
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
dgl_exec_cmdqueue(struct dg_cmdpart *queue, int game, struct dg_user *me)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
struct dg_cmdpart *tmp = queue;
|
||||||
|
|
||||||
|
if (!queue) return 1;
|
||||||
|
|
||||||
|
while (tmp) {
|
||||||
|
char *p1 = NULL; /* FIXME: should probably use fixed-size buffers instead of doing strdup() every time */
|
||||||
|
char *p2 = NULL;
|
||||||
|
if (tmp->param1) p1 = strdup(dgl_format_str(game, me, tmp->param1));
|
||||||
|
if (tmp->param2) p2 = strdup(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_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 (!(newfile = fopen (p2, "w"))) return 1;
|
||||||
|
if (!(cannedf = fopen (p1, "r"))) return 1;
|
||||||
|
while ((bytes = fread (buf, 1, 1024, cannedf)) > 0) {
|
||||||
|
if (fwrite (buf, 1, bytes, newfile) != bytes) {
|
||||||
|
if (ferror (newfile)) {
|
||||||
|
fclose (cannedf);
|
||||||
|
fclose (newfile);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fclose (cannedf);
|
||||||
|
fclose (newfile);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case DGLCMD_SETENV:
|
||||||
|
if (p1 && p2) mysetenv(p1, p2, 1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
free(p1);
|
||||||
|
free(p2);
|
||||||
|
|
||||||
|
tmp = tmp->next;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
sort_game_username(const void *g1, const void *g2)
|
sort_game_username(const void *g1, const void *g2)
|
||||||
|
@ -397,4 +463,8 @@ create_config ()
|
||||||
globalconfig.shed_gid = 60; /* games gid in debian */
|
globalconfig.shed_gid = 60; /* games gid in debian */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (tmp = 0; tmp < NUM_DGLTIMES; tmp++) {
|
||||||
|
globalconfig.cmdqueue[0] = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,9 +54,36 @@ passwd = "/dgl-login"
|
||||||
lockfile = "/dgl-lock"
|
lockfile = "/dgl-lock"
|
||||||
|
|
||||||
|
|
||||||
|
#
|
||||||
|
# define some commands that are run when something happens. format is
|
||||||
|
# commands [ <time> ] = <command> ...
|
||||||
|
#
|
||||||
|
# <time> can be one of:
|
||||||
|
# login = when user logs in
|
||||||
|
# dglstart = when someone telnets in
|
||||||
|
# register = when a new user is registered
|
||||||
|
# gamestart = when a game is started
|
||||||
|
#
|
||||||
|
# <command> is:
|
||||||
|
# mkdir "foo" = creates a directory "foo"
|
||||||
|
# chdir "foo" = changes current work dir to "foo"
|
||||||
|
# cp "foo" "bar" = copies file "foo" to "bar"
|
||||||
|
# unlink "foo" = deletes file "foo"
|
||||||
|
# setenv "foo "bar" = sets environment variable "foo" to "bar"
|
||||||
|
#
|
||||||
|
# eg. commands[login] = mkdir "foo", unlink "bar", setenv "Z" "foo"
|
||||||
|
#
|
||||||
|
|
||||||
|
# create the user's ttyrec dir when they register
|
||||||
|
commands[register] = cmdmkdir "%rttyrec/%n"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Next, we'll define one game's data:
|
# Next, we'll define one game's data:
|
||||||
|
|
||||||
|
DEFINE {
|
||||||
|
|
||||||
# From inside the jail, the location of the binary to be launched.
|
# From inside the jail, the location of the binary to be launched.
|
||||||
game_path = "/bin/nethackstub"
|
game_path = "/bin/nethackstub"
|
||||||
|
@ -93,13 +120,19 @@ savefilefmt = "/var/games/nethack/save/%u%n.gz"
|
||||||
# each game you define here must have it's own.
|
# each game you define here must have it's own.
|
||||||
inprogressdir = "inprogress-nethackstub/"
|
inprogressdir = "inprogress-nethackstub/"
|
||||||
|
|
||||||
|
# We can also define per-game commands, that are executed
|
||||||
|
# when teh game starts:
|
||||||
|
# commands = chdir "/dgldir", mkdir "foo_%u_%g"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# the second game
|
# the second game
|
||||||
#
|
#
|
||||||
#
|
#
|
||||||
game_num = 1
|
|
||||||
|
DEFINE {
|
||||||
|
|
||||||
game_path = "/bin/nethack"
|
game_path = "/bin/nethack"
|
||||||
game_name = "NetHack 3.4.3"
|
game_name = "NetHack 3.4.3"
|
||||||
|
@ -118,13 +151,15 @@ rc_fmt = "%rrcfiles/%n.nethackrc"
|
||||||
savefilefmt = "/var/games/nethack/save/%u%n.gz"
|
savefilefmt = "/var/games/nethack/save/%u%n.gz"
|
||||||
inprogressdir = "inprogress-nethack/"
|
inprogressdir = "inprogress-nethack/"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
#
|
#
|
||||||
# third game
|
# third game
|
||||||
#
|
#
|
||||||
#
|
#
|
||||||
|
|
||||||
game_num = 2
|
DEFINE {
|
||||||
|
|
||||||
game_path = "/bin/crawlss017"
|
game_path = "/bin/crawlss017"
|
||||||
game_name = "Crawl Stone Soup 0.1.7"
|
game_name = "Crawl Stone Soup 0.1.7"
|
||||||
short_name = "Cr017"
|
short_name = "Cr017"
|
||||||
|
@ -148,12 +183,15 @@ rc_template = "/dgl-default-rcfile.crawl"
|
||||||
rc_fmt = "%rrcfiles/%n.crawlrc"
|
rc_fmt = "%rrcfiles/%n.crawlrc"
|
||||||
inprogressdir = "inprogress-crawlss017/"
|
inprogressdir = "inprogress-crawlss017/"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
#
|
#
|
||||||
# fourth game
|
# fourth game
|
||||||
#
|
#
|
||||||
#
|
#
|
||||||
game_num = 3
|
|
||||||
|
DEFINE {
|
||||||
|
|
||||||
game_path = "/bin/crawlss020"
|
game_path = "/bin/crawlss020"
|
||||||
game_name = "Crawl Stone Soup 0.2.0"
|
game_name = "Crawl Stone Soup 0.2.0"
|
||||||
short_name = "Cr020"
|
short_name = "Cr020"
|
||||||
|
@ -177,4 +215,4 @@ rc_template = "/dgl-default-rcfile.crawl"
|
||||||
rc_fmt = "/crawlss020/plr/%n/%n.crawlrc"
|
rc_fmt = "/crawlss020/plr/%n/%n.crawlrc"
|
||||||
inprogressdir = "inprogress-crawlss020/"
|
inprogressdir = "inprogress-crawlss020/"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue