blob: dcc8cc5fc19d545dc4e127eac98640848b55a483 [file] [log] [blame]
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/common/extensions/extension_l10n_util.h"
#include <set>
#include <string>
#include <vector>
#include "app/l10n_util.h"
#include "base/file_util.h"
#include "base/linked_ptr.h"
#include "base/string_util.h"
#include "base/values.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/extensions/extension_file_util.h"
#include "chrome/common/extensions/extension.h"
#include "chrome/common/extensions/extension_constants.h"
#include "chrome/common/extensions/extension_message_bundle.h"
#include "chrome/common/json_value_serializer.h"
#include "unicode/uloc.h"
namespace errors = extension_manifest_errors;
namespace keys = extension_manifest_keys;
static std::string* GetProcessLocale() {
static std::string locale;
return &locale;
}
namespace extension_l10n_util {
void SetProcessLocale(const std::string& locale) {
*(GetProcessLocale()) = locale;
}
std::string GetDefaultLocaleFromManifest(const DictionaryValue& manifest,
std::string* error) {
std::string default_locale;
if (manifest.GetString(keys::kDefaultLocale, &default_locale))
return default_locale;
*error = errors::kInvalidDefaultLocale;
return "";
}
bool ShouldRelocalizeManifest(const ExtensionInfo& info) {
DictionaryValue* manifest = info.extension_manifest.get();
if (!manifest)
return false;
if (!manifest->HasKey(keys::kDefaultLocale))
return false;
std::string manifest_current_locale;
manifest->GetString(keys::kCurrentLocale, &manifest_current_locale);
return manifest_current_locale != CurrentLocaleOrDefault();
}
// Localizes manifest value for a given key.
static bool LocalizeManifestValue(const std::wstring& key,
const ExtensionMessageBundle& messages,
DictionaryValue* manifest,
std::string* error) {
std::string result;
if (!manifest->GetString(key, &result))
return true;
if (!messages.ReplaceMessages(&result, error))
return false;
manifest->SetString(key, result);
return true;
}
bool LocalizeManifest(const ExtensionMessageBundle& messages,
DictionaryValue* manifest,
std::string* error) {
// Don't localize themes.
if (manifest->HasKey(keys::kTheme))
return true;
// Initialize name.
std::string result;
if (!manifest->GetString(keys::kName, &result)) {
*error = errors::kInvalidName;
return false;
}
if (!LocalizeManifestValue(keys::kName, messages, manifest, error)) {
return false;
}
// Initialize description.
if (!LocalizeManifestValue(keys::kDescription, messages, manifest, error))
return false;
// Initialize browser_action.default_title
std::wstring key(keys::kBrowserAction);
key.append(L".");
key.append(keys::kPageActionDefaultTitle);
if (!LocalizeManifestValue(key, messages, manifest, error))
return false;
// Initialize page_action.default_title
key.assign(keys::kPageAction);
key.append(L".");
key.append(keys::kPageActionDefaultTitle);
if (!LocalizeManifestValue(key, messages, manifest, error))
return false;
// Add current locale key to the manifest, so we can overwrite prefs
// with new manifest when chrome locale changes.
manifest->SetString(keys::kCurrentLocale, CurrentLocaleOrDefault());
return true;
}
bool LocalizeExtension(Extension* extension,
DictionaryValue* manifest,
std::string* error) {
DCHECK(manifest);
std::string default_locale = GetDefaultLocaleFromManifest(*manifest, error);
scoped_ptr<ExtensionMessageBundle> message_bundle(
extension_file_util::LoadExtensionMessageBundle(
extension->path(), default_locale, error));
if (!message_bundle.get() && !error->empty())
return false;
if (message_bundle.get() &&
!LocalizeManifest(*message_bundle, manifest, error))
return false;
return true;
}
bool AddLocale(const std::set<std::string>& chrome_locales,
const FilePath& locale_folder,
const std::string& locale_name,
std::set<std::string>* valid_locales,
std::string* error) {
// Accept name that starts with a . but don't add it to the list of supported
// locales.
if (locale_name.find(".") == 0)
return true;
if (chrome_locales.find(locale_name) == chrome_locales.end()) {
// Fail if there is an extension locale that's not in the Chrome list.
*error = StringPrintf("Supplied locale %s is not supported.",
locale_name.c_str());
return false;
}
// Check if messages file is actually present (but don't check content).
if (file_util::PathExists(
locale_folder.Append(Extension::kMessagesFilename))) {
valid_locales->insert(locale_name);
} else {
*error = StringPrintf("Catalog file is missing for locale %s.",
locale_name.c_str());
return false;
}
return true;
}
std::string NormalizeLocale(const std::string& locale) {
std::string normalized_locale(locale);
std::replace(normalized_locale.begin(), normalized_locale.end(), '-', '_');
return normalized_locale;
}
std::string CurrentLocaleOrDefault() {
std::string current_locale =
NormalizeLocale(g_browser_process->GetApplicationLocale());
if (current_locale.empty())
current_locale = "en";
return current_locale;
}
void GetParentLocales(const std::string& current_locale,
std::vector<std::string>* parent_locales) {
std::string locale(NormalizeLocale(current_locale));
const int kNameCapacity = 256;
char parent[kNameCapacity];
base::strlcpy(parent, locale.c_str(), kNameCapacity);
parent_locales->push_back(parent);
UErrorCode err = U_ZERO_ERROR;
while (uloc_getParent(parent, parent, kNameCapacity, &err) > 0) {
if (U_FAILURE(err))
break;
parent_locales->push_back(parent);
}
}
// Extends list of Chrome locales to them and their parents, so we can do
// proper fallback.
static void GetAllLocales(std::set<std::string>* all_locales) {
const std::vector<std::string>& available_locales =
l10n_util::GetAvailableLocales();
// Add all parents of the current locale to the available locales set.
// I.e. for sr_Cyrl_RS we add sr_Cyrl_RS, sr_Cyrl and sr.
for (size_t i = 0; i < available_locales.size(); ++i) {
std::vector<std::string> result;
GetParentLocales(available_locales[i], &result);
all_locales->insert(result.begin(), result.end());
}
}
bool GetValidLocales(const FilePath& locale_path,
std::set<std::string>* valid_locales,
std::string* error) {
static std::set<std::string> chrome_locales;
GetAllLocales(&chrome_locales);
// Enumerate all supplied locales in the extension.
file_util::FileEnumerator locales(locale_path,
false,
file_util::FileEnumerator::DIRECTORIES);
FilePath locale_folder;
while (!(locale_folder = locales.Next()).empty()) {
std::string locale_name =
WideToASCII(locale_folder.BaseName().ToWStringHack());
if (!AddLocale(chrome_locales,
locale_folder,
locale_name,
valid_locales,
error)) {
return false;
}
}
if (valid_locales->empty()) {
*error = extension_manifest_errors::kLocalesNoValidLocaleNamesListed;
return false;
}
return true;
}
// Loads contents of the messages file for given locale. If file is not found,
// or there was parsing error we return NULL and set |error|.
// Caller owns the returned object.
static DictionaryValue* LoadMessageFile(const FilePath& locale_path,
const std::string& locale,
std::string* error) {
std::string extension_locale = locale;
FilePath file = locale_path.AppendASCII(extension_locale)
.Append(Extension::kMessagesFilename);
JSONFileValueSerializer messages_serializer(file);
Value *dictionary = messages_serializer.Deserialize(error);
if (!dictionary && error->empty()) {
// JSONFileValueSerializer just returns NULL if file cannot be found. It
// doesn't set the error, so we have to do it.
*error = StringPrintf("Catalog file is missing for locale %s.",
extension_locale.c_str());
}
return static_cast<DictionaryValue*>(dictionary);
}
ExtensionMessageBundle* LoadMessageCatalogs(
const FilePath& locale_path,
const std::string& default_locale,
const std::string& application_locale,
const std::set<std::string>& valid_locales,
std::string* error) {
// Order locales to load as current_locale, first_parent, ..., default_locale.
std::vector<std::string> all_fallback_locales;
if (!application_locale.empty() && application_locale != default_locale)
GetParentLocales(application_locale, &all_fallback_locales);
all_fallback_locales.push_back(default_locale);
std::vector<linked_ptr<DictionaryValue> > catalogs;
for (size_t i = 0; i < all_fallback_locales.size(); ++i) {
// Skip all parent locales that are not supplied.
if (valid_locales.find(all_fallback_locales[i]) == valid_locales.end())
continue;
linked_ptr<DictionaryValue> catalog(
LoadMessageFile(locale_path, all_fallback_locales[i], error));
if (!catalog.get()) {
// If locale is valid, but messages.json is corrupted or missing, return
// an error.
return false;
} else {
catalogs.push_back(catalog);
}
}
return ExtensionMessageBundle::Create(catalogs, error);
}
} // namespace extension_l10n_util