/* * Copyright (C) 2011 Google Inc. All rights reserved. * Copyright (C) 2015 Apple 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 "config.h" #include "PageRuntimeAgent.h" #include "Document.h" #include "Frame.h" #include "InspectorPageAgent.h" #include "InstrumentingAgents.h" #include "JSDOMWindowBase.h" #include "MainFrame.h" #include "Page.h" #include "PageConsoleClient.h" #include "ScriptController.h" #include "ScriptState.h" #include "SecurityOrigin.h" #include #include #include using Inspector::Protocol::Runtime::ExecutionContextDescription; using namespace Inspector; namespace WebCore { PageRuntimeAgent::PageRuntimeAgent(PageAgentContext& context, InspectorPageAgent* pageAgent) : InspectorRuntimeAgent(context) , m_frontendDispatcher(std::make_unique(context.frontendRouter)) , m_backendDispatcher(Inspector::RuntimeBackendDispatcher::create(context.backendDispatcher, this)) , m_pageAgent(pageAgent) , m_inspectedPage(context.inspectedPage) { } void PageRuntimeAgent::didCreateFrontendAndBackend(Inspector::FrontendRouter*, Inspector::BackendDispatcher*) { } void PageRuntimeAgent::willDestroyFrontendAndBackend(Inspector::DisconnectReason reason) { String unused; disable(unused); InspectorRuntimeAgent::willDestroyFrontendAndBackend(reason); } void PageRuntimeAgent::enable(ErrorString& errorString) { if (enabled()) return; InspectorRuntimeAgent::enable(errorString); // Only report existing contexts if the page did commit load, otherwise we may // unintentionally initialize contexts in the frames which may trigger some listeners // that are expected to be triggered only after the load is committed, see http://crbug.com/131623 if (m_mainWorldContextCreated) reportExecutionContextCreation(); } void PageRuntimeAgent::disable(ErrorString& errorString) { if (!enabled()) return; InspectorRuntimeAgent::disable(errorString); } void PageRuntimeAgent::didCreateMainWorldContext(Frame& frame) { m_mainWorldContextCreated = true; if (!enabled()) return; String frameId = m_pageAgent->frameId(&frame); JSC::ExecState* scriptState = mainWorldExecState(&frame); notifyContextCreated(frameId, scriptState, nullptr, true); } InjectedScript PageRuntimeAgent::injectedScriptForEval(ErrorString& errorString, const int* executionContextId) { if (!executionContextId) { JSC::ExecState* scriptState = mainWorldExecState(&m_inspectedPage.mainFrame()); InjectedScript result = injectedScriptManager().injectedScriptFor(scriptState); if (result.hasNoValue()) errorString = ASCIILiteral("Internal error: main world execution context not found."); return result; } InjectedScript injectedScript = injectedScriptManager().injectedScriptForId(*executionContextId); if (injectedScript.hasNoValue()) errorString = ASCIILiteral("Execution context with given id not found."); return injectedScript; } void PageRuntimeAgent::muteConsole() { PageConsoleClient::mute(); } void PageRuntimeAgent::unmuteConsole() { PageConsoleClient::unmute(); } void PageRuntimeAgent::reportExecutionContextCreation() { Vector> isolatedContexts; for (Frame* frame = &m_inspectedPage.mainFrame(); frame; frame = frame->tree().traverseNext()) { if (!frame->script().canExecuteScripts(NotAboutToExecuteScript)) continue; String frameId = m_pageAgent->frameId(frame); JSC::ExecState* scriptState = mainWorldExecState(frame); notifyContextCreated(frameId, scriptState, nullptr, true); frame->script().collectIsolatedContexts(isolatedContexts); if (isolatedContexts.isEmpty()) continue; for (auto& context : isolatedContexts) notifyContextCreated(frameId, context.first, context.second, false); isolatedContexts.clear(); } } void PageRuntimeAgent::notifyContextCreated(const String& frameId, JSC::ExecState* scriptState, SecurityOrigin* securityOrigin, bool isPageContext) { ASSERT(securityOrigin || isPageContext); InjectedScript result = injectedScriptManager().injectedScriptFor(scriptState); if (result.hasNoValue()) return; int executionContextId = injectedScriptManager().injectedScriptIdFor(scriptState); String name = securityOrigin ? securityOrigin->toRawString() : String(); m_frontendDispatcher->executionContextCreated(ExecutionContextDescription::create() .setId(executionContextId) .setIsPageContext(isPageContext) .setName(name) .setFrameId(frameId) .release()); } } // namespace WebCore