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'
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:
-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 {} \;
où 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
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 {} +