summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--webkit/appcache/appcache_request_handler.cc10
-rw-r--r--webkit/appcache/appcache_request_handler_unittest.cc105
2 files changed, 109 insertions, 6 deletions
diff --git a/webkit/appcache/appcache_request_handler.cc b/webkit/appcache/appcache_request_handler.cc
index 064b378..a90d6d7 100644
--- a/webkit/appcache/appcache_request_handler.cc
+++ b/webkit/appcache/appcache_request_handler.cc
@@ -137,6 +137,16 @@ AppCacheURLRequestJob* AppCacheRequestHandler::MaybeLoadFallbackForResponse(
int code_major = request->GetResponseCode() / 100;
if (code_major !=4 && code_major != 5)
return NULL;
+
+ // Servers can override the fallback behavior with a response header.
+ const std::string kFallbackOverrideHeader(
+ "x-chromium-appcache-fallback-override");
+ const std::string kFallbackOverrideValue(
+ "disallow-fallback");
+ std::string header_value;
+ request->GetResponseHeaderByName(kFallbackOverrideHeader, &header_value);
+ if (header_value == kFallbackOverrideValue)
+ return NULL;
}
// 6.9.6, step 4: If this results in a 4xx or 5xx status code
diff --git a/webkit/appcache/appcache_request_handler_unittest.cc b/webkit/appcache/appcache_request_handler_unittest.cc
index c9fbe0e..a40d294 100644
--- a/webkit/appcache/appcache_request_handler_unittest.cc
+++ b/webkit/appcache/appcache_request_handler_unittest.cc
@@ -10,6 +10,7 @@
#include "base/threading/thread.h"
#include "base/synchronization/waitable_event.h"
#include "net/base/net_errors.h"
+#include "net/http/http_response_headers.h"
#include "net/url_request/url_request.h"
#include "net/url_request/url_request_error_job.h"
#include "testing/gtest/include/gtest/gtest.h"
@@ -68,16 +69,41 @@ class AppCacheRequestHandlerTest : public testing::Test {
Method method_;
};
- // Subclasses to simulate particular response codes so test cases can
+ // Subclasses to simulate particular responses so test cases can
// exercise fallback code paths.
+ class MockURLRequestDelegate : public net::URLRequest::Delegate {
+ virtual void OnResponseStarted(net::URLRequest* request) {}
+ virtual void OnReadCompleted(net::URLRequest* request, int bytes_read) {}
+ };
+
class MockURLRequestJob : public net::URLRequestJob {
public:
- MockURLRequestJob(net::URLRequest* request, int response_code)
- : net::URLRequestJob(request), response_code_(response_code) {}
- virtual void Start() {}
- virtual int GetResponseCode() const { return response_code_; }
+ MockURLRequestJob(
+ net::URLRequest* request, int response_code)
+ : net::URLRequestJob(request),
+ response_code_(response_code),
+ has_response_info_(false) {}
+ MockURLRequestJob(
+ net::URLRequest* request, const net::HttpResponseInfo& info)
+ : net::URLRequestJob(request),
+ response_code_(info.headers->response_code()),
+ has_response_info_(true),
+ response_info_(info) {}
+ virtual void Start() {
+ NotifyHeadersComplete();
+ }
+ virtual int GetResponseCode() const {
+ return response_code_;
+ }
+ virtual void GetResponseInfo(net::HttpResponseInfo* info) {
+ if (!has_response_info_)
+ return;
+ *info = response_info_;
+ }
int response_code_;
+ bool has_response_info_;
+ net::HttpResponseInfo response_info_;
};
class MockURLRequest : public net::URLRequest {
@@ -88,10 +114,19 @@ class AppCacheRequestHandlerTest : public testing::Test {
mock_factory_job_ = new MockURLRequestJob(this, http_response_code);
Start();
DCHECK(!mock_factory_job_);
- // All our simulation need to do satisfy are the following two DCHECKs
+ // All our simulation needs to satisfy are the following two DCHECKs
DCHECK(status().is_success());
DCHECK_EQ(http_response_code, GetResponseCode());
}
+
+ void SimulateResponseInfo(const net::HttpResponseInfo& info) {
+ mock_factory_job_ = new MockURLRequestJob(this, info);
+ set_delegate(&delegate_); // needed to get the info back out
+ Start();
+ DCHECK(!mock_factory_job_);
+ }
+
+ MockURLRequestDelegate delegate_;
};
static net::URLRequestJob* MockHttpJobFactory(net::URLRequest* request,
@@ -328,6 +363,59 @@ class AppCacheRequestHandlerTest : public testing::Test {
TestFinished();
}
+ // MainResource_FallbackOverride --------------------------------------------
+
+ void MainResource_FallbackOverride() {
+ PushNextTask(NewRunnableMethod(
+ this,
+ &AppCacheRequestHandlerTest::Verify_MainResource_FallbackOverride));
+
+ request_.reset(new MockURLRequest(GURL("http://blah/fallback-override")));
+ handler_.reset(host_->CreateRequestHandler(request_.get(),
+ ResourceType::MAIN_FRAME));
+ EXPECT_TRUE(handler_.get());
+
+ mock_storage()->SimulateFindMainResource(
+ AppCacheEntry(),
+ GURL("http://blah/fallbackurl"),
+ AppCacheEntry(AppCacheEntry::EXPLICIT, 1),
+ 1, GURL("http://blah/manifest/"));
+
+ job_ = handler_->MaybeLoadResource(request_.get());
+ EXPECT_TRUE(job_.get());
+ EXPECT_TRUE(job_->is_waiting());
+
+ // We have to wait for completion of storage->FindResponseForMainRequest.
+ ScheduleNextTask();
+ }
+
+ void Verify_MainResource_FallbackOverride() {
+ EXPECT_FALSE(job_->is_waiting());
+ EXPECT_TRUE(job_->is_delivering_network_response());
+
+ // When the request is restarted, the existing job is dropped so a
+ // real network job gets created. We expect NULL here which will cause
+ // the net library to create a real job.
+ job_ = handler_->MaybeLoadResource(request_.get());
+ EXPECT_FALSE(job_);
+
+ // Simulate an http error of the real network job, but with custom
+ // headers that override the fallback behavior.
+ const char kOverrideHeaders[] =
+ "HTTP/1.1 404 BOO HOO\0"
+ "x-chromium-appcache-fallback-override: disallow-fallback\0"
+ "\0";
+ net::HttpResponseInfo info;
+ info.headers = new net::HttpResponseHeaders(
+ std::string(kOverrideHeaders, arraysize(kOverrideHeaders)));
+ request_->SimulateResponseInfo(info);
+
+ job_ = handler_->MaybeLoadFallbackForResponse(request_.get());
+ EXPECT_FALSE(job_);
+
+ TestFinished();
+ }
+
// SubResource_Miss_WithNoCacheSelected ----------------------------------
void SubResource_Miss_WithNoCacheSelected() {
@@ -697,6 +785,11 @@ TEST_F(AppCacheRequestHandlerTest, MainResource_Fallback) {
RunTestOnIOThread(&AppCacheRequestHandlerTest::MainResource_Fallback);
}
+TEST_F(AppCacheRequestHandlerTest, MainResource_FallbackOverride) {
+ RunTestOnIOThread(
+ &AppCacheRequestHandlerTest::MainResource_FallbackOverride);
+}
+
TEST_F(AppCacheRequestHandlerTest, SubResource_Miss_WithNoCacheSelected) {
RunTestOnIOThread(
&AppCacheRequestHandlerTest::SubResource_Miss_WithNoCacheSelected);