2014-08-25 19:38:12 +02:00
/*
* Copyright ( c ) 2008 Kungliga Tekniska Högskolan
* ( Royal Institute of Technology , Stockholm , Sweden ) .
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
*
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
*
* 3. Neither the name of the Institute nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ` ` AS IS ' ' AND
* ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED . IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
* DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT
* LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE .
*/
2014-10-29 11:48:59 +01:00
# include "replace.h"
# include "system/filesys.h"
# include "system/wait.h"
2020-01-15 12:40:38 +01:00
# include "lib/util/sys_rw.h"
2014-08-25 19:38:12 +02:00
# ifdef HAVE_PTY_H
# include <pty.h>
# endif
# ifdef HAVE_UTIL_H
# include <util.h>
# endif
2014-12-10 12:23:04 +00:00
# ifdef HAVE_BSD_LIBUTIL_H
# include <bsd/libutil.h>
# elif defined HAVE_LIBUTIL_H
2014-08-25 19:38:12 +02:00
# include <libutil.h>
# endif
# ifdef STREAMSPTY
# include <stropts.h>
# endif /* STREAMPTY */
# include <popt.h>
2015-02-13 13:47:14 -08:00
# ifdef HAVE_ERR_H
2014-08-25 19:38:12 +02:00
# include <err.h>
2015-02-13 13:47:14 -08:00
# else
2015-03-12 15:07:38 +00:00
const char progname [ ] = " unknown program " ;
2021-02-01 18:30:15 +01:00
static void err ( int eval , const char * fmt , . . . ) PRINTF_ATTRIBUTE ( 2 , 0 ) ;
static void errx ( int eval , const char * fmt , . . . ) PRINTF_ATTRIBUTE ( 2 , 0 ) ;
2015-03-12 15:07:38 +00:00
static void err ( int eval , const char * fmt , . . . )
{
int err_errno = errno ;
va_list ap ;
fprintf ( stderr , " %s: " , progname ) ;
va_start ( ap , fmt ) ;
vfprintf ( stderr , fmt , ap ) ;
va_end ( ap ) ;
fprintf ( stderr , " : %s \n " , strerror ( err_errno ) ) ;
exit ( eval ) ;
}
static void errx ( int eval , const char * fmt , . . . )
{
va_list ap ;
fprintf ( stderr , " %s: " , progname ) ;
va_start ( ap , fmt ) ;
vfprintf ( stderr , fmt , ap ) ;
va_end ( ap ) ;
fprintf ( stderr , " \n " ) ;
exit ( eval ) ;
}
2015-02-13 13:47:14 -08:00
# endif
2014-08-25 19:38:12 +02:00
struct command {
enum { CMD_EXPECT = 0 , CMD_SEND , CMD_PASSWORD } type ;
unsigned int lineno ;
char * str ;
struct command * next ;
} ;
/*
*
*/
static struct command * commands , * * next = & commands ;
static sig_atomic_t alarmset = 0 ;
static int opt_timeout = 10 ;
static int opt_verbose ;
static int master ;
static int slave ;
static char line [ 256 ] = { 0 } ;
static void caught_signal ( int signo )
{
alarmset = signo ;
}
static void open_pty ( void )
{
# ifdef _AIX
printf ( " implement open_pty \n " ) ;
exit ( 77 ) ;
# endif
# if defined(HAVE_OPENPTY) || defined(__linux) || defined(__osf__) /* XXX */
if ( openpty ( & master , & slave , line , 0 , 0 ) = = 0 )
return ;
# endif /* HAVE_OPENPTY .... */
# ifdef STREAMSPTY
{
char * clone [ ] = {
" /dev/ptc " ,
" /dev/ptmx " ,
" /dev/ptm " ,
" /dev/ptym/clone " ,
NULL
} ;
char * * q ;
for ( q = clone ; * q ; q + + ) {
master = open ( * q , O_RDWR ) ;
if ( master > = 0 ) {
# ifdef HAVE_GRANTPT
grantpt ( master ) ;
# endif
# ifdef HAVE_UNLOCKPT
unlockpt ( master ) ;
# endif
strlcpy ( line , ptsname ( master ) , sizeof ( line ) ) ;
slave = open ( line , O_RDWR ) ;
if ( slave < 0 )
errx ( 1 , " failed to open slave when using %s " , * q ) ;
ioctl ( slave , I_PUSH , " ptem " ) ;
ioctl ( slave , I_PUSH , " ldterm " ) ;
return ;
}
}
}
# endif /* STREAMSPTY */
/* more cases, like open /dev/ptmx, etc */
exit ( 77 ) ;
}
/*
*
*/
static char * iscmd ( const char * buf , const char * s )
{
size_t len = strlen ( s ) ;
if ( strncmp ( buf , s , len ) ! = 0 ) {
return NULL ;
}
return strdup ( buf + len ) ;
}
static void parse_configuration ( const char * fn )
{
struct command * c ;
char s [ 1024 ] ;
char * str ;
unsigned int lineno = 0 ;
FILE * cmd ;
cmd = fopen ( fn , " r " ) ;
if ( cmd = = NULL )
err ( 1 , " open: %s " , fn ) ;
while ( fgets ( s , sizeof ( s ) , cmd ) ! = NULL ) {
s [ strcspn ( s , " # \n " ) ] = ' \0 ' ;
lineno + + ;
c = calloc ( 1 , sizeof ( * c ) ) ;
if ( c = = NULL )
errx ( 1 , " malloc " ) ;
c - > lineno = lineno ;
( * next ) = c ;
next = & ( c - > next ) ;
if ( ( str = iscmd ( s , " expect " ) ) ! = NULL ) {
c - > type = CMD_EXPECT ;
c - > str = str ;
} else if ( ( str = iscmd ( s , " send " ) ) ! = NULL ) {
c - > type = CMD_SEND ;
c - > str = str ;
} else if ( ( str = iscmd ( s , " password " ) ) ! = NULL ) {
c - > type = CMD_PASSWORD ;
c - > str = str ;
} else
errx ( 1 , " Invalid command on line %d: %s " , lineno , s ) ;
}
fclose ( cmd ) ;
}
/*
*
*/
static int eval_parent ( pid_t pid )
{
struct command * c ;
char in ;
size_t len = 0 ;
ssize_t sret ;
for ( c = commands ; c ! = NULL ; c = c - > next ) {
switch ( c - > type ) {
case CMD_EXPECT :
if ( opt_verbose ) {
printf ( " [expecting %s] \n " , c - > str ) ;
}
len = 0 ;
alarm ( opt_timeout ) ;
while ( ( sret = read ( master , & in , sizeof ( in ) ) ) > 0 ) {
alarm ( opt_timeout ) ;
printf ( " %c " , in ) ;
if ( c - > str [ len ] ! = in ) {
len = 0 ;
continue ;
}
len + + ;
if ( c - > str [ len ] = = ' \0 ' ) {
break ;
}
}
alarm ( 0 ) ;
if ( alarmset = = SIGALRM ) {
errx ( 1 , " timeout waiting for %s (line %u) " ,
c - > str , c - > lineno ) ;
} else if ( alarmset ) {
errx ( 1 , " got a signal %d waiting for %s (line %u) " ,
( int ) alarmset , c - > str , c - > lineno ) ;
}
if ( sret < = 0 ) {
errx ( 1 , " end command while waiting for %s (line %u) " ,
c - > str , c - > lineno ) ;
}
break ;
case CMD_SEND :
case CMD_PASSWORD : {
size_t i = 0 ;
const char * msg = ( c - > type = = CMD_PASSWORD ) ? " **** " : c - > str ;
if ( opt_verbose ) {
printf ( " [send %s] \n " , msg ) ;
}
len = strlen ( c - > str ) ;
while ( i < len ) {
if ( c - > str [ i ] = = ' \\ ' & & i < len - 1 ) {
char ctrl ;
i + + ;
switch ( c - > str [ i ] ) {
case ' n ' :
ctrl = ' \n ' ;
break ;
case ' r ' :
ctrl = ' \r ' ;
break ;
case ' t ' :
ctrl = ' \t ' ;
break ;
default :
errx ( 1 ,
" unknown control char %c (line %u) " ,
c - > str [ i ] ,
c - > lineno ) ;
}
if ( sys_write ( master , & ctrl , 1 ) ! = 1 ) {
errx ( 1 , " command refused input (line %u) " , c - > lineno ) ;
}
} else {
if ( sys_write ( master , & c - > str [ i ] , 1 ) ! = 1 ) {
errx ( 1 , " command refused input (line %u) " , c - > lineno ) ;
}
}
i + + ;
}
break ;
}
default :
abort ( ) ;
}
}
while ( read ( master , & in , sizeof ( in ) ) > 0 ) {
printf ( " %c " , in ) ;
}
if ( opt_verbose ) {
printf ( " [end of program] \n " ) ;
}
/*
* Fetch status from child
*/
{
int ret , status ;
ret = waitpid ( pid , & status , 0 ) ;
if ( ret = = - 1 ) {
err ( 1 , " waitpid " ) ;
}
if ( WIFEXITED ( status ) & & WEXITSTATUS ( status ) ) {
return WEXITSTATUS ( status ) ;
} else if ( WIFSIGNALED ( status ) ) {
printf ( " killed by signal: %d \n " , WTERMSIG ( status ) ) ;
return 1 ;
}
}
return 0 ;
}
/*
*
*/
struct poptOption long_options [ ] = {
POPT_AUTOHELP
2018-12-13 11:23:00 +01:00
{
. longName = " timeout " ,
. shortName = ' t ' ,
. argInfo = POPT_ARG_INT ,
. arg = & opt_timeout ,
. val = ' t ' ,
} ,
{
. longName = " verbose " ,
. shortName = ' v ' ,
. argInfo = POPT_ARG_NONE ,
. arg = & opt_verbose ,
. val = ' v ' ,
} ,
2014-08-25 19:38:12 +02:00
POPT_TABLEEND
} ;
int main ( int argc , const char * * argv )
{
int optidx = 0 ;
pid_t pid ;
2019-08-19 13:09:25 +02:00
poptContext pc = NULL ;
2014-08-25 19:38:12 +02:00
const char * instruction_file ;
2014-09-26 09:06:59 +02:00
const char * * args ;
2014-08-25 19:38:12 +02:00
const char * program ;
2014-09-26 09:06:59 +02:00
char * const * program_args ;
2014-08-25 19:38:12 +02:00
pc = poptGetContext ( " texpect " ,
argc ,
argv ,
long_options ,
POPT_CONTEXT_POSIXMEHARDER ) ;
if ( argc = = 1 ) {
poptPrintHelp ( pc , stderr , 0 ) ;
2019-08-19 13:09:25 +02:00
goto out ;
2014-08-25 19:38:12 +02:00
}
while ( ( optidx = poptGetNextOpt ( pc ) ) ! = - 1 ) {
2021-09-09 18:14:36 +02:00
switch ( optidx ) {
case POPT_ERROR_BADOPT :
fprintf ( stderr , " \n Invalid option %s: %s \n \n " ,
poptBadOption ( pc , 0 ) , poptStrerror ( optidx ) ) ;
poptPrintUsage ( pc , stderr , 0 ) ;
exit ( 1 ) ;
}
2014-08-25 19:38:12 +02:00
}
instruction_file = poptGetArg ( pc ) ;
2014-09-26 09:06:59 +02:00
args = poptGetArgs ( pc ) ;
2019-05-03 16:15:30 +12:00
if ( args = = NULL ) {
poptPrintHelp ( pc , stderr , 0 ) ;
2019-08-19 13:09:25 +02:00
goto out ;
2019-05-03 16:15:30 +12:00
}
2014-09-26 09:06:59 +02:00
program_args = ( char * const * ) discard_const_p ( char * , args ) ;
2014-08-25 19:38:12 +02:00
program = program_args [ 0 ] ;
if ( opt_verbose ) {
int i ;
printf ( " Using instruction_file: %s \n " , instruction_file ) ;
printf ( " Executing '%s' " , program ) ;
2019-05-03 16:15:30 +12:00
for ( i = 0 ; program_args [ i ] ! = NULL ; i + + ) {
2014-08-25 19:38:12 +02:00
printf ( " '%s' " , program_args [ i ] ) ;
}
printf ( " \n " ) ;
}
parse_configuration ( instruction_file ) ;
open_pty ( ) ;
pid = fork ( ) ;
switch ( pid ) {
case - 1 :
err ( 1 , " Failed to fork " ) ;
2017-07-26 16:51:39 +02:00
/* Never reached */
2019-08-19 13:09:25 +02:00
goto out ;
2014-08-25 19:38:12 +02:00
case 0 :
if ( setsid ( ) < 0 )
err ( 1 , " setsid " ) ;
dup2 ( slave , STDIN_FILENO ) ;
dup2 ( slave , STDOUT_FILENO ) ;
dup2 ( slave , STDERR_FILENO ) ;
2020-01-15 12:37:22 +01:00
closefrom ( STDERR_FILENO + 1 ) ;
2014-08-25 19:38:12 +02:00
/* texpect <expect_instructions> <progname> [<args>] */
execvp ( program , program_args ) ;
err ( 1 , " Failed to exec: %s " , program ) ;
2017-07-26 16:51:39 +02:00
/* Never reached */
2019-08-19 13:09:25 +02:00
goto out ;
2014-08-25 19:38:12 +02:00
default :
close ( slave ) ;
{
struct sigaction sa ;
sa . sa_handler = caught_signal ;
sa . sa_flags = 0 ;
sigemptyset ( & sa . sa_mask ) ;
sigaction ( SIGALRM , & sa , NULL ) ;
}
2019-08-19 13:09:25 +02:00
poptFreeContext ( pc ) ;
2014-08-25 19:38:12 +02:00
return eval_parent ( pid ) ;
}
2017-07-26 16:51:39 +02:00
/* Never reached */
2019-08-19 13:09:25 +02:00
out :
poptFreeContext ( pc ) ;
2017-07-26 16:51:39 +02:00
return 1 ;
2014-08-25 19:38:12 +02:00
}