X



シェルスクリプト総合 その26 [無断転載禁止]©2ch.net
■ このスレッドは過去ログ倉庫に格納されています
0001名無しさん@お腹いっぱい。
垢版 |
2017/03/20(月) 12:07:26.41
シェルスクリプトの総合スレです。
□お約束
・特記なき場合はBourne Shell(/bin/sh)がデフォルトです。
 bash/zsh/ksh/ashなどに依存する場合は明示しましょう。
 Linuxユーザは/bin/shの正体がbashまたはdashなので特に注意。
 FreeBSDユーザは/bin/shの正体がashなので注意。
 v7 shに一番近くて、現役のshは、OpenSolaris由来のheirloom sh。
  http://src.illumos.org/source/xref/illumos-gate/usr/src/cmd/sh/
  http://heirloom.sourceforge.net/sh.html
・csh/tcshのシェルスクリプトは推奨されません。
 (理由は「csh-whynot」でググれ)
・UNIXにはシェルスクリプトに便利な小さなコマンドがいろいろあります。
 manや参考リンクを見ましょう。
 aproposないしはman -kでそれらしい単語による簡単な検索もできます。
・シェルで使えるワイルドカード等は正規表現ではありません。
 正規表現の話題はスレ違い(正規表現スレへ)
・シェルスクリプトのことをシェルってゆーな
□初心者へのアドバイス:
・適した道具を判断するのも頭の重要な使い方。シェルスクリプトよりも
 awkまたはperlの方が適した処理にはそちらを使いましょう。
・知らないコマンドが出てきたらmanを引きましょう。
・思い通りに動かないときは、まずは sh -x でトレースしましょう。
□回答者への注意事項:
・シェルスクリプトでの処理方法を質問しているのに、よくわからずに
 「そういうのはperl使いましょう」と回答するのはやめましょう。
 安易にperlに逃げずにシェルスクリプトで処理するのが頭のいいやり方。
前スレ
シェルスクリプト総合 その25
http://echo.2ch.net/test/read.cgi/unix/1439563321/
0587586
垢版 |
2018/03/06(火) 23:29:38.39
5行目あたりの日本語がおかしい。けど国語力ないので自然に直すの無理。
察してくれ。
0588名無しさん@お腹いっぱい。
垢版 |
2018/03/06(火) 23:43:28.43
>>586
なるほど!
#!/bin/sh が必ずbashにリンクされてるとは限らないから、
拡張表現で書かなくて済む場合は確かに可搬性を優先した方がいいですね。

ありがとうございました。
0589名無しさん@お腹いっぱい。
垢版 |
2018/03/06(火) 23:49:32.51
>>583
mdfindを知らんので問題解決に直接ならないけれど`done < ${2}`ではなく`done < "$2"`のほうがいいよ〜
もっと言うなら`cat "$2" | while read LINE;`のほうがいいかもね〜
* POSIX sh互換シェルは変数名に数字が来たらそこで読み取りを中断するので波括弧で変数名の範囲を明示しなくていい
* 変数$LINEに^Dとかが渡された時catコマンドと<リダイレクトで違いがある
短いんだけど参考: http://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html#tag_18_07_01
0590588
垢版 |
2018/03/06(火) 23:58:01.42
あー!
さらに調べたら、よくわかりました。
#!/bin/sh でbashを起動したらbashの拡張とか使えなくなっちゃうんですね。
だからシェバンを書き換える必要があると…
そういう意味だったんですね。
(ちゃんとmanにかいてありました)

ためになりました。
0591名無しさん@お腹いっぱい。
垢版 |
2018/03/07(水) 07:20:57.39
>>589
ん?どゆこと?
$ cat file | cmd
$ cmd < file
で違いがあるってこと?そうは思えないんだけど…
0594watarie
垢版 |
2018/03/07(水) 09:23:48.67
>>589さん
0595watarie
垢版 |
2018/03/07(水) 09:32:33.42
>>589さん
ありがとうございます。ちょっと英語で読解が難しいので時間を作って該当の箇所を読ませていただこうかと思います。
{2}の箇所を"2"に変更しました。
<の箇所をcatに変更しました。
非常に目に優しいくなりました。
読み込む箇所なので、怪しいと感じます。結果としては変わりませんでしたが、参照の箇所は興味がありますので読ませていただこうと思っています。英語で弱気になりますが^^;
ありがとうございます。
0596watarie
垢版 |
2018/03/07(水) 09:37:28.43
>>593さん
すみません。確かにそうですよね。
初心者過ぎで逃げを当然にしてたバチがあたったようです。
吐き出したエラーは、
cp: /Volumes/Macintosh: No such file or directory
cp: HD: No such file or directory
などのファイルパスが、スペース区切りになってcp出来ませんでしたよというエラーでした。
ただ、検索に漏れた該当のファイルは何度めかにはヒットしますので上記のエラーとはならなかったということに。
spotlight特有のものだからと諦めるしかないのか・・・

