Avi Drissman | 4e1b7bc3 | 2022-09-15 14:03:50 | [diff] [blame] | 1 | // Copyright 2013 The Chromium Authors |
[email protected] | ee75b899 | 2012-01-27 07:53:57 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
vkuzkokov | cbabd58 | 2014-11-06 13:53:54 | [diff] [blame] | 5 | #include "content/shell/browser/shell_devtools_manager_delegate.h" |
[email protected] | ee75b899 | 2012-01-27 07:53:57 | [diff] [blame] | 6 | |
avi | 66a0772 | 2015-12-25 23:38:12 | [diff] [blame] | 7 | #include <stdint.h> |
| 8 | |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 9 | #include <vector> |
| 10 | |
pfeldman | f7c18d024 | 2016-09-08 19:54:26 | [diff] [blame] | 11 | #include "base/atomicops.h" |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 12 | #include "base/command_line.h" |
Weizhong Xia | a10850f | 2024-04-24 18:53:00 | [diff] [blame] | 13 | #include "base/containers/contains.h" |
[email protected] | 0e60fce | 2014-06-04 22:27:20 | [diff] [blame] | 14 | #include "base/files/file_path.h" |
Avi Drissman | adac2199 | 2023-01-11 23:46:39 | [diff] [blame] | 15 | #include "base/functional/bind.h" |
| 16 | #include "base/functional/callback.h" |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 17 | #include "base/strings/string_number_conversions.h" |
[email protected] | 852b34ba | 2013-10-03 16:29:06 | [diff] [blame] | 18 | #include "base/strings/utf_string_conversions.h" |
avi | 66a0772 | 2015-12-25 23:38:12 | [diff] [blame] | 19 | #include "build/build_config.h" |
yzshen | 6051dfd3 | 2015-10-20 19:23:25 | [diff] [blame] | 20 | #include "content/public/browser/browser_context.h" |
[email protected] | 852b34ba | 2013-10-03 16:29:06 | [diff] [blame] | 21 | #include "content/public/browser/devtools_agent_host.h" |
Johannes Henkel | 9d14e4b8 | 2020-01-28 01:29:41 | [diff] [blame] | 22 | #include "content/public/browser/devtools_agent_host_client_channel.h" |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 23 | #include "content/public/browser/devtools_socket_factory.h" |
[email protected] | 852b34ba | 2013-10-03 16:29:06 | [diff] [blame] | 24 | #include "content/public/browser/favicon_status.h" |
| 25 | #include "content/public/browser/navigation_entry.h" |
| 26 | #include "content/public/browser/render_view_host.h" |
[email protected] | eb04099 | 2012-10-10 16:56:00 | [diff] [blame] | 27 | #include "content/public/browser/web_contents.h" |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 28 | #include "content/public/common/content_switches.h" |
[email protected] | eb04099 | 2012-10-10 16:56:00 | [diff] [blame] | 29 | #include "content/public/common/url_constants.h" |
Weizhong Xia | a10850f | 2024-04-24 18:53:00 | [diff] [blame] | 30 | #include "content/shell/browser/protocol/shell_devtools_session.h" |
[email protected] | de7d61ff | 2013-08-20 11:30:41 | [diff] [blame] | 31 | #include "content/shell/browser/shell.h" |
dgozman | 102fee9 | 2015-04-20 15:45:46 | [diff] [blame] | 32 | #include "content/shell/common/shell_content_client.h" |
dgozman | 454746093 | 2017-06-22 03:04:36 | [diff] [blame] | 33 | #include "content/shell/common/shell_switches.h" |
thakis | da47dac | 2017-02-27 14:35:30 | [diff] [blame] | 34 | #include "content/shell/grit/shell_resources.h" |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 35 | #include "net/base/net_errors.h" |
mikecirone | f22f981 | 2016-10-04 03:40:19 | [diff] [blame] | 36 | #include "net/log/net_log_source.h" |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 37 | #include "net/socket/tcp_server_socket.h" |
[email protected] | ee75b899 | 2012-01-27 07:53:57 | [diff] [blame] | 38 | #include "ui/base/resource/resource_bundle.h" |
| 39 | |
Xiaohan Wang | bd08442 | 2022-01-15 18:47:51 | [diff] [blame] | 40 | #if BUILDFLAG(IS_ANDROID) |
[email protected] | ed6635e | 2012-10-27 00:44:09 | [diff] [blame] | 41 | #include "content/public/browser/android/devtools_auth.h" |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 42 | #include "net/socket/unix_domain_server_socket_posix.h" |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 43 | #endif |
[email protected] | ed6635e | 2012-10-27 00:44:09 | [diff] [blame] | 44 | |
vkuzkokov | cbabd58 | 2014-11-06 13:53:54 | [diff] [blame] | 45 | namespace content { |
[email protected] | 852b34ba | 2013-10-03 16:29:06 | [diff] [blame] | 46 | |
[email protected] | ed6635e | 2012-10-27 00:44:09 | [diff] [blame] | 47 | namespace { |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 48 | |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 49 | const int kBackLog = 10; |
| 50 | |
pfeldman | f7c18d024 | 2016-09-08 19:54:26 | [diff] [blame] | 51 | base::subtle::Atomic32 g_last_used_port; |
| 52 | |
Xiaohan Wang | bd08442 | 2022-01-15 18:47:51 | [diff] [blame] | 53 | #if BUILDFLAG(IS_ANDROID) |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 54 | class UnixDomainServerSocketFactory : public content::DevToolsSocketFactory { |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 55 | public: |
| 56 | explicit UnixDomainServerSocketFactory(const std::string& socket_name) |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 57 | : socket_name_(socket_name) {} |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 58 | |
Peter Boström | 9b03653 | 2021-10-28 23:37:28 | [diff] [blame] | 59 | UnixDomainServerSocketFactory(const UnixDomainServerSocketFactory&) = delete; |
| 60 | UnixDomainServerSocketFactory& operator=( |
| 61 | const UnixDomainServerSocketFactory&) = delete; |
| 62 | |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 63 | private: |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 64 | // content::DevToolsSocketFactory. |
dcheng | 6003e0b | 2016-04-09 18:42:34 | [diff] [blame] | 65 | std::unique_ptr<net::ServerSocket> CreateForHttpServer() override { |
| 66 | std::unique_ptr<net::UnixDomainServerSocket> socket( |
danakj | c988f0be | 2019-05-17 20:48:01 | [diff] [blame] | 67 | new net::UnixDomainServerSocket( |
| 68 | base::BindRepeating(&CanUserConnectToDevTools), |
| 69 | true /* use_abstract_namespace */)); |
tfarina | a7b245d | 2016-02-02 02:03:49 | [diff] [blame] | 70 | if (socket->BindAndListen(socket_name_, kBackLog) != net::OK) |
Lei Zhang | df291f6 | 2021-04-14 17:23:44 | [diff] [blame] | 71 | return nullptr; |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 72 | |
tfarina | a7b245d | 2016-02-02 02:03:49 | [diff] [blame] | 73 | return std::move(socket); |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 74 | } |
| 75 | |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 76 | std::unique_ptr<net::ServerSocket> CreateForTethering( |
| 77 | std::string* out_name) override { |
| 78 | return nullptr; |
| 79 | } |
| 80 | |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 81 | std::string socket_name_; |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 82 | }; |
| 83 | #else |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 84 | class TCPServerSocketFactory : public content::DevToolsSocketFactory { |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 85 | public: |
avi | 66a0772 | 2015-12-25 23:38:12 | [diff] [blame] | 86 | TCPServerSocketFactory(const std::string& address, uint16_t port) |
| 87 | : address_(address), port_(port) {} |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 88 | |
Peter Boström | 9b03653 | 2021-10-28 23:37:28 | [diff] [blame] | 89 | TCPServerSocketFactory(const TCPServerSocketFactory&) = delete; |
| 90 | TCPServerSocketFactory& operator=(const TCPServerSocketFactory&) = delete; |
| 91 | |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 92 | private: |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 93 | // content::DevToolsSocketFactory. |
dcheng | 6003e0b | 2016-04-09 18:42:34 | [diff] [blame] | 94 | std::unique_ptr<net::ServerSocket> CreateForHttpServer() override { |
| 95 | std::unique_ptr<net::ServerSocket> socket( |
mikecirone | f22f981 | 2016-10-04 03:40:19 | [diff] [blame] | 96 | new net::TCPServerSocket(nullptr, net::NetLogSource())); |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 97 | if (socket->ListenWithAddressAndPort(address_, port_, kBackLog) != net::OK) |
Lei Zhang | df291f6 | 2021-04-14 17:23:44 | [diff] [blame] | 98 | return nullptr; |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 99 | |
pfeldman | f7c18d024 | 2016-09-08 19:54:26 | [diff] [blame] | 100 | net::IPEndPoint endpoint; |
| 101 | if (socket->GetLocalAddress(&endpoint) == net::OK) |
| 102 | base::subtle::NoBarrier_Store(&g_last_used_port, endpoint.port()); |
| 103 | |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 104 | return socket; |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 105 | } |
| 106 | |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 107 | std::unique_ptr<net::ServerSocket> CreateForTethering( |
| 108 | std::string* out_name) override { |
| 109 | return nullptr; |
| 110 | } |
| 111 | |
vkuzkokov | 7942676 | 2015-01-13 08:03:00 | [diff] [blame] | 112 | std::string address_; |
avi | 66a0772 | 2015-12-25 23:38:12 | [diff] [blame] | 113 | uint16_t port_; |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 114 | }; |
| 115 | #endif |
| 116 | |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 117 | std::unique_ptr<content::DevToolsSocketFactory> CreateSocketFactory() { |
avi | 83883c8 | 2014-12-23 00:08:49 | [diff] [blame] | 118 | const base::CommandLine& command_line = |
| 119 | *base::CommandLine::ForCurrentProcess(); |
Xiaohan Wang | bd08442 | 2022-01-15 18:47:51 | [diff] [blame] | 120 | #if BUILDFLAG(IS_ANDROID) |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 121 | std::string socket_name = "content_shell_devtools_remote"; |
| 122 | if (command_line.HasSwitch(switches::kRemoteDebuggingSocketName)) { |
| 123 | socket_name = command_line.GetSwitchValueASCII( |
| 124 | switches::kRemoteDebuggingSocketName); |
| 125 | } |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 126 | return std::unique_ptr<content::DevToolsSocketFactory>( |
byungchul | 38c3ae7 | 2014-08-25 23:27:46 | [diff] [blame] | 127 | new UnixDomainServerSocketFactory(socket_name)); |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 128 | #else |
| 129 | // See if the user specified a port on the command line (useful for |
| 130 | // automation). If not, use an ephemeral port by specifying 0. |
avi | 66a0772 | 2015-12-25 23:38:12 | [diff] [blame] | 131 | uint16_t port = 0; |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 132 | if (command_line.HasSwitch(switches::kRemoteDebuggingPort)) { |
| 133 | int temp_port; |
| 134 | std::string port_str = |
| 135 | command_line.GetSwitchValueASCII(switches::kRemoteDebuggingPort); |
| 136 | if (base::StringToInt(port_str, &temp_port) && |
yzshen | 6051dfd3 | 2015-10-20 19:23:25 | [diff] [blame] | 137 | temp_port >= 0 && temp_port < 65535) { |
avi | 66a0772 | 2015-12-25 23:38:12 | [diff] [blame] | 138 | port = static_cast<uint16_t>(temp_port); |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 139 | } else { |
| 140 | DLOG(WARNING) << "Invalid http debugger port number " << temp_port; |
| 141 | } |
| 142 | } |
Tiago Vignatti | a293c2c | 2023-03-28 20:31:04 | [diff] [blame] | 143 | // By default listen to incoming DevTools connections on localhost. |
| 144 | std::string address_str = net::IPAddress::IPv4Localhost().ToString(); |
| 145 | if (command_line.HasSwitch(switches::kRemoteDebuggingAddress)) { |
| 146 | net::IPAddress address; |
| 147 | address_str = |
| 148 | command_line.GetSwitchValueASCII(switches::kRemoteDebuggingAddress); |
| 149 | if (!address.AssignFromIPLiteral(address_str)) { |
| 150 | DLOG(WARNING) << "Invalid devtools server address: " << address_str; |
| 151 | } |
| 152 | } |
pfeldman | f1a1694 | 2016-09-02 21:28:00 | [diff] [blame] | 153 | return std::unique_ptr<content::DevToolsSocketFactory>( |
Tiago Vignatti | a293c2c | 2023-03-28 20:31:04 | [diff] [blame] | 154 | new TCPServerSocketFactory(address_str, port)); |
[email protected] | ed6635e | 2012-10-27 00:44:09 | [diff] [blame] | 155 | #endif |
[email protected] | 90e6c541 | 2013-04-16 22:45:25 | [diff] [blame] | 156 | } |
[email protected] | 852b34ba | 2013-10-03 16:29:06 | [diff] [blame] | 157 | |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 158 | } // namespace |
vkuzkokov | cbabd58 | 2014-11-06 13:53:54 | [diff] [blame] | 159 | |
dgozman | 252e18d | 2014-09-22 12:40:06 | [diff] [blame] | 160 | // ShellDevToolsManagerDelegate ---------------------------------------------- |
| 161 | |
vkuzkokov | cbabd58 | 2014-11-06 13:53:54 | [diff] [blame] | 162 | // static |
pfeldman | f7c18d024 | 2016-09-08 19:54:26 | [diff] [blame] | 163 | int ShellDevToolsManagerDelegate::GetHttpHandlerPort() { |
| 164 | return base::subtle::NoBarrier_Load(&g_last_used_port); |
| 165 | } |
| 166 | |
| 167 | // static |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 168 | void ShellDevToolsManagerDelegate::StartHttpHandler( |
vkuzkokov | cbabd58 | 2014-11-06 13:53:54 | [diff] [blame] | 169 | BrowserContext* browser_context) { |
| 170 | std::string frontend_url; |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 171 | DevToolsAgentHost::StartRemoteDebuggingServer( |
Pavel Feldman | c8a484b5 | 2018-02-07 21:07:32 | [diff] [blame] | 172 | CreateSocketFactory(), browser_context->GetPath(), base::FilePath()); |
Pavel Feldman | 2b11e235 | 2017-10-25 05:24:01 | [diff] [blame] | 173 | |
| 174 | const base::CommandLine& command_line = |
| 175 | *base::CommandLine::ForCurrentProcess(); |
| 176 | if (command_line.HasSwitch(switches::kRemoteDebuggingPipe)) |
Dmitry Gozman | 8bbf58d | 2020-11-13 22:48:26 | [diff] [blame] | 177 | DevToolsAgentHost::StartRemoteDebuggingPipeHandler(base::OnceClosure()); |
vkuzkokov | cbabd58 | 2014-11-06 13:53:54 | [diff] [blame] | 178 | } |
| 179 | |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 180 | // static |
| 181 | void ShellDevToolsManagerDelegate::StopHttpHandler() { |
| 182 | DevToolsAgentHost::StopRemoteDebuggingServer(); |
| 183 | } |
| 184 | |
Pavel Feldman | 43f56b7c | 2016-08-30 00:04:35 | [diff] [blame] | 185 | ShellDevToolsManagerDelegate::ShellDevToolsManagerDelegate( |
| 186 | BrowserContext* browser_context) |
| 187 | : browser_context_(browser_context) { |
dgozman | 252e18d | 2014-09-22 12:40:06 | [diff] [blame] | 188 | } |
| 189 | |
| 190 | ShellDevToolsManagerDelegate::~ShellDevToolsManagerDelegate() { |
| 191 | } |
| 192 | |
Andrey Lushnikov | 36299bc | 2018-08-23 22:09:54 | [diff] [blame] | 193 | BrowserContext* ShellDevToolsManagerDelegate::GetDefaultBrowserContext() { |
| 194 | return browser_context_; |
| 195 | } |
| 196 | |
Andrey Lushnikov | df165198 | 2018-09-14 19:00:37 | [diff] [blame] | 197 | void ShellDevToolsManagerDelegate::ClientAttached( |
Johannes Henkel | 9d14e4b8 | 2020-01-28 01:29:41 | [diff] [blame] | 198 | content::DevToolsAgentHostClientChannel* channel) { |
Andrey Lushnikov | df165198 | 2018-09-14 19:00:37 | [diff] [blame] | 199 | // Make sure we don't receive notifications twice for the same client. |
Weizhong Xia | a10850f | 2024-04-24 18:53:00 | [diff] [blame] | 200 | CHECK(!base::Contains(sessions_, channel)); |
| 201 | sessions_.emplace( |
| 202 | channel, |
| 203 | std::make_unique<shell::protocol::ShellDevToolsSession>( |
| 204 | base::raw_ref<BrowserContext>::from_ptr(browser_context_), channel)); |
Andrey Lushnikov | df165198 | 2018-09-14 19:00:37 | [diff] [blame] | 205 | } |
| 206 | |
| 207 | void ShellDevToolsManagerDelegate::ClientDetached( |
Johannes Henkel | 9d14e4b8 | 2020-01-28 01:29:41 | [diff] [blame] | 208 | content::DevToolsAgentHostClientChannel* channel) { |
Weizhong Xia | a10850f | 2024-04-24 18:53:00 | [diff] [blame] | 209 | sessions_.erase(channel); |
| 210 | } |
| 211 | |
| 212 | void ShellDevToolsManagerDelegate::HandleCommand( |
| 213 | content::DevToolsAgentHostClientChannel* channel, |
| 214 | base::span<const uint8_t> message, |
| 215 | NotHandledCallback callback) { |
| 216 | auto& session = sessions_.at(channel); |
| 217 | session->HandleCommand(message, std::move(callback)); |
Andrey Lushnikov | df165198 | 2018-09-14 19:00:37 | [diff] [blame] | 218 | } |
| 219 | |
Danil Somsikov | d5b3684 | 2022-12-16 11:24:50 | [diff] [blame] | 220 | scoped_refptr<DevToolsAgentHost> ShellDevToolsManagerDelegate::CreateNewTarget( |
| 221 | const GURL& url, |
Michael Thiessen | 08b5aad | 2025-03-14 08:27:22 | [diff] [blame] | 222 | content::DevToolsManagerDelegate::TargetType target_type, |
| 223 | bool new_window) { |
danakj | 24577b1 | 2020-05-13 22:38:18 | [diff] [blame] | 224 | Shell* shell = Shell::CreateNewWindow(browser_context_, url, nullptr, |
| 225 | Shell::GetShellDefaultSize()); |
Danil Somsikov | daa9772bb | 2023-08-03 10:02:10 | [diff] [blame] | 226 | return target_type == content::DevToolsManagerDelegate::kTab |
| 227 | ? DevToolsAgentHost::GetOrCreateForTab(shell->web_contents()) |
| 228 | : DevToolsAgentHost::GetOrCreateFor(shell->web_contents()); |
Pavel Feldman | 43f56b7c | 2016-08-30 00:04:35 | [diff] [blame] | 229 | } |
| 230 | |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 231 | std::string ShellDevToolsManagerDelegate::GetDiscoveryPageHTML() { |
Tiago Vignatti | 9bbb4a5 | 2023-04-06 22:05:57 | [diff] [blame] | 232 | #if BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_IOS) |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 233 | return std::string(); |
| 234 | #else |
Kristian Kirs | b9e27051 | 2020-07-08 15:21:20 | [diff] [blame] | 235 | return ui::ResourceBundle::GetSharedInstance().LoadDataResourceString( |
| 236 | IDR_CONTENT_SHELL_DEVTOOLS_DISCOVERY_PAGE); |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 237 | #endif |
| 238 | } |
| 239 | |
Pavel Feldman | c8a484b5 | 2018-02-07 21:07:32 | [diff] [blame] | 240 | bool ShellDevToolsManagerDelegate::HasBundledFrontendResources() { |
Tiago Vignatti | 9bbb4a5 | 2023-04-06 22:05:57 | [diff] [blame] | 241 | #if BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_IOS) |
Pavel Feldman | c8a484b5 | 2018-02-07 21:07:32 | [diff] [blame] | 242 | return false; |
Nico Weber | 2771b0f | 2021-08-16 17:16:24 | [diff] [blame] | 243 | #else |
Pavel Feldman | c8a484b5 | 2018-02-07 21:07:32 | [diff] [blame] | 244 | return true; |
Nico Weber | 2771b0f | 2021-08-16 17:16:24 | [diff] [blame] | 245 | #endif |
pfeldman | 1062876 | 2016-09-08 07:59:26 | [diff] [blame] | 246 | } |
| 247 | |
[email protected] | ee75b899 | 2012-01-27 07:53:57 | [diff] [blame] | 248 | } // namespace content |