geekbot/Geekbot.net/Lib/Localization/TranslationHandler.cs

199 lines
8 KiB
C#
Raw Normal View History

2017-11-15 01:08:20 +01:00
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading.Tasks;
2017-11-16 00:51:36 +01:00
using Discord.Commands;
using Geekbot.net.Database;
using Geekbot.net.Database.Models;
using Geekbot.net.Lib.Extensions;
using Geekbot.net.Lib.Logger;
2018-04-30 23:44:19 +02:00
using Utf8Json;
2017-11-15 01:08:20 +01:00
namespace Geekbot.net.Lib.Localization
2017-11-15 01:08:20 +01:00
{
public class TranslationHandler : ITranslationHandler
{
private readonly DatabaseContext _database;
private readonly IGeekbotLogger _logger;
private readonly Dictionary<ulong, string> _serverLanguages;
private Dictionary<string, Dictionary<string, Dictionary<string, List<string>>>> _translations;
public TranslationHandler(DatabaseContext database, IGeekbotLogger logger)
2017-11-15 01:08:20 +01:00
{
_database = database;
2017-11-15 01:08:20 +01:00
_logger = logger;
_logger.Information(LogSource.Geekbot, "Loading Translations");
2017-11-15 01:08:20 +01:00
LoadTranslations();
_serverLanguages = new Dictionary<ulong, string>();
2017-11-15 01:08:20 +01:00
}
private void LoadTranslations()
{
try
{
var translationFile = File.ReadAllText(Path.GetFullPath("./Lib/Localization/Translations.json"));
var rawTranslations = JsonSerializer.Deserialize<Dictionary<string, Dictionary<string, Dictionary<string, List<string>>>>>(translationFile);
var sortedPerLanguage = new Dictionary<string, Dictionary<string, Dictionary<string, List<string>>>>();
2017-11-16 00:51:36 +01:00
foreach (var command in rawTranslations)
{
foreach (var str in command.Value)
{
foreach (var lang in str.Value)
{
if (!sortedPerLanguage.ContainsKey(lang.Key))
{
var commandDict = new Dictionary<string, Dictionary<string, List<string>>>();
var strDict = new Dictionary<string, List<string>>
{
{str.Key, lang.Value}
};
2017-11-16 00:51:36 +01:00
commandDict.Add(command.Key, strDict);
sortedPerLanguage.Add(lang.Key, commandDict);
}
if (!sortedPerLanguage[lang.Key].ContainsKey(command.Key))
{
var strDict = new Dictionary<string, List<string>>
{
{str.Key, lang.Value}
};
2017-11-16 00:51:36 +01:00
sortedPerLanguage[lang.Key].Add(command.Key, strDict);
}
if (!sortedPerLanguage[lang.Key][command.Key].ContainsKey(str.Key))
{
sortedPerLanguage[lang.Key][command.Key].Add(str.Key, lang.Value);
}
2017-11-16 00:51:36 +01:00
}
}
}
_translations = sortedPerLanguage;
SupportedLanguages = new List<string>();
2017-11-16 00:51:36 +01:00
foreach (var lang in sortedPerLanguage)
2017-11-15 01:08:20 +01:00
{
SupportedLanguages.Add(lang.Key);
2017-11-15 01:08:20 +01:00
}
}
catch (Exception e)
{
_logger.Error(LogSource.Geekbot, "Failed to load Translations", e);
2018-05-06 02:00:45 +02:00
Environment.Exit(GeekbotExitCode.TranslationsFailed.GetHashCode());
2017-11-15 01:08:20 +01:00
}
}
2017-11-16 00:51:36 +01:00
private async Task<string> GetServerLanguage(ulong guildId)
2017-11-15 01:08:20 +01:00
{
try
2017-11-15 01:08:20 +01:00
{
string lang;
try
2017-11-15 01:08:20 +01:00
{
lang = _serverLanguages[guildId];
if (!string.IsNullOrEmpty(lang))
{
return lang;
}
throw new Exception();
2017-11-15 01:08:20 +01:00
}
catch
2017-11-15 01:08:20 +01:00
{
lang = (await GetGuild(guildId)).Language ?? "EN";
_serverLanguages[guildId] = lang;
return lang;
2017-11-15 01:08:20 +01:00
}
}
catch (Exception e)
{
_logger.Error(LogSource.Geekbot, "Could not get guild language", e);
return "EN";
}
2017-11-15 01:08:20 +01:00
}
public async Task<string> GetString(ulong guildId, string command, string stringName)
2017-11-15 01:08:20 +01:00
{
var serverLang = await GetServerLanguage(guildId);
return GetStrings(serverLang, command, stringName).First();
}
public List<string> GetStrings(string language, string command, string stringName)
{
var translation = _translations[language][command][stringName];
if (!string.IsNullOrWhiteSpace(translation.First())) return translation;
2017-11-15 01:08:20 +01:00
translation = _translations[command][stringName]["EN"];
if (string.IsNullOrWhiteSpace(translation.First()))
2017-11-15 01:08:20 +01:00
{
_logger.Warning(LogSource.Geekbot, $"No translation found for {command} - {stringName}");
2017-11-15 01:08:20 +01:00
}
return translation;
}
private async Task<Dictionary<string, List<string>>> GetDict(ICommandContext context)
2017-11-16 00:51:36 +01:00
{
try
{
var command = context.Message.Content.Split(' ').First().TrimStart('!').ToLower();
var serverLanguage = await GetServerLanguage(context.Guild?.Id ?? 0);
return _translations[serverLanguage][command];
2017-11-16 00:51:36 +01:00
}
catch (Exception e)
{
2018-05-06 02:00:45 +02:00
_logger.Error(LogSource.Geekbot, "No translations for command found", e);
return new Dictionary<string, List<string>>();
2017-11-16 00:51:36 +01:00
}
}
public async Task<TranslationGuildContext> GetGuildContext(ICommandContext context)
{
var dict = await GetDict(context);
var language = await GetServerLanguage(context.Guild?.Id ?? 0);
return new TranslationGuildContext(this, language, dict);
}
public async Task<Dictionary<string, string>> GetDict(ICommandContext context, string command)
{
try
{
var serverLanguage = await GetServerLanguage(context.Guild?.Id ?? 0);
return _translations[serverLanguage][command]
.ToDictionary(dict => dict.Key, dict => dict.Value.First());
}
catch (Exception e)
{
2018-05-06 02:00:45 +02:00
_logger.Error(LogSource.Geekbot, "No translations for command found", e);
return new Dictionary<string, string>();
}
}
2017-11-16 00:51:36 +01:00
public async Task<bool> SetLanguage(ulong guildId, string language)
2017-11-15 01:08:20 +01:00
{
try
{
if (!SupportedLanguages.Contains(language)) return false;
var guild = await GetGuild(guildId);
guild.Language = language;
_database.GuildSettings.Update(guild);
2017-11-15 01:08:20 +01:00
_serverLanguages[guildId] = language;
return true;
}
catch (Exception e)
{
_logger.Error(LogSource.Geekbot, "Error while changing language", e);
2017-11-15 01:08:20 +01:00
return false;
}
}
public List<string> SupportedLanguages { get; private set; }
private async Task<GuildSettingsModel> GetGuild(ulong guildId)
{
var guild = _database.GuildSettings.FirstOrDefault(g => g.GuildId.Equals(guildId.AsLong()));
if (guild != null) return guild;
_database.GuildSettings.Add(new GuildSettingsModel
{
GuildId = guildId.AsLong()
});
await _database.SaveChangesAsync();
return _database.GuildSettings.FirstOrDefault(g => g.GuildId.Equals(guildId.AsLong()));
}
2017-11-15 01:08:20 +01:00
}
}