J'essaie de remplacer des textes dans des fichiers texte par sed
mais je ne sais pas comment le faire avec plusieurs fichiers.
J'utilise:
sed -i -- 's/SOME_TEXT/SOME_TEXT_TO_REPLACE/g /path/to/file/target_text_file
Avant de partir avec plusieurs fichiers, j’imprimais les chemins des fichiers texte ciblés dans un fichier texte avec cette commande:
find /path/to/files/ -name "target_text_file" > /home/user/Desktop/target_files_list.txt
Maintenant, je veux exécuter sed
en fonction de target_files_list.txt
.
Vous pouvez parcourir le fichier en utilisant la boucle while ... do
:
$ while read i; do printf "Current line: %s\n" "$i"; done < target_files_list.txt
Dans votre cas, vous devez remplacer printf ...
par la commande sed
que vous souhaitez.
$ while read i; do sed -i -- 's/SOME_TEXT/SOME_TEXT_TO_REPLACE/g' "$i"; done < target_files_list.txt
Toutefois, notez que vous pouvez obtenir ce que vous voulez en utilisant uniquement find
:
$ find /path/to/files/ -name "target_text_file" -exec sed -i -- 's/SOME_TEXT/SOME_TEXT_TO_REPLACE/g' {} \;
Vous pouvez en savoir plus sur l'option -exec
en exécutant man find | less '+/-exec '
:
-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 section for examples of the use of the -exec option. The specified command is run once for each matched file. The command is executed in the starting directory. There are unavoidable security problems surrounding use of the -exec action; you should use the -execdir option instead.
Comme correctement noté par les utilisateurs terdon et dessert dans les commentaires, il est nécessaire d'utiliser -r
avec read
car il gérera correctement les barres obliques inverses. C'est aussi rapporté par shellcheck
:
$ cat << EOF >> do.sh
#!/usr/bin/env sh
while read i; do printf "$i\n"; done < target_files_list.txt
EOF
$ ~/.cabal/bin/shellcheck do.sh
In do.sh line 2:
while read i; do printf "\n"; done < target_files_list.txt
^-- SC2162: read without -r will mangle backslashes.
Donc ça devrait être:
$ while read -r i; do sed -i -- 's/SOME_TEXT/SOME_TEXT_TO_REPLACE/g' "$i"; done < target_files_list.txt
Une solution consisterait à utiliser xargs
:
xargs -a target_files_list.txt -d '\n' sed -i -- 's/SOME_TEXT/TEXT_TO_REPLACE/g'
De man xargs
:
-a file, --arg-file=file
Read items from file instead of standard input.
--delimiter=delim, -d delim
Input items are terminated by the specified character. The
specified delimiter may be a single character, a C-style charac‐
ter escape such as \n, or an octal or hexadecimal escape code.
Utilisez simplement une boucle for
.
IFS=$'\n' # Very important! Splits files on newline instead of space.
for file in $(cat files.txt); do
sed ...
done
Notez que vous rencontrerez des problèmes si vous rencontrez des fichiers avec des nouvelles lignes (!) Dans leurs noms. (: