| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 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 "chrome/renderer/extensions/chrome_v8_context.h" | 5 #include "chrome/renderer/extensions/chrome_v8_context.h" |
| 6 | 6 |
| 7 #include "base/debug/trace_event.h" | |
| 8 #include "base/logging.h" | 7 #include "base/logging.h" |
| 9 #include "base/memory/scoped_ptr.h" | 8 #include "base/memory/scoped_ptr.h" |
| 10 #include "base/strings/string_split.h" | 9 #include "base/strings/string_split.h" |
| 11 #include "base/values.h" | 10 #include "base/values.h" |
| 12 #include "chrome/common/extensions/api/extension_api.h" | 11 #include "chrome/common/extensions/api/extension_api.h" |
| 13 #include "chrome/common/extensions/extension.h" | 12 #include "chrome/common/extensions/extension.h" |
| 14 #include "chrome/common/extensions/extension_set.h" | 13 #include "chrome/common/extensions/extension_set.h" |
| 15 #include "chrome/common/extensions/features/base_feature_provider.h" | 14 #include "chrome/common/extensions/features/base_feature_provider.h" |
| 16 #include "chrome/renderer/extensions/chrome_v8_extension.h" | 15 #include "chrome/renderer/extensions/chrome_v8_extension.h" |
| 17 #include "chrome/renderer/extensions/module_system.h" | 16 #include "chrome/renderer/extensions/module_system.h" |
| 18 #include "chrome/renderer/extensions/user_script_slave.h" | 17 #include "chrome/renderer/extensions/user_script_slave.h" |
| 19 #include "content/public/renderer/render_view.h" | 18 #include "content/public/renderer/render_view.h" |
| 20 #include "content/public/renderer/v8_value_converter.h" | 19 #include "content/public/renderer/v8_value_converter.h" |
| 21 #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" | 20 #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" |
| 21 #include "third_party/WebKit/Source/WebKit/chromium/public/WebScopedMicrotaskSup
pression.h" |
| 22 #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" | 22 #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" |
| 23 #include "v8/include/v8.h" | 23 #include "v8/include/v8.h" |
| 24 | 24 |
| 25 using content::V8ValueConverter; | 25 using content::V8ValueConverter; |
| 26 | 26 |
| 27 namespace extensions { | 27 namespace extensions { |
| 28 | 28 |
| 29 namespace { | 29 namespace { |
| 30 | |
| 31 const char kChromeHidden[] = "chromeHidden"; | 30 const char kChromeHidden[] = "chromeHidden"; |
| 32 const char kValidateCallbacks[] = "validateCallbacks"; | |
| 33 const char kValidateAPI[] = "validateAPI"; | |
| 34 | |
| 35 } // namespace | 31 } // namespace |
| 36 | 32 |
| 37 ChromeV8Context::ChromeV8Context(v8::Handle<v8::Context> v8_context, | 33 ChromeV8Context::ChromeV8Context(v8::Handle<v8::Context> v8_context, |
| 38 WebKit::WebFrame* web_frame, | 34 WebKit::WebFrame* web_frame, |
| 39 const Extension* extension, | 35 const Extension* extension, |
| 40 Feature::Context context_type) | 36 Feature::Context context_type) |
| 41 : v8_context_(v8_context), | 37 : v8_context_(v8_context), |
| 42 web_frame_(web_frame), | 38 web_frame_(web_frame), |
| 43 extension_(extension), | 39 extension_(extension), |
| 44 context_type_(context_type) { | 40 context_type_(context_type) { |
| 45 VLOG(1) << "Created context:\n" | 41 VLOG(1) << "Created context:\n" |
| 46 << " extension id: " << GetExtensionID() << "\n" | 42 << " extension id: " << GetExtensionID() << "\n" |
| 47 << " frame: " << web_frame_ << "\n" | 43 << " frame: " << web_frame_ << "\n" |
| 48 << " context type: " << GetContextTypeDescription(); | 44 << " context type: " << GetContextTypeDescription(); |
| 49 } | 45 } |
| 50 | 46 |
| 51 ChromeV8Context::~ChromeV8Context() { | 47 ChromeV8Context::~ChromeV8Context() { |
| 52 VLOG(1) << "Destroyed context for extension\n" | 48 VLOG(1) << "Destroyed context for extension\n" |
| 53 << " extension id: " << GetExtensionID(); | 49 << " extension id: " << GetExtensionID(); |
| 54 Invalidate(); | 50 Invalidate(); |
| 55 } | 51 } |
| 56 | 52 |
| 57 void ChromeV8Context::Invalidate() { | 53 void ChromeV8Context::Invalidate() { |
| 58 if (v8_context_.get().IsEmpty()) | 54 if (!is_valid()) |
| 59 return; | 55 return; |
| 60 if (module_system_) | 56 if (module_system_) |
| 61 module_system_->Invalidate(); | 57 module_system_->Invalidate(); |
| 62 web_frame_ = NULL; | 58 web_frame_ = NULL; |
| 63 v8_context_.reset(); | 59 v8_context_.reset(); |
| 64 } | 60 } |
| 65 | 61 |
| 66 std::string ChromeV8Context::GetExtensionID() { | 62 std::string ChromeV8Context::GetExtensionID() { |
| 67 return extension_ ? extension_->id() : std::string(); | 63 return extension_ ? extension_->id() : std::string(); |
| 68 } | 64 } |
| 69 | 65 |
| 70 // static | 66 // static |
| 71 v8::Handle<v8::Value> ChromeV8Context::GetOrCreateChromeHidden( | 67 v8::Handle<v8::Value> ChromeV8Context::GetOrCreateChromeHidden( |
| 72 v8::Handle<v8::Context> context) { | 68 v8::Handle<v8::Context> context) { |
| 73 v8::Local<v8::Object> global = context->Global(); | 69 v8::Local<v8::Object> global = context->Global(); |
| 74 v8::Local<v8::Value> hidden = global->GetHiddenValue( | 70 v8::Local<v8::Value> hidden = global->GetHiddenValue( |
| 75 v8::String::New(kChromeHidden)); | 71 v8::String::New(kChromeHidden)); |
| 76 | 72 |
| 77 if (hidden.IsEmpty() || hidden->IsUndefined()) { | 73 if (hidden.IsEmpty() || hidden->IsUndefined()) { |
| 78 hidden = v8::Object::New(); | 74 hidden = v8::Object::New(); |
| 79 global->SetHiddenValue(v8::String::New(kChromeHidden), hidden); | 75 global->SetHiddenValue(v8::String::New(kChromeHidden), hidden); |
| 80 | |
| 81 if (DCHECK_IS_ON()) { | |
| 82 // Tell bindings.js to validate callbacks and events against their schema | |
| 83 // definitions. | |
| 84 v8::Local<v8::Object>::Cast(hidden)->Set( | |
| 85 v8::String::New(kValidateCallbacks), v8::True()); | |
| 86 // Tell bindings.js to validate API for ambiguity. | |
| 87 v8::Local<v8::Object>::Cast(hidden)->Set( | |
| 88 v8::String::New(kValidateAPI), v8::True()); | |
| 89 } | |
| 90 } | 76 } |
| 91 | 77 |
| 92 DCHECK(hidden->IsObject()); | 78 DCHECK(hidden->IsObject()); |
| 93 return v8::Local<v8::Object>::Cast(hidden); | 79 return v8::Local<v8::Object>::Cast(hidden); |
| 94 } | 80 } |
| 95 | 81 |
| 96 v8::Handle<v8::Value> ChromeV8Context::GetChromeHidden() const { | 82 v8::Handle<v8::Value> ChromeV8Context::GetChromeHidden() const { |
| 97 v8::Local<v8::Object> global = v8_context_->Global(); | 83 v8::Local<v8::Object> global = v8_context_->Global(); |
| 98 return global->GetHiddenValue(v8::String::New(kChromeHidden)); | 84 return global->GetHiddenValue(v8::String::New(kChromeHidden)); |
| 99 } | 85 } |
| 100 | 86 |
| 101 content::RenderView* ChromeV8Context::GetRenderView() const { | 87 content::RenderView* ChromeV8Context::GetRenderView() const { |
| 102 if (web_frame_ && web_frame_->view()) | 88 if (web_frame_ && web_frame_->view()) |
| 103 return content::RenderView::FromWebView(web_frame_->view()); | 89 return content::RenderView::FromWebView(web_frame_->view()); |
| 104 else | 90 else |
| 105 return NULL; | 91 return NULL; |
| 106 } | 92 } |
| 107 | 93 |
| 108 bool ChromeV8Context::CallChromeHiddenMethod( | 94 v8::Local<v8::Value> ChromeV8Context::CallFunction( |
| 109 const std::string& function_name, | 95 v8::Handle<v8::Function> function, |
| 110 int argc, | 96 int argc, |
| 111 v8::Handle<v8::Value>* argv, | 97 v8::Handle<v8::Value> argv[]) const { |
| 112 v8::Handle<v8::Value>* result) const { | 98 v8::HandleScope handle_scope; |
| 113 // ChromeV8ContextSet calls Invalidate() and then schedules a task to delete | 99 WebKit::WebScopedMicrotaskSuppression suppression; |
| 114 // this object. This check prevents a race from attempting to execute script | 100 if (!is_valid()) |
| 115 // on a NULL web_frame_. | 101 return handle_scope.Close(v8::Undefined()); |
| 102 |
| 103 v8::Handle<v8::Object> global = v8_context()->Global(); |
| 116 if (!web_frame_) | 104 if (!web_frame_) |
| 117 return false; | 105 return handle_scope.Close(function->Call(global, argc, argv)); |
| 118 | 106 return handle_scope.Close( |
| 119 v8::Context::Scope context_scope(v8_context_.get()); | |
| 120 | |
| 121 // Look up the function name, which may be a sub-property like | |
| 122 // "Port.dispatchOnMessage" in the hidden global variable. | |
| 123 v8::Local<v8::Value> value = v8::Local<v8::Value>::New(GetChromeHidden()); | |
| 124 if (value.IsEmpty()) | |
| 125 return false; | |
| 126 | |
| 127 std::vector<std::string> components; | |
| 128 base::SplitStringDontTrim(function_name, '.', &components); | |
| 129 for (size_t i = 0; i < components.size(); ++i) { | |
| 130 if (!value.IsEmpty() && value->IsObject()) { | |
| 131 value = v8::Local<v8::Object>::Cast(value)->Get( | |
| 132 v8::String::New(components[i].c_str())); | |
| 133 } | |
| 134 } | |
| 135 | |
| 136 if (value.IsEmpty() || !value->IsFunction()) { | |
| 137 VLOG(1) << "Could not execute chrome hidden method: " << function_name; | |
| 138 return false; | |
| 139 } | |
| 140 | |
| 141 TRACE_EVENT1("v8", "v8.callChromeHiddenMethod", | |
| 142 "function_name", function_name); | |
| 143 | |
| 144 v8::Local<v8::Function> function = v8::Local<v8::Function>::Cast(value); | |
| 145 v8::Handle<v8::Value> result_temp = | |
| 146 web_frame_->callFunctionEvenIfScriptDisabled(function, | 107 web_frame_->callFunctionEvenIfScriptDisabled(function, |
| 147 v8::Object::New(), | 108 global, |
| 148 argc, | 109 argc, |
| 149 argv); | 110 argv)); |
| 150 if (result) | |
| 151 *result = result_temp; | |
| 152 | |
| 153 return true; | |
| 154 } | 111 } |
| 155 | 112 |
| 156 bool ChromeV8Context::IsAnyFeatureAvailableToContext( | 113 bool ChromeV8Context::IsAnyFeatureAvailableToContext( |
| 157 const std::string& api_name) { | 114 const std::string& api_name) { |
| 158 return ExtensionAPI::GetSharedInstance()->IsAnyFeatureAvailableToContext( | 115 return ExtensionAPI::GetSharedInstance()->IsAnyFeatureAvailableToContext( |
| 159 api_name, | 116 api_name, |
| 160 context_type_, | 117 context_type_, |
| 161 UserScriptSlave::GetDataSourceURLForFrame(web_frame_)); | 118 UserScriptSlave::GetDataSourceURLForFrame(web_frame_)); |
| 162 } | 119 } |
| 163 | 120 |
| 164 Feature::Availability ChromeV8Context::GetAvailability( | 121 Feature::Availability ChromeV8Context::GetAvailability( |
| 165 const std::string& api_name) { | 122 const std::string& api_name) { |
| 166 return ExtensionAPI::GetSharedInstance()->IsAvailable( | 123 return ExtensionAPI::GetSharedInstance()->IsAvailable( |
| 167 api_name, | 124 api_name, |
| 168 extension_, | 125 extension_, |
| 169 context_type_, | 126 context_type_, |
| 170 UserScriptSlave::GetDataSourceURLForFrame(web_frame_)); | 127 UserScriptSlave::GetDataSourceURLForFrame(web_frame_)); |
| 171 } | 128 } |
| 172 | 129 |
| 173 void ChromeV8Context::DispatchOnUnloadEvent() { | 130 void ChromeV8Context::DispatchOnUnloadEvent() { |
| 174 v8::HandleScope handle_scope; | 131 module_system_->CallModuleMethod("on_unload", "dispatch"); |
| 175 CallChromeHiddenMethod("dispatchOnUnload", 0, NULL, NULL); | |
| 176 } | 132 } |
| 177 | 133 |
| 178 std::string ChromeV8Context::GetContextTypeDescription() { | 134 std::string ChromeV8Context::GetContextTypeDescription() { |
| 179 switch (context_type_) { | 135 switch (context_type_) { |
| 180 case Feature::UNSPECIFIED_CONTEXT: return "UNSPECIFIED"; | 136 case Feature::UNSPECIFIED_CONTEXT: return "UNSPECIFIED"; |
| 181 case Feature::BLESSED_EXTENSION_CONTEXT: return "BLESSED_EXTENSION"; | 137 case Feature::BLESSED_EXTENSION_CONTEXT: return "BLESSED_EXTENSION"; |
| 182 case Feature::UNBLESSED_EXTENSION_CONTEXT: return "UNBLESSED_EXTENSION"; | 138 case Feature::UNBLESSED_EXTENSION_CONTEXT: return "UNBLESSED_EXTENSION"; |
| 183 case Feature::CONTENT_SCRIPT_CONTEXT: return "CONTENT_SCRIPT"; | 139 case Feature::CONTENT_SCRIPT_CONTEXT: return "CONTENT_SCRIPT"; |
| 184 case Feature::WEB_PAGE_CONTEXT: return "WEB_PAGE"; | 140 case Feature::WEB_PAGE_CONTEXT: return "WEB_PAGE"; |
| 185 } | 141 } |
| (...skipping 14 matching lines...) Expand all Loading... |
| 200 | 156 |
| 201 scoped_ptr<V8ValueConverter> converter(V8ValueConverter::create()); | 157 scoped_ptr<V8ValueConverter> converter(V8ValueConverter::create()); |
| 202 v8::Handle<v8::Value> argv[] = { | 158 v8::Handle<v8::Value> argv[] = { |
| 203 v8::Integer::New(request_id), | 159 v8::Integer::New(request_id), |
| 204 v8::String::New(name.c_str()), | 160 v8::String::New(name.c_str()), |
| 205 v8::Boolean::New(success), | 161 v8::Boolean::New(success), |
| 206 converter->ToV8Value(&response, v8_context_.get()), | 162 converter->ToV8Value(&response, v8_context_.get()), |
| 207 v8::String::New(error.c_str()) | 163 v8::String::New(error.c_str()) |
| 208 }; | 164 }; |
| 209 | 165 |
| 210 v8::Handle<v8::Value> retval; | 166 v8::Handle<v8::Value> retval = module_system_->CallModuleMethod( |
| 211 CHECK(CallChromeHiddenMethod("handleResponse", arraysize(argv), argv, | 167 "sendRequest", "handleResponse", arraysize(argv), argv); |
| 212 &retval)); | 168 |
| 213 // In debug, the js will validate the callback parameters and return a | 169 // In debug, the js will validate the callback parameters and return a |
| 214 // string if a validation error has occured. | 170 // string if a validation error has occured. |
| 215 if (DCHECK_IS_ON()) { | 171 if (DCHECK_IS_ON()) { |
| 216 if (!retval.IsEmpty() && !retval->IsUndefined()) { | 172 if (!retval.IsEmpty() && !retval->IsUndefined()) { |
| 217 std::string error = *v8::String::AsciiValue(retval); | 173 std::string error = *v8::String::AsciiValue(retval); |
| 218 DCHECK(false) << error; | 174 DCHECK(false) << error; |
| 219 } | 175 } |
| 220 } | 176 } |
| 221 } | 177 } |
| 222 | 178 |
| 179 bool ChromeV8Context::is_valid() const { |
| 180 return !v8_context_.get().IsEmpty(); |
| 181 } |
| 182 |
| 223 } // namespace extensions | 183 } // namespace extensions |
| OLD | NEW |