1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179
|
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/translate/core/browser/translate_script.h"
#include "base/command_line.h"
#include "base/functional/bind.h"
#include "base/location.h"
#include "base/logging.h"
#include "base/strings/escape.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/task/single_thread_task_runner.h"
#include "build/build_config.h"
#include "components/grit/components_resources.h"
#include "components/translate/core/browser/translate_url_fetcher.h"
#include "components/translate/core/browser/translate_url_util.h"
#include "components/translate/core/common/translate_switches.h"
#include "components/translate/core/common/translate_util.h"
#include "components/variations/variations_associated_data.h"
#include "google_apis/google_api_keys.h"
#include "net/base/url_util.h"
#include "ui/base/resource/resource_bundle.h"
namespace translate {
namespace {
const int kExpirationDelayDays = 1;
} // namespace
const char TranslateScript::kScriptURL[] =
"https://translate.googleapis.com/translate_a/element.js";
const char TranslateScript::kRequestHeader[] =
"Google-Translate-Element-Mode: library";
const char TranslateScript::kAlwaysUseSslQueryName[] = "aus";
const char TranslateScript::kAlwaysUseSslQueryValue[] = "true";
const char TranslateScript::kCallbackQueryName[] = "cb";
const char TranslateScript::kCallbackQueryValue[] =
"cr.googleTranslate.onTranslateElementLoad";
const char TranslateScript::kCssLoaderCallbackQueryName[] = "clc";
const char TranslateScript::kCssLoaderCallbackQueryValue[] =
"cr.googleTranslate.onLoadCSS";
const char TranslateScript::kJavascriptLoaderCallbackQueryName[] = "jlc";
const char TranslateScript::kJavascriptLoaderCallbackQueryValue[] =
"cr.googleTranslate.onLoadJavascript";
TranslateScript::TranslateScript()
: expiration_delay_(base::Days(kExpirationDelayDays)) {}
TranslateScript::~TranslateScript() = default;
void TranslateScript::Request(RequestCallback callback, bool is_incognito) {
script_fetch_start_time_ = base::Time::Now().InMillisecondsFSinceUnixEpoch();
DCHECK(data_.empty()) << "Do not fetch the script if it is already fetched";
callback_list_.AddUnsafe(std::move(callback));
if (fetcher_) {
// If there is already a request in progress, do nothing. |callback| will be
// run on completion.
return;
}
GURL translate_script_url = GetTranslateScriptURL();
fetcher_ = std::make_unique<TranslateURLFetcher>();
fetcher_->set_extra_request_header(kRequestHeader);
fetcher_->Request(translate_script_url,
base::BindOnce(&TranslateScript::OnScriptFetchComplete,
base::Unretained(this)),
is_incognito);
}
// static
GURL TranslateScript::GetTranslateScriptURL() {
GURL translate_script_url;
// Check if command-line contains an alternative URL for translate service.
const base::CommandLine& command_line =
*base::CommandLine::ForCurrentProcess();
if (command_line.HasSwitch(translate::switches::kTranslateScriptURL)) {
translate_script_url = GURL(command_line.GetSwitchValueASCII(
translate::switches::kTranslateScriptURL));
if (!translate_script_url.is_valid()) {
LOG(WARNING) << "The following translate URL specified at the "
<< "command-line is invalid: "
<< translate_script_url.spec();
translate_script_url = GURL();
} else {
LOG(WARNING) << "Using custom translate URL: "
<< translate_script_url.spec();
}
}
// Use default URL when command-line argument is not specified, or specified
// URL is invalid.
if (translate_script_url.is_empty())
translate_script_url = GURL(kScriptURL);
translate_script_url = net::AppendQueryParameter(
translate_script_url, kCallbackQueryName, kCallbackQueryValue);
translate_script_url = net::AppendQueryParameter(
translate_script_url, kAlwaysUseSslQueryName, kAlwaysUseSslQueryValue);
translate_script_url = net::AppendQueryParameter(
translate_script_url, kCssLoaderCallbackQueryName,
kCssLoaderCallbackQueryValue);
translate_script_url = net::AppendQueryParameter(
translate_script_url, kJavascriptLoaderCallbackQueryName,
kJavascriptLoaderCallbackQueryValue);
translate_script_url = AddHostLocaleToUrl(translate_script_url);
translate_script_url = AddApiKeyToUrl(translate_script_url);
return translate_script_url;
}
void TranslateScript::OnScriptFetchComplete(bool success,
const std::string& data) {
std::unique_ptr<const TranslateURLFetcher> delete_ptr(std::move(fetcher_));
if (success) {
DCHECK(data_.empty());
// Insert variable definitions on API Key and security origin.
data_ = base::StringPrintf("var translateApiKey = '%s';\n",
google_apis::GetAPIKey().c_str());
// Insert server params to pass experimental params to google translate
// server.
std::string server_params;
std::map<std::string, std::string> params;
base::StringAppendF(
&data_, "var gtTimeInfo = {'fetchStart': %0.f, 'fetchEnd': %0.f};\n",
script_fetch_start_time_,
base::Time::Now().InMillisecondsFSinceUnixEpoch());
base::StringAppendF(&data_, "var serverParams = '%s';\n",
server_params.c_str());
GURL security_origin = translate::GetTranslateSecurityOrigin();
base::StringAppendF(&data_, "var securityOrigin = '%s';\n",
security_origin.spec().c_str());
// Load embedded translate.js.
data_.append(ui::ResourceBundle::GetSharedInstance().LoadDataResourceString(
IDR_TRANSLATE_JS));
#if BUILDFLAG(IS_IOS)
// Append snippet to install callbacks on translate.js if available.
const char* install_callbacks =
"try {"
" __gCrWeb.translate.installCallbacks();"
"} catch (error) {};";
data_.append(install_callbacks);
#endif // BUILDFLAG(IS_IOS)
// Wrap |data| in try/catch block to handle unexpected script errors.
static constexpr char kFormat[] =
"try {"
" %s;"
"} catch (error) {"
" cr.googleTranslate.onTranslateElementError(error);"
"};";
base::StringAppendF(&data_, kFormat, data.c_str());
// We'll expire the cached script after some time, to make sure long
// running browsers still get fixes that might get pushed with newer
// scripts.
base::SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
FROM_HERE,
base::BindOnce(&TranslateScript::Clear,
weak_method_factory_.GetWeakPtr()),
expiration_delay_);
}
callback_list_.Notify(success);
}
} // namespace translate
|