diff options
Diffstat (limited to 'spec/bundler/quality_spec.rb')
| -rw-r--r-- | spec/bundler/quality_spec.rb | 261 |
1 files changed, 261 insertions, 0 deletions
diff --git a/spec/bundler/quality_spec.rb b/spec/bundler/quality_spec.rb new file mode 100644 index 0000000000..16b7f18788 --- /dev/null +++ b/spec/bundler/quality_spec.rb @@ -0,0 +1,261 @@ +# frozen_string_literal: true + +require "set" + +RSpec.describe "The library itself" do + def check_for_git_merge_conflicts(filename) + merge_conflicts_regex = / + <<<<<<<| + =======| + >>>>>>> + /x + + failing_lines = [] + each_line(filename) do |line, number| + failing_lines << number + 1 if line&.match?(merge_conflicts_regex) + end + + return if failing_lines.empty? + "#{filename} has unresolved git merge conflicts on lines #{failing_lines.join(", ")}" + end + + def check_for_tab_characters(filename) + # Because Go uses hard tabs + return if filename.end_with?(".go.tt") + + failing_lines = [] + each_line(filename) do |line, number| + failing_lines << number + 1 if line.include?("\t") + end + + return if failing_lines.empty? + "#{filename} has tab characters on lines #{failing_lines.join(", ")}" + end + + def check_for_extra_spaces(filename) + failing_lines = [] + each_line(filename) do |line, number| + next if /^\s+#.*\s+\n$/.match?(line) + failing_lines << number + 1 if /\s+\n$/.match?(line) + end + + return if failing_lines.empty? + "#{filename} has spaces on the EOL on lines #{failing_lines.join(", ")}" + end + + def check_for_extraneous_quotes(filename) + failing_lines = [] + each_line(filename) do |line, number| + failing_lines << number + 1 if /\u{2019}/.match?(line) + end + + return if failing_lines.empty? + "#{filename} has an extraneous quote on lines #{failing_lines.join(", ")}" + end + + def check_for_expendable_words(filename) + failing_line_message = [] + useless_words = %w[ + actually + basically + clearly + just + obviously + really + simply + ] + pattern = /\b#{Regexp.union(useless_words)}\b/i + + each_line(filename) do |line, number| + next unless word_found = pattern.match(line) + failing_line_message << "#{filename}:#{number.succ} has '#{word_found}'. Avoid using these kinds of weak modifiers." + end + + failing_line_message unless failing_line_message.empty? + end + + def check_for_specific_pronouns(filename) + failing_line_message = [] + specific_pronouns = /\b(he|she|his|hers|him|her|himself|herself)\b/i + + each_line(filename) do |line, number| + next unless word_found = specific_pronouns.match(line) + failing_line_message << "#{filename}:#{number.succ} has '#{word_found}'. Use more generic pronouns in documentation." + end + + failing_line_message unless failing_line_message.empty? + end + + it "has no malformed whitespace" do + exempt = /\.gitmodules|fixtures|vendor|LICENSE|vcr_cassettes|rbreadline\.diff|index\.txt$/ + error_messages = [] + tracked_files.each do |filename| + next if filename&.match?(exempt) + error_messages << check_for_tab_characters(filename) + error_messages << check_for_extra_spaces(filename) + end + expect(error_messages.compact).to be_well_formed + end + + it "has no extraneous quotes" do + exempt = /vendor|vcr_cassettes|LICENSE|rbreadline\.diff/ + error_messages = [] + tracked_files.each do |filename| + next if filename&.match?(exempt) + error_messages << check_for_extraneous_quotes(filename) + end + expect(error_messages.compact).to be_well_formed + end + + it "does not include any unresolved merge conflicts" do + error_messages = [] + exempt = %r{lock/lockfile_spec|quality_spec|vcr_cassettes|\.ronn|lockfile_parser} + tracked_files.each do |filename| + next if filename&.match?(exempt) + error_messages << check_for_git_merge_conflicts(filename) + end + expect(error_messages.compact).to be_well_formed + end + + it "maintains language quality of the documentation" do + error_messages = [] + man_tracked_files.each do |filename| + error_messages << check_for_expendable_words(filename) + error_messages << check_for_specific_pronouns(filename) + end + expect(error_messages.compact).to be_well_formed + end + + it "maintains language quality of sentences used in source code" do + error_messages = [] + exempt = /vendor|vcr_cassettes|CODE_OF_CONDUCT/ + lib_tracked_files.each do |filename| + next if filename&.match?(exempt) + error_messages << check_for_expendable_words(filename) + error_messages << check_for_specific_pronouns(filename) + end + expect(error_messages.compact).to be_well_formed + end + + it "documents all used settings" do + exemptions = %w[ + gem.changelog + gem.ci + gem.coc + gem.linter + gem.mit + gem.bundle + gem.rubocop + gem.test + git.allow_insecure + inline + trust-policy + ] + + all_settings = Hash.new {|h, k| h[k] = [] } + documented_settings = [] + + Bundler::Settings::BOOL_KEYS.each {|k| all_settings[k] << "in Bundler::Settings::BOOL_KEYS" } + Bundler::Settings::NUMBER_KEYS.each {|k| all_settings[k] << "in Bundler::Settings::NUMBER_KEYS" } + Bundler::Settings::ARRAY_KEYS.each {|k| all_settings[k] << "in Bundler::Settings::ARRAY_KEYS" } + Bundler::Settings::STRING_KEYS.each {|k| all_settings[k] << "in Bundler::Settings::STRING_KEYS" } + + key_pattern = /([a-z\._-]+)/i + lib_tracked_files.each do |filename| + each_line(filename) do |line, number| + line.scan(/Bundler\.settings\[:#{key_pattern}\]/).flatten.each {|s| all_settings[s] << "referenced at `#{filename}:#{number.succ}`" } + end + end + settings_section = File.read(source_root.join("lib/bundler/man/bundle-config.1.ronn")).split(/^## /).find {|section| section.start_with?("LIST OF AVAILABLE KEYS") } + documented_settings = settings_section.scan(/^\* `#{key_pattern}`/).flatten + + documented_settings.each do |s| + all_settings.delete(s) + expect(exemptions.delete(s)).to be_nil, "setting #{s} was exempted but was actually documented" + end + + exemptions.each do |s| + expect(all_settings.delete(s)).to be_truthy, "setting #{s} was exempted but unused" + end + error_messages = all_settings.map do |setting, refs| + "The `#{setting}` setting is undocumented\n\t- #{refs.join("\n\t- ")}\n" + end + + expect(error_messages.sort).to be_well_formed + + expect(documented_settings).to be_sorted + end + + it "can still be built" do + with_built_bundler do |gem_path| + expect(File.exist?(gem_path)).to be true + end + end + + it "ships the correct set of files" do + git_list = tracked_files.reject {|f| f.start_with?("spec/") } + + gem_list = loaded_gemspec.files + gem_list.map! {|f| f.sub(%r{\Aexe/}, "libexec/") } if ruby_core? + + expect(git_list).to match_array(gem_list) + end + + it "does not contain any warnings" do + exclusions = %w[ + lib/bundler/capistrano.rb + lib/bundler/deployment.rb + lib/bundler/gem_tasks.rb + lib/bundler/vlad.rb + ] + files_to_require = lib_tracked_files.grep(/\.rb$/) - exclusions + files_to_require.reject! {|f| f.start_with?("lib/bundler/vendor") } + files_to_require.map! {|f| File.expand_path(f, source_root) } + files_to_require.sort! + sys_exec("ruby -w") do |input, _, _| + files_to_require.each do |f| + input.puts "require '#{f}'" + end + end + + warnings = stdboth.split("\n") + # ignore warnings around deprecated Object#=~ method in RubyGems + warnings.reject! {|w| w =~ %r{rubygems\/version.rb.*deprecated\ Object#=~} } + + expect(warnings).to be_well_formed + end + + it "does not use require internally, but require_relative" do + exempt = %r{templates/|\.5|\.1|vendor/} + all_bad_requires = [] + lib_tracked_files.each do |filename| + next if filename&.match?(exempt) + each_line(filename) do |line, number| + line.scan(/^ *require "bundler/).each { all_bad_requires << "#{filename}:#{number.succ}" } + end + end + + expect(all_bad_requires).to be_empty, "#{all_bad_requires.size} internal requires that should use `require_relative`: #{all_bad_requires}" + end + + # We don't want our artifice code to activate bundler, but it needs to use the + # namespaced implementation of `Net::HTTP`. So we duplicate the file in + # bundler that loads that. + it "keeps vendored_net_http spec code in sync with the lib implementation" do + lib_implementation_path = File.join(source_lib_dir, "bundler", "vendored_net_http.rb") + expect(File.exist?(lib_implementation_path)).to be_truthy + lib_code = File.read(lib_implementation_path) + + spec_implementation_path = File.join(spec_dir, "support", "vendored_net_http.rb") + expect(File.exist?(spec_implementation_path)).to be_truthy + spec_code = File.read(spec_implementation_path) + + expect(lib_code).to eq(spec_code) + end + + private + + def each_line(filename, &block) + File.readlines(File.expand_path(filename, source_root), encoding: "UTF-8").each_with_index(&block) + end +end |
