2001-03-18 13:24:57 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2001-03-18 13:24:57 +00:00
Samba readline wrapper implementation
2001-03-18 23:41:53 +00:00
Copyright ( C ) Simo Sorce 2001
2001-03-18 13:24:57 +00:00
Copyright ( C ) Andrew Tridgell 2001
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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2001-03-18 13:24:57 +00:00
( 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
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2001-03-18 13:24:57 +00:00
*/
# include "includes.h"
2002-09-25 15:19:00 +00:00
# ifdef HAVE_LIBREADLINE
# ifdef HAVE_READLINE_READLINE_H
# include <readline / readline.h>
# ifdef HAVE_READLINE_HISTORY_H
# include <readline / history.h>
# endif
# else
# ifdef HAVE_READLINE_H
# include <readline.h>
# ifdef HAVE_HISTORY_H
# include <history.h>
# endif
# else
# undef HAVE_LIBREADLINE
# endif
# endif
# endif
2001-11-19 04:18:45 +00:00
# ifdef HAVE_NEW_LIBREADLINE
# define RL_COMPLETION_CAST (rl_completion_func_t *)
# else
/* This type is missing from libreadline<4.0 (approximately) */
2001-11-19 07:39:35 +00:00
# define RL_COMPLETION_CAST
2001-11-19 04:18:45 +00:00
# endif /* HAVE_NEW_LIBREADLINE */
2008-10-03 14:58:41 -07:00
static bool smb_rl_done ;
2008-10-05 13:12:31 +02:00
# if HAVE_LIBREADLINE
/*
* MacOS / X does not have rl_done in readline . h , but
* readline . so has it
*/
extern int rl_done ;
# endif
2008-10-03 14:58:41 -07:00
void smb_readline_done ( void )
{
smb_rl_done = true ;
# if HAVE_LIBREADLINE
rl_done = 1 ;
# endif
}
2001-03-18 13:24:57 +00:00
/****************************************************************************
2002-01-19 22:54:13 +00:00
Display the prompt and wait for input . Call callback ( ) regularly
2001-03-18 13:24:57 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-01-19 22:54:13 +00:00
2006-02-03 22:19:41 +00:00
static char * smb_readline_replacement ( const char * prompt , void ( * callback ) ( void ) ,
2003-08-11 19:11:43 +00:00
char * * ( completion_fn ) ( const char * text , int start , int end ) )
2001-03-18 13:24:57 +00:00
{
2001-03-18 23:41:53 +00:00
fd_set fds ;
2007-12-06 17:16:33 -08:00
char * line = NULL ;
2001-03-18 23:41:53 +00:00
struct timeval timeout ;
2003-02-08 00:08:23 +00:00
int fd = x_fileno ( x_stdin ) ;
2002-01-19 22:54:13 +00:00
char * ret ;
2001-03-18 23:41:53 +00:00
2006-06-15 23:51:20 +00:00
/* Prompt might be NULL in non-interactive mode. */
if ( prompt ) {
x_fprintf ( x_stdout , " %s " , prompt ) ;
x_fflush ( x_stdout ) ;
}
2001-03-18 13:24:57 +00:00
2007-12-06 17:16:33 -08:00
line = ( char * ) SMB_MALLOC ( BUFSIZ ) ;
if ( ! line ) {
return NULL ;
2007-11-15 14:19:52 -08:00
}
2008-10-03 14:58:41 -07:00
while ( ! smb_rl_done ) {
2001-03-18 23:41:53 +00:00
timeout . tv_sec = 5 ;
timeout . tv_usec = 0 ;
2001-03-18 13:24:57 +00:00
2001-03-18 23:41:53 +00:00
FD_ZERO ( & fds ) ;
FD_SET ( fd , & fds ) ;
2007-11-15 14:19:52 -08:00
2002-01-31 23:26:12 +00:00
if ( sys_select_intr ( fd + 1 , & fds , NULL , NULL , & timeout ) = = 1 ) {
2007-11-15 14:19:52 -08:00
ret = x_fgets ( line , BUFSIZ , x_stdin ) ;
2007-12-06 17:16:33 -08:00
if ( ret = = 0 ) {
SAFE_FREE ( line ) ;
}
2001-03-18 23:41:53 +00:00
return ret ;
}
2007-12-06 17:16:33 -08:00
if ( callback ) {
2002-01-19 22:54:13 +00:00
callback ( ) ;
2007-12-06 17:16:33 -08:00
}
2001-03-18 23:41:53 +00:00
}
2008-10-29 15:55:45 -07:00
SAFE_FREE ( line ) ;
2008-10-03 14:58:41 -07:00
return NULL ;
2001-07-20 07:46:39 +00:00
}
/****************************************************************************
2002-01-19 22:54:13 +00:00
Display the prompt and wait for input . Call callback ( ) regularly .
2001-07-20 07:46:39 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-01-19 22:54:13 +00:00
2007-12-06 17:16:33 -08:00
char * smb_readline ( const char * prompt , void ( * callback ) ( void ) ,
2003-08-11 19:11:43 +00:00
char * * ( completion_fn ) ( const char * text , int start , int end ) )
2001-07-20 07:46:39 +00:00
{
2006-06-15 23:51:20 +00:00
char * ret ;
2007-10-18 17:40:25 -07:00
bool interactive ;
2006-06-15 23:51:20 +00:00
interactive = isatty ( x_fileno ( x_stdin ) ) | | getenv ( " CLI_FORCE_INTERACTIVE " ) ;
if ( ! interactive ) {
2007-12-06 17:16:33 -08:00
return smb_readline_replacement ( NULL , callback , completion_fn ) ;
2006-06-15 23:51:20 +00:00
}
2001-07-20 07:46:39 +00:00
# if HAVE_LIBREADLINE
2002-01-19 22:54:13 +00:00
2006-06-15 23:51:20 +00:00
/* Aargh! Readline does bizzare things with the terminal width
that mucks up expect ( 1 ) . Set CLI_NO_READLINE in the environment
to force readline not to be used . */
if ( getenv ( " CLI_NO_READLINE " ) )
return smb_readline_replacement ( prompt , callback , completion_fn ) ;
if ( completion_fn ) {
/* The callback prototype has changed slightly between
different versions of Readline , so the same function
works in all of them to date , but we get compiler
warnings in some . */
rl_attempted_completion_function = RL_COMPLETION_CAST completion_fn ;
}
2006-10-11 13:59:03 +00:00
# if HAVE_DECL_RL_EVENT_HOOK
2006-06-15 23:51:20 +00:00
if ( callback )
rl_event_hook = ( Function * ) callback ;
2006-10-11 13:59:03 +00:00
# endif
2006-06-15 23:51:20 +00:00
ret = readline ( prompt ) ;
if ( ret & & * ret )
add_history ( ret ) ;
# else
ret = smb_readline_replacement ( prompt , callback , completion_fn ) ;
2001-03-18 13:24:57 +00:00
# endif
2006-06-15 23:51:20 +00:00
return ret ;
2001-03-18 13:24:57 +00:00
}
2003-05-12 18:12:31 +00:00
/****************************************************************************
* return line buffer text
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
const char * smb_readline_get_line_buffer ( void )
{
# if defined(HAVE_LIBREADLINE)
return rl_line_buffer ;
# else
return NULL ;
# endif
}
/****************************************************************************
* set completion append character
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void smb_readline_ca_char ( char c )
{
# if defined(HAVE_LIBREADLINE)
rl_completion_append_character = c ;
# endif
}
2001-03-18 13:24:57 +00:00
/****************************************************************************
history
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-09 19:12:18 +00:00
int cmd_history ( void )
2001-03-18 13:24:57 +00:00
{
2006-10-11 13:59:03 +00:00
# if defined(HAVE_LIBREADLINE) && defined(HAVE_HISTORY_LIST)
2001-03-18 13:24:57 +00:00
HIST_ENTRY * * hlist ;
int i ;
2001-03-18 23:41:53 +00:00
hlist = history_list ( ) ;
2007-12-06 17:16:33 -08:00
2001-03-18 13:24:57 +00:00
for ( i = 0 ; hlist & & hlist [ i ] ; i + + ) {
DEBUG ( 0 , ( " %d: %s \n " , i , hlist [ i ] - > line ) ) ;
}
# else
DEBUG ( 0 , ( " no history without readline support \n " ) ) ;
# endif
2001-10-09 19:12:18 +00:00
return 0 ;
2001-03-18 13:24:57 +00:00
}