- add sources.
[platform/framework/web/crosswalk.git] / src / content / browser / browser_url_handler_impl.cc
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "content/browser/browser_url_handler_impl.h"
6
7 #include "base/command_line.h"
8 #include "base/strings/string_util.h"
9 #include "content/browser/frame_host/debug_urls.h"
10 #include "content/browser/webui/web_ui_impl.h"
11 #include "content/public/browser/content_browser_client.h"
12 #include "content/public/common/content_switches.h"
13 #include "content/public/common/url_constants.h"
14 #include "url/gurl.h"
15
16 namespace content {
17
18 // Handles rewriting view-source URLs for what we'll actually load.
19 static bool HandleViewSource(GURL* url, BrowserContext* browser_context) {
20   if (url->SchemeIs(kViewSourceScheme)) {
21     // Load the inner URL instead.
22     *url = GURL(url->GetContent());
23
24     // Bug 26129: limit view-source to view the content and not any
25     // other kind of 'active' url scheme like 'javascript' or 'data'.
26     static const char* const allowed_sub_schemes[] = {
27       kHttpScheme, kHttpsScheme, chrome::kFtpScheme,
28       chrome::kChromeDevToolsScheme, chrome::kChromeUIScheme,
29       chrome::kFileScheme, chrome::kFileSystemScheme
30     };
31
32     bool is_sub_scheme_allowed = false;
33     for (size_t i = 0; i < arraysize(allowed_sub_schemes); i++) {
34       if (url->SchemeIs(allowed_sub_schemes[i])) {
35         is_sub_scheme_allowed = true;
36         break;
37       }
38     }
39
40     if (!is_sub_scheme_allowed) {
41       *url = GURL(kAboutBlankURL);
42       return false;
43     }
44
45     return true;
46   }
47   return false;
48 }
49
50 // Turns a non view-source URL into the corresponding view-source URL.
51 static bool ReverseViewSource(GURL* url, BrowserContext* browser_context) {
52   // No action necessary if the URL is already view-source:
53   if (url->SchemeIs(kViewSourceScheme))
54     return false;
55
56   url_canon::Replacements<char> repl;
57   repl.SetScheme(kViewSourceScheme,
58                  url_parse::Component(0, strlen(kViewSourceScheme)));
59   repl.SetPath(url->spec().c_str(),
60       url_parse::Component(0, url->spec().size()));
61   *url = url->ReplaceComponents(repl);
62   return true;
63 }
64
65 static bool DebugURLHandler(GURL* url, BrowserContext* browser_context) {
66   // If running inside the Telemetry test harness, allow automated
67   // navigations to access browser-side debug URLs. They must use the
68   // chrome:// scheme, since the about: scheme won't be rewritten in
69   // this code path.
70   if (CommandLine::ForCurrentProcess()->HasSwitch(
71           switches::kEnableGpuBenchmarking)) {
72     if (HandleDebugURL(*url, PAGE_TRANSITION_FROM_ADDRESS_BAR)) {
73       return true;
74     }
75   }
76
77   // Circumvent processing URLs that the renderer process will handle.
78   return *url == GURL(kChromeUICrashURL) ||
79          *url == GURL(kChromeUIHangURL) ||
80          *url == GURL(kChromeUIKillURL) ||
81          *url == GURL(kChromeUIShorthangURL);
82 }
83
84 // static
85 BrowserURLHandler* BrowserURLHandler::GetInstance() {
86   return BrowserURLHandlerImpl::GetInstance();
87 }
88
89 // static
90 BrowserURLHandler::URLHandler BrowserURLHandler::null_handler() {
91   // Required for VS2010: http://connect.microsoft.com/VisualStudio/feedback/details/520043/error-converting-from-null-to-a-pointer-type-in-std-pair
92   return NULL;
93 }
94
95 // static
96 BrowserURLHandlerImpl* BrowserURLHandlerImpl::GetInstance() {
97   return Singleton<BrowserURLHandlerImpl>::get();
98 }
99
100 BrowserURLHandlerImpl::BrowserURLHandlerImpl() {
101   AddHandlerPair(&DebugURLHandler, BrowserURLHandlerImpl::null_handler());
102
103   GetContentClient()->browser()->BrowserURLHandlerCreated(this);
104
105   // view-source:
106   AddHandlerPair(&HandleViewSource, &ReverseViewSource);
107 }
108
109 BrowserURLHandlerImpl::~BrowserURLHandlerImpl() {
110 }
111
112 void BrowserURLHandlerImpl::AddHandlerPair(URLHandler handler,
113                                            URLHandler reverse_handler) {
114   url_handlers_.push_back(HandlerPair(handler, reverse_handler));
115 }
116
117 void BrowserURLHandlerImpl::RewriteURLIfNecessary(
118     GURL* url,
119     BrowserContext* browser_context,
120     bool* reverse_on_redirect) {
121   for (size_t i = 0; i < url_handlers_.size(); ++i) {
122     URLHandler handler = *url_handlers_[i].first;
123     if (handler && handler(url, browser_context)) {
124       *reverse_on_redirect = (url_handlers_[i].second != NULL);
125       return;
126     }
127   }
128 }
129
130 bool BrowserURLHandlerImpl::ReverseURLRewrite(
131     GURL* url, const GURL& original, BrowserContext* browser_context) {
132   for (size_t i = 0; i < url_handlers_.size(); ++i) {
133     URLHandler reverse_rewriter = *url_handlers_[i].second;
134     if (reverse_rewriter) {
135       GURL test_url(original);
136       URLHandler handler = *url_handlers_[i].first;
137       if (!handler) {
138         if (reverse_rewriter(url, browser_context))
139           return true;
140       } else if (handler(&test_url, browser_context)) {
141         return reverse_rewriter(url, browser_context);
142       }
143     }
144   }
145   return false;
146 }
147
148 }  // namespace content