mirror of
https://gitea.invidious.io/iv-org/shard-ameba.git
synced 2024-08-15 00:53:29 +00:00
Add Lint/Formatting
rule
This commit is contained in:
parent
e6ebca7a5b
commit
3bc8bda008
2 changed files with 113 additions and 0 deletions
48
spec/ameba/rule/lint/formatting_spec.cr
Normal file
48
spec/ameba/rule/lint/formatting_spec.cr
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
require "../../../spec_helper"
|
||||||
|
|
||||||
|
module Ameba::Rule::Lint
|
||||||
|
describe Formatting do
|
||||||
|
subject = Formatting.new
|
||||||
|
|
||||||
|
it "passes if source is formatted" do
|
||||||
|
expect_no_issues subject, <<-CRYSTAL
|
||||||
|
def method(a, b)
|
||||||
|
a + b
|
||||||
|
end
|
||||||
|
|
||||||
|
CRYSTAL
|
||||||
|
end
|
||||||
|
|
||||||
|
it "reports if source is not formatted" do
|
||||||
|
expect_issue subject, <<-CRYSTAL
|
||||||
|
def method(a,b)
|
||||||
|
# ^{} error: Use built-in formatter to format this source
|
||||||
|
end
|
||||||
|
CRYSTAL
|
||||||
|
end
|
||||||
|
|
||||||
|
context "properties" do
|
||||||
|
context "#fail_on_error" do
|
||||||
|
it "passes on formatter errors by default" do
|
||||||
|
rule = Formatting.new
|
||||||
|
|
||||||
|
expect_no_issues rule, <<-CRYSTAL
|
||||||
|
def method(a, b)
|
||||||
|
a + b
|
||||||
|
CRYSTAL
|
||||||
|
end
|
||||||
|
|
||||||
|
it "reports on formatter errors when enabled" do
|
||||||
|
rule = Formatting.new
|
||||||
|
rule.fail_on_error = true
|
||||||
|
|
||||||
|
expect_issue rule, <<-CRYSTAL
|
||||||
|
def method(a, b)
|
||||||
|
a + b
|
||||||
|
# ^ error: Error while formatting: expecting identifier 'end', not 'EOF'
|
||||||
|
CRYSTAL
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
65
src/ameba/rule/lint/formatting.cr
Normal file
65
src/ameba/rule/lint/formatting.cr
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
require "compiler/crystal/formatter"
|
||||||
|
|
||||||
|
module Ameba::Rule::Lint
|
||||||
|
# A rule that verifies syntax formatting according to the
|
||||||
|
# Crystal's built-in formatter.
|
||||||
|
#
|
||||||
|
# For example, this syntax is invalid:
|
||||||
|
#
|
||||||
|
# def foo(a,b,c=0)
|
||||||
|
# #foobar
|
||||||
|
# a+b+c
|
||||||
|
# end
|
||||||
|
#
|
||||||
|
# And should be properly written:
|
||||||
|
#
|
||||||
|
# def foo(a, b, c = 0)
|
||||||
|
# # foobar
|
||||||
|
# a + b + c
|
||||||
|
# end
|
||||||
|
#
|
||||||
|
# YAML configuration example:
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# Lint/Formatting:
|
||||||
|
# Enabled: true
|
||||||
|
# FailOnError: false
|
||||||
|
# ```
|
||||||
|
class Formatting < Base
|
||||||
|
properties do
|
||||||
|
description "Reports not formatted sources"
|
||||||
|
fail_on_error false
|
||||||
|
end
|
||||||
|
|
||||||
|
MSG = "Use built-in formatter to format this source"
|
||||||
|
MSG_ERROR = "Error while formatting: %s"
|
||||||
|
|
||||||
|
private LOCATION = {1, 1}
|
||||||
|
|
||||||
|
def test(source)
|
||||||
|
source_code = source.code
|
||||||
|
result = Crystal.format(source_code, source.path)
|
||||||
|
return if result == source_code
|
||||||
|
|
||||||
|
source_lines = source_code.lines
|
||||||
|
return if source_lines.empty?
|
||||||
|
|
||||||
|
end_location = {
|
||||||
|
source_lines.size,
|
||||||
|
source_lines.last.size + 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_for LOCATION, MSG do |corrector|
|
||||||
|
corrector.replace(LOCATION, end_location, result)
|
||||||
|
end
|
||||||
|
rescue ex : Crystal::SyntaxException
|
||||||
|
if fail_on_error?
|
||||||
|
issue_for({ex.line_number, ex.column_number}, MSG_ERROR % ex.message)
|
||||||
|
end
|
||||||
|
rescue ex
|
||||||
|
if fail_on_error?
|
||||||
|
issue_for(LOCATION, MSG_ERROR % ex.message)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue