2022-04-10 07:57:02 +00:00
|
|
|
#include "messagebundles.hpp"
|
|
|
|
|
|
|
|
#include <cstring>
|
|
|
|
#include <unicode/calendar.h>
|
|
|
|
#include <unicode/errorcode.h>
|
|
|
|
#include <yaml-cpp/yaml.h>
|
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
|
|
|
|
namespace l10n
|
|
|
|
{
|
|
|
|
MessageBundles::MessageBundles(const std::vector<icu::Locale>& preferredLocales, icu::Locale& fallbackLocale)
|
|
|
|
: mFallbackLocale(fallbackLocale)
|
|
|
|
{
|
|
|
|
setPreferredLocales(preferredLocales);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MessageBundles::setPreferredLocales(const std::vector<icu::Locale>& preferredLocales)
|
|
|
|
{
|
|
|
|
mPreferredLocales.clear();
|
|
|
|
mPreferredLocaleStrings.clear();
|
|
|
|
for (const icu::Locale& loc : preferredLocales)
|
|
|
|
{
|
|
|
|
mPreferredLocales.push_back(loc);
|
2022-04-11 12:37:22 +00:00
|
|
|
mPreferredLocaleStrings.emplace_back(loc.getName());
|
2022-04-10 07:57:02 +00:00
|
|
|
// Try without variant or country if they are specified, starting with the most specific
|
|
|
|
if (strcmp(loc.getVariant(), "") != 0)
|
|
|
|
{
|
|
|
|
icu::Locale withoutVariant(loc.getLanguage(), loc.getCountry());
|
|
|
|
mPreferredLocales.push_back(withoutVariant);
|
2022-04-11 12:37:22 +00:00
|
|
|
mPreferredLocaleStrings.emplace_back(withoutVariant.getName());
|
2022-04-10 07:57:02 +00:00
|
|
|
}
|
|
|
|
if (strcmp(loc.getCountry(), "") != 0)
|
|
|
|
{
|
|
|
|
icu::Locale withoutCountry(loc.getLanguage());
|
|
|
|
mPreferredLocales.push_back(withoutCountry);
|
2022-04-11 12:37:22 +00:00
|
|
|
mPreferredLocaleStrings.emplace_back(withoutCountry.getName());
|
2022-04-10 07:57:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string getErrorText(const UParseError& parseError)
|
|
|
|
{
|
|
|
|
icu::UnicodeString preContext(parseError.preContext), postContext(parseError.postContext);
|
|
|
|
std::string parseErrorString;
|
|
|
|
preContext.toUTF8String(parseErrorString);
|
|
|
|
postContext.toUTF8String(parseErrorString);
|
|
|
|
return parseErrorString;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool checkSuccess(
|
|
|
|
const icu::ErrorCode& status, const std::string& message, const UParseError parseError = UParseError())
|
|
|
|
{
|
|
|
|
if (status.isFailure())
|
|
|
|
{
|
|
|
|
std::string errorText = getErrorText(parseError);
|
2022-04-11 12:37:22 +00:00
|
|
|
if (!errorText.empty())
|
2022-04-10 07:57:02 +00:00
|
|
|
{
|
|
|
|
Log(Debug::Error) << message << ": " << status.errorName() << " in \"" << errorText << "\"";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log(Debug::Error) << message << ": " << status.errorName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return status.isSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MessageBundles::load(std::istream& input, const icu::Locale& lang, const std::string& path)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
YAML::Node data = YAML::Load(input);
|
|
|
|
std::string localeName = lang.getName();
|
2023-03-11 23:34:17 +00:00
|
|
|
const icu::Locale& langOrEn = localeName == "gmst" ? icu::Locale::getEnglish() : lang;
|
2022-04-10 07:57:02 +00:00
|
|
|
for (const auto& it : data)
|
|
|
|
{
|
2022-04-11 14:59:46 +00:00
|
|
|
const auto key = it.first.as<std::string>();
|
|
|
|
const auto value = it.second.as<std::string>();
|
2022-07-01 20:29:36 +00:00
|
|
|
icu::UnicodeString pattern = icu::UnicodeString::fromUTF8(icu::StringPiece(value.data(), value.size()));
|
2022-04-10 07:57:02 +00:00
|
|
|
icu::ErrorCode status;
|
|
|
|
UParseError parseError;
|
2023-03-11 23:34:17 +00:00
|
|
|
icu::MessageFormat message(pattern, langOrEn, parseError, status);
|
2022-04-10 07:57:02 +00:00
|
|
|
if (checkSuccess(status,
|
|
|
|
std::string("Failed to create message ") + key + " for locale " + lang.getName(), parseError))
|
|
|
|
{
|
|
|
|
mBundles[localeName].insert(std::make_pair(key, message));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (std::exception& e)
|
|
|
|
{
|
|
|
|
Log(Debug::Error) << "Can not load " << path << ": " << e.what();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const icu::MessageFormat* MessageBundles::findMessage(std::string_view key, const std::string& localeName) const
|
|
|
|
{
|
|
|
|
auto iter = mBundles.find(localeName);
|
|
|
|
if (iter != mBundles.end())
|
|
|
|
{
|
|
|
|
auto message = iter->second.find(key.data());
|
|
|
|
if (message != iter->second.end())
|
|
|
|
{
|
|
|
|
return &(message->second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string MessageBundles::formatMessage(
|
|
|
|
std::string_view key, const std::map<std::string, icu::Formattable>& args) const
|
|
|
|
{
|
|
|
|
std::vector<icu::UnicodeString> argNames;
|
|
|
|
std::vector<icu::Formattable> argValues;
|
2022-04-11 12:37:22 +00:00
|
|
|
for (auto& [k, v] : args)
|
2022-04-10 07:57:02 +00:00
|
|
|
{
|
2022-07-01 20:29:36 +00:00
|
|
|
argNames.push_back(icu::UnicodeString::fromUTF8(icu::StringPiece(k.data(), k.size())));
|
2022-04-11 12:37:22 +00:00
|
|
|
argValues.push_back(v);
|
2022-04-10 07:57:02 +00:00
|
|
|
}
|
|
|
|
return formatMessage(key, argNames, argValues);
|
|
|
|
}
|
|
|
|
|
2023-03-11 23:34:17 +00:00
|
|
|
static std::string loadGmst(
|
|
|
|
const std::function<std::string(std::string_view)> gmstLoader, const icu::MessageFormat* message)
|
|
|
|
{
|
|
|
|
icu::UnicodeString gmstNameUnicode;
|
|
|
|
std::string gmstName;
|
|
|
|
icu::ErrorCode success;
|
|
|
|
message->format(nullptr, nullptr, 0, gmstNameUnicode, success);
|
|
|
|
gmstNameUnicode.toUTF8String(gmstName);
|
|
|
|
if (gmstLoader)
|
|
|
|
return gmstLoader(gmstName);
|
|
|
|
else
|
|
|
|
return "GMST:" + gmstName;
|
|
|
|
}
|
|
|
|
|
2022-04-10 07:57:02 +00:00
|
|
|
std::string MessageBundles::formatMessage(std::string_view key, const std::vector<icu::UnicodeString>& argNames,
|
|
|
|
const std::vector<icu::Formattable>& args) const
|
|
|
|
{
|
|
|
|
icu::UnicodeString result;
|
|
|
|
std::string resultString;
|
|
|
|
icu::ErrorCode success;
|
|
|
|
|
|
|
|
const icu::MessageFormat* message = nullptr;
|
|
|
|
for (auto& loc : mPreferredLocaleStrings)
|
|
|
|
{
|
|
|
|
message = findMessage(key, loc);
|
|
|
|
if (message)
|
2023-03-11 23:34:17 +00:00
|
|
|
{
|
|
|
|
if (loc == "gmst")
|
|
|
|
return loadGmst(mGmstLoader, message);
|
2022-04-10 07:57:02 +00:00
|
|
|
break;
|
2023-03-11 23:34:17 +00:00
|
|
|
}
|
2022-04-10 07:57:02 +00:00
|
|
|
}
|
|
|
|
// If no requested locales included the message, try the fallback locale
|
|
|
|
if (!message)
|
|
|
|
message = findMessage(key, mFallbackLocale.getName());
|
|
|
|
|
|
|
|
if (message)
|
|
|
|
{
|
2022-04-11 12:37:22 +00:00
|
|
|
if (!args.empty() && !argNames.empty())
|
2022-09-03 15:41:35 +00:00
|
|
|
message->format(argNames.data(), args.data(), args.size(), result, success);
|
2022-04-10 07:57:02 +00:00
|
|
|
else
|
|
|
|
message->format(nullptr, nullptr, args.size(), result, success);
|
|
|
|
checkSuccess(success, std::string("Failed to format message ") + key.data());
|
|
|
|
result.toUTF8String(resultString);
|
|
|
|
return resultString;
|
|
|
|
}
|
2022-04-11 12:37:22 +00:00
|
|
|
icu::Locale defaultLocale(nullptr);
|
|
|
|
if (!mPreferredLocales.empty())
|
2022-04-10 07:57:02 +00:00
|
|
|
{
|
|
|
|
defaultLocale = mPreferredLocales[0];
|
|
|
|
}
|
|
|
|
UParseError parseError;
|
2022-05-06 19:57:41 +00:00
|
|
|
icu::MessageFormat defaultMessage(
|
|
|
|
icu::UnicodeString::fromUTF8(icu::StringPiece(key.data(), key.size())), defaultLocale, parseError, success);
|
2022-04-10 07:57:02 +00:00
|
|
|
if (!checkSuccess(success, std::string("Failed to create message ") + key.data(), parseError))
|
|
|
|
// If we can't parse the key as a pattern, just return the key
|
|
|
|
return std::string(key);
|
|
|
|
|
2022-04-11 12:37:22 +00:00
|
|
|
if (!args.empty() && !argNames.empty())
|
2022-09-03 15:41:35 +00:00
|
|
|
defaultMessage.format(argNames.data(), args.data(), args.size(), result, success);
|
2022-04-10 07:57:02 +00:00
|
|
|
else
|
|
|
|
defaultMessage.format(nullptr, nullptr, args.size(), result, success);
|
|
|
|
checkSuccess(success, std::string("Failed to format message ") + key.data());
|
|
|
|
result.toUTF8String(resultString);
|
|
|
|
return resultString;
|
|
|
|
}
|
|
|
|
}
|