forked from Shopify/erb_lint
-
Notifications
You must be signed in to change notification settings - Fork 0
/
hard_coded_string.rb
144 lines (117 loc) · 4.22 KB
/
hard_coded_string.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
# frozen_string_literal: true
require "set"
require "better_html/tree/tag"
require "active_support/core_ext/string/inflections"
module ERBLint
module Linters
# Checks for hardcoded strings. Useful if you want to ensure a string can be translated using i18n.
class HardCodedString < Linter
include LinterRegistry
ForbiddenCorrector = Class.new(StandardError)
MissingCorrector = Class.new(StandardError)
MissingI18nLoadPath = Class.new(StandardError)
ALLOWED_CORRECTORS = ["I18nCorrector", "RuboCop::Corrector::I18n::HardCodedString"]
NON_TEXT_TAGS = Set.new(["script", "style", "xmp", "iframe", "noembed", "noframes", "listing"])
NO_TRANSLATION_NEEDED = Set.new([
" ",
"&",
"<",
">",
""",
"©",
"®",
"™",
"…",
"—",
"•",
"“",
"”",
"‘",
"’",
"←",
"→",
"↓",
"↑",
" ",
" ",
" ",
"×",
])
class ConfigSchema < LinterConfig
property :corrector, accepts: Hash, required: false, default: -> { {} }
property :i18n_load_path, accepts: String, required: false, default: ""
end
self.config_schema = ConfigSchema
def run(processed_source)
hardcoded_strings = processed_source.ast.descendants(:text).each_with_object([]) do |text_node, to_check|
next if non_text_tag?(processed_source, text_node)
offended_strings = text_node.to_a.select { |node| relevant_node(node) }
offended_strings.each do |offended_string|
offended_string.split("\n").each do |str|
to_check << [text_node, str] if check_string?(str)
end
end
end
hardcoded_strings.compact.each do |text_node, offended_str|
range = find_range(text_node, offended_str)
source_range = processed_source.to_source_range(range)
add_offense(
source_range,
message(source_range.source)
)
end
end
def find_range(node, str)
match = node.loc.source.match(Regexp.new(Regexp.quote(str.strip)))
return unless match
range_begin = match.begin(0) + node.loc.begin_pos
range_end = match.end(0) + node.loc.begin_pos
(range_begin...range_end)
end
def autocorrect(processed_source, offense)
string = offense.source_range.source
return unless (klass = load_corrector)
return unless string.strip.length > 1
node = ::RuboCop::AST::StrNode.new(:str, [string])
corrector = klass.new(node, processed_source.filename, corrector_i18n_load_path, offense.source_range)
corrector.autocorrect(tag_start: "<%= ", tag_end: " %>")
rescue MissingCorrector, MissingI18nLoadPath
nil
end
private
def check_string?(str)
string = str.gsub(/\s*/, "")
string.length > 1 && !NO_TRANSLATION_NEEDED.include?(string)
end
def load_corrector
corrector_name = @config["corrector"].fetch("name") { raise MissingCorrector }
raise ForbiddenCorrector unless ALLOWED_CORRECTORS.include?(corrector_name)
require @config["corrector"].fetch("path") { raise MissingCorrector }
corrector_name.safe_constantize
end
def corrector_i18n_load_path
@config["corrector"].fetch("i18n_load_path") { raise MissingI18nLoadPath }
end
def non_text_tag?(processed_source, text_node)
ast = processed_source.parser.ast.to_a
index = ast.find_index(text_node)
previous_node = ast[index - 1]
if previous_node.type == :tag
tag = BetterHtml::Tree::Tag.from_node(previous_node)
NON_TEXT_TAGS.include?(tag.name) && !tag.closing?
end
end
def relevant_node(inner_node)
if inner_node.is_a?(String)
inner_node.strip.empty? ? false : inner_node
else
false
end
end
def message(string)
stripped_string = string.strip
"String not translated: #{stripped_string}"
end
end
end
end