2019-05-02 18:17:27 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2018-09-04 17:57:20 -04:00
|
|
|
require 'i18n/i18n_interpolation_keys_finder'
|
|
|
|
require 'yaml'
|
|
|
|
|
|
|
|
class LocaleFileChecker
|
2018-09-06 10:54:15 -04:00
|
|
|
TYPE_MISSING_INTERPOLATION_KEYS = 1
|
|
|
|
TYPE_UNSUPPORTED_INTERPOLATION_KEYS = 2
|
|
|
|
TYPE_MISSING_PLURAL_KEYS = 3
|
2018-09-06 11:27:17 -04:00
|
|
|
TYPE_INVALID_MESSAGE_FORMAT = 4
|
2021-07-14 07:26:12 -04:00
|
|
|
TYPE_INVALID_MARKDOWN_LINK = 5
|
2018-09-04 17:57:20 -04:00
|
|
|
|
|
|
|
def check(locale)
|
|
|
|
@errors = {}
|
|
|
|
@locale = locale.to_s
|
|
|
|
|
|
|
|
locale_files.each do |locale_path|
|
|
|
|
next unless reference_path = reference_file(locale_path)
|
|
|
|
|
|
|
|
@relative_locale_path = Pathname.new(locale_path).relative_path_from(Pathname.new(Rails.root)).to_s
|
|
|
|
@locale_yaml = YAML.load_file(locale_path)
|
|
|
|
@reference_yaml = YAML.load_file(reference_path)
|
|
|
|
|
2021-07-14 07:26:12 -04:00
|
|
|
next if @locale_yaml.blank? || @locale_yaml.first[1].blank?
|
|
|
|
|
2020-06-14 18:15:08 -04:00
|
|
|
check_interpolation_keys
|
|
|
|
check_plural_keys
|
2018-09-06 11:27:17 -04:00
|
|
|
check_message_format
|
2021-07-14 07:26:12 -04:00
|
|
|
check_markdown_links
|
2018-09-04 17:57:20 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
@errors
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
YML_DIRS = ["config/locales", "plugins/**/locales"]
|
|
|
|
PLURALS_FILE = "config/locales/plurals.rb"
|
|
|
|
REFERENCE_LOCALE = "en"
|
|
|
|
REFERENCE_PLURAL_KEYS = ["one", "other"]
|
|
|
|
|
|
|
|
# Some languages should always use %{count} in pluralized strings.
|
|
|
|
# https://meta.discourse.org/t/always-use-count-variable-when-translating-pluralized-strings/83969
|
2019-05-12 17:48:57 -04:00
|
|
|
FORCE_PLURAL_COUNT_LOCALES = ["bs", "fr", "lt", "lv", "ru", "sl", "sr", "uk"]
|
2018-09-04 17:57:20 -04:00
|
|
|
|
|
|
|
def locale_files
|
|
|
|
YML_DIRS.map { |dir| Dir["#{Rails.root}/#{dir}/{client,server}.#{@locale}.yml"] }.flatten
|
|
|
|
end
|
|
|
|
|
|
|
|
def reference_file(path)
|
|
|
|
path = path.gsub(/\.\w{2,}\.yml$/, ".#{REFERENCE_LOCALE}.yml")
|
|
|
|
path if File.exists?(path)
|
|
|
|
end
|
|
|
|
|
|
|
|
def traverse_hash(hash, parent_keys, &block)
|
|
|
|
hash.each do |key, value|
|
|
|
|
keys = parent_keys.dup << key
|
|
|
|
|
|
|
|
if value.is_a?(Hash)
|
|
|
|
traverse_hash(value, keys, &block)
|
|
|
|
else
|
|
|
|
yield(keys, value, hash)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def check_interpolation_keys
|
|
|
|
traverse_hash(@locale_yaml, []) do |keys, value|
|
|
|
|
reference_value = reference_value(keys)
|
|
|
|
next if reference_value.nil?
|
|
|
|
|
|
|
|
if pluralized = reference_value_pluralized?(reference_value)
|
|
|
|
if keys.last == "one" && !FORCE_PLURAL_COUNT_LOCALES.include?(@locale)
|
|
|
|
reference_value = reference_value["one"]
|
|
|
|
else
|
|
|
|
reference_value = reference_value["other"]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
reference_interpolation_keys = I18nInterpolationKeysFinder.find(reference_value.to_s)
|
|
|
|
locale_interpolation_keys = I18nInterpolationKeysFinder.find(value.to_s)
|
|
|
|
|
|
|
|
missing_keys = reference_interpolation_keys - locale_interpolation_keys
|
|
|
|
unsupported_keys = locale_interpolation_keys - reference_interpolation_keys
|
|
|
|
|
|
|
|
# English strings often don't use the %{count} variable within the "one" key,
|
|
|
|
# but it's perfectly fine for other locales to use it.
|
|
|
|
unsupported_keys.delete("count") if pluralized && keys.last == "one"
|
|
|
|
|
|
|
|
# Not all locales need the %{count} variable within the "one" key.
|
|
|
|
if pluralized && keys.last == "one" && !FORCE_PLURAL_COUNT_LOCALES.include?(@locale)
|
|
|
|
missing_keys.delete("count")
|
|
|
|
end
|
|
|
|
|
2018-09-06 10:54:15 -04:00
|
|
|
add_error(keys, TYPE_MISSING_INTERPOLATION_KEYS, missing_keys, pluralized: pluralized) unless missing_keys.empty?
|
|
|
|
add_error(keys, TYPE_UNSUPPORTED_INTERPOLATION_KEYS, unsupported_keys, pluralized: pluralized) unless unsupported_keys.empty?
|
2018-09-04 17:57:20 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-07-14 07:26:12 -04:00
|
|
|
def check_markdown_links
|
|
|
|
traverse_hash(@locale_yaml, []) do |keys, value|
|
|
|
|
next if value.is_a?(Array)
|
|
|
|
|
|
|
|
if /\[.*?\]\s+\(.*?\)/.match?(value)
|
|
|
|
add_error(keys, TYPE_INVALID_MARKDOWN_LINK, nil, pluralized: false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-09-04 17:57:20 -04:00
|
|
|
def check_plural_keys
|
|
|
|
known_parent_keys = Set.new
|
|
|
|
|
|
|
|
traverse_hash(@locale_yaml, []) do |keys, _, parent|
|
|
|
|
keys = keys[0..-2]
|
|
|
|
parent_key = keys.join(".")
|
|
|
|
next if known_parent_keys.include?(parent_key)
|
|
|
|
known_parent_keys << parent_key
|
|
|
|
|
|
|
|
reference_value = reference_value(keys)
|
|
|
|
next if reference_value.nil? || !reference_value_pluralized?(reference_value)
|
|
|
|
|
|
|
|
expected_plural_keys = plural_keys[@locale]
|
|
|
|
actual_plural_keys = parent.is_a?(Hash) ? parent.keys : []
|
|
|
|
missing_plural_keys = expected_plural_keys - actual_plural_keys
|
|
|
|
|
2018-09-06 10:54:15 -04:00
|
|
|
add_error(keys, TYPE_MISSING_PLURAL_KEYS, missing_plural_keys, pluralized: true) unless missing_plural_keys.empty?
|
2018-09-04 17:57:20 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-09-06 11:27:17 -04:00
|
|
|
def check_message_format
|
2019-02-19 08:42:58 -05:00
|
|
|
mf_locale, mf_filename = JsLocaleHelper.find_message_format_locale([@locale], fallback_to_english: true)
|
2018-09-06 11:27:17 -04:00
|
|
|
|
|
|
|
traverse_hash(@locale_yaml, []) do |keys, value|
|
|
|
|
next unless keys.last.ends_with?("_MF")
|
|
|
|
|
|
|
|
begin
|
|
|
|
JsLocaleHelper.with_context do |ctx|
|
|
|
|
ctx.load(mf_filename) if File.exist?(mf_filename)
|
|
|
|
ctx.eval("mf = new MessageFormat('#{mf_locale}');")
|
|
|
|
ctx.eval("mf.precompile(mf.parse(#{value.to_s.inspect}))")
|
|
|
|
end
|
|
|
|
rescue MiniRacer::EvalError => error
|
|
|
|
error_message = error.message.sub(/at undefined[:\d]+/, "").strip
|
|
|
|
add_error(keys, TYPE_INVALID_MESSAGE_FORMAT, error_message, pluralized: false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-09-04 17:57:20 -04:00
|
|
|
def reference_value(keys)
|
|
|
|
value = @reference_yaml[REFERENCE_LOCALE]
|
|
|
|
|
|
|
|
keys[1..-2].each do |key|
|
|
|
|
value = value[key]
|
|
|
|
return nil if value.nil?
|
|
|
|
end
|
|
|
|
|
|
|
|
reference_value_pluralized?(value) ? value : value[keys.last]
|
|
|
|
end
|
|
|
|
|
|
|
|
def reference_value_pluralized?(value)
|
|
|
|
value.is_a?(Hash) &&
|
|
|
|
value.keys.sort == REFERENCE_PLURAL_KEYS &&
|
|
|
|
value.keys.all? { |k| value[k].is_a?(String) }
|
|
|
|
end
|
|
|
|
|
|
|
|
def plural_keys
|
|
|
|
@plural_keys ||= begin
|
2021-10-27 04:39:28 -04:00
|
|
|
eval(File.read("#{Rails.root}/#{PLURALS_FILE}")).map do |locale, value| # rubocop:disable Security/Eval
|
2018-09-04 17:57:20 -04:00
|
|
|
[locale.to_s, value[:i18n][:plural][:keys].map(&:to_s)]
|
|
|
|
end.to_h
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-09-06 10:54:15 -04:00
|
|
|
def add_error(keys, type, details, pluralized:)
|
2018-09-04 17:57:20 -04:00
|
|
|
@errors[@relative_locale_path] ||= []
|
2018-09-06 10:54:15 -04:00
|
|
|
|
|
|
|
if pluralized
|
|
|
|
joined_key = keys[1..-2].join(".") << " [#{keys.last}]"
|
|
|
|
else
|
|
|
|
joined_key = keys[1..-1].join(".")
|
|
|
|
end
|
|
|
|
|
2018-09-04 17:57:20 -04:00
|
|
|
@errors[@relative_locale_path] << {
|
2018-09-06 10:54:15 -04:00
|
|
|
key: joined_key,
|
2018-09-04 17:57:20 -04:00
|
|
|
type: type,
|
|
|
|
details: details.to_s
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|