ユーザ用ツール

サイト用ツール


shellscript

差分

このページの2つのバージョン間の差分を表示します。

この比較画面へのリンク

両方とも前のリビジョン前のリビジョン
次のリビジョン
前のリビジョン
shellscript [2019/07/08 06:24] – [テクニック] nullponshellscript [2025/10/11 16:48] (現在) nullpon
行 1: 行 1:
 ====== ShellScript ====== ====== ShellScript ======
 +
 +===== bash =====
 +
 +シェルスクリプトはbashをターゲットに作成する。POSIX互換のシェルの方がポータビリティが高いが今時bashがない環境は少ない。
 +
 +<code>
 +#!/bin/bash -eu
 +
 +...
 +</code>
 +
 +そんなbashがない環境の代表がalpine。dockerでシェルスクリプトを起動するならdebianやubuntuが良いだろう
  
 ===== shの実行オプション ===== ===== shの実行オプション =====
行 6: 行 18:
   * -x 変数を展開した状態で実際に実行されるコマンドを標準エラー出力に書き出す   * -x 変数を展開した状態で実際に実行されるコマンドを標準エラー出力に書き出す
   * -u 未定義の変数があれば処理を打ち切る   * -u 未定義の変数があれば処理を打ち切る
 +  * -C noclobberオプションを有効にする(リダイレクトでファイルを上書きしようとするとエラーになる、追記はOK)
  
 shebangに追加されることが多い shebangに追加されることが多い
行 17: 行 30:
 ===== testコマンド ===== ===== testコマンド =====
  
-<code shellscript>+<code bash>
 test -f ./hoge.txt && echo "hoge.txtあるよ" test -f ./hoge.txt && echo "hoge.txtあるよ"
  
行 42: 行 55:
 ==== [[ ]] ==== ==== [[ ]] ====
  
-bashの拡張、bashならこちらを使った方がバグが少ないので+testコマンドbash拡張。単語分割やパス展開しないので、変数をクオートで囲む必要な
  
-単語分割やパス展開しないので、変数をクオートで囲む必要ない+<code bash> 
 +HOGE="a b c"
  
