pages:howtos:bash:bash-confusion-with-file-redirection-and-sudo
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
pages:howtos:bash:bash-confusion-with-file-redirection-and-sudo [2021/02/11 14:16] – [Source] mischerh | pages:howtos:bash:bash-confusion-with-file-redirection-and-sudo [2021/12/09 21:57] (current) – [So how to actually redirect output to a file as root?] rokkitlawnchair | ||
---|---|---|---|
Line 19: | Line 19: | ||
Next we will review the code of two popular shells: GNU/Bash and the Z shell, to see how they realize this functionality in code. | Next we will review the code of two popular shells: GNU/Bash and the Z shell, to see how they realize this functionality in code. | ||
- | How do popular shells implement this | + | ===== How do popular shells implement this ===== |
- | ------------------------------------ | + | GNU/Bash uses the [[https:// |
- | + | ||
- | GNU/Bash uses the GNU/Bison project for specifying the input command syntax. The syntax file is written in the yacc format. You can find the special syntax described in the parse.y file, in the root directory of GNU/Bash source. There are two special types defined: redirection and redirection_list. Here is an excerpt from parse.y file which defines how redirecting stdout to a file works or how passing a file to stdin works: | + | |
+ | <sxh bash> | ||
redirection: | redirection: | ||
{ | { | ||
Line 30: | Line 29: | ||
$$ = make_redirection (source, r_output_direction, | $$ = make_redirection (source, r_output_direction, | ||
} | } | ||
- | | '<' | + | |
{ | { | ||
source.dest = 0; | source.dest = 0; | ||
Line 36: | Line 35: | ||
$$ = make_redirection (source, r_input_direction, | $$ = make_redirection (source, r_input_direction, | ||
} | } | ||
- | | ... | + | |
+ | </ | ||
And here is the excerpt which shows that the redirection list goes after a command: | And here is the excerpt which shows that the redirection list goes after a command: | ||
+ | <sxh bash> | ||
command: | command: | ||
{ $$ = clean_simple_command ($1); } | { $$ = clean_simple_command ($1); } | ||
Line 50: | Line 49: | ||
} | } | ||
| ... | | ... | ||
+ | </ | ||
- | + | My point is that you can see that GNU/Bash really has a special syntax for file redirection, | |
- | My point is that you can see that GNU/Bash really has a special syntax for file redirection, | + | <sxh cpp> |
/* | /* | ||
* cmd : { redir } ( for | case | if | while | repeat | | * cmd : { redir } ( for | case | if | while | repeat | | ||
Line 60: | Line 59: | ||
* zsh_construct is passed through to par_subsh(), | * zsh_construct is passed through to par_subsh(), | ||
*/ | */ | ||
+ | </ | ||
Indeed, redirection is parsed later in the function that is responsible for parsing the command: | Indeed, redirection is parsed later in the function that is responsible for parsing the command: | ||
+ | <sxh cpp> | ||
static int | static int | ||
par_cmd(int *cmplx, int zsh_construct) | par_cmd(int *cmplx, int zsh_construct) | ||
Line 106: | Line 105: | ||
return 1; | return 1; | ||
} | } | ||
+ | </ | ||
- | As you can see, the redirections may be at either the end or the beginning of the command. Let me repeat again: file redirection is special syntax and it is not passed to the actual thing being run. The GNU/Bash shell supports the file redirection syntax at the beginning of the line as well but I have just decided to not include it for brevity. | + | As you can see, the redirections may be at either the end or the beginning of the command. Let me repeat again: file redirection is **special syntax** and it is not passed to the actual thing being run. The GNU/Bash shell supports the file redirection syntax at the beginning of the line as well but I have just decided to not include it for brevity. |
- | + | ||
- | So how to actually redirect output to a file as root? | + | |
- | ----------------------------------------------------- | + | |
- | + | ||
- | I guess the most popular solution to this is to simply use a pipe to redirect output to a file. I am not sure but probably the program, tee(1), was made for this purpose. Or it was made as an extension of the tee system call but still it is the perfect tool to solve our issue. So, the solution to this problem would be: | + | |
+ | ===== So how to actually redirect output to a file as root? ===== | ||
+ | I guess the most popular solution to this is to simply use a pipe to redirect output to a file. I am not sure but probably the program, [[https:// | ||
+ | <sxh bash> | ||
echo " | echo " | ||
+ | </ | ||
+ | This will at first (probably, depending on your set up) ask for your password. Then, “hi” will be written to a file descriptor which will get passed on to tee(1). Then, [[https:// | ||
- | This will at first (probably, depending on your set up) ask for your password. Then, “hi” will be written to a file descriptor which will get passed on to tee(1). Then, tee(1) will dump everything into /tmp/test. The >&- (or >/ | + | Another way to solve this is to pass more commands to run via root either by using the **-s** //sudo// option. For example, this works: |
- | + | <sxh bash> | |
- | Another way to solve this is to pass more commands to run via root either by using the -s sudo option. For example, this works: | + | |
sudo -s <<EOF | sudo -s <<EOF | ||
exec >/ | exec >/ | ||
echo " | echo " | ||
EOF | EOF | ||
+ | </ | ||
- | + | Or, run another shell using //sudo// which will redirect everything to ///tmp/test//: | |
- | Or, run another shell using sudo which will redirect everything to / | + | <sxh bash> |
sudo /bin/bash -c 'exec >/ | sudo /bin/bash -c 'exec >/ | ||
+ | </ | ||
- | + | Or, run a script with //sudo// which will internally redirect | |
- | Or, run a script with sudo which will internally redirect stdout to / | + | <sxh bash> |
cat > | cat > | ||
exec >/ | exec >/ | ||
Line 141: | Line 139: | ||
chmod +x ./script.sh | chmod +x ./script.sh | ||
sudo ./script.sh | sudo ./script.sh | ||
+ | </ | ||
Obviously, there are more (complex) ways how you could achieve this but these are the main methods. You are free to adopt these examples to your own case. Please comment if you find any errors or if you want to add anything about this topic. | Obviously, there are more (complex) ways how you could achieve this but these are the main methods. You are free to adopt these examples to your own case. Please comment if you find any errors or if you want to add anything about this topic. | ||
+ | ---- | ||
+ | ~~DISCUSSION~~ |
pages/howtos/bash/bash-confusion-with-file-redirection-and-sudo.1613052961.txt.gz · Last modified: 2021/02/11 14:16 by mischerh