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
|
// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/webui/settings/settings_utils.h"
#include <stddef.h>
#include "base/containers/span.h"
#include "base/environment.h"
#include "base/files/file_util.h"
#include "base/functional/bind.h"
#include "base/memory/weak_ptr.h"
#include "base/nix/xdg_util.h"
#include "base/process/launch.h"
#include "base/task/thread_pool.h"
#include "base/threading/scoped_blocking_call.h"
#include "build/build_config.h"
#include "chrome/browser/tab_contents/tab_util.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/web_contents.h"
using content::BrowserThread;
using content::OpenURLParams;
using content::Referrer;
namespace {
const char* const kCinnamonProxyConfigCommand[] = {"cinnamon-settings",
"network"};
// Command used to configure GNOME 2 proxy settings.
const char* const kGNOME2ProxyConfigCommand[] = {"gnome-network-properties"};
// In GNOME 3, we might need to run gnome-control-center instead. We try this
// only after gnome-network-properties is not found, because older GNOME also
// has this but it doesn't do the same thing. See below where we use it.
const char* const kGNOME3ProxyConfigCommand[] = {"gnome-control-center",
"network"};
// KDE3, 4, and 5 are only slightly different, but incompatible. Go figure.
const char* const kKDE3ProxyConfigCommand[] = {"kcmshell", "proxy"};
const char* const kKDE4ProxyConfigCommand[] = {"kcmshell4", "proxy"};
const char* const kKDE5ProxyConfigCommand[] = {"kcmshell5", "proxy"};
const char* const kKDE6ProxyConfigCommand[] = {"kcmshell6", "kcm_proxy"};
// In Deepin OS, we might need to run dde-control-center instead.
const char* const kDeepinProxyConfigCommand[] = {"dde-control-center", "-m",
"network"};
const char* const kCosmicProxyConfigCommand[] = {"cosmic-settings", "network"};
// The URL for Linux proxy configuration help when not running under a
// supported desktop environment.
constexpr char kLinuxProxyConfigUrl[] = "chrome://linux-proxy-config";
// Show the proxy config URL in the given tab.
void ShowLinuxProxyConfigUrl(base::WeakPtr<content::WebContents> web_contents,
bool launched) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
if (launched) {
return;
}
std::unique_ptr<base::Environment> env(base::Environment::Create());
const char* name = base::nix::GetDesktopEnvironmentName(env.get());
if (name) {
LOG(ERROR) << "Could not find " << name << " network settings in $PATH";
}
OpenURLParams params(GURL(kLinuxProxyConfigUrl), Referrer(),
WindowOpenDisposition::NEW_FOREGROUND_TAB,
ui::PAGE_TRANSITION_LINK, false);
if (web_contents) {
web_contents->OpenURL(params, /*navigation_handle_callback=*/{});
}
}
// Start the given proxy configuration utility.
bool StartProxyConfigUtil(base::span<const char* const> command) {
base::ScopedBlockingCall scoped_blocking_call(FROM_HERE,
base::BlockingType::MAY_BLOCK);
// base::LaunchProcess() returns true ("success") if the fork()
// succeeds, but not necessarily the exec(). We'd like to be able to
// use StartProxyConfigUtil() to search possible options and stop on
// success, so we search $PATH first to predict whether the exec is
// expected to succeed.
std::unique_ptr<base::Environment> env(base::Environment::Create());
if (!base::ExecutableExistsInPath(env.get(), command[0])) {
return false;
}
std::vector<std::string> argv;
for (const char* arg : command) {
argv.push_back(arg);
}
base::Process process = base::LaunchProcess(argv, base::LaunchOptions());
if (!process.IsValid()) {
LOG(ERROR) << "StartProxyConfigUtil failed to start " << command[0];
return false;
}
base::EnsureProcessGetsReaped(std::move(process));
return true;
}
// Detect, and if possible, start the appropriate proxy config utility. On
// failure to do so, show the Linux proxy config URL in a new tab instead.
bool DetectAndStartProxyConfigUtil() {
base::ScopedBlockingCall scoped_blocking_call(FROM_HERE,
base::BlockingType::MAY_BLOCK);
std::unique_ptr<base::Environment> env(base::Environment::Create());
bool launched = false;
switch (base::nix::GetDesktopEnvironment(env.get())) {
case base::nix::DESKTOP_ENVIRONMENT_CINNAMON:
launched = StartProxyConfigUtil(kCinnamonProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_DEEPIN:
launched = StartProxyConfigUtil(kDeepinProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_GNOME:
case base::nix::DESKTOP_ENVIRONMENT_PANTHEON:
case base::nix::DESKTOP_ENVIRONMENT_UKUI:
case base::nix::DESKTOP_ENVIRONMENT_UNITY: {
launched = StartProxyConfigUtil(kGNOME2ProxyConfigCommand);
if (!launched) {
// We try this second, even though it's the newer way, because this
// command existed in older versions of GNOME, but it didn't do the
// same thing. The older command is gone though, so this should do
// the right thing. (Also some distributions have blurred the lines
// between GNOME 2 and 3, so we can't necessarily detect what the
// right thing is based on indications of which version we have.)
launched = StartProxyConfigUtil(kGNOME3ProxyConfigCommand);
}
break;
}
case base::nix::DESKTOP_ENVIRONMENT_KDE3:
launched = StartProxyConfigUtil(kKDE3ProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_KDE4:
launched = StartProxyConfigUtil(kKDE4ProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_KDE5:
launched = StartProxyConfigUtil(kKDE5ProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_KDE6:
launched = StartProxyConfigUtil(kKDE6ProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_COSMIC:
launched = StartProxyConfigUtil(kCosmicProxyConfigCommand);
break;
case base::nix::DESKTOP_ENVIRONMENT_XFCE:
case base::nix::DESKTOP_ENVIRONMENT_LXQT:
case base::nix::DESKTOP_ENVIRONMENT_OTHER:
break;
}
return launched;
}
} // namespace
namespace settings_utils {
void ShowNetworkProxySettings(content::WebContents* web_contents) {
base::ThreadPool::PostTaskAndReplyWithResult(
FROM_HERE, {base::TaskPriority::USER_VISIBLE, base::MayBlock()},
base::BindOnce(&DetectAndStartProxyConfigUtil),
base::BindOnce(&ShowLinuxProxyConfigUrl, web_contents->GetWeakPtr()));
}
} // namespace settings_utils
|