【 getopt 】コマンド――オプションを解析する(応用編)Linux基本コマンドTips(380)

本連載は、Linuxのコマンドについて、基本書式からオプション、具体的な実行例までを紹介していきます。今回は、シェルスクリプト内などでオプションを解析する「getopt」コマンドです。

» 2020年02月20日 05時00分 公開
[西村めぐみ@IT]

この記事は会員限定です。会員登録(無料)すると全てご覧いただけます。

「Linux基本コマンドTips」のインデックス

Linux基本コマンドTips一覧

 本連載は、Linuxのコマンドについて、基本書式からオプション、具体的な実行例までを紹介していきます。今回は、連載第379回に続き、シェルスクリプト内などでオプションを解析する「getopt」コマンドです。

getoptコマンドとは?

 「getopt」は、シェルスクリプト内などでオプションを解析するコマンドです。自作のシェルスクリプトで「-a」のような「ハイフン+アルファベット1文字」のオプションを扱う際に便利です。「-f ファイル名」のように引数を取るオプションも解析できます。

 よく似た名前で用途も同じ「getopts」コマンドとはどこが異なるのでしょうか(連載第378回)。まず、getoptsコマンドはbashのビルトインコマンドです。さらにgetoptsコマンドでは、「--」から始まるロングオプションを解析できません。

 getoptコマンドは外部コマンド(/usr/bin/getopt)であり、ロングオプションの解析も可能です。

 なお、getoptは、CentOS 8やUbuntu 18などに収録されているGNU版(util-linuxパッケージ収録、※1)の他に、macOSなどに収録されているBSD版が存在します。BSD版のgetoptは空白や特殊記号を含む引数をうまく扱えないという問題があるので注意してください(※2)。getoptコマンド以外の手段を検討した方がよいかもしれません。以下の解説はGNU版(CentOS 8版)のgetoptコマンドを使用しています。

※1 どの版を利用しているのかを調べるには「getopt --verion」を実行する。
※2 BSD版のmanには、この問題の解決が難しいことを次のように記している。「Arguments containing white space or embedded shell metacharacters generally will not survive intact; this looks easy to fix but isn't. People trying to fix getopt or the example in this manpage should check the history of this file in FreeBSD.」



 この他、オプションを解析する際には、引数を1つずらす「shift」コマンドを使うことも可能です。



コマンドの書式

getopt [オプション] オプション文字列 引数

getopt [オプション] -o オプション文字列 -- 引数

getopt [オプション] -o オプション文字列 -l 長いオプション -- 引数

※ [ ]は省略可能な引数を示しています。




getoptの主なオプション

