Некоторое время назад я написал сценарий, который перемещает файлы и каталоги из Downloads
в .Downloads
, если они старше 3 дней, и удалите их из этого каталога через 30 дней. Он работает нормально, но только для файлов без пробелов.
Я исследовал и обнаружил, что команда find
, которую я использую в сценарии, не работает, как я ожидал, для любого файла или каталога с пробелами в их имени.
Вот что делает find
:
Я ожидал увидеть, что команда find
также найдет файлы с пробелами.
Вот сценарий:
#! /bin/bash
# set -x
export DISPLAY=:0.0
# true - delete, else - move
function process(){
if [ "$2" = "DELETE" ]; then
rm -r "$1" && notify-send "$3 $1 deleted!"
else
mv "$1" "../.Downloads/$1" && notify-send "$3 $1 moved to ~/.Downloads/$1!"
fi
}
# remove empty directories
for emptyDir in `find ~/Desktop/ ~/Downloads/ -empty -type d`; do
notify-send "Directoy $emptyDir was deleted, because was empty!"
done
find ~/Desktop/ ~/Downloads/ -empty -type d -delete
# remove / move old files / directorie
if [ -z "$1" ] || [ "${1,,}" != "delete" ] && [ "${1,,}" != "move" ]; then
echo "Give as parameter mode ( delete / move )"
exit
fi
if [ "${1,,}" == "delete" ]; then
day=30
path=".Downloads"
mode="DELETE"
else
day=2
path="Downloads"
mode="MOVE"
cr
if [ ! -d "~/.Downloads" ]; then
mkdir -p ~/.Downloads
fi
fi
cd ~/$path
for element in *
do
if [ -d "$element" ]; then
if [ $(find "$element" -type f -mtime -$day | wc -l) -eq 0 ]; then
process "$element" "$mode" "Directory"
fi
else
if [ $(find `pwd` -name "$element" -mtime +$day | wc -l) -gt 0 ]; then
process "$element" "$mode" "File"
fi
fi
done
Я прошу вас сказать мне, что я делаю неправильно.
Заранее спасибо!
tl; dr: это не пробелы, а скобки 1
В тесте команды find
-name
используются выражения оболочки оболочки - добавление кавычек вокруг аргумента предотвращает интерпретацию специальных символов glob вашей оболочкой , но find
по-прежнему требует их экранирования.
Пр.
$ touch 'filename with [brackets] in it'
$ find . -name 'filename with [brackets] in it'
$
(результатов нет - потому что [скобки]
означает любой отдельный символ в наборе b
, r
, a
, ] c
, k
, e
, t
, s
); тогда как
$ find . -name 'filename with \[brackets\] in it'
./filename with [brackets] in it
Если вам нужно реализовать это программно, вы могли бы использовать printf
оболочки bash для добавления требуемых экранирований:
$ element='filename with [brackets] in it'
$ find . -name "$(printf '%q' "$element")"
./filename with [brackets] in it
однако у вас БУДЕТ проблема с пробелами в строке
для emptyDir в `find ~ / Desktop / ~ / Downloads / -empty -type d`; do
существует ряд других проблем, например цитируемая ~
в [! -d "~ / .Downloads"]
не будет расширяться до $ HOME
- обычно вам следует избегать ~
в скриптах