// 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 "net/url_request/view_cache_helper.h" #include "base/bind.h" #include "base/bind_helpers.h" #include "base/strings/stringprintf.h" #include "net/base/escape.h" #include "net/base/io_buffer.h" #include "net/base/net_errors.h" #include "net/disk_cache/disk_cache.h" #include "net/http/http_cache.h" #include "net/http/http_response_headers.h" #include "net/http/http_response_info.h" #include "net/url_request/url_request_context.h" #define VIEW_CACHE_HEAD \ "<html><meta charset=\"utf-8\">" \ "<meta http-equiv=\"Content-Security-Policy\" " \ " content=\"object-src 'none'; script-src 'none' 'unsafe-eval'\">" \ "<body><table>" #define VIEW_CACHE_TAIL \ "</table></body></html>" namespace net { namespace { std::string FormatEntryInfo(disk_cache::Entry* entry, const std::string& url_prefix) { std::string key = entry->GetKey(); GURL url = GURL(url_prefix + key); std::string row = "<tr><td><a href=\"" + url.spec() + "\">" + EscapeForHTML(key) + "</a></td></tr>"; return row; } } // namespace. ViewCacheHelper::ViewCacheHelper() : context_(NULL), disk_cache_(NULL), entry_(NULL), buf_len_(0), index_(0), data_(NULL), next_state_(STATE_NONE), weak_factory_(this) { } ViewCacheHelper::~ViewCacheHelper() { if (entry_) entry_->Close(); } int ViewCacheHelper::GetEntryInfoHTML(const std::string& key, const URLRequestContext* context, std::string* out, const CompletionCallback& callback) { return GetInfoHTML(key, context, std::string(), out, callback); } int ViewCacheHelper::GetContentsHTML(const URLRequestContext* context, const std::string& url_prefix, std::string* out, const CompletionCallback& callback) { return GetInfoHTML(std::string(), context, url_prefix, out, callback); } // static void ViewCacheHelper::HexDump(const char *buf, size_t buf_len, std::string* result) { const size_t kMaxRows = 16; int offset = 0; const unsigned char *p; while (buf_len) { base::StringAppendF(result, "%08x: ", offset); offset += kMaxRows; p = (const unsigned char *) buf; size_t i; size_t row_max = std::min(kMaxRows, buf_len); // print hex codes: for (i = 0; i < row_max; ++i) base::StringAppendF(result, "%02x ", *p++); for (i = row_max; i < kMaxRows; ++i) result->append(" "); result->append(" "); // print ASCII glyphs if possible: p = (const unsigned char *) buf; for (i = 0; i < row_max; ++i, ++p) { if (*p < 0x7F && *p > 0x1F) { AppendEscapedCharForHTML(*p, result); } else { result->push_back('.'); } } result->push_back('\n'); buf += row_max; buf_len -= row_max; } } //----------------------------------------------------------------------------- int ViewCacheHelper::GetInfoHTML(const std::string& key, const URLRequestContext* context, const std::string& url_prefix, std::string* out, const CompletionCallback& callback) { DCHECK(callback_.is_null()); DCHECK(context); key_ = key; context_ = context; url_prefix_ = url_prefix; data_ = out; next_state_ = STATE_GET_BACKEND; int rv = DoLoop(OK); if (rv == ERR_IO_PENDING) callback_ = callback; return rv; } void ViewCacheHelper::DoCallback(int rv) { DCHECK_NE(ERR_IO_PENDING, rv); DCHECK(!callback_.is_null()); callback_.Run(rv); callback_.Reset(); } void ViewCacheHelper::HandleResult(int rv) { DCHECK_NE(ERR_IO_PENDING, rv); DCHECK_NE(ERR_FAILED, rv); context_ = NULL; if (!callback_.is_null()) DoCallback(rv); } int ViewCacheHelper::DoLoop(int result) { DCHECK(next_state_ != STATE_NONE); int rv = result; do { State state = next_state_; next_state_ = STATE_NONE; switch (state) { case STATE_GET_BACKEND: DCHECK_EQ(OK, rv); rv = DoGetBackend(); break; case STATE_GET_BACKEND_COMPLETE: rv = DoGetBackendComplete(rv); break; case STATE_OPEN_NEXT_ENTRY: DCHECK_EQ(OK, rv); rv = DoOpenNextEntry(); break; case STATE_OPEN_NEXT_ENTRY_COMPLETE: rv = DoOpenNextEntryComplete(rv); break; case STATE_OPEN_ENTRY: DCHECK_EQ(OK, rv); rv = DoOpenEntry(); break; case STATE_OPEN_ENTRY_COMPLETE: rv = DoOpenEntryComplete(rv); break; case STATE_READ_RESPONSE: DCHECK_EQ(OK, rv); rv = DoReadResponse(); break; case STATE_READ_RESPONSE_COMPLETE: rv = DoReadResponseComplete(rv); break; case STATE_READ_DATA: DCHECK_EQ(OK, rv); rv = DoReadData(); break; case STATE_READ_DATA_COMPLETE: rv = DoReadDataComplete(rv); break; default: NOTREACHED() << "bad state"; rv = ERR_FAILED; break; } } while (rv != ERR_IO_PENDING && next_state_ != STATE_NONE); if (rv != ERR_IO_PENDING) HandleResult(rv); return rv; } int ViewCacheHelper::DoGetBackend() { next_state_ = STATE_GET_BACKEND_COMPLETE; if (!context_->http_transaction_factory()) return ERR_FAILED; HttpCache* http_cache = context_->http_transaction_factory()->GetCache(); if (!http_cache) return ERR_FAILED; return http_cache->GetBackend( &disk_cache_, base::Bind(&ViewCacheHelper::OnIOComplete, base::Unretained(this))); } int ViewCacheHelper::DoGetBackendComplete(int result) { if (result == ERR_FAILED) { data_->append("no disk cache"); return OK; } DCHECK_EQ(OK, result); if (key_.empty()) { data_->assign(VIEW_CACHE_HEAD); DCHECK(!iter_); next_state_ = STATE_OPEN_NEXT_ENTRY; return OK; } next_state_ = STATE_OPEN_ENTRY; return OK; } int ViewCacheHelper::DoOpenNextEntry() { next_state_ = STATE_OPEN_NEXT_ENTRY_COMPLETE; if (!iter_) iter_ = disk_cache_->CreateIterator(); return iter_->OpenNextEntry(&entry_, base::Bind(&ViewCacheHelper::OnIOComplete, base::Unretained(this))); } int ViewCacheHelper::DoOpenNextEntryComplete(int result) { if (result == ERR_FAILED) { data_->append(VIEW_CACHE_TAIL); return OK; } DCHECK_EQ(OK, result); data_->append(FormatEntryInfo(entry_, url_prefix_)); entry_->Close(); entry_ = NULL; next_state_ = STATE_OPEN_NEXT_ENTRY; return OK; } int ViewCacheHelper::DoOpenEntry() { next_state_ = STATE_OPEN_ENTRY_COMPLETE; return disk_cache_->OpenEntry( key_, &entry_, base::Bind(&ViewCacheHelper::OnIOComplete, base::Unretained(this))); } int ViewCacheHelper::DoOpenEntryComplete(int result) { if (result == ERR_FAILED) { data_->append("no matching cache entry for: " + EscapeForHTML(key_)); return OK; } data_->assign(VIEW_CACHE_HEAD); data_->append(EscapeForHTML(entry_->GetKey())); next_state_ = STATE_READ_RESPONSE; return OK; } int ViewCacheHelper::DoReadResponse() { next_state_ = STATE_READ_RESPONSE_COMPLETE; buf_len_ = entry_->GetDataSize(0); if (!buf_len_) return buf_len_; buf_ = new IOBuffer(buf_len_); return entry_->ReadData( 0, 0, buf_.get(), buf_len_, base::Bind(&ViewCacheHelper::OnIOComplete, weak_factory_.GetWeakPtr())); } int ViewCacheHelper::DoReadResponseComplete(int result) { if (result && result == buf_len_) { HttpResponseInfo response; bool truncated; if (HttpCache::ParseResponseInfo( buf_->data(), buf_len_, &response, &truncated) && response.headers.get()) { if (truncated) data_->append("<pre>RESPONSE_INFO_TRUNCATED</pre>"); data_->append("<hr><pre>"); data_->append(EscapeForHTML(response.headers->GetStatusLine())); data_->push_back('\n'); void* iter = NULL; std::string name, value; while (response.headers->EnumerateHeaderLines(&iter, &name, &value)) { data_->append(EscapeForHTML(name)); data_->append(": "); data_->append(EscapeForHTML(value)); data_->push_back('\n'); } data_->append("</pre>"); } } index_ = 0; next_state_ = STATE_READ_DATA; return OK; } int ViewCacheHelper::DoReadData() { data_->append("<hr><pre>"); next_state_ = STATE_READ_DATA_COMPLETE; buf_len_ = entry_->GetDataSize(index_); if (!buf_len_) return buf_len_; buf_ = new IOBuffer(buf_len_); return entry_->ReadData( index_, 0, buf_.get(), buf_len_, base::Bind(&ViewCacheHelper::OnIOComplete, weak_factory_.GetWeakPtr())); } int ViewCacheHelper::DoReadDataComplete(int result) { if (result && result == buf_len_) { HexDump(buf_->data(), buf_len_, data_); } data_->append("</pre>"); index_++; if (index_ < HttpCache::kNumCacheEntryDataIndices) { next_state_ = STATE_READ_DATA; } else { data_->append(VIEW_CACHE_TAIL); entry_->Close(); entry_ = NULL; } return OK; } void ViewCacheHelper::OnIOComplete(int result) { DoLoop(result); } } // namespace net.