| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* bashhist.c -- bash interface to the GNU history library. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | /* Copyright (C) 1993-2009 Free Software Foundation, Inc.
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |    This file is part of GNU Bash, the Bourne Again SHell. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    Bash 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 3 of the License, or | 
					
						
							|  |  |  |    (at your option) any later version. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    Bash 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. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |    along with Bash.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "config.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HISTORY)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_UNISTD_H)
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #  ifdef _MINIX
 | 
					
						
							|  |  |  | #    include <sys/types.h>
 | 
					
						
							|  |  |  | #  endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #  include <unistd.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "bashtypes.h"
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include "bashansi.h"
 | 
					
						
							|  |  |  | #include "posixstat.h"
 | 
					
						
							|  |  |  | #include "filecntl.h"
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #include "bashintl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (SYSLOG_HISTORY)
 | 
					
						
							|  |  |  | #  include <syslog.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "shell.h"
 | 
					
						
							|  |  |  | #include "flags.h"
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "input.h"
 | 
					
						
							|  |  |  | #include "parser.h"	/* for the struct dstack stuff. */
 | 
					
						
							|  |  |  | #include "pathexp.h"	/* for the struct ignorevar stuff */
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #include "bashhist.h"	/* matching prototypes and declarations */
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "builtins/common.h"
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include <readline/history.h>
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #include <glob/glob.h>
 | 
					
						
							|  |  |  | #include <glob/strmatch.h>
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (READLINE)
 | 
					
						
							|  |  |  | #  include "bashline.h"
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | extern int rl_done, rl_dispatching;	/* should really include readline.h */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (errno)
 | 
					
						
							|  |  |  | extern int errno; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static int histignore_item_func __P((struct ign *)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static int check_history_control __P((char *)); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static void hc_erasedups __P((char *)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static void really_add_history __P((char *)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct ignorevar histignore = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   "HISTIGNORE", | 
					
						
							|  |  |  |   (struct ign *)0, | 
					
						
							|  |  |  |   0, | 
					
						
							|  |  |  |   (char *)0, | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   (sh_iv_item_func_t *)histignore_item_func, | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define HIGN_EXPAND 0x01
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Declarations of bash history variables. */ | 
					
						
							|  |  |  | /* Non-zero means to remember lines typed to the shell on the history
 | 
					
						
							|  |  |  |    list.  This is different than the user-controlled behaviour; this | 
					
						
							|  |  |  |    becomes zero when we read lines from a file, for example. */ | 
					
						
							|  |  |  | int remember_on_history = 1; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | int enable_history_list = 1;	/* value for `set -o history' */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | /* The number of lines that Bash has added to this history session.  The
 | 
					
						
							|  |  |  |    difference between the number of the top element in the history list | 
					
						
							|  |  |  |    (offset from history_base) and the number of lines in the history file. | 
					
						
							|  |  |  |    Appending this session's history to the history file resets this to 0. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int history_lines_this_session; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* The number of lines that Bash has read from the history file. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int history_lines_in_file; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (BANG_HISTORY)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Non-zero means do no history expansion on this line, regardless
 | 
					
						
							|  |  |  |    of what history_expansion says. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int history_expansion_inhibited; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | /* With the old default, every line was saved in the history individually.
 | 
					
						
							|  |  |  |    I.e., if the user enters: | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	bash$ for i in a b c | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	> do | 
					
						
							|  |  |  | 	> echo $i | 
					
						
							|  |  |  | 	> done | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |    Each line will be individually saved in the history. | 
					
						
							|  |  |  | 	bash$ history | 
					
						
							|  |  |  | 	10  for i in a b c | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	11  do | 
					
						
							|  |  |  | 	12  echo $i | 
					
						
							|  |  |  | 	13  done | 
					
						
							|  |  |  | 	14  history | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |    If the variable command_oriented_history is set, multiple lines | 
					
						
							|  |  |  |    which form one command will be saved as one history entry. | 
					
						
							|  |  |  | 	bash$ for i in a b c | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	> do | 
					
						
							|  |  |  | 	> echo $i | 
					
						
							|  |  |  | 	> done | 
					
						
							|  |  |  | 	bash$ history | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	10  for i in a b c | 
					
						
							|  |  |  |     do | 
					
						
							|  |  |  |     echo $i | 
					
						
							|  |  |  |     done | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	11  history | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |    The user can then recall the whole command all at once instead | 
					
						
							|  |  |  |    of just being able to recall one line at a time. | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |    This is now enabled by default. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int command_oriented_history = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | /* Set to 1 if the first line of a possibly-multi-line command was saved
 | 
					
						
							|  |  |  |    in the history list.  Managed by maybe_add_history(), but global so | 
					
						
							|  |  |  |    the history-manipluating builtins can see it. */ | 
					
						
							|  |  |  | int current_command_first_line_saved = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Non-zero means to store newlines in the history list when using
 | 
					
						
							|  |  |  |    command_oriented_history rather than trying to use semicolons. */ | 
					
						
							|  |  |  | int literal_history; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Non-zero means to append the history to the history file at shell
 | 
					
						
							|  |  |  |    exit, even if the history has been stifled. */ | 
					
						
							|  |  |  | int force_append_history; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | /* A nit for picking at history saving.  Flags have the following values:
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    Value == 0 means save all lines parsed by the shell on the history. | 
					
						
							|  |  |  |    Value & HC_IGNSPACE means save all lines that do not start with a space. | 
					
						
							|  |  |  |    Value & HC_IGNDUPS means save all lines that do not match the last | 
					
						
							|  |  |  |    line saved. | 
					
						
							|  |  |  |    Value & HC_ERASEDUPS means to remove all other matching lines from the | 
					
						
							|  |  |  |    history list before saving the latest line. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int history_control; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | /* Set to 1 if the last command was added to the history list successfully
 | 
					
						
							|  |  |  |    as a separate history entry; set to 0 if the line was ignored or added | 
					
						
							|  |  |  |    to a previous entry as part of command-oriented-history processing. */ | 
					
						
							|  |  |  | int hist_last_line_added; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | /* Set to 1 if builtins/history.def:push_history added the last history
 | 
					
						
							|  |  |  |    entry. */ | 
					
						
							|  |  |  | int hist_last_line_pushed; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (READLINE)
 | 
					
						
							|  |  |  | /* If non-zero, and readline is being used, the user is offered the
 | 
					
						
							|  |  |  |    chance to re-edit a failed history expansion. */ | 
					
						
							|  |  |  | int history_reediting; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* If non-zero, and readline is being used, don't directly execute a
 | 
					
						
							|  |  |  |    line with history substitution.  Reload it into the editing buffer | 
					
						
							|  |  |  |    instead and let the user further edit and confirm with a newline. */ | 
					
						
							|  |  |  | int hist_verify; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* READLINE */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | /* Non-zero means to not save function definitions in the history list. */ | 
					
						
							|  |  |  | int dont_save_function_defs; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Variables declared in other files used here. */ | 
					
						
							|  |  |  | extern int current_command_line_count; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | extern struct dstack dstack; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static int bash_history_inhibit_expansion __P((char *, int)); | 
					
						
							|  |  |  | #if defined (READLINE)
 | 
					
						
							|  |  |  | static void re_edit __P((char *)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | static int history_expansion_p __P((char *)); | 
					
						
							|  |  |  | static int shell_comment __P((char *)); | 
					
						
							|  |  |  | static int should_expand __P((char *)); | 
					
						
							|  |  |  | static HIST_ENTRY *last_history_entry __P((void)); | 
					
						
							|  |  |  | static char *expand_histignore_pattern __P((char *)); | 
					
						
							|  |  |  | static int history_should_ignore __P((char *)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | /* Is the history expansion starting at string[i] one that should not
 | 
					
						
							|  |  |  |    be expanded? */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | bash_history_inhibit_expansion (string, i) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int i; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* The shell uses ! as a pattern negation character in globbing [...]
 | 
					
						
							|  |  |  |      expressions, so let those pass without expansion. */ | 
					
						
							|  |  |  |   if (i > 0 && (string[i - 1] == '[') && member (']', string + i + 1)) | 
					
						
							|  |  |  |     return (1); | 
					
						
							|  |  |  |   /* The shell uses ! as the indirect expansion character, so let those
 | 
					
						
							|  |  |  |      expansions pass as well. */ | 
					
						
							|  |  |  |   else if (i > 1 && string[i - 1] == '{' && string[i - 2] == '$' && | 
					
						
							|  |  |  | 	     member ('}', string + i + 1)) | 
					
						
							|  |  |  |     return (1); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #if defined (EXTENDED_GLOB)
 | 
					
						
							|  |  |  |   else if (extended_glob && i > 1 && string[i+1] == '(' && member (')', string + i + 2)) | 
					
						
							|  |  |  |     return (1); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | void | 
					
						
							|  |  |  | bash_initialize_history () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   history_quotes_inhibit_expansion = 1; | 
					
						
							|  |  |  |   history_search_delimiter_chars = ";&()|<>"; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   history_inhibit_expansion_function = bash_history_inhibit_expansion; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (BANG_HISTORY)
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   sv_histchars ("histchars"); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | bash_history_reinit (interact) | 
					
						
							|  |  |  |      int interact; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if defined (BANG_HISTORY)
 | 
					
						
							|  |  |  |   history_expansion = interact != 0; | 
					
						
							|  |  |  |   history_expansion_inhibited = 1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   remember_on_history = enable_history_list = interact != 0; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   history_inhibit_expansion_function = bash_history_inhibit_expansion; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | bash_history_disable () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   remember_on_history = 0; | 
					
						
							|  |  |  | #if defined (BANG_HISTORY)
 | 
					
						
							|  |  |  |   history_expansion_inhibited = 1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | bash_history_enable () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   remember_on_history = 1; | 
					
						
							|  |  |  | #if defined (BANG_HISTORY)
 | 
					
						
							|  |  |  |   history_expansion_inhibited = 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   history_inhibit_expansion_function = bash_history_inhibit_expansion; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   sv_history_control ("HISTCONTROL"); | 
					
						
							|  |  |  |   sv_histignore ("HISTIGNORE"); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Load the history list from the history file. */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | load_history () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *hf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Truncate history file for interactive shells which desire it.
 | 
					
						
							|  |  |  |      Note that the history file is automatically truncated to the | 
					
						
							|  |  |  |      size of HISTSIZE if the user does not explicitly set the size | 
					
						
							|  |  |  |      differently. */ | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   set_if_not ("HISTSIZE", "500"); | 
					
						
							|  |  |  |   sv_histsize ("HISTSIZE"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   set_if_not ("HISTFILESIZE", get_string_value ("HISTSIZE")); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   sv_histsize ("HISTFILESIZE"); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Read the history in HISTFILE into the history list. */ | 
					
						
							|  |  |  |   hf = get_string_value ("HISTFILE"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (hf && *hf && file_exists (hf)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       read_history (hf); | 
					
						
							|  |  |  |       using_history (); | 
					
						
							|  |  |  |       history_lines_in_file = where_history (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | void | 
					
						
							|  |  |  | bash_clear_history () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   clear_history (); | 
					
						
							|  |  |  |   history_lines_this_session = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Delete and free the history list entry at offset I. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | bash_delete_histent (i) | 
					
						
							|  |  |  |      int i; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HIST_ENTRY *discard; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   discard = remove_history (i); | 
					
						
							|  |  |  |   if (discard) | 
					
						
							|  |  |  |     free_history_entry (discard); | 
					
						
							|  |  |  |   history_lines_this_session--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | bash_delete_last_history () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i; | 
					
						
							|  |  |  |   HIST_ENTRY **hlist, *histent; | 
					
						
							|  |  |  |   int r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   hlist = history_list (); | 
					
						
							|  |  |  |   if (hlist == NULL) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; hlist[i]; i++) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   i--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* History_get () takes a parameter that must be offset by history_base. */ | 
					
						
							|  |  |  |   histent = history_get (history_base + i);	/* Don't free this */ | 
					
						
							|  |  |  |   if (histent == NULL) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   r = bash_delete_histent (i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (where_history () > history_length) | 
					
						
							|  |  |  |     history_set_pos (history_length); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #ifdef INCLUDE_UNUSED
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Write the existing history out to the history file. */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | save_history () | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *hf; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   hf = get_string_value ("HISTFILE"); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (hf && *hf && file_exists (hf)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Append only the lines that occurred this session to
 | 
					
						
							|  |  |  | 	 the history file. */ | 
					
						
							|  |  |  |       using_history (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (history_lines_this_session < where_history () || force_append_history) | 
					
						
							|  |  |  | 	append_history (history_lines_this_session, hf); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	write_history (hf); | 
					
						
							|  |  |  |       sv_histsize ("HISTFILESIZE"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | maybe_append_history (filename) | 
					
						
							|  |  |  |      char *filename; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int fd, result; | 
					
						
							|  |  |  |   struct stat buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   result = EXECUTION_SUCCESS; | 
					
						
							|  |  |  |   if (history_lines_this_session && (history_lines_this_session < where_history ())) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* If the filename was supplied, then create it if necessary. */ | 
					
						
							|  |  |  |       if (stat (filename, &buf) == -1 && errno == ENOENT) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	  fd = open (filename, O_WRONLY|O_CREAT, 0600); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (fd < 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      builtin_error (_("%s: cannot create: %s"), filename, strerror (errno)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      return (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  close (fd); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       result = append_history (history_lines_this_session, filename); | 
					
						
							|  |  |  |       history_lines_in_file += history_lines_this_session; | 
					
						
							|  |  |  |       history_lines_this_session = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (result); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* If this is an interactive shell, then append the lines executed
 | 
					
						
							|  |  |  |    this session to the history file. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | maybe_save_shell_history () | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int result; | 
					
						
							|  |  |  |   char *hf; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   result = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (history_lines_this_session) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       hf = get_string_value ("HISTFILE"); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (hf && *hf) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* If the file doesn't exist, then create it. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  if (file_exists (hf) == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      int file; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	      file = open (hf, O_CREAT | O_TRUNC | O_WRONLY, 0600); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      if (file != -1) | 
					
						
							|  |  |  | 		close (file); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Now actually append the lines if the history hasn't been
 | 
					
						
							|  |  |  | 	     stifled.  If the history has been stifled, rewrite the | 
					
						
							|  |  |  | 	     history file. */ | 
					
						
							|  |  |  | 	  using_history (); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (history_lines_this_session <= where_history () || force_append_history) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      result = append_history (history_lines_this_session, hf); | 
					
						
							|  |  |  | 	      history_lines_in_file += history_lines_this_session; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      result = write_history (hf); | 
					
						
							|  |  |  | 	      history_lines_in_file = history_lines_this_session; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  history_lines_this_session = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  sv_histsize ("HISTFILESIZE"); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (READLINE)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Tell readline () that we have some text for it to edit. */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | re_edit (text) | 
					
						
							|  |  |  |      char *text; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (bash_input.type == st_stdin) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     bash_re_edit (text); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #endif /* READLINE */
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Return 1 if this line needs history expansion. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | history_expansion_p (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *s; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (s = line; *s; s++) | 
					
						
							|  |  |  |     if (*s == history_expansion_char || *s == history_subst_char) | 
					
						
							|  |  |  |       return 1; | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Do pre-processing on LINE.  If PRINT_CHANGES is non-zero, then
 | 
					
						
							|  |  |  |    print the results of expanding the line if there were any changes. | 
					
						
							|  |  |  |    If there is an error, return NULL, otherwise the expanded line is | 
					
						
							|  |  |  |    returned.  If ADDIT is non-zero the line is added to the history | 
					
						
							|  |  |  |    list after history expansion.  ADDIT is just a suggestion; | 
					
						
							|  |  |  |    REMEMBER_ON_HISTORY can veto, and does. | 
					
						
							|  |  |  |    Right now this does history expansion. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | pre_process_line (line, print_changes, addit) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  |      int print_changes, addit; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *history_value; | 
					
						
							|  |  |  |   char *return_value; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int expanded; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return_value = line; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   expanded = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #  if defined (BANG_HISTORY)
 | 
					
						
							|  |  |  |   /* History expand the line.  If this results in no errors, then
 | 
					
						
							|  |  |  |      add that line to the history if ADDIT is non-zero. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (!history_expansion_inhibited && history_expansion && history_expansion_p (line)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       expanded = history_expand (line, &history_value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (expanded) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (print_changes) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (expanded < 0) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		internal_error ("%s", history_value); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #if defined (READLINE)
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      else if (hist_verify == 0 || expanded == 2) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		fprintf (stderr, "%s\n", history_value); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* If there was an error, return NULL. */ | 
					
						
							|  |  |  | 	  if (expanded < 0 || expanded == 2)	/* 2 == print only */ | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #    if defined (READLINE)
 | 
					
						
							|  |  |  | 	      if (expanded == 2 && rl_dispatching == 0 && *history_value) | 
					
						
							|  |  |  | #    else	      
 | 
					
						
							|  |  |  | 	      if (expanded == 2 && *history_value) | 
					
						
							|  |  |  | #    endif /* !READLINE */
 | 
					
						
							|  |  |  | 		maybe_add_history (history_value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      free (history_value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #    if defined (READLINE)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      /* New hack.  We can allow the user to edit the
 | 
					
						
							|  |  |  | 		 failed history expansion. */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      if (history_reediting && expanded < 0 && rl_done) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		re_edit (line); | 
					
						
							|  |  |  | #    endif /* READLINE */
 | 
					
						
							|  |  |  | 	      return ((char *)NULL); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #    if defined (READLINE)
 | 
					
						
							|  |  |  | 	  if (hist_verify && expanded == 1) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      re_edit (history_value); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      return ((char *)NULL); | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #    endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Let other expansions know that return_value can be free'ed,
 | 
					
						
							|  |  |  | 	 and that a line has been added to the history list.  Note | 
					
						
							|  |  |  | 	 that we only add lines that have something in them. */ | 
					
						
							|  |  |  |       expanded = 1; | 
					
						
							|  |  |  |       return_value = history_value; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #  endif /* BANG_HISTORY */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (addit && remember_on_history && *return_value) | 
					
						
							|  |  |  |     maybe_add_history (return_value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (expanded == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return_value = savestring (line); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (return_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | /* Return 1 if the first non-whitespace character in LINE is a `#', indicating
 | 
					
						
							|  |  |  |  * that the line is a shell comment. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | shell_comment (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (p = line; p && *p && whitespace (*p); p++) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   return (p && *p == '#'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #ifdef INCLUDE_UNUSED
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | /* Remove shell comments from LINE.  A `#' and anything after it is a comment.
 | 
					
						
							|  |  |  |    This isn't really useful yet, since it doesn't handle quoting. */ | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | filter_comments (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (p = line; p && *p && *p != '#'; p++) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   if (p && *p == '#') | 
					
						
							|  |  |  |     *p = '\0'; | 
					
						
							|  |  |  |   return (line); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | /* Check LINE against what HISTCONTROL says to do.  Returns 1 if the line
 | 
					
						
							|  |  |  |    should be saved; 0 if it should be discarded. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | check_history_control (line) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   HIST_ENTRY *temp; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   int r; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   if (history_control == 0) | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* ignorespace or ignoreboth */ | 
					
						
							|  |  |  |   if ((history_control & HC_IGNSPACE) && *line == ' ') | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* ignoredups or ignoreboth */ | 
					
						
							|  |  |  |   if (history_control & HC_IGNDUPS) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       using_history (); | 
					
						
							|  |  |  |       temp = previous_history (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       r = (temp == 0 || STREQ (temp->line, line) == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       using_history (); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (r == 0) | 
					
						
							|  |  |  | 	return r; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Remove all entries matching LINE from the history list.  Triggered when
 | 
					
						
							|  |  |  |    HISTCONTROL includes `erasedups'. */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | hc_erasedups (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HIST_ENTRY *temp; | 
					
						
							|  |  |  |   int r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   using_history (); | 
					
						
							|  |  |  |   while (temp = previous_history ()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (STREQ (temp->line, line)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  r = where_history (); | 
					
						
							|  |  |  | 	  remove_history (r); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   using_history (); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Add LINE to the history list, handling possibly multi-line compound
 | 
					
						
							|  |  |  |    commands.  We note whether or not we save the first line of each command | 
					
						
							|  |  |  |    (which is usually the entire command and history entry), and don't add | 
					
						
							|  |  |  |    the second and subsequent lines of a multi-line compound command if we | 
					
						
							|  |  |  |    didn't save the first line.  We don't usually save shell comment lines in | 
					
						
							|  |  |  |    compound commands in the history, because they could have the effect of | 
					
						
							|  |  |  |    commenting out the rest of the command when the entire command is saved as | 
					
						
							|  |  |  |    a single history entry (when COMMAND_ORIENTED_HISTORY is enabled).  If | 
					
						
							|  |  |  |    LITERAL_HISTORY is set, we're saving lines in the history with embedded | 
					
						
							|  |  |  |    newlines, so it's OK to save comment lines.  We also make sure to save | 
					
						
							|  |  |  |    multiple-line quoted strings or other constructs. */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | maybe_add_history (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   hist_last_line_added = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Don't use the value of history_control to affect the second
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      and subsequent lines of a multi-line command (old code did | 
					
						
							|  |  |  |      this only when command_oriented_history is enabled). */ | 
					
						
							|  |  |  |   if (current_command_line_count > 1) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (current_command_first_line_saved && | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  (literal_history || dstack.delimiter_depth != 0 || shell_comment (line) == 0)) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	bash_add_history (line); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   /* This is the first line of a (possible multi-line) command.  Note whether
 | 
					
						
							|  |  |  |      or not we should save the first line and remember it. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   current_command_first_line_saved = check_add_history (line, 0); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | /* Just check LINE against HISTCONTROL and HISTIGNORE and add it to the
 | 
					
						
							|  |  |  |    history if it's OK.  Used by `history -s' as well as maybe_add_history(). | 
					
						
							|  |  |  |    Returns 1 if the line was saved in the history, 0 otherwise. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | check_add_history (line, force) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  |      int force; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (check_history_control (line) && history_should_ignore (line) == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       /* We're committed to saving the line.  If the user has requested it,
 | 
					
						
							|  |  |  | 	 remove other matching lines from the history. */ | 
					
						
							|  |  |  |       if (history_control & HC_ERASEDUPS) | 
					
						
							|  |  |  | 	hc_erasedups (line); | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (force) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  really_add_history (line); | 
					
						
							|  |  |  | 	  using_history (); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	bash_add_history (line); | 
					
						
							|  |  |  |       return 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (SYSLOG_HISTORY)
 | 
					
						
							|  |  |  | #define SYSLOG_MAXLEN 600
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | bash_syslog_history (line) | 
					
						
							|  |  |  |      const char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char trunc[SYSLOG_MAXLEN]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (strlen(line) < SYSLOG_MAXLEN) | 
					
						
							|  |  |  |     syslog (SYSLOG_FACILITY|SYSLOG_LEVEL, "HISTORY: PID=%d UID=%d %s", getpid(), current_user.uid, line); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       strncpy (trunc, line, SYSLOG_MAXLEN); | 
					
						
							|  |  |  |       trunc[SYSLOG_MAXLEN - 1] = '\0'; | 
					
						
							|  |  |  |       syslog (SYSLOG_FACILITY|SYSLOG_LEVEL, "HISTORY (TRUNCATED): PID=%d UID=%d %s", getpid(), current_user.uid, trunc); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |      	 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Add a line to the history list.
 | 
					
						
							|  |  |  |    The variable COMMAND_ORIENTED_HISTORY controls the style of history | 
					
						
							|  |  |  |    remembering;  when non-zero, and LINE is not the first line of a | 
					
						
							|  |  |  |    complete parser construct, append LINE to the last history line instead | 
					
						
							|  |  |  |    of adding it as a new line. */ | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | bash_add_history (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int add_it, offset, curlen; | 
					
						
							|  |  |  |   HIST_ENTRY *current, *old; | 
					
						
							|  |  |  |   char *chars_to_add, *new_line; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   add_it = 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (command_oriented_history && current_command_line_count > 1) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       chars_to_add = literal_history ? "\n" : history_delimiting_chars (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       using_history (); | 
					
						
							|  |  |  |       current = previous_history (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (current) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* If the previous line ended with an escaped newline (escaped
 | 
					
						
							|  |  |  | 	     with backslash, but otherwise unquoted), then remove the quoted | 
					
						
							|  |  |  | 	     newline, since that is what happens when the line is parsed. */ | 
					
						
							|  |  |  | 	  curlen = strlen (current->line); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (dstack.delimiter_depth == 0 && current->line[curlen - 1] == '\\' && | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      current->line[curlen - 2] != '\\') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      current->line[curlen - 1] = '\0'; | 
					
						
							|  |  |  | 	      curlen--; | 
					
						
							|  |  |  | 	      chars_to_add = ""; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	  new_line = (char *)xmalloc (1 | 
					
						
							|  |  |  | 				      + curlen | 
					
						
							|  |  |  | 				      + strlen (line) | 
					
						
							|  |  |  | 				      + strlen (chars_to_add)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  sprintf (new_line, "%s%s%s", current->line, chars_to_add, line); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  offset = where_history (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  old = replace_history_entry (offset, new_line, current->data); | 
					
						
							|  |  |  | 	  free (new_line); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (old) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    free_history_entry (old); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  add_it = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (add_it) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     really_add_history (line); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (SYSLOG_HISTORY)
 | 
					
						
							|  |  |  |   bash_syslog_history (line); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   using_history (); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | really_add_history (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   hist_last_line_added = 1; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   hist_last_line_pushed = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   add_history (line); | 
					
						
							|  |  |  |   history_lines_this_session++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | int | 
					
						
							|  |  |  | history_number () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   using_history (); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   return (remember_on_history ? history_base + where_history () : 1); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | should_expand (s) | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (p = s; p && *p; p++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*p == '\\') | 
					
						
							|  |  |  | 	p++; | 
					
						
							|  |  |  |       else if (*p == '&') | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | histignore_item_func (ign) | 
					
						
							|  |  |  |      struct ign *ign; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (should_expand (ign->val)) | 
					
						
							|  |  |  |     ign->flags |= HIGN_EXPAND; | 
					
						
							|  |  |  |   return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | setup_history_ignore (varname) | 
					
						
							|  |  |  |      char *varname; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   setup_ignore_patterns (&histignore); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static HIST_ENTRY * | 
					
						
							|  |  |  | last_history_entry () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HIST_ENTRY *he; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   using_history (); | 
					
						
							|  |  |  |   he = previous_history (); | 
					
						
							|  |  |  |   using_history (); | 
					
						
							|  |  |  |   return he; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | last_history_line () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HIST_ENTRY *he; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   he = last_history_entry (); | 
					
						
							|  |  |  |   if (he == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  |   return he->line; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | expand_histignore_pattern (pat) | 
					
						
							|  |  |  |      char *pat; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HIST_ENTRY *phe; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   char *ret; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   phe = last_history_entry (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (phe == (HIST_ENTRY *)0) | 
					
						
							|  |  |  |     return (savestring (pat)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   ret = strcreplace (pat, '&', phe->line, 1); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return 1 if we should not put LINE into the history according to the
 | 
					
						
							|  |  |  |    patterns in HISTIGNORE. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | history_should_ignore (line) | 
					
						
							|  |  |  |      char *line; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i, match; | 
					
						
							|  |  |  |   char *npat; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (histignore.num_ignores == 0) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = match = 0; i < histignore.num_ignores; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (histignore.ignores[i].flags & HIGN_EXPAND) | 
					
						
							|  |  |  | 	npat = expand_histignore_pattern (histignore.ignores[i].val); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	npat = histignore.ignores[i].val; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       match = strmatch (npat, line, FNMATCH_EXTFLAG) != FNM_NOMATCH; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (histignore.ignores[i].flags & HIGN_EXPAND) | 
					
						
							|  |  |  | 	free (npat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (match) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return match; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* HISTORY */
 |