// Copyright 2015 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "src/inspector/v8-injected-script-host.h" #include "src/base/macros.h" #include "src/inspector/injected-script-native.h" #include "src/inspector/string-util.h" #include "src/inspector/v8-debugger.h" #include "src/inspector/v8-inspector-impl.h" #include "src/inspector/v8-internal-value-type.h" #include "src/inspector/v8-value-copier.h" #include "include/v8-inspector.h" namespace v8_inspector { namespace { void setFunctionProperty(v8::Local<v8::Context> context, v8::Local<v8::Object> obj, const char* name, v8::FunctionCallback callback, v8::Local<v8::External> external) { v8::Local<v8::String> funcName = toV8StringInternalized(context->GetIsolate(), name); v8::Local<v8::Function> func; if (!v8::Function::New(context, callback, external, 0, v8::ConstructorBehavior::kThrow) .ToLocal(&func)) return; func->SetName(funcName); createDataProperty(context, obj, funcName, func); } V8InspectorImpl* unwrapInspector( const v8::FunctionCallbackInfo<v8::Value>& info) { DCHECK(!info.Data().IsEmpty()); DCHECK(info.Data()->IsExternal()); V8InspectorImpl* inspector = static_cast<V8InspectorImpl*>(info.Data().As<v8::External>()->Value()); DCHECK(inspector); return inspector; } } // namespace v8::Local<v8::Object> V8InjectedScriptHost::create( v8::Local<v8::Context> context, V8InspectorImpl* inspector) { v8::Isolate* isolate = inspector->isolate(); v8::Local<v8::Object> injectedScriptHost = v8::Object::New(isolate); bool success = injectedScriptHost->SetPrototype(context, v8::Null(isolate)) .FromMaybe(false); DCHECK(success); USE(success); v8::Local<v8::External> debuggerExternal = v8::External::New(isolate, inspector); setFunctionProperty(context, injectedScriptHost, "nullifyPrototype", V8InjectedScriptHost::nullifyPrototypeCallback, debuggerExternal); setFunctionProperty(context, injectedScriptHost, "internalConstructorName", V8InjectedScriptHost::internalConstructorNameCallback, debuggerExternal); setFunctionProperty( context, injectedScriptHost, "formatAccessorsAsProperties", V8InjectedScriptHost::formatAccessorsAsProperties, debuggerExternal); setFunctionProperty(context, injectedScriptHost, "subtype", V8InjectedScriptHost::subtypeCallback, debuggerExternal); setFunctionProperty(context, injectedScriptHost, "getInternalProperties", V8InjectedScriptHost::getInternalPropertiesCallback, debuggerExternal); setFunctionProperty(context, injectedScriptHost, "objectHasOwnProperty", V8InjectedScriptHost::objectHasOwnPropertyCallback, debuggerExternal); setFunctionProperty(context, injectedScriptHost, "bind", V8InjectedScriptHost::bindCallback, debuggerExternal); setFunctionProperty(context, injectedScriptHost, "proxyTargetValue", V8InjectedScriptHost::proxyTargetValueCallback, debuggerExternal); return injectedScriptHost; } void V8InjectedScriptHost::nullifyPrototypeCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { CHECK(info.Length() == 1 && info[0]->IsObject()); v8::Isolate* isolate = info.GetIsolate(); info[0] .As<v8::Object>() ->SetPrototype(isolate->GetCurrentContext(), v8::Null(isolate)) .ToChecked(); } void V8InjectedScriptHost::internalConstructorNameCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { if (info.Length() < 1 || !info[0]->IsObject()) return; v8::Local<v8::Object> object = info[0].As<v8::Object>(); info.GetReturnValue().Set(object->GetConstructorName()); } void V8InjectedScriptHost::formatAccessorsAsProperties( const v8::FunctionCallbackInfo<v8::Value>& info) { DCHECK_EQ(info.Length(), 2); info.GetReturnValue().Set(false); if (!info[1]->IsFunction()) return; // Check that function is user-defined. if (info[1].As<v8::Function>()->ScriptId() != v8::UnboundScript::kNoScriptId) return; info.GetReturnValue().Set( unwrapInspector(info)->client()->formatAccessorsAsProperties(info[0])); } void V8InjectedScriptHost::subtypeCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { if (info.Length() < 1) return; v8::Isolate* isolate = info.GetIsolate(); v8::Local<v8::Value> value = info[0]; if (value->IsObject()) { v8::Local<v8::Value> internalType = v8InternalValueTypeFrom( isolate->GetCurrentContext(), v8::Local<v8::Object>::Cast(value)); if (internalType->IsString()) { info.GetReturnValue().Set(internalType); return; } } if (value->IsArray() || value->IsArgumentsObject()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "array")); return; } if (value->IsTypedArray()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "typedarray")); return; } if (value->IsDate()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "date")); return; } if (value->IsRegExp()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "regexp")); return; } if (value->IsMap() || value->IsWeakMap()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "map")); return; } if (value->IsSet() || value->IsWeakSet()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "set")); return; } if (value->IsMapIterator() || value->IsSetIterator()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "iterator")); return; } if (value->IsGeneratorObject()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "generator")); return; } if (value->IsNativeError()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "error")); return; } if (value->IsProxy()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "proxy")); return; } if (value->IsPromise()) { info.GetReturnValue().Set(toV8StringInternalized(isolate, "promise")); return; } std::unique_ptr<StringBuffer> subtype = unwrapInspector(info)->client()->valueSubtype(value); if (subtype) { info.GetReturnValue().Set(toV8String(isolate, subtype->string())); return; } } void V8InjectedScriptHost::getInternalPropertiesCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { if (info.Length() < 1) return; std::unordered_set<String16> allowedProperties; if (info[0]->IsBooleanObject() || info[0]->IsNumberObject() || info[0]->IsStringObject() || info[0]->IsSymbolObject()) { allowedProperties.insert(String16("[[PrimitiveValue]]")); } else if (info[0]->IsPromise()) { allowedProperties.insert(String16("[[PromiseStatus]]")); allowedProperties.insert(String16("[[PromiseValue]]")); } else if (info[0]->IsGeneratorObject()) { allowedProperties.insert(String16("[[GeneratorStatus]]")); } else if (info[0]->IsMapIterator() || info[0]->IsSetIterator()) { allowedProperties.insert(String16("[[IteratorHasMore]]")); allowedProperties.insert(String16("[[IteratorIndex]]")); allowedProperties.insert(String16("[[IteratorKind]]")); allowedProperties.insert(String16("[[Entries]]")); } else if (info[0]->IsMap() || info[0]->IsWeakMap() || info[0]->IsSet() || info[0]->IsWeakSet()) { allowedProperties.insert(String16("[[Entries]]")); } if (!allowedProperties.size()) return; v8::Isolate* isolate = info.GetIsolate(); v8::Local<v8::Array> allProperties; if (!unwrapInspector(info) ->debugger() ->internalProperties(isolate->GetCurrentContext(), info[0]) .ToLocal(&allProperties) || !allProperties->IsArray() || allProperties->Length() % 2 != 0) return; { v8::Local<v8::Context> context = isolate->GetCurrentContext(); v8::TryCatch tryCatch(isolate); v8::Isolate::DisallowJavascriptExecutionScope throwJs( isolate, v8::Isolate::DisallowJavascriptExecutionScope::THROW_ON_FAILURE); v8::Local<v8::Array> properties = v8::Array::New(isolate); if (tryCatch.HasCaught()) return; uint32_t outputIndex = 0; for (uint32_t i = 0; i < allProperties->Length(); i += 2) { v8::Local<v8::Value> key; if (!allProperties->Get(context, i).ToLocal(&key)) continue; if (tryCatch.HasCaught()) { tryCatch.Reset(); continue; } String16 keyString = toProtocolStringWithTypeCheck(key); if (keyString.isEmpty() || allowedProperties.find(keyString) == allowedProperties.end()) continue; v8::Local<v8::Value> value; if (!allProperties->Get(context, i + 1).ToLocal(&value)) continue; if (tryCatch.HasCaught()) { tryCatch.Reset(); continue; } createDataProperty(context, properties, outputIndex++, key); createDataProperty(context, properties, outputIndex++, value); } info.GetReturnValue().Set(properties); } } void V8InjectedScriptHost::objectHasOwnPropertyCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { if (info.Length() < 2 || !info[0]->IsObject() || !info[1]->IsString()) return; bool result = info[0] .As<v8::Object>() ->HasOwnProperty(info.GetIsolate()->GetCurrentContext(), v8::Local<v8::String>::Cast(info[1])) .FromMaybe(false); info.GetReturnValue().Set(v8::Boolean::New(info.GetIsolate(), result)); } void V8InjectedScriptHost::bindCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { if (info.Length() < 2 || !info[1]->IsString()) return; InjectedScriptNative* injectedScriptNative = InjectedScriptNative::fromInjectedScriptHost(info.GetIsolate(), info.Holder()); if (!injectedScriptNative) return; v8::Local<v8::Context> context = info.GetIsolate()->GetCurrentContext(); v8::Local<v8::String> v8groupName = info[1]->ToString(context).ToLocalChecked(); String16 groupName = toProtocolStringWithTypeCheck(v8groupName); int id = injectedScriptNative->bind(info[0], groupName); info.GetReturnValue().Set(id); } void V8InjectedScriptHost::proxyTargetValueCallback( const v8::FunctionCallbackInfo<v8::Value>& info) { if (info.Length() != 1 || !info[0]->IsProxy()) { UNREACHABLE(); return; } v8::Local<v8::Object> target = info[0].As<v8::Proxy>(); while (target->IsProxy()) target = v8::Local<v8::Proxy>::Cast(target)->GetTarget(); info.GetReturnValue().Set(target); } } // namespace v8_inspector