mirror of
https://github.com/discourse/discourse.git
synced 2025-01-12 19:43:53 +08:00
69205cb1e5
This configuration makes it so that a missing translation will raise an error during test execution. Better discover there than after deploy.
331 lines
11 KiB
Ruby
331 lines
11 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
RSpec.describe "translate accelerator" do
|
|
before do
|
|
@original_i18n_load_path = I18n.load_path.dup
|
|
I18n.load_path += Dir["#{Rails.root}/spec/fixtures/i18n/translate_accelerator.*.yml"]
|
|
I18n.reload!
|
|
end
|
|
|
|
after do
|
|
I18n.load_path = @original_i18n_load_path
|
|
I18n.reload!
|
|
end
|
|
|
|
def override_translation(locale, key, value)
|
|
expect(I18n.exists?(key, locale)).to eq(true)
|
|
override = TranslationOverride.upsert!(locale, key, value)
|
|
expect(override.persisted?).to eq(true)
|
|
end
|
|
|
|
it "supports raising if requested, and cache bypasses" do
|
|
expect { I18n.t("i_am_an_unknown_key99", raise: true) }.to raise_error(
|
|
I18n::MissingTranslationData,
|
|
)
|
|
|
|
allow_missing_translations do
|
|
orig = I18n.t("i_am_an_unknown_key99")
|
|
|
|
expect(I18n.t("i_am_an_unknown_key99").object_id).to eq(orig.object_id)
|
|
expect(I18n.t("i_am_an_unknown_key99")).to eq("Translation missing: en.i_am_an_unknown_key99")
|
|
end
|
|
end
|
|
|
|
it "has the same 'translation missing' message as upstream" do
|
|
allow_missing_translations do
|
|
expect(I18n.t("this_key_does_not_exist")).to eq(
|
|
I18n.translate_no_cache("this_key_does_not_exist"),
|
|
)
|
|
end
|
|
end
|
|
|
|
it "returns the correct language" do
|
|
expect(I18n.t("foo", locale: :en)).to eq("Foo in :en")
|
|
expect(I18n.t("foo", locale: :de)).to eq("Foo in :de")
|
|
|
|
I18n.with_locale(:en) { expect(I18n.t("foo")).to eq("Foo in :en") }
|
|
|
|
I18n.with_locale(:de) { expect(I18n.t("foo")).to eq("Foo in :de") }
|
|
end
|
|
|
|
it "converts language keys to symbols" do
|
|
expect(I18n.t("foo", locale: :en)).to eq("Foo in :en")
|
|
expect(I18n.t("foo", locale: "en")).to eq("Foo in :en")
|
|
|
|
expect(I18n.instance_variable_get(:@loaded_locales)).to contain_exactly(:en)
|
|
end
|
|
|
|
it "overrides for both string and symbol keys" do
|
|
key = "user.email.not_allowed"
|
|
text_overridden = "foobar"
|
|
|
|
expect(I18n.t(key)).to be_present
|
|
|
|
override_translation("en", key, text_overridden)
|
|
|
|
expect(I18n.t(key)).to eq(text_overridden)
|
|
expect(I18n.t(key.to_sym)).to eq(text_overridden)
|
|
end
|
|
|
|
describe ".overrides_by_locale" do
|
|
it "should cache overrides for each locale" do
|
|
override_translation("en", "got", "summer")
|
|
override_translation("zh_TW", "got", "冬季")
|
|
|
|
I18n.overrides_by_locale("en")
|
|
I18n.overrides_by_locale("zh_TW")
|
|
|
|
expect(I18n.instance_variable_get(:@overrides_by_site)).to eq(
|
|
"default" => {
|
|
en: {
|
|
"got" => "summer",
|
|
},
|
|
zh_TW: {
|
|
"got" => "冬季",
|
|
},
|
|
},
|
|
)
|
|
end
|
|
end
|
|
|
|
describe "plugins" do
|
|
before do
|
|
DiscoursePluginRegistry.register_locale(
|
|
"foo",
|
|
name: "Foo",
|
|
nativeName: "Foo Bar",
|
|
plural: {
|
|
keys: %i[one few other],
|
|
rule:
|
|
lambda do |n|
|
|
return :one if n == 1
|
|
return :few if n < 10
|
|
:other
|
|
end,
|
|
},
|
|
)
|
|
|
|
LocaleSiteSetting.reset!
|
|
I18n.reload!
|
|
end
|
|
|
|
after do
|
|
DiscoursePluginRegistry.reset!
|
|
LocaleSiteSetting.reset!
|
|
end
|
|
|
|
it "loads plural rules from plugins" do
|
|
I18n.locale = :foo
|
|
|
|
expect(I18n.t("i18n.plural.keys")).to eq(%i[one few other])
|
|
expect(I18n.t("items", count: 1)).to eq("one item")
|
|
expect(I18n.t("items", count: 3)).to eq("some items")
|
|
expect(I18n.t("items", count: 20)).to eq("20 items")
|
|
end
|
|
end
|
|
|
|
describe "with overrides" do
|
|
before { I18n.locale = :en }
|
|
|
|
it "returns the overridden key" do
|
|
override_translation("en", "foo", "Overwritten foo")
|
|
expect(I18n.t("foo")).to eq("Overwritten foo")
|
|
|
|
override_translation("en", "foo", "new value")
|
|
expect(I18n.t("foo")).to eq("new value")
|
|
end
|
|
|
|
it "returns the overridden key after switching the locale" do
|
|
override_translation("en", "foo", "Overwritten foo in EN")
|
|
override_translation("de", "foo", "Overwritten foo in DE")
|
|
|
|
expect(I18n.t("foo")).to eq("Overwritten foo in EN")
|
|
I18n.locale = :de
|
|
expect(I18n.t("foo")).to eq("Overwritten foo in DE")
|
|
end
|
|
|
|
it "can be searched" do
|
|
override_translation("en", "wat", "Overwritten value")
|
|
expect(I18n.search("wat")).to include("wat" => "Overwritten value")
|
|
expect(I18n.search("Overwritten")).to include("wat" => "Overwritten value")
|
|
|
|
override_translation("en", "wat", "Overwritten with (parentheses)")
|
|
expect(I18n.search("Overwritten with (")).to include(
|
|
"wat" => "Overwritten with (parentheses)",
|
|
)
|
|
end
|
|
|
|
it "supports disabling" do
|
|
orig_title = I18n.t("title")
|
|
override_translation("en", "title", "overridden title")
|
|
|
|
I18n.overrides_disabled { expect(I18n.t("title")).to eq(orig_title) }
|
|
|
|
expect(I18n.t("title")).to eq("overridden title")
|
|
end
|
|
|
|
it "supports interpolation" do
|
|
override_translation("en", "world", "my %{world}")
|
|
expect(I18n.t("world", world: "foo")).to eq("my foo")
|
|
end
|
|
|
|
it "supports interpolation named count" do
|
|
override_translation("en", "wat", "goodbye %{count}")
|
|
expect(I18n.t("wat", count: 123)).to eq("goodbye 123")
|
|
end
|
|
|
|
it "ignores interpolation named count if it is not applicable" do
|
|
override_translation("en", "wat", "bar")
|
|
expect(I18n.t("wat", count: 1)).to eq("bar")
|
|
end
|
|
|
|
it "supports one and other" do
|
|
override_translation("en", "items.one", "one fish")
|
|
override_translation("en", "items.other", "%{count} fishies")
|
|
expect(I18n.t("items", count: 13)).to eq("13 fishies")
|
|
expect(I18n.t("items", count: 1)).to eq("one fish")
|
|
end
|
|
|
|
it "works with strings and symbols for non-pluralized string when count is given" do
|
|
override_translation("en", "fish", "trout")
|
|
expect(I18n.t(:fish, count: 1)).to eq("trout")
|
|
expect(I18n.t("fish", count: 1)).to eq("trout")
|
|
end
|
|
|
|
it "supports one and other with fallback locale" do
|
|
override_translation("en_GB", "items.one", "one fish")
|
|
override_translation("en_GB", "items.other", "%{count} fishies")
|
|
|
|
I18n.with_locale(:en_GB) do
|
|
expect(I18n.t("items", count: 13)).to eq("13 fishies")
|
|
expect(I18n.t("items", count: 1)).to eq("one fish")
|
|
end
|
|
end
|
|
|
|
it "supports one and other when only a single pluralization key is overridden" do
|
|
override_translation("en", "keys.magic.other", "no magic keys")
|
|
expect(I18n.t("keys.magic", count: 1)).to eq("one magic key")
|
|
expect(I18n.t("keys.magic", count: 2)).to eq("no magic keys")
|
|
end
|
|
|
|
it "returns the overridden text when falling back" do
|
|
override_translation("en", "got", "summer")
|
|
expect(I18n.t("got")).to eq("summer")
|
|
expect(I18n.with_locale(:zh_TW) { I18n.t("got") }).to eq("summer")
|
|
|
|
override_translation("en", "throne", "%{title} is the new queen")
|
|
expect(I18n.t("throne", title: "snow")).to eq("snow is the new queen")
|
|
expect(I18n.with_locale(:en) { I18n.t("throne", title: "snow") }).to eq(
|
|
"snow is the new queen",
|
|
)
|
|
end
|
|
|
|
it "returns override if it exists before falling back" do
|
|
expect(I18n.t("got", default: "")).to eq("winter")
|
|
expect(I18n.with_locale(:ru) { I18n.t("got", default: "") }).to eq("winter")
|
|
|
|
override_translation("ru", "got", "summer")
|
|
expect(I18n.t("got", default: "")).to eq("winter")
|
|
expect(I18n.with_locale(:ru) { I18n.t("got", default: "") }).to eq("summer")
|
|
end
|
|
|
|
it "does not affect ActiveModel::Naming#human" do
|
|
Fish = Class.new(ActiveRecord::Base)
|
|
|
|
override_translation("en", "fish", "fake fish")
|
|
expect(Fish.model_name.human).to eq("Fish")
|
|
end
|
|
|
|
it "works when the override contains an interpolation key" do
|
|
expect(I18n.t("foo_with_variable")).to eq("Foo in :en with %{variable}")
|
|
I18n.with_locale(:de) do
|
|
expect(I18n.t("foo_with_variable")).to eq("Foo in :de with %{variable}")
|
|
end
|
|
|
|
override_translation("en", "foo_with_variable", "Override in :en with %{variable}")
|
|
expect(I18n.t("foo_with_variable")).to eq("Override in :en with %{variable}")
|
|
I18n.with_locale(:de) do
|
|
expect(I18n.t("foo_with_variable")).to eq("Foo in :de with %{variable}")
|
|
end
|
|
|
|
override_translation("de", "foo_with_variable", "Override in :de with %{variable}")
|
|
expect(I18n.t("foo_with_variable")).to eq("Override in :en with %{variable}")
|
|
I18n.with_locale(:de) do
|
|
expect(I18n.t("foo_with_variable")).to eq("Override in :de with %{variable}")
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "translation precedence" do
|
|
def translation_should_equal(key, expected_value)
|
|
I18n.locale = :en
|
|
expect(I18n.t(key, locale: :de)).to eq(expected_value)
|
|
expect(I18n.search(key, locale: :de)[key]).to eq(expected_value)
|
|
|
|
I18n.locale = :de
|
|
expect(I18n.t(key)).to eq(expected_value)
|
|
expect(I18n.search(key)[key]).to eq(expected_value)
|
|
end
|
|
|
|
context "with existing translations in current locale and fallback locale" do
|
|
context "with overrides in both locales" do
|
|
it "should return the override from the current locale" do
|
|
override_translation("de", "foo", "Override of foo in :de")
|
|
override_translation("en", "foo", "Override of foo in :en")
|
|
translation_should_equal("foo", "Override of foo in :de")
|
|
end
|
|
end
|
|
|
|
context "with override only in current locale" do
|
|
it "should return the override from the current locale" do
|
|
override_translation("de", "foo", "Override of foo in :de")
|
|
translation_should_equal("foo", "Override of foo in :de")
|
|
end
|
|
end
|
|
|
|
context "with override only in fallback locale" do
|
|
it "should return the translation from the current locale" do
|
|
override_translation("en", "foo", "Override of foo in :en")
|
|
translation_should_equal("foo", "Foo in :de")
|
|
end
|
|
end
|
|
|
|
context "with no overrides" do
|
|
it "should return the translation from the current locale" do
|
|
translation_should_equal("foo", "Foo in :de")
|
|
end
|
|
end
|
|
end
|
|
|
|
context "with existing translation in fallback locale" do
|
|
context "with overrides in both locales" do
|
|
it "should return the override from the current locale" do
|
|
override_translation("de", "fish", "Override of fish in :de")
|
|
override_translation("en", "fish", "Override of fish in :en")
|
|
translation_should_equal("fish", "Override of fish in :de")
|
|
end
|
|
end
|
|
|
|
context "with override only in current locale" do
|
|
it "should return the override from the current locale" do
|
|
override_translation("de", "fish", "Override of fish in :de")
|
|
translation_should_equal("fish", "Override of fish in :de")
|
|
end
|
|
end
|
|
|
|
context "with override only in fallback locale" do
|
|
it "should return the translation from the current locale" do
|
|
override_translation("en", "fish", "Override of fish in :en")
|
|
translation_should_equal("fish", "Override of fish in :en")
|
|
end
|
|
end
|
|
|
|
context "with no overrides" do
|
|
it "should return the translation from the fallback locale" do
|
|
translation_should_equal("fish", "original fish")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|