短いオプション 長いオプション 意味
-o オプション --options オプション オプションとして使用したい文字を指定する(本文を参照
-l オプション --longoptions オプション 長いオプションとして使用したい文字列を指定する(-oと併用、複数ある場合は「,」で区切るか「-l」オプションを複数回使用する、本文を参照
-a --alternative 単一の「-」で始まるロングオプションを許可する
-s シェルの種類 --shell シェルの種類 使用するシェルを「sh」「bash」「csh」「tcsh」から指定する(デフォルトはbash、※3)
-n --name 名前 getoptを使うプログラムの名前を指定する(エラーメッセージを表示する際に使用する)
-q --quiet エラーメッセージを表示しない
-Q --quiet-output 通常の出力を省略する
-u --unquoted 出力にクオートを付けない

※3 シェルによってエスケープの処理が異なるため、bash(デフォルト)以外のシェルスクリプトでgetoptコマンドを使用する場合に指定する。なお、zshの場合、「zparseopts」コマンド(man zshmodules参照)で長いオプションを解析可能。





長いオプションを解析する

 getoptコマンドを使って長いオプション(ロングオプション)を解析したい場合は、「getopt -o オプション文字列 -l 長いオプション文字列 -- 引数」のようにします(画面1)。引数の部分に解析したい文字列を置きます。長いオプションが複数ある場合は「,」記号で区切るか、「-l」を複数回使用します。

 「-o」は短いオプション用です(連載第379回)。短いオプションが不要の場合は「-o ""」のようにします(※4)。

※4 「-o」を使わない場合、「getopt -l 長いオプション文字列 "" -- 引数」のように、最後(-lなどのオプション指定の後)に「""」を指定する必要がある。



 なお、長いオプションを使う際、「省略表記」に対応しています。例えば、「--file」「--quiet」「--quote」の3つが設定されている場合、「--f」で「--file」と見なし、「--qui」で「--quiet」と見なします。

コマンド実行例

getopt -o abc -l aaa,bbb -- 解析したい文字列

(短いオプション「-a」「-b」「-c」、長いオプション「--aaa」「--bbb」を使う場合を解析する)

getopt -o "" -l aaa,bbb -- 解析したい文字列

(短いオプションなし、長いオプション「--aaa」「--bbb」を使う場合を解析する)


画面1 画面1 長いオプションを解析したところ コマンドラインからgetoptコマンドを使った例


長いオプションでハイフンが1つの場合を許容する

 長いオプションは「--file」のようにハイフンを2つ使います。しかしgetoptコマンドで解析する際、「-file」のようにハイフンが1つであっても許容したい、という場合は「-a」オプションを使用します。

コマンド実行例

getopt -a -o abc -l aaa,bbb -- 解析したい文字列

(短いオプションとして「-a」「-b」「-c」、長いオプション「--aaa」「-aaa」「--bbb」「-bbb」を使う場合を解析する)

getopt -a -o "" -l aaa,bbb -- 解析したい文字列

(短いオプションなし、長いオプション「--aaa」「-aaa」「--bbb」「-bbb」を使う場合を解析する)


 画面2では、「-a」を指定した場合と指定しなかった場合の結果の違いを比較しています。

画面2 画面2 長いオプションを解析したところ ハイフンが1つの場合の処理を比較した


引数付きの長いオプションを使用する

 長いオプションを引数付きにしたい場合、オプションの後ろに「:」を付けます。例えば、「-l aaa,bbb,ccc」を解析する際、「aaa」と「ccc」を引数付きにするのであれば「-l aaa:,bbb,ccc:」のようにします(画面3)。

 引数が必須ではない場合、つまりあってもなくてもよいという場合は「::」とします。

 長いオプションに対する引数は、「--file=ファイル名」のように「=」を付けて指定しますが、「=」を省略して「--file ファイル名」のようにすることもあります。getoptコマンドではどちらの指定方法も利用できます(※5)。

※5 man getoptには次のように記述されている。「If the option has a required argument, it may be written directly after the long option name, separated by '=', or as the next argument (i.e. separated by whitespace on the command line).」



コマンド実行例

getopt -o "" -l aaa:,bbb,ccc: -- 解析したい文字列

(短いオプションなし、引数が必須の長いオプション「--aaa」、引数がない「--bbb」、引数が必須の「-ccc」を使う場合を解析する)

getopt -o "" -l aaa::,bbb,ccc: -- 解析したい文字列

(短いオプションなし、引数があってもなくてもよい長いオプション「--aaa」、引数がない「--bbb」、引数が必須の「-ccc」を使う場合を解析する)


画面3 画面3 引数付きの長いオプションを解析したところ

 引数が必須ではない場合、オプションに対する引数の指定方法によって解析結果が変わります。

 「--file ファイル名」のようにオプションとスペースで区切った場合はオプションへの引数ではなく、コマンドライン全体の引数として解釈されます。getoptで解析した結果の順番が変わることに注意してください(画面4)。

画面4 画面4 引数が必須ではない場合のオプションを解析したところ


シェルスクリプトで応用する(参考)

 getoptコマンドの出力を加工することで、より分かりやすくオプションや引数の解析結果を表示できます。ここでは長いオプションを扱った例を紹介します。

 スクリプト1(opt6.sh)では「flag_a」という変数にあらかじめ「false」という値をセットしておき、実行時に「-a」「-aaa」「--aaa」が指定されていた場合に値を「true」に変更しています(※6)。「flag_b」「flag_c」も同様です。

 「-n $(basename $0)」の部分については連載第379回の「修正版のopt4.sh」を参照してください。

※6 true、falseとも値ではなくコマンド(/bin/trueと/bin/false、連載第318回参照)。



 「--aaa」は引数を取るかもしれないので、「$2」で値を参照しています。先頭文字が「-」ではなかった場合は引数と見なし、変数arg_aに格納して「shift」コマンドで次に進めています(※7)。arg_cについても同様な措置をしています(画面5)。

※7 このように処理してしまうと、オプションの引数に「-」から始まる文字列を指定できないということになってしまう。スクリプトの用途によっては「=」を指定している場合は「-」から始まる引数を受け付けるようにする必要があるだろう。getoptを通すと「=」を使用したかどうかが判別できなくなるため、getoptを使わずに引数を処理する必要がある。



#! /bin/bash
# 短いオプション-a,-b.-cと長いオプション--aaa,--bbb,-cccを使用
# -aと-aaa、-bと-bbb、-cと-cccは同じ意味だが、
# 長いオプション--aaaと--cccは引数を取る(--cccは引数が必須)
# 以上がこのシェルスクリプトの処理内容(脚注も参照のこと)
OPTIONS=`getopt -n $(basename $0) -a -o abc -l aaa::,bbb,ccc: -- "$@"`
eval set -- "$OPTIONS"
flag_a=false #変数「flag_a」にコマンド名「false」をセット
flag_b=false
flag_c=false
while [ $# -gt 0 ]
do
  case $1 in
    -a | -aaa | --aaa) flag_a=true;
                   if [ "${2::1}" != "-" ]; then
                     arg_a=$2; shift;
                   fi;;
    -b | -bbb | --bbb) flag_b=true;;
    -c | -ccc | --ccc) flag_c=true;
                   if [ "${2::1}" != "-" ]; then
                     arg_c=$2; shift;
                   fi;;
    --) shift; break;;
  esac
  shift
