2020-04-21 11:21:10 -04:00
# frozen_string_literal: true
module RuboCop
module Cop
class StaticTranslationDefinition < RuboCop :: Cop :: Cop
2021-03-25 23:09:21 -04:00
MSG = " The text you're translating will be already in the translated form when it's assigned to the constant. When a users changes the locale, these texts won't be translated again. Consider moving the translation logic to a method. "
2020-04-21 11:21:10 -04:00
TRANSLATION_METHODS = % i [ _ s_ n_ ] . freeze
def_node_matcher :translation_method? , << ~ PATTERN
2021-11-19 19:14:01 -05:00
( send _ _ str * )
2020-04-21 11:21:10 -04:00
PATTERN
def_node_matcher :lambda_node? , << ~ PATTERN
2021-11-19 19:14:01 -05:00
( send _ :lambda )
PATTERN
def_node_matcher :struct_constant_assignment? , << ~ PATTERN
( casgn _ _ ` (const _ :Struct))
2020-04-21 11:21:10 -04:00
PATTERN
def on_send ( node )
return unless translation_method? ( node )
method_name = node . children [ 1 ]
return unless TRANSLATION_METHODS . include? ( method_name )
2020-09-11 05:08:44 -04:00
translation_memoized = false
2020-04-21 11:21:10 -04:00
node . each_ancestor do | ancestor |
receiver , _ = * ancestor
break if lambda_node? ( receiver ) # translations defined in lambda nodes should be allowed
2021-11-19 19:14:01 -05:00
if constant_assignment? ( ancestor ) && ! struct_constant_assignment? ( ancestor )
2020-04-21 11:21:10 -04:00
add_offense ( node , location : :expression )
break
end
2020-09-11 05:08:44 -04:00
translation_memoized = true if memoization? ( ancestor )
if translation_memoized && class_method_definition? ( ancestor )
add_offense ( node , location : :expression )
break
end
2020-04-21 11:21:10 -04:00
end
end
private
def constant_assignment? ( node )
node . type == :casgn
end
2020-09-11 05:08:44 -04:00
def memoization? ( node )
node . type == :or_asgn
end
def class_method_definition? ( node )
node . type == :defs
end
2020-04-21 11:21:10 -04:00
end
end
end