1 // Copyright 2014 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.
5 #include "content/renderer/web_ui_mojo_context_state.h"
8 #include "base/stl_util.h"
9 #include "content/public/renderer/resource_fetcher.h"
10 #include "content/renderer/web_ui_runner.h"
11 #include "gin/converter.h"
12 #include "gin/modules/module_registry.h"
13 #include "gin/per_context_data.h"
14 #include "gin/public/context_holder.h"
15 #include "gin/try_catch.h"
16 #include "mojo/bindings/js/core.h"
17 #include "mojo/bindings/js/handle.h"
18 #include "mojo/bindings/js/support.h"
19 #include "third_party/WebKit/public/platform/WebURLResponse.h"
20 #include "third_party/WebKit/public/web/WebFrame.h"
21 #include "third_party/WebKit/public/web/WebScriptSource.h"
24 using v8::HandleScope
;
27 using v8::ObjectTemplate
;
34 // All modules have this prefixed to them when downloading.
35 // TODO(sky): move this into some common place.
36 const char kModulePrefix
[] = "chrome://mojo/";
38 void RunMain(base::WeakPtr
<gin::Runner
> runner
,
39 mojo::ScopedMessagePipeHandle
* handle
,
40 v8::Handle
<v8::Value
> module
) {
41 v8::Isolate
* isolate
= runner
->GetContextHolder()->isolate();
42 v8::Handle
<v8::Function
> start
;
43 CHECK(gin::ConvertFromV8(isolate
, module
, &start
));
44 v8::Handle
<v8::Value
> args
[] = {
45 gin::ConvertToV8(isolate
, mojo::Handle(handle
->release().value())) };
46 runner
->Call(start
, runner
->global(), 1, args
);
51 // WebUIMojo -------------------------------------------------------------------
53 WebUIMojoContextState::WebUIMojoContextState(blink::WebFrame
* frame
,
54 v8::Handle
<v8::Context
> context
)
56 module_added_(false) {
57 gin::PerContextData
* context_data
= gin::PerContextData::From(context
);
58 gin::ContextHolder
* context_holder
= context_data
->context_holder();
59 runner_
.reset(new WebUIRunner(frame_
, context_holder
));
60 gin::Runner::Scope
scoper(runner_
.get());
61 gin::ModuleRegistry::From(context
)->AddObserver(this);
62 runner_
->RegisterBuiltinModules();
63 gin::ModuleRegistry::InstallGlobals(context
->GetIsolate(), context
->Global());
64 // Warning |frame| may be destroyed.
65 // TODO(sky): add test for this.
68 WebUIMojoContextState::~WebUIMojoContextState() {
69 gin::Runner::Scope
scoper(runner_
.get());
70 gin::ModuleRegistry::From(
71 runner_
->GetContextHolder()->context())->RemoveObserver(this);
74 void WebUIMojoContextState::SetHandle(mojo::ScopedMessagePipeHandle handle
) {
75 gin::ContextHolder
* context_holder
= runner_
->GetContextHolder();
76 mojo::ScopedMessagePipeHandle
* passed_handle
=
77 new mojo::ScopedMessagePipeHandle(handle
.Pass());
78 gin::ModuleRegistry::From(context_holder
->context())->LoadModule(
79 context_holder
->isolate(),
81 base::Bind(RunMain
, runner_
->GetWeakPtr(), base::Owned(passed_handle
)));
84 void WebUIMojoContextState::FetchModules(const std::vector
<std::string
>& ids
) {
85 gin::Runner::Scope
scoper(runner_
.get());
86 gin::ContextHolder
* context_holder
= runner_
->GetContextHolder();
87 gin::ModuleRegistry
* registry
= gin::ModuleRegistry::From(
88 context_holder
->context());
89 for (size_t i
= 0; i
< ids
.size(); ++i
) {
90 if (fetched_modules_
.find(ids
[i
]) == fetched_modules_
.end() &&
91 registry
->available_modules().count(ids
[i
]) == 0) {
97 void WebUIMojoContextState::FetchModule(const std::string
& id
) {
98 const GURL
url(kModulePrefix
+ id
);
99 // TODO(sky): better error checks here?
100 DCHECK(url
.is_valid() && !url
.is_empty());
101 DCHECK(fetched_modules_
.find(id
) == fetched_modules_
.end());
102 fetched_modules_
.insert(id
);
103 ResourceFetcher
* fetcher
= ResourceFetcher::Create(url
);
104 module_fetchers_
.push_back(fetcher
);
105 fetcher
->Start(frame_
, blink::WebURLRequest::TargetIsScript
,
106 base::Bind(&WebUIMojoContextState::OnFetchModuleComplete
,
107 base::Unretained(this), fetcher
));
110 void WebUIMojoContextState::OnFetchModuleComplete(
111 ResourceFetcher
* fetcher
,
112 const blink::WebURLResponse
& response
,
113 const std::string
& data
) {
114 DCHECK_EQ(kModulePrefix
,
115 response
.url().string().utf8().substr(0, arraysize(kModulePrefix
) - 1));
116 const std::string module
=
117 response
.url().string().utf8().substr(arraysize(kModulePrefix
) - 1);
118 // We can't delete fetch right now as the arguments to this function come from
119 // it and are used below. Instead use a scope_ptr to cleanup.
120 scoped_ptr
<ResourceFetcher
> deleter(fetcher
);
121 module_fetchers_
.weak_erase(
122 std::find(module_fetchers_
.begin(), module_fetchers_
.end(), fetcher
));
125 return; // TODO(sky): log something?
128 runner_
->Run(data
, module
);
131 void WebUIMojoContextState::OnDidAddPendingModule(
132 const std::string
& id
,
133 const std::vector
<std::string
>& dependencies
) {
134 FetchModules(dependencies
);
136 gin::ContextHolder
* context_holder
= runner_
->GetContextHolder();
137 gin::ModuleRegistry
* registry
= gin::ModuleRegistry::From(
138 context_holder
->context());
139 registry
->AttemptToLoadMoreModules(context_holder
->isolate());
142 } // namespace content