/* * 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 "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/string-util.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/tracing/trace-event.h" #include "include/v8-inspector.h" namespace v8_inspector { namespace V8RuntimeAgentImplState { static const char customObjectFormatterEnabled[] = "customObjectFormatterEnabled"; static const char runtimeEnabled[] = "runtimeEnabled"; }; using protocol::Runtime::RemoteObject; namespace { template class ProtocolPromiseHandler { public: static void add(V8InspectorImpl* inspector, v8::Local context, v8::MaybeLocal value, const String16& notPromiseError, int contextGroupId, int executionContextId, const String16& objectGroup, bool returnByValue, bool generatePreview, std::unique_ptr callback) { if (value.IsEmpty()) { callback->sendFailure(Response::InternalError()); return; } if (!value.ToLocalChecked()->IsPromise()) { callback->sendFailure(Response::Error(notPromiseError)); return; } v8::MicrotasksScope microtasks_scope(inspector->isolate(), v8::MicrotasksScope::kRunMicrotasks); v8::Local promise = v8::Local::Cast(value.ToLocalChecked()); Callback* rawCallback = callback.get(); ProtocolPromiseHandler* handler = new ProtocolPromiseHandler( inspector, contextGroupId, executionContextId, objectGroup, returnByValue, generatePreview, std::move(callback)); v8::Local wrapper = handler->m_wrapper.Get(inspector->isolate()); v8::Local thenCallbackFunction = v8::Function::New(context, thenCallback, wrapper, 0, v8::ConstructorBehavior::kThrow) .ToLocalChecked(); if (promise->Then(context, thenCallbackFunction).IsEmpty()) { rawCallback->sendFailure(Response::InternalError()); return; } v8::Local catchCallbackFunction = v8::Function::New(context, catchCallback, wrapper, 0, v8::ConstructorBehavior::kThrow) .ToLocalChecked(); if (promise->Catch(context, catchCallbackFunction).IsEmpty()) { rawCallback->sendFailure(Response::InternalError()); return; } } private: static void thenCallback(const v8::FunctionCallbackInfo& info) { ProtocolPromiseHandler* handler = static_cast*>( info.Data().As()->Value()); DCHECK(handler); v8::Local value = info.Length() > 0 ? info[0] : v8::Local::Cast(v8::Undefined(info.GetIsolate())); std::unique_ptr wrappedValue( handler->wrapObject(value)); if (!wrappedValue) return; handler->m_callback->sendSuccess( std::move(wrappedValue), Maybe()); } static void catchCallback(const v8::FunctionCallbackInfo& info) { ProtocolPromiseHandler* handler = static_cast*>( info.Data().As()->Value()); DCHECK(handler); v8::Local value = info.Length() > 0 ? info[0] : v8::Local::Cast(v8::Undefined(info.GetIsolate())); std::unique_ptr wrappedValue( handler->wrapObject(value)); if (!wrappedValue) return; std::unique_ptr stack = handler->m_inspector->debugger()->captureStackTrace(true); std::unique_ptr exceptionDetails = protocol::Runtime::ExceptionDetails::create() .setExceptionId(handler->m_inspector->nextExceptionId()) .setText("Uncaught (in promise)") .setLineNumber(stack && !stack->isEmpty() ? stack->topLineNumber() : 0) .setColumnNumber( stack && !stack->isEmpty() ? stack->topColumnNumber() : 0) .setException(wrappedValue->clone()) .build(); if (stack) exceptionDetails->setStackTrace(stack->buildInspectorObjectImpl()); if (stack && !stack->isEmpty()) exceptionDetails->setScriptId(toString16(stack->topScriptId())); handler->m_callback->sendSuccess(std::move(wrappedValue), std::move(exceptionDetails)); } ProtocolPromiseHandler(V8InspectorImpl* inspector, int contextGroupId, int executionContextId, const String16& objectGroup, bool returnByValue, bool generatePreview, std::unique_ptr callback) : m_inspector(inspector), m_contextGroupId(contextGroupId), m_executionContextId(executionContextId), m_objectGroup(objectGroup), m_returnByValue(returnByValue), m_generatePreview(generatePreview), m_callback(std::move(callback)), m_wrapper(inspector->isolate(), v8::External::New(inspector->isolate(), this)) { m_wrapper.SetWeak(this, cleanup, v8::WeakCallbackType::kParameter); } static void cleanup( const v8::WeakCallbackInfo>& data) { if (!data.GetParameter()->m_wrapper.IsEmpty()) { data.GetParameter()->m_wrapper.Reset(); data.SetSecondPassCallback(cleanup); } else { data.GetParameter()->m_callback->sendFailure( Response::Error("Promise was collected")); delete data.GetParameter(); } } std::unique_ptr wrapObject( v8::Local value) { InjectedScript::ContextScope scope(m_inspector, m_contextGroupId, m_executionContextId); Response response = scope.initialize(); if (!response.isSuccess()) { m_callback->sendFailure(response); return nullptr; } std::unique_ptr wrappedValue; response = scope.injectedScript()->wrapObject( value, m_objectGroup, m_returnByValue, m_generatePreview, &wrappedValue); if (!response.isSuccess()) { m_callback->sendFailure(response); return nullptr; } return wrappedValue; } V8InspectorImpl* m_inspector; int m_contextGroupId; int m_executionContextId; String16 m_objectGroup; bool m_returnByValue; bool m_generatePreview; std::unique_ptr m_callback; v8::Global m_wrapper; }; template bool wrapEvaluateResultAsync(InjectedScript* injectedScript, v8::MaybeLocal maybeResultValue, const v8::TryCatch& tryCatch, const String16& objectGroup, bool returnByValue, bool generatePreview, Callback* 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; } 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, 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_inspector, m_session->contextGroupId(), 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(); bool evalIsDisabled = !scope.context()->IsCodeGenerationFromStringsAllowed(); // Temporarily enable allow evals for inspector. if (evalIsDisabled) scope.context()->AllowCodeGenerationFromStrings(true); v8::MaybeLocal maybeResultValue; v8::Local script; if (m_inspector->compileScript(scope.context(), expression, String16()) .ToLocal(&script)) { maybeResultValue = m_inspector->runCompiledScript(scope.context(), script); } if (evalIsDisabled) scope.context()->AllowCodeGenerationFromStrings(false); // 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; } ProtocolPromiseHandler::add( m_inspector, scope.context(), maybeResultValue, "Result of the evaluation is not a promise", m_session->contextGroupId(), scope.injectedScript()->context()->contextId(), objectGroup.fromMaybe(""), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), std::move(callback)); } void V8RuntimeAgentImpl::awaitPromise( const String16& promiseObjectId, Maybe returnByValue, Maybe generatePreview, std::unique_ptr callback) { InjectedScript::ObjectScope scope(m_inspector, m_session->contextGroupId(), promiseObjectId); Response response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } ProtocolPromiseHandler::add( m_inspector, scope.context(), scope.object(), "Could not find promise with given id", m_session->contextGroupId(), scope.injectedScript()->context()->contextId(), scope.objectGroupName(), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), std::move(callback)); } void V8RuntimeAgentImpl::callFunctionOn( const String16& objectId, const String16& expression, Maybe> optionalArguments, Maybe silent, Maybe returnByValue, Maybe generatePreview, Maybe userGesture, Maybe awaitPromise, std::unique_ptr callback) { InjectedScript::ObjectScope scope(m_inspector, m_session->contextGroupId(), objectId); Response response = scope.initialize(); if (!response.isSuccess()) { callback->sendFailure(response); return; } 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 = scope.injectedScript()->resolveCallArgument(arguments->get(i), &argumentValue); if (!response.isSuccess()) { callback->sendFailure(response); return; } argv[i] = argumentValue; } } if (silent.fromMaybe(false)) scope.ignoreExceptionsAndMuteConsole(); if (userGesture.fromMaybe(false)) scope.pretendUserGesture(); v8::MaybeLocal maybeFunctionValue; v8::Local functionScript; if (m_inspector ->compileScript(scope.context(), "(" + expression + ")", String16()) .ToLocal(&functionScript)) { maybeFunctionValue = m_inspector->runCompiledScript(scope.context(), functionScript); } // 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 (scope.tryCatch().HasCaught()) { wrapEvaluateResultAsync(scope.injectedScript(), maybeFunctionValue, scope.tryCatch(), scope.objectGroupName(), 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 = m_inspector->callFunction( functionValue.As(), scope.context(), scope.object(), 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.fromMaybe(false) || scope.tryCatch().HasCaught()) { wrapEvaluateResultAsync(scope.injectedScript(), maybeResultValue, scope.tryCatch(), scope.objectGroupName(), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), callback.get()); return; } ProtocolPromiseHandler::add( m_inspector, scope.context(), maybeResultValue, "Result of the function call is not a promise", m_session->contextGroupId(), scope.injectedScript()->context()->contextId(), scope.objectGroupName(), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), 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_inspector, m_session->contextGroupId(), objectId); Response response = scope.initialize(); if (!response.isSuccess()) return response; scope.ignoreExceptionsAndMuteConsole(); 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(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_inspector, m_session->contextGroupId(), 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); m_session->setCustomObjectFormatterEnabled(enabled); 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_inspector, m_session->contextGroupId(), 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_inspector, m_session->contextGroupId(), 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 = m_inspector->runCompiledScript(scope.context(), script); // 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; } ProtocolPromiseHandler::add( m_inspector, scope.context(), maybeResultValue.ToLocalChecked(), "Result of the script execution is not a promise", m_session->contextGroupId(), scope.injectedScript()->context()->contextId(), objectGroup.fromMaybe(""), returnByValue.fromMaybe(false), generatePreview.fromMaybe(false), std::move(callback)); } 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); } 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_inspector->disableStackCapturingIfNeeded(); m_session->discardInjectedScripts(); reset(); m_inspector->client()->endEnsureAllContextsInGroup( m_session->contextGroupId()); return Response::OK(); } void V8RuntimeAgentImpl::reset() { m_compiledScripts.clear(); if (m_enabled) { if (const V8InspectorImpl::ContextByIdMap* contexts = m_inspector->contextGroup(m_session->contextGroupId())) { for (auto& idContext : *contexts) idContext.second->setReported(false); } m_frontend.executionContextsCleared(); } } void V8RuntimeAgentImpl::reportExecutionContextCreated( InspectedContext* context) { if (!m_enabled) return; context->setReported(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()) { context->setReported(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