blob: 3cf9f13554428e707f148aaa7cb977b871121a7e [file] [log] [blame]
// Copyright 2017 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 "services/network/cache_url_loader.h"
#include "base/bind.h"
#include "base/macros.h"
#include "base/strings/string_util.h"
#include "base/threading/thread_task_runner_handle.h"
#include "mojo/common/data_pipe_utils.h"
#include "net/url_request/view_cache_helper.h"
namespace network {
namespace {
class CacheURLLoader {
public:
CacheURLLoader(const GURL& url,
net::URLRequestContext* request_context,
mojom::URLLoaderClientPtr client)
: client_(std::move(client)) {
scoped_refptr<net::HttpResponseHeaders> headers(
new net::HttpResponseHeaders("HTTP/1.1 200 OK"));
ResourceResponseHead resource_response;
resource_response.headers = headers;
resource_response.mime_type = "text/html";
client_->OnReceiveResponse(resource_response, base::nullopt, nullptr);
auto url_prefix = url.GetWithEmptyPath().spec();
std::string cache_key = url.spec().substr(url_prefix.size());
int rv;
if (cache_key.empty()) {
rv = cache_helper_.GetContentsHTML(
request_context, url_prefix, &data_,
base::Bind(&CacheURLLoader::DataAvailable, base::Unretained(this)));
} else {
rv = cache_helper_.GetEntryInfoHTML(
cache_key, request_context, &data_,
base::Bind(&CacheURLLoader::DataAvailable, base::Unretained(this)));
}
if (rv != net::ERR_IO_PENDING)
DataAvailable(rv);
}
~CacheURLLoader() {}
private:
void DataAvailable(int result) {
DCHECK_EQ(net::OK, result);
mojo::DataPipe data_pipe(data_.size());
CHECK(
mojo::common::BlockingCopyFromString(data_, data_pipe.producer_handle));
client_->OnStartLoadingResponseBody(std::move(data_pipe.consumer_handle));
URLLoaderCompletionStatus status(net::OK);
status.encoded_data_length = data_.size();
status.encoded_body_length = data_.size();
client_->OnComplete(status);
// So we don't delete |this| in the constructor.
base::ThreadTaskRunnerHandle::Get()->DeleteSoon(FROM_HERE, this);
}
std::string data_;
mojom::URLLoaderClientPtr client_;
net::ViewCacheHelper cache_helper_;
DISALLOW_COPY_AND_ASSIGN(CacheURLLoader);
};
} // namespace
void StartCacheURLLoader(const GURL& url,
net::URLRequestContext* request_context,
mojom::URLLoaderClientPtr client) {
new CacheURLLoader(url, request_context, std::move(client));
}
} // namespace network