1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149
|
// Copyright 2014 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 "content/browser/loader/navigation_resource_handler.h"
#include "base/logging.h"
#include "content/browser/devtools/devtools_netlog_observer.h"
#include "content/browser/loader/navigation_url_loader_impl_core.h"
#include "content/browser/loader/resource_request_info_impl.h"
#include "content/browser/resource_context_impl.h"
#include "content/browser/streams/stream.h"
#include "content/browser/streams/stream_context.h"
#include "content/public/browser/resource_controller.h"
#include "content/public/browser/stream_handle.h"
#include "content/public/common/resource_response.h"
#include "net/base/net_errors.h"
#include "net/url_request/url_request.h"
namespace content {
NavigationResourceHandler::NavigationResourceHandler(
net::URLRequest* request,
NavigationURLLoaderImplCore* core)
: ResourceHandler(request),
core_(core) {
core_->set_resource_handler(this);
}
NavigationResourceHandler::~NavigationResourceHandler() {
if (core_) {
core_->NotifyRequestFailed(net::ERR_ABORTED);
DetachFromCore();
}
}
void NavigationResourceHandler::Cancel() {
controller()->Cancel();
core_ = nullptr;
}
void NavigationResourceHandler::FollowRedirect() {
controller()->Resume();
}
void NavigationResourceHandler::SetController(ResourceController* controller) {
writer_.set_controller(controller);
ResourceHandler::SetController(controller);
}
bool NavigationResourceHandler::OnUploadProgress(uint64 position, uint64 size) {
return true;
}
bool NavigationResourceHandler::OnRequestRedirected(
const net::RedirectInfo& redirect_info,
ResourceResponse* response,
bool* defer) {
DCHECK(core_);
// TODO(davidben): Perform a CSP check here, and anything else that would have
// been done renderer-side.
DevToolsNetLogObserver::PopulateResponseInfo(request(), response);
core_->NotifyRequestRedirected(redirect_info, response);
*defer = true;
return true;
}
bool NavigationResourceHandler::OnResponseStarted(ResourceResponse* response,
bool* defer) {
DCHECK(core_);
ResourceRequestInfoImpl* info = GetRequestInfo();
// If the BufferedResourceHandler intercepted this request and converted it
// into a download, it will still call OnResponseStarted and immediately
// cancel. Ignore the call; OnReadCompleted will happen shortly.
//
// TODO(davidben): Move the dispatch out of BufferedResourceHandler. Perhaps
// all the way to the UI thread. Downloads, user certificates, etc., should be
// dispatched at the navigation layer.
if (info->IsDownload() || info->is_stream())
return true;
StreamContext* stream_context =
GetStreamContextForResourceContext(info->GetContext());
writer_.InitializeStream(stream_context->registry(),
request()->url().GetOrigin());
// Detach from the loader; at this point, the request is now owned by the
// StreamHandle.
DevToolsNetLogObserver::PopulateResponseInfo(request(), response);
core_->NotifyResponseStarted(response, writer_.stream()->CreateHandle());
DetachFromCore();
return true;
}
bool NavigationResourceHandler::OnWillStart(const GURL& url, bool* defer) {
return true;
}
bool NavigationResourceHandler::OnBeforeNetworkStart(const GURL& url,
bool* defer) {
return true;
}
bool NavigationResourceHandler::OnWillRead(scoped_refptr<net::IOBuffer>* buf,
int* buf_size,
int min_size) {
writer_.OnWillRead(buf, buf_size, min_size);
return true;
}
bool NavigationResourceHandler::OnReadCompleted(int bytes_read, bool* defer) {
writer_.OnReadCompleted(bytes_read, defer);
return true;
}
void NavigationResourceHandler::OnResponseCompleted(
const net::URLRequestStatus& status,
const std::string& security_info,
bool* defer) {
// If the request has already committed, close the stream and leave it as-is.
//
// TODO(davidben): The net error code should be passed through StreamWriter
// down to the stream's consumer. See https://crbug.com/426162.
if (writer_.stream()) {
writer_.Finalize();
return;
}
if (core_) {
DCHECK_NE(net::OK, status.error());
core_->NotifyRequestFailed(status.error());
DetachFromCore();
}
}
void NavigationResourceHandler::OnDataDownloaded(int bytes_downloaded) {
NOTREACHED();
}
void NavigationResourceHandler::DetachFromCore() {
DCHECK(core_);
core_->set_resource_handler(nullptr);
core_ = nullptr;
}
} // namespace content
|