シェルスクリプト総合 その27
■ このスレッドは過去ログ倉庫に格納されています
シェルスクリプトの総合スレです。 □お約束 ・特記なき場合はBourne Shell(/bin/sh)もしくはPOSIX準拠の互換シェルがデフォルトです。 bash/zsh/ksh/ash/dash/yash/poshなどの専用機能に依存する場合は明示しましょう。 Linuxユーザは/bin/shの正体がbashまたはdashなので特に注意。 FreeBSDユーザは/bin/shの正体がashなので注意。 ・POSIXについてのリンクは https://en.wikipedia.org/wiki/POSIX にまとめられています 最新の仕様はこちらへ http://pubs.opengroup.org/onlinepubs/9699919799/ (左上の「Shell & Utilities」 から参照することができます。) ・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に逃げずにシェルスクリプトで処理するのが頭のいいやり方。 質問に対して問題が間違ってるといちゃもんをつけるのもやめましょう 前スレ シェルスクリプト総合 その26 https://mevius.5ch.net/test/read.cgi/unix/1489979246/ >>37 ,38 それは知っていてのだったがな そこここにはそれ流があると思うよ。LinuxとかBSDのソース見たらいまだにC流で短かったりするし。シェルスクリプトなら>>20 っていう意見があってもいいだろう、そもそもいろいろ記号的なもんだから なのにいきなりオブジェクト指向とか言い出し馬鹿にするのがTPOをわきまえない馬鹿だなっていう シェルスクリプトがオブジェクト指向ならそれはそれでオブジェクト指向言語というかそれでのフレームワークなどでの名前付け方がってのはわかるけど 名前に拘るならNOPも略さずにNoOperationで 名前の長さとオブジェクト指向は独立した話だからなあ。 オブジェクト指向って言うからにはクラスか、あるいは型のプロトタイプか、どっちかは必須だろう。 >>38 あまりにも的確な分析にワロタ。まさか御本人様ですか?w まあ文字だけのコミュニケーションだといろいろと伝わりにくいことが多いから、お互い気を付けないとね。 秒を分に変更するにはどんなコマンドを使ったらいいですか? 3000という文字があったとして、50にしたいです date -d "3000" +"%M"だとダメでした $(( 3000 / 60 )) `expr 3000 / 60` 普通に算術で date使ってもある意味無理やりなのは、 Linux date -d "@3000" +"%M" BSD date -r 3000 +%M >>44 は@が抜けてる >>45 >>46 どうもありがとうございます!! date -d "@3000" +"%M" これだと、59分以上を表せなかったです。 >>49 そうだね。 $ echo 3000/60 |bc 50 $ echo 4000/60 |bc 66 だんだん数値が大きくなっていって閏年で悩むんだろうなあ >>50 bcのマニュアル読もう! $ printf '%.0f\n' "$(echo '4000/60' | bc -l)" 67 >>53 えっ、今回の要件は「商」を計算したいのだと思ったけど。 秒を分に変換するのに、勝手に四捨五入したらダメだろ すいません。 いま、誤ってログインシェルの設定ファイル(~/.bashrc)を空にしてしまいました。 しかし、まだログアウトしていないため、今稼動しているシェルはその設定を保持しています。 例えばエイリアスを知りたければ`alias`とやればいいはずですが、ほかにもシェル関数や環境変数なども読み込んでいます。 これらの値を今取得するにはどうすればいいですかね。 >>57 set 環境変数とシェル関数(posixモードの場合は変数のみ) set -o オプション設定のon/off状態 関数はsetで出てこね? printenvなんてあったのね。env使ってた。diffってみたが違いがわからんな。printenvの意味を後で調べてみよう みなさまありがとうございます。 set -oおよびprintenv(これはenvと同じ出力でした。なんとなくprintenvを使いました)でかなりの部分を取り戻すことができました。 ただ、set -oで出力されるオプション設定の項目数が少ないように思うんだけども、どうなんですかね。 allexport off braceexpand on emacs off errexit off errtrace off functrace off hashall on histexpand on history on ignoreeof off interactive-comments on keyword off monitor on noclobber off noexec off noglob off nolog off notify off nounset off onecmd off physical off pipefail off posix off privileged off verbose off vi on xtrace off なお、これらの内allexportやxtraceなどは設定した覚えがありません。システム設定(/etc/bash.bashrcあたり)でも設定されてません。不思議です……。 厚かましいのですが、できれば全てのオプションの状態を把握したいです。 すいません。shoptで行けました。ほとんど修復できました。感謝します。 >>63 っていうか、バックアップから.bashrcを拾って戻せるようにしておきなよ 言われなくても百も承知だろうけど、バックアップは大事だよ >>64 ですよね……。 なぜかバックアップとシンボリックリンクを張るという馬鹿な真似をしてました。 それはもうバックアップとは言わねぇか。 次にパソコンを買った時はLVMやらで柔軟なパーティションを組んでやります ドットファイルみたいにちょくちょく書き換えしててなくなると困るファイルは RCSなんかで管理しとくといいんじゃない? >>66 ホームぶっ飛ばしたバカを見たことがある。 多分 rm -fr ./work を rm -fr . /work って打ったんだろう。 ★★★共産●主義者とは言葉の通じない赤犬であり、決して人間と見なしてはならない!時たま本当らしいことを言うとしても、それはより大きな嘘を覆い隠す為の罠である。★★★ ● この掲示板(万有サロン)に優秀な書き込みをして、総額148万円の賞金をゲット●しよう!(*^^)v http://jbbs.livedoor.jp/study/37 ●29/ →リンクが不良なら、検索窓に入れる!● /home直下にhoge00〜hoge99までの名前がついているディレクトリがあるとします 他にも/home直下にはディレクトリが存在します ディレクトリ名を検索しながら、hoge00-hoge99の名前にマッチした複数のディレクトリの中の、ある特定のディレクトリに既存のファイルであるtestという名前のファイルをコピーしたいです ある特定の、という意味は ・hoge○○ディレクトリ直下にtestというファイルがあったら→スルー ・testというファイル名のファイルがhoge○○直下になければ、既存のtestを該当hoge○○ディレクトリ直下にコピー このようなシェルスクリプトを書きたいのですが教えてください forで回してifで分岐してcpするだけだと思うけど、 自分ではどこまでできてんの? >>73 ifとcpは使うんだろうなと思っていました forもやはり必要そうですね、、 それを使わずにどうやろうというんだ 要するに連番ディレクトリに全てtestファイルがあるようにしたい、ただし既存のtestファイルは上書きしない、でいいのか filehoge="test" [ ! -f "$filehoge" ] exit 1 seq -f 'hoge%02g' 1 99 | while read hogedir do [ -d "$hogedir" ] && [ ! -f "$hogedir/$filehoge" ] && { cp "$filehoge" "$hogedir"; echo "copied into $hogedir"; } done seq -f 'hoge%02g' 0 99 の間違い。00からなので >>77 >>78 もの凄く高度ですね… 自分には理解できるかわかりませんがじっくり読ませてください ありがとうございます お題として遊ぶなら、findしてexec内でやっつけるとかかな findは存在を確認するのは楽だろうけど、存在しないことを確認するのは面倒じゃない? >>77 [ ! -f "$filehoge" ] && exit 1 の間違いもあった。&&抜けてた、すまん。 >>79 ん?まあ、for(forはアレでアレの回避があったような気がするがすぐ忘れる。 | while read hoge しか使わないので) と if(明示するだけだけど)で書けばいいよ ... for hogedir in $(seq -f 'hoge%02g' 1 99) ... if [ -d "$hogedir" ] && [ ! -f "$hogedir/$filehoge" ]; then cp "$filehoge" "$hogedir" ... fi あー、" で括るだけか。アレの場合も for hogedir in "$(seq -f 'hoge%02g' 1 99)" が、よりかな やっぱりダメなのかな?? touch 'hoge 01';touch 'hoge$02' for name in "$(/bin/ls -1)" /bin/ls -1 | while read name やっぱforはメンドクサイから、安心簡単な while でいいやっ >>82 考えるのに役立つアドバイスありがとうございます >>83 >>84>>85 更新してくださりありがとうございますm(_ _)m 読ませていただきます 初心者ですが面白くなってきました 今日中に仕上げられるよう頑張ります >>82 それはexecの中でやればいいんじゃ お題とは別に、存在しないことの確認だけだったら、 検索条件に入れればいいだけだし 質問です。 シェルスクリプトファイルを実行する方法として 1. 利用者に応じた実行権限を賦与する $ chmod +x scriptfile.sh $ ./scriptfile.sh 2. sh(1)コマンドの引数として実行する $ sh scriptfile.sh という方法がありますが,どちらがより良いか教えてください(というかスクリプトの実行結果自体にはたして違いはあるのでしょうか)。 今まで一つ目の方法しか知りませんでしたが,もし二つ目になんら欠点がなければよりコマンドの数が少ない (極端に言えばchmod(1)コマンドがなくてもできる)後者の方法を使っていきたいです。 どうかよろしくおねがいします。 psで見てみ。1 でも、そのシェルスクリプトファイルのシバン(*1)で指定したインタプリタ(シェル)を、引数としてそのシェルスクリプトファイルをで起動してるから 1はコマンド、2はスクリプトファイルっていう違い。/usr/binなどの中のコマンドでも実態はシェルスクリプトやPerlスクリプト等なのがあるけど、そんなの気にしてない気づいてなくコマンドとして使ってたりするだろ? まあ、コマンドとして必要じゃない(コマンドの数が少ないって意味がわからんが一時的とかか?)んだったら、別にスクリプトファイルとしてで全然普通 1) お約束事項のような #!/bin/sh もう俺は気にしてなくて #!/bin/bash だけど >>82 hogefile="test" find . -maxdepth 1 -type d -name 'hoge[0-9][0-9]' -exec bash -c "test -f {}/$hogefile || cp $hogefile {}" \; こんな感じとか?俺は、俺も、そうはしないけど。-exec は使ったことないのでどこまでできるのかよくわからんな。なのでbashスクリプトにした/逃げた >>91 ありがとうございます。 ps(1)コマンドで確認したところ,たしかに両方ともにシェルが始めに起動されていました。 (素朴な疑問なんですが,PIDが連続しているのは偶然ですか? 原理的に必然ですか?) 1の方法だと一見シェルは起動されていないように見えるけれども,シバンに書かれたシェルがまず起動するということですね。 ということは全く違いはないという理解でよいでしょうか。 >>92 なるほど。1の方法は永続的,2の方法は試験目的で一時的に作ったファイルなどに対し用いるということですね。 シェルプロセスが勝手に起動されるか明示的に起動するかで、なんかあった(rcやprofileを読まないとか起動時の環境設定の動作)が今回のようなのは関係ないだろう、気にすんなw PIDはだいたい普通にインクリメント。なにかプロセスが起動するごとにそれの新しいPIDはインクリメントされた値かな OSによってはセキュリティ的な理由でPIDをある程度ランダムに割り当てるので連続にはならない。 BSD系はどれもデフォルトだとランダムじゃないかな。 PIDが若いから特権があるわけでもないしPIDが推測しやすいからっていうセキュリティ懸念ってなんだろ? cat >/tmp/tmpfile.$$ みたいに手抜きな一時ファイル作成してると シンボリックリンク攻撃くらうからそのリスクを減らせたりする。 ちゃんと攻撃避ける正しい書き方してれば関係ない話だけど、 世の中にテキトーなスクリプトは尽きないから。 なるほど。OSのセキュリティではなくPIDを利用したナニかに対するか ん?PIDを知ることができる(当然知ることは普通にできる)ってとこがキモで推測しやすいのとは別のような。まあ、いいや、ナニかあるなってことで ああ、シンボリックリンク攻撃を誤解してた。事前に(大量に)シンボリックリンク作ってしまうわけね。なるほど 知り合いから教えてもらったパソコン一台でお金持ちになれるやり方 時間がある方はみてもいいかもしれません グーグルで検索するといいかも『ネットで稼ぐ方法 モニアレフヌノ』 GXGZT シンボリック攻撃ってシンボリックリンクを作成する時にパーミッション関連の隙を突いて本来保護されているコマンドを改竄することだよね。 大量に作成する必要なないんじゃないかしらん。もちろんある種の陽動にはなるだろうけど。 >>77 >>78 これで一発で成功しました 驚いています このコードをもっと理解すると共に、自分も先輩のようなプログラマーになれるよう勉強を続けていきたいです マジで助かり、勉強になりました 本当にありがとうございました もちろん>>83 の修正も加えさせていただきました コードをくだすった方と同じ方かはわかりませんがそもそも>>75 でこちらの仕様を把握してくださったのがすごい 要求仕様、要件定義の大切さも少しわかりました そしてコードはこちらの望む通りの結果を出して一発で動きましたよと >>105 別人だ。俺はスマホだからここにコードは書けん >>106 そうでしたかm(_ _)m いずれにせよスレの複数の先輩方にお世話になりました しかも勉強になります あのなぁ、お前さんここをどこの板だと思っているんだい? 歴史あんだぜ まあその話はもうほんとに一昔前の出来事になっちまったけどな なんだったか、2chの鯖に関する貢献があったような 当時からいたわけじゃないからどっかのサイトで知った話だが Flash黄金期にその話がFlash化されてたから見て知ったな read.cgiの改良作業だよね。 直接関わってはいないけど、当時はリアルタイムで見ていたし、手元の どこかに運営から住民に開示されたread.cgiも残ってると思う。 アクセスが増えて通信量が増加して、当時借りてたホスティング業者から 追い出されそうになってたんだけど、HTTPの通信をgzipで圧縮して通信量を 削減しようという話だったかと思う。 ただ、UNIX板でも話をして作業には参加していたものの、プログラム板と かの方でもガリガリやっていたような記憶がある。 それとは別に SYN flood 攻撃に対し SYN cookie 有効化で対抗した時はUNIX板で主に議論してた気がする。 リアルタイムで見てたというか発言してたけど(SYN cacheと比較した得失とか書いた気が)、 こういうのはあくまで個人としての行動なので板自慢に繋げるのは好きじゃないな。 むしろ統一とかとれてなくてカオスなのが5chの良いところだと思うので。 >>112 gzip圧縮ってapacheの設定じゃないの? 他の顧客がいるからミドルウェアで圧縮できない、というホスティング屋を使っていたんじゃなかったのかな(後から知った話だから、違うかもしれない) なれ合いですらなくて、おっさんが勝手に郷愁に浸ってるだけという。 $ cat a.file > b.file とやるのと $ cp a.file b.file とやるので違いが生じる場合はありますか? >>120 b.fileというディレクトリがあった場合の動作 >>123 によると cat a.file > b.file/ ってやるといいんだって 意図してディレクトリじゃないからつけろもなんもないのにな Linuxのは実装はしてるみたいだな。デフォではないから話の流れでの実装してるとは言えんけど シェルスクリプトの勉強をするのにおすすめの本はありますか? お知恵お借りしたいです。 awk '($3 !~ /ここにパイプ区切りで記述したい/) && ($4 ~ /01/)' output.txt 上記のawkで条件に当てはまるファイルを絞り込みたいのですが その1 別ファイルからのawkの結果を $aaa に設定してある その2 $aaa は複数列の出力なので set コマンドにて位置パラメータに分けてある その3 各変数を展開して文字列としてパイプで区切ってawkを使いたい。 以上のようなことをやってみたのですがどうもうまく行きません。 また、違うやり方もあるようでしたらご教示いただけるとありがたいです。 よろしくお願いします。 > $aaa は複数列の出力なので set コマンドにて位置パラメータに分けてある これはどういう意味? set はシェル自身の設定だと思うんだけども cat ZZZ.txt |grep XXX hoge fuga と複数行でしたので aaa=$(cat ZZZ.txt |grep XXX) としてみました。 そして set $aaa と設定して $1 $2 $3 ...と変数が自動で割り振られるなら増減しても対応できるのでは考えました。 (本来は0個のときもあるのでそれも考慮しなければならないんですが) そして $1 等に入っている変数(hogeやfuga) を展開してパイプで区切れれば動くと思うんです。 ごめん俺の理解が悪いと思うのだがあなたのやりたいことが見えない。 他の人にまかせるわ。 もしくはZZZ.txtの内容をプライバシーに障らない程度に具体的に教えてほしい。 「ZZZ.txtからgrepで取り出した値をパイプで連結して処理する」という文言からは ZZZ.txtには命令が一行ずつ書かれているように受け取れるのだが、そうじゃないよね。 それとも「パイプで繋ぐ」というのはシェルのパイプ実行ではなく単に文字列としてパイプを区切りとして使うという意味? もしそうなら $ cat ZZZ.txt | grep XXX | tr '\n' '|' | sed 's/|$//1' ↑こういうので行けるけど。違うでしょ? 説明がヘタですみません。ZZZ.txtの中身はyoutube等の URL タイトル 日付 がタブ区切りで複数行書かれているファイルです。 123.com hoge 02 456.com fuga 05 789.com peke 08 のような感じです。 それらから条件から当てはまる行や列を抜き出してパイプ区切りにしたいのです。 パイプの解釈は hoge|fuga で hoge or fuga と展開して awkのパターン入れて読ませたい訳です。 >>137 なるほどね。こういうこと? (ZZZ.txtの代わりにヒアドキュメント使ってる) $ cat <<. | awk '($2 ~ /(hoge|fuga)/) && ($3 ~ /02/) {print}' example111.com hoge 02 example222.com hoge 03 456.com fuga 05 456example.com fuga 02 789.com peke 08 . example111.com hoge 02 456example.com fuga 02 ■ このスレッドは過去ログ倉庫に格納されています
read.cgi ver 07.5.4 2024/05/19 Walang Kapalit ★ | Donguri System Team 5ちゃんねる