mirror of
https://gitea.invidious.io/iv-org/shard-ameba.git
synced 2024-08-15 00:53:29 +00:00
Merge pull request #321 from crystal-ameba/Sija/lint-missing-block-argument-rule
This commit is contained in:
commit
b78e2aebc5
5 changed files with 94 additions and 3 deletions
|
@ -2,7 +2,7 @@ require "../../spec_helper"
|
||||||
require "file_utils"
|
require "file_utils"
|
||||||
|
|
||||||
module Ameba
|
module Ameba
|
||||||
private def with_formatter
|
private def with_formatter(&)
|
||||||
io = IO::Memory.new
|
io = IO::Memory.new
|
||||||
formatter = Formatter::TODOFormatter.new(io)
|
formatter = Formatter::TODOFormatter.new(io)
|
||||||
|
|
||||||
|
|
42
spec/ameba/rule/lint/missing_block_argument_spec.cr
Normal file
42
spec/ameba/rule/lint/missing_block_argument_spec.cr
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
require "../../../spec_helper"
|
||||||
|
|
||||||
|
module Ameba::Rule::Lint
|
||||||
|
subject = MissingBlockArgument.new
|
||||||
|
|
||||||
|
describe MissingBlockArgument do
|
||||||
|
it "passes if the block argument is defined" do
|
||||||
|
expect_no_issues subject, <<-CRYSTAL
|
||||||
|
def foo(&)
|
||||||
|
yield 42
|
||||||
|
end
|
||||||
|
|
||||||
|
def bar(&block)
|
||||||
|
yield 24
|
||||||
|
end
|
||||||
|
|
||||||
|
def baz(a, b, c, &block)
|
||||||
|
yield a, b, c
|
||||||
|
end
|
||||||
|
CRYSTAL
|
||||||
|
end
|
||||||
|
|
||||||
|
it "reports if the block argument is missing" do
|
||||||
|
expect_issue subject, <<-CRYSTAL
|
||||||
|
def foo
|
||||||
|
# ^^^ error: Missing anonymous block argument. Use `&` as an argument name to indicate yielding method.
|
||||||
|
yield 42
|
||||||
|
end
|
||||||
|
|
||||||
|
def bar
|
||||||
|
# ^^^ error: Missing anonymous block argument. Use `&` as an argument name to indicate yielding method.
|
||||||
|
yield 24
|
||||||
|
end
|
||||||
|
|
||||||
|
def baz(a, b, c)
|
||||||
|
# ^^^ error: Missing anonymous block argument. Use `&` as an argument name to indicate yielding method.
|
||||||
|
yield a, b, c
|
||||||
|
end
|
||||||
|
CRYSTAL
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
49
src/ameba/rule/lint/missing_block_argument.cr
Normal file
49
src/ameba/rule/lint/missing_block_argument.cr
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
module Ameba::Rule::Lint
|
||||||
|
# A rule that disallows yielding method definitions without block argument.
|
||||||
|
#
|
||||||
|
# For example, this is considered invalid:
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# def foo
|
||||||
|
# yield 42
|
||||||
|
# end
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# And has to be written as the following:
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# def foo(&)
|
||||||
|
# yield 42
|
||||||
|
# end
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# YAML configuration example:
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# Lint/MissingBlockArgument:
|
||||||
|
# Enabled: true
|
||||||
|
# ```
|
||||||
|
class MissingBlockArgument < Base
|
||||||
|
include AST::Util
|
||||||
|
|
||||||
|
properties do
|
||||||
|
description "Disallows yielding method definitions without block argument"
|
||||||
|
end
|
||||||
|
|
||||||
|
MSG = "Missing anonymous block argument. Use `&` as an argument " \
|
||||||
|
"name to indicate yielding method."
|
||||||
|
|
||||||
|
def test(source)
|
||||||
|
AST::ScopeVisitor.new self, source
|
||||||
|
end
|
||||||
|
|
||||||
|
def test(source, node : Crystal::Def, scope : AST::Scope)
|
||||||
|
return if !scope.yields? || node.block_arg
|
||||||
|
|
||||||
|
return unless location = node.name_location
|
||||||
|
end_location = name_end_location(node)
|
||||||
|
|
||||||
|
issue_for location, end_location, MSG
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -179,7 +179,7 @@ module Ameba
|
||||||
|
|
||||||
private MAX_ITERATIONS = 200
|
private MAX_ITERATIONS = 200
|
||||||
|
|
||||||
private def loop_unless_infinite(source, corrected_issues)
|
private def loop_unless_infinite(source, corrected_issues, &)
|
||||||
# Keep track of the state of the source. If a rule modifies the source
|
# Keep track of the state of the source. If a rule modifies the source
|
||||||
# and another rule undoes it producing identical source we have an
|
# and another rule undoes it producing identical source we have an
|
||||||
# infinite loop.
|
# infinite loop.
|
||||||
|
|
|
@ -111,7 +111,7 @@ class Ameba::Source::Rewriter
|
||||||
end
|
end
|
||||||
|
|
||||||
# Similar to `@children.bsearch_index || size` except allows for a starting point
|
# Similar to `@children.bsearch_index || size` except allows for a starting point
|
||||||
protected def bsearch_child_index(from = 0)
|
protected def bsearch_child_index(from = 0, &)
|
||||||
size = @children.size
|
size = @children.size
|
||||||
(from...size).bsearch { |i| yield @children[i] } || size
|
(from...size).bsearch { |i| yield @children[i] } || size
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue