// 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 "config.h" #include "webkit/glue/resource_fetcher.h" #include "base/compiler_specific.h" MSVC_PUSH_WARNING_LEVEL(0); #include "FrameLoader.h" #include "FrameLoaderClient.h" #include "ResourceHandle.h" #include "ResourceRequest.h" MSVC_POP_WARNING(); #undef LOG #include "base/logging.h" #include "webkit/glue/glue_util.h" #include "net/url_request/url_request_status.h" using WebCore::ResourceError; using WebCore::ResourceHandle; using WebCore::ResourceResponse; ResourceFetcher::ResourceFetcher(const GURL& url, WebCore::Frame* frame, Delegate* d) : url_(url), delegate_(d), completed_(false) { // Can't do anything without a frame. However, delegate can be NULL (so we // can do a http request and ignore the results). DCHECK(frame); Start(frame); } ResourceFetcher::~ResourceFetcher() { if (!completed_ && loader_.get()) loader_->cancel(); loader_ = NULL; } void ResourceFetcher::Cancel() { if (!completed_) { loader_->cancel(); completed_ = true; } } void ResourceFetcher::Start(WebCore::Frame* frame) { WebCore::FrameLoader* frame_loader = frame->loader(); if (!frame_loader) { // We put this on a 0 timer so the callback happens async (consistent with // regular fetches). start_failed_timer_.reset(new StartFailedTimer(this, &ResourceFetcher::StartFailed)); start_failed_timer_->startOneShot(0); return; } WebCore::ResourceRequest request(webkit_glue::GURLToKURL(url_)); WebCore::ResourceResponse response; frame_loader->client()->dispatchWillSendRequest(NULL, 0, request, response); loader_ = ResourceHandle::create(request, this, NULL, false, false); } void ResourceFetcher::StartFailed(StartFailedTimer* timer) { didFail(NULL, ResourceError()); } ///////////////////////////////////////////////////////////////////////////// // ResourceHandleClient methods void ResourceFetcher::didReceiveResponse(ResourceHandle* resource_handle, const ResourceResponse& response) { ASSERT(!completed_); // It's safe to use the ResourceResponse copy constructor // (xmlhttprequest.cpp uses it). response_ = response; } void ResourceFetcher::didReceiveData(ResourceHandle* resource_handle, const char* data, int length, int total_length) { ASSERT(!completed_); if (length <= 0) return; data_.append(data, length); } void ResourceFetcher::didFinishLoading(ResourceHandle* resource_handle) { ASSERT(!completed_); completed_ = true; if (delegate_) delegate_->OnURLFetchComplete(response_, data_); } void ResourceFetcher::didFail(ResourceHandle* resource_handle, const ResourceError& error) { ASSERT(!completed_); completed_ = true; // Go ahead and tell our delegate that we're done. Send an empty // ResourceResponse and string. if (delegate_) delegate_->OnURLFetchComplete(ResourceResponse(), std::string()); } ///////////////////////////////////////////////////////////////////////////// // A resource fetcher with a timeout ResourceFetcherWithTimeout::ResourceFetcherWithTimeout( const GURL& url, WebCore::Frame* frame, double timeout_secs, Delegate* d) : ResourceFetcher(url, frame, d) { timeout_timer_.reset(new FetchTimer(this, &ResourceFetcherWithTimeout::TimeoutFired)); timeout_timer_->startOneShot(timeout_secs); } void ResourceFetcherWithTimeout::TimeoutFired(FetchTimer* timer) { if (!completed_) { loader_->cancel(); didFail(NULL, ResourceError()); } }