Merge pull request #314 from crystal-ameba/Sija/style-query-bool-methods-rule

This commit is contained in:
Sijawusz Pur Rahnama 2022-12-08 13:14:19 +01:00 committed by GitHub
commit 150ba6c70f
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 159 additions and 9 deletions

View file

@ -0,0 +1,72 @@
require "../../../spec_helper"
module Ameba::Rule::Style
subject = QueryBoolMethods.new
describe QueryBoolMethods do
it "passes for valid cases" do
expect_no_issues subject, <<-CRYSTAL
class Foo
class_property? foo = true
property? foo = true
property foo2 : Bool? = true
setter panda = true
end
module Bar
class_getter? bar : Bool = true
getter? bar : Bool
getter bar2 : Bool? = true
setter panda : Bool = true
def initialize(@bar = true)
end
end
CRYSTAL
end
it "reports only valid properties" do
expect_issue subject, <<-CRYSTAL
class Foo
class_property? foo = true
class_property bar = true
# ^^^ error: Consider using 'class_property?' for 'bar'
class_property baz = true
# ^^^ error: Consider using 'class_property?' for 'baz'
end
CRYSTAL
end
{% for call in %w[getter class_getter property class_property] %}
it "reports `{{ call.id }}` assign with Bool" do
expect_issue subject, <<-CRYSTAL, call: {{ call }}
class Foo
%{call} foo = true
_{call} # ^^^ error: Consider using '%{call}?' for 'foo'
end
CRYSTAL
end
it "reports `{{ call.id }}` type declaration assign with Bool" do
expect_issue subject, <<-CRYSTAL, call: {{ call }}
class Foo
%{call} foo : Bool = true
_{call} # ^ error: Consider using '%{call}?' for 'foo'
end
CRYSTAL
end
it "reports `{{ call.id }}` type declaration with Bool" do
expect_issue subject, <<-CRYSTAL, call: {{ call }}
class Foo
%{call} foo : Bool
_{call} # ^ error: Consider using '%{call}?' for 'foo'
def initialize(@foo = true)
end
end
CRYSTAL
end
{% end %}
end
end

View file

@ -58,6 +58,13 @@ module Ameba::AST::Util
is_literal is_literal
end end
# Returns `true` if current `node` is a `Crystal::Path`
# matching given *name*, `false` otherwise.
def path_named?(node, name) : Bool
node.is_a?(Crystal::Path) &&
name == node.names.join("::")
end
# Returns a source code for the current node. # Returns a source code for the current node.
# This method uses `node.location` and `node.end_location` # This method uses `node.location` and `node.end_location`
# to determine and cut a piece of source of the node. # to determine and cut a piece of source of the node.

View file

@ -122,11 +122,10 @@ module Ameba::AST
# `false` if not. # `false` if not.
def used_in_macro?(scope = @scope) def used_in_macro?(scope = @scope)
scope.inner_scopes.each do |inner_scope| scope.inner_scopes.each do |inner_scope|
return true if MacroReferenceFinder.new(inner_scope.node, node.name).references return true if MacroReferenceFinder.new(inner_scope.node, node.name).references?
end end
return true if MacroReferenceFinder.new(scope.node, node.name).references return true if MacroReferenceFinder.new(scope.node, node.name).references?
return true if (outer_scope = scope.outer_scope) && return true if (outer_scope = scope.outer_scope) && used_in_macro?(outer_scope)
used_in_macro?(outer_scope)
false false
end end
@ -169,7 +168,7 @@ module Ameba::AST
end end
private class MacroReferenceFinder < Crystal::Visitor private class MacroReferenceFinder < Crystal::Visitor
property references = false property? references = false
def initialize(node, @reference : String = reference) def initialize(node, @reference : String = reference)
node.accept self node.accept self

View file

@ -20,19 +20,19 @@ module Ameba::Rule::Style
# Enabled: true # Enabled: true
# ``` # ```
class IsANil < Base class IsANil < Base
include AST::Util
properties do properties do
description "Disallows calls to `is_a?(Nil)` in favor of `nil?`" description "Disallows calls to `is_a?(Nil)` in favor of `nil?`"
end end
MSG = "Use `nil?` instead of `is_a?(Nil)`" MSG = "Use `nil?` instead of `is_a?(Nil)`"
PATH_NIL_NAMES = %w(Nil)
def test(source, node : Crystal::IsA) def test(source, node : Crystal::IsA)
return if node.nil_check? return if node.nil_check?
const = node.const const = node.const
return unless const.is_a?(Crystal::Path) return unless path_named?(const, "Nil")
return unless const.names == PATH_NIL_NAMES
issue_for const, MSG issue_for const, MSG
end end

View file

@ -0,0 +1,72 @@
module Ameba::Rule::Style
# A rule that disallows boolean properties without the `?` suffix - defined
# using `Object#(class_)property` or `Object#(class_)getter` macros.
#
# Favour this:
#
# ```
# class Person
# property? deceased = false
# getter? witty = true
# end
# ```
#
# Over this:
#
# ```
# class Person
# property deceased = false
# getter witty = true
# end
# ```
#
# YAML configuration example:
#
# ```
# Style/QueryBoolMethods:
# Enabled: true
# ```
class QueryBoolMethods < Base
include AST::Util
properties do
description "Reports boolean properties without the `?` suffix"
end
MSG = "Consider using '%s?' for '%s'"
CALL_NAMES = %w[getter class_getter property class_property]
def test(source, node : Crystal::ClassDef | Crystal::ModuleDef)
calls =
case body = node.body
when Crystal::Call
[body] if body.name.in?(CALL_NAMES)
when Crystal::Expressions
body.expressions
.select(Crystal::Call)
.select!(&.name.in?(CALL_NAMES))
end
return unless calls
calls.each do |exp|
exp.args.each do |arg|
name_node, is_bool =
case arg
when Crystal::Assign
{arg.target, arg.value.is_a?(Crystal::BoolLiteral)}
when Crystal::TypeDeclaration
{arg.var, path_named?(arg.declared_type, "Bool")}
else
{nil, false}
end
if name_node && is_bool
issue_for name_node, MSG % {exp.name, name_node}
end
end
end
end
end
end