Como posso executar um comando específico para cada resultado de busca?

43

Como alguém executaria um comando específico para cada arquivo encontrado usando o comando find ? Para o propósito da pergunta, digamos que eu simplesmente gostaria de excluir cada arquivo encontrado por find .

    
por FailedDev 02.12.2011 / 11:33

5 respostas

52

Editar: Embora a resposta a seguir explique o caso geral de uso, devo observar que a exclusão de arquivos e diretórios é um caso especial. Em vez de usar a construção -execdir rm {} \; , use apenas -delete , como em:

find -iname '*.txt' -delete

Isso lida com vários casos de borda em que você pode não pensar em incluir os arquivos e diretórios de pedidos que precisam ser excluídos para não se depararem com erros. Para outros casos de uso ...

A melhor maneira de manipular os comandos em execução dos resultados de uma descoberta é geralmente usar as várias opções -exec no comando find . Em particular, você deve tentar usar -execdir sempre que possível, pois ele é executado dentro do diretório do arquivo que foi encontrado e é geralmente mais seguro (no sentido de evitar erros estúpidos sendo desastrosos) do que outras opções.

As opções -exec são seguidas pelo comando que você gostaria de executar com {} denotando o local onde o arquivo encontrado por find deve ser incluído e terminado por \; para executar o comando uma vez para cada arquivo ou + para substituir {} por uma lista de argumentos de todas as correspondências. Observe que o terminador ponto-e-vírgula é escapado de forma que não seja entendido pelo shell como um separador que leva a um novo comando.

Digamos que você esteja encontrando todos os arquivos de texto:

find -iname '*.txt' -execdir rm {} \;

Aqui está o bit relevante do manual de localização ( man find ):

   -exec command ;
          Execute  command;  true  if 0 status is returned.  All following
          arguments to find are taken to be arguments to the command until
          an  argument  consisting of ‘;’ is encountered.  The string ‘{}’
          is replaced by the current file name being processed  everywhere
          it occurs in the arguments to the command, not just in arguments
          where it is alone, as in some versions of find.  Both  of  these
          constructions might need to be escaped (with a ‘\’) or quoted to
          protect them from expansion by the shell.  See the EXAMPLES sec-
          tion for examples of the use of the -exec option.  The specified
          command is run once for each matched file.  The command is  exe-
          cuted  in  the starting directory.   There are unavoidable secu-
          rity problems surrounding use of the -exec  action;  you  should
          use the -execdir option instead.


   -exec command {} +
          This  variant  of the -exec action runs the specified command on
          the selected files, but the command line is built  by  appending
          each  selected file name at the end; the total number of invoca-
          tions of the command will  be  much  less  than  the  number  of
          matched  files.   The command line is built in much the same way
          that xargs builds its command lines.  Only one instance of  ‘{}’
          is  allowed  within the command.  The command is executed in the
          starting directory.


   -execdir command ;

   -execdir command {} +
          Like -exec, but the specified command is run from the  subdirec-
          tory  containing  the  matched  file,  which is not normally the
          directory in which you started find.  This a  much  more  secure
          method  for invoking commands, as it avoids race conditions dur-
          ing resolution of the paths to the matched files.  As  with  the
          -exec action, the ‘+’ form of -execdir will build a command line
          to process more than one matched file, but any given  invocation
          of command will only list files that exist in the same subdirec-
          tory.  If you use this option, you must ensure that  your  $PATH
          environment  variable  does  not  reference  ‘.’;  otherwise, an
          attacker can run any commands they like by leaving an  appropri-
          ately-named  file in a directory in which you will run -execdir.
          The same applies to having entries in $PATH which are  empty  or
          which are not absolute directory names.
    
por 02.12.2011 / 11:39
6

Uma alternativa é canalizar a saída e analisá-la com os comandos subsequentes. A única maneira segura de fazer isso é usar a opção -print0 , que diz a find para usar um caractere nulo como delimitador de resultados. Os comandos de recebimento devem ter a capacidade de reconhecer a entrada delimitada nula. Exemplo:

find /home/phunehehe -iregex '.*\.png$' -print0 | xargs -0 file

Observe que a opção -0 informa xargs para tratar a entrada como delimitada por nulo.

    
por 02.12.2011 / 13:00
3

O Find tem um comando delete embutido, se isso é tudo que você precisa fazer.

find . -name "*.txt" -delete

Qualquer arquivo .txt encontrado será excluído usando o comando acima.

    
por 02.12.2011 / 17:47
2

Eu estava procurando por uma resposta para isso e me deparei com essa discussão. As respostas me deram uma ideia de como eu poderia alcançá-lo. Suponha que você queira encontrar o mediainfo de todos os arquivos JPEG

Isso acrescentaria mediainfo " no início e " no final de cada arquivo correspondente (para escapar dos caracteres especiais o máximo possível), colocá-lo em um script e executar o script:

find . -name *.jpg | sed -e 's/^/mediainfo "/g;' | sed -e 's/$/"/g;' > foo.sh && sh foo.sh

Caso você esteja preocupado, algo pode dar errado, você pode ignorar o redirecionamento da saída para um arquivo e apenas ver o resultado no terminal antes de executar o script.

    
por 18.11.2017 / 23:46
1

Você pode fazer isso usando o comando xargs . xargs essencialmente executa um comando uma vez para cada instrução de sua entrada padrão. Portanto, se você precisar excluir todos os arquivos .jpg em um diretório, por exemplo, uma maneira rápida na linha de comando é:

$ find ./ -name "*.jpg" | xargs rm 

Você também pode usar o backtick (acima do botão Tab) para fazer isso (observe que esse é o caractere de backquote, não o caractere de aspas simples):

$ rm 'find ./ -name "*.jpg"'

Observe que, devido à maneira como xargs e shells processam sua entrada, o método xargs só funciona se nenhum dos nomes de arquivos e diretórios envolvidos contiver espaços em branco ou qualquer um dos \"' ; o método backquote funciona apenas se nenhum dos nomes de arquivos e diretórios envolvidos contiver espaço em branco ou qualquer um dos \[?* .

    
por 02.12.2011 / 11:39