これがひょっとしたら手がかりになるかもしれませんが、わたしの力では全く・・・
0597watarie
垢版 |
2018/03/07(水) 09:45:34.86
すみません、mdfindの検索結果のファイルパスが
スペースで割れてしまってcpに渡されたということでした。たぶん^^;
0598watarie
垢版 |
2018/03/07(水) 12:55:37.74
解けました!ありがとうございます!
ダブルコーテーションを%の箇所で囲うのではなく、
1行取り出した先でパイプして囲えば良いと思い以下を試したところ出来ました。
mdfind "kMDItemFSName == "${LINE}" || kMDItemDisplayName == "${LINE%.*}"" | awk 'NR==1' | sed -e 's/^\(.*\)$/\"\1\"/' | xargs -J % cp -p % "$1" 2>/Users/mac10/Desktop/out
エラー出力素晴らしいです!
ありがとうございました!
0601名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 00:41:50.66
>>575
今更だけど、あるんだよね。誰も教えたくなかったんだね。
0603名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 07:30:07.37
ヒントくらい出すか……。でもあいつに教えたくないんだよなぁ。
卑屈だけど、気持は分かるだろ?
パイプに名前(ry
0604名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 16:44:02.99
それ使うと出力側はこんな感じかな
while :;do sed -e 's/\(.*\)/^[[32m\1^[[m/' np1 & sed -e 's/\(.*\)/^[[33m\1^[[m/' np2 >&2; done
0605名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 20:31:17.47
>>604
野暮な指摘だが、sedを使わない
printf '\e[32m%s\[0m' "$(cat)"
の方が分かりやすくね? つーか printf ってそういう「整形した出力」の為のコマンドだろうし。
0606604
垢版 |
2018/03/08(木) 20:32:49.31
>>695
ミスった… printf '\e[32m%s\e[0m' "$(cat)" だ。
こうすると "$(cat)" の部分で np1 を受けt[以下略]
0607名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 20:51:46.21
>>603
名前付きパイプ(mkfifo)はプロセスが2個あって
片方が出力側プロセス、片方側が入力側プロセスに
別れない限りストリーミングにすることができない
(同一プロセスでは出力しながら入力ができないため)

今回の場合、標準出力と標準エラー出力の2つがあるから
出力側プロセス1個と入力側プロセス2個の合計3プロセスが必要になる
つまり別プロセスとして起動しなければいけない

もちろん出力の完了待ちのwaitも必要だし、作成したパイプの削除も必要。
特にCTRL+Cで中断された場合とか
また万全を期してパイプ名が被ることも考慮してランダムな名前にしたい

ここまで言えば名前付きパイプを使った方法は実装するのに行数がかかり
面倒な方法だと言うことがわかるだろう?

また>>554で書いたとおりbusyboxで動かす必要があるのだが、
そもそもbusyboxでmkfifoが存在しない場合もある
↑俺にとってはこっちのほうが重要
0611名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 21:57:56.36
うわ。まだ居たのかこのキ​チガイ…
まあ放置が一番いいな。俺のこのレスも彼を発狂させる原因になるやもしれんし。
0612名無しさん@お腹いっぱい。
垢版 |
2018/03/08(木) 22:15:10.19
まあ反論なく放置される(=俺の勝ち逃げ)という状況ってのが
俺にとっては一番であることには間違いないが


で俺がレスしたのは、放置してないからだよ?w
0613604
垢版 |
2018/03/08(木) 22:33:08.10
>>606
あ、しまった printf '\e' は GNU/Linux じゃないと動かないみたい。
Debian使ってんのがバレてしまったw
FreeBSDとかでも動作させる為には printf '\033' とやる必要がある。
0619名無しさん@お腹いっぱい。
垢版 |
2018/03/09(金) 23:00:06.75
そういや>>616で思い出したけど、
busyboxのsedってなんか動きがおかしいんだよね。

(echo 1; sleep 5; echo 2; sleep 5) | sed 's/a/a/'

例えばこれを実行すると、1が表示されるのは5秒後になる
debianだとすぐに表示されるのに。一行遅れて表示される

だからsedは今回の俺の用途には使えなくてreadを使ってる
0622名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 01:06:54.08
> openSUSE の busybox では記述通りの動きでした。
もしかして、sedはbusyboxじゃない方のsedになってない?

WSL上のUbuntuに入れたbusyboxのsed
↓つまりこれを実行するとやっぱり5秒後に1と表示される
(echo 1; sleep 5; echo 2; sleep 5) | busybox sed 's/a/a/'

やっぱりbusyboxが実装してる簡易版sedの問題な気がする
0625名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 09:07:03.16
おーほんとだ。Busybox v1.22.1 on Debianでも同じような動きになる。
ここまでハッキリしたバグに遭遇したの久し振りだわ
単に離れてただけかもしれんが
0626名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 09:38:16.90
なにやら興味がある人がいるようでw

なんで俺の用途で使えないかというと、例えばこんな感じに
ユーザーのインタラクティブな入力を変換して出力したいから

↓ >(実際には画面に表示されない) の行がユーザーの入力

$ sh | sed 's/a/a/'
> echo 1
> echo 2
1
> echo 3
2

見ての通りユーザーが入力したコマンドの結果が一つ遅れて表示される
sedのよくあるユースケースでは問題ないんだろうけどね
0627名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 12:15:21.67
>>626
挙動が変わるツールは珍しいなと思いました。
busybox ash の中で sh | sed 's/a/a/' とすると、
Debian (v1.22.1) まとも
Ubuntu (v1.22.1) 遅れる
openSUSE (v1.21.1) まとも(sh プロンプトが表示される)
CentOS (v1.15.1) まとも(sh プロンプトが表示される)
Windows (v1.29.0) 遅れる
でした。
ビルド時に引いているライブラリの差異なのでしょうか...
0628名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 12:28:15.59
バッファリングが影響してそうなのかもしれないけど
正規表現の実装の都合な気もする。
はたまたカーネルが関係してるのかシェルの設定でなんかあるのか

正直、そういう違いが有るんだね。気をつけておくよと
事実を受け止めるぐらいでで深追いする気力はないw

話変わるけど、何もしない関数の定義、
foo() :

もしくは、echoするだけの関数の定義
foo(): echo foo

この書き方ってPOSIX準拠だよね?
bashやzshではエラーになった
0630名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 18:45:48.71
>>628
POSIX準拠なのかはよくわかりませんが、sh ならば

foo()
foo1() echo "$1"

で、sh でも bash でも動かすなら

foo() { :; }
foo1() { echo "$1"; }

でいけると思います。
0631名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 23:10:12.04
訂正
× foo(): echo foo
○ foo() echo foo


>>629
ごめん、zshでは試していなかった。
今zsh入れてたマシン壊れてるんだったw
別の環境にzsh入れたらたしかに動いたね

dash、ash、zsh では動いた
bashだけ動かなかった
0632名無しさん@お腹いっぱい。
垢版 |
2018/03/10(土) 23:17:51.17
func() :
↑これ確かにシェバンを #! /bin/sh にすると通るけど
#! /usr/bin/env bash にすると撥ねられるな。
http://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html#tag_18_09_05
一応現行POSIXのシェル関数の定義はここに書いてあるけど、func() : は恐らく適合してるな。
多分bashの文法チェックが少しばかりおせっかいすぎるんだろう。
でもPOSIXに準拠した記法なのにエラーになるって嫌だなぁ
0633632
垢版 |
2018/03/10(土) 23:21:43.98
あれ。yashでも「syntax error: a function body must be a compound command」って撥ねられるわ。
yashはかなり厳密にPOSIXに従ってる筈なんだけど…
0634名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 00:25:29.87
またちょうどよくシバンの話がw

シェルスクリプトのファイルを直接実行するのではなく
ライブラリとして他のスクリプトから読み込むだけの場合
シバンいらないよね? またその場合の拡張子ってどうしてる?

シンタックスハイライトとか自動で適用させたいんだけど
余計なものはできるだけ排除したい
0635名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 00:28:14.22
>>632
そうなんだよね。あと

> fname ( ) compound-command [io-redirect ...]

ここにio-redirectが書けたのがびっくり
どういう時に使えば良いんだろうか?
0636632
垢版 |
2018/03/11(日) 01:05:34.01
>>635

俺は未インストールコマンドの処理に使ってる

check_command()
{
type "$1" > /dev/null 2>&1 || {
printf '%s%s\n' \
"$1" 'をインストールして下さい。'
exit 1
} >&2

return $?
}

check_command "curl"

こうするとエラーメッセージが標準出力に出ないからパイプ繋いでてもおかしなことになりにくい。
もっとも |& ←みたいなパイプなら無理だけど、そもそも標準エラー出力ごとパイプに使うなんて稀だろうという推測。
0637名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 01:46:36.03
>>636
いや、そうではなくて関数定義の最後に書けるんだなーって話

その例で言えば、こういう書き方ってこと
(関数全体を別の所に出力したいときには便利か、うーん使うかな?)

check_command()
{
 type "$1" > /dev/null 2>&1 || {
  printf '%sをインストールして下さい。\n' "$1"
  exit 1
 }
} >&2

# ついでにどうでもいい所を俺の趣味で変更したw
# exitもreturnに変えたい。もしくはexit_if_command_not_foundみたいな関数名にしたい


> もっとも |& ←みたいなパイプなら無理だけど、そもそも標準エラー出力ごとパイプに使うなんて稀だろうという推測。

/dev/ttyに出力するって手もあるよ

$ (echo stdout-a; echo stderr-a >&2; echo tty-a >/dev/tty) |& sed 's/a/b/'
tty-a
stdout-b
stderr-b
0638名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 02:00:39.65
>>633
yash覚えた

最低限機能しかももってないと思われるashで動けば
Bourne Shell 系全部でうごくだろうと思っていたから
そういうこともあるんだなーって思った
0639632
垢版 |
2018/03/11(日) 02:47:31.38
>>637
あ,そういうことか。勘違いしてたわw それはちょっと使い道思い付かん

ところで,
return $?
↑これ消した理由なに? C言語をやってるせいか関数が終了ステータスを返さないと不安で仕方無いんだけども,
シェルスクリプトだと要らないのかな。今まで特にreturn行のせいで不具合が発生したこともなかったけれど。
0640名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 11:42:23.58
>>639
何も書いてなければ、その時点の$?がそのまま返るだろ?

> C言語をやってるせいか関数が終了ステータスを返さないと不安で仕方無いんだけども,
Ruby(最後に評価された値を返す)やってるとreturnなんて書かないんだがw
0641名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 11:44:13.17
> 今まで特にreturn行のせいで不具合が発生したこともなかったけれど。
まあ不具合は発生しないだろうね。

関数の最後でなにもしないなら、その時点の $? が返る所を
$?を取り出してreturnの引数で$?を返すようにしただけだからw
0645名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 13:59:18.74
Cはハードに近いところを表現するのに適してるから絶対に無くならないけどね。
化石だけど生活には欠かせない燃料みたいなもんかな。
Cで書かれてるシェルのスレで否定しちゃうのはちょっと痛いね。
0646名無しさん@お腹いっぱい。
垢版 |
2018/03/11(日) 14:52:52.42
俺はCを否定してないぞ。C以外を使っていれば
returnしなくていい言語だってあることを知るだろうし、
returnを書くことに拘る必要はないってことだ
0652名無しさん@お腹いっぱい。
垢版 |
2018/03/12(月) 09:24:03.09
話をぶった切ってすまんが、なるべく依存関係が少ない・かつインタラクティブがある程度機能的なPOSIX互換シェルって何があるかな。
ksh98とかyashとかは使ったことあるけど、ああいう感じでもうちょい開発が盛んなやつとか知らない?
0653名無しさん@お腹いっぱい。
垢版 |
2018/03/12(月) 17:35:16.53
連続足し算だけど
#!/bin/bash

while true
do
add=expr 0
read x
add=$(($add + $x))
echo '----------'
echo ' ' $add
done

でできるが
line 5: 0: command not found
1
----------
1
./tasizan-renzoku.bat: line 5: 0: command not found
2
----------
3
./tasizan-renzoku.bat: line 5: 0: command not found

と 気色悪い。 どうすれば正解なの?
0655名無しさん@お腹いっぱい。
垢版 |
2018/03/12(月) 17:48:20.94
すごいね、以下でエラーがなくなった。

#!/bin/bash
while true
do
add=(expr 0)
read x
add=$(($add + $x))
echo '----------'
echo ' ' $add
done
0656名無しさん@お腹いっぱい。
垢版 |
2018/03/12(月) 17:49:39.62
エラーがなくなった が 足し算の合計もなくなった
0657名無しさん@お腹いっぱい。
垢版 |
2018/03/12(月) 17:58:34.26
a=0
while read x; do
a=$((a + x))
echo ----------
echo ' ' $a
done
0658名無しさん@お腹いっぱい。
垢版 |
2018/03/12(月) 18:02:05.44
ありがとう。 合計を外側で定義することがわからなかった。
こうなると 電卓より便利!
0664名無しさん@お腹いっぱい。
垢版 |
2018/03/17(土) 03:48:34.81
for i in `seq -f %02g 1 10`

do
echo ${i}
done


このfor文の中で01の時に1、02の時に2を指定する方法ってありませんか?
簡潔に言うと先頭の0を取ったものを使いたいです
0665名無しさん@お腹いっぱい。
垢版 |
2018/03/17(土) 08:33:32.12
>>662
どうしてデバッグが楽なのか知らんが、
少なくともprintfデバッグすると
終了コードが0に変わるからデバッグしづらいな
0672名無しさん@お腹いっぱい。
垢版 |
2018/03/17(土) 23:05:13.47
> 拡張POSIX準拠らしいけど

拡張POSIXってしう仕様があるの?
POSIXを拡張したもので、POSIX非準拠って意味だと思ってたんだけど?
0673名無しさん@お腹いっぱい。
垢版 |
2018/03/18(日) 00:55:12.66
シェルスクリプトってさ、POSIXが標準だけど
実質bashがデファクトスタンダードって考えて良いのかな?
zshやfishを使っていてもbashはインストールされていると考えていい?

いや、あるコマンド作ろうと思ってるんだけど、
基本はPOSIXで動くようにするけど、bashがインストールされていれば
拡張機能が使えるようになる。という仕組みは
ほぼすべての人が拡張機能使えると考えて良いんだろうかなって思って
0676名無しさん@お腹いっぱい。
垢版 |
2018/03/18(日) 01:15:31.51
>>674
いやbash前提じゃないんだ。POSIX シェル前提。POSIX シェルだけで動く。
だけどbashがインストールされていれば拡張機能が使える
その拡張機能っていうのも、本質的な機能じゃなくて
なんていうかな、同じコマンドで実行結果も同じだけど情報が詳細に取れるって感じ

BSD系といってもMacはデフォルトでbashになったのは知ってる。
パスに関してはシバンではなくbashコマンド経由で実行すれば良いと思ってる
(切替可能なようにするのでどっちみちシバンには頼らないと思うし)
0678名無しさん@お腹いっぱい。
垢版 |
2018/03/18(日) 09:35:45.26
POSIX規定外のことをPOSIX拡張って書くのは誤解のもとだからやめた方がいい。
伝統的にシェルにはなくてPOSIXで増えた機能のことをPOSIX拡張とよぶことがあるので、意味が真逆。
0680名無しさん@お腹いっぱい。
垢版 |
2018/03/18(日) 12:02:31.33
  ★★★The● best way to eliminate too much gap between rich and poor, is to decide the tax● rate of the progressive tax in the referendum(Root Tax).★★★

  この掲示板(万有サロン)に優●秀な書き込みをして、総額148万円の賞金をゲットしよう!(*^^)v
●  http://jbbs.livedoor.jp/study/3729/ →リンクが不良なら、検索窓に入●れる!
0683名無しさん@お腹いっぱい。
垢版 |
2018/03/18(日) 15:53:34.91
>>682
見分け方は本質的なところじゃないんで
俺にとってはどうでもいい話だけど、
シェルの判定ではなくて機能チェックで見分けるよ

ブラウザをUserAgentで判定するのではなく
使いたい機能が使えるかどうかで判定するのと似たようなやり方ね

> ps -p $$ -o comm=
ちなみにそれcygwinではエラーになった
0684682
垢版 |
2018/03/18(日) 22:58:03.92
>>683
bash や zsh の機能を多く使うならば、その都度機能が動くかどうか判定するより、
起動シェルを見て分岐したほうが手間がないという意味合いで書きました。

cygwin の ps は -o オプションは無いのですね。失礼しました。
freebsd の環境で hoge.sh に ps -p $$ -o comm= と書いて、
/usr/local/bin/bash ./hoge.sh とすると、bash と表示されます。
(debian とかの ps でも ok)

cygwin の /bin/sh は /bin/bash なのでシェルを見るよりその都度使用する機能を判定するほうがよさそうですね。
0685名無しさん@お腹いっぱい。
垢版 |
2018/03/19(月) 06:19:05.05
> > ps -p $$ -o comm=
> ちなみにそれcygwinではエラーになった

マジで!? POSIXでも定義されてるんだけどなぁ
0686名無しさん@お腹いっぱい。
垢版 |
2018/03/19(月) 14:02:30.98
壊れた動画探しに
ffmpeg -i input -f null /dev/null 2>&1 | grep -m 1 -Ii error
ffmpeg -i input -f null /dev/null 2>&1 | grep -q -i error
とやったのですけれど、grepにヒットしてもffmpegの動画読み込みが止まりません。
grepにヒットしたらffmpegを止めるにはどうすればいいのか。
いいアイデアはないですか。
■ このスレッドは過去ログ倉庫に格納されています

ニューススポーツなんでも実況