Index: chrome/browser/local_discovery/privet_url_fetcher.cc |
diff --git a/chrome/browser/local_discovery/privet_url_fetcher.cc b/chrome/browser/local_discovery/privet_url_fetcher.cc |
index 349555086c6c7c569d022307267fdb998d41b69a..d56bfe2a1855fc2a6efff1e2b76775c28ef630fa 100644 |
--- a/chrome/browser/local_discovery/privet_url_fetcher.cc |
+++ b/chrome/browser/local_discovery/privet_url_fetcher.cc |
@@ -10,6 +10,7 @@ |
#include "base/json/json_reader.h" |
#include "base/message_loop/message_loop.h" |
#include "base/rand_util.h" |
+#include "base/strings/stringprintf.h" |
#include "chrome/browser/browser_process.h" |
#include "chrome/browser/local_discovery/privet_constants.h" |
#include "content/public/browser/browser_thread.h" |
@@ -20,40 +21,80 @@ namespace local_discovery { |
namespace { |
const char kXPrivetTokenHeaderPrefix[] = "X-Privet-Token: "; |
+const char kRangeHeaderFormat[] = "Range: bytes=%d-%d"; |
const char kXPrivetEmptyToken[] = "\"\""; |
const int kPrivetMaxRetries = 20; |
const int kPrivetTimeoutOnError = 5; |
+const int kHTTPErrorCodeInvalidXPrivetToken = 418; |
+ |
+std::string MakeRangeHeader(int start, int end) { |
+ DCHECK_GE(start, 0); |
+ DCHECK_GT(end, 0); |
+ DCHECK_GT(end, start); |
+ return base::StringPrintf(kRangeHeaderFormat, start, end); |
} |
+} // namespace |
+ |
void PrivetURLFetcher::Delegate::OnNeedPrivetToken( |
PrivetURLFetcher* fetcher, |
const TokenCallback& callback) { |
OnError(fetcher, TOKEN_ERROR); |
} |
+bool PrivetURLFetcher::Delegate::OnRawData(PrivetURLFetcher* fetcher, |
+ bool response_is_file, |
+ const std::string& data_string, |
+ const base::FilePath& data_file) { |
+ return false; |
+} |
+ |
PrivetURLFetcher::PrivetURLFetcher( |
const std::string& token, |
const GURL& url, |
net::URLFetcher::RequestType request_type, |
net::URLRequestContextGetter* request_context, |
PrivetURLFetcher::Delegate* delegate) |
- : privet_access_token_(token), url_(url), request_type_(request_type), |
- request_context_(request_context), delegate_(delegate), |
- do_not_retry_on_transient_error_(false), allow_empty_privet_token_(false), |
- tries_(0), weak_factory_(this) { |
+ : privet_access_token_(token), |
+ url_(url), |
+ request_type_(request_type), |
+ request_context_(request_context), |
+ delegate_(delegate), |
+ do_not_retry_on_transient_error_(false), |
+ allow_empty_privet_token_(false), |
+ has_byte_range_(false), |
+ make_response_file_(false), |
+ byte_range_start_(0), |
+ byte_range_end_(0), |
+ tries_(0), |
+ weak_factory_(this) { |
} |
PrivetURLFetcher::~PrivetURLFetcher() { |
} |
void PrivetURLFetcher::DoNotRetryOnTransientError() { |
+ DCHECK(tries_ == 0); |
do_not_retry_on_transient_error_ = true; |
} |
void PrivetURLFetcher::AllowEmptyPrivetToken() { |
+ DCHECK(tries_ == 0); |
allow_empty_privet_token_ = true; |
} |
+void PrivetURLFetcher::SaveResponseToFile() { |
+ DCHECK(tries_ == 0); |
+ make_response_file_ = true; |
+} |
+ |
+void PrivetURLFetcher::SetByteRange(int start, int end) { |
+ DCHECK(tries_ == 0); |
+ byte_range_start_ = start; |
+ byte_range_end_ = end; |
+ has_byte_range_ = true; |
+} |
+ |
void PrivetURLFetcher::Try() { |
tries_++; |
if (tries_ < kPrivetMaxRetries) { |
@@ -66,6 +107,16 @@ void PrivetURLFetcher::Try() { |
url_fetcher_->SetRequestContext(request_context_); |
url_fetcher_->AddExtraRequestHeader(std::string(kXPrivetTokenHeaderPrefix) + |
token); |
+ if (has_byte_range_) { |
+ url_fetcher_->AddExtraRequestHeader( |
+ MakeRangeHeader(byte_range_start_, byte_range_end_)); |
+ } |
+ |
+ if (make_response_file_) { |
+ url_fetcher_->SaveResponseToTemporaryFile( |
+ content::BrowserThread::GetMessageLoopProxyForThread( |
+ content::BrowserThread::FILE)); |
+ } |
// URLFetcher requires us to set upload data for POST requests. |
if (request_type_ == net::URLFetcher::POST) { |
@@ -119,6 +170,56 @@ void PrivetURLFetcher::OnURLFetchComplete(const net::URLFetcher* source) { |
return; |
} |
+ if (!OnURLFetchCompleteDoNotParseData(source)) { |
+ // Byte ranges should only be used when we're not parsing the data |
+ // as JSON. |
+ DCHECK(!has_byte_range_); |
+ |
+ // We should only be saving raw data to a file. |
+ DCHECK(!make_response_file_); |
+ |
+ OnURLFetchCompleteParseData(source); |
+ } |
+} |
+ |
+// Note that this function returns "true" in error cases to indicate |
+// that it has fully handled the responses. |
+bool PrivetURLFetcher::OnURLFetchCompleteDoNotParseData( |
+ const net::URLFetcher* source) { |
+ if (source->GetResponseCode() == kHTTPErrorCodeInvalidXPrivetToken) { |
+ RequestTokenRefresh(); |
+ return true; |
+ } |
+ |
+ if (source->GetResponseCode() != net::HTTP_OK && |
+ source->GetResponseCode() != net::HTTP_PARTIAL_CONTENT) { |
+ delegate_->OnError(this, RESPONSE_CODE_ERROR); |
+ return true; |
+ } |
+ |
+ if (make_response_file_) { |
+ base::FilePath response_file_path; |
+ |
+ if (!source->GetResponseAsFilePath(true, &response_file_path)) { |
+ delegate_->OnError(this, URL_FETCH_ERROR); |
+ return true; |
+ } |
+ |
+ return delegate_->OnRawData(this, true, std::string(), response_file_path); |
+ } else { |
+ std::string response_str; |
+ |
+ if (!source->GetResponseAsString(&response_str)) { |
+ delegate_->OnError(this, URL_FETCH_ERROR); |
+ return true; |
+ } |
+ |
+ return delegate_->OnRawData(this, false, response_str, base::FilePath()); |
+ } |
+} |
+ |
+void PrivetURLFetcher::OnURLFetchCompleteParseData( |
+ const net::URLFetcher* source) { |
if (source->GetResponseCode() != net::HTTP_OK) { |
delegate_->OnError(this, RESPONSE_CODE_ERROR); |
return; |