web-dev-qa-db-fra.com

Comment utiliser l'attribut 'if' dans '-exec' de 'find'?

J'ai un répertoire avec plusieurs paires de fichiers. Chaque fichier non-texte a un partenaire de description textuelle. Par exemple, un répertoire peut ressembler à ceci:

a.jpg
a.jpg.text
b.ogv
b.ogv.text
cd ef.JpG
cd ef.JpG.text

Je veux confirmer qu'il n'y a pas de fichiers texte en vrac (fichiers de description de quelque chose déjà supprimé). J'ai donc essayé d'exécuter ce qui suit:

find . -name '*.text' -exec if [ ! -f `basename -s .text {}` ]; then echo {}; fi \;

Cependant, maintenant je reçois une erreur:

bash: syntax error near unexpected token `then'
3
v010dya

Vous ne pouvez simplement pas utiliser if à l'intérieur d'une action -exec de find de la manière que vous souhaitez. Ceci pour les raisons suivantes:

  • L'action -exec attend une commande . Cela vient de man find (quelque part sur la ligne 697 de mon terminal):
    -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.
  • if n'est pas une commande comme vous le pensez, mais un mot-clé Shell . Voir le résultat de la commande type if.

Maintenant, pour accomplir ce que vous souhaitez, vous n'avez pas vraiment besoin d'utiliser if à l'intérieur de -exec. Faites juste le test à l'intérieur de -exec, puis utilisez -print (voir man find pour plus d'informations):

find . -name '*.text' -exec $Shell -c '[ ! -f ${1%.*} ]' $Shell '{}' ';' -print

Une autre façon serait d'utiliser un script bash, comme suit:

find . -name '*.text' -exec my_if {} \;

cat ~/bin/my_if donne la sortie suivante dans mon cas:

#!/bin/bash
if [ ! -f $(basename -s .text "$1") ]; then echo "$1"; fi

Enfin, je pense que toutes les personnes utilisant normalement bash utiliseraient:

for f in *.text; do if [ ! -f $(basename -s .text "$f") ]; then echo "$f"; fi; done
4
Radu Rădeanu

Voici un moyen non optimisé d'utiliser une if dans une clause exec:

find . -name '*.text' -exec sh -c \
    'if [ ! -f "$(dirname "$1")/$(basename "$1" .text)" ]; then echo == $1; fi' sh {} \;

et voici quelque chose de bien meilleur qui ne lance pas un shell par fichier:

find . -name '*.text' -exec sh -c \
    'for i do if [ ! -f "${i%.text}" ]; then echo == $i; fi;done' sh {} +
3
jlliagre