問題のあるシェルスクリプト
あなたがよりよく理解できるように私がやろうとしていることを説明します。私のディレクトリに100個の.torrentファイルがあるとしましょう。 BitTorrent クライアントに追加すると、そのうちの 2 つがそれぞれ xxx.epub と yyy.epub をダウンロードしますが、100 個のうちの 2 つかどうかはわかりません。
したがって、私のスクリプトが行うことは、(1)find
すべての.torrentファイルを繰り返し、pwd
各.torrentファイルを渡すことです。その後、transmission-show
.torrentファイルを解析し、人間が読める形式でメタデータを出力します。次にそれを使用して、急流はawk
ダウンロードするファイル名を取得し、それをリスト.txtに対して実行します。これには、私たちが探しているファイル名(xxx.epubとyyy.epub)が含まれています。
文書: findtor-array.sh
#! /bin/bash
#
# Search .torrent file based on 'Name' field.
#
# USAGE:
# cd ~/myspace # location of .torrent files
# Run `findtor ~/list.txt` (if `findtor.sh` is placed in `~/bin` or `~/.local/bin`)
# Turn the list of file names from ~/list.txt (or any file passed as argument) into an array
readarray -t FILE_NAMES_TO_SEARCH < "$1"
# For each file name from the list...
for FILE_NAME in "${FILE_NAMES_TO_SEARCH[@]}"
do
# In `pwd` and 1 directory-level under, look for .torrent files and search them for the file name
find . -maxdepth 2 -name '*.torrent' -type f -exec bash -c "transmission-show \"\$1\" | awk '/^Name\: / || /^File\: /' | awk -F ': ' '\$2 ~ \"$FILE_NAME\" {getline; print}'" _ {} \; >> ~/torrents.txt
# The `transmission-show` command included in `find`, on it own, for clarity:
# transmission-show xxx.torrent | awk '/^Name: / || /^File: /' | awk -F ': ' '$2 ~ "SEARCH STRING" {getline; print}'
done
私はそのプロセスが単純で正しく行われていると思います(確認していないことを除いて)。しかし、タスク全体がスクリプトに比べて多すぎるようです。スクリプトを実行してから一定時間が経過すると、I Ctrl+ Citまでこれらのエラーが発生し続けるためです。
_: -c: line 0: unexpected EOF while looking for matching `"'
_: -c: line 1: syntax error: unexpected end of file
これらの「スケーリング」の問題はありますか?私が見逃している部分は何であり、それを解決するにはどうすればよいですか?
答え1
FILE_NAME
コマンドオプションに直接渡されますbash -c
。引用符/シェルコードを含めると、問題が発生する可能性があります。実際、-exec
find
FILE_NAME
任意のコード実行可能。例: この特別なケースでは、入力ファイルに次の行を含めることができます。'; echo "run commands";'
代わりにbash -c
ループvarを位置引数として渡してください。たとえば、
find . -maxdepth 2 -name '*.torrent' -type f -exec sh -c '
transmission-show "$2" |
awk -v search="$1" '\''/^Name: / {name = substr($0,7)} /^File: / && name ~ search {print; exit}'\' \
_ "$FILE_NAME" {} \;
また、各ファイルのすべてのクエリを繰り返すことは非効率的です。ファイルを繰り返して、次のように検索することを検討してくださいgrep -f file
。
find . -maxdepth 2 -name '*.torrent' -type f -exec sh -c '
file=$1
shift
if transmission-show "$file" | head -n 1 | cut -d" " -f2- | grep -q "$@"; then
printf "%s\n" "$file"
fi' _ {} "$@" \;
またはfind
:
for file in *.torrent */*.torrent; do
if transmission-show "$file" | head -n 1 | cut -d' ' -f2- | grep -q "$@"; then
printf '%s\n' "$file"
fi
done
- 上記はすべての引数をに渡すので、
grep
使用法は固定文字列などのfindtor -f ~/list.txt
リストからパターンを取得することです。-F
-e expression
答え2
@Kusalanandaの提案、回答(@guestと@Jetchisel)に基づいています。Kevinの詳細な回答、私はこれを思い出しました:
#! /bin/bash
#
# Search for 'Name' field match in torrent metadata for all .torrent files in
# current directory and directories 1-level below.
#
# USAGE e.g.:
# cd ~/torrent-files # location of .torrent files
# Run `~/findtor.sh ~/list.txt`
# Get one file name at a time ($FILE_NAME_TO_SEARCH) to search for from list.txt
# provided as argument to this script.
while IFS= read -r FILE_NAME_TO_SEARCH; do
# `find` .torrent files in current directory and directories 1-level under
# it. `-print0` to print the full file name on the standard output, followed
# by a null character (instead of the newline character that `-print` uses).
#
# While that's happening, we'll again use read, this time to pass one
# .torrent file at a time (from output of `find`) to `transmission-show`
# for the latter to output the metadata of the torrent file, followed by
# `awk` commands to look for the file name match ($FILE_NAME_TO_SEARCH) from
# list.txt.
find . -maxdepth 2 -name '*.torrent' -type f -print0 |
while IFS= read -r -d '' TORRENT_NAME; do
transmission-show "$TORRENT_NAME" | awk '/^Name: / || /^File: /' | awk -F ': ' -v search_string="$FILE_NAME_TO_SEARCH" '$2 ~ search_string {getline; print}';
done >> ~/torrents-found.txt
done < "$1"
私はこれを実行しましたが、これまではうまくいくようです。参加してくださった皆さんに心から感謝します!
ベストを尽くしましたが、修正や追加の提案を歓迎します。
答え3
私はこのように書きます。
#!/usr/bin/env bash
pattern_file="$1"
while IFS= read -r -d '' file; do
transmission-show "$file" | awk .... "$pattern_file" ##: Figure out how to do the awk with a file rather than looping through an array.
done < <(find . -maxdepth 2 -name '*.torrent' -type f -print0)
これは引用地獄を避けるべきです:-)
まあ、そうではないかもしれませんnullglob
。
編集する:
find コマンドを試して、元のスクリプトと一緒に使用します。
find . -maxdepth 2 -name '*.torrent' -type f -exec bash -c 'transmission-show "$1" | awk "/^Name\: / || /^File\: /" | awk -F ": " "\$2 ~ \"$FILE_NAME\" {getline; print}"' _ {} + >> ~/torrents.txt