blob: 9192ba9d8412ab6e7ca4fe0fbecbd02354a49ced [file] [log] [blame]
/*
* Copyright (C) 2010, 2015 Apple Inc. All rights reserved.
* Copyright (C) 2010 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:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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 "InspectorApplicationCacheAgent.h"
#include "ApplicationCacheHost.h"
#include "DocumentLoader.h"
#include "Frame.h"
#include "FrameLoader.h"
#include "InspectorPageAgent.h"
#include "InstrumentingAgents.h"
#include "LoaderStrategy.h"
#include "Page.h"
#include "PlatformStrategies.h"
namespace WebCore {
using namespace Inspector;
InspectorApplicationCacheAgent::InspectorApplicationCacheAgent(PageAgentContext& context)
: InspectorAgentBase("ApplicationCache"_s, context)
, m_frontendDispatcher(makeUnique<Inspector::ApplicationCacheFrontendDispatcher>(context.frontendRouter))
, m_backendDispatcher(Inspector::ApplicationCacheBackendDispatcher::create(context.backendDispatcher, this))
, m_inspectedPage(context.inspectedPage)
{
}
InspectorApplicationCacheAgent::~InspectorApplicationCacheAgent() = default;
void InspectorApplicationCacheAgent::didCreateFrontendAndBackend(FrontendRouter*, BackendDispatcher*)
{
}
void InspectorApplicationCacheAgent::willDestroyFrontendAndBackend(Inspector::DisconnectReason)
{
disable();
}
Protocol::ErrorStringOr<void> InspectorApplicationCacheAgent::enable()
{
if (m_instrumentingAgents.enabledApplicationCacheAgent() == this)
return makeUnexpected("ApplicationCache domain already enabled"_s);
m_instrumentingAgents.setEnabledApplicationCacheAgent(this);
// We need to pass initial navigator.onOnline.
networkStateChanged();
return { };
}
Protocol::ErrorStringOr<void> InspectorApplicationCacheAgent::disable()
{
if (m_instrumentingAgents.enabledApplicationCacheAgent() != this)
return makeUnexpected("ApplicationCache domain already disabled"_s);
m_instrumentingAgents.setEnabledApplicationCacheAgent(nullptr);
return { };
}
void InspectorApplicationCacheAgent::updateApplicationCacheStatus(Frame* frame)
{
auto* pageAgent = m_instrumentingAgents.enabledPageAgent();
if (!pageAgent)
return;
if (!frame)
return;
auto* documentLoader = frame->loader().documentLoader();
if (!documentLoader)
return;
auto& host = documentLoader->applicationCacheHost();
int status = host.status();
auto manifestURL = host.applicationCacheInfo().manifest.string();
m_frontendDispatcher->applicationCacheStatusUpdated(pageAgent->frameId(frame), manifestURL, status);
}
void InspectorApplicationCacheAgent::networkStateChanged()
{
m_frontendDispatcher->networkStateUpdated(platformStrategies()->loaderStrategy()->isOnLine());
}
Expected<Ref<JSON::ArrayOf<Protocol::ApplicationCache::FrameWithManifest>>, Protocol::ErrorString> InspectorApplicationCacheAgent::getFramesWithManifests()
{
auto* pageAgent = m_instrumentingAgents.enabledPageAgent();
if (!pageAgent)
return makeUnexpected("Page domain must be enabled"_s);
auto result = JSON::ArrayOf<Protocol::ApplicationCache::FrameWithManifest>::create();
m_inspectedPage.forEachFrame([&](Frame& frame) {
auto* documentLoader = frame.loader().documentLoader();
if (!documentLoader)
return;
auto& host = documentLoader->applicationCacheHost();
String manifestURL = host.applicationCacheInfo().manifest.string();
if (!manifestURL.isEmpty()) {
result->addItem(Protocol::ApplicationCache::FrameWithManifest::create()
.setFrameId(pageAgent->frameId(&frame))
.setManifestURL(manifestURL)
.setStatus(static_cast<int>(host.status()))
.release());
}
});
return result;
}
DocumentLoader* InspectorApplicationCacheAgent::assertFrameWithDocumentLoader(Protocol::ErrorString& errorString, const Inspector::Protocol::Network::FrameId& frameId)
{
auto* pageAgent = m_instrumentingAgents.enabledPageAgent();
if (!pageAgent) {
errorString = "Page domain must be enabled"_s;
return nullptr;
}
auto* frame = pageAgent->assertFrame(errorString, frameId);
if (!frame)
return nullptr;
return InspectorPageAgent::assertDocumentLoader(errorString, frame);
}
Expected<String, Protocol::ErrorString> InspectorApplicationCacheAgent::getManifestForFrame(const Inspector::Protocol::Network::FrameId& frameId)
{
Protocol::ErrorString errorString;
DocumentLoader* documentLoader = assertFrameWithDocumentLoader(errorString, frameId);
if (!documentLoader)
return makeUnexpected(errorString);
return documentLoader->applicationCacheHost().applicationCacheInfo().manifest.string();
}
Expected<Ref<Protocol::ApplicationCache::ApplicationCache>, Protocol::ErrorString> InspectorApplicationCacheAgent::getApplicationCacheForFrame(const Inspector::Protocol::Network::FrameId& frameId)
{
Protocol::ErrorString errorString;
auto* documentLoader = assertFrameWithDocumentLoader(errorString, frameId);
if (!documentLoader)
return makeUnexpected(errorString);
auto& host = documentLoader->applicationCacheHost();
return buildObjectForApplicationCache(host.resourceList(), host.applicationCacheInfo());
}
Ref<Protocol::ApplicationCache::ApplicationCache> InspectorApplicationCacheAgent::buildObjectForApplicationCache(const Vector<ApplicationCacheHost::ResourceInfo>& applicationCacheResources, const ApplicationCacheHost::CacheInfo& applicationCacheInfo)
{
return Protocol::ApplicationCache::ApplicationCache::create()
.setManifestURL(applicationCacheInfo.manifest.string())
.setSize(applicationCacheInfo.size)
.setCreationTime(applicationCacheInfo.creationTime)
.setUpdateTime(applicationCacheInfo.updateTime)
.setResources(buildArrayForApplicationCacheResources(applicationCacheResources))
.release();
}
Ref<JSON::ArrayOf<Protocol::ApplicationCache::ApplicationCacheResource>> InspectorApplicationCacheAgent::buildArrayForApplicationCacheResources(const Vector<ApplicationCacheHost::ResourceInfo>& applicationCacheResources)
{
auto result = JSON::ArrayOf<Protocol::ApplicationCache::ApplicationCacheResource>::create();
for (auto& info : applicationCacheResources)
result->addItem(buildObjectForApplicationCacheResource(info));
return result;
}
Ref<Protocol::ApplicationCache::ApplicationCacheResource> InspectorApplicationCacheAgent::buildObjectForApplicationCacheResource(const ApplicationCacheHost::ResourceInfo& resourceInfo)
{
auto types = makeString(resourceInfo.isMaster ? "Master " : "",
resourceInfo.isManifest ? "Manifest " : "",
resourceInfo.isFallback ? "Fallback " : "",
resourceInfo.isForeign ? "Foreign " : "",
resourceInfo.isExplicit ? "Explicit " : "");
return Protocol::ApplicationCache::ApplicationCacheResource::create()
.setUrl(resourceInfo.resource.string())
.setSize(static_cast<int>(resourceInfo.size))
.setType(types)
.release();
}
} // namespace WebCore