module SCSSLint # Finds and aggregates all lints found by running the registered linters # against a set of SCSS files. class Runner attr_reader :lints, :files # @param config [Config] def initialize(config) @config = config @lints = [] @linters = LinterRegistry.linters.select { |linter| @config.linter_enabled?(linter) } @linters.map!(&:new) end # @param files [Array] list of file object/path hashes def run(files) @files = files @files.each do |file| find_lints(file) end end private # @param file [Hash] # @option file [String] File object # @option path [String] path to File (determines which Linter config to apply) def find_lints(file) # rubocop:disable AbcSize options = file.merge(preprocess_command: @config.options['preprocess_command'], preprocess_files: @config.options['preprocess_files']) engine = Engine.new(options) @linters.each do |linter| begin run_linter(linter, engine, file[:path]) rescue StandardError => e raise SCSSLint::Exceptions::LinterError, "#{linter.class} raised unexpected error linting file #{file[:path]}: " \ "'#{e.message}'", e.backtrace end end rescue Sass::SyntaxError => e @lints << Lint.new(Linter::Syntax.new, e.sass_filename, Location.new(e.sass_line), "Syntax Error: #{e}", :error) rescue FileEncodingError => e @lints << Lint.new(Linter::Encoding.new, file[:path], Location.new, e.to_s, :error) end # For stubbing in tests. def run_linter(linter, engine, file_path) return if @config.excluded_file_for_linter?(file_path, linter) @lints += linter.run(engine, @config.linter_options(linter)) end end end