OLD | NEW |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "mojo/shell/application_instance.h" | 5 #include "mojo/shell/application_instance.h" |
6 | 6 |
7 #include "base/bind.h" | 7 #include "base/bind.h" |
8 #include "base/stl_util.h" | 8 #include "base/stl_util.h" |
9 #include "mojo/application/public/interfaces/content_handler.mojom.h" | 9 #include "mojo/application/public/interfaces/content_handler.mojom.h" |
10 #include "mojo/common/common_type_converters.h" | 10 #include "mojo/common/common_type_converters.h" |
11 #include "mojo/common/url_type_converters.h" | 11 #include "mojo/common/url_type_converters.h" |
12 #include "mojo/shell/application_manager.h" | 12 #include "mojo/shell/application_manager.h" |
13 #include "mojo/shell/content_handler_connection.h" | 13 #include "mojo/shell/content_handler_connection.h" |
14 | 14 |
15 namespace mojo { | 15 namespace mojo { |
16 namespace shell { | 16 namespace shell { |
| 17 namespace { |
| 18 |
| 19 // It's valid to specify mojo: URLs in the filter either as mojo:foo or |
| 20 // mojo://foo/ - but we store the filter in the latter form. |
| 21 CapabilityFilter CanonicalizeFilter(const CapabilityFilter& filter) { |
| 22 CapabilityFilter canonicalized; |
| 23 for (CapabilityFilter::const_iterator it = filter.begin(); |
| 24 it != filter.end(); |
| 25 ++it) { |
| 26 if (it->first == "*") |
| 27 canonicalized[it->first] = it->second; |
| 28 else |
| 29 canonicalized[GURL(it->first).spec()] = it->second; |
| 30 } |
| 31 return canonicalized; |
| 32 } |
| 33 |
| 34 } // namespace |
17 | 35 |
18 ApplicationInstance::ApplicationInstance( | 36 ApplicationInstance::ApplicationInstance( |
19 ApplicationPtr application, | 37 ApplicationPtr application, |
20 ApplicationManager* manager, | 38 ApplicationManager* manager, |
| 39 const Identity& originator_identity, |
21 const Identity& identity, | 40 const Identity& identity, |
| 41 const CapabilityFilter& filter, |
22 uint32_t requesting_content_handler_id, | 42 uint32_t requesting_content_handler_id, |
23 const base::Closure& on_application_end) | 43 const base::Closure& on_application_end) |
24 : manager_(manager), | 44 : manager_(manager), |
| 45 originator_identity_(originator_identity), |
25 identity_(identity), | 46 identity_(identity), |
26 allow_any_application_(identity.filter().size() == 1 && | 47 filter_(CanonicalizeFilter(filter)), |
27 identity.filter().count("*") == 1), | 48 allow_any_application_(filter.size() == 1 && filter.count("*") == 1), |
28 requesting_content_handler_id_(requesting_content_handler_id), | 49 requesting_content_handler_id_(requesting_content_handler_id), |
29 on_application_end_(on_application_end), | 50 on_application_end_(on_application_end), |
30 application_(application.Pass()), | 51 application_(application.Pass()), |
31 binding_(this), | 52 binding_(this), |
32 queue_requests_(false) { | 53 queue_requests_(false) { |
33 binding_.set_connection_error_handler([this]() { OnConnectionError(); }); | 54 binding_.set_connection_error_handler([this]() { OnConnectionError(); }); |
34 } | 55 } |
35 | 56 |
36 ApplicationInstance::~ApplicationInstance() { | 57 ApplicationInstance::~ApplicationInstance() { |
37 for (auto request : queued_client_requests_) | 58 for (auto request : queued_client_requests_) |
38 request->connect_callback().Run(kInvalidContentHandlerID); | 59 request->connect_callback().Run(kInvalidContentHandlerID); |
39 STLDeleteElements(&queued_client_requests_); | 60 STLDeleteElements(&queued_client_requests_); |
40 } | 61 } |
41 | 62 |
42 void ApplicationInstance::InitializeApplication() { | 63 void ApplicationInstance::InitializeApplication() { |
43 ShellPtr shell; | 64 ShellPtr shell; |
44 binding_.Bind(GetProxy(&shell)); | 65 binding_.Bind(GetProxy(&shell)); |
45 application_->Initialize(shell.Pass(), identity_.url().spec()); | 66 application_->Initialize(shell.Pass(), identity_.url.spec()); |
46 } | 67 } |
47 | 68 |
48 void ApplicationInstance::ConnectToClient( | 69 void ApplicationInstance::ConnectToClient( |
49 scoped_ptr<ConnectToApplicationParams> params) { | 70 scoped_ptr<ConnectToApplicationParams> params) { |
50 if (queue_requests_) { | 71 if (queue_requests_) { |
51 queued_client_requests_.push_back(params.release()); | 72 queued_client_requests_.push_back(params.release()); |
52 return; | 73 return; |
53 } | 74 } |
54 | 75 |
55 CallAcceptConnection(params.Pass()); | 76 CallAcceptConnection(params.Pass()); |
56 } | 77 } |
57 | 78 |
58 // Shell implementation: | 79 // Shell implementation: |
59 void ApplicationInstance::ConnectToApplication( | 80 void ApplicationInstance::ConnectToApplication( |
60 URLRequestPtr app_request, | 81 URLRequestPtr app_request, |
61 InterfaceRequest<ServiceProvider> services, | 82 InterfaceRequest<ServiceProvider> services, |
62 ServiceProviderPtr exposed_services, | 83 ServiceProviderPtr exposed_services, |
63 CapabilityFilterPtr filter, | 84 CapabilityFilterPtr filter, |
64 const ConnectToApplicationCallback& callback) { | 85 const ConnectToApplicationCallback& callback) { |
65 std::string url_string = app_request->url.To<std::string>(); | 86 std::string url_string = app_request->url.To<std::string>(); |
66 GURL url(url_string); | 87 GURL url(url_string); |
67 if (!url.is_valid()) { | 88 if (!url.is_valid()) { |
68 LOG(ERROR) << "Error: invalid URL: " << url_string; | 89 LOG(ERROR) << "Error: invalid URL: " << url_string; |
69 callback.Run(kInvalidContentHandlerID); | 90 callback.Run(kInvalidContentHandlerID); |
70 return; | 91 return; |
71 } | 92 } |
72 if (allow_any_application_ || | 93 if (allow_any_application_ || filter_.find(url.spec()) != filter_.end()) { |
73 identity_.filter().find(url.spec()) != identity_.filter().end()) { | |
74 CapabilityFilter capability_filter = GetPermissiveCapabilityFilter(); | 94 CapabilityFilter capability_filter = GetPermissiveCapabilityFilter(); |
75 if (!filter.is_null()) | 95 if (!filter.is_null()) |
76 capability_filter = filter->filter.To<CapabilityFilter>(); | 96 capability_filter = filter->filter.To<CapabilityFilter>(); |
77 | 97 |
78 scoped_ptr<ConnectToApplicationParams> params( | 98 scoped_ptr<ConnectToApplicationParams> params( |
79 new ConnectToApplicationParams); | 99 new ConnectToApplicationParams); |
80 params->SetSource(this); | 100 params->SetOriginatorInfo(this); |
81 params->SetTargetURLRequest( | 101 params->SetURLInfo(app_request.Pass()); |
82 app_request.Pass(), | |
83 Identity(GURL(app_request->url), std::string(), capability_filter)); | |
84 params->set_services(services.Pass()); | 102 params->set_services(services.Pass()); |
85 params->set_exposed_services(exposed_services.Pass()); | 103 params->set_exposed_services(exposed_services.Pass()); |
| 104 params->set_filter(capability_filter); |
86 params->set_connect_callback(callback); | 105 params->set_connect_callback(callback); |
87 manager_->ConnectToApplication(params.Pass()); | 106 manager_->ConnectToApplication(params.Pass()); |
88 } else { | 107 } else { |
89 LOG(WARNING) << "CapabilityFilter prevented connection from: " << | 108 LOG(WARNING) << "CapabilityFilter prevented connection from: " << |
90 identity_.url() << " to: " << url.spec(); | 109 identity_.url << " to: " << url.spec(); |
91 callback.Run(kInvalidContentHandlerID); | 110 callback.Run(kInvalidContentHandlerID); |
92 } | 111 } |
93 } | 112 } |
94 | 113 |
95 void ApplicationInstance::QuitApplication() { | 114 void ApplicationInstance::QuitApplication() { |
96 queue_requests_ = true; | 115 queue_requests_ = true; |
97 application_->OnQuitRequested( | 116 application_->OnQuitRequested( |
98 base::Bind(&ApplicationInstance::OnQuitRequestedResult, | 117 base::Bind(&ApplicationInstance::OnQuitRequestedResult, |
99 base::Unretained(this))); | 118 base::Unretained(this))); |
100 } | 119 } |
101 | 120 |
102 void ApplicationInstance::CallAcceptConnection( | 121 void ApplicationInstance::CallAcceptConnection( |
103 scoped_ptr<ConnectToApplicationParams> params) { | 122 scoped_ptr<ConnectToApplicationParams> params) { |
104 params->connect_callback().Run(requesting_content_handler_id_); | 123 params->connect_callback().Run(requesting_content_handler_id_); |
105 AllowedInterfaces interfaces; | 124 AllowedInterfaces interfaces; |
106 interfaces.insert("*"); | 125 interfaces.insert("*"); |
107 if (!params->source().is_null()) | 126 if (!params->originator_identity().is_null()) |
108 interfaces = GetAllowedInterfaces(params->source().filter(), identity_); | 127 interfaces = GetAllowedInterfaces(params->originator_filter(), identity_); |
109 | 128 |
110 application_->AcceptConnection( | 129 application_->AcceptConnection( |
111 params->source().url().spec(), params->TakeServices(), | 130 params->originator_identity().url.spec(), params->TakeServices(), |
112 params->TakeExposedServices(), Array<String>::From(interfaces).Pass(), | 131 params->TakeExposedServices(), Array<String>::From(interfaces).Pass(), |
113 params->target().url().spec()); | 132 params->app_url().spec()); |
114 } | 133 } |
115 | 134 |
116 void ApplicationInstance::OnConnectionError() { | 135 void ApplicationInstance::OnConnectionError() { |
117 std::vector<ConnectToApplicationParams*> queued_client_requests; | 136 std::vector<ConnectToApplicationParams*> queued_client_requests; |
118 queued_client_requests_.swap(queued_client_requests); | 137 queued_client_requests_.swap(queued_client_requests); |
119 auto manager = manager_; | 138 auto manager = manager_; |
120 manager_->OnApplicationInstanceError(this); | 139 manager_->OnApplicationInstanceError(this); |
121 //|this| is deleted. | 140 //|this| is deleted. |
122 | 141 |
123 // If any queued requests came to shell during time it was shutting down, | 142 // If any queued requests came to shell during time it was shutting down, |
124 // start them now. | 143 // start them now. |
125 for (auto request : queued_client_requests) { | 144 for (auto request : queued_client_requests) { |
126 // Unfortunately, it is possible that |request->target_url_request()| is | 145 // Unfortunately, it is possible that |request->app_url_request()| is null |
127 // null at this point. Consider the following sequence: | 146 // at this point. Consider the following sequence: |
128 // 1) connect_request_1 arrives at the application manager; the manager | 147 // 1) connect_request_1 arrives at the application manager; the manager |
129 // decides to fetch the app. | 148 // decides to fetch the app. |
130 // 2) connect_request_2 arrives for the same app; because the app is not | 149 // 2) connect_request_2 arrives for the same app; because the app is not |
131 // running yet, the manager decides to fetch the app again. | 150 // running yet, the manager decides to fetch the app again. |
132 // 3) The fetch for step (1) completes and an application instance app_a is | 151 // 3) The fetch for step (1) completes and an application instance app_a is |
133 // registered. | 152 // registered. |
134 // 4) app_a goes into two-phase shutdown. | 153 // 4) app_a goes into two-phase shutdown. |
135 // 5) The fetch for step (2) completes; the manager finds that there is a | 154 // 5) The fetch for step (2) completes; the manager finds that there is a |
136 // running app already, so it connects to app_a. | 155 // running app already, so it connects to app_a. |
137 // 6) connect_request_2 is queued (and eventually gets here), but its | 156 // 6) connect_request_2 is queued (and eventually gets here), but its |
138 // original_request field was already lost to NetworkFetcher at step (2). | 157 // original_request field was already lost to NetworkFetcher at step (2). |
139 // | 158 // |
140 // TODO(yzshen): It seems we should register a pending application instance | 159 // TODO(yzshen): It seems we should register a pending application instance |
141 // before starting the fetch. So at step (2) the application manager knows | 160 // before starting the fetch. So at step (2) the application manager knows |
142 // that it can wait for the first fetch to complete instead of doing a | 161 // that it can wait for the first fetch to complete instead of doing a |
143 // second one directly. | 162 // second one directly. |
144 if (!request->target_url_request()) { | 163 if (!request->app_url_request()) { |
145 URLRequestPtr url_request = mojo::URLRequest::New(); | 164 URLRequestPtr url_request = mojo::URLRequest::New(); |
146 url_request->url = request->target().url().spec(); | 165 url_request->url = request->app_url().spec(); |
147 request->SetTargetURLRequest(url_request.Pass(), request->target()); | 166 request->SetURLInfo(url_request.Pass()); |
148 } | 167 } |
149 manager->ConnectToApplication(make_scoped_ptr(request)); | 168 manager->ConnectToApplication(make_scoped_ptr(request)); |
150 } | 169 } |
151 } | 170 } |
152 | 171 |
153 void ApplicationInstance::OnQuitRequestedResult(bool can_quit) { | 172 void ApplicationInstance::OnQuitRequestedResult(bool can_quit) { |
154 if (can_quit) | 173 if (can_quit) |
155 return; | 174 return; |
156 | 175 |
157 queue_requests_ = false; | 176 queue_requests_ = false; |
158 for (auto request : queued_client_requests_) | 177 for (auto request : queued_client_requests_) |
159 CallAcceptConnection(make_scoped_ptr(request)); | 178 CallAcceptConnection(make_scoped_ptr(request)); |
160 | 179 |
161 queued_client_requests_.clear(); | 180 queued_client_requests_.clear(); |
162 } | 181 } |
163 | 182 |
164 } // namespace shell | 183 } // namespace shell |
165 } // namespace mojo | 184 } // namespace mojo |
OLD | NEW |