mirror of
https://github.com/mperham/sidekiq.git
synced 2022-11-09 13:52:34 -05:00
Improve parsing of HTTP_ACCEPT_LANGUAGE (continue Nate's work) (#3449)
* Add test case for sv locale * Use Rack::Utils to parse locale header * Take "q" value into account * Make '*' match the default locale. * Add test for available_locales * Correct test case sv -> en * Add missing test cases for Safari requests * Add missing require needed to run a single test file * Reimplement WebHelpers#locale to handle regions in header Implementation inspired by: https://github.com/iain/http_accept_language/blob/master/lib/http_accept_language/parser.rb Also see: https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4 * Add docs and references * Add failing test cases for pt-br, pt-pt, pt (examples taken from Chrome & Safari) * Add more test cases for Mac + Chrome + UK English + US English * Make test cases for 'pt-PT,pt;q=0.8,en-US;q=0.6,en;q=0.4' and 'pt-pt' pass * Make special case 'ru,en' work (equal qvalues)
This commit is contained in:
parent
af591b9686
commit
3b96430046
2 changed files with 77 additions and 12 deletions
|
@ -24,6 +24,7 @@ module Sidekiq
|
||||||
def clear_caches
|
def clear_caches
|
||||||
@@strings = nil
|
@@strings = nil
|
||||||
@@locale_files = nil
|
@@locale_files = nil
|
||||||
|
@@available_locales = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def locale_files
|
def locale_files
|
||||||
|
@ -32,6 +33,10 @@ module Sidekiq
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def available_locales
|
||||||
|
@@available_locales ||= locale_files.map { |path| File.basename(path, '.yml') }.uniq
|
||||||
|
end
|
||||||
|
|
||||||
def find_locale_files(lang)
|
def find_locale_files(lang)
|
||||||
locale_files.select { |file| file =~ /\/#{lang}\.yml$/ }
|
locale_files.select { |file| file =~ /\/#{lang}\.yml$/ }
|
||||||
end
|
end
|
||||||
|
@ -73,20 +78,36 @@ module Sidekiq
|
||||||
text_direction == 'rtl'
|
text_direction == 'rtl'
|
||||||
end
|
end
|
||||||
|
|
||||||
# Given a browser request Accept-Language header like
|
# See https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4
|
||||||
# "fr-FR,fr;q=0.8,en-US;q=0.6,en;q=0.4,ru;q=0.2", this function
|
def user_preferred_languages
|
||||||
# will return "fr" since that's the first code with a matching
|
languages = env['HTTP_ACCEPT_LANGUAGE'.freeze]
|
||||||
# locale in web/locales
|
languages.to_s.downcase.gsub(/\s+/, '').split(',').map do |language|
|
||||||
|
locale, quality = language.split(';q=', 2)
|
||||||
|
locale = nil if locale == '*' # Ignore wildcards
|
||||||
|
quality = quality ? quality.to_f : 1.0
|
||||||
|
[locale, quality]
|
||||||
|
end.sort do |(_, left), (_, right)|
|
||||||
|
right <=> left
|
||||||
|
end.map(&:first).compact
|
||||||
|
end
|
||||||
|
|
||||||
|
# Given an Accept-Language header like "fr-FR,fr;q=0.8,en-US;q=0.6,en;q=0.4,ru;q=0.2"
|
||||||
|
# this method will try to best match the available locales to the user's preferred languages.
|
||||||
|
#
|
||||||
|
# Inspiration taken from https://github.com/iain/http_accept_language/blob/master/lib/http_accept_language/parser.rb
|
||||||
def locale
|
def locale
|
||||||
@locale ||= begin
|
@locale ||= begin
|
||||||
locale = 'en'.freeze
|
matched_locale = user_preferred_languages.map do |preferred|
|
||||||
languages = env['HTTP_ACCEPT_LANGUAGE'.freeze] || 'en'.freeze
|
preferred_language = preferred.split('-', 2).first
|
||||||
languages.downcase.split(','.freeze).each do |lang|
|
|
||||||
next if lang == '*'.freeze
|
lang_group = available_locales.select do |available|
|
||||||
lang = lang.split(';'.freeze)[0]
|
preferred_language == available.split('-', 2).first
|
||||||
break locale = lang if find_locale_files(lang).any?
|
end
|
||||||
end
|
|
||||||
locale
|
lang_group.find { |lang| lang == preferred } || lang_group.min_by(&:length)
|
||||||
|
end.compact.first
|
||||||
|
|
||||||
|
matched_locale || 'en'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
require_relative 'helper'
|
require_relative 'helper'
|
||||||
|
require 'sidekiq/web'
|
||||||
|
|
||||||
class TestWebHelpers < Sidekiq::Test
|
class TestWebHelpers < Sidekiq::Test
|
||||||
|
|
||||||
|
@ -42,13 +43,56 @@ class TestWebHelpers < Sidekiq::Test
|
||||||
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'zh-CN,zh;q=0.8,en-US;q=0.6,en;q=0.4,ru;q=0.2')
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'zh-CN,zh;q=0.8,en-US;q=0.6,en;q=0.4,ru;q=0.2')
|
||||||
assert_equal 'zh-cn', obj.locale
|
assert_equal 'zh-cn', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'en-US,sv-SE;q=0.8,sv;q=0.6,en;q=0.4')
|
||||||
|
assert_equal 'en', obj.locale
|
||||||
|
|
||||||
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'nb-NO,nb;q=0.2')
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'nb-NO,nb;q=0.2')
|
||||||
assert_equal 'nb', obj.locale
|
assert_equal 'nb', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'en-us')
|
||||||
|
assert_equal 'en', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'sv-se')
|
||||||
|
assert_equal 'sv', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'pt-BR,pt;q=0.8,en-US;q=0.6,en;q=0.4')
|
||||||
|
assert_equal 'pt-br', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'pt-PT,pt;q=0.8,en-US;q=0.6,en;q=0.4')
|
||||||
|
assert_equal 'pt', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'pt-br')
|
||||||
|
assert_equal 'pt-br', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'pt-pt')
|
||||||
|
assert_equal 'pt', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'pt')
|
||||||
|
assert_equal 'pt', obj.locale
|
||||||
|
|
||||||
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'en-us; *')
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'en-us; *')
|
||||||
assert_equal 'en', obj.locale
|
assert_equal 'en', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'en-US,en;q=0.8')
|
||||||
|
assert_equal 'en', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'en-GB,en-US;q=0.8,en;q=0.6')
|
||||||
|
assert_equal 'en', obj.locale
|
||||||
|
|
||||||
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => 'ru,en')
|
||||||
|
assert_equal 'ru', obj.locale
|
||||||
|
|
||||||
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => '*')
|
obj = Helpers.new('HTTP_ACCEPT_LANGUAGE' => '*')
|
||||||
assert_equal 'en', obj.locale
|
assert_equal 'en', obj.locale
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_available_locales
|
||||||
|
obj = Helpers.new
|
||||||
|
expected = %w(
|
||||||
|
ar cs da de el en es fa fr he hi it ja
|
||||||
|
ko nb nl pl pt-br pt ru sv ta uk ur
|
||||||
|
zh-cn zh-tw
|
||||||
|
)
|
||||||
|
assert_equal expected, obj.available_locales
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Add table
Reference in a new issue