-<code sh> +[ $HOGE = "a b c" ] && echo "一致"  # => Error, bash: [: too many arguments
-$ HOGE="a b c" +
-[ $HOGE == "a b c" ] && echo "一致" +
-bash: [: too many arguments +
-$ [[ $HOGE == "a b c" ]] && echo "一致" +
-一致+
  
-$ HOGE=hoge.* +"$HOGE= "a b c" ] && echo "一致"  # => 一致 
-[ $HOGE == "hoge.*" ] && echo "一致" || echo "不一致" + 
-不一致 +[[ $HOGE = "a b c" ]] && echo "一致"  # => 一致
-[[ $HOGE == "hoge.*" ]] && echo "一致" || echo "不一致" +
-一致+
 </code> </code>
 +
 +<code bash>
 +HOGE="hoge.*"
 +
 +[ $HOGE = "hoge.*" ] && echo "一致" || echo "不一致"  # => 不一致
 +
 +[ "$HOGE" = "hoge.*" ] && echo "一致" || echo "不一致"  # => 一致
 +
 +[[ $HOGE = "hoge.*" ]] && echo "一致" || echo "不一致"  # =>一致
 +</code>
 +
 +zshでは ''['' も単語分割やパス展開しない拡張版に置き換えられている
  
 ===== read ===== ===== read =====
行 65: 行 84:
   read line   read line
   echo $line   echo $line
 +
 +プロンプトを出して入力を促す
 +
 +  echo -n "User Name: "
 +  read username
 +  echo $username
 +  
 +zshでは以下のようにかける
 +
 +  read "username?User Name: "
 +  echo $username
 +  
 +入力値をマスクしたい場合はread -s(決定時の改行も出力されなくなるのでechoを挟むとよい)
 +
 +  read -s "password?Password: "
 +  tty -s && echo
 +  echo $password | shasum -a 256
 +
 +
      
 標準入力が尽きるまで行を読み取る(Ctrl-DでEOFを送信) 標準入力が尽きるまで行を読み取る(Ctrl-DでEOFを送信)
行 89: 行 127:
      
      
-===== テクニック =====+===== trap =====
  
-==== 終了時処理を実行 ====+==== シグナルのハンドリング ====
  
-例えばmktempで作成した一時ファイルをスクリプト終了時に削除するなど+<code bash> 
 +trap "echo trap sigint" INT
  
-<code> +while :; do sleep 1; done 
-hoge=/tmp/hogehoge+</code>
  
 +SIGINTをトラップすると、Ctrl-cで終了できなくなるので注意。またSIGKILLはトラップできない。
 +
 +==== EXITで終了時処理 ====
 +
 +基本はシグナルのハンドリングだが、EXITとERRという疑似シグナルが利用できる。例えばmktempで作成した一時ファイルをスクリプト終了時に削除するなどの処理をEXITと組合わせて書ける。
 +
 +<code bash>
 +hoge=/tmp/hogehoge
 mktemp $hoge mktemp $hoge
 +
  
 trap "rm $hoge" EXIT  trap "rm $hoge" EXIT 
 </code> </code>
 +  
 +===== テクニック =====
 +
 ==== スクリプトファイル名を取得 ==== ==== スクリプトファイル名を取得 ====
  
行 131: 行 182:
 0-9A-Za-zのランダムな文字列を作成する。mac用 0-9A-Za-zのランダムな文字列を作成する。mac用
  
-<code> +<code bash
-cat /dev/random | base64 | fold -w 16 | egrep -v "[+/]" | head -n 1+cat /dev/random | base64 | fold -w 16 | egrep -v "[+/]" | head -n 1 
 +</code>
  
-head -c 100 /dev/random | base64 | tr -d "+/" | head -c 16+<code bash> 
 +head -c 100 /dev/random | base64 | tr -d "+/" | head -c 16
 </code> </code>
  
 セキュアな必要がないならば セキュアな必要がないならば
  
-<code> +<code bash
-shasum <(ls -al) | cut -d " " -f 1 +shasum <(ls -al) | cut -d " " -f 1 
 </code> </code>
  
行 146: 行 199:
 ===== 変数参照 ===== ===== 変数参照 =====
  
-<code>+<code bash>
 ${HOGE:-wang}   # HOGEに値がある場合はその値が展開される。HOGEが空の場合は、wangが展開される ${HOGE:-wang}   # HOGEに値がある場合はその値が展開される。HOGEが空の場合は、wangが展開される
-${HOGE:+wang}   # HOGEに値がある場合はその値とwangが展開される。HOGEが空の場合は何もしない+${HOGE:+wang}   # HOGEに値がある場合はwangが展開される。HOGEが空の場合は何もしない
 ${HOGE:=wang}   # HOGEに値がある場合はその値が展開される。HOGEが空の場合は、wangと展開し、さらにHOGEにwangを代入 ${HOGE:=wang}   # HOGEに値がある場合はその値が展開される。HOGEが空の場合は、wangと展開し、さらにHOGEにwangを代入
 ${HOGE:?wang}   # HOGEに値がある場合はその値が展開される。HOGEが空の場合は、wangを標準エラー出力に書き出して exit 1 ${HOGE:?wang}   # HOGEに値がある場合はその値が展開される。HOGEが空の場合は、wangを標準エラー出力に書き出して exit 1
 </code> </code>
  
 +===== シェル変数の加工 =====
 +
 +指定位置で切り取り
 +
 +<code bash>
 +x=abcdefg
 +echo ${x:1:4}      # => bcde
 +</code>
 +
 +変数の値の先頭・末尾でマッチした部分を削除する
 +
 +<code bash>
 +x=hogehogefugafuga
 +
 +echo ${x#hoge}       # => hogefugafuga 前方一致削除
 +
 +echo ${x%fuga}       # => hogehogefuga 後方一致削除
 +
 +echo ${x#h*e}      # => hogefugafuga * はワイルドカード
 +
 +echo ${x##h*e}     # => fugafuga ##はワイルドカードを前方最長一致にする
 +
 +echo ${x%f*a}        # => hogehogefuga * はワイルドカード
 +
 +echo ${x%%f*a}       # => hogehoge %%はワイルドカードを後方最長一致にする
 +</code>
 +
 +文字列置換
 +
 +<code bash>
 +echo ${var/hoge/HOGE}   # => HOGEhogefugafuga
 +
 +echo ${var//hoge/HOGE}  # => HOGEHOGEfugafuga グローバルマッチ
 +</code>
 +
 +
 +使用例:まとめてリネーム、拡張子まとめて変更
 +<code bash>
 +$ ls 
 +a.txt b.txt c.txt
 +
 +$ for i in `ls`; do mv $i ${i%.txt}.php; done
 +
 +$ls 
 +a.php b.php c.php
 +</code>
 ===== パラメータ展開 ===== ===== パラメータ展開 =====
  
  
-<code>+<code bash>
 HOGE="hoge.fuga.piyo" HOGE="hoge.fuga.piyo"
 echo ${HOGE#*.}          # 最短前方一致削除 => fuga.piyo echo ${HOGE#*.}          # 最短前方一致削除 => fuga.piyo
行 170: 行 269:
 コロンは何もしないビルトインコマンド コロンは何もしないビルトインコマンド
  
-<code>+<code bash>
 : hogehoge      # 何も起こらない : hogehoge      # 何も起こらない
 </code> </code>
行 176: 行 275:
 変数HOGEが空だった場合、nyanを代入する 変数HOGEが空だった場合、nyanを代入する
  
-<code>+<code bash>
 : ${HOGE:=nyan} : ${HOGE:=nyan}
 </code> </code>
行 182: 行 281:
 変数HOGEが空だった場合、Error HOGE is emptyと出力して終了 変数HOGEが空だった場合、Error HOGE is emptyと出力して終了
  
-<code>+<code bash>
 : ${HOGE:?"Error HOGE is empty"} : ${HOGE:?"Error HOGE is empty"}
 </code> </code>
行 188: 行 287:
 可読性を考えると普通にifで書いた方がいいかも… 可読性を考えると普通にifで書いた方がいいかも…
  
-<code> +<code bash
-if [ -z "${HOGE}]; then+if [[ -z ${HOGE} ]]; then
   echo "Error HOGE is empty" >&2   echo "Error HOGE is empty" >&2
 fi fi
行 207: 行 306:
 ファイルディスクリプタ0番がterminalに紐づいている(=パイプやリダイレクトによるデータの入力がされていない) ファイルディスクリプタ0番がterminalに紐づいている(=パイプやリダイレクトによるデータの入力がされていない)
  
-<code>+<code bash>
 if [ -t 0 ]; then if [ -t 0 ]; then
    # パイプやリダイレクトされてない    # パイプやリダイレクトされてない
行 224: 行 323:
 ''$1'' ''$2'' で引数を取得できる。引数の数は ''$#'' で取得できる。引数の数が2桁になる場合は ''${10}'' のように {} で括る。 ''$1'' ''$2'' で引数を取得できる。引数の数は ''$#'' で取得できる。引数の数が2桁になる場合は ''${10}'' のように {} で括る。
  
-<code>+<code bash>
 # arg1.sh # arg1.sh
 echo $# echo $#
行 231: 行 330:
 </code> </code>
  
-<code>+<code bash>
 $ ./arg1.sh a b c d $ ./arg1.sh a b c d
 4 4
行 241: 行 340:
 ''$@'' または ''$*'' でシェルスクリプトの引数全体を取得できる。  ''$@'' または ''$*'' でシェルスクリプトの引数全体を取得できる。 
  
-<code>+<code bash>
 # arg2.sh # arg2.sh
 echo $@ echo $@
行 247: 行 346:
 </code> </code>
  
-<code>+<code bash>
 $ ./arg2.sh 1 2 3 $ ./arg2.sh 1 2 3
 1 2 3 1 2 3
行 253: 行 352:
 </code> </code>
  
-''$@'' と ''$*'' はダブルクオートで括った場合の動作が異なる。''$*''引数が一つの値にまとめられてしまう。ほとんどの場合、''$@'' の挙動が求められるので、''$@'' を使っておけば間違いない+''$@'' と ''$*'' はダブルクオートで括った場合の動作が異なる。''"$@"''は''"1" "2" "3"''、''"$*"''は''"1 2 3"''と展開される。
  
-<code>+ 
 +''$*'' は引数が一つの値にまとめられてしまう。ほとんどの場合、''$@'' の挙動が求められるので、''$@'' を使っておけば間違いない 
 + 
 +<code bash>
 # arg3.sh # arg3.sh
 node -p -e 'process.argv.slice(1).join()' -- "$*" node -p -e 'process.argv.slice(1).join()' -- "$*"
行 264: 行 366:
 </code> </code>
  
-<code>+<code bash>
 $ ./arg3.sh 1 2 3 $ ./arg3.sh 1 2 3
 1 2 3 1 2 3
行 274: 行 376:
 ''$*'' をダブルクオートで括った場合、空白スペースで結合される。結合文字は環境変数IFSで変更できるが影響範囲が大きいのであまり変えない方が良いだろう。 ''$*'' をダブルクオートで括った場合、空白スペースで結合される。結合文字は環境変数IFSで変更できるが影響範囲が大きいのであまり変えない方が良いだろう。
  
-<code>+<code bash>
 # arg4.sh # arg4.sh
 echo "$*"  echo "$*" 
行 281: 行 383:
 </code> </code>
  
-<code>+<code bash>
 $ ./arg4.sh 1 2 3 $ ./arg4.sh 1 2 3
 1 2 3 1 2 3
行 288: 行 390:
  
 これらの挙動はシェル関数の引数でも同様である。 これらの挙動はシェル関数の引数でも同様である。
 +
 +===== ヒアドキュメント =====
 +
 +変数に格納
 +
 +<code bash>
 +val=$(cat << 'EOS'
 +hoge
 +fuga
 +piyo
 +EOS
 +)
 +</code>
 +
 +
 +標準入力に渡す
 +
 +<code bash>
 +while read -r line
 +do
 +  echo $line
 +done << 'EOS'
 +hoge
 +fuga
 +piyo
 +EOS
 +)
 +</code>
 +
 +
 +一度変数を経由する。''<<<''はヒアストリングと呼ばれるものでPOSIXには定義されておらずshでは使用できない。bashやzshで使用できる
 +
 +<code bash>
 +val=$(cat << 'EOS'
 +hoge
 +fuga
 +piyo
 +EOS
 +)
 +
 +while read -r line
 +do
 +  echo $line
 +done <<< $val
 +</code>
 +
 +クオートの有無は変数展開するかしないか
 +
 +<code bash>
 +val='test'
 +
 +while read -r line
 +do
 +    echo $line
 +done << "EOF"
 +${val}
 +${val}
 +${val}
 +EOF
 +</code>
 +
 +クオートありの場合は変数展開されずに以下のように出力される
 +
 +<code>
 +${val}
 +${val}
 +${val}
 +</code>
 +
shellscript.1562567041.txt.gz · 最終更新: by nullpon