diff --git a/spec/clim/stdout_spec/files/exception_message.cr b/spec/clim/stdout_spec/files/required.cr similarity index 84% rename from spec/clim/stdout_spec/files/exception_message.cr rename to spec/clim/stdout_spec/files/required.cr index 99d8aca8..91de0d9e 100644 --- a/spec/clim/stdout_spec/files/exception_message.cr +++ b/spec/clim/stdout_spec/files/required.cr @@ -4,6 +4,8 @@ module Hello class Cli < Clim main do usage "hello " + help short: "-h" + version "Version 0.1.0", short: "-v" option "--prefix ", type: String, desc: "Prefix.", required: true argument "arg1", type: String, desc: "argument1", required: true run do |opts, args| diff --git a/spec/clim/stdout_spec/stdout_spec.cr b/spec/clim/stdout_spec/stdout_spec.cr index 76f4492a..22df3520 100644 --- a/spec/clim/stdout_spec/stdout_spec.cr +++ b/spec/clim/stdout_spec/stdout_spec.cr @@ -189,7 +189,7 @@ describe "STDOUT spec, " do DISPLAY end it "exception message. (Required options)" do - `crystal run spec/clim/stdout_spec/files/exception_message.cr --no-color -- `.should eq <<-DISPLAY + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- `.should eq <<-DISPLAY ERROR: Required options. "--prefix " Please see the `--help`. @@ -197,7 +197,7 @@ describe "STDOUT spec, " do DISPLAY end it "exception message. (Option that requires an argument)" do - `crystal run spec/clim/stdout_spec/files/exception_message.cr --no-color -- --prefix`.should eq <<-DISPLAY + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix`.should eq <<-DISPLAY ERROR: Option that requires an argument. "--prefix" Please see the `--help`. @@ -205,11 +205,177 @@ describe "STDOUT spec, " do DISPLAY end it "exception message. (Option that requires an argument)" do - `crystal run spec/clim/stdout_spec/files/exception_message.cr --no-color -- --prefix=foo`.should eq <<-DISPLAY + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix=foo`.should eq <<-DISPLAY ERROR: Required arguments. "arg1" Please see the `--help`. DISPLAY end + it "For --help, options and arguments are not required." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --help`.should eq <<-DISPLAY + + Command Line Interface Tool. + + Usage: + + hello + + Options: + + --prefix Prefix. [type:String] [required] + -h, --help Show this help. + -v, --version Show version. + + Arguments: + + 01. arg1 argument1 [type:String] [required] + + + DISPLAY + end + it "For -h, options and arguments are not required." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- -h`.should eq <<-DISPLAY + + Command Line Interface Tool. + + Usage: + + hello + + Options: + + --prefix Prefix. [type:String] [required] + -h, --help Show this help. + -v, --version Show version. + + Arguments: + + 01. arg1 argument1 [type:String] [required] + + + DISPLAY + end + it "If --help is specified along with options or arguments, a help message will be displayed." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix=foo arg1 --help `.should eq <<-DISPLAY + + Command Line Interface Tool. + + Usage: + + hello + + Options: + + --prefix Prefix. [type:String] [required] + -h, --help Show this help. + -v, --version Show version. + + Arguments: + + 01. arg1 argument1 [type:String] [required] + + + DISPLAY + end + it "If -h is specified along with options or arguments, a help message will be displayed." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix=foo arg1 -h `.should eq <<-DISPLAY + + Command Line Interface Tool. + + Usage: + + hello + + Options: + + --prefix Prefix. [type:String] [required] + -h, --help Show this help. + -v, --version Show version. + + Arguments: + + 01. arg1 argument1 [type:String] [required] + + + DISPLAY + end + it "For --version, options and arguments are not required." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --version`.should eq <<-DISPLAY + Version 0.1.0 + + DISPLAY + end + it "For -v, options and arguments are not required." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- -v`.should eq <<-DISPLAY + Version 0.1.0 + + DISPLAY + end + it "If --version is specified along with options or arguments, a version message will be displayed." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix=foo arg1 --version `.should eq <<-DISPLAY + Version 0.1.0 + + DISPLAY + end + it "If -v is specified along with options or arguments, a version message will be displayed." do + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix=foo arg1 -v `.should eq <<-DISPLAY + Version 0.1.0 + + DISPLAY + end + it "For --bash-completion, options and arguments are not required." do + expected_regex = <<-'DISPLAY' + _.*\(\) + \{ + \ \ \ \ local\ program\=\$\{COMP_WORDS\[0\]\} + \ \ \ \ local\ cmd\=\$\{COMP_WORDS\[1\]\} + \ \ \ \ local\ cur\="\$\{COMP_WORDS\[COMP_CWORD\]\}" + \ \ \ \ local\ prev\="\$\{COMP_WORDS\[COMP_CWORD\-1\]\}" + \ \ \ \ local\ cword\="\$\{COMP_CWORD\}" + + \ \ \ \ + if\ \[\[\ "\$\{prev\}"\ \=\=\ ".*"\ \]\]\ ;\ then + \ \ \ \ COMPREPLY\=\(\ \$\(compgen\ \-W\ "\-v\ \-\-version\ \-\-prefix\ \-h\ \-\-help"\ \-\-\ \$\{cur\}\)\ \) + else + \ \ \ \ COMPREPLY\=\(\ \$\(compgen\ \-f\ \$\{cur\}\)\ \) + fi + + + \ \ \ \ return\ 0 + \} + + complete\ \-F\ _.* .* + + DISPLAY + + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --bash-completion`.should match(/#{expected_regex}/) + end + it "If --bash-completion is specified along with options or arguments, a bash completion string will be displayed." do + expected_regex = <<-'DISPLAY' + _.*\(\) + \{ + \ \ \ \ local\ program\=\$\{COMP_WORDS\[0\]\} + \ \ \ \ local\ cmd\=\$\{COMP_WORDS\[1\]\} + \ \ \ \ local\ cur\="\$\{COMP_WORDS\[COMP_CWORD\]\}" + \ \ \ \ local\ prev\="\$\{COMP_WORDS\[COMP_CWORD\-1\]\}" + \ \ \ \ local\ cword\="\$\{COMP_CWORD\}" + + \ \ \ \ + if\ \[\[\ "\$\{prev\}"\ \=\=\ ".*"\ \]\]\ ;\ then + \ \ \ \ COMPREPLY\=\(\ \$\(compgen\ \-W\ "\-v\ \-\-version\ \-\-prefix\ \-h\ \-\-help"\ \-\-\ \$\{cur\}\)\ \) + else + \ \ \ \ COMPREPLY\=\(\ \$\(compgen\ \-f\ \$\{cur\}\)\ \) + fi + + + \ \ \ \ return\ 0 + \} + + complete\ \-F\ _.* .* + + DISPLAY + + `crystal run spec/clim/stdout_spec/files/required.cr --no-color -- --prefix=foo arg1 --bash-completion`.should match(/#{expected_regex}/) + end end diff --git a/src/clim/command/arguments.cr b/src/clim/command/arguments.cr index e25613ad..2fa0f3f3 100644 --- a/src/clim/command/arguments.cr +++ b/src/clim/command/arguments.cr @@ -26,10 +26,10 @@ class Clim def set_argv(@argv : Array(String)) end - def required_validate!(options : Options) - if options.responds_to?(:help) - return if options.help - end + def required_validate!(opts : Options) + return if opts.responds_to?(:help) && opts.help + return if opts.responds_to?(:version) && opts.version + return if opts.responds_to?(:bash_completion) && opts.bash_completion return if invalid_required_names.empty? raise ClimInvalidOptionException.new "Required arguments. \"#{invalid_required_names.join("\", \"")}\"" end diff --git a/src/clim/command/options.cr b/src/clim/command/options.cr index 04a403df..5c44d826 100644 --- a/src/clim/command/options.cr +++ b/src/clim/command/options.cr @@ -57,9 +57,9 @@ class Clim def required_validate! opts = self.dup - if opts.responds_to?(:help) - return if opts.help - end + return if opts.responds_to?(:help) && opts.help + return if opts.responds_to?(:version) && opts.version + return if opts.responds_to?(:bash_completion) && opts.bash_completion return if invalid_required_names.empty? raise ClimInvalidOptionException.new "Required options. \"#{invalid_required_names.join("\", \"")}\"" end