2025-01-22 23:17:22 +00:00
|
|
|
# typed: strict
|
2022-08-05 15:35:25 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require "forwardable"
|
|
|
|
|
|
|
|
module RuboCop
|
|
|
|
module Cop
|
|
|
|
module Cask
|
|
|
|
# This cop audits variables in casks.
|
|
|
|
#
|
2024-04-26 20:55:51 +02:00
|
|
|
# ### Example
|
2022-08-05 15:35:25 -04:00
|
|
|
#
|
2024-04-26 20:55:51 +02:00
|
|
|
# ```ruby
|
|
|
|
# # bad
|
|
|
|
# cask do
|
|
|
|
# arch = Hardware::CPU.intel? ? "darwin" : "darwin-arm64"
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # good
|
|
|
|
# cask 'foo' do
|
|
|
|
# arch arm: "darwin-arm64", intel: "darwin"
|
|
|
|
# end
|
|
|
|
# ```
|
2022-08-05 15:35:25 -04:00
|
|
|
class Variables < Base
|
|
|
|
extend Forwardable
|
|
|
|
extend AutoCorrector
|
|
|
|
include CaskHelp
|
|
|
|
|
2025-01-22 23:17:22 +00:00
|
|
|
sig { override.params(cask_block: RuboCop::Cask::AST::CaskBlock).void }
|
2022-08-05 15:35:25 -04:00
|
|
|
def on_cask(cask_block)
|
2025-01-22 23:17:22 +00:00
|
|
|
@cask_block = T.let(cask_block, T.nilable(RuboCop::Cask::AST::CaskBlock))
|
2022-08-05 15:35:25 -04:00
|
|
|
add_offenses
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def_delegator :@cask_block, :cask_node
|
|
|
|
|
2025-01-22 23:17:22 +00:00
|
|
|
sig { void }
|
2022-08-05 15:35:25 -04:00
|
|
|
def add_offenses
|
|
|
|
variable_assignment(cask_node) do |node, var_name, arch_condition, true_node, false_node|
|
|
|
|
arm_node, intel_node = if arch_condition == :arm?
|
|
|
|
[true_node, false_node]
|
|
|
|
else
|
|
|
|
[false_node, true_node]
|
|
|
|
end
|
|
|
|
|
|
|
|
replacement_string = if var_name == :arch
|
|
|
|
"arch "
|
|
|
|
else
|
|
|
|
"#{var_name} = on_arch_conditional "
|
|
|
|
end
|
|
|
|
replacement_parameters = []
|
|
|
|
replacement_parameters << "arm: #{arm_node.source}" unless blank_node?(arm_node)
|
|
|
|
replacement_parameters << "intel: #{intel_node.source}" unless blank_node?(intel_node)
|
|
|
|
replacement_string += replacement_parameters.join(", ")
|
|
|
|
|
2023-05-08 11:02:44 +02:00
|
|
|
add_offense(node, message: "Use `#{replacement_string}` instead of `#{node.source}`.") do |corrector|
|
2022-08-05 15:35:25 -04:00
|
|
|
corrector.replace(node, replacement_string)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2025-01-22 23:17:22 +00:00
|
|
|
sig { params(node: RuboCop::AST::Node).returns(T::Boolean) }
|
2022-08-05 15:35:25 -04:00
|
|
|
def blank_node?(node)
|
|
|
|
case node.type
|
|
|
|
when :str
|
2025-01-22 23:17:22 +00:00
|
|
|
node.str_content.empty?
|
2022-08-05 15:35:25 -04:00
|
|
|
when :nil
|
|
|
|
true
|
|
|
|
else
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def_node_search :variable_assignment, <<~PATTERN
|
|
|
|
$(lvasgn $_ (if (send (const (const nil? :Hardware) :CPU) ${:arm? :intel?}) $_ $_))
|
|
|
|
PATTERN
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|