require 'pp'
require 'yaml'
require 'simplabs/excellent/parsing/parser'
require 'simplabs/excellent/parsing/code_processor'
module Simplabs
module Excellent
# The Runner is the interface to invoke parsing and processing of source code. You can pass either a String containing the code to process or the
# name of a file to read the code to process from.
class Runner
DEFAULT_CONFIG = {
:AssignmentInConditionalCheck => { },
:CaseMissingElseCheck => { },
:ClassLineCountCheck => { :threshold => 300 },
:ClassNameCheck => { :pattern => /^[A-Z][a-zA-Z0-9]*$/ },
:SingletonVariableCheck => { },
:CyclomaticComplexityBlockCheck => { :complexity => 4 },
:CyclomaticComplexityMethodCheck => { :complexity => 8 },
:EmptyRescueBodyCheck => { },
:ForLoopCheck => { },
:MethodLineCountCheck => { :line_count => 20 },
:MethodNameCheck => { :pattern => /^[_a-z<>=\[|+-\/\*`]+[_a-z0-9_<>=~@\[\]]*[=!\?]?$/ },
:ModuleLineCountCheck => { :line_count => 300 },
:ModuleNameCheck => { :pattern => /^[A-Z][a-zA-Z0-9]*$/ },
:ParameterNumberCheck => { :parameter_count => 3 },
:FlogMethodCheck => { },
:FlogBlockCheck => { },
:FlogClassCheck => { },
:'Rails::AttrProtectedCheck' => { },
:'Rails::AttrAccessibleCheck' => { }
}
attr_accessor :config #:nodoc:
# Initializes a Runner
#
# ==== Parameters
#
# * checks - The checks to apply - pass instances of the various check classes. If no checks are specified, all checks will be applied.
def initialize(*checks)
@config = DEFAULT_CONFIG
@checks = checks unless checks.empty?
@parser = Parsing::Parser.new
end
# Processes the +code+ and sets the file name of the warning to +filename+
#
# ==== Parameters
#
# * filename - The name of the file the code was read from.
# * code - The code to process (String).
def check(filename, code)
@checks ||= load_checks
@processor ||= Parsing::CodeProcessor.new(@checks)
node = parse(filename, code)
@processor.process(node)
end
# Processes the +code+, setting the file name of the warnings to '+dummy-file.rb+'
#
# ==== Parameters
#
# * code - The code to process (String).
def check_code(code)
check('dummy-file.rb', code)
end
# Processes the file +filename+. The code will be read from the file.
#
# ==== Parameters
#
# * filename - The name of the file to read the code from.
def check_file(filename)
check(filename, File.read(filename))
end
# Processes the passed +paths+
#
# ==== Parameters
#
# * paths - The paths to process (specify file names or directories; will recursively process all ruby files if a directory are given).
# * formatter - The formatter to use. If a formatter is specified, its +start+, +file+, +warning+ and +end+ methods will be called
def check_paths(paths, formatter = nil)
formatter.start if formatter
collect_files(paths).each do |path|
check_file(path)
format_file_and_warnings(formatter, path) if formatter
end
formatter.end if formatter
end
# Gets the warnings that were produced by the checks.
def warnings
@checks ||= []
@checks.collect { |check| check.warnings }.flatten
end
private
def parse(filename, code)
@parser.parse(code, filename)
end
def load_checks
check_objects = []
DEFAULT_CONFIG.each_pair do |key, value|
klass = eval("Simplabs::Excellent::Checks::#{key.to_s}")
check_objects << (value.empty? ? klass.new : klass.new(value))
end
check_objects
end
def collect_files(paths)
files = []
paths.each do |path|
if File.file?(path)
files << path
elsif File.directory?(path)
files += Dir.glob(File.join(path, '**/*.rb'))
else
raise ArgumentError.new("#{path} is neither a File nor a directory!")
end
end
files
end
def format_file_and_warnings(formatter, filename)
warnings = @checks.map { |check| check.warnings_for(filename) }.flatten
if warnings.length > 0
formatter.file(filename) do |formatter|
warnings.sort { |x, y| x.line_number <=> y.line_number }.each { |warning| formatter.warning(warning) }
end
end
end
end
end
end