Note:
- This answer probably goes deeper than the use case warrants, and
find 2>/dev/null
may be good enough in many situations. It may still be of interest for a cross-platform perspective and for its discussion of some advanced shell techniques in the interest of finding a solution that is as robust as possible, even though the cases guarded against may be largely hypothetical.
If your shell is bash
or zsh
, there’s a solution that is robust while being reasonably simple, using only POSIX-compliant find
features; while bash
itself is not part of POSIX, most modern Unix platforms come with it, making this solution widely portable:
find . > files_and_folders 2> >(grep -v 'Permission denied' >&2)
Note:
-
If your system is configured to show localized error messages, prefix the
find
calls below withLC_ALL=C
(LC_ALL=C find ...
) to ensure that English messages are reported, so thatgrep -v 'Permission denied'
works as intended. Invariably, however, any error messages that do get displayed will then be in English as well. -
>(...)
is a (rarely used) output process substitution that allows redirecting output (in this case, stderr output (2>
) to the stdin of the command inside>(...)
.
In addition tobash
andzsh
,ksh
supports them as well in principle, but trying to combine them with redirection from stderr, as is done here (2> >(...)
), appears to be silently ignored (inksh 93u+
).-
grep -v 'Permission denied'
filters out (-v
) all lines (from thefind
command’s stderr stream) that contain the phrasePermission denied
and outputs the remaining lines to stderr (>&2
). -
Note: There’s a small chance that some of
grep
‘s output may arrive afterfind
completes, because the overall command doesn’t wait for the command inside>(...)
to finish. Inbash
, you can prevent this by appending| cat
to the command.
-
This approach is:
-
robust:
grep
is only applied to error messages (and not to a combination of file paths and error messages, potentially leading to false positives), and error messages other than permission-denied ones are passed through, to stderr. -
side-effect free:
find
‘s exit code is preserved: the inability to access at least one of the filesystem items encountered results in exit code1
(although that won’t tell you whether errors other than permission-denied ones occurred (too)).
POSIX-compliant solutions:
Fully POSIX-compliant solutions either have limitations or require additional work.
If find
‘s output is to be captured in a file anyway (or suppressed altogether), then the pipeline-based solution from Jonathan Leffler’s answer is simple, robust, and POSIX-compliant:
find . 2>&1 >files_and_folders | grep -v 'Permission denied' >&2
Note that the order of the redirections matters: 2>&1
must come first.
Capturing stdout output in a file up front allows 2>&1
to send only error messages through the pipeline, which grep
can then unambiguously operate on.
The only downside is that the overall exit code will be the grep
command’s, not find
‘s, which in this case means: if there are no errors at all or only permission-denied errors, the exit code will be 1
(signaling failure), otherwise (errors other than permission-denied ones) 0
— which is the opposite of the intent.
That said, find
‘s exit code is rarely used anyway, as it often conveys little information beyond fundamental failure such as passing a non-existent path.
However, the specific case of even only some of the input paths being inaccessible due to lack of permissions is reflected in find
‘s exit code (in both GNU and BSD find
): if a permissions-denied error occurs for any of the files processed, the exit code is set to 1
.
The following variation addresses that:
find . 2>&1 >files_and_folders | { grep -v 'Permission denied' >&2; [ $? -eq 1 ]; }
Now, the exit code indicates whether any errors other than Permission denied
occurred: 1
if so, 0
otherwise.
In other words: the exit code now reflects the true intent of the command: success (0
) is reported, if no errors at all or only permission-denied errors occurred.
This is arguably even better than just passing find
‘s exit code through, as in the solution at the top.
gniourf_gniourf in the comments proposes a (still POSIX-compliant) generalization of this solution using sophisticated redirections, which works even with the default behavior of printing the file paths to stdout:
{ find . 3>&2 2>&1 1>&3 | grep -v 'Permission denied' >&3; } 3>&2 2>&1
In short: Custom file descriptor 3
is used to temporarily swap stdout (1
) and stderr (2
), so that error messages alone can be piped to grep
via stdout.
Without these redirections, both data (file paths) and error messages would be piped to grep
via stdout, and grep
would then not be able to distinguish between error message Permission denied
and a (hypothetical) file whose name happens to contain the phrase Permission denied
.
As in the first solution, however, the the exit code reported will be grep
‘s, not find
‘s, but the same fix as above can be applied.
Notes on the existing answers:
-
There are several points to note about Michael Brux’s answer,
find . ! -readable -prune -o -print
:-
It requires GNU
find
; notably, it won’t work on macOS. Of course, if you only ever need the command to work with GNUfind
, this won’t be a problem for you. -
Some
Permission denied
errors may still surface:find ! -readable -prune
reports such errors for the child items of directories for which the current user does haver
permission, but lacksx
(executable) permission. The reason is that because the directory itself is readable,-prune
is not executed, and the attempt to descend into that directory then triggers the error messages. That said, the typical case is for ther
permission to be missing. -
Note: The following point is a matter of philosophy and/or specific use case, and you may decide it is not relevant to you and that the command fits your needs well, especially if simply printing the paths is all you do:
- If you conceptualize the filtering of the permission-denied error messages a separate task that you want to be able to apply to any
find
command, then the opposite approach of proactively preventing permission-denied errors requires introducing «noise» into thefind
command, which also introduces complexity and logical pitfalls. - For instance, the most up-voted comment on Michael’s answer (as of this writing) attempts to show how to extend the command by including a
-name
filter, as follows:
find . ! -readable -prune -o -name '*.txt'
This, however, does not work as intended, because the trailing-print
action is required (an explanation can be found in this answer). Such subtleties can introduce bugs.
- If you conceptualize the filtering of the permission-denied error messages a separate task that you want to be able to apply to any
-
-
The first solution in Jonathan Leffler’s answer,
find . 2>/dev/null > files_and_folders
, as he himself states, blindly silences all error messages (and the workaround is cumbersome and not fully robust, as he also explains). Pragmatically speaking, however, it is the simplest solution, as you may be content to assume that any and all errors would be permission-related. -
mist’s answer,
sudo find . > files_and_folders
, is concise and pragmatic, but ill-advised for anything other than merely printing filenames, for security reasons: because you’re running as the root user, «you risk having your whole system being messed up by a bug in find or a malicious version, or an incorrect invocation which writes something unexpectedly, which could not happen if you ran this with normal privileges» (from a comment on mist’s answer by tripleee). -
The 2nd solution in viraptor’s answer,
find . 2>&1 | grep -v 'Permission denied' > some_file
runs the risk of false positives (due to sending a mix of stdout and stderr through the pipeline), and, potentially, instead of reporting non-permission-denied errors via stderr, captures them alongside the output paths in the output file.
you can filter out messages to stderr
. I prefer to redirect them to stdout
like this.
find / -name art 2>&1 | grep -v "Permission denied"
Explanation:
In short, all regular output goes to standard output (stdout
). All error messages to standard error (stderr
).
grep
usually finds/prints the specified string, the -v
inverts this, so it finds/prints every string that doesn’t contain «Permission denied». All of your output from the find command, including error messages usually sent to stderr
(file descriptor 2) go now to stdout
(file descriptor 1) and then get filtered by the grep
command.
This assumes you are using the bash/sh
shell.
Under tcsh/csh
you would use
find / -name art |& grep ....
Модераторы: /dev/random, Модераторы разделов
-
pavelvat
- Сообщения: 64
- ОС: Arch Linux i686
find — каким ключом отключить сообщения «Отказано в доступе»?
Код: Выделить всё
[user@myhost ~]$ find / -name "*http_cache_*"
find: `/proc/tty/driver': Отказано в доступе
find: `/proc/1/task/1/fd': Отказано в доступе
find: `/proc/1/task/1/fdinfo': Отказано в доступе
find: `/proc/1/fd': Отказано в доступе
find: `/proc/1/fdinfo': Отказано в доступе
find: `/proc/2/task/2/fd': Отказано в доступе
.
.
.
/usr/share/kde4/services/http_cache_cleaner.desktop
.
.
.
find: `/usr/lib/mozilla/extensions': Отказано в доступе
.
.
.
/usr/lib/kde4/libexec/kio_http_cache_cleaner
/usr/lib/libkdeinit4_kio_http_cache_cleaner.so
find: `/lost+found': Отказано в доступе
find: `/boot/lost+found': Отказано в доступе
-
flank’er
- Сообщения: 496
- Статус: слаковщик
- ОС: Slackware64
-
drBatty
- Сообщения: 8735
- Статус: GPG ID: 4DFBD1D6 дом горит, козёл не видит…
- ОС: Slackware-current
-
Контактная информация:
Re: find — каким ключом отключить сообщения «Отказано в доступе»?
Сообщение
drBatty » 02.02.2010 09:52
wilgib писал(а): ↑
02.02.2010 01:04
2 > /dev/null ?
Спасибо.
Но у этого способа есть недостаток — убираются все сообщения об ошибках, а нужно только не замусоривать вывод сообщениями о директориях к которым нет доступа. Я подумал, что разработчики find предусмотрели такую возможность, но беглый просмотр
как-бы это сообщение об ошибке, как его можно убрать? Я-бы тоже не стал так делать.
Впрочем лекарство есть:
запустите find с ключём -type d и проверяйте второй find только в директориях в которых у вас есть доступ. Впрочем, проще отфильтровать эти сообщения, как сказано в прошлом посте.
-
pavelvat
- Сообщения: 64
- ОС: Arch Linux i686
Re: find — каким ключом отключить сообщения «Отказано в доступе»?
Сообщение
pavelvat » 03.02.2010 12:23
drBatty писал(а): ↑
02.02.2010 09:52
wilgib писал(а): ↑
02.02.2010 01:04
2 > /dev/null ?
Спасибо.
Но у этого способа есть недостаток — убираются все сообщения об ошибках, а нужно только не замусоривать вывод сообщениями о директориях к которым нет доступа. Я подумал, что разработчики find предусмотрели такую возможность, но беглый просмотркак-бы это сообщение об ошибке, как его можно убрать? Я-бы тоже не стал так делать.
Впрочем лекарство есть:
запустите find с ключём -type d и проверяйте второй find только в директориях в которых у вас есть доступ. Впрочем, проще отфильтровать эти сообщения, как сказано в прошлом посте.
Пожалуй, способ указанный flank’er наиболее простой, но и у него есть недостаток — если локаль произвольная.
Всё же интересно, можно ли добиться желаемого только средствами команды find, без организации конвеера?
-
drBatty
- Сообщения: 8735
- Статус: GPG ID: 4DFBD1D6 дом горит, козёл не видит…
- ОС: Slackware-current
- Контактная информация:
Re: find — каким ключом отключить сообщения «Отказано в доступе»?
Сообщение
drBatty » 03.02.2010 12:46
wilgib писал(а): ↑
03.02.2010 12:23
если локаль произвольная.
Код: Выделить всё
drb@localhost:~$ ( LANG=C; find /etc -name "XXX" )
find: /etc/cups/ssl: Permission denied
find: /etc/samba/private: Permission denied
find: /etc/openvpn/certs: Permission denied
find: /etc/openvpn/keys: Permission denied
find: /etc/bluetooth/passkeys: Permission denied
drb@localhost:~$ find /etc -name "XXX"
find: /etc/cups/ssl: Отказано в доступе
find: /etc/samba/private: Отказано в доступе
find: /etc/openvpn/certs: Отказано в доступе
find: /etc/openvpn/keys: Отказано в доступе
find: /etc/bluetooth/passkeys: Отказано в доступе
wilgib писал(а): ↑
03.02.2010 12:23
Всё же интересно, можно ли добиться желаемого только средствами команды find, без организации конвеера?
ИМХО нет, разве что использовать -perm, -uid, что-бы не лазать в закрытые папки.
-
pavelvat
- Сообщения: 64
- ОС: Arch Linux i686
Re: find — каким ключом отключить сообщения «Отказано в доступе»?
Сообщение
pavelvat » 03.02.2010 15:11
drBatty писал(а): ↑
03.02.2010 12:46
wilgib писал(а): ↑
03.02.2010 12:23
если локаль произвольная.
Код: Выделить всё
drb@localhost:~$ ( LANG=C; find /etc -name "XXX" ) find: /etc/cups/ssl: Permission denied find: /etc/samba/private: Permission denied find: /etc/openvpn/certs: Permission denied find: /etc/openvpn/keys: Permission denied find: /etc/bluetooth/passkeys: Permission denied drb@localhost:~$ find /etc -name "XXX" find: /etc/cups/ssl: Отказано в доступе find: /etc/samba/private: Отказано в доступе find: /etc/openvpn/certs: Отказано в доступе find: /etc/openvpn/keys: Отказано в доступе find: /etc/bluetooth/passkeys: Отказано в доступе
Действительно, LANG=C решает проблему с произвольной локалью.
Да я же знал и про перенаправление, и про /dev/null
Но вот так работают шаблоны мышления — искал соответствующий параметр у программы, а посмотреть на проблему с другой стороны не сообразил..
Но вот с таким еще не встречался, спасибо:
Цитировать
&>файл — перенаправить стандартный вывод (stdout) и ошибки в файл
2>&1 | tee файл — копировать вывод ошибок в stdout, показывать на экране и одновременно писать в файл
In bash, 2> /dev/null
would actually work, as long as you add it as part of the find command itself, i.e., before the pipe.
For example, this will work:
find -type f -printf "%s %pn" 2> /dev/null | sort -nr | head -n 20
With tcsh, there’s no straightforward way to do the same, but man tcsh mentions a workaround:
The shell cannot presently redirect diagnostic output without also redirecting standard output, but ‘(command > output-file) >& error-file‘ is often an acceptable workaround. Either output-file or error-file may be ‘/dev/tty’ to send output to the terminal.
For a plain find command, this gives:
(find > /dev/tty) >& /dev/null
When using pipes, in contrast to bash, you have to place > /dev/tty
after the last pipe to avoid the Ambiguous output redirect. error message.
This will work in tcsh:
(find -type f -printf "%s %pn" | sort -nr | head -n 20 > /dev/tty) >& /dev/null