- add sources.
[platform/framework/web/crosswalk.git] / src / webkit / browser / fileapi / file_system_url_request_job.cc
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "webkit/browser/fileapi/file_system_url_request_job.h"
6
7 #include <vector>
8
9 #include "base/bind.h"
10 #include "base/compiler_specific.h"
11 #include "base/files/file_path.h"
12 #include "base/files/file_util_proxy.h"
13 #include "base/message_loop/message_loop.h"
14 #include "base/message_loop/message_loop_proxy.h"
15 #include "base/platform_file.h"
16 #include "base/threading/thread_restrictions.h"
17 #include "base/time/time.h"
18 #include "build/build_config.h"
19 #include "net/base/file_stream.h"
20 #include "net/base/io_buffer.h"
21 #include "net/base/mime_util.h"
22 #include "net/base/net_errors.h"
23 #include "net/base/net_util.h"
24 #include "net/http/http_response_headers.h"
25 #include "net/http/http_response_info.h"
26 #include "net/http/http_util.h"
27 #include "net/url_request/url_request.h"
28 #include "url/gurl.h"
29 #include "webkit/browser/blob/file_stream_reader.h"
30 #include "webkit/browser/fileapi/file_system_context.h"
31 #include "webkit/browser/fileapi/file_system_operation_runner.h"
32 #include "webkit/common/fileapi/file_system_util.h"
33
34 using net::NetworkDelegate;
35 using net::URLRequest;
36 using net::URLRequestJob;
37 using net::URLRequestStatus;
38
39 namespace fileapi {
40
41 static net::HttpResponseHeaders* CreateHttpResponseHeaders() {
42   // HttpResponseHeaders expects its input string to be terminated by two NULs.
43   static const char kStatus[] = "HTTP/1.1 200 OK\0";
44   static const size_t kStatusLen = arraysize(kStatus);
45
46   net::HttpResponseHeaders* headers =
47       new net::HttpResponseHeaders(std::string(kStatus, kStatusLen));
48
49   // Tell WebKit never to cache this content.
50   std::string cache_control(net::HttpRequestHeaders::kCacheControl);
51   cache_control.append(": no-cache");
52   headers->AddHeader(cache_control);
53
54   return headers;
55 }
56
57 FileSystemURLRequestJob::FileSystemURLRequestJob(
58     URLRequest* request,
59     NetworkDelegate* network_delegate,
60     FileSystemContext* file_system_context)
61     : URLRequestJob(request, network_delegate),
62       file_system_context_(file_system_context),
63       is_directory_(false),
64       remaining_bytes_(0),
65       weak_factory_(this) {
66 }
67
68 FileSystemURLRequestJob::~FileSystemURLRequestJob() {}
69
70 void FileSystemURLRequestJob::Start() {
71   base::MessageLoop::current()->PostTask(
72       FROM_HERE,
73       base::Bind(&FileSystemURLRequestJob::StartAsync,
74                  weak_factory_.GetWeakPtr()));
75 }
76
77 void FileSystemURLRequestJob::Kill() {
78   reader_.reset();
79   URLRequestJob::Kill();
80   weak_factory_.InvalidateWeakPtrs();
81 }
82
83 bool FileSystemURLRequestJob::ReadRawData(net::IOBuffer* dest, int dest_size,
84                                           int *bytes_read) {
85   DCHECK_NE(dest_size, 0);
86   DCHECK(bytes_read);
87   DCHECK_GE(remaining_bytes_, 0);
88
89   if (reader_.get() == NULL)
90     return false;
91
92   if (remaining_bytes_ < dest_size)
93     dest_size = static_cast<int>(remaining_bytes_);
94
95   if (!dest_size) {
96     *bytes_read = 0;
97     return true;
98   }
99
100   const int rv = reader_->Read(dest, dest_size,
101                                base::Bind(&FileSystemURLRequestJob::DidRead,
102                                           weak_factory_.GetWeakPtr()));
103   if (rv >= 0) {
104     // Data is immediately available.
105     *bytes_read = rv;
106     remaining_bytes_ -= rv;
107     DCHECK_GE(remaining_bytes_, 0);
108     return true;
109   }
110   if (rv == net::ERR_IO_PENDING)
111     SetStatus(URLRequestStatus(URLRequestStatus::IO_PENDING, 0));
112   else
113     NotifyFailed(rv);
114   return false;
115 }
116
117 bool FileSystemURLRequestJob::GetMimeType(std::string* mime_type) const {
118   DCHECK(request_);
119   DCHECK(url_.is_valid());
120   base::FilePath::StringType extension = url_.path().Extension();
121   if (!extension.empty())
122     extension = extension.substr(1);
123   return net::GetWellKnownMimeTypeFromExtension(extension, mime_type);
124 }
125
126 void FileSystemURLRequestJob::SetExtraRequestHeaders(
127     const net::HttpRequestHeaders& headers) {
128   std::string range_header;
129   if (headers.GetHeader(net::HttpRequestHeaders::kRange, &range_header)) {
130     std::vector<net::HttpByteRange> ranges;
131     if (net::HttpUtil::ParseRangeHeader(range_header, &ranges)) {
132       if (ranges.size() == 1) {
133         byte_range_ = ranges[0];
134       } else {
135         // We don't support multiple range requests in one single URL request.
136         // TODO(adamk): decide whether we want to support multiple range
137         // requests.
138         NotifyFailed(net::ERR_REQUEST_RANGE_NOT_SATISFIABLE);
139       }
140     }
141   }
142 }
143
144 void FileSystemURLRequestJob::GetResponseInfo(net::HttpResponseInfo* info) {
145   if (response_info_)
146     *info = *response_info_;
147 }
148
149 int FileSystemURLRequestJob::GetResponseCode() const {
150   if (response_info_)
151     return 200;
152   return URLRequestJob::GetResponseCode();
153 }
154
155 void FileSystemURLRequestJob::StartAsync() {
156   if (!request_)
157     return;
158   DCHECK(!reader_.get());
159   url_ = file_system_context_->CrackURL(request_->url());
160   if (!file_system_context_->CanServeURLRequest(url_)) {
161     // In incognito mode the API is not usable and there should be no data.
162     NotifyFailed(net::ERR_FILE_NOT_FOUND);
163     return;
164   }
165   file_system_context_->operation_runner()->GetMetadata(
166       url_,
167       base::Bind(&FileSystemURLRequestJob::DidGetMetadata,
168                  weak_factory_.GetWeakPtr()));
169 }
170
171 void FileSystemURLRequestJob::DidGetMetadata(
172     base::PlatformFileError error_code,
173     const base::PlatformFileInfo& file_info) {
174   if (error_code != base::PLATFORM_FILE_OK) {
175     NotifyFailed(error_code == base::PLATFORM_FILE_ERROR_INVALID_URL ?
176                  net::ERR_INVALID_URL : net::ERR_FILE_NOT_FOUND);
177     return;
178   }
179
180   // We may have been orphaned...
181   if (!request_)
182     return;
183
184   is_directory_ = file_info.is_directory;
185
186   if (!byte_range_.ComputeBounds(file_info.size)) {
187     NotifyFailed(net::ERR_REQUEST_RANGE_NOT_SATISFIABLE);
188     return;
189   }
190
191   if (is_directory_) {
192     NotifyHeadersComplete();
193     return;
194   }
195
196   remaining_bytes_ = byte_range_.last_byte_position() -
197                      byte_range_.first_byte_position() + 1;
198   DCHECK_GE(remaining_bytes_, 0);
199
200   DCHECK(!reader_.get());
201   reader_ = file_system_context_->CreateFileStreamReader(
202       url_, byte_range_.first_byte_position(), base::Time());
203
204   set_expected_content_size(remaining_bytes_);
205   response_info_.reset(new net::HttpResponseInfo());
206   response_info_->headers = CreateHttpResponseHeaders();
207   NotifyHeadersComplete();
208 }
209
210 void FileSystemURLRequestJob::DidRead(int result) {
211   if (result > 0)
212     SetStatus(URLRequestStatus());  // Clear the IO_PENDING status
213   else if (result == 0)
214     NotifyDone(URLRequestStatus());
215   else
216     NotifyFailed(result);
217
218   remaining_bytes_ -= result;
219   DCHECK_GE(remaining_bytes_, 0);
220
221   NotifyReadComplete(result);
222 }
223
224 bool FileSystemURLRequestJob::IsRedirectResponse(GURL* location,
225                                                  int* http_status_code) {
226   if (is_directory_) {
227     // This happens when we discovered the file is a directory, so needs a
228     // slash at the end of the path.
229     std::string new_path = request_->url().path();
230     new_path.push_back('/');
231     GURL::Replacements replacements;
232     replacements.SetPathStr(new_path);
233     *location = request_->url().ReplaceComponents(replacements);
234     *http_status_code = 301;  // simulate a permanent redirect
235     return true;
236   }
237
238   return false;
239 }
240
241 void FileSystemURLRequestJob::NotifyFailed(int rv) {
242   NotifyDone(URLRequestStatus(URLRequestStatus::FAILED, rv));
243 }
244
245 }  // namespace fileapi