// Copyright (c) 2012 The Chromium 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 "webkit/plugins/npapi/plugin_stream_url.h" #include "net/http/http_response_headers.h" #include "webkit/plugins/npapi/plugin_host.h" #include "webkit/plugins/npapi/plugin_instance.h" #include "webkit/plugins/npapi/plugin_lib.h" #include "webkit/plugins/npapi/webplugin.h" namespace webkit { namespace npapi { PluginStreamUrl::PluginStreamUrl( unsigned long resource_id, const GURL &url, PluginInstance *instance, bool notify_needed, void *notify_data) : PluginStream(instance, url.spec().c_str(), notify_needed, notify_data), url_(url), id_(resource_id) { } bool PluginStreamUrl::Close(NPReason reason) { // Protect the stream against it being destroyed or the whole plugin instance // being destroyed within the destroy stream handler. scoped_refptr protect(this); CancelRequest(); bool result = PluginStream::Close(reason); instance()->RemoveStream(this); return result; } WebPluginResourceClient* PluginStreamUrl::AsResourceClient() { return static_cast(this); } void PluginStreamUrl::CancelRequest() { if (id_ > 0) { if (instance()->webplugin()) { instance()->webplugin()->CancelResource(id_); } id_ = 0; } } void PluginStreamUrl::WillSendRequest(const GURL& url, int http_status_code) { if (notify_needed()) { // If the plugin participates in HTTP url redirect handling then notify it. if (net::HttpResponseHeaders::IsRedirectResponseCode(http_status_code) && instance()->handles_url_redirects()) { pending_redirect_url_ = url.spec(); instance()->NPP_URLRedirectNotify(url.spec().c_str(), http_status_code, notify_data()); return; } } url_ = url; UpdateUrl(url.spec().c_str()); } void PluginStreamUrl::DidReceiveResponse(const std::string& mime_type, const std::string& headers, uint32 expected_length, uint32 last_modified, bool request_is_seekable) { // Protect the stream against it being destroyed or the whole plugin instance // being destroyed within the new stream handler. scoped_refptr protect(this); bool opened = Open(mime_type, headers, expected_length, last_modified, request_is_seekable); if (!opened) { CancelRequest(); instance()->RemoveStream(this); } else { if (id_ > 0) instance()->webplugin()->SetDeferResourceLoading(id_, false); } } void PluginStreamUrl::DidReceiveData(const char* buffer, int length, int data_offset) { if (!open()) return; // Protect the stream against it being destroyed or the whole plugin instance // being destroyed within the write handlers scoped_refptr protect(this); if (length > 0) { // The PluginStreamUrl instance could get deleted if the plugin fails to // accept data in NPP_Write. if (Write(const_cast(buffer), length, data_offset) > 0) { if (id_ > 0) instance()->webplugin()->SetDeferResourceLoading(id_, false); } } } void PluginStreamUrl::DidFinishLoading() { if (!seekable()) { Close(NPRES_DONE); } } void PluginStreamUrl::DidFail() { Close(NPRES_NETWORK_ERR); } bool PluginStreamUrl::IsMultiByteResponseExpected() { return seekable(); } int PluginStreamUrl::ResourceId() { return id_; } PluginStreamUrl::~PluginStreamUrl() { if (instance() && instance()->webplugin()) { instance()->webplugin()->ResourceClientDeleted(AsResourceClient()); } } } // namespace npapi } // namespace webkit