LocalizationManager.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  1. using System;
  2. using System.Collections.Concurrent;
  3. using System.Collections.Generic;
  4. using System.Globalization;
  5. using System.IO;
  6. using System.Linq;
  7. using System.Reflection;
  8. using System.Text;
  9. using System.Threading.Tasks;
  10. using MediaBrowser.Controller.Configuration;
  11. using MediaBrowser.Model.Entities;
  12. using MediaBrowser.Model.Extensions;
  13. using MediaBrowser.Model.Globalization;
  14. using MediaBrowser.Model.IO;
  15. using MediaBrowser.Model.Serialization;
  16. using Microsoft.Extensions.Logging;
  17. namespace Emby.Server.Implementations.Localization
  18. {
  19. /// <summary>
  20. /// Class LocalizationManager
  21. /// </summary>
  22. public class LocalizationManager : ILocalizationManager
  23. {
  24. /// <summary>
  25. /// The _configuration manager
  26. /// </summary>
  27. private readonly IServerConfigurationManager _configurationManager;
  28. /// <summary>
  29. /// The us culture
  30. /// </summary>
  31. private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
  32. private readonly Dictionary<string, Dictionary<string, ParentalRating>> _allParentalRatings =
  33. new Dictionary<string, Dictionary<string, ParentalRating>>(StringComparer.OrdinalIgnoreCase);
  34. private readonly IFileSystem _fileSystem;
  35. private readonly IJsonSerializer _jsonSerializer;
  36. private readonly ILogger _logger;
  37. private static readonly Assembly _assembly = typeof(LocalizationManager).Assembly;
  38. /// <summary>
  39. /// Initializes a new instance of the <see cref="LocalizationManager" /> class.
  40. /// </summary>
  41. /// <param name="configurationManager">The configuration manager.</param>
  42. /// <param name="fileSystem">The file system.</param>
  43. /// <param name="jsonSerializer">The json serializer.</param>
  44. public LocalizationManager(
  45. IServerConfigurationManager configurationManager,
  46. IFileSystem fileSystem,
  47. IJsonSerializer jsonSerializer,
  48. ILoggerFactory loggerFactory)
  49. {
  50. _configurationManager = configurationManager;
  51. _fileSystem = fileSystem;
  52. _jsonSerializer = jsonSerializer;
  53. _logger = loggerFactory.CreateLogger(nameof(LocalizationManager));
  54. }
  55. public async Task LoadAll()
  56. {
  57. const string ratingsResource = "Emby.Server.Implementations.Ratings.";
  58. Directory.CreateDirectory(LocalizationPath);
  59. var existingFiles = GetRatingsFiles(LocalizationPath).Select(Path.GetFileName);
  60. // Extract from the assembly
  61. foreach (var resource in _assembly.GetManifestResourceNames()
  62. .Where(i => i.StartsWith(ratingsResource)))
  63. {
  64. string filename = "ratings-" + resource.Substring(ratingsResource.Length);
  65. if (!existingFiles.Contains(filename))
  66. {
  67. using (var stream = _assembly.GetManifestResourceStream(resource))
  68. {
  69. string target = Path.Combine(LocalizationPath, filename);
  70. _logger.LogInformation("Extracting ratings to {0}", target);
  71. using (var fs = _fileSystem.GetFileStream(target, FileOpenMode.Create, FileAccessMode.Write, FileShareMode.Read))
  72. {
  73. await stream.CopyToAsync(fs);
  74. }
  75. }
  76. }
  77. }
  78. foreach (var file in GetRatingsFiles(LocalizationPath))
  79. {
  80. await LoadRatings(file);
  81. }
  82. LoadAdditionalRatings();
  83. await LoadCultures();
  84. }
  85. private void LoadAdditionalRatings()
  86. {
  87. LoadRatings("au", new[]
  88. {
  89. new ParentalRating("AU-G", 1),
  90. new ParentalRating("AU-PG", 5),
  91. new ParentalRating("AU-M", 6),
  92. new ParentalRating("AU-MA15+", 7),
  93. new ParentalRating("AU-M15+", 8),
  94. new ParentalRating("AU-R18+", 9),
  95. new ParentalRating("AU-X18+", 10),
  96. new ParentalRating("AU-RC", 11)
  97. });
  98. LoadRatings("be", new[]
  99. {
  100. new ParentalRating("BE-AL", 1),
  101. new ParentalRating("BE-MG6", 2),
  102. new ParentalRating("BE-6", 3),
  103. new ParentalRating("BE-9", 5),
  104. new ParentalRating("BE-12", 6),
  105. new ParentalRating("BE-16", 8)
  106. });
  107. LoadRatings("de", new[]
  108. {
  109. new ParentalRating("DE-0", 1),
  110. new ParentalRating("FSK-0", 1),
  111. new ParentalRating("DE-6", 5),
  112. new ParentalRating("FSK-6", 5),
  113. new ParentalRating("DE-12", 7),
  114. new ParentalRating("FSK-12", 7),
  115. new ParentalRating("DE-16", 8),
  116. new ParentalRating("FSK-16", 8),
  117. new ParentalRating("DE-18", 9),
  118. new ParentalRating("FSK-18", 9)
  119. });
  120. LoadRatings("ru", new[]
  121. {
  122. new ParentalRating("RU-0+", 1),
  123. new ParentalRating("RU-6+", 3),
  124. new ParentalRating("RU-12+", 7),
  125. new ParentalRating("RU-16+", 9),
  126. new ParentalRating("RU-18+", 10)
  127. });
  128. }
  129. private void LoadRatings(string country, ParentalRating[] ratings)
  130. {
  131. _allParentalRatings[country] = ratings.ToDictionary(i => i.Name);
  132. }
  133. private IEnumerable<string> GetRatingsFiles(string directory)
  134. => _fileSystem.GetFilePaths(directory, false)
  135. .Where(i => string.Equals(Path.GetExtension(i), ".csv", StringComparison.OrdinalIgnoreCase))
  136. .Where(i => Path.GetFileName(i).StartsWith("ratings-", StringComparison.OrdinalIgnoreCase));
  137. /// <summary>
  138. /// Gets the localization path.
  139. /// </summary>
  140. /// <value>The localization path.</value>
  141. public string LocalizationPath
  142. => Path.Combine(_configurationManager.ApplicationPaths.ProgramDataPath, "localization");
  143. public string NormalizeFormKD(string text)
  144. => text.Normalize(NormalizationForm.FormKD);
  145. private CultureDto[] _cultures;
  146. /// <summary>
  147. /// Gets the cultures.
  148. /// </summary>
  149. /// <returns>IEnumerable{CultureDto}.</returns>
  150. public CultureDto[] GetCultures()
  151. => _cultures;
  152. private async Task LoadCultures()
  153. {
  154. List<CultureDto> list = new List<CultureDto>();
  155. const string path = "Emby.Server.Implementations.Localization.iso6392.txt";
  156. using (var stream = _assembly.GetManifestResourceStream(path))
  157. using (var reader = new StreamReader(stream))
  158. {
  159. while (!reader.EndOfStream)
  160. {
  161. var line = await reader.ReadLineAsync();
  162. if (string.IsNullOrWhiteSpace(line))
  163. {
  164. continue;
  165. }
  166. var parts = line.Split('|');
  167. if (parts.Length == 5)
  168. {
  169. string name = parts[3];
  170. if (string.IsNullOrWhiteSpace(name))
  171. {
  172. continue;
  173. }
  174. string twoCharName = parts[2];
  175. if (string.IsNullOrWhiteSpace(twoCharName))
  176. {
  177. continue;
  178. }
  179. string[] threeletterNames;
  180. if (string.IsNullOrWhiteSpace(parts[1]))
  181. {
  182. threeletterNames = new [] { parts[0] };
  183. }
  184. else
  185. {
  186. threeletterNames = new [] { parts[0], parts[1] };
  187. }
  188. list.Add(new CultureDto
  189. {
  190. DisplayName = name,
  191. Name = name,
  192. ThreeLetterISOLanguageNames = threeletterNames,
  193. TwoLetterISOLanguageName = twoCharName
  194. });
  195. }
  196. }
  197. }
  198. _cultures = list.ToArray();
  199. }
  200. public CultureDto FindLanguageInfo(string language)
  201. => GetCultures()
  202. .FirstOrDefault(i =>
  203. string.Equals(i.DisplayName, language, StringComparison.OrdinalIgnoreCase)
  204. || string.Equals(i.Name, language, StringComparison.OrdinalIgnoreCase)
  205. || i.ThreeLetterISOLanguageNames.Contains(language, StringComparer.OrdinalIgnoreCase)
  206. || string.Equals(i.TwoLetterISOLanguageName, language, StringComparison.OrdinalIgnoreCase));
  207. /// <summary>
  208. /// Gets the countries.
  209. /// </summary>
  210. /// <returns>IEnumerable{CountryInfo}.</returns>
  211. public Task<CountryInfo[]> GetCountries()
  212. => _jsonSerializer.DeserializeFromStreamAsync<CountryInfo[]>(
  213. _assembly.GetManifestResourceStream("Emby.Server.Implementations.Localization.countries.json"));
  214. /// <summary>
  215. /// Gets the parental ratings.
  216. /// </summary>
  217. /// <returns>IEnumerable{ParentalRating}.</returns>
  218. public IEnumerable<ParentalRating> GetParentalRatings()
  219. => GetParentalRatingsDictionary().Values;
  220. /// <summary>
  221. /// Gets the parental ratings dictionary.
  222. /// </summary>
  223. /// <returns>Dictionary{System.StringParentalRating}.</returns>
  224. private Dictionary<string, ParentalRating> GetParentalRatingsDictionary()
  225. {
  226. var countryCode = _configurationManager.Configuration.MetadataCountryCode;
  227. if (string.IsNullOrEmpty(countryCode))
  228. {
  229. countryCode = "us";
  230. }
  231. return GetRatings(countryCode) ?? GetRatings("us");
  232. }
  233. /// <summary>
  234. /// Gets the ratings.
  235. /// </summary>
  236. /// <param name="countryCode">The country code.</param>
  237. private Dictionary<string, ParentalRating> GetRatings(string countryCode)
  238. {
  239. _allParentalRatings.TryGetValue(countryCode, out var value);
  240. return value;
  241. }
  242. /// <summary>
  243. /// Loads the ratings.
  244. /// </summary>
  245. /// <param name="file">The file.</param>
  246. /// <returns>Dictionary{System.StringParentalRating}.</returns>
  247. private async Task LoadRatings(string file)
  248. {
  249. Dictionary<string, ParentalRating> dict = new Dictionary<string, ParentalRating>(StringComparer.OrdinalIgnoreCase);
  250. using (var str = File.OpenRead(file))
  251. using (var reader = new StreamReader(str))
  252. {
  253. string line;
  254. while ((line = await reader.ReadLineAsync()) != null)
  255. {
  256. if (string.IsNullOrWhiteSpace(line))
  257. {
  258. continue;
  259. }
  260. string[] parts = line.Split(',');
  261. if (parts.Length == 2
  262. && int.TryParse(parts[1], NumberStyles.Integer, UsCulture, out var value))
  263. {
  264. dict.Add(parts[0], (new ParentalRating { Name = parts[0], Value = value }));
  265. }
  266. #if DEBUG
  267. _logger.LogWarning("Misformed line in {Path}", file);
  268. #endif
  269. }
  270. }
  271. var countryCode = Path.GetFileNameWithoutExtension(file).Split('-')[1];
  272. _allParentalRatings[countryCode] = dict;
  273. }
  274. private static readonly string[] _unratedValues = { "n/a", "unrated", "not rated" };
  275. /// <summary>
  276. /// Gets the rating level.
  277. /// </summary>
  278. public int? GetRatingLevel(string rating)
  279. {
  280. if (string.IsNullOrEmpty(rating))
  281. {
  282. throw new ArgumentNullException(nameof(rating));
  283. }
  284. if (_unratedValues.Contains(rating, StringComparer.OrdinalIgnoreCase))
  285. {
  286. return null;
  287. }
  288. // Fairly common for some users to have "Rated R" in their rating field
  289. rating = rating.Replace("Rated ", string.Empty, StringComparison.OrdinalIgnoreCase);
  290. var ratingsDictionary = GetParentalRatingsDictionary();
  291. if (ratingsDictionary.TryGetValue(rating, out ParentalRating value))
  292. {
  293. return value.Value;
  294. }
  295. // If we don't find anything check all ratings systems
  296. foreach (var dictionary in _allParentalRatings.Values)
  297. {
  298. if (dictionary.TryGetValue(rating, out value))
  299. {
  300. return value.Value;
  301. }
  302. }
  303. // Try splitting by : to handle "Germany: FSK 18"
  304. var index = rating.IndexOf(':');
  305. if (index != -1)
  306. {
  307. rating = rating.Substring(index).TrimStart(':').Trim();
  308. if (!string.IsNullOrWhiteSpace(rating))
  309. {
  310. return GetRatingLevel(rating);
  311. }
  312. }
  313. // TODO: Further improve by normalizing out all spaces and dashes
  314. return null;
  315. }
  316. public bool HasUnicodeCategory(string value, UnicodeCategory category)
  317. {
  318. foreach (var chr in value)
  319. {
  320. if (char.GetUnicodeCategory(chr) == category)
  321. {
  322. return true;
  323. }
  324. }
  325. return false;
  326. }
  327. public string GetLocalizedString(string phrase)
  328. {
  329. return GetLocalizedString(phrase, _configurationManager.Configuration.UICulture);
  330. }
  331. public string GetLocalizedString(string phrase, string culture)
  332. {
  333. if (string.IsNullOrEmpty(culture))
  334. {
  335. culture = _configurationManager.Configuration.UICulture;
  336. }
  337. if (string.IsNullOrEmpty(culture))
  338. {
  339. culture = DefaultCulture;
  340. }
  341. var dictionary = GetLocalizationDictionary(culture);
  342. if (dictionary.TryGetValue(phrase, out var value))
  343. {
  344. return value;
  345. }
  346. return phrase;
  347. }
  348. private const string DefaultCulture = "en-US";
  349. private readonly ConcurrentDictionary<string, Dictionary<string, string>> _dictionaries =
  350. new ConcurrentDictionary<string, Dictionary<string, string>>(StringComparer.OrdinalIgnoreCase);
  351. public Dictionary<string, string> GetLocalizationDictionary(string culture)
  352. {
  353. if (string.IsNullOrEmpty(culture))
  354. {
  355. throw new ArgumentNullException(nameof(culture));
  356. }
  357. const string prefix = "Core";
  358. var key = prefix + culture;
  359. return _dictionaries.GetOrAdd(key,
  360. f => GetDictionary(prefix, culture, DefaultCulture + ".json").GetAwaiter().GetResult());
  361. }
  362. private async Task<Dictionary<string, string>> GetDictionary(string prefix, string culture, string baseFilename)
  363. {
  364. if (string.IsNullOrEmpty(culture))
  365. {
  366. throw new ArgumentNullException(nameof(culture));
  367. }
  368. var dictionary = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
  369. var namespaceName = GetType().Namespace + "." + prefix;
  370. await CopyInto(dictionary, namespaceName + "." + baseFilename);
  371. await CopyInto(dictionary, namespaceName + "." + GetResourceFilename(culture));
  372. return dictionary;
  373. }
  374. private async Task CopyInto(IDictionary<string, string> dictionary, string resourcePath)
  375. {
  376. using (var stream = _assembly.GetManifestResourceStream(resourcePath))
  377. {
  378. var dict = await _jsonSerializer.DeserializeFromStreamAsync<Dictionary<string, string>>(stream);
  379. foreach (var key in dict.Keys)
  380. {
  381. dictionary[key] = dict[key];
  382. }
  383. }
  384. }
  385. private static string GetResourceFilename(string culture)
  386. {
  387. var parts = culture.Split('-');
  388. if (parts.Length == 2)
  389. {
  390. culture = parts[0].ToLowerInvariant() + "-" + parts[1].ToUpperInvariant();
  391. }
  392. else
  393. {
  394. culture = culture.ToLowerInvariant();
  395. }
  396. return culture + ".json";
  397. }
  398. public LocalizationOption[] GetLocalizationOptions()
  399. => new LocalizationOption[]
  400. {
  401. new LocalizationOption("Arabic", "ar"),
  402. new LocalizationOption("Belarusian (Belarus)", "be-BY"),
  403. new LocalizationOption("Bulgarian (Bulgaria)", "bg-BG"),
  404. new LocalizationOption("Catalan", "ca"),
  405. new LocalizationOption("Chinese Simplified", "zh-CN"),
  406. new LocalizationOption("Chinese Traditional", "zh-TW"),
  407. new LocalizationOption("Chinese Traditional (Hong Kong)", "zh-HK"),
  408. new LocalizationOption("Croatian", "hr"),
  409. new LocalizationOption("Czech", "cs"),
  410. new LocalizationOption("Danish", "da"),
  411. new LocalizationOption("Dutch", "nl"),
  412. new LocalizationOption("English (United Kingdom)", "en-GB"),
  413. new LocalizationOption("English (United States)", "en-US"),
  414. new LocalizationOption("Finnish", "fi"),
  415. new LocalizationOption("French", "fr"),
  416. new LocalizationOption("French (Canada)", "fr-CA"),
  417. new LocalizationOption("German", "de"),
  418. new LocalizationOption("Greek", "el"),
  419. new LocalizationOption("Hebrew", "he"),
  420. new LocalizationOption("Hindi (India)", "hi-IN"),
  421. new LocalizationOption("Hungarian", "hu"),
  422. new LocalizationOption("Indonesian", "id"),
  423. new LocalizationOption("Italian", "it"),
  424. new LocalizationOption("Japanese", "ja"),
  425. new LocalizationOption("Kazakh", "kk"),
  426. new LocalizationOption("Korean", "ko"),
  427. new LocalizationOption("Lithuanian", "lt-LT"),
  428. new LocalizationOption("Malay", "ms"),
  429. new LocalizationOption("Norwegian Bokmål", "nb"),
  430. new LocalizationOption("Persian", "fa"),
  431. new LocalizationOption("Polish", "pl"),
  432. new LocalizationOption("Portuguese (Brazil)", "pt-BR"),
  433. new LocalizationOption("Portuguese (Portugal)", "pt-PT"),
  434. new LocalizationOption("Romanian", "ro"),
  435. new LocalizationOption("Russian", "ru"),
  436. new LocalizationOption("Slovak", "sk"),
  437. new LocalizationOption("Slovenian (Slovenia)", "sl-SI"),
  438. new LocalizationOption("Spanish", "es"),
  439. new LocalizationOption("Spanish (Latin America)", "es-419"),
  440. new LocalizationOption("Spanish (Mexico)", "es-MX"),
  441. new LocalizationOption("Swedish", "sv"),
  442. new LocalizationOption("Swiss German", "gsw"),
  443. new LocalizationOption("Turkish", "tr"),
  444. new LocalizationOption("Ukrainian", "uk"),
  445. new LocalizationOption("Vietnamese", "vi")
  446. };
  447. }
  448. }