From ea1ce3e831a1c975eb122b7a9dc780d8f08c463a Mon Sep 17 00:00:00 2001 From: Andrew Nordman Date: Sat, 23 Feb 2019 12:48:50 -0600 Subject: [PATCH] Fixes regression with CLI experience This adds a flag matcher in addition to OptionParser that ensures that we only parse the flags leading up to the command we attempt to execute. Resolves #375 --- lib/dotenv/cli.rb | 57 +++++++++++++++++++++++++++++------------ spec/dotenv/cli_spec.rb | 16 ++++++++++++ 2 files changed, 57 insertions(+), 16 deletions(-) diff --git a/lib/dotenv/cli.rb b/lib/dotenv/cli.rb index dbe70e69..446318e7 100644 --- a/lib/dotenv/cli.rb +++ b/lib/dotenv/cli.rb @@ -6,10 +6,12 @@ module Dotenv # The CLI is a class responsible of handling all the command line interface # logic. class CLI - attr_reader :argv + attr_reader :argv, :exec_args, :parser_args, :filenames def initialize(argv = []) @argv = argv.dup + @filenames = [] + @flag_matchers = [] end def run @@ -20,48 +22,71 @@ def run rescue Errno::ENOENT => e abort e.message else - exec(*@argv) unless @argv.empty? + exec(*@exec_args) unless @exec_args.empty? end end private def parse_argv!(argv) - @filenames = [] - - OptionParser.new do |parser| - parser.banner = "Usage: dotenv [options]" - parser.separator "" - add_options(parser) - end.parse!(argv) + parser = create_option_parser + add_options(parser, @flag_matchers) + @parser_args, @exec_args = split_argv(argv.join(" "), @flag_matchers) + parser.parse! @parser_args @filenames end - def add_options(parser) - add_files_option(parser) - add_help_option(parser) - add_version_option(parser) + def add_options(parser, flag_matchers) + add_files_option(parser, flag_matchers) + add_help_option(parser, flag_matchers) + add_version_option(parser, flag_matchers) end - def add_files_option(parser) + def add_files_option(parser, flag_matchers) + flag_matchers.push("-f \\S+") parser.on("-f FILES", Array, "List of env files to parse") do |list| @filenames = list end end - def add_help_option(parser) + def add_help_option(parser, flag_matchers) + flag_matchers.push("-h", "--help") parser.on("-h", "--help", "Display help") do puts parser exit end end - def add_version_option(parser) + def add_version_option(parser, flag_matchers) + flag_matchers.push("-v", "--version") parser.on("-v", "--version", "Show version") do puts "dotenv #{Dotenv::VERSION}" exit end end + + # Detect dotenv flags vs executable args so we can parse properly and still + # take advantage of OptionParser for dotenv flags + def split_argv(arg_string, matchers) + matcher = /^((?:#{matchers.join("|")})\s?)?(.+)?$/ + data = matcher.match(arg_string) + dotenv_args = [] + exec_args = [] + + unless data.nil? + dotenv_args = (!data[1].nil? ? data[1].split(" ") : []) + exec_args = (!data[2].nil? ? data[2].split(" ") : []) + end + + [dotenv_args, exec_args] + end + + def create_option_parser + OptionParser.new do |parser| + parser.banner = "Usage: dotenv [options]" + parser.separator "" + end + end end end diff --git a/spec/dotenv/cli_spec.rb b/spec/dotenv/cli_spec.rb index 0f2b6dc8..b69ff43a 100644 --- a/spec/dotenv/cli_spec.rb +++ b/spec/dotenv/cli_spec.rb @@ -38,6 +38,22 @@ def run(*args) expect(ENV).to have_key("QUOTED") end + it "does not consume non-dotenv flags by accident" do + cli = Dotenv::CLI.new(["-f", "plain.env", "foo", "--switch"]) + cli.send(:parse_argv!, cli.argv) + + expect(cli.filenames).to eql(["plain.env"]) + expect(cli.exec_args).to eql(["foo", "--switch"]) + end + + it "does not consume dotenv flags from subcommand" do + cli = Dotenv::CLI.new(["foo", "-f", "something"]) + cli.send(:parse_argv!, cli.argv) + + expect(cli.filenames).to eql([]) + expect(cli.exec_args).to eql(["foo", "-f", "something"]) + end + # Capture output to $stdout and $stderr def capture_output(&_block) original_stderr = $stderr