Fallback to a variant if the language in general is not supported
authorTusooa Zhu <tusooa@kazv.moe>
Thu, 3 Mar 2022 00:59:11 +0000 (19:59 -0500)
committerFloatingGhost <hannah@coffee-and-dreams.uk>
Wed, 29 Jun 2022 19:47:10 +0000 (20:47 +0100)
For an example, here, zh is not supported, but zh_Hans and zh_Hant
are. If the user asks for zh, we should choose a variant for them
instead of fallbacking to default.

Some browsers (e.g. Firefox) does not allow users to customize
their language codes. For example, there is no zh-Hans, but only
zh, zh-CN, zh-TW, zh-HK, etc. This provides a workaround for
those users suffering from bad design decisions.

lib/pleroma/web/gettext.ex
lib/pleroma/web/plugs/set_locale_plug.ex
test/pleroma/web/plugs/set_locale_plug_test.exs

index 828b98b153d68ad555bc2f0c1a78ce7765b51741..cfd92f991e29a196ffbac153d8cb579281ebe2e7 100644 (file)
@@ -49,6 +49,20 @@ defmodule Pleroma.Web.Gettext do
     |> Enum.member?(locale)
   end
 
+  def variant?(locale), do: String.contains?(locale, "_")
+
+  def supported_variants_of_locale(locale) do
+    cond do
+      variant?(locale) ->
+        [locale]
+      supports_locale?(locale) ->
+        [locale]
+      true ->
+        Gettext.known_locales(Pleroma.Web.Gettext)
+        |> Enum.filter(fn l -> String.starts_with?(l, locale <> "_") end)
+    end
+  end
+
   def locale_or_default(locale) do
     if supports_locale?(locale) do
       locale
index 4c6e44fb59353e86db2d7ad0a7dab26e2261ae1b..78ae566c778bcd7ab63ae4f4f97374c91c357f9b 100644 (file)
@@ -20,6 +20,12 @@ defmodule Pleroma.Web.Plugs.SetLocalePlug do
     conn
     |> extract_preferred_language()
     |> normalize_language_codes()
+    |> first_supported()
+  end
+
+  defp first_supported(locales) do
+    locales
+    |> Enum.flat_map(&Pleroma.Web.Gettext.supported_variants_of_locale/1)
     |> Enum.find(&supported_locale?/1)
   end
 
index 043d7eb1822e5572c55b58a81ca1183b2c0105de..349326c245878c1ceb48deb30504c87fc8c8ecc4 100644 (file)
@@ -47,6 +47,20 @@ defmodule Pleroma.Web.Plugs.SetLocalePlugTest do
     assert %{locale: "zh_Hans"} == conn.assigns
   end
 
+  test "fallback to some variant of the language if the unqualified language is not supported" do
+    conn =
+      :get
+      |> conn("/cofe")
+      |> Conn.put_req_header(
+        "accept-language",
+        "zh;q=0.9, en;q=0.8, *;q=0.5"
+      )
+      |> SetLocalePlug.call([])
+
+    assert "zh_" <> _ = Gettext.get_locale()
+    assert %{locale: "zh_" <> _} = conn.assigns
+  end
+
   test "use supported locale from cookie" do
     conn =
       :get