// Copyright (c) 2006-2008 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 #include "net/url_request/url_request_test_job.h" #include "base/message_loop.h" #include "base/string_util.h" #include "net/base/io_buffer.h" #include "net/base/net_errors.h" #include "net/http/http_response_headers.h" #include "net/url_request/url_request.h" // This emulates the global message loop for the test URL request class, since // this is only test code, it's probably not too dangerous to have this static // object. static std::vector< scoped_refptr > pending_jobs; // static getters for known URLs GURL URLRequestTestJob::test_url_1() { return GURL("test:url1"); } GURL URLRequestTestJob::test_url_2() { return GURL("test:url2"); } GURL URLRequestTestJob::test_url_3() { return GURL("test:url3"); } GURL URLRequestTestJob::test_url_error() { return GURL("test:error"); } // static getters for known URL responses std::string URLRequestTestJob::test_data_1() { return std::string("Test One"); } std::string URLRequestTestJob::test_data_2() { return std::string("Test Two Two"); } std::string URLRequestTestJob::test_data_3() { return std::string("Test Three Three Three"); } // static getter for simple response headers std::string URLRequestTestJob::test_headers() { const char headers[] = "HTTP/1.1 200 OK\0" "Content-type: text/html\0" "\0"; return std::string(headers, arraysize(headers)); } // static getter for redirect response headers std::string URLRequestTestJob::test_redirect_headers() { const char headers[] = "HTTP/1.1 302 MOVED\0" "Location: somewhere\0" "\0"; return std::string(headers, arraysize(headers)); } // static getter for error response headers std::string URLRequestTestJob::test_error_headers() { const char headers[] = "HTTP/1.1 500 BOO HOO\0" "\0"; return std::string(headers, arraysize(headers)); } // static URLRequestJob* URLRequestTestJob::Factory(URLRequest* request, const std::string& scheme) { return new URLRequestTestJob(request); } URLRequestTestJob::URLRequestTestJob(URLRequest* request) : URLRequestJob(request), auto_advance_(false), stage_(WAITING), offset_(0), async_buf_(NULL), async_buf_size_(0) { } URLRequestTestJob::URLRequestTestJob(URLRequest* request, bool auto_advance) : URLRequestJob(request), auto_advance_(auto_advance), stage_(WAITING), offset_(0), async_buf_(NULL), async_buf_size_(0) { } URLRequestTestJob::URLRequestTestJob(URLRequest* request, const std::string& response_headers, const std::string& response_data, bool auto_advance) : URLRequestJob(request), auto_advance_(auto_advance), stage_(WAITING), response_headers_(new net::HttpResponseHeaders(response_headers)), response_data_(response_data), offset_(0), async_buf_(NULL), async_buf_size_(0) { } URLRequestTestJob::~URLRequestTestJob() { } bool URLRequestTestJob::GetMimeType(std::string* mime_type) const { DCHECK(mime_type); if (!response_headers_) return false; return response_headers_->GetMimeType(mime_type); } void URLRequestTestJob::Start() { // Start reading asynchronously so that all error reporting and data // callbacks happen as they would for network requests. MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod( this, &URLRequestTestJob::StartAsync)); } void URLRequestTestJob::StartAsync() { if (!response_headers_) { response_headers_ = new net::HttpResponseHeaders(test_headers()); if (request_->url().spec() == test_url_1().spec()) { response_data_ = test_data_1(); stage_ = DATA_AVAILABLE; // Simulate a synchronous response for this one. } else if (request_->url().spec() == test_url_2().spec()) { response_data_ = test_data_2(); } else if (request_->url().spec() == test_url_3().spec()) { response_data_ = test_data_3(); } else { // unexpected url, return error // FIXME(brettw) we may want to use WININET errors or have some more types // of errors NotifyDone(URLRequestStatus(URLRequestStatus::FAILED, net::ERR_INVALID_URL)); // FIXME(brettw): this should emulate a network error, and not just fail // initiating a connection return; } } AdvanceJob(); this->NotifyHeadersComplete(); } bool URLRequestTestJob::ReadRawData(net::IOBuffer* buf, int buf_size, int *bytes_read) { if (stage_ == WAITING) { async_buf_ = buf; async_buf_size_ = buf_size; SetStatus(URLRequestStatus(URLRequestStatus::IO_PENDING, 0)); return false; } DCHECK(bytes_read); *bytes_read = 0; if (offset_ >= static_cast(response_data_.length())) { return true; // done reading } int to_read = buf_size; if (to_read + offset_ > static_cast(response_data_.length())) to_read = static_cast(response_data_.length()) - offset_; memcpy(buf->data(), &response_data_.c_str()[offset_], to_read); offset_ += to_read; *bytes_read = to_read; return true; } void URLRequestTestJob::GetResponseInfo(net::HttpResponseInfo* info) { if (response_headers_) info->headers = response_headers_; } bool URLRequestTestJob::IsRedirectResponse(GURL* location, int* http_status_code) { if (!response_headers_) return false; std::string value; if (!response_headers_->IsRedirect(&value)) return false; *location = request_->url().Resolve(value); *http_status_code = response_headers_->response_code(); return true; } void URLRequestTestJob::Kill() { stage_ = DONE; URLRequestJob::Kill(); } void URLRequestTestJob::ProcessNextOperation() { switch (stage_) { case WAITING: stage_ = DATA_AVAILABLE; // OK if ReadRawData wasn't called yet. if (async_buf_) { int bytes_read; if (!ReadRawData(async_buf_, async_buf_size_, &bytes_read)) NOTREACHED() << "This should not return false in DATA_AVAILABLE."; SetStatus(URLRequestStatus()); // clear the io pending flag NotifyReadComplete(bytes_read); } break; case DATA_AVAILABLE: stage_ = ALL_DATA; // done sending data break; case ALL_DATA: stage_ = DONE; return; case DONE: return; default: NOTREACHED() << "Invalid stage"; return; } AdvanceJob(); } void URLRequestTestJob::AdvanceJob() { if (auto_advance_) { MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod( this, &URLRequestTestJob::ProcessNextOperation)); return; } pending_jobs.push_back(scoped_refptr(this)); } // static bool URLRequestTestJob::ProcessOnePendingMessage() { if (pending_jobs.empty()) return false; scoped_refptr next_job(pending_jobs[0]); pending_jobs.erase(pending_jobs.begin()); DCHECK(!next_job->auto_advance()); // auto_advance jobs should be in this q next_job->ProcessNextOperation(); return true; }