UPDATE
Thanks a lot for the answers and comments. Thanks to @Fravadona for the help I managed to copy it and change a few things. This is the actual code:
{ exec 0>&9; } 1> /dev/null 2>&1 || exec 9> /dev/tty
checklist() {
(( $# >= 1 )) || return 1
{
printf '%s' "$2"
#(( $# > 1 )) #&& printf ' %q' "${@:2}" I think I don't need this line
printf '...\n'
} >&9
if "$1"; then
printf '%s\n' "$3" >&9
else
printf '%s\n' "$4" >&9
fi
}
checklist "ping -c 3 google.es" "Trying to ping google" "Ping is successfull!" "Error trying to ping google!"
checklist lsblk "Trying to list disks" "The command ended succesfully!" "Error trying to list disks!"
The first argument is the actual command, the 2nd is the string I want to show while the command is running, the 3rd is when the command ends successfully and the 4th is when there's some error.
The output I'm getting is:
Trying to ping google...
Error trying to ping google!
Trying to list disks...
The command ended succesfully!
The lsblk command is working because there's only 1 word but the ping command doesn't work because it has more words. How can I do it? I've tried with single quotes, backward apostrophe, brackets... I think I'm missing something.
The logs are working fine.
Thank you for your help!
ORIGINAL
I'm doing an script in bash and I want to redirect the stderr and stdout to a single file (output.log) and the stderr to another one (error.log). In the terminal, it has to show only the echo commands I want from the script.
I just want to make like a checklist. Redirect stdout and stderr from the commands to a file and stderr to diferent file. Once you have this, the only thing left is checking if the command is successful or not. This is easy just checking the stderr file.
I did something similar to what I want but I have to execute the script like this:
{ . script.sh 2>&1 1>&3 | tee error.log; } > output.log 3>&1
But in each command that I want to show in the terminal, I have to add:
| tee -a /dev/tty
A minor problem with this method is that tput civis and tput cnorm to hide and show the cursor is not working
It would be nice if I can execute the script like this, but this is not required as long as tput works:
. script.sh
I'm showing an example with ping command and another one to show you what I want:
echo "Trying ping..." | tee -a /dev/tty # Show to terminal
if ! ping -c 3 google.es; then # Check if the ping is successful or not, redirecting all outputs to files
echo "Error trying to ping" | tee -a /dev/tty # Show to terminal
else
echo "Ping finished successfully!" | tee -a /dev/tty # Show to terminal
fi
echo "Trying {cmd}..." | tee -a /dev/tty # Show to terminal
if ! {cmd2}; then # Check if the {cmd2} is successful or not, redirecting all outputs to files
echo "Error trying to {cmd2}..." | tee -a /dev/tty # Show to terminal
else
echo "{cmd2} finished successfully!" | tee -a /dev/tty # Show to terminal
fi
.
.
.
.
The output would be and I want is:
Trying ping...
Ping finished successfully!
Trying {cmd2}...
Error trying to {cmd2}!
.
.
.
.
If there's another way to make that checklist I am all ears.
Thank you for your time :)
PS: I will do functions to refactor the code, don't worry about that. For example a function to check if the command is successful or not.
UPDATE
If you could refactor
script.sh
like this:Then, when you run it without sourcing:
On the terminal you'll get:
In error.log you'll have:
In output.log there'll be:
OLD ANSWER
A standard way would be:
To understand it you need to know that the "redirections" of a command are executed from right to left.
Let's consider what happens with
cmd 2>&1 1>&3
.The content of
1
is "moved" to3
.The content
2
is "moved" to1
.As there is no more redirections for
cmd
, the contents of1
,2
(empty) and3
are then "consumed" by the rest of the script.Now, what would happen if we change the order of the redirections with
cmd 1>&3 2>&1
?The content of
2
is "moved" to1
The content of
1
(which also contains a copy of2
) is then "moved" to3
.As there is no more redirections for
cmd
, the contents of the file descriptors1
(empty),2
(empty) and3
are then "consumed" by the rest of the script.