/* * Copyright (C) 2011 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "src/inspector/v8-runtime-agent-impl.h" #include #include "src/debug/debug-interface.h" #include "src/inspector/injected-script.h" #include "src/inspector/inspected-context.h" #include "src/inspector/protocol/Protocol.h" #include "src/inspector/remote-object-id.h" #include "src/inspector/v8-console-message.h" #include "src/inspector/v8-debugger-agent-impl.h" #include "src/inspector/v8-debugger.h" #include "src/inspector/v8-inspector-impl.h" #include "src/inspector/v8-inspector-session-impl.h" #include "src/inspector/v8-stack-trace-impl.h" #include "src/inspector/v8-value-utils.h" #include "src/tracing/trace-event.h" #include "include/v8-inspector.h" namespace v8_inspector { namespace V8RuntimeAgentImplState { static const char customObjectFormatterEnabled[] = "customObjectFormatterEnabled"; static const char runtimeEnabled[] = "runtimeEnabled"; static const char bindings[] = "bindings"; }; using protocol::Runtime::RemoteObject; namespace { template class EvaluateCallbackWrapper : public EvaluateCallback { public: static std::unique_ptr wrap( std::unique_ptr callback) { return std::unique_ptr( new EvaluateCallbackWrapper(std::move(callback))); } void sendSuccess(std::unique_ptr result, protocol::Maybe exceptionDetails) override { return m_callback->sendSuccess(std::move(result), std::move(exceptionDetails)); } void sendFailure(const protocol::DispatchResponse& response) override { return m_callback->sendFailure(response); } private: explicit EvaluateCallbackWrapper(std::unique_ptr callback) : m_callback(std::move(callback)) {} std::unique_ptr m_callback; }; template bool wrapEvaluateResultAsync(InjectedScript* injectedScript, v8::MaybeLocal maybeResultValue, const v8::TryCatch& tryCatch, const String16& objectGroup, bool returnByValue, bool generatePreview, ProtocolCallback* callback) { std::unique_ptr result; Maybe exceptionDetails; Response response = injectedScript->wrapEvaluateResult( maybeResultValue, tryCatch, objectGroup, returnByValue, generatePreview, &result, &exceptionDetails); if (response.isSuccess()) { callback->sendSuccess(std::move(result), std::move(exceptionDetails)); return true; } callback->sendFailure(response); return false; } void innerCallFunctionOn( V8InspectorSessionImpl* session, InjectedScript::Scope& scope, v8::Local recv, const String16& expression, Maybe> optionalArguments, bool silent, bool returnByValue, bool generatePreview, bool userGesture, bool awaitPromise, const String16& objectGroup, std::unique_ptr callback) { V8InspectorImpl* inspector = session->inspector(); std::unique_ptr[]> argv = nullptr; int argc = 0; if (optionalArguments.isJust()) { protocol::Array* arguments = optionalArguments.fromJust(); argc = static_cast(arguments->length()); argv.reset(new v8::Local[argc]); for (int i = 0; i < argc; ++i) { v8::Local argumentValue; Response response = scope.injectedScript()->resolveCallArgument( arguments->get(i), &argumentValue); if (!response.isSuccess()) { callback->sendFailure(response); return; } argv[i] = argumentValue; } } if (silent) scope.ignoreExceptionsAndMuteConsole(); if (userGesture) scope.pretendUserGesture(); // Temporarily enable allow evals for inspector. scope.allowCodeGenerationFromStrings(); v8::MaybeLocal maybeFunctionValue; v8::Local functionScript; if (inspector ->compileScript(scope.context(), "(" + expression + ")", String16()) .ToLocal(&functionScript)) { v8::MicrotasksScope microtasksScope(inspector->isolate(), v8::MicrotasksScope::kRunMicrotasks); maybeFunctionValue = functionScript->Run(scope.context()); } // Re-initialize after running client's code, as it could have destroyed // context or session. Response response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (scope.tryCatch().HasCaught()) { wrapEvaluateResultAsync(scope.injectedScript(), maybeFunctionValue, scope.tryCatch(), objectGroup, false, false, callback.get()); return; } v8::Local functionValue; if (!maybeFunctionValue.ToLocal(&functionValue) || !functionValue->IsFunction()) { callback->sendFailure( Response::Error("Given expression does not evaluate to a function")); return; } v8::MaybeLocal maybeResultValue; { v8::MicrotasksScope microtasksScope(inspector->isolate(), v8::MicrotasksScope::kRunMicrotasks); maybeResultValue = functionValue.As()->Call( scope.context(), recv, argc, argv.get()); } // Re-initialize after running client's code, as it could have destroyed // context or session. response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (!awaitPromise || scope.tryCatch().HasCaught()) { wrapEvaluateResultAsync(scope.injectedScript(), maybeResultValue, scope.tryCatch(), objectGroup, returnByValue, generatePreview, callback.get()); return; } scope.injectedScript()->addPromiseCallback( session, maybeResultValue, objectGroup, returnByValue, generatePreview, EvaluateCallbackWrapper::wrap( std::move(callback))); } Response ensureContext(V8InspectorImpl* inspector, int contextGroupId, Maybe executionContextId, int* contextId) { if (executionContextId.isJust()) { *contextId = executionContextId.fromJust(); } else { v8::HandleScope handles(inspector->isolate()); v8::Local defaultContext = inspector->client()->ensureDefaultContextInGroup(contextGroupId); if (defaultContext.IsEmpty()) return Response::Error("Cannot find default execution context"); *contextId = InspectedContext::contextId(defaultContext); } return Response::OK(); } } // namespace V8RuntimeAgentImpl::V8RuntimeAgentImpl( V8InspectorSessionImpl* session, protocol::FrontendChannel* FrontendChannel, protocol::DictionaryValue* state) : m_session(session), m_state(state), m_frontend(FrontendChannel), m_inspector(session->inspector()), m_enabled(false) {} V8RuntimeAgentImpl::~V8RuntimeAgentImpl() {} void V8RuntimeAgentImpl::evaluate( const String16& expression, Maybe objectGroup, Maybe includeCommandLineAPI, Maybe silent, Maybe executionContextId, Maybe returnByValue, Maybe generatePreview, Maybe userGesture, Maybe awaitPromise, Maybe throwOnSideEffect, Maybe timeout, std::unique_ptr callback) { TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"), "EvaluateScript"); int contextId = 0; Response response = ensureContext(m_inspector, m_session->contextGroupId(), std::move(executionContextId), &contextId); if (!response.isSuccess()) { callback->sendFailure(response); return; } InjectedScript::ContextScope scope(m_session, contextId); response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (silent.fromMaybe(false)) scope.ignoreExceptionsAndMuteConsole(); if (userGesture.fromMaybe(false)) scope.pretendUserGesture(); if (includeCommandLineAPI.fromMaybe(false)) scope.installCommandLineAPI(); // Temporarily enable allow evals for inspector. scope.allowCodeGenerationFromStrings(); v8::MaybeLocal maybeResultValue; { V8InspectorImpl::EvaluateScope evaluateScope(m_inspector->isolate()); if (timeout.isJust()) { response = evaluateScope.setTimeout(timeout.fromJust() / 1000.0); if (!response.isSuccess()) { callback->sendFailure(response); return; } } v8::MicrotasksScope microtasksScope(m_inspector->isolate(), v8::MicrotasksScope::kRunMicrotasks); maybeResultValue = v8::debug::EvaluateGlobal( m_inspector->isolate(), toV8String(m_inspector->isolate(), expression), throwOnSideEffect.fromMaybe(false)); } // Run microtasks before returning result. // Re-initialize after running client's code, as it could have destroyed // context or session. response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (!awaitPromise.fromMaybe(false) || scope.tryCatch().HasCaught()) { wrapEvaluateResultAsync(scope.injectedScript(), maybeResultValue, scope.tryCatch(), objectGroup.fromMaybe(""), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), callback.get()); return; } scope.injectedScript()->addPromiseCallback( m_session, maybeResultValue, objectGroup.fromMaybe(""), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), EvaluateCallbackWrapper::wrap(std::move(callback))); } void V8RuntimeAgentImpl::awaitPromise( const String16& promiseObjectId, Maybe returnByValue, Maybe generatePreview, std::unique_ptr callback) { InjectedScript::ObjectScope scope(m_session, promiseObjectId); Response response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (!scope.object()->IsPromise()) { callback->sendFailure( Response::Error("Could not find promise with given id")); return; } scope.injectedScript()->addPromiseCallback( m_session, scope.object(), scope.objectGroupName(), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), EvaluateCallbackWrapper::wrap(std::move(callback))); } void V8RuntimeAgentImpl::callFunctionOn( const String16& expression, Maybe objectId, Maybe> optionalArguments, Maybe silent, Maybe returnByValue, Maybe generatePreview, Maybe userGesture, Maybe awaitPromise, Maybe executionContextId, Maybe objectGroup, std::unique_ptr callback) { if (objectId.isJust() && executionContextId.isJust()) { callback->sendFailure(Response::Error( "ObjectId must not be specified together with executionContextId")); return; } if (!objectId.isJust() && !executionContextId.isJust()) { callback->sendFailure(Response::Error( "Either ObjectId or executionContextId must be specified")); return; } if (objectId.isJust()) { InjectedScript::ObjectScope scope(m_session, objectId.fromJust()); Response response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } innerCallFunctionOn( m_session, scope, scope.object(), expression, std::move(optionalArguments), silent.fromMaybe(false), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), userGesture.fromMaybe(false), awaitPromise.fromMaybe(false), objectGroup.isJust() ? objectGroup.fromMaybe(String16()) : scope.objectGroupName(), std::move(callback)); } else { int contextId = 0; Response response = ensureContext(m_inspector, m_session->contextGroupId(), std::move(executionContextId.fromJust()), &contextId); if (!response.isSuccess()) { callback->sendFailure(response); return; } InjectedScript::ContextScope scope(m_session, contextId); response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } innerCallFunctionOn( m_session, scope, scope.context()->Global(), expression, std::move(optionalArguments), silent.fromMaybe(false), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), userGesture.fromMaybe(false), awaitPromise.fromMaybe(false), objectGroup.fromMaybe(""), std::move(callback)); } } Response V8RuntimeAgentImpl::getProperties( const String16& objectId, Maybe ownProperties, Maybe accessorPropertiesOnly, Maybe generatePreview, std::unique_ptr>* result, Maybe>* internalProperties, Maybe* exceptionDetails) { using protocol::Runtime::InternalPropertyDescriptor; InjectedScript::ObjectScope scope(m_session, objectId); Response response = scope.initialize(); if (!response.isSuccess()) return response; scope.ignoreExceptionsAndMuteConsole(); v8::MicrotasksScope microtasks_scope(m_inspector->isolate(), v8::MicrotasksScope::kRunMicrotasks); if (!scope.object()->IsObject()) return Response::Error("Value with given id is not an object"); v8::Local object = scope.object().As(); response = scope.injectedScript()->getProperties( object, scope.objectGroupName(), ownProperties.fromMaybe(false), accessorPropertiesOnly.fromMaybe(false), generatePreview.fromMaybe(false), result, exceptionDetails); if (!response.isSuccess()) return response; if (exceptionDetails->isJust() || accessorPropertiesOnly.fromMaybe(false)) return Response::OK(); v8::Local propertiesArray; if (!m_inspector->debugger() ->internalProperties(scope.context(), scope.object()) .ToLocal(&propertiesArray)) { return Response::InternalError(); } std::unique_ptr> propertiesProtocolArray = protocol::Array::create(); for (uint32_t i = 0; i < propertiesArray->Length(); i += 2) { v8::Local name; if (!propertiesArray->Get(scope.context(), i).ToLocal(&name) || !name->IsString()) { return Response::InternalError(); } v8::Local value; if (!propertiesArray->Get(scope.context(), i + 1).ToLocal(&value)) return Response::InternalError(); std::unique_ptr wrappedValue; protocol::Response response = scope.injectedScript()->wrapObject( value, scope.objectGroupName(), false, false, &wrappedValue); if (!response.isSuccess()) return response; propertiesProtocolArray->addItem( InternalPropertyDescriptor::create() .setName( toProtocolString(m_inspector->isolate(), name.As())) .setValue(std::move(wrappedValue)) .build()); } if (propertiesProtocolArray->length()) *internalProperties = std::move(propertiesProtocolArray); return Response::OK(); } Response V8RuntimeAgentImpl::releaseObject(const String16& objectId) { InjectedScript::ObjectScope scope(m_session, objectId); Response response = scope.initialize(); if (!response.isSuccess()) return response; scope.injectedScript()->releaseObject(objectId); return Response::OK(); } Response V8RuntimeAgentImpl::releaseObjectGroup(const String16& objectGroup) { m_session->releaseObjectGroup(objectGroup); return Response::OK(); } Response V8RuntimeAgentImpl::runIfWaitingForDebugger() { m_inspector->client()->runIfWaitingForDebugger(m_session->contextGroupId()); return Response::OK(); } Response V8RuntimeAgentImpl::setCustomObjectFormatterEnabled(bool enabled) { m_state->setBoolean(V8RuntimeAgentImplState::customObjectFormatterEnabled, enabled); if (!m_enabled) return Response::Error("Runtime agent is not enabled"); m_session->setCustomObjectFormatterEnabled(enabled); return Response::OK(); } Response V8RuntimeAgentImpl::setMaxCallStackSizeToCapture(int size) { if (size < 0) { return Response::Error("maxCallStackSizeToCapture should be non-negative"); } V8StackTraceImpl::maxCallStackSizeToCapture = size; return Response::OK(); } Response V8RuntimeAgentImpl::discardConsoleEntries() { V8ConsoleMessageStorage* storage = m_inspector->ensureConsoleMessageStorage(m_session->contextGroupId()); storage->clear(); return Response::OK(); } Response V8RuntimeAgentImpl::compileScript( const String16& expression, const String16& sourceURL, bool persistScript, Maybe executionContextId, Maybe* scriptId, Maybe* exceptionDetails) { if (!m_enabled) return Response::Error("Runtime agent is not enabled"); int contextId = 0; Response response = ensureContext(m_inspector, m_session->contextGroupId(), std::move(executionContextId), &contextId); if (!response.isSuccess()) return response; InjectedScript::ContextScope scope(m_session, contextId); response = scope.initialize(); if (!response.isSuccess()) return response; if (!persistScript) m_inspector->debugger()->muteScriptParsedEvents(); v8::Local script; bool isOk = m_inspector->compileScript(scope.context(), expression, sourceURL) .ToLocal(&script); if (!persistScript) m_inspector->debugger()->unmuteScriptParsedEvents(); if (!isOk) { if (scope.tryCatch().HasCaught()) { response = scope.injectedScript()->createExceptionDetails( scope.tryCatch(), String16(), false, exceptionDetails); if (!response.isSuccess()) return response; return Response::OK(); } else { return Response::Error("Script compilation failed"); } } if (!persistScript) return Response::OK(); String16 scriptValueId = String16::fromInteger(script->GetUnboundScript()->GetId()); std::unique_ptr> global( new v8::Global(m_inspector->isolate(), script)); m_compiledScripts[scriptValueId] = std::move(global); *scriptId = scriptValueId; return Response::OK(); } void V8RuntimeAgentImpl::runScript( const String16& scriptId, Maybe executionContextId, Maybe objectGroup, Maybe silent, Maybe includeCommandLineAPI, Maybe returnByValue, Maybe generatePreview, Maybe awaitPromise, std::unique_ptr callback) { if (!m_enabled) { callback->sendFailure(Response::Error("Runtime agent is not enabled")); return; } auto it = m_compiledScripts.find(scriptId); if (it == m_compiledScripts.end()) { callback->sendFailure(Response::Error("No script with given id")); return; } int contextId = 0; Response response = ensureContext(m_inspector, m_session->contextGroupId(), std::move(executionContextId), &contextId); if (!response.isSuccess()) { callback->sendFailure(response); return; } InjectedScript::ContextScope scope(m_session, contextId); response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (silent.fromMaybe(false)) scope.ignoreExceptionsAndMuteConsole(); std::unique_ptr> scriptWrapper = std::move(it->second); m_compiledScripts.erase(it); v8::Local script = scriptWrapper->Get(m_inspector->isolate()); if (script.IsEmpty()) { callback->sendFailure(Response::Error("Script execution failed")); return; } if (includeCommandLineAPI.fromMaybe(false)) scope.installCommandLineAPI(); v8::MaybeLocal maybeResultValue; { v8::MicrotasksScope microtasksScope(m_inspector->isolate(), v8::MicrotasksScope::kRunMicrotasks); maybeResultValue = script->Run(scope.context()); } // Re-initialize after running client's code, as it could have destroyed // context or session. response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } if (!awaitPromise.fromMaybe(false) || scope.tryCatch().HasCaught()) { wrapEvaluateResultAsync(scope.injectedScript(), maybeResultValue, scope.tryCatch(), objectGroup.fromMaybe(""), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), callback.get()); return; } scope.injectedScript()->addPromiseCallback( m_session, maybeResultValue.ToLocalChecked(), objectGroup.fromMaybe(""), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), EvaluateCallbackWrapper::wrap(std::move(callback))); } Response V8RuntimeAgentImpl::queryObjects( const String16& prototypeObjectId, Maybe objectGroup, std::unique_ptr* objects) { InjectedScript::ObjectScope scope(m_session, prototypeObjectId); Response response = scope.initialize(); if (!response.isSuccess()) return response; if (!scope.object()->IsObject()) { return Response::Error("Prototype should be instance of Object"); } v8::Local resultArray = m_inspector->debugger()->queryObjects( scope.context(), v8::Local::Cast(scope.object())); return scope.injectedScript()->wrapObject( resultArray, objectGroup.fromMaybe(scope.objectGroupName()), false, false, objects); } Response V8RuntimeAgentImpl::globalLexicalScopeNames( Maybe executionContextId, std::unique_ptr>* outNames) { int contextId = 0; Response response = ensureContext(m_inspector, m_session->contextGroupId(), std::move(executionContextId), &contextId); if (!response.isSuccess()) return response; InjectedScript::ContextScope scope(m_session, contextId); response = scope.initialize(); if (!response.isSuccess()) return response; v8::PersistentValueVector names(m_inspector->isolate()); v8::debug::GlobalLexicalScopeNames(scope.context(), &names); *outNames = protocol::Array::create(); for (size_t i = 0; i < names.Size(); ++i) { (*outNames)->addItem( toProtocolString(m_inspector->isolate(), names.Get(i))); } return Response::OK(); } Response V8RuntimeAgentImpl::getIsolateId(String16* outIsolateId) { char buf[40]; std::snprintf(buf, sizeof(buf), "%" PRIx64, m_inspector->isolateId()); *outIsolateId = buf; return Response::OK(); } Response V8RuntimeAgentImpl::getHeapUsage(double* out_usedSize, double* out_totalSize) { v8::HeapStatistics stats; m_inspector->isolate()->GetHeapStatistics(&stats); *out_usedSize = stats.used_heap_size(); *out_totalSize = stats.total_heap_size(); return Response::OK(); } void V8RuntimeAgentImpl::terminateExecution( std::unique_ptr callback) { m_inspector->debugger()->terminateExecution(std::move(callback)); } Response V8RuntimeAgentImpl::addBinding(const String16& name, Maybe executionContextId) { if (!m_state->getObject(V8RuntimeAgentImplState::bindings)) { m_state->setObject(V8RuntimeAgentImplState::bindings, protocol::DictionaryValue::create()); } protocol::DictionaryValue* bindings = m_state->getObject(V8RuntimeAgentImplState::bindings); if (bindings->booleanProperty(name, false)) return Response::OK(); if (executionContextId.isJust()) { int contextId = executionContextId.fromJust(); InspectedContext* context = m_inspector->getContext(m_session->contextGroupId(), contextId); if (!context) { return Response::Error( "Cannot find execution context with given executionContextId"); } addBinding(context, name); // false means that we should not add this binding later. bindings->setBoolean(name, false); return Response::OK(); } bindings->setBoolean(name, true); m_inspector->forEachContext( m_session->contextGroupId(), [&name, this](InspectedContext* context) { addBinding(context, name); }); return Response::OK(); } void V8RuntimeAgentImpl::bindingCallback( const v8::FunctionCallbackInfo& info) { v8::Isolate* isolate = info.GetIsolate(); if (info.Length() != 1 || !info[0]->IsString()) { info.GetIsolate()->ThrowException(toV8String( isolate, "Invalid arguments: should be exactly one string.")); return; } V8InspectorImpl* inspector = static_cast(v8::debug::GetInspector(isolate)); int contextId = InspectedContext::contextId(isolate->GetCurrentContext()); int contextGroupId = inspector->contextGroupId(contextId); String16 name = toProtocolString(isolate, v8::Local::Cast(info.Data())); String16 payload = toProtocolString(isolate, v8::Local::Cast(info[0])); inspector->forEachSession( contextGroupId, [&name, &payload, &contextId](V8InspectorSessionImpl* session) { session->runtimeAgent()->bindingCalled(name, payload, contextId); }); } void V8RuntimeAgentImpl::addBinding(InspectedContext* context, const String16& name) { v8::HandleScope handles(m_inspector->isolate()); v8::Local localContext = context->context(); v8::Local global = localContext->Global(); v8::Local v8Name = toV8String(m_inspector->isolate(), name); v8::Local functionValue; v8::MicrotasksScope microtasks(m_inspector->isolate(), v8::MicrotasksScope::kDoNotRunMicrotasks); if (v8::Function::New(localContext, bindingCallback, v8Name) .ToLocal(&functionValue)) { v8::Maybe success = global->Set(localContext, v8Name, functionValue); USE(success); } } Response V8RuntimeAgentImpl::removeBinding(const String16& name) { protocol::DictionaryValue* bindings = m_state->getObject(V8RuntimeAgentImplState::bindings); if (!bindings) return Response::OK(); bindings->remove(name); return Response::OK(); } void V8RuntimeAgentImpl::bindingCalled(const String16& name, const String16& payload, int executionContextId) { protocol::DictionaryValue* bindings = m_state->getObject(V8RuntimeAgentImplState::bindings); if (!bindings || !bindings->get(name)) return; m_frontend.bindingCalled(name, payload, executionContextId); } void V8RuntimeAgentImpl::addBindings(InspectedContext* context) { if (!m_enabled) return; protocol::DictionaryValue* bindings = m_state->getObject(V8RuntimeAgentImplState::bindings); if (!bindings) return; for (size_t i = 0; i < bindings->size(); ++i) { if (!bindings->at(i).second) continue; addBinding(context, bindings->at(i).first); } } void V8RuntimeAgentImpl::restore() { if (!m_state->booleanProperty(V8RuntimeAgentImplState::runtimeEnabled, false)) return; m_frontend.executionContextsCleared(); enable(); if (m_state->booleanProperty( V8RuntimeAgentImplState::customObjectFormatterEnabled, false)) m_session->setCustomObjectFormatterEnabled(true); m_inspector->forEachContext( m_session->contextGroupId(), [this](InspectedContext* context) { addBindings(context); }); } Response V8RuntimeAgentImpl::enable() { if (m_enabled) return Response::OK(); m_inspector->client()->beginEnsureAllContextsInGroup( m_session->contextGroupId()); m_enabled = true; m_state->setBoolean(V8RuntimeAgentImplState::runtimeEnabled, true); m_inspector->enableStackCapturingIfNeeded(); m_session->reportAllContexts(this); V8ConsoleMessageStorage* storage = m_inspector->ensureConsoleMessageStorage(m_session->contextGroupId()); for (const auto& message : storage->messages()) { if (!reportMessage(message.get(), false)) break; } return Response::OK(); } Response V8RuntimeAgentImpl::disable() { if (!m_enabled) return Response::OK(); m_enabled = false; m_state->setBoolean(V8RuntimeAgentImplState::runtimeEnabled, false); m_state->remove(V8RuntimeAgentImplState::bindings); m_inspector->disableStackCapturingIfNeeded(); m_session->setCustomObjectFormatterEnabled(false); reset(); m_inspector->client()->endEnsureAllContextsInGroup( m_session->contextGroupId()); if (m_session->debuggerAgent() && !m_session->debuggerAgent()->enabled()) { m_session->debuggerAgent()->setAsyncCallStackDepth(0); } return Response::OK(); } void V8RuntimeAgentImpl::reset() { m_compiledScripts.clear(); if (m_enabled) { int sessionId = m_session->sessionId(); m_inspector->forEachContext(m_session->contextGroupId(), [&sessionId](InspectedContext* context) { context->setReported(sessionId, false); }); m_frontend.executionContextsCleared(); } } void V8RuntimeAgentImpl::reportExecutionContextCreated( InspectedContext* context) { if (!m_enabled) return; context->setReported(m_session->sessionId(), true); std::unique_ptr description = protocol::Runtime::ExecutionContextDescription::create() .setId(context->contextId()) .setName(context->humanReadableName()) .setOrigin(context->origin()) .build(); if (!context->auxData().isEmpty()) description->setAuxData(protocol::DictionaryValue::cast( protocol::StringUtil::parseJSON(context->auxData()))); m_frontend.executionContextCreated(std::move(description)); } void V8RuntimeAgentImpl::reportExecutionContextDestroyed( InspectedContext* context) { if (m_enabled && context->isReported(m_session->sessionId())) { context->setReported(m_session->sessionId(), false); m_frontend.executionContextDestroyed(context->contextId()); } } void V8RuntimeAgentImpl::inspect( std::unique_ptr objectToInspect, std::unique_ptr hints) { if (m_enabled) m_frontend.inspectRequested(std::move(objectToInspect), std::move(hints)); } void V8RuntimeAgentImpl::messageAdded(V8ConsoleMessage* message) { if (m_enabled) reportMessage(message, true); } bool V8RuntimeAgentImpl::reportMessage(V8ConsoleMessage* message, bool generatePreview) { message->reportToFrontend(&m_frontend, m_session, generatePreview); m_frontend.flush(); return m_inspector->hasConsoleMessageStorage(m_session->contextGroupId()); } } // namespace v8_inspector