done
if $flag_a; then
  echo "-a 指定あり(引数=$arg_a)"
else
  echo "-a 指定なし"
fi
if $flag_b; then
  echo "-b 指定あり"
else
  echo "-b 指定なし"
fi
if $flag_c; then
  echo "-c 指定あり(引数=$arg_c)"
else
  echo "-c 指定なし"
fi
while [ $# -gt 0 ]
do
  echo "引数=$1"
  shift
done
スクリプト1(opt6.sh)
画面5 画面5 長いオプションや引数の内容を分かりやすく表示したところ

 シェルスクリプトでgetoptsやgetoptを使う場合は、それぞれの解析方法の特徴を把握し、実際に利用する前に十分なテストを行うようにしましょう。

 引数を取るオプションを使いたい場合は、getoptなどに任せきらず、自分で解析する方がよい場合もあります。自作のスクリプトの場合、そもそもオプションに使う文字または文字列が決まっているため、オプションごとに必要な判定を行うことができるでしょう。



筆者紹介

西村 めぐみ(にしむら めぐみ)

元々はDOSユーザーで「DOS版UNIX-like tools」を愛用。ソフトハウスに勤務し生産管理のパッケージソフトウェアの開発およびサポート業務を担当、その後ライターになる。著書に『図解でわかるLinux』『らぶらぶLinuxシリーズ』『Accessではじめるデータベース超入門[改訂2版]』『macOSコマンド入門』など。地方自治体の在宅就業支援事業にてMicrosoft Officeの教材作成およびeラーニング指導を担当。会社などの"PCヘルパー"やピンポイント研修なども行っている。


Copyright © ITmedia, Inc. All Rights Reserved.

スポンサーからのお知らせPR

注目のテーマ

Microsoft & Windows最前線2025
AI for エンジニアリング
ローコード/ノーコード セントラル by @IT - ITエンジニアがビジネスの中心で活躍する組織へ
Cloud Native Central by @IT - スケーラブルな能力を組織に
システム開発ノウハウ 【発注ナビ】PR
あなたにおすすめの記事PR

RSSについて

アイティメディアIDについて

メールマガジン登録

@ITのメールマガジンは、 もちろん、すべて無料です。ぜひメールマガジンをご購読ください。