# typed: strict # frozen_string_literal: true require 'rubocop/packs/private' module RuboCop module Packs # Pack-level rubocop and rubocop_todo YML files are named differently because they are not integrated # into rubocop in the standard way. For example, we could call these the standard `.rubocop.yml` and # `.rubocop_todo.yml`. However, this introduces a number of path relativity issues (https://docs.rubocop.org/rubocop/configuration.html#path-relativity) # that make this approach not possible. Therefore, for pack level rubocops, we name them in a way that mirrors packwerk `package_todo.yml` files # for consistency and to ensure that thes are not read by rubocop except via the ERB templating mechanism. PACK_LEVEL_RUBOCOP_YML = 'package_rubocop.yml' PACK_LEVEL_RUBOCOP_TODO_YML = 'package_rubocop_todo.yml' class Error < StandardError; end extend T::Sig # Your code goes here... PROJECT_ROOT = T.let(Pathname.new(__dir__).parent.parent.expand_path.freeze, Pathname) CONFIG_DEFAULT = T.let(PROJECT_ROOT.join('config', 'default.yml').freeze, Pathname) CONFIG = T.let(YAML.safe_load(CONFIG_DEFAULT.read).freeze, T.untyped) private_constant(:CONFIG_DEFAULT, :PROJECT_ROOT) class Offense < T::Struct extend T::Sig const :cop_name, String const :filepath, String sig { returns(ParsePackwerk::Package) } def pack ParsePackwerk.package_from_path(filepath) end end sig { params(paths: T::Array[String], cop_names: T::Array[String]).returns(T::Array[Offense]) } def self.offenses_for(paths:, cop_names:) path_arguments = paths.join(' ') cop_arguments = cop_names.join(',') # I think we can potentially use `RuboCop::CLI.new(args)` for this to avoid shelling out and starting another process that needs to reload the bundle args = [path_arguments, "--only=#{cop_arguments}", '--format=json'] puts "Executing: bundle exec rubocop #{args.join(' ')}" json = JSON.parse(Private.execute_rubocop(args)) offenses = T.let([], T::Array[Offense]) json['files'].each do |file_hash| filepath = file_hash['path'] file_hash['offenses'].each do |offense_hash| offenses << Offense.new( cop_name: offense_hash['cop_name'], filepath: filepath ) end end offenses end # # Ideally, this is API that is available to us via `rubocop` itself. # That is: the ability to preserve the location of `.rubocop_todo.yml` files and associate # exclusions with the closest ancestor `.rubocop_todo.yml` # sig { params(packs: T::Array[ParsePackwerk::Package]).void } def self.auto_generate_rubocop_todo(packs:) offenses = offenses_for( paths: packs.map(&:name).reject { |name| name == ParsePackwerk::ROOT_PACKAGE_NAME }, cop_names: config.permitted_pack_level_cops ) offenses.group_by(&:pack).each do |pack, offenses_for_pack| rubocop_todo_yml = pack.directory.join(PACK_LEVEL_RUBOCOP_TODO_YML) if rubocop_todo_yml.exist? rubocop_todo = YAML.load_file(rubocop_todo_yml) else rubocop_todo = {} end offenses_for_pack.each do |offense| rubocop_todo[offense.cop_name] ||= { 'Exclude' => [] } rubocop_todo[offense.cop_name]['Exclude'] << offense.filepath end next if rubocop_todo.empty? rubocop_todo_yml.write(YAML.dump(rubocop_todo)) end end # # Ideally, this is API that is available to us via `rubocop` itself. # That is: the ability to preserve the location of `.rubocop_todo.yml` files and associate # exclusions with the closest ancestor `.rubocop_todo.yml` # sig { params(packs: T::Array[ParsePackwerk::Package]).void } def self.set_default_rubocop_yml(packs:) packs.each do |pack| rubocop_yml = Pathname.new(pack.directory.join(PACK_LEVEL_RUBOCOP_YML)) rubocop_yml_hash = {} config.required_pack_level_cops.each do |cop| rubocop_yml_hash[cop] = { 'Enabled' => true } end formatted_yml = YAML.dump(rubocop_yml_hash). # Find lines of the form \nCopDepartment/CopName: and add a new line before it. gsub(%r{^(\w+/\w+:)}, "\n\\1"). # Remove the `---` header at the top of the file gsub("---\n\n", '') rubocop_yml.write(formatted_yml) end end sig { params(root_pathname: String).returns(String) } # It would be great if rubocop (upstream) could take in a glob for `inherit_from`, which # would allow us to delete this method and this additional complexity. def self.pack_based_rubocop_config(root_pathname: Bundler.root) rubocop_config = {} # We do this because when the ERB is evaluated Dir.pwd is at the directory containing the YML. # Ideally rubocop wouldn't change the PWD before invoking this method. Dir.chdir(root_pathname) do ParsePackwerk.all.each do |package| next if package.name == ParsePackwerk::ROOT_PACKAGE_NAME rubocop_todo = package.directory.join(PACK_LEVEL_RUBOCOP_TODO_YML) if rubocop_todo.exist? loaded_rubocop_todo = YAML.load_file(rubocop_todo) loaded_rubocop_todo.each do |cop_name, key_config| rubocop_config[cop_name] ||= {} rubocop_config[cop_name]['Exclude'] ||= [] rubocop_config[cop_name]['Exclude'] += key_config['Exclude'] end end pack_rubocop = package.directory.join(PACK_LEVEL_RUBOCOP_YML) next unless pack_rubocop.exist? loaded_pack_rubocop = YAML.load_file(pack_rubocop) loaded_pack_rubocop.each do |cop_name, key_config| next unless key_config['Enabled'] rubocop_config[cop_name] ||= {} rubocop_config[cop_name]['Include'] ||= [] rubocop_config[cop_name]['Include'] << package.directory.join('**/*').to_s end end end YAML.dump(rubocop_config) end sig { void } def self.bust_cache! config.bust_cache! Private.bust_cache! end sig { params(blk: T.proc.params(arg0: Private::Configuration).void).void } def self.configure(&blk) yield(config) end sig { returns(Private::Configuration) } def self.config Private.load_client_configuration @config = T.let(@config, T.nilable(Private::Configuration)) @config ||= Private::Configuration.new end # We can remove this function once package_protections is fully deprecated sig { params(rule: String).returns(T::Set[String]) } def self.exclude_for_rule(rule) Private.exclude_for_rule(rule) end # # Note: when we add per-pack `.rubocop.yml` files, we'll want to add some validations here # to restrict what cops are permitted to be configured in those files. # We might also want further (configurable?) constraints *requiring* that the "permitted pack level cops" be specified explicitly. # sig { returns(T::Array[String]) } def self.validate errors = T.let([], T::Array[String]) ParsePackwerk.all.each do |package| next if package.name == ParsePackwerk::ROOT_PACKAGE_NAME errors += Private.validate_rubocop_todo_yml(package) errors += Private.validate_rubocop_yml(package) errors += Private.validate_failure_mode_strict(package) end errors end end end