How can I tell (in ~/.bashrc) if I'm running in interactive mode, or, say, executing a command over ssh. I want to avoid printing of ANSI escape sequences in .bashrc if it's the latter.
6 Answers
According to man bash:
PS1 is set and $- includes i if bash is interactive, allowing a shell script or a startup file to test this state.
So you can use:
if [[ $- == *i* ]]
then
do_interactive_stuff
fi
Also:
When an interactive shell that is not a login shell is started, bash reads and executes commands from /etc/bash.bashrc and ~/.bashrc, if these files exist.
So ~/.bashrc is only sourced for interactive shells. Sometimes, people source it from ~/.bash_profile or ~/.profile which is incorrect since it interferes with the expected behavior. If you want to simplify maintenance of code that is common, you should use a separate file to contain the common code and source it independently from both rc files.
It's best if there's no output to stdout from login rc files such as ~/.bash_profile or ~/.profile since it can interfere with the proper operation of rsync for example.
In any case, it's still a good idea to test for interactivity since incorrect configuration may exist.
- 64,083
the test tool can check for this (from the man page):
-t FD True if FD is opened on a terminal.
So you can use for example:
if [ -t 0 ] ; then
echo stdin is a terminal
.....
fi
or
if [ -t 1 ] ; then
echo stdout is a terminal
fi
- 593
I typically look at the output of the program tty.
If you're on a tty, it will tell you which tty you're on. If you're not in interactive mode, it will typically tell you something like "not a tty".
- 12,104
This is how red hat does it... Guessing it's the fastest way...
if [ "${-#*i}" == "$-" ]; then
It means get the bash parameters, do a substring removal, shortest possible, match everything between the beginning and i. Then check if it's the same as the original bash parameters.
Check you did your job by connecting to the user using sftp, it will fail if non interactive sessions have output
- 269
This is half-an-answer. It's more of an addon/reference to the other answers.
"man bash" definition
I find the definition hard to understand: https://manpages.debian.org/bookworm/bash/bash.1.en.html BASH Invocation
An interactive shell is one started without non-option arguments (unless -s
is specified) and without the -c option, whose standard input and error are
both connected to terminals (as determined by isatty(3)), or one started with
the -i option. PS1 is set and $- includes i if bash is interactive, allowing a
shell script or a startup file to test this state.
Bash manual
2024-05-13, UPDATE: The full GNU manual has a much longer section. Didn't read it yet.
https://www.gnu.org/software/bash/manual/bash.html#Interactive-Shells Interactive Shells
ABS lists several methods
The ABS (Advanced Bash Scripting Guide) lists several methods. None of them are great. And they are not equivalent.
https://tldp.org/LDP/abs/html/intandnonint.html Interactive and non-interactive shells and scripts
FYI: My trouble on Rocky
I've had some trouble on "Rocky Linux release 9.2 (Blue Onyx)" with a service startup script that insisted it was running interactively.
There was a trap 17 (signal SIGCHLD) in there. This traps whenever "Job Control" is enabled. And by default: Job Control is disabled inside non-interactive scripts and enabled inside interactive scripts.
Now for some reason Job Control enabled itself on this taste of Linux. And therefor the SIGCHLD trapped and the startup script failed. This worked fine on a different distro. I'm still in the process of analyzing this. So that's all I know for now.
- 2,045