// Copyright (c) 2012 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. #ifndef NET_BASE_UPLOAD_FILE_ELEMENT_READER_H_ #define NET_BASE_UPLOAD_FILE_ELEMENT_READER_H_ #include #include #include "base/compiler_specific.h" #include "base/files/file.h" #include "base/files/file_path.h" #include "base/gtest_prod_util.h" #include "base/macros.h" #include "base/memory/ref_counted.h" #include "base/memory/weak_ptr.h" #include "base/time/time.h" #include "net/base/net_export.h" #include "net/base/upload_element_reader.h" namespace base { class TaskRunner; } namespace net { class FileStream; // An UploadElementReader implementation for file. class NET_EXPORT UploadFileElementReader : public UploadElementReader { public: // |file| must be valid and opened for reading. On Windows, the file must have // been opened with File::FLAG_ASYNC, and elsewhere it must have ben opened // without it. |path| is never validated or used to re-open the file. It's // only used as the return value for path(). // |task_runner| is used to perform file operations. It must not be NULL. // // TODO(mmenke): Remove |task_runner| argument, and use the TaskScheduler // instead. UploadFileElementReader(base::TaskRunner* task_runner, base::File file, const base::FilePath& path, uint64_t range_offset, uint64_t range_length, const base::Time& expected_modification_time); // Same a above, but takes a FilePath instead. // TODO(mmenke): Remove if all consumers can be switched to the first // constructor. UploadFileElementReader(base::TaskRunner* task_runner, const base::FilePath& path, uint64_t range_offset, uint64_t range_length, const base::Time& expected_modification_time); ~UploadFileElementReader() override; const base::FilePath& path() const { return path_; } uint64_t range_offset() const { return range_offset_; } uint64_t range_length() const { return range_length_; } const base::Time& expected_modification_time() const { return expected_modification_time_; } // UploadElementReader overrides: const UploadFileElementReader* AsFileReader() const override; int Init(const CompletionCallback& callback) override; uint64_t GetContentLength() const override; uint64_t BytesRemaining() const override; int Read(IOBuffer* buf, int buf_length, const CompletionCallback& callback) override; private: enum class State { // No async operation is pending. IDLE, // The ordered sequence of events started by calling Init(). // Opens file. State is skipped if file already open. OPEN, OPEN_COMPLETE, SEEK, GET_FILE_INFO, GET_FILE_INFO_COMPLETE, // There is no READ state as reads are always started immediately on Read(). READ_COMPLETE, }; FRIEND_TEST_ALL_PREFIXES(ElementsUploadDataStreamTest, FileSmallerThanLength); FRIEND_TEST_ALL_PREFIXES(HttpNetworkTransactionTest, UploadFileSmallerThanLength); int DoLoop(int result); int DoOpen(); int DoOpenComplete(int result); int DoSeek(); int DoGetFileInfo(int result); int DoGetFileInfoComplete(int result); int DoReadComplete(int result); void OnIOComplete(int result); // Sets an value to override the result for GetContentLength(). // Used for tests. struct NET_EXPORT_PRIVATE ScopedOverridingContentLengthForTests { explicit ScopedOverridingContentLengthForTests(uint64_t value); ~ScopedOverridingContentLengthForTests(); }; scoped_refptr task_runner_; const base::FilePath path_; const uint64_t range_offset_; const uint64_t range_length_; const base::Time expected_modification_time_; std::unique_ptr file_stream_; uint64_t content_length_; uint64_t bytes_remaining_; // File information. Only valid during GET_FILE_INFO_COMPLETE state. base::File::Info file_info_; State next_state_; CompletionCallback pending_callback_; // True if Init() was called while an async operation was in progress. bool init_called_while_operation_pending_; base::WeakPtrFactory weak_ptr_factory_; DISALLOW_COPY_AND_ASSIGN(UploadFileElementReader); }; } // namespace net #endif // NET_BASE_UPLOAD_FILE_ELEMENT_READER_H_