bundler 1.15.3 → 1.15.4

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of bundler might be problematic. Click here for more details.

checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 4ec85619b6e7291fe9e96c8a11183f6ed2879017
4
- data.tar.gz: f98a0bfaca1056ce8f0ad6fe4170e4e2e22f78af
3
+ metadata.gz: 4017f3a02c248780eb86afcd89c12249064f3dbc
4
+ data.tar.gz: 96679694c04c77c28b3c01d6e017fb6273eb857a
5
5
  SHA512:
6
- metadata.gz: 6bc7bd6c3273f58a62086e5d9e59347615c17f2367aa97bbeaddef0764f698c22c83e08eca1af2152518a14b156f4f2c1e1b42e7da3842d3f1399e53cacd5c92
7
- data.tar.gz: 74ae15403d2abb3681b4e15cc19ccdc9d957b02fbc7cf590453a90cc53616ac37b488e777472a45f22daef14fb7871e2121ddcfb496b44fb41993eabec0b1152
6
+ metadata.gz: 70122e1565c9ee2d2998872fb0f9a30e65749a2f07ef9661f340e68db0244c53a0d62384061afdc05f259b9fccc40a7cf4a70623251fe60bb09384d536863357
7
+ data.tar.gz: 1aeca0caca38ecfb8acd47c1cfe0f18e19e1521b68f77b797e3df57f6a6ab54f3e16d550f2cb3bf75124f060e92c1371f9128509b9e9123de7d6167e5c13596c
@@ -1,3 +1,12 @@
1
+ ## 1.15.4 (2017-08-19)
2
+
3
+ Bugfixes:
4
+
5
+ - handle file conflicts gracefully in `bundle gem` (@rafaelfranca, @segiddins)
6
+ - bundler will fail gracefully when the bundle path contains the system path separator (#5485, ajwann)
7
+ - failed gem downloads will be retried consistently across different RubyGems versions (@shayonj)
8
+ - `bundle pristine` will respect build options while re-building native extensions (@NickLaMuro)
9
+
1
10
  ## 1.15.3 (2017-07-21)
2
11
 
3
12
  Bugfixes:
@@ -91,9 +91,6 @@ module Bundler
91
91
  end
92
92
  end
93
93
 
94
- # Ensure `bundle help --no-color` is valid
95
- all_commands["help"].disable_class_options = false
96
-
97
94
  def self.handle_no_command_error(command, has_namespace = $thor_runner)
98
95
  if Bundler.feature_flag.plugins? && Bundler::Plugin.command?(command)
99
96
  return Bundler::Plugin.exec_command(command, ARGV[1..-1])
@@ -141,9 +141,10 @@ module Bundler
141
141
  end
142
142
 
143
143
  executables.each do |file|
144
- path = target.join(file)
145
- executable = (path.stat.mode | 0o111)
146
- path.chmod(executable)
144
+ SharedHelpers.filesystem_access(target.join(file)) do |path|
145
+ executable = (path.stat.mode | 0o111)
146
+ path.chmod(executable)
147
+ end
147
148
  end
148
149
 
149
150
  if Bundler.git_present?
@@ -4,29 +4,32 @@ require "bundler/cli/common"
4
4
  module Bundler
5
5
  class CLI::Pristine
6
6
  def run
7
+ definition = Bundler.definition
8
+ definition.validate_runtime!
9
+ installer = Bundler::Installer.new(Bundler.root, definition)
10
+
7
11
  Bundler.load.specs.each do |spec|
8
12
  next if spec.name == "bundler" # Source::Rubygems doesn't install bundler
9
13
 
10
14
  gem_name = "#{spec.name} (#{spec.version}#{spec.git_version})"
11
15
  gem_name += " (#{spec.platform})" if !spec.platform.nil? && spec.platform != Gem::Platform::RUBY
12
16
 
13
- case spec.source
17
+ case source = spec.source
14
18
  when Source::Rubygems
15
19
  cached_gem = spec.cache_file
16
20
  unless File.exist?(cached_gem)
17
21
  Bundler.ui.error("Failed to pristine #{gem_name}. Cached gem #{cached_gem} does not exist.")
18
22
  next
19
23
  end
20
-
21
- FileUtils.rm_rf spec.full_gem_path
22
- spec.source.install(spec, :force => true)
23
24
  when Source::Git
24
- git_source = spec.source
25
- git_source.remote!
26
- git_source.install(spec, :force => true)
25
+ source.remote!
27
26
  else
28
27
  Bundler.ui.warn("Cannot pristine #{gem_name}. Gem is sourced from local path.")
28
+ next
29
29
  end
30
+ FileUtils.rm_rf spec.full_gem_path
31
+
32
+ Bundler::GemInstaller.new(spec, installer, false, 0, true).install_from_spec
30
33
  end
31
34
  end
32
35
  end
@@ -300,7 +300,7 @@ module Bundler
300
300
  def download_gem(spec, uri, path)
301
301
  uri = Bundler.settings.mirror_for(uri)
302
302
  fetcher = Gem::RemoteFetcher.new(configuration[:http_proxy])
303
- Bundler::Retry.new("download gem #{uri}", Gem::RemoteFetcher::FetchError).attempts do
303
+ Bundler::Retry.new("download gem from #{uri}").attempts do
304
304
  fetcher.download(spec, uri, path)
305
305
  end
306
306
  end
@@ -736,7 +736,9 @@ module Bundler
736
736
  uri = Bundler.settings.mirror_for(uri)
737
737
  fetcher = gem_remote_fetcher
738
738
  fetcher.headers = { "X-Gemfile-Source" => spec.remote.original_uri.to_s } if spec.remote.original_uri
739
- fetcher.download(spec, uri, path)
739
+ Bundler::Retry.new("download gem from #{uri}").attempts do
740
+ fetcher.download(spec, uri, path)
741
+ end
740
742
  end
741
743
 
742
744
  def gem_remote_fetcher
@@ -172,6 +172,17 @@ module Bundler
172
172
 
173
173
  private
174
174
 
175
+ def validate_bundle_path
176
+ return unless Bundler.bundle_path.to_s.include?(File::PATH_SEPARATOR)
177
+ message = "Your bundle path contains a '#{File::PATH_SEPARATOR}', " \
178
+ "which is the path separator for your system. Bundler cannot " \
179
+ "function correctly when the Bundle path contains the " \
180
+ "system's PATH separator. Please change your " \
181
+ "bundle path to not include '#{File::PATH_SEPARATOR}'." \
182
+ "\nYour current bundle path is '#{Bundler.bundle_path}'."
183
+ raise Bundler::PathError, message
184
+ end
185
+
175
186
  def find_gemfile
176
187
  given = ENV["BUNDLE_GEMFILE"]
177
188
  return given if given && !given.empty?
@@ -222,6 +233,7 @@ module Bundler
222
233
  end
223
234
 
224
235
  def set_path
236
+ validate_bundle_path
225
237
  paths = (ENV["PATH"] || "").split(File::PATH_SEPARATOR)
226
238
  paths.unshift "#{Bundler.bundle_path}/bin"
227
239
  ENV["PATH"] = paths.uniq.join(File::PATH_SEPARATOR)
@@ -158,10 +158,6 @@ class Bundler::Thor
158
158
  end
159
159
  alias_method :option, :method_option
160
160
 
161
- def disable_class_options
162
- @disable_class_options = true
163
- end
164
-
165
161
  # Prints help information for the given command.
166
162
  #
167
163
  # ==== Parameters
@@ -241,6 +237,9 @@ class Bundler::Thor
241
237
  invoke_args.unshift "help" if opts.delete("--help") || opts.delete("-h")
242
238
  invoke subcommand_class, *invoke_args
243
239
  end
240
+ subcommand_class.commands.each do |_meth, command|
241
+ command.ancestor_name = subcommand
242
+ end
244
243
  end
245
244
  alias_method :subtask, :subcommand
246
245
 
@@ -326,12 +325,31 @@ class Bundler::Thor
326
325
  command && stop_on_unknown_option.include?(command.name.to_sym)
327
326
  end
328
327
 
328
+ # Disable the check for required options for the given commands.
329
+ # This is useful if you have a command that does not need the required options
330
+ # to work, like help.
331
+ #
332
+ # ==== Parameters
333
+ # Symbol ...:: A list of commands that should be affected.
334
+ def disable_required_check!(*command_names)
335
+ disable_required_check.merge(command_names)
336
+ end
337
+
338
+ def disable_required_check?(command) #:nodoc:
339
+ command && disable_required_check.include?(command.name.to_sym)
340
+ end
341
+
329
342
  protected
330
343
 
331
344
  def stop_on_unknown_option #:nodoc:
332
345
  @stop_on_unknown_option ||= Set.new
333
346
  end
334
347
 
348
+ # help command has the required check disabled by default.
349
+ def disable_required_check #:nodoc:
350
+ @disable_required_check ||= Set.new([:help])
351
+ end
352
+
335
353
  # The method responsible for dispatching given the args.
336
354
  def dispatch(meth, given_args, given_opts, config) #:nodoc: # rubocop:disable MethodLength
337
355
  meth ||= retrieve_command_name(given_args)
@@ -390,12 +408,12 @@ class Bundler::Thor
390
408
  @usage ||= nil
391
409
  @desc ||= nil
392
410
  @long_desc ||= nil
393
- @disable_class_options ||= nil
411
+ @hide ||= nil
394
412
 
395
413
  if @usage && @desc
396
414
  base_class = @hide ? Bundler::Thor::HiddenCommand : Bundler::Thor::Command
397
- commands[meth] = base_class.new(meth, @desc, @long_desc, @usage, method_options, @disable_class_options)
398
- @usage, @desc, @long_desc, @method_options, @hide, @disable_class_options = nil
415
+ commands[meth] = base_class.new(meth, @desc, @long_desc, @usage, method_options)
416
+ @usage, @desc, @long_desc, @method_options, @hide = nil
399
417
  true
400
418
  elsif all_commands[meth] || meth == "method_missing"
401
419
  true
@@ -477,7 +495,6 @@ class Bundler::Thor
477
495
  map HELP_MAPPINGS => :help
478
496
 
479
497
  desc "help [COMMAND]", "Describe available commands or one specific command"
480
- disable_class_options
481
498
  def help(command = nil, subcommand = false)
482
499
  if command
483
500
  if self.class.subcommands.include? command
@@ -1,4 +1,3 @@
1
- require "fileutils"
2
1
  require "uri"
3
2
  require "bundler/vendor/thor/lib/thor/core_ext/io_binary_read"
4
3
  require "bundler/vendor/thor/lib/thor/actions/create_file"
@@ -141,7 +140,7 @@ class Bundler::Thor
141
140
  end
142
141
  end
143
142
 
144
- message = "Could not find #{file.inspect} in any of your source paths. "
143
+ message = "Could not find #{file.inspect} in any of your source paths. ".dup
145
144
 
146
145
  unless self.class.source_root
147
146
  message << "Please invoke #{self.class.name}.source_root(PATH) with the PATH containing your templates. "
@@ -175,6 +174,7 @@ class Bundler::Thor
175
174
 
176
175
  # If the directory doesnt exist and we're not pretending
177
176
  if !File.exist?(destination_root) && !pretend
177
+ require "fileutils"
178
178
  FileUtils.mkdir_p(destination_root)
179
179
  end
180
180
 
@@ -182,6 +182,7 @@ class Bundler::Thor
182
182
  # In pretend mode, just yield down to the block
183
183
  block.arity == 1 ? yield(destination_root) : yield
184
184
  else
185
+ require "fileutils"
185
186
  FileUtils.cd(destination_root) { block.arity == 1 ? yield(destination_root) : yield }
186
187
  end
187
188
 
@@ -251,7 +252,9 @@ class Bundler::Thor
251
252
 
252
253
  say_status :run, desc, config.fetch(:verbose, true)
253
254
 
254
- !options[:pretend] && config[:capture] ? `#{command}` : system(command.to_s)
255
+ unless options[:pretend]
256
+ config[:capture] ? `#{command}` : system(command.to_s)
257
+ end
255
258
  end
256
259
 
257
260
  # Executes a ruby script (taking into account WIN32 platform quirks).
@@ -58,6 +58,7 @@ class Bundler::Thor
58
58
 
59
59
  def invoke!
60
60
  invoke_with_conflict_check do
61
+ require "fileutils"
61
62
  FileUtils.mkdir_p(File.dirname(destination))
62
63
  File.open(destination, "wb") { |f| f.write render }
63
64
  end
@@ -38,6 +38,7 @@ class Bundler::Thor
38
38
 
39
39
  def invoke!
40
40
  invoke_with_conflict_check do
41
+ require "fileutils"
41
42
  FileUtils.mkdir_p(File.dirname(destination))
42
43
  # Create a symlink by default
43
44
  config[:symbolic] = true if config[:symbolic].nil?
@@ -48,12 +48,14 @@ class Bundler::Thor
48
48
 
49
49
  def invoke!
50
50
  invoke_with_conflict_check do
51
+ require "fileutils"
51
52
  ::FileUtils.mkdir_p(destination)
52
53
  end
53
54
  end
54
55
 
55
56
  def revoke!
56
57
  say_status :remove, :red
58
+ require "fileutils"
57
59
  ::FileUtils.rm_rf(destination) if !pretend? && exists?
58
60
  given_destination
59
61
  end
@@ -112,11 +114,17 @@ class Bundler::Thor
112
114
  if exists?
113
115
  on_conflict_behavior(&block)
114
116
  else
115
- say_status :create, :green
116
117
  yield unless pretend?
118
+ say_status :create, :green
117
119
  end
118
120
 
119
121
  destination
122
+ rescue Errno::EISDIR, Errno::EEXIST
123
+ on_file_clash_behavior
124
+ end
125
+
126
+ def on_file_clash_behavior
127
+ say_status :file_clash, :red
120
128
  end
121
129
 
122
130
  # What to do when the destination file already exists.
@@ -1,5 +1,4 @@
1
1
  require "erb"
2
- require "open-uri"
3
2
 
4
3
  class Bundler::Thor
5
4
  module Actions
@@ -78,7 +77,12 @@ class Bundler::Thor
78
77
  config = args.last.is_a?(Hash) ? args.pop : {}
79
78
  destination = args.first
80
79
 
81
- source = File.expand_path(find_in_source_paths(source.to_s)) unless source =~ %r{^https?\://}
80
+ if source =~ %r{^https?\://}
81
+ require "open-uri"
82
+ else
83
+ source = File.expand_path(find_in_source_paths(source.to_s))
84
+ end
85
+
82
86
  render = open(source) { |input| input.binmode.read }
83
87
 
84
88
  destination ||= if block_given?
@@ -113,7 +117,9 @@ class Bundler::Thor
113
117
  context = config.delete(:context) || instance_eval("binding")
114
118
 
115
119
  create_file destination, nil, config do
116
- content = CapturableERB.new(::File.binread(source), nil, "-", "@output_buffer").result(context)
120
+ content = CapturableERB.new(::File.binread(source), nil, "-", "@output_buffer").tap do |erb|
121
+ erb.filename = source
122
+ end.result(context)
117
123
  content = yield(content) if block
118
124
  content
119
125
  end
@@ -134,7 +140,10 @@ class Bundler::Thor
134
140
  return unless behavior == :invoke
135
141
  path = File.expand_path(path, destination_root)
136
142
  say_status :chmod, relative_to_original_destination_root(path), config.fetch(:verbose, true)
137
- FileUtils.chmod_R(mode, path) unless options[:pretend]
143
+ unless options[:pretend]
144
+ require "fileutils"
145
+ FileUtils.chmod_R(mode, path)
146
+ end
138
147
  end
139
148
 
140
149
  # Prepend text to a file. Since it depends on insert_into_file, it's reversible.
@@ -204,6 +213,29 @@ class Bundler::Thor
204
213
  insert_into_file(path, *(args << config), &block)
205
214
  end
206
215
 
216
+ # Injects text right after the module definition. Since it depends on
217
+ # insert_into_file, it's reversible.
218
+ #
219
+ # ==== Parameters
220
+ # path<String>:: path of the file to be changed
221
+ # module_name<String|Class>:: the module to be manipulated
222
+ # data<String>:: the data to append to the class, can be also given as a block.
223
+ # config<Hash>:: give :verbose => false to not log the status.
224
+ #
225
+ # ==== Examples
226
+ #
227
+ # inject_into_module "app/helpers/application_helper.rb", ApplicationHelper, " def help; 'help'; end\n"
228
+ #
229
+ # inject_into_module "app/helpers/application_helper.rb", ApplicationHelper do
230
+ # " def help; 'help'; end\n"
231
+ # end
232
+ #
233
+ def inject_into_module(path, module_name, *args, &block)
234
+ config = args.last.is_a?(Hash) ? args.pop : {}
235
+ config[:after] = /module #{module_name}\n|module #{module_name} .*\n/
236
+ insert_into_file(path, *(args << config), &block)
237
+ end
238
+
207
239
  # Run a regular expression replacement on a file.
208
240
  #
209
241
  # ==== Parameters
@@ -288,7 +320,10 @@ class Bundler::Thor
288
320
  path = File.expand_path(path, destination_root)
289
321
 
290
322
  say_status :remove, relative_to_original_destination_root(path), config.fetch(:verbose, true)
291
- ::FileUtils.rm_rf(path) if !options[:pretend] && File.exist?(path)
323
+ if !options[:pretend] && File.exist?(path)
324
+ require "fileutils"
325
+ ::FileUtils.rm_rf(path)
326
+ end
292
327
  end
293
328
  alias_method :remove_dir, :remove_file
294
329
 
@@ -305,8 +340,10 @@ class Bundler::Thor
305
340
  with_output_buffer { yield(*args) }
306
341
  end
307
342
 
308
- def with_output_buffer(buf = "") #:nodoc:
309
- self.output_buffer, old_buffer = buf, output_buffer
343
+ def with_output_buffer(buf = "".dup) #:nodoc:
344
+ raise ArgumentError, "Buffer can not be a frozen object" if buf.frozen?
345
+ old_buffer = output_buffer
346
+ self.output_buffer = buf
310
347
  yield
311
348
  output_buffer
312
349
  ensure
@@ -319,7 +356,7 @@ class Bundler::Thor
319
356
  def set_eoutvar(compiler, eoutvar = "_erbout")
320
357
  compiler.put_cmd = "#{eoutvar}.concat"
321
358
  compiler.insert_cmd = "#{eoutvar}.concat"
322
- compiler.pre_cmd = ["#{eoutvar} = ''"]
359
+ compiler.pre_cmd = ["#{eoutvar} = ''.dup"]
323
360
  compiler.post_cmd = [eoutvar]
324
361
  end
325
362
  end
@@ -53,7 +53,13 @@ class Bundler::Thor
53
53
  replacement + '\0'
54
54
  end
55
55
 
56
- replace!(/#{flag}/, content, config[:force])
56
+ if exists?
57
+ replace!(/#{flag}/, content, config[:force])
58
+ else
59
+ unless pretend?
60
+ raise Bundler::Thor::Error, "The file #{ destination } does not appear to exist"
61
+ end
62
+ end
57
63
  end
58
64
 
59
65
  def revoke!
@@ -91,8 +97,8 @@ class Bundler::Thor
91
97
  # Adds the content to the file.
92
98
  #
93
99
  def replace!(regexp, string, force)
94
- return if base.options[:pretend]
95
- content = File.binread(destination)
100
+ return if pretend?
101
+ content = File.read(destination)
96
102
  if force || !content.include?(replacement)
97
103
  content.gsub!(regexp, string)
98
104
  File.open(destination, "wb") { |file| file.write(content) }
@@ -42,7 +42,7 @@ class Bundler::Thor
42
42
  # config<Hash>:: Configuration for this Bundler::Thor class.
43
43
  #
44
44
  def initialize(args = [], local_options = {}, config = {})
45
- parse_options = config[:current_command] && config[:current_command].disable_class_options ? {} : self.class.class_options
45
+ parse_options = self.class.class_options
46
46
 
47
47
  # The start method splits inbound arguments at the first argument
48
48
  # that looks like an option (starts with - or --). It then calls
@@ -65,7 +65,8 @@ class Bundler::Thor
65
65
  # declared options from the array. This will leave us with
66
66
  # a list of arguments that weren't declared.
67
67
  stop_on_unknown = self.class.stop_on_unknown_option? config[:current_command]
68
- opts = Bundler::Thor::Options.new(parse_options, hash_options, stop_on_unknown)
68
+ disable_required_check = self.class.disable_required_check? config[:current_command]
69
+ opts = Bundler::Thor::Options.new(parse_options, hash_options, stop_on_unknown, disable_required_check)
69
70
  self.options = opts.parse(array_options)
70
71
  self.options = config[:class_options].merge(options) if config[:class_options]
71
72
 
@@ -150,6 +151,21 @@ class Bundler::Thor
150
151
  !!check_unknown_options
151
152
  end
152
153
 
154
+ # If you want to raise an error when the default value of an option does not match
155
+ # the type call check_default_type!
156
+ # This is disabled by default for compatibility.
157
+ def check_default_type!
158
+ @check_default_type = true
159
+ end
160
+
161
+ def check_default_type #:nodoc:
162
+ @check_default_type ||= from_superclass(:check_default_type, false)
163
+ end
164
+
165
+ def check_default_type? #:nodoc:
166
+ !!check_default_type
167
+ end
168
+
153
169
  # If true, option parsing is suspended as soon as an unknown option or a
154
170
  # regular argument is encountered. All remaining arguments are passed to
155
171
  # the command as regular arguments.
@@ -157,6 +173,12 @@ class Bundler::Thor
157
173
  false
158
174
  end
159
175
 
176
+ # If true, option set will not suspend the execution of the command when
177
+ # a required option is not provided.
178
+ def disable_required_check?(command_name) #:nodoc:
179
+ false
180
+ end
181
+
160
182
  # If you want only strict string args (useful when cascading thor classes),
161
183
  # call strict_args_position! This is disabled by default to allow dynamic
162
184
  # invocations.
@@ -477,7 +499,8 @@ class Bundler::Thor
477
499
  alias_method :handle_no_task_error, :handle_no_command_error
478
500
 
479
501
  def handle_argument_error(command, error, args, arity) #:nodoc:
480
- msg = "ERROR: \"#{basename} #{command.name}\" was called with "
502
+ name = [command.ancestor_name, command.name].compact.join(" ")
503
+ msg = "ERROR: \"#{basename} #{name}\" was called with ".dup
481
504
  msg << "no arguments" if args.empty?
482
505
  msg << "arguments " << args.inspect unless args.empty?
483
506
  msg << "\nUsage: #{banner(command).inspect}"
@@ -541,7 +564,7 @@ class Bundler::Thor
541
564
  # options<Hash>:: Described in both class_option and method_option.
542
565
  # scope<Hash>:: Options hash that is being built up
543
566
  def build_option(name, options, scope) #:nodoc:
544
- scope[name] = Bundler::Thor::Option.new(name, options)
567
+ scope[name] = Bundler::Thor::Option.new(name, options.merge(:check_default_type => check_default_type?))
545
568
  end
546
569
 
547
570
  # Receives a hash of options, parse them and add to the scope. This is a
@@ -1,9 +1,9 @@
1
1
  class Bundler::Thor
2
- class Command < Struct.new(:name, :description, :long_description, :usage, :options, :disable_class_options)
2
+ class Command < Struct.new(:name, :description, :long_description, :usage, :options, :ancestor_name)
3
3
  FILE_REGEXP = /^#{Regexp.escape(File.dirname(__FILE__))}/
4
4
 
5
- def initialize(name, description, long_description, usage, options = nil, disable_class_options = false)
6
- super(name.to_s, description, long_description, usage, options || {}, disable_class_options)
5
+ def initialize(name, description, long_description, usage, options = nil)
6
+ super(name.to_s, description, long_description, usage, options || {})
7
7
  end
8
8
 
9
9
  def initialize_copy(other) #:nodoc:
@@ -39,13 +39,15 @@ class Bundler::Thor
39
39
  # Returns the formatted usage by injecting given required arguments
40
40
  # and required options into the given usage.
41
41
  def formatted_usage(klass, namespace = true, subcommand = false)
42
- if namespace
42
+ if ancestor_name
43
+ formatted = "#{ancestor_name} ".dup # add space
44
+ elsif namespace
43
45
  namespace = klass.namespace
44
- formatted = "#{namespace.gsub(/^(default)/, '')}:"
46
+ formatted = "#{namespace.gsub(/^(default)/, '')}:".dup
45
47
  end
46
- formatted = "#{klass.namespace.split(':').last} " if subcommand
48
+ formatted ||= "#{klass.namespace.split(':').last} ".dup if subcommand
47
49
 
48
- formatted ||= ""
50
+ formatted ||= "".dup
49
51
 
50
52
  # Add usage with required arguments
51
53
  formatted << if klass && !klass.arguments.empty?
@@ -51,6 +51,18 @@ class Bundler::Thor
51
51
  self
52
52
  end
53
53
 
54
+ def reverse_merge(other)
55
+ self.class.new(other).merge(self)
56
+ end
57
+
58
+ def reverse_merge!(other_hash)
59
+ replace(reverse_merge(other_hash))
60
+ end
61
+
62
+ def replace(other_hash)
63
+ super(other_hash)
64
+ end
65
+
54
66
  # Convert to a Hash with String keys.
55
67
  def to_hash
56
68
  Hash.new(default).merge!(self)
@@ -205,7 +205,7 @@ class Bundler::Thor::Group
205
205
  alias_method :printable_tasks, :printable_commands
206
206
 
207
207
  def handle_argument_error(command, error, _args, arity) #:nodoc:
208
- msg = "#{basename} #{command.name} takes #{arity} argument"
208
+ msg = "#{basename} #{command.name} takes #{arity} argument".dup
209
209
  msg << "s" if arity > 1
210
210
  msg << ", but it should not."
211
211
  raise error, msg
@@ -23,6 +23,8 @@ class Bundler::Thor
23
23
  if echo?
24
24
  $stdin.gets
25
25
  else
26
+ # Lazy-load io/console since it is gem-ified as of 2.3
27
+ require "io/console" if RUBY_VERSION > "1.9.2"
26
28
  $stdin.noecho(&:gets)
27
29
  end
28
30
  end
@@ -5,6 +5,7 @@ class Bundler::Thor
5
5
  VALID_TYPES = [:boolean, :numeric, :hash, :array, :string]
6
6
 
7
7
  def initialize(name, options = {})
8
+ @check_default_type = options[:check_default_type]
8
9
  options[:required] = false unless options.key?(:required)
9
10
  super
10
11
  @lazy_default = options[:lazy_default]
@@ -80,12 +81,12 @@ class Bundler::Thor
80
81
 
81
82
  def usage(padding = 0)
82
83
  sample = if banner && !banner.to_s.empty?
83
- "#{switch_name}=#{banner}"
84
+ "#{switch_name}=#{banner}".dup
84
85
  else
85
86
  switch_name
86
87
  end
87
88
 
88
- sample = "[#{sample}]" unless required?
89
+ sample = "[#{sample}]".dup unless required?
89
90
 
90
91
  if boolean?
91
92
  sample << ", [#{dasherize('no-' + human_name)}]" unless (name == "force") || name.start_with?("no-")
@@ -110,7 +111,7 @@ class Bundler::Thor
110
111
 
111
112
  def validate!
112
113
  raise ArgumentError, "An option cannot be boolean and required." if boolean? && required?
113
- validate_default_type!
114
+ validate_default_type! if @check_default_type
114
115
  end
115
116
 
116
117
  def validate_default_type!
@@ -127,8 +128,7 @@ class Bundler::Thor
127
128
  @default.class.name.downcase.to_sym
128
129
  end
129
130
 
130
- # TODO: This should raise an ArgumentError in a future version of Bundler::Thor
131
- warn "Expected #{@type} default value for '#{switch_name}'; got #{@default.inspect} (#{default_type})" unless default_type == @type
131
+ raise ArgumentError, "Expected #{@type} default value for '#{switch_name}'; got #{@default.inspect} (#{default_type})" unless default_type == @type
132
132
  end
133
133
 
134
134
  def dasherized?
@@ -18,19 +18,20 @@ class Bundler::Thor
18
18
  when Hash
19
19
  "--#{key} #{value.map { |k, v| "#{k}:#{v}" }.join(' ')}"
20
20
  when nil, false
21
- ""
21
+ nil
22
22
  else
23
23
  "--#{key} #{value.inspect}"
24
24
  end
25
- end.join(" ")
25
+ end.compact.join(" ")
26
26
  end
27
27
 
28
28
  # Takes a hash of Bundler::Thor::Option and a hash with defaults.
29
29
  #
30
30
  # If +stop_on_unknown+ is true, #parse will stop as soon as it encounters
31
31
  # an unknown option or a regular argument.
32
- def initialize(hash_options = {}, defaults = {}, stop_on_unknown = false)
32
+ def initialize(hash_options = {}, defaults = {}, stop_on_unknown = false, disable_required_check = false)
33
33
  @stop_on_unknown = stop_on_unknown
34
+ @disable_required_check = disable_required_check
34
35
  options = hash_options.values
35
36
  super(options)
36
37
 
@@ -111,7 +112,7 @@ class Bundler::Thor
111
112
  end
112
113
  end
113
114
 
114
- check_requirement!
115
+ check_requirement! unless @disable_required_check
115
116
 
116
117
  assigns = Bundler::Thor::CoreExt::HashWithIndifferentAccess.new(@assigns)
117
118
  assigns.freeze
@@ -188,7 +189,7 @@ class Bundler::Thor
188
189
  shift
189
190
  false
190
191
  else
191
- true
192
+ !no_or_skip?(switch)
192
193
  end
193
194
  else
194
195
  @switches.key?(switch) || !no_or_skip?(switch)
@@ -2,8 +2,6 @@ require "bundler/vendor/thor/lib/thor"
2
2
  require "bundler/vendor/thor/lib/thor/group"
3
3
  require "bundler/vendor/thor/lib/thor/core_ext/io_binary_read"
4
4
 
5
- require "fileutils"
6
- require "open-uri"
7
5
  require "yaml"
8
6
  require "digest/md5"
9
7
  require "pathname"
@@ -104,6 +102,7 @@ class Bundler::Thor::Runner < Bundler::Thor #:nodoc: # rubocop:disable ClassLeng
104
102
  if package == :file
105
103
  File.open(destination, "w") { |f| f.puts contents }
106
104
  else
105
+ require "fileutils"
107
106
  FileUtils.cp_r(name, destination)
108
107
  end
109
108
 
@@ -120,6 +119,7 @@ class Bundler::Thor::Runner < Bundler::Thor #:nodoc: # rubocop:disable ClassLeng
120
119
  def uninstall(name)
121
120
  raise Error, "Can't find module '#{name}'" unless thor_yaml[name]
122
121
  say "Uninstalling #{name}."
122
+ require "fileutils"
123
123
  FileUtils.rm_rf(File.join(thor_root, (thor_yaml[name][:filename]).to_s))
124
124
 
125
125
  thor_yaml.delete(name)
@@ -138,6 +138,7 @@ class Bundler::Thor::Runner < Bundler::Thor #:nodoc: # rubocop:disable ClassLeng
138
138
  self.options = options.merge("as" => name)
139
139
 
140
140
  if File.directory? File.expand_path(name)
141
+ require "fileutils"
141
142
  FileUtils.rm_rf(File.join(thor_root, old_filename))
142
143
 
143
144
  thor_yaml.delete(old_filename)
@@ -194,6 +195,7 @@ private
194
195
  yaml_file = File.join(thor_root, "thor.yml")
195
196
 
196
197
  unless File.exist?(yaml_file)
198
+ require "fileutils"
197
199
  FileUtils.mkdir_p(thor_root)
198
200
  yaml_file = File.join(thor_root, "thor.yml")
199
201
  FileUtils.touch(yaml_file)
@@ -1,6 +1,3 @@
1
- require "tempfile"
2
- require "io/console" if RUBY_VERSION > "1.9.2"
3
-
4
1
  class Bundler::Thor
5
2
  module Shell
6
3
  class Basic
@@ -110,7 +107,7 @@ class Bundler::Thor
110
107
  status = set_color status, color, true if color
111
108
 
112
109
  buffer = "#{status}#{spaces}#{message}"
113
- buffer << "\n" unless buffer.end_with?("\n")
110
+ buffer = "#{buffer}\n" unless buffer.end_with?("\n")
114
111
 
115
112
  stdout.print(buffer)
116
113
  stdout.flush
@@ -165,7 +162,7 @@ class Bundler::Thor
165
162
  colwidth = options[:colwidth]
166
163
  options[:truncate] = terminal_width if options[:truncate] == true
167
164
 
168
- formats << "%-#{colwidth + 2}s" if colwidth
165
+ formats << "%-#{colwidth + 2}s".dup if colwidth
169
166
  start = colwidth ? 1 : 0
170
167
 
171
168
  colcount = array.max { |a, b| a.size <=> b.size }.size
@@ -177,9 +174,9 @@ class Bundler::Thor
177
174
  maximas << maxima
178
175
  formats << if index == colcount - 1
179
176
  # Don't output 2 trailing spaces when printing the last column
180
- "%-s"
177
+ "%-s".dup
181
178
  else
182
- "%-#{maxima + 2}s"
179
+ "%-#{maxima + 2}s".dup
183
180
  end
184
181
  end
185
182
 
@@ -187,7 +184,7 @@ class Bundler::Thor
187
184
  formats << "%s"
188
185
 
189
186
  array.each do |row|
190
- sentence = ""
187
+ sentence = "".dup
191
188
 
192
189
  row.each_with_index do |column, index|
193
190
  maxima = maximas[index]
@@ -255,6 +252,9 @@ class Bundler::Thor
255
252
  )
256
253
 
257
254
  case answer
255
+ when nil
256
+ say ""
257
+ return true
258
258
  when is?(:yes), is?(:force), ""
259
259
  return true
260
260
  when is?(:no), is?(:skip)
@@ -350,6 +350,7 @@ class Bundler::Thor
350
350
  def show_diff(destination, content) #:nodoc:
351
351
  diff_cmd = ENV["THOR_DIFF"] || ENV["RAILS_DIFF"] || "diff -u"
352
352
 
353
+ require "tempfile"
353
354
  Tempfile.open(File.basename(destination), File.dirname(destination)) do |temp|
354
355
  temp.write content
355
356
  temp.rewind
@@ -411,7 +412,7 @@ class Bundler::Thor
411
412
 
412
413
  return unless result
413
414
 
414
- result.strip!
415
+ result = result.strip
415
416
 
416
417
  if default && result == ""
417
418
  default
@@ -1,3 +1,3 @@
1
1
  class Bundler::Thor
2
- VERSION = "0.19.4"
2
+ VERSION = "0.20.0"
3
3
  end
@@ -7,7 +7,7 @@ module Bundler
7
7
  # We're doing this because we might write tests that deal
8
8
  # with other versions of bundler and we are unsure how to
9
9
  # handle this better.
10
- VERSION = "1.15.3" unless defined?(::Bundler::VERSION)
10
+ VERSION = "1.15.4" unless defined?(::Bundler::VERSION)
11
11
 
12
12
  def self.overwrite_loaded_gem_version
13
13
  begin
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: bundler
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.15.3
4
+ version: 1.15.4
5
5
  platform: ruby
6
6
  authors:
7
7
  - André Arko
@@ -17,7 +17,7 @@ authors:
17
17
  autorequire:
18
18
  bindir: exe
19
19
  cert_chain: []
20
- date: 2017-07-21 00:00:00.000000000 Z
20
+ date: 2017-08-19 00:00:00.000000000 Z
21
21
  dependencies:
22
22
  - !ruby/object:Gem::Dependency
23
23
  name: automatiek