2004-01-04 03:54:01 +01:00
/* dgamelaunch.c
2003-12-30 22:30:32 +01:00
*
2004-01-25 23:34:35 +01:00
* ( c ) 2001 - 4 M . Drew Streib < dtype @ dtype . org >
* also parts ( c ) 2003 - 4 Joshua Kwan < joshk @ triplehelix . org > ,
2004-01-04 03:54:01 +01:00
* Brett Carrington < brettcar @ segvio . org > ,
* Jilles Tjoelker < jilles @ stack . nl >
*
2003-12-30 22:30:32 +01:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
2004-01-04 03:54:01 +01:00
* See this program in action at http : //alt.org/nethack/
2003-12-30 22:30:32 +01:00
*
* This is a little wrapper for nethack ( and soon other programs ) that
* will allow them to be run from a telnetd session , chroot , shed privs ,
* make a simple login , then play the game .
*/
2004-01-02 05:35:46 +01:00
# define _GNU_SOURCE
2004-01-30 03:49:29 +01:00
2003-12-31 03:39:10 +01:00
# include "dgamelaunch.h"
2004-06-18 19:57:29 +02:00
# include "config.h"
2004-01-30 03:49:29 +01:00
# include "ttyplay.h"
# include "ttyrec.h"
2003-12-31 03:39:10 +01:00
2003-12-30 22:30:32 +01:00
/* a request from the author: please leave some remnance of
* ' based on dgamelaunch version xxx ' in any derivative works , or
* even keep the line the same altogether . I ' m probably happy
* to make any changes you need . */
/* ************************************************************* */
/* ************************************************************* */
/* ************************************************************* */
/* program stuff */
# include <sys/types.h>
2004-01-02 21:47:53 +01:00
# include <sys/time.h>
2004-01-23 19:14:48 +01:00
# include <sys/wait.h>
2004-01-25 23:34:35 +01:00
# include <sys/ioctl.h> /* ttyrec */
# include <sys/stat.h>
2004-07-04 01:34:13 +02:00
# include <libgen.h>
2004-01-25 23:34:35 +01:00
# include <stdlib.h>
# include <curses.h>
2004-01-02 21:47:53 +01:00
2007-12-27 20:52:48 +01:00
# ifdef USE_SQLITE3
# include <sqlite3.h>
# endif
2004-01-02 21:47:53 +01:00
# ifndef __FreeBSD__
2004-03-10 01:18:30 +01:00
# ifdef __APPLE__
# include <unistd.h>
# else
# include <crypt.h>
# endif
2004-01-02 21:47:53 +01:00
# else
# include <libutil.h>
# endif
2004-01-02 22:00:54 +01:00
# ifdef __linux__
# include <pty.h>
2004-01-03 03:10:35 +01:00
# endif
2004-01-12 18:32:47 +01:00
# include <fcntl.h>
2004-01-05 02:25:38 +01:00
# include <pwd.h>
2004-01-02 22:00:54 +01:00
# include <grp.h>
2004-01-02 21:47:53 +01:00
# include <time.h>
2003-12-30 22:30:32 +01:00
# include <errno.h>
# include <dirent.h>
# include <string.h>
2004-01-02 21:47:53 +01:00
# include <signal.h>
2003-12-30 22:30:32 +01:00
# include <assert.h>
# include <ctype.h>
# include <unistd.h>
# include <termios.h>
2004-01-05 02:25:38 +01:00
extern FILE * yyin ;
extern int yyparse ( ) ;
2004-01-25 23:44:34 +01:00
extern int editor_main ( int argc , char * * argv ) ;
2003-12-30 22:30:32 +01:00
/* global variables */
2004-07-03 21:19:28 +02:00
char * __progname ;
2007-03-29 22:14:55 +02:00
/*char rcfilename[80];*/
2003-12-30 22:30:32 +01:00
2007-12-27 20:52:48 +01:00
# ifndef USE_SQLITE3
2003-12-30 22:30:32 +01:00
int f_num = 0 ;
2003-12-31 02:58:56 +01:00
struct dg_user * * users = NULL ;
2007-12-27 20:52:48 +01:00
# endif
2003-12-31 02:58:56 +01:00
struct dg_user * me = NULL ;
2003-12-31 20:47:49 +01:00
struct dg_banner banner ;
2003-12-30 22:30:32 +01:00
2007-12-27 20:52:48 +01:00
struct dg_user *
cpy_me ( struct dg_user * me )
{
struct dg_user * tmp = malloc ( sizeof ( struct dg_user ) ) ;
if ( tmp & & me ) {
2007-12-30 09:29:28 +01:00
# ifdef USE_SQLITE3
2007-12-27 20:52:48 +01:00
tmp - > id = me - > id ;
2007-12-30 09:29:28 +01:00
# endif
2007-12-27 20:52:48 +01:00
if ( me - > username ) tmp - > username = strdup ( me - > username ) ;
if ( me - > email ) tmp - > email = strdup ( me - > email ) ;
if ( me - > env ) tmp - > env = strdup ( me - > env ) ;
if ( me - > password ) tmp - > password = strdup ( me - > password ) ;
tmp - > flags = me - > flags ;
}
return tmp ;
}
2004-06-18 20:16:22 +02:00
# ifndef HAVE_SETENV
2004-01-16 03:00:47 +01:00
int
mysetenv ( const char * name , const char * value , int overwrite )
{
int retval ;
char * buf = NULL ;
if ( getenv ( name ) = = NULL | | overwrite )
{
size_t len = strlen ( name ) + 1 + strlen ( value ) + 1 ; /* NAME=VALUE\0 */
buf = malloc ( len ) ;
snprintf ( buf , len , " %s=%s " , name , value ) ;
retval = putenv ( buf ) ;
}
else
retval = - 1 ;
return retval ;
}
2004-01-23 17:40:51 +01:00
# else /* use native setenv */
# define mysetenv setenv
2004-06-18 20:16:22 +02:00
# endif
2004-01-16 03:00:47 +01:00
2003-12-30 22:30:32 +01:00
/* ************************************************************* */
/* for ttyrec */
void
2004-02-20 18:57:29 +01:00
ttyrec_getpty ( )
2003-12-30 22:30:32 +01:00
{
2004-06-18 20:04:36 +02:00
# ifdef HAVE_OPENPTY
2004-02-20 18:57:29 +01:00
if ( openpty ( & master , & slave , NULL , NULL , NULL ) = = - 1 )
graceful_exit ( 62 ) ;
2004-01-02 21:47:53 +01:00
# else
2003-12-31 20:47:49 +01:00
if ( ( master = open ( " /dev/ptmx " , O_RDWR ) ) < 0 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 62 ) ;
2004-02-20 18:57:29 +01:00
grantpt ( master ) ;
unlockpt ( master ) ;
if ( ( slave = open ( ( const char * ) ptsname ( master ) , O_RDWR ) ) < 0 )
{
graceful_exit ( 65 ) ;
}
# endif
ioctl ( slave , TIOCSWINSZ , ( char * ) & win ) ;
tcsetattr ( slave , TCSANOW , & tt ) ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2004-01-25 23:34:35 +01:00
char *
2003-12-30 22:30:32 +01:00
gen_ttyrec_filename ( )
{
2003-12-31 20:47:49 +01:00
time_t rawtime ;
struct tm * ptm ;
2004-01-25 23:34:35 +01:00
char * ttyrec_filename = calloc ( 100 , sizeof ( char ) ) ;
2003-12-31 20:47:49 +01:00
/* append time to filename */
time ( & rawtime ) ;
ptm = gmtime ( & rawtime ) ;
snprintf ( ttyrec_filename , 100 , " %04i-%02i-%02i.%02i:%02i:%02i.ttyrec " ,
ptm - > tm_year + 1900 , ptm - > tm_mon + 1 , ptm - > tm_mday ,
ptm - > tm_hour , ptm - > tm_min , ptm - > tm_sec ) ;
2004-01-25 23:34:35 +01:00
return ttyrec_filename ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2004-04-21 18:22:55 +02:00
char *
gen_nhext_filename ( )
{
time_t rawtime ;
struct tm * ptm ;
char * nhext_filename = calloc ( 100 , sizeof ( char ) ) ;
/* append time to filename */
time ( & rawtime ) ;
ptm = gmtime ( & rawtime ) ;
snprintf ( nhext_filename , 100 , " %04i-%02i-%02i.%02i:%02i:%02i.nhext " ,
ptm - > tm_year + 1900 , ptm - > tm_mon + 1 , ptm - > tm_mday ,
ptm - > tm_hour , ptm - > tm_min , ptm - > tm_sec ) ;
return nhext_filename ;
}
/* ************************************************************* */
2004-01-28 17:31:37 +01:00
char *
2007-03-29 22:14:55 +02:00
gen_inprogress_lock ( int game , pid_t pid , char * ttyrec_filename )
2003-12-30 22:30:32 +01:00
{
2004-03-04 14:58:03 +01:00
char * lockfile = NULL , filebuf [ 80 ] ;
2003-12-31 20:47:49 +01:00
int fd ;
2004-01-28 17:31:37 +01:00
size_t len ;
2004-01-30 18:32:12 +01:00
struct flock fl = { 0 } ;
2004-01-04 01:59:48 +01:00
2004-03-04 14:58:03 +01:00
snprintf ( filebuf , sizeof ( filebuf ) , " %d \n %d \n %d \n " ,
pid , win . ws_row , win . ws_col ) ;
2004-01-04 03:19:12 +01:00
2004-01-30 18:32:12 +01:00
fl . l_type = F_WRLCK ;
fl . l_whence = SEEK_SET ;
fl . l_start = 0 ;
fl . l_len = 0 ;
2007-03-31 17:17:26 +02:00
len = strlen ( globalconfig . dglroot ) + strlen ( myconfig [ game ] - > inprogressdir ) + strlen ( me - > username ) + strlen ( ttyrec_filename ) + 13 ;
2004-01-28 17:31:37 +01:00
lockfile = calloc ( len , sizeof ( char ) ) ;
2007-03-29 22:14:55 +02:00
snprintf ( lockfile , len , " %s%s%s:%s " , globalconfig . dglroot , myconfig [ game ] - > inprogressdir ,
2003-12-31 20:47:49 +01:00
me - > username , ttyrec_filename ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
fd = open ( lockfile , O_WRONLY | O_CREAT , 0644 ) ;
2004-01-04 01:59:48 +01:00
if ( fcntl ( fd , F_SETLKW , & fl ) = = - 1 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 68 ) ;
2004-01-04 03:03:26 +01:00
2004-03-04 14:58:03 +01:00
write ( fd , filebuf , strlen ( filebuf ) ) ;
2004-01-28 17:31:37 +01:00
return lockfile ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
void
2004-01-02 07:14:34 +01:00
catch_sighup ( int signum )
2003-12-30 22:30:32 +01:00
{
2004-01-25 23:34:35 +01:00
if ( child )
2003-12-31 20:47:49 +01:00
{
sleep ( 10 ) ;
2004-01-25 23:34:35 +01:00
kill ( child , SIGHUP ) ;
2003-12-31 20:47:49 +01:00
sleep ( 5 ) ;
}
2004-01-04 02:45:21 +01:00
graceful_exit ( 2 ) ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2003-12-31 20:47:49 +01:00
void
2007-03-29 22:14:55 +02:00
loadbanner ( int game , struct dg_banner * ban )
2003-12-31 20:47:49 +01:00
{
2003-12-31 21:21:00 +01:00
FILE * bannerfile ;
2003-12-31 20:47:49 +01:00
char buf [ 80 ] ;
memset ( buf , 0 , 80 ) ;
2007-03-29 22:14:55 +02:00
bannerfile = fopen ( globalconfig . banner , " r " ) ;
2003-12-31 20:47:49 +01:00
2003-12-31 21:21:00 +01:00
if ( ! bannerfile )
2003-12-31 20:47:49 +01:00
{
2004-01-05 02:55:08 +01:00
size_t len ;
2003-12-31 20:47:49 +01:00
ban - > len = 2 ;
ban - > lines = malloc ( sizeof ( char * ) ) ;
ban - > lines [ 0 ] =
2004-06-18 19:57:29 +02:00
strdup ( " ### dgamelaunch " PACKAGE_VERSION
2003-12-31 20:47:49 +01:00
" - network console game launcher " ) ;
2007-03-29 22:14:55 +02:00
len = strlen ( globalconfig . banner ) + ARRAY_SIZE ( " ### NOTE: administrator has not installed a file " ) ;
2004-01-05 02:55:08 +01:00
ban - > lines [ 1 ] = malloc ( len ) ;
2007-03-29 22:14:55 +02:00
snprintf ( ban - > lines [ 1 ] , len , " ### NOTE: administrator has not installed a %s file " , globalconfig . banner ) ;
2003-12-31 20:47:49 +01:00
return ;
}
ban - > len = 0 ;
2003-12-31 21:21:00 +01:00
while ( fgets ( buf , 80 , bannerfile ) ! = NULL )
2003-12-31 20:47:49 +01:00
{
2004-01-06 03:08:53 +01:00
char * loc , * b = buf ;
char bufnew [ 80 ] ;
memset ( bufnew , 0 , 80 ) ;
2003-12-31 20:47:49 +01:00
ban - > len + + ;
ban - > lines = realloc ( ban - > lines , sizeof ( char * ) * ban - > len ) ;
2004-01-06 03:08:53 +01:00
if ( strstr ( b , " $VERSION " ) )
{
int i = 0 ;
while ( ( loc = strstr ( b , " $VERSION " ) ) ! = NULL )
{
for ( ; i < 80 ; i + + )
2003-12-31 20:47:49 +01:00
{
if ( loc ! = b )
bufnew [ i ] = * ( b + + ) ;
else
{
2004-06-18 19:57:29 +02:00
strlcat ( bufnew , PACKAGE_VERSION , 80 ) ;
b + = 8 ; /* skip the whole $PACKAGE_VERSION string */
i + = ARRAY_SIZE ( PACKAGE_VERSION ) - 1 ;
2004-01-06 03:08:53 +01:00
break ;
2003-12-31 20:47:49 +01:00
}
if ( strlen ( b ) = = 0 )
break ;
2004-01-06 03:08:53 +01:00
}
}
if ( * b )
strlcat ( bufnew , b , 80 ) ;
ban - > lines [ ban - > len - 1 ] = strdup ( bufnew ) ;
}
2003-12-31 20:47:49 +01:00
else
ban - > lines [ ban - > len - 1 ] = strdup ( buf ) ;
memset ( buf , 0 , 80 ) ;
2004-01-01 03:06:19 +01:00
2004-02-05 20:38:17 +01:00
if ( ban - > len = = 11 ) /* menu itself needs 13 lines, 24 - 11 */
2004-01-06 06:03:30 +01:00
break ;
}
2004-01-01 03:06:19 +01:00
fclose ( bannerfile ) ;
2003-12-31 20:47:49 +01:00
}
void
drawbanner ( unsigned int start_line , unsigned int howmany )
{
static short loaded_banner = 0 ;
unsigned int i ;
if ( ! loaded_banner )
{
2007-03-29 22:14:55 +02:00
loadbanner ( 0 , & banner ) ;
2003-12-31 20:47:49 +01:00
loaded_banner = 1 ;
}
if ( howmany > banner . len | | howmany = = 0 )
howmany = banner . len ;
for ( i = 0 ; i < howmany ; i + + )
mvaddstr ( start_line + i , 1 , banner . lines [ i ] ) ;
}
2004-01-02 05:35:46 +01:00
void
2007-03-29 22:14:55 +02:00
inprogressmenu ( int gameid )
2004-01-02 05:35:46 +01:00
{
2004-03-04 15:38:02 +01:00
int i , menuchoice , len = 20 , offset = 0 , doresizewin = 0 ;
2007-12-24 19:09:15 +01:00
dg_sortmode sortmode = SORTMODE_NONE ;
2003-12-31 20:47:49 +01:00
time_t ctime ;
2004-01-02 05:35:46 +01:00
struct dg_game * * games ;
2004-04-21 18:22:55 +02:00
char ttyrecname [ 130 ] , * replacestr = NULL , gametype [ 10 ] ;
int is_nhext [ 14 ] ;
2004-03-04 15:38:02 +01:00
sigset_t oldmask , toblock ;
2003-12-31 20:47:49 +01:00
2007-03-29 22:14:55 +02:00
games = populate_games ( gameid , & len ) ;
2007-12-24 19:09:15 +01:00
games = sort_games ( games , len , sortmode ) ;
2003-12-31 20:47:49 +01:00
2004-01-02 05:35:46 +01:00
while ( 1 )
2003-12-31 20:47:49 +01:00
{
2004-03-12 17:24:18 +01:00
if ( len = = 0 )
offset = 0 ;
else
{
while ( offset > = len & & offset > = 14 )
offset - = 14 ;
}
2004-03-21 17:53:07 +01:00
erase ( ) ;
2003-12-31 20:47:49 +01:00
drawbanner ( 1 , 1 ) ;
mvprintw ( 3 , 1 ,
2004-02-03 03:55:24 +01:00
" During playback, hit 'q' to return here, 'm' to send mail (requires login), " ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( 4 , 1 ,
2004-02-03 03:55:24 +01:00
" 's' to toggle graphic-set stripping for DEC, IBM, and none. " ) ;
2004-03-04 15:38:02 +01:00
mvaddstr ( 5 , 1 , " The following games are in progress: (use uppercase to try to change size) " ) ;
2003-12-31 20:47:49 +01:00
/* clean old games and list good ones */
i = 0 ;
2004-01-02 05:35:46 +01:00
for ( i = 0 ; i < 14 ; i + + )
2003-12-31 20:47:49 +01:00
{
2004-01-02 05:35:46 +01:00
if ( i + offset > = len )
break ;
2004-04-21 18:22:55 +02:00
is_nhext [ i ] = ! strcmp ( games [ i + offset ] - > ttyrec_fn + strlen ( games [ i + offset ] - > ttyrec_fn ) - 6 , " .nhext " ) ;
if ( is_nhext [ i ] )
strcpy ( gametype , " NhExt " ) ;
else
snprintf ( gametype , sizeof gametype , " %3dx%3d " ,
games [ i + offset ] - > ws_col , games [ i + offset ] - > ws_row ) ;
2007-04-01 12:11:38 +02:00
mvprintw ( 7 + i , 1 , " %c) %-15s %-5s (%s) %s %s (%ldm %lds idle) " ,
i + 97 , games [ i + offset ] - > name , myconfig [ games [ i + offset ] - > gamenum ] - > shortname , gametype ,
2004-01-02 05:35:46 +01:00
games [ i + offset ] - > date , games [ i + offset ] - > time ,
( time ( & ctime ) - games [ i + offset ] - > idle_time ) / 60 ,
( time ( & ctime ) - games [ i + offset ] - > idle_time ) % 60 ) ;
2003-12-31 20:47:49 +01:00
}
2007-12-24 19:09:15 +01:00
mvprintw ( 22 , 1 , " 's' and 'S' change sort mode (current: %s) " , SORTMODE_NAME [ sortmode ] ) ;
2004-03-01 22:17:05 +01:00
if ( len > 0 )
mvprintw ( 21 , 1 , " (%d-%d of %d) " , offset + 1 , offset + i , len ) ;
2004-01-02 05:35:46 +01:00
mvaddstr ( 23 , 1 ,
2004-02-07 09:42:26 +01:00
" Watch which game? (any key refreshes, 'q' quits, '>'/'<' for more/less) => " ) ;
2003-12-31 20:47:49 +01:00
refresh ( ) ;
2004-03-04 15:38:02 +01:00
switch ( ( menuchoice = getch ( ) ) )
2003-12-31 20:47:49 +01:00
{
2004-01-03 04:51:29 +01:00
case ' > ' :
2004-01-02 05:35:46 +01:00
if ( ( offset + 14 ) > = len )
break ;
2003-12-31 20:47:49 +01:00
else
2004-01-02 05:35:46 +01:00
offset + = 14 ;
break ;
2004-01-01 20:57:58 +01:00
2004-01-03 04:51:29 +01:00
case ' < ' :
2004-01-02 05:35:46 +01:00
if ( ( offset - 14 ) < 0 )
break ;
else
offset - = 14 ;
break ;
2004-01-01 20:57:58 +01:00
2004-03-04 15:38:02 +01:00
case ' q ' : case ' Q ' :
2004-01-02 05:35:46 +01:00
return ;
2004-01-01 03:06:19 +01:00
2007-12-24 19:09:15 +01:00
case ' s ' :
if ( sortmode < ( NUM_SORTMODES - 1 ) ) sortmode + + ; else sortmode = SORTMODE_NONE ;
break ;
case ' S ' :
if ( sortmode > SORTMODE_NONE ) sortmode - - ; else sortmode = ( NUM_SORTMODES - 1 ) ;
break ;
2004-08-23 14:59:33 +02:00
case 12 : case 18 : /* ^L, ^R */
clear ( ) ;
break ;
2004-01-02 05:35:46 +01:00
default :
2004-03-04 15:38:02 +01:00
doresizewin = 0 ;
if ( isupper ( menuchoice ) )
{
doresizewin = 1 ;
menuchoice = tolower ( menuchoice ) ;
}
if ( ( menuchoice - ' a ' ) > = 0 & & ( menuchoice - ' a ' ) < i )
2004-01-02 05:35:46 +01:00
{
2004-04-21 18:22:55 +02:00
if ( is_nhext [ menuchoice - 97 ] ) /* Cannot watch NhExt game */
break ;
2004-01-02 05:35:46 +01:00
/* valid choice has been made */
2007-03-29 22:14:55 +02:00
snprintf ( ttyrecname , 130 , " %sttyrec/%s " , globalconfig . dglroot ,
2004-02-19 22:10:23 +01:00
games [ menuchoice - 97 + offset ] - > ttyrec_fn ) ;
2004-01-02 05:35:46 +01:00
chosen_name = strdup ( games [ menuchoice - 97 + offset ] - > name ) ;
2003-12-31 20:47:49 +01:00
2004-01-02 05:35:46 +01:00
/* reuse thie char* */
replacestr = strchr ( ttyrecname , ' : ' ) ;
if ( ! replacestr )
2004-01-04 02:45:21 +01:00
graceful_exit ( 145 ) ;
2004-01-02 05:35:46 +01:00
replacestr [ 0 ] = ' / ' ;
clear ( ) ;
refresh ( ) ;
endwin ( ) ;
2004-03-04 15:38:02 +01:00
if ( doresizewin )
{
/*
* Let curses deal with the resize later . Perhaps this is
* not the best way .
*/
sigemptyset ( & toblock ) ;
sigaddset ( & toblock , SIGWINCH ) ;
sigprocmask ( SIG_BLOCK , & toblock , & oldmask ) ;
printf ( " \033 [8;%d;%dt " ,
games [ menuchoice - 97 + offset ] - > ws_row ,
games [ menuchoice - 97 + offset ] - > ws_col ) ;
fflush ( stdout ) ;
}
2004-02-03 03:55:24 +01:00
ttyplay_main ( ttyrecname , 1 ) ;
2004-01-26 09:29:04 +01:00
initcurses ( ) ;
2004-03-04 15:38:02 +01:00
if ( doresizewin )
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
2004-01-02 05:35:46 +01:00
}
2003-12-31 20:47:49 +01:00
}
2007-03-29 22:14:55 +02:00
games = populate_games ( gameid , & len ) ;
2007-12-24 19:09:15 +01:00
games = sort_games ( games , len , sortmode ) ;
2003-12-31 20:47:49 +01:00
}
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2004-03-06 21:34:27 +01:00
/*
* Check email address , returns 1 if valid , 0 otherwise .
* Doesn ' t recognize addresses with parts in double - quotes .
* Addresses with a colon in them are always rejected .
*/
int
check_email ( char * s )
{
char * atomchars = " !#$%&'*+-/=?^_`{|}~ " " 0123456789 "
" abcdefghijklmnopqrstuvwxyz " " ABCDEFGHIJKLMNOPQRSTUVWXYZ " ;
int f ;
if ( * s = = ' @ ' )
return 0 ;
while ( * s ! = ' \0 ' & & * s ! = ' @ ' )
{
if ( strchr ( atomchars , * s ) = = NULL )
return 0 ;
s + + ;
if ( * s = = ' . ' )
s + + ;
}
if ( * s = = ' \0 ' )
return 0 ;
s + + ;
f = 0 ;
while ( * s ! = ' \0 ' )
{
if ( strchr ( atomchars , * s ) = = NULL )
return 0 ;
s + + ;
if ( * s = = ' . ' )
s + + , f = 1 ;
}
return f ;
}
2004-02-05 20:38:17 +01:00
void
change_email ( )
{
char buf [ 81 ] ;
clear ( ) ;
2004-02-05 20:50:58 +01:00
for ( ; ; )
2004-02-05 20:38:17 +01:00
{
drawbanner ( 1 , 1 ) ;
mvprintw ( 3 , 1 , " Your current email is: %s " , me - > email ) ;
mvaddstr ( 4 , 1 , " Please enter a new one (max 80 chars; blank line aborts) " ) ;
mvaddstr ( 6 , 1 , " => " ) ;
mygetnstr ( buf , 80 , 1 ) ;
2007-12-30 10:00:16 +01:00
if ( * buf = = ' \0 ' )
2004-02-05 20:38:17 +01:00
return ;
else if ( ! strcmp ( me - > email , buf ) )
{
clear ( ) ;
mvaddstr ( 8 , 1 , " That's the same one as before. Try again? " ) ;
move ( 1 , 1 ) ;
}
2004-03-06 21:34:27 +01:00
else if ( check_email ( buf ) )
2004-02-05 20:38:17 +01:00
{
mvprintw ( 8 , 1 , " Changing email address to '%s'. Confirm (y/n): " , buf ) ;
if ( getch ( ) = = ' y ' )
{
free ( me - > email ) ;
me - > email = strdup ( buf ) ;
writefile ( 0 ) ;
return ;
}
else
{
mvaddstr ( 9 , 1 , " No changes made. Press any key to continue... " ) ;
getch ( ) ;
return ;
}
}
else
{
clear ( ) ;
mvaddstr ( 8 , 1 , " That doesn't look like an email address to me. " ) ;
move ( 1 , 1 ) ;
}
}
}
2004-01-04 13:16:05 +01:00
int
2004-02-13 23:30:59 +01:00
changepw ( int dowrite )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
char buf [ 21 ] ;
int error = 2 ;
2004-01-03 04:51:29 +01:00
/* A precondition is that struct `me' exists because we can be not-yet-logged-in. */
2003-12-31 20:47:49 +01:00
if ( ! me )
2004-01-04 02:45:21 +01:00
graceful_exit ( 122 ) ; /* Die. */
2003-12-31 20:47:49 +01:00
while ( error )
{
char repeatbuf [ 21 ] ;
clear ( ) ;
drawbanner ( 1 , 1 ) ;
mvprintw ( 5 , 1 ,
" Please enter a%s password. Remember that this is sent over the net " ,
loggedin ? " new " : " " ) ;
mvaddstr ( 6 , 1 ,
" in plaintext, so make it something new and expect it to be relatively " ) ;
mvaddstr ( 7 , 1 , " insecure. " ) ;
mvaddstr ( 8 , 1 ,
" 20 character max. No ':' characters. Blank line to abort. " ) ;
mvaddstr ( 10 , 1 , " => " ) ;
if ( error = = 1 )
{
mvaddstr ( 15 , 1 , " Sorry, the passwords don't match. Try again. " ) ;
move ( 10 , 4 ) ;
}
refresh ( ) ;
2004-01-19 12:58:31 +01:00
mygetnstr ( buf , 20 , 0 ) ;
2003-12-31 20:47:49 +01:00
2007-12-30 10:00:16 +01:00
if ( * buf = = ' \0 ' )
2004-01-04 13:16:05 +01:00
return 0 ;
2003-12-31 20:47:49 +01:00
2004-01-01 03:06:19 +01:00
if ( strchr ( buf , ' : ' ) ! = NULL )
2004-01-04 02:45:21 +01:00
graceful_exit ( 112 ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( 12 , 1 , " And again: " ) ;
mvaddstr ( 13 , 1 , " => " ) ;
2004-01-19 12:58:31 +01:00
mygetnstr ( repeatbuf , 20 , 0 ) ;
2003-12-31 20:47:49 +01:00
if ( ! strcmp ( buf , repeatbuf ) )
error = 0 ;
else
error = 1 ;
}
2004-02-05 20:38:17 +01:00
free ( me - > password ) ;
2003-12-31 20:47:49 +01:00
me - > password = strdup ( crypt ( buf , buf ) ) ;
2004-02-13 23:30:59 +01:00
2004-02-13 23:36:17 +01:00
if ( dowrite )
2004-02-13 23:30:59 +01:00
writefile ( 0 ) ;
2004-01-04 13:16:05 +01:00
return 1 ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
void
domailuser ( char * username )
{
2003-12-31 20:47:49 +01:00
unsigned int len , i ;
2004-02-22 23:40:33 +01:00
char * spool_fn , message [ 81 ] ;
2003-12-31 20:47:49 +01:00
FILE * user_spool = NULL ;
time_t now ;
int mail_empty = 1 ;
2007-03-29 22:14:55 +02:00
int game ;
2004-01-30 18:32:12 +01:00
struct flock fl = { 0 } ;
fl . l_type = F_WRLCK ;
fl . l_whence = SEEK_SET ;
fl . l_start = 0 ;
fl . l_len = 0 ;
2003-12-31 20:47:49 +01:00
assert ( loggedin ) ;
2007-03-29 22:14:55 +02:00
game = 0 ; /*TODO: find_curr_player_game(username) */
if ( strlen ( myconfig [ game ] - > spool ) < 1 ) return ;
len = strlen ( myconfig [ game ] - > spool ) + strlen ( username ) + 1 ;
2003-12-31 20:47:49 +01:00
spool_fn = malloc ( len + 1 ) ;
time ( & now ) ;
2007-03-29 22:14:55 +02:00
snprintf ( spool_fn , len + 1 , " %s/%s " , myconfig [ game ] - > spool , username ) ;
2003-12-31 20:47:49 +01:00
/* print the enter your message line */
clear ( ) ;
drawbanner ( 1 , 1 ) ;
mvaddstr ( 5 , 1 ,
" Enter your message here. It is to be one line only and 80 characters or less. " ) ;
mvaddstr ( 7 , 1 , " => " ) ;
2004-01-19 12:58:31 +01:00
mygetnstr ( message , 80 , 1 ) ;
2003-12-31 20:47:49 +01:00
for ( i = 0 ; i < strlen ( message ) ; i + + )
{
if ( message [ i ] ! = ' ' & & message [ i ] ! = ' \n ' & & message [ i ] ! = ' \t ' )
mail_empty = 0 ;
}
if ( mail_empty )
{
mvaddstr ( 9 , 1 , " This scroll appears to be blank.--More-- " ) ;
mvaddstr ( 10 , 1 , " (Aborting your message.) " ) ;
getch ( ) ;
return ;
}
if ( ( user_spool = fopen ( spool_fn , " a " ) ) = = NULL )
{
mvaddstr ( 9 , 1 ,
" You fall into the water! You sink like a rock.--More-- " ) ;
mvprintw ( 10 , 1 ,
" (I couldn't open %s'%c spool file for some reason, so I'm giving up.) " ,
username , ( username [ strlen ( username ) - 1 ] ! = ' s ' ) ? ' s ' : 0 ) ;
getch ( ) ;
return ;
}
mvaddstr ( 9 , 1 , " Getting a lock on the mailspool... " ) ;
refresh ( ) ;
2004-01-04 01:21:39 +01:00
while ( fcntl ( fileno ( user_spool ) , F_SETLK , & fl ) = = - 1 )
{
2004-01-04 03:19:12 +01:00
if ( errno ! = EAGAIN )
{
mvaddstr ( 10 , 1 ,
" Received a weird error from fcntl, so I'm giving up. " ) ;
getch ( ) ;
return ;
}
sleep ( 1 ) ;
2004-01-04 01:21:39 +01:00
}
2003-12-31 20:47:49 +01:00
fprintf ( user_spool , " %s:%s \n " , me - > username , message ) ;
2004-01-06 14:35:38 +01:00
/*
* Don ' t unlock the file ourselves , this way it will be done automatically
* after all data has been written . ( Using file locking with stdio is icky . )
*/
2003-12-31 20:47:49 +01:00
fclose ( user_spool ) ;
2004-01-06 14:35:38 +01:00
mvaddstr ( 9 , 1 , " Message sent successfully " ) ;
2004-01-27 18:31:26 +01:00
move ( 9 , 26 ) ; /* Pedantry! */
2004-01-06 14:35:38 +01:00
refresh ( ) ;
sleep ( 2 ) ;
2003-12-31 20:47:49 +01:00
return ;
2003-12-30 22:30:32 +01:00
}
2007-03-29 22:14:55 +02:00
void
drawgamemenu ( int game )
{
static int flood = 0 ;
if ( loggedin ) {
clear ( ) ;
drawbanner ( 1 , 0 ) ;
mvprintw ( banner . len + 2 , 1 , " Logged in as: %s " , me - > username ) ;
mvaddstr ( banner . len + 4 , 1 , " c) Change password " ) ;
mvaddstr ( banner . len + 5 , 1 , " e) Change email address " ) ;
mvaddstr ( banner . len + 6 , 1 , " w) Watch games in progress " ) ;
if ( myconfig [ game ] - > rcfile )
mvprintw ( banner . len + 7 , 1 , " o) Edit options for %s " , myconfig [ game ] - > game_name ) ;
mvprintw ( banner . len + 8 , 1 , " p) Play %s " , myconfig [ game ] - > game_name ) ;
mvaddstr ( banner . len + 9 , 1 , " q) Return to previous menu " ) ;
mvaddstr ( banner . len + 11 , 1 , " => " ) ;
refresh ( ) ;
}
/* for retarded clients */
flood + + ;
if ( flood > = 20 )
{
endwin ( ) ;
graceful_exit ( 119 ) ;
}
}
2003-12-30 22:30:32 +01:00
void
drawmenu ( )
{
2003-12-31 20:47:49 +01:00
static int flood = 0 ;
2007-03-29 22:14:55 +02:00
int game = 0 ;
2003-12-31 20:47:49 +01:00
clear ( ) ;
drawbanner ( 1 , 0 ) ;
if ( loggedin )
{
mvprintw ( banner . len + 2 , 1 , " Logged in as: %s " , me - > username ) ;
mvaddstr ( banner . len + 4 , 1 , " c) Change password " ) ;
2004-02-05 20:38:17 +01:00
mvaddstr ( banner . len + 5 , 1 , " e) Change email address " ) ;
2007-03-29 22:14:55 +02:00
mvaddstr ( banner . len + 6 , 1 , " w) Watch games in progress " ) ;
if ( num_games > = 1 )
for ( game = 0 ; ( game < = num_games ) & & myconfig & & myconfig [ game ] & & myconfig [ game ] - > game_name ; game + + )
mvprintw ( banner . len + 7 + game , 1 , " %c) Go to %s menu " , ' 1 ' + game , myconfig [ game ] - > game_name ) ;
else {
if ( myconfig [ 0 ] - > rcfile )
mvprintw ( banner . len + 7 , 1 , " o) Edit options " ) ;
mvprintw ( banner . len + 8 , 1 , " p) Play %s " , myconfig [ 0 ] - > game_name ) ;
}
mvaddstr ( banner . len + 9 + game , 1 , " q) Quit " ) ;
mvaddstr ( banner . len + 11 + game , 1 , " => " ) ;
2003-12-31 20:47:49 +01:00
}
else
{
mvaddstr ( banner . len + 2 , 1 , " Not logged in. " ) ;
mvaddstr ( banner . len + 4 , 1 , " l) Login " ) ;
2007-04-09 18:42:14 +02:00
if ( globalconfig . allow_registration )
mvaddstr ( banner . len + 5 , 1 , " r) Register new user " ) ;
else
mvaddstr ( banner . len + 5 , 1 , " Register new user - DISABLED " ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( banner . len + 6 , 1 , " w) Watch games in progress " ) ;
mvaddstr ( banner . len + 7 , 1 , " q) Quit " ) ;
2007-03-29 22:14:55 +02:00
mvaddstr ( banner . len + 11 , 1 , " => " ) ;
2003-12-31 20:47:49 +01:00
}
refresh ( ) ;
/* for retarded clients */
flood + + ;
if ( flood > = 20 )
2004-01-05 02:25:38 +01:00
{
endwin ( ) ;
2004-01-04 02:45:21 +01:00
graceful_exit ( 119 ) ;
2004-01-05 02:25:38 +01:00
}
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
void
freefile ( )
{
2007-12-27 20:52:48 +01:00
# ifndef USE_SQLITE3
2003-12-31 20:47:49 +01:00
int i ;
/* free existing mem, clear existing entries */
for ( i = 0 ; i < f_num ; i + + )
{
2004-01-26 09:29:04 +01:00
if ( users [ i ] ! = me )
{
free ( users [ i ] - > password ) ;
free ( users [ i ] - > username ) ;
free ( users [ i ] - > email ) ;
free ( users [ i ] - > env ) ;
free ( users [ i ] ) ;
}
2003-12-31 20:47:49 +01:00
}
if ( users )
free ( users ) ;
users = NULL ;
f_num = 0 ;
2007-12-27 20:52:48 +01:00
# endif
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
void
2004-01-26 09:29:04 +01:00
initcurses ( )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
initscr ( ) ;
cbreak ( ) ;
2004-01-19 12:58:31 +01:00
noecho ( ) ;
2003-12-31 20:47:49 +01:00
nonl ( ) ;
intrflush ( stdscr , FALSE ) ;
keypad ( stdscr , TRUE ) ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2004-06-12 02:11:18 +02:00
void
autologin ( char * user , char * pass )
{
2007-12-27 20:52:48 +01:00
struct dg_user * tmp ;
tmp = userexist ( user , 0 ) ;
if ( tmp ) {
me = cpy_me ( tmp ) ;
if ( passwordgood ( pass ) ) {
loggedin = 1 ;
setproctitle ( " %s " , me - > username ) ;
2008-01-01 18:03:05 +01:00
dgl_exec_cmdqueue ( globalconfig . cmdqueue [ DGLTIME_LOGIN ] , 0 , me ) ;
2007-12-27 20:52:48 +01:00
}
}
2004-06-12 02:11:18 +02:00
}
2003-12-30 22:30:32 +01:00
void
2004-01-31 04:04:58 +01:00
loginprompt ( int from_ttyplay )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
char user_buf [ 22 ] , pw_buf [ 22 ] ;
2007-12-30 09:29:28 +01:00
int error = 2 ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
loggedin = 0 ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
while ( error )
{
clear ( ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
drawbanner ( 1 , 1 ) ;
2003-12-30 22:30:32 +01:00
2004-01-31 04:04:58 +01:00
if ( from_ttyplay = = 1 )
mvaddstr ( 4 , 1 , " This operation requires you to be logged in. " ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( 5 , 1 ,
2004-01-31 04:04:58 +01:00
" Please enter your username. (blank entry aborts) " ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( 7 , 1 , " => " ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
if ( error = = 1 )
{
mvaddstr ( 9 , 1 , " There was a problem with your last entry. " ) ;
move ( 7 , 4 ) ;
}
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
refresh ( ) ;
2003-12-30 22:30:32 +01:00
2007-03-31 13:10:07 +02:00
/* keep this at 20 chars for hysterical raisins */
2004-01-19 12:58:31 +01:00
mygetnstr ( user_buf , 20 , 1 ) ;
2003-12-30 22:30:32 +01:00
2007-12-30 10:00:16 +01:00
if ( * user_buf = = ' \0 ' )
2003-12-31 20:47:49 +01:00
return ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
error = 1 ;
2003-12-30 22:30:32 +01:00
2007-12-27 20:52:48 +01:00
{
struct dg_user * tmpme ;
if ( ( tmpme = userexist ( user_buf , 0 ) ) ) {
me = cpy_me ( tmpme ) ;
error = 0 ;
}
}
2003-12-31 20:47:49 +01:00
}
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
clear ( ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
drawbanner ( 1 , 1 ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
mvaddstr ( 5 , 1 , " Please enter your password. " ) ;
mvaddstr ( 7 , 1 , " => " ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
refresh ( ) ;
2003-12-30 22:30:32 +01:00
2004-01-19 12:58:31 +01:00
mygetnstr ( pw_buf , 20 , 0 ) ;
2003-12-30 22:30:32 +01:00
2004-01-02 22:00:54 +01:00
if ( passwordgood ( pw_buf ) )
2003-12-31 20:47:49 +01:00
{
loggedin = 1 ;
2004-06-28 21:41:06 +02:00
setproctitle ( " %s " , me - > username ) ;
2008-01-01 18:03:05 +01:00
dgl_exec_cmdqueue ( globalconfig . cmdqueue [ DGLTIME_LOGIN ] , 0 , me ) ;
2003-12-31 20:47:49 +01:00
}
2004-04-22 17:04:15 +02:00
else
2004-01-31 04:10:03 +01:00
{
2004-04-22 17:04:15 +02:00
me = NULL ;
if ( from_ttyplay = = 1 )
{
mvaddstr ( 9 , 1 , " Login failed. Returning to game. " ) ;
refresh ( ) ;
sleep ( 2 ) ;
}
}
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
void
newuser ( )
{
2004-02-03 00:28:17 +01:00
char buf [ 1024 ] , dirname [ 100 ] ;
2003-12-31 20:47:49 +01:00
int error = 2 ;
unsigned int i ;
loggedin = 0 ;
2003-12-30 22:30:32 +01:00
2007-12-27 20:52:48 +01:00
# ifndef USE_SQLITE3
2007-03-29 22:14:55 +02:00
if ( f_num > = globalconfig . max )
2004-01-21 17:31:40 +01:00
{
clear ( ) ;
drawbanner ( 1 , 1 ) ;
mvaddstr ( 5 , 1 , " Sorry, too many users have registered now. " ) ;
mvaddstr ( 6 , 1 , " You might email the server administrator. " ) ;
mvaddstr ( 7 , 1 , " Press return to return to the menu. " ) ;
getch ( ) ;
return ;
}
2007-12-27 20:52:48 +01:00
# endif
2004-01-21 17:31:40 +01:00
2003-12-31 20:47:49 +01:00
if ( me )
free ( me ) ;
2003-12-30 22:30:32 +01:00
2004-02-17 02:00:22 +01:00
me = calloc ( 1 , sizeof ( struct dg_user ) ) ;
2003-12-31 02:58:56 +01:00
2003-12-31 20:47:49 +01:00
while ( error )
{
clear ( ) ;
2003-12-30 23:41:47 +01:00
2007-03-31 13:10:07 +02:00
sprintf ( buf , " %i character max. " , globalconfig . max_newnick_len ) ;
2003-12-31 20:47:49 +01:00
drawbanner ( 1 , 1 ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
mvaddstr ( 5 , 1 , " Welcome new user. Please enter a username. " ) ;
mvaddstr ( 6 , 1 ,
" Only characters and numbers are allowed, with no spaces. " ) ;
2007-03-31 13:10:07 +02:00
mvaddstr ( 7 , 1 , buf ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( 9 , 1 , " => " ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
if ( error = = 1 )
{
mvaddstr ( 11 , 1 , " There was a problem with your last entry. " ) ;
move ( 9 , 4 ) ;
}
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
refresh ( ) ;
2003-12-30 22:30:32 +01:00
2007-03-31 13:10:07 +02:00
mygetnstr ( buf , globalconfig . max_newnick_len , 1 ) ;
2007-12-27 20:52:48 +01:00
2007-12-30 10:00:16 +01:00
if ( * buf = = ' \0 ' ) {
2007-12-27 20:52:48 +01:00
free ( me ) ;
me = NULL ;
return ;
}
if ( ! userexist ( buf , 1 ) ) {
error = 0 ;
} else
error = 1 ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
for ( i = 0 ; i < strlen ( buf ) ; i + + )
{
2004-01-16 03:00:47 +01:00
if ( ! isalnum ( ( int ) buf [ i ] ) )
2003-12-31 20:47:49 +01:00
error = 1 ;
}
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
if ( strlen ( buf ) < 2 )
error = 1 ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
if ( strlen ( buf ) = = 0 )
2004-01-05 02:25:38 +01:00
{
free ( me ) ;
2004-02-17 01:58:21 +01:00
me = NULL ;
2003-12-31 20:47:49 +01:00
return ;
2004-01-05 02:25:38 +01:00
}
2003-12-31 20:47:49 +01:00
}
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
me - > username = strdup ( buf ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
/* password step */
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
clear ( ) ;
2003-12-30 22:30:32 +01:00
2004-02-13 23:30:59 +01:00
if ( ! changepw ( 0 ) ) /* Calling changepw instead to prompt twice. */
2004-01-04 13:16:05 +01:00
{
2004-01-05 02:25:38 +01:00
free ( me - > username ) ;
2004-01-04 13:16:05 +01:00
free ( me ) ;
me = NULL ;
return ;
}
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
/* email step */
2003-12-30 22:30:32 +01:00
2004-03-06 21:34:27 +01:00
error = 2 ;
while ( error ! = 0 )
{
clear ( ) ;
2003-12-30 22:30:32 +01:00
2004-03-06 21:34:27 +01:00
drawbanner ( 1 , 1 ) ;
2003-12-30 22:30:32 +01:00
2004-03-06 21:34:27 +01:00
mvaddstr ( 5 , 1 , " Please enter your email address. " ) ;
mvaddstr ( 6 , 1 , " This is sent _nowhere_ but will be used if you ask "
" the sysadmin for lost " ) ;
mvaddstr ( 7 , 1 , " password help. Please use a correct one. It only "
" benefits you. " ) ;
mvaddstr ( 8 , 1 , " 80 character max. No ':' characters. Blank line "
" aborts. " ) ;
mvaddstr ( 10 , 1 , " => " ) ;
2003-12-30 22:30:32 +01:00
2004-03-06 21:34:27 +01:00
if ( error = = 1 )
{
mvaddstr ( 12 , 1 , " There was a problem with your last entry. " ) ;
move ( 10 , 4 ) ;
}
2003-12-30 22:30:32 +01:00
2004-03-06 21:34:27 +01:00
refresh ( ) ;
mygetnstr ( buf , 80 , 1 ) ;
if ( check_email ( buf ) )
error = 0 ;
else
error = 1 ;
2007-12-30 10:00:16 +01:00
if ( * buf = = ' \0 ' )
2004-03-06 21:34:27 +01:00
{
free ( me - > username ) ;
free ( me - > password ) ;
free ( me ) ;
me = NULL ;
return ;
}
}
2004-01-05 02:25:38 +01:00
2003-12-31 20:47:49 +01:00
me - > email = strdup ( buf ) ;
me - > env = calloc ( 1 , 1 ) ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
loggedin = 1 ;
2003-12-30 22:30:32 +01:00
2004-06-28 21:41:06 +02:00
setproctitle ( " %s " , me - > username ) ;
2004-02-17 19:33:13 +01:00
2008-01-01 18:03:05 +01:00
dgl_exec_cmdqueue ( globalconfig . cmdqueue [ DGLTIME_REGISTER ] , 0 , me ) ;
2004-02-03 00:28:17 +01:00
/* create their ttyrec dir */
2008-01-01 18:03:05 +01:00
/*
2007-03-29 22:14:55 +02:00
snprintf ( dirname , 100 , " %sttyrec/%s " , globalconfig . dglroot , me - > username ) ;
2004-02-03 00:28:17 +01:00
if ( access ( dirname , F_OK ) ! = 0 )
mkdir ( dirname , 0755 ) ;
2008-01-01 18:03:05 +01:00
*/
2003-12-31 20:47:49 +01:00
writefile ( 1 ) ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
int
2004-01-02 22:00:54 +01:00
passwordgood ( char * cpw )
2003-12-30 22:30:32 +01:00
{
2004-06-14 20:06:25 +02:00
assert ( me ! = NULL ) ;
2004-01-02 22:00:54 +01:00
2003-12-31 20:47:49 +01:00
if ( ! strncmp ( crypt ( cpw , cpw ) , me - > password , 13 ) )
return 1 ;
if ( ! strncmp ( cpw , me - > password , 20 ) )
return 1 ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
return 0 ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
int
readfile ( int nolock )
{
2007-12-27 20:52:48 +01:00
# ifndef USE_SQLITE3
2003-12-31 20:47:49 +01:00
FILE * fp = NULL , * fpl = NULL ;
char buf [ 1200 ] ;
2004-01-30 18:32:12 +01:00
struct flock fl = { 0 } ;
2004-01-04 01:59:48 +01:00
2004-01-30 18:32:12 +01:00
fl . l_type = F_RDLCK ;
fl . l_whence = SEEK_SET ;
fl . l_start = 0 ;
fl . l_len = 0 ;
memset ( buf , 1024 , 0 ) ;
2003-12-31 20:47:49 +01:00
/* read new stuff */
if ( ! nolock )
{
2007-03-31 23:29:45 +02:00
fpl = fopen ( globalconfig . lockfile , " r " ) ;
2003-12-31 20:47:49 +01:00
if ( ! fpl )
2004-01-04 02:45:21 +01:00
graceful_exit ( 106 ) ;
2004-01-04 03:19:12 +01:00
if ( fcntl ( fileno ( fpl ) , F_SETLKW , & fl ) = = - 1 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 114 ) ;
2003-12-31 20:47:49 +01:00
}
2007-03-31 23:18:32 +02:00
fp = fopen ( globalconfig . passwd , " r " ) ;
2003-12-31 20:47:49 +01:00
if ( ! fp )
2004-01-04 02:45:21 +01:00
graceful_exit ( 106 ) ;
2003-12-31 20:47:49 +01:00
/* once per name in the file */
while ( fgets ( buf , 1200 , fp ) )
{
char * b = buf , * n = buf ;
users = realloc ( users , sizeof ( struct dg_user * ) * ( f_num + 1 ) ) ;
users [ f_num ] = malloc ( sizeof ( struct dg_user ) ) ;
users [ f_num ] - > username = ( char * ) calloc ( 22 , sizeof ( char ) ) ;
users [ f_num ] - > email = ( char * ) calloc ( 82 , sizeof ( char ) ) ;
users [ f_num ] - > password = ( char * ) calloc ( 22 , sizeof ( char ) ) ;
users [ f_num ] - > env = ( char * ) calloc ( 1026 , sizeof ( char ) ) ;
/* name field, must be valid */
while ( * b ! = ' : ' )
{
2004-01-16 03:00:47 +01:00
if ( ! isalnum ( ( int ) * b ) )
2003-12-31 20:47:49 +01:00
return 1 ;
users [ f_num ] - > username [ ( b - n ) ] = * b ;
b + + ;
if ( ( b - n ) > = 21 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 100 ) ;
2003-12-31 20:47:49 +01:00
}
/* advance to next field */
n = b + 1 ;
b = n ;
/* email field */
while ( * b ! = ' : ' )
{
users [ f_num ] - > email [ ( b - n ) ] = * b ;
b + + ;
if ( ( b - n ) > 80 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 101 ) ;
2003-12-31 20:47:49 +01:00
}
/* advance to next field */
n = b + 1 ;
b = n ;
/* pw field */
while ( * b ! = ' : ' )
{
users [ f_num ] - > password [ ( b - n ) ] = * b ;
b + + ;
if ( ( b - n ) > = 20 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 102 ) ;
2003-12-31 20:47:49 +01:00
}
/* advance to next field */
n = b + 1 ;
b = n ;
/* env field */
while ( ( * b ! = ' \n ' ) & & ( * b ! = 0 ) & & ( * b ! = EOF ) )
{
users [ f_num ] - > env [ ( b - n ) ] = * b ;
b + + ;
if ( ( b - n ) > = 1024 )
2004-01-04 02:45:21 +01:00
graceful_exit ( 102 ) ;
2003-12-31 20:47:49 +01:00
}
f_num + + ;
/* prevent a buffer overrun here */
2007-03-29 22:14:55 +02:00
if ( f_num > globalconfig . max )
2004-02-01 09:20:28 +01:00
{
fprintf ( stderr , " ERROR: number of users in database exceeds maximum. Exiting. \n " ) ;
2004-01-04 02:45:21 +01:00
graceful_exit ( 109 ) ;
2004-02-01 09:20:28 +01:00
}
2003-12-31 20:47:49 +01:00
}
if ( ! nolock )
fclose ( fpl ) ;
fclose ( fp ) ;
2007-12-27 20:52:48 +01:00
# endif
2003-12-31 20:47:49 +01:00
return 0 ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2007-12-27 20:52:48 +01:00
# ifndef USE_SQLITE3
2007-12-30 09:29:28 +01:00
struct dg_user * userexist_tmp_me = NULL ;
struct dg_user *
2007-03-31 13:10:07 +02:00
userexist ( char * cname , int isnew )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
int i ;
2003-12-30 22:30:32 +01:00
2007-12-30 09:29:28 +01:00
if ( userexist_tmp_me ) {
free ( userexist_tmp_me - > username ) ;
free ( userexist_tmp_me - > email ) ;
free ( userexist_tmp_me - > env ) ;
free ( userexist_tmp_me - > password ) ;
free ( userexist_tmp_me ) ;
userexist_tmp_me = NULL ;
}
2003-12-31 20:47:49 +01:00
for ( i = 0 ; i < f_num ; i + + )
{
2007-12-30 09:29:28 +01:00
if ( ! strncasecmp ( cname , users [ i ] - > username , ( isnew ? globalconfig . max_newnick_len : 20 ) ) ) {
userexist_tmp_me = cpy_me ( users [ i ] ) ;
return userexist_tmp_me ;
}
2003-12-31 20:47:49 +01:00
}
2003-12-30 22:30:32 +01:00
2007-12-30 09:29:28 +01:00
return NULL ;
2003-12-30 22:30:32 +01:00
}
2007-12-27 20:52:48 +01:00
# else
struct dg_user * userexist_tmp_me = NULL ;
static int
userexist_callback ( void * NotUsed , int argc , char * * argv , char * * colname )
{
int i ;
NotUsed = NULL ;
userexist_tmp_me = malloc ( sizeof ( struct dg_user ) ) ;
for ( i = 0 ; i < argc ; i + + ) {
if ( ! strcmp ( colname [ i ] , " username " ) )
userexist_tmp_me - > username = strdup ( argv [ i ] ) ;
else if ( ! strcmp ( colname [ i ] , " email " ) )
userexist_tmp_me - > email = strdup ( argv [ i ] ) ;
else if ( ! strcmp ( colname [ i ] , " env " ) )
userexist_tmp_me - > env = strdup ( argv [ i ] ) ;
else if ( ! strcmp ( colname [ i ] , " password " ) )
userexist_tmp_me - > password = strdup ( argv [ i ] ) ;
else if ( ! strcmp ( colname [ i ] , " flags " ) )
userexist_tmp_me - > flags = atoi ( argv [ i ] ) ;
else if ( ! strcmp ( colname [ i ] , " id " ) )
userexist_tmp_me - > id = atoi ( argv [ i ] ) ;
}
return 0 ;
}
struct dg_user *
userexist ( char * cname , int isnew )
{
sqlite3 * db ;
char * errmsg = NULL ;
2007-12-28 00:44:09 +01:00
int ret , retry = 10 ;
2007-12-27 20:52:48 +01:00
char * qbuf ;
char tmpbuf [ 32 ] ;
strncpy ( tmpbuf , cname , ( isnew ? globalconfig . max_newnick_len : 20 ) ) ;
2007-12-28 11:01:35 +01:00
/* Check that the nick doesn't interfere with already registered nicks */
if ( isnew & & ( strlen ( cname ) > = globalconfig . max_newnick_len ) )
strcat ( tmpbuf , " % " ) ;
qbuf = sqlite3_mprintf ( " select * from dglusers where username like '%q' limit 1 " , tmpbuf ) ;
2007-12-27 20:52:48 +01:00
2007-12-30 09:44:02 +01:00
ret = sqlite3_open ( USE_SQLITE_DB , & db ) ; /* FIXME: use globalconfig->passwd? */
2007-12-27 20:52:48 +01:00
if ( ret ) {
sqlite3_close ( db ) ;
graceful_exit ( 109 ) ;
}
if ( userexist_tmp_me ) {
free ( userexist_tmp_me - > username ) ;
free ( userexist_tmp_me - > email ) ;
free ( userexist_tmp_me - > env ) ;
free ( userexist_tmp_me - > password ) ;
free ( userexist_tmp_me ) ;
userexist_tmp_me = NULL ;
}
2007-12-28 00:44:09 +01:00
do {
ret = sqlite3_exec ( db , qbuf , userexist_callback , 0 , & errmsg ) ;
if ( ret = = SQLITE_BUSY ) sleep ( 1 ) ;
} while ( ( ret = = SQLITE_BUSY ) & & ( retry - - > 0 ) ) ;
2007-12-27 20:52:48 +01:00
if ( ret ! = SQLITE_OK ) {
sqlite3_close ( db ) ;
graceful_exit ( 108 ) ;
}
sqlite3_close ( db ) ;
sqlite3_free ( qbuf ) ;
return userexist_tmp_me ;
}
# endif
2003-12-30 22:30:32 +01:00
/* ************************************************************* */
void
2007-03-29 22:14:55 +02:00
write_canned_rcfile ( int game , char * target )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
FILE * canned , * newfile ;
2004-01-05 02:55:08 +01:00
char buf [ 1024 ] , * rfn ;
size_t bytes , len ;
2007-03-29 22:14:55 +02:00
len = strlen ( myconfig [ game ] - > rcfile ) + 2 ;
2004-01-05 02:55:08 +01:00
rfn = malloc ( len ) ;
2007-03-29 22:14:55 +02:00
snprintf ( rfn , len , " /%s " , myconfig [ game ] - > rcfile ) ;
2003-12-31 20:47:49 +01:00
if ( ! ( newfile = fopen ( target , " w " ) ) )
{
bail :
mvaddstr ( 13 , 1 ,
2004-02-01 09:10:00 +01:00
" You don't know how to write that! You write \" %s was here \" and the scroll disappears. " ) ;
2003-12-31 20:47:49 +01:00
mvaddstr ( 14 , 1 ,
2007-03-29 22:14:55 +02:00
" (Sorry, but I couldn't open one of the config files. This is a bug.) " ) ;
2003-12-31 20:47:49 +01:00
return ;
}
2004-01-05 02:55:08 +01:00
if ( ! ( canned = fopen ( rfn , " r " ) ) )
2003-12-31 20:47:49 +01:00
goto bail ;
2004-01-05 02:55:08 +01:00
free ( rfn ) ;
2003-12-31 20:47:49 +01:00
while ( ( bytes = fread ( buf , 1 , 1024 , canned ) ) > 0 )
{
if ( fwrite ( buf , 1 , bytes , newfile ) ! = bytes )
{
if ( ferror ( newfile ) )
{
mvaddstr ( 13 , 1 , " Your hand slips while engraving. " ) ;
mvaddstr ( 14 , 1 ,
" (Encountered a problem writing the new file. This is a bug.) " ) ;
fclose ( canned ) ;
fclose ( newfile ) ;
return ;
}
}
}
fclose ( canned ) ;
fclose ( newfile ) ;
2003-12-30 22:30:32 +01:00
}
2003-12-31 02:58:56 +01:00
2003-12-30 22:30:32 +01:00
void
2007-03-29 22:14:55 +02:00
editoptions ( int game )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
FILE * rcfile ;
char * myargv [ 3 ] ;
2004-01-23 19:14:48 +01:00
pid_t editor ;
2003-12-30 22:30:32 +01:00
2007-03-29 22:14:55 +02:00
rcfile = fopen ( dgl_format_str ( game , me , myconfig [ game ] - > rc_fmt ) , " r " ) ;
if ( ! rcfile )
write_canned_rcfile ( game , dgl_format_str ( game , me , myconfig [ game ] - > rc_fmt ) ) ;
2003-12-31 02:58:56 +01:00
2004-01-31 04:04:58 +01:00
/* use whatever editor_main to edit */
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
myargv [ 0 ] = " " ;
2007-03-29 22:14:55 +02:00
myargv [ 1 ] = dgl_format_str ( game , me , myconfig [ game ] - > rc_fmt ) ;
2003-12-31 20:47:49 +01:00
myargv [ 2 ] = 0 ;
2003-12-30 22:30:32 +01:00
2003-12-31 20:47:49 +01:00
endwin ( ) ;
2004-01-23 19:14:48 +01:00
editor = fork ( ) ;
if ( editor = = - 1 )
{
perror ( " fork " ) ;
graceful_exit ( 114 ) ;
}
else if ( editor = = 0 )
{
2004-01-25 23:44:34 +01:00
editor_main ( 2 , myargv ) ;
2004-01-23 19:14:48 +01:00
exit ( 0 ) ;
}
else
waitpid ( editor , NULL , 0 ) ;
2003-12-31 20:47:49 +01:00
refresh ( ) ;
2003-12-30 22:30:32 +01:00
}
/* ************************************************************* */
2007-12-27 20:52:48 +01:00
# ifndef USE_SQLITE3
2003-12-30 22:30:32 +01:00
void
writefile ( int requirenew )
{
2003-12-31 20:47:49 +01:00
FILE * fp , * fpl ;
int i = 0 ;
int my_done = 0 ;
2004-01-30 18:32:12 +01:00
struct flock fl = { 0 } ;
2004-02-17 01:53:28 +01:00
sigset_t oldmask , toblock ;
2004-01-30 18:32:12 +01:00
fl . l_type = F_WRLCK ;
fl . l_whence = SEEK_SET ;
fl . l_start = 0 ;
fl . l_len = 0 ;
2003-12-31 20:47:49 +01:00
2004-02-17 01:53:28 +01:00
sigemptyset ( & toblock ) ;
sigaddset ( & toblock , SIGHUP ) ;
sigaddset ( & toblock , SIGINT ) ;
sigaddset ( & toblock , SIGQUIT ) ;
sigaddset ( & toblock , SIGTERM ) ;
sigprocmask ( SIG_BLOCK , & toblock , & oldmask ) ;
2007-03-31 23:29:45 +02:00
fpl = fopen ( globalconfig . lockfile , " r+ " ) ;
2003-12-31 20:47:49 +01:00
if ( ! fpl )
2004-02-17 01:53:28 +01:00
{
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
graceful_exit ( 115 ) ;
}
2004-01-12 18:32:47 +01:00
if ( fcntl ( fileno ( fpl ) , F_SETLK , & fl ) )
2004-02-17 01:53:28 +01:00
{
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
graceful_exit ( 107 ) ;
}
2003-12-31 20:47:49 +01:00
2004-01-12 18:32:47 +01:00
fl . l_type = F_UNLCK ;
2003-12-31 20:47:49 +01:00
freefile ( ) ;
readfile ( 1 ) ;
2007-03-31 23:18:32 +02:00
fp = fopen ( globalconfig . passwd , " w " ) ;
2003-12-31 20:47:49 +01:00
if ( ! fp )
2004-02-17 01:53:28 +01:00
{
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
graceful_exit ( 104 ) ;
}
2003-12-31 20:47:49 +01:00
for ( i = 0 ; i < f_num ; i + + )
{
2004-01-05 18:37:21 +01:00
if ( loggedin & & ! strncmp ( me - > username , users [ i ] - > username , 20 ) )
2003-12-31 20:47:49 +01:00
{
if ( requirenew )
{
/* this is if someone managed to register at the same time
* as someone else . just die . */
2004-02-17 01:53:28 +01:00
fclose ( fp ) ;
fclose ( fpl ) ;
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
2004-01-04 02:45:21 +01:00
graceful_exit ( 111 ) ;
2003-12-31 20:47:49 +01:00
}
fprintf ( fp , " %s:%s:%s:%s \n " , me - > username , me - > email , me - > password ,
me - > env ) ;
my_done = 1 ;
}
else
{
fprintf ( fp , " %s:%s:%s:%s \n " , users [ i ] - > username , users [ i ] - > email ,
users [ i ] - > password , users [ i ] - > env ) ;
}
}
if ( loggedin & & ! my_done )
{ /* new entry */
2007-03-29 22:14:55 +02:00
if ( f_num < globalconfig . max )
2004-01-21 17:31:40 +01:00
fprintf ( fp , " %s:%s:%s:%s \n " , me - > username , me - > email , me - > password ,
me - > env ) ;
else /* Oops, someone else registered the last available slot first */
2004-02-17 01:53:28 +01:00
{
fclose ( fp ) ;
fclose ( fpl ) ;
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
graceful_exit ( 116 ) ;
}
2003-12-31 20:47:49 +01:00
}
fclose ( fp ) ;
fclose ( fpl ) ;
2004-02-17 01:53:28 +01:00
sigprocmask ( SIG_SETMASK , & oldmask , NULL ) ;
2003-12-30 22:30:32 +01:00
}
2007-12-27 20:52:48 +01:00
# else
void
writefile ( int requirenew )
{
sqlite3 * db ;
char * errmsg = NULL ;
2007-12-28 00:44:09 +01:00
int ret , retry = 10 ;
2007-12-27 20:52:48 +01:00
char * qbuf ;
char tmpbuf [ 32 ] ;
if ( requirenew ) {
qbuf = sqlite3_mprintf ( " insert into dglusers (username, email, env, password, flags) values ('%q', '%q', '%q', '%q', %li) " , me - > username , me - > email , me - > env , me - > password , me - > flags ) ;
} else {
qbuf = sqlite3_mprintf ( " update dglusers set username='%q', email='%q', env='%q', password='%q', flags=%li where id=%i " , me - > username , me - > email , me - > env , me - > password , me - > flags , me - > id ) ;
}
2007-12-30 09:44:02 +01:00
ret = sqlite3_open ( USE_SQLITE_DB , & db ) ; /* FIXME: use globalconfig->passwd? */
2007-12-27 20:52:48 +01:00
if ( ret ) {
sqlite3_close ( db ) ;
graceful_exit ( 107 ) ;
}
2007-12-28 00:44:09 +01:00
do {
ret = sqlite3_exec ( db , qbuf , NULL , NULL , & errmsg ) ;
if ( ret = = SQLITE_BUSY ) sleep ( 1 ) ;
} while ( ( ret = = SQLITE_BUSY ) & & ( retry - - > 0 ) ) ;
2007-12-27 20:52:48 +01:00
sqlite3_free ( qbuf ) ;
if ( ret ! = SQLITE_OK ) {
sqlite3_close ( db ) ;
graceful_exit ( 106 ) ;
}
sqlite3_close ( db ) ;
}
# endif
2003-12-30 22:30:32 +01:00
2004-01-04 02:45:21 +01:00
/* ************************************************************* */
/* ************************************************************* */
2003-12-30 22:30:32 +01:00
/* ************************************************************* */
/* ************************************************************* */
/* ************************************************************* */
2007-03-29 22:14:55 +02:00
2004-01-28 17:31:37 +01:00
int
2007-03-29 22:14:55 +02:00
purge_stale_locks ( int game )
2004-01-04 03:03:26 +01:00
{
2004-01-04 03:19:12 +01:00
DIR * pdir ;
2004-01-04 03:03:26 +01:00
struct dirent * dent ;
2004-01-05 02:55:08 +01:00
char * dir ;
size_t len ;
2004-01-28 17:31:37 +01:00
short firsttime = 1 ;
2004-01-05 02:55:08 +01:00
2007-03-29 22:14:55 +02:00
len = strlen ( globalconfig . dglroot ) + strlen ( myconfig [ game ] - > inprogressdir ) + 1 ;
2004-01-05 02:55:08 +01:00
dir = malloc ( len ) ;
2007-03-29 22:14:55 +02:00
snprintf ( dir , len , " %s%s " , globalconfig . dglroot , myconfig [ game ] - > inprogressdir ) ;
2004-01-05 02:55:08 +01:00
if ( ! ( pdir = opendir ( dir ) ) )
2004-01-04 03:19:12 +01:00
graceful_exit ( 200 ) ;
2004-01-05 02:55:08 +01:00
free ( dir ) ;
2004-01-04 03:19:12 +01:00
while ( ( dent = readdir ( pdir ) ) ! = NULL )
{
FILE * ipfile ;
2004-01-04 03:35:40 +01:00
char * colon , * fn ;
2004-01-04 03:19:12 +01:00
char buf [ 16 ] ;
pid_t pid ;
2004-01-04 03:35:40 +01:00
size_t len ;
int seconds = 0 ;
2007-03-29 22:14:55 +02:00
2004-01-04 23:49:29 +01:00
if ( ! strcmp ( dent - > d_name , " . " ) | | ! strcmp ( dent - > d_name , " .. " ) )
continue ;
2004-01-04 03:28:57 +01:00
2004-01-04 03:19:12 +01:00
colon = strchr ( dent - > d_name , ' : ' ) ;
/* should never happen */
if ( ! colon )
graceful_exit ( 201 ) ;
2004-02-16 00:19:39 +01:00
if ( colon - dent - > d_name ! = strlen ( me - > username ) )
continue ;
2004-01-04 03:19:12 +01:00
if ( strncmp ( dent - > d_name , me - > username , colon - dent - > d_name ) )
continue ;
2004-01-04 03:03:26 +01:00
2007-06-11 21:40:51 +02:00
len = strlen ( dent - > d_name ) + strlen ( globalconfig . dglroot ) + strlen ( myconfig [ game ] - > inprogressdir ) + 1 ;
2004-01-04 03:35:40 +01:00
fn = malloc ( len ) ;
2007-03-29 22:14:55 +02:00
snprintf ( fn , len , " %s%s%s " , globalconfig . dglroot , myconfig [ game ] - > inprogressdir , dent - > d_name ) ;
fprintf ( stderr , " ERR:'%s' \n " , fn ) ;
2004-01-04 23:49:29 +01:00
2004-01-04 03:35:40 +01:00
if ( ! ( ipfile = fopen ( fn , " r " ) ) )
2004-01-04 03:19:12 +01:00
graceful_exit ( 202 ) ;
2004-01-04 03:03:26 +01:00
2004-01-04 03:19:12 +01:00
if ( fgets ( buf , 16 , ipfile ) = = NULL )
graceful_exit ( 203 ) ;
2004-01-04 03:03:26 +01:00
2004-01-04 03:19:12 +01:00
fclose ( ipfile ) ;
2004-01-04 03:35:40 +01:00
2004-01-28 17:31:37 +01:00
if ( firsttime )
{
clear ( ) ;
drawbanner ( 1 , 1 ) ;
2004-02-01 09:10:00 +01:00
# define HUP_WAIT 10 /* seconds before HUPPING */
mvprintw ( 3 , 1 ,
2004-04-03 21:27:42 +02:00
" There are some stale %s processes, will recover in %d seconds. " ,
2007-03-29 22:14:55 +02:00
myconfig [ game ] - > game_name , HUP_WAIT ) ;
2004-01-28 17:31:37 +01:00
mvaddstr ( 4 , 1 ,
" Press a key NOW if you don't want this to happen! " ) ;
2007-03-29 22:14:55 +02:00
move ( 3 , 51 + strlen ( myconfig [ game ] - > game_name ) ) ; /* pedantry */
2004-01-28 17:31:37 +01:00
halfdelay ( 10 ) ;
2007-03-29 22:14:55 +02:00
2004-02-01 09:10:00 +01:00
for ( seconds = HUP_WAIT - 1 ; seconds > = 0 ; seconds - - )
2004-01-28 17:31:37 +01:00
{
if ( getch ( ) ! = ERR )
{
nocbreak ( ) ; /* leave half-delay */
cbreak ( ) ;
return 0 ;
}
2007-03-29 22:14:55 +02:00
mvprintw ( 3 , 50 + strlen ( myconfig [ game ] - > game_name ) , " %d%s " , seconds , ( seconds > 9 ) ? " " : " " ) ;
2004-01-28 17:31:37 +01:00
}
nocbreak ( ) ;
cbreak ( ) ;
2007-03-29 22:14:55 +02:00
2004-01-28 17:31:37 +01:00
firsttime = 0 ;
}
2004-01-30 01:56:46 +01:00
clear ( ) ;
2004-01-04 03:56:09 +01:00
refresh ( ) ;
2004-01-04 03:03:26 +01:00
2004-01-04 03:19:12 +01:00
pid = atoi ( buf ) ;
2004-01-04 03:03:26 +01:00
2004-01-04 03:19:12 +01:00
kill ( pid , SIGHUP ) ;
2004-01-04 03:03:26 +01:00
2004-01-04 03:19:12 +01:00
errno = 0 ;
2004-01-04 03:03:26 +01:00
2004-01-04 03:19:12 +01:00
/* Wait for it to stop running */
2004-01-04 03:31:00 +01:00
seconds = 0 ;
2004-01-04 03:19:12 +01:00
while ( kill ( pid , 0 ) = = 0 )
{
seconds + + ;
sleep ( 1 ) ;
if ( seconds = = 10 )
{
2004-04-03 21:27:42 +02:00
mvprintw ( 3 , 1 ,
2007-03-29 22:14:55 +02:00
" Couldn't terminate one of your stale %s processes gracefully. " , myconfig [ game ] - > game_name ) ;
2004-01-04 03:19:12 +01:00
mvaddstr ( 4 , 1 , " Force its termination? [yn] " ) ;
if ( tolower ( getch ( ) ) = = ' y ' )
{
kill ( pid , SIGTERM ) ;
break ;
}
2004-01-04 23:49:29 +01:00
else
{
endwin ( ) ;
2004-04-03 21:27:42 +02:00
fprintf ( stderr , " Sorry, no %s for you now, please "
2007-03-29 22:14:55 +02:00
" contact the admin. \n " , myconfig [ game ] - > game_name ) ;
2004-01-04 23:49:29 +01:00
graceful_exit ( 1 ) ;
}
2004-01-04 03:19:12 +01:00
}
}
2004-01-04 03:56:09 +01:00
/* Don't remove the lock file until the process is dead. */
unlink ( fn ) ;
free ( fn ) ;
2004-01-04 03:19:12 +01:00
}
closedir ( pdir ) ;
2004-01-28 17:31:37 +01:00
return 1 ;
}
2007-03-29 22:14:55 +02:00
int
gamemenuloop ( int game )
2004-01-28 17:31:37 +01:00
{
int userchoice = 0 ;
2007-03-29 22:14:55 +02:00
while ( 1 )
2004-01-28 17:31:37 +01:00
{
2007-03-29 22:14:55 +02:00
drawgamemenu ( game ) ;
2004-01-28 17:31:37 +01:00
userchoice = getch ( ) ;
switch ( tolower ( userchoice ) )
{
2007-03-29 22:14:55 +02:00
default :
break ;
case ' p ' :
if ( loggedin )
return 1 ;
break ;
2004-01-28 17:31:37 +01:00
case ' c ' :
if ( loggedin )
2004-02-13 23:30:59 +01:00
changepw ( 1 ) ;
2004-01-28 17:31:37 +01:00
break ;
2004-02-05 20:38:17 +01:00
case ' e ' :
if ( loggedin )
change_email ( ) ;
break ;
2004-01-28 17:31:37 +01:00
case ' w ' :
2007-03-29 22:14:55 +02:00
inprogressmenu ( game ) ;
2004-01-28 17:31:37 +01:00
break ;
2007-03-29 22:14:55 +02:00
case ' o ' :
if ( loggedin & & myconfig [ game ] - > rcfile )
editoptions ( game ) ;
break ;
2004-01-28 17:31:37 +01:00
case ' q ' :
2007-03-29 22:14:55 +02:00
return 0 ;
/* endwin ();
graceful_exit ( 0 ) ; */
2004-01-28 17:31:37 +01:00
/* break; */
case ' r ' :
2007-04-09 18:42:14 +02:00
if ( ! loggedin & & globalconfig . allow_registration )
2004-01-28 17:31:37 +01:00
newuser ( ) ;
break ;
case ' l ' :
if ( ! loggedin ) /* not visible to loggedin */
2004-01-31 04:04:58 +01:00
loginprompt ( 0 ) ;
2004-01-28 17:31:37 +01:00
}
}
2007-03-29 22:14:55 +02:00
return 0 ;
2004-01-04 03:03:26 +01:00
}
2007-03-29 22:14:55 +02:00
int
menuloop ( void )
{
int userchoice = 0 ;
while ( 1 )
{
drawmenu ( ) ;
userchoice = getch ( ) ;
if ( ( num_games > = 1 ) & & loggedin & & ( userchoice > = ' 1 ' ) & & ( userchoice < = ( ' 1 ' + num_games ) ) ) {
int game = userchoice - ' 1 ' ;
if ( myconfig [ game ] & & myconfig [ game ] - > game_name ) {
if ( gamemenuloop ( game ) ) return game ;
}
} else {
switch ( tolower ( userchoice ) )
{
default :
break ;
case ' c ' :
if ( loggedin )
changepw ( 1 ) ;
break ;
case ' e ' :
if ( loggedin )
change_email ( ) ;
break ;
case ' w ' :
inprogressmenu ( - 1 ) ;
break ;
case ' o ' :
if ( loggedin & & ( num_games = = 0 ) & & myconfig [ 0 ] - > rcfile )
editoptions ( 0 ) ;
break ;
case ' p ' :
if ( loggedin & & ( num_games = = 0 ) )
return 0 ;
break ;
case ' q ' :
endwin ( ) ;
graceful_exit ( 0 ) ;
/* break; */
case ' r ' :
2008-01-01 13:05:08 +01:00
if ( ! loggedin & & globalconfig . allow_registration )
2007-03-29 22:14:55 +02:00
newuser ( ) ;
break ;
case ' l ' :
if ( ! loggedin ) /* not visible to loggedin */
loginprompt ( 0 ) ;
}
}
}
return - 1 ;
}
2004-04-21 18:22:55 +02:00
int
authenticate ( )
{
int i , len , me_index ;
char user_buf [ 22 ] , pw_buf [ 22 ] ;
struct dg_game * * games ;
/* We use simple password authentication, rather than challenge/response. */
printf ( " \n " ) ;
fflush ( stdout ) ;
fgets ( user_buf , sizeof ( user_buf ) , stdin ) ;
len = strlen ( user_buf ) ;
if ( user_buf [ len - 1 ] = = ' \n ' )
user_buf [ - - len ] = ' \0 ' ;
else
{
fprintf ( stderr , " Username too long (max 20 chars). \n " ) ;
return 1 ;
}
fgets ( pw_buf , sizeof ( pw_buf ) , stdin ) ;
len = strlen ( pw_buf ) ;
if ( pw_buf [ len - 1 ] = = ' \n ' )
pw_buf [ - - len ] = ' \0 ' ;
else
{
fprintf ( stderr , " Password too long (max 20 chars). \n " ) ;
return 1 ;
}
2007-12-27 20:52:48 +01:00
{
struct dg_user * tmpme ;
if ( ( tmpme = userexist ( user_buf , 0 ) ) ) {
me = cpy_me ( tmpme ) ;
if ( passwordgood ( pw_buf ) )
{
games = populate_games ( - 1 , & len ) ;
for ( i = 0 ; i < len ; i + + )
if ( ! strcmp ( games [ i ] - > name , user_buf ) )
{
fprintf ( stderr , " Game already in progress. \n " ) ;
return 1 ;
}
win . ws_row = win . ws_col = 0 ;
gen_inprogress_lock ( 0 , getppid ( ) , gen_nhext_filename ( ) ) ;
return 0 ;
}
}
}
2004-04-21 18:22:55 +02:00
sleep ( 2 ) ;
fprintf ( stderr , " Login failed. \n " ) ;
return 1 ;
}
2003-12-30 22:30:32 +01:00
int
2004-01-05 02:57:25 +01:00
main ( int argc , char * * argv )
2003-12-30 22:30:32 +01:00
{
2003-12-31 20:47:49 +01:00
/* for chroot and program execution */
2004-09-12 10:58:32 +02:00
char atrcfilename [ 81 ] , * spool , * p , * auth = NULL ;
2003-12-31 20:47:49 +01:00
unsigned int len ;
2007-03-31 18:21:58 +02:00
int c , i ;
2004-04-21 18:22:55 +02:00
int nhext = 0 , nhauth = 0 ;
2007-03-29 22:14:55 +02:00
int userchoice ;
2007-04-01 14:09:42 +02:00
char * tmp ;
2003-12-31 20:47:49 +01:00
2004-07-03 21:19:28 +02:00
# ifndef HAVE_SETPROCTITLE
/* save argc, argv */
char * * saved_argv ;
2007-03-31 18:21:58 +02:00
int saved_argc ;
2004-07-03 21:19:28 +02:00
saved_argc = argc ;
saved_argv = malloc ( sizeof ( char * * ) * ( argc + 1 ) ) ;
for ( i = 0 ; i < argc ; i + + )
saved_argv [ i ] = strdup ( argv [ i ] ) ;
saved_argv [ i ] = ' \0 ' ;
compat_init_setproctitle ( argc , argv ) ;
argv = saved_argv ;
# endif
2004-09-12 10:58:32 +02:00
p = getenv ( " USER " ) ;
if ( p & & * p ! = ' \0 ' )
auth = strdup ( p ) ;
/* else auth is still NULL */
2004-07-03 21:19:28 +02:00
__progname = basename ( strdup ( argv [ 0 ] ) ) ;
2004-06-22 18:36:09 +02:00
while ( ( c = getopt ( argc , argv , " qh:pf:ae " ) ) ! = - 1 )
2004-01-31 07:23:02 +01:00
{
2004-02-01 21:53:08 +01:00
switch ( c )
2004-02-01 21:46:08 +01:00
{
2004-02-01 21:53:08 +01:00
case ' q ' :
silent = 1 ; break ;
2004-04-21 18:22:55 +02:00
case ' a ' :
nhauth = 1 ; break ;
case ' e ' :
nhext = 1 ; break ;
2004-02-02 23:57:54 +01:00
case ' f ' :
if ( config )
{
if ( ! silent )
fprintf ( stderr , " warning: using %s \n " , argv [ optind ] ) ;
free ( config ) ;
}
config = strdup ( optarg ) ;
break ;
2004-02-01 21:53:08 +01:00
default :
break ; /*ignore */
2004-02-01 21:46:08 +01:00
}
2004-02-01 21:53:08 +01:00
}
2004-06-22 18:36:09 +02:00
while ( optind < argc )
{
size_t len = strlen ( argv [ optind ] ) ;
memset ( argv [ optind + + ] , 0 , len ) ;
}
2004-06-28 21:41:06 +02:00
setproctitle ( " (not logged in) " ) ;
2004-06-22 18:36:09 +02:00
2004-01-05 02:55:08 +01:00
create_config ( ) ;
2003-12-31 20:47:49 +01:00
/* signal handlers */
signal ( SIGHUP , catch_sighup ) ;
2004-02-20 18:57:29 +01:00
( void ) tcgetattr ( 0 , & tt ) ;
2004-03-04 14:30:12 +01:00
if ( - 1 = = ioctl ( 0 , TIOCGWINSZ , ( char * ) & win ) | | win . ws_row < 4 | |
win . ws_col < 4 ) /* Rudimentary validity check */
{
win . ws_row = 24 ;
win . ws_col = 80 ;
win . ws_xpixel = win . ws_col * 8 ;
win . ws_ypixel = win . ws_row * 8 ;
}
2003-12-31 20:47:49 +01:00
2004-02-20 18:57:29 +01:00
/* get master tty just before chroot (lives in /dev) */
2004-04-21 18:22:55 +02:00
if ( ! nhext & & ! nhauth )
ttyrec_getpty ( ) ;
2003-12-31 20:47:49 +01:00
2007-03-29 22:14:55 +02:00
if ( geteuid ( ) ! = globalconfig . shed_uid )
2003-12-31 20:47:49 +01:00
{
2004-04-21 18:22:55 +02:00
/* chroot */
2007-03-29 22:14:55 +02:00
if ( chroot ( globalconfig . chroot ) )
2004-04-21 18:22:55 +02:00
{
perror ( " cannot change root directory " ) ;
graceful_exit ( 1 ) ;
}
2003-12-31 20:47:49 +01:00
2004-04-21 18:22:55 +02:00
if ( chdir ( " / " ) )
{
perror ( " cannot chdir to root directory " ) ;
graceful_exit ( 1 ) ;
}
2003-12-31 20:47:49 +01:00
2004-04-21 18:22:55 +02:00
/* shed privs. this is done immediately after chroot. */
2007-03-29 22:14:55 +02:00
if ( setgroups ( 1 , & globalconfig . shed_gid ) = = - 1 )
2004-04-21 18:22:55 +02:00
{
perror ( " setgroups " ) ;
graceful_exit ( 1 ) ;
}
2004-01-02 21:39:02 +01:00
2007-03-29 22:14:55 +02:00
if ( setgid ( globalconfig . shed_gid ) = = - 1 )
2004-04-21 18:22:55 +02:00
{
perror ( " setgid " ) ;
graceful_exit ( 1 ) ;
}
2007-03-29 22:14:55 +02:00
if ( setuid ( globalconfig . shed_uid ) = = - 1 )
2004-04-21 18:22:55 +02:00
{
perror ( " setuid " ) ;
graceful_exit ( 1 ) ;
}
2004-01-03 04:51:29 +01:00
}
2004-01-02 21:39:02 +01:00
2008-01-01 18:03:05 +01:00
dgl_exec_cmdqueue ( globalconfig . cmdqueue [ DGLTIME_DGLSTART ] , 0 , NULL ) ;
2004-04-21 18:22:55 +02:00
if ( nhext )
2004-01-03 04:51:29 +01:00
{
2004-04-21 18:22:55 +02:00
char * myargv [ 3 ] ;
2007-03-29 22:14:55 +02:00
myargv [ 0 ] = myconfig [ 0 ] - > game_path ;
2004-04-21 18:22:55 +02:00
myargv [ 1 ] = " --proxy " ;
myargv [ 2 ] = 0 ;
2007-03-29 22:14:55 +02:00
execvp ( myconfig [ 0 ] - > game_path , myargv ) ;
perror ( myconfig [ 0 ] - > game_path ) ;
2004-01-04 02:45:21 +01:00
graceful_exit ( 1 ) ;
2004-01-03 04:51:29 +01:00
}
2003-12-31 20:47:49 +01:00
/* simple login routine, uses ncurses */
if ( readfile ( 0 ) )
2004-01-04 02:45:21 +01:00
graceful_exit ( 110 ) ;
2003-12-31 20:47:49 +01:00
2004-04-21 18:22:55 +02:00
if ( nhauth )
graceful_exit ( authenticate ( ) ) ;
2004-06-12 02:11:18 +02:00
if ( auth )
{
char * user , * pass , * p ;
p = strchr ( auth , ' : ' ) ;
if ( p )
{
pass = p + 1 ;
2004-06-13 05:05:47 +02:00
if ( * pass ! = ' \0 ' )
2004-06-12 02:11:18 +02:00
{
* p = ' \0 ' ;
user = auth ;
autologin ( user , pass ) ;
}
}
}
2004-01-26 09:29:04 +01:00
initcurses ( ) ;
2007-03-29 22:14:55 +02:00
userchoice = menuloop ( ) ;
2003-12-31 20:47:49 +01:00
assert ( loggedin ) ;
2007-03-29 22:14:55 +02:00
if ( ( userchoice > = 0 ) & & ( userchoice < = num_games ) ) {
while ( ! purge_stale_locks ( userchoice ) ) {
userchoice = gamemenuloop ( userchoice ) ;
}
if ( ! ( ( userchoice > = 0 ) & & ( userchoice < = num_games ) ) )
graceful_exit ( 1 ) ;
} else {
graceful_exit ( 1 ) ;
}
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 ) ) ;
}
2004-01-04 03:31:00 +01:00
2004-06-28 21:41:06 +02:00
setproctitle ( " %s [playing] " , me - > username ) ;
2004-01-04 03:20:54 +01:00
endwin ( ) ;
2004-01-06 15:27:56 +01:00
signal ( SIGWINCH , SIG_DFL ) ;
2004-01-04 03:19:12 +01:00
2008-01-01 18:03:05 +01:00
/* 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 ) ;
2008-01-01 18:36:46 +01:00
/*
2007-03-29 22:14:55 +02:00
if ( ! backup_savefile ( userchoice ) )
2004-03-17 15:04:31 +01:00
graceful_exit ( 5 ) ;
2008-01-01 18:36:46 +01:00
*/
2004-03-17 15:04:31 +01:00
2003-12-31 20:47:49 +01:00
/* environment */
2007-03-29 22:14:55 +02:00
if ( myconfig [ userchoice ] - > rcfile ) {
snprintf ( atrcfilename , 81 , " @%s " , dgl_format_str ( userchoice , me , myconfig [ userchoice ] - > rc_fmt ) ) ;
mysetenv ( " NETHACKOPTIONS " , atrcfilename , 1 ) ;
}
2003-12-31 20:47:49 +01:00
2007-03-29 22:14:55 +02:00
len = strlen ( myconfig [ userchoice ] - > spool ) + strlen ( me - > username ) + 1 ;
2003-12-31 20:47:49 +01:00
spool = malloc ( len + 1 ) ;
2007-03-29 22:14:55 +02:00
snprintf ( spool , len + 1 , " %s/%s " , myconfig [ userchoice ] - > spool , me - > username ) ;
2003-12-31 20:47:49 +01:00
2004-01-16 03:00:47 +01:00
mysetenv ( " MAIL " , spool , 1 ) ;
mysetenv ( " SIMPLEMAIL " , " 1 " , 1 ) ;
2003-12-31 20:47:49 +01:00
/* don't let the mail file grow */
if ( access ( spool , F_OK ) = = 0 )
unlink ( spool ) ;
free ( spool ) ;
2007-03-29 22:14:55 +02:00
/* fix the variables in the arguments */
for ( i = 0 ; i < myconfig [ userchoice ] - > num_args ; i + + ) {
2007-04-01 14:09:42 +02:00
tmp = strdup ( dgl_format_str ( userchoice , me , myconfig [ userchoice ] - > bin_args [ i ] ) ) ;
2007-03-29 22:14:55 +02:00
free ( myconfig [ userchoice ] - > bin_args [ i ] ) ;
myconfig [ userchoice ] - > bin_args [ i ] = tmp ;
}
2007-04-07 01:13:06 +02:00
if ( myconfig [ userchoice ] - > mkdir ) {
tmp = strdup ( dgl_format_str ( userchoice , me , myconfig [ userchoice ] - > mkdir ) ) ;
free ( myconfig [ userchoice ] - > mkdir ) ;
myconfig [ userchoice ] - > mkdir = tmp ;
}
2007-04-01 14:09:42 +02:00
2007-04-07 01:13:06 +02:00
if ( myconfig [ userchoice ] - > chdir ) {
tmp = strdup ( dgl_format_str ( userchoice , me , myconfig [ userchoice ] - > chdir ) ) ;
free ( myconfig [ userchoice ] - > chdir ) ;
myconfig [ userchoice ] - > chdir = tmp ;
}
2007-04-01 14:09:42 +02:00
2003-12-31 20:47:49 +01:00
/* launch program */
2007-03-29 22:14:55 +02:00
ttyrec_main ( userchoice , me - > username , gen_ttyrec_filename ( ) ) ;
2003-12-31 20:47:49 +01:00
/* NOW we can safely kill this */
freefile ( ) ;
2007-03-29 22:14:55 +02:00
/*
printf ( " config:'%s' \n " , config ) ;
printf ( " chroot:'%s' \n " , globalconfig . chroot ) ;
printf ( " gamepath:'%s' \n " , myconfig [ userchoice ] - > game_path ) ;
printf ( " game:'%s' \n " , myconfig [ userchoice ] - > game_name ) ;
printf ( " dglroot:'%s' \n " , globalconfig . dglroot ) ;
2007-03-31 23:29:45 +02:00
printf ( " lockfile:'%s' \n " , globalconfig . lockfile ) ;
2007-03-31 23:18:32 +02:00
printf ( " passwd:'%s' \n " , globalconfig . passwd ) ;
2007-03-29 22:14:55 +02:00
printf ( " banner:'%s' \n " , globalconfig . banner ) ;
printf ( " rcfile:'%s' \n " , myconfig [ userchoice ] - > rcfile ) ;
printf ( " spool:'%s' \n " , myconfig [ userchoice ] - > spool ) ;
printf ( " savefilefmt:'%s' \n " , myconfig [ userchoice ] - > savefilefmt ) ;
printf ( " dgl_format_str(savefilefmt):'%s' \n " , dgl_format_str ( userchoice , myconfig [ userchoice ] - > savefilefmt ) ) ;
printf ( " inprogressdir:'%s' \n " , myconfig [ userchoice ] - > inprogressdir ) ;
*/
2003-12-31 20:47:49 +01:00
if ( me )
free ( me ) ;
2008-01-01 18:03:05 +01:00
/* FIXME: free data in globalconfig */
2004-01-04 02:45:21 +01:00
graceful_exit ( 1 ) ;
2007-03-29 22:14:55 +02:00
2003-12-31 20:47:49 +01:00
return 1 ;
2003-12-30 22:30:32 +01:00
}