|  | // Copyright 2014 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 CONTENT_BROWSER_CACHE_STORAGE_CACHE_STORAGE_CACHE_H_ | 
|  | #define CONTENT_BROWSER_CACHE_STORAGE_CACHE_STORAGE_CACHE_H_ | 
|  |  | 
|  | #include <stdint.h> | 
|  |  | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/callback.h" | 
|  | #include "base/files/file_path.h" | 
|  | #include "base/id_map.h" | 
|  | #include "base/macros.h" | 
|  | #include "base/memory/ref_counted.h" | 
|  | #include "base/memory/weak_ptr.h" | 
|  | #include "content/common/cache_storage/cache_storage_types.h" | 
|  | #include "content/common/service_worker/service_worker_types.h" | 
|  | #include "net/base/io_buffer.h" | 
|  | #include "net/disk_cache/disk_cache.h" | 
|  | #include "storage/common/quota/quota_status_code.h" | 
|  |  | 
|  | namespace net { | 
|  | class URLRequestContextGetter; | 
|  | class IOBufferWithSize; | 
|  | } | 
|  |  | 
|  | namespace storage { | 
|  | class BlobDataHandle; | 
|  | class BlobStorageContext; | 
|  | class QuotaManagerProxy; | 
|  | } | 
|  |  | 
|  | namespace content { | 
|  |  | 
|  | class CacheStorageBlobToDiskCache; | 
|  | class CacheMetadata; | 
|  | class CacheStorageScheduler; | 
|  | class TestCacheStorageCache; | 
|  |  | 
|  | // Represents a ServiceWorker Cache as seen in | 
|  | // https://slightlyoff.github.io/ServiceWorker/spec/service_worker/ The | 
|  | // asynchronous methods are executed serially (except for Size). Callbacks to | 
|  | // the public functions will be called so long as the cache object lives. | 
|  | class CONTENT_EXPORT CacheStorageCache | 
|  | : public base::RefCounted<CacheStorageCache> { | 
|  | public: | 
|  | using ErrorCallback = base::Callback<void(CacheStorageError)>; | 
|  | using ResponseCallback = | 
|  | base::Callback<void(CacheStorageError, | 
|  | std::unique_ptr<ServiceWorkerResponse>, | 
|  | std::unique_ptr<storage::BlobDataHandle>)>; | 
|  | using Responses = std::vector<ServiceWorkerResponse>; | 
|  | using BlobDataHandles = std::vector<storage::BlobDataHandle>; | 
|  | using ResponsesCallback = | 
|  | base::Callback<void(CacheStorageError, | 
|  | std::unique_ptr<Responses>, | 
|  | std::unique_ptr<BlobDataHandles>)>; | 
|  | using Requests = std::vector<ServiceWorkerFetchRequest>; | 
|  | using RequestsCallback = | 
|  | base::Callback<void(CacheStorageError, std::unique_ptr<Requests>)>; | 
|  | using SizeCallback = base::Callback<void(int64_t)>; | 
|  |  | 
|  | enum EntryIndex { INDEX_HEADERS = 0, INDEX_RESPONSE_BODY, INDEX_SIDE_DATA }; | 
|  |  | 
|  | static scoped_refptr<CacheStorageCache> CreateMemoryCache( | 
|  | const GURL& origin, | 
|  | const std::string& cache_name, | 
|  | scoped_refptr<net::URLRequestContextGetter> request_context_getter, | 
|  | scoped_refptr<storage::QuotaManagerProxy> quota_manager_proxy, | 
|  | base::WeakPtr<storage::BlobStorageContext> blob_context); | 
|  | static scoped_refptr<CacheStorageCache> CreatePersistentCache( | 
|  | const GURL& origin, | 
|  | const std::string& cache_name, | 
|  | const base::FilePath& path, | 
|  | scoped_refptr<net::URLRequestContextGetter> request_context_getter, | 
|  | scoped_refptr<storage::QuotaManagerProxy> quota_manager_proxy, | 
|  | base::WeakPtr<storage::BlobStorageContext> blob_context); | 
|  |  | 
|  | // Returns ERROR_TYPE_NOT_FOUND if not found. | 
|  | void Match(std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const ResponseCallback& callback); | 
|  |  | 
|  | // Returns CACHE_STORAGE_OK and matched responses in this cache. If there are | 
|  | // no responses, returns CACHE_STORAGE_OK and an empty vector. | 
|  | void MatchAll(std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const CacheStorageCacheQueryParams& match_params, | 
|  | const ResponsesCallback& callback); | 
|  |  | 
|  | // Writes the side data (ex: V8 code cache) for the specified cache entry. | 
|  | // If it doesn't exist, or the |expected_response_time| differs from the | 
|  | // entry's, CACHE_STORAGE_ERROR_NOT_FOUND is returned. | 
|  | // Note: This "side data" is same meaning as "metadata" in HTTPCache. We use | 
|  | // "metadata" in cache_storage.proto for the pair of headers of a request and | 
|  | // a response. To avoid the confusion we use "side data" here. | 
|  | void WriteSideData(const CacheStorageCache::ErrorCallback& callback, | 
|  | const GURL& url, | 
|  | base::Time expected_response_time, | 
|  | scoped_refptr<net::IOBuffer> buffer, | 
|  | int buf_len); | 
|  |  | 
|  | // Runs given batch operations. This corresponds to the Batch Cache Operations | 
|  | // algorithm in the spec. | 
|  | // | 
|  | // |operations| cannot mix PUT and DELETE operations and cannot contain | 
|  | // multiple DELETE operations. | 
|  | // | 
|  | // In the case of the PUT operation, puts request and response objects in the | 
|  | // cache and returns OK when all operations are successfully completed. | 
|  | // In the case of the DELETE operation, returns ERROR_NOT_FOUND if a specified | 
|  | // entry is not found. Otherwise deletes it and returns OK. | 
|  | // | 
|  | // TODO(nhiroki): This function should run all operations atomically. | 
|  | // http://crbug.com/486637 | 
|  | void BatchOperation(const std::vector<CacheStorageBatchOperation>& operations, | 
|  | const ErrorCallback& callback); | 
|  | void BatchDidOneOperation(const base::Closure& barrier_closure, | 
|  | ErrorCallback* callback, | 
|  | CacheStorageError error); | 
|  | void BatchDidAllOperations(std::unique_ptr<ErrorCallback> callback); | 
|  |  | 
|  | // TODO(jkarlin): Have keys take an optional ServiceWorkerFetchRequest. | 
|  | // Returns CACHE_STORAGE_OK and a vector of requests if there are no errors. | 
|  | void Keys(const RequestsCallback& callback); | 
|  |  | 
|  | // Closes the backend. Future operations that require the backend | 
|  | // will exit early. Close should only be called once per CacheStorageCache. | 
|  | void Close(const base::Closure& callback); | 
|  |  | 
|  | // The size of the cache's contents. This runs in parallel with other Cache | 
|  | // operations. This is because QuotaManager is a dependency of the Put | 
|  | // operation and QuotaManager calls Size. If the cache isn't yet initialized, | 
|  | // runs immediately after initialization, before any pending operations in the | 
|  | // scheduler are run. | 
|  | void Size(const SizeCallback& callback); | 
|  |  | 
|  | // Gets the cache's size, closes the backend, and then runs |callback| with | 
|  | // the cache's size. | 
|  | void GetSizeThenClose(const SizeCallback& callback); | 
|  |  | 
|  | base::FilePath path() const { return path_; } | 
|  |  | 
|  | base::WeakPtr<CacheStorageCache> AsWeakPtr(); | 
|  |  | 
|  | private: | 
|  | friend class base::RefCounted<CacheStorageCache>; | 
|  | friend class TestCacheStorageCache; | 
|  |  | 
|  | struct OpenAllEntriesContext; | 
|  | struct MatchAllContext; | 
|  | struct KeysContext; | 
|  | struct PutContext; | 
|  |  | 
|  | // The backend progresses from uninitialized, to open, to closed, and cannot | 
|  | // reverse direction.  The open step may be skipped. | 
|  | enum BackendState { | 
|  | BACKEND_UNINITIALIZED,  // No backend, create backend on first operation. | 
|  | BACKEND_OPEN,           // Backend can be used. | 
|  | BACKEND_CLOSED          // Backend cannot be used.  All ops should fail. | 
|  | }; | 
|  |  | 
|  | using Entries = std::vector<disk_cache::Entry*>; | 
|  | using ScopedBackendPtr = std::unique_ptr<disk_cache::Backend>; | 
|  | using BlobToDiskCacheIDMap = | 
|  | IDMap<CacheStorageBlobToDiskCache, IDMapOwnPointer>; | 
|  | using OpenAllEntriesCallback = | 
|  | base::Callback<void(std::unique_ptr<OpenAllEntriesContext>, | 
|  | CacheStorageError)>; | 
|  |  | 
|  | CacheStorageCache( | 
|  | const GURL& origin, | 
|  | const std::string& cache_name, | 
|  | const base::FilePath& path, | 
|  | scoped_refptr<net::URLRequestContextGetter> request_context_getter, | 
|  | scoped_refptr<storage::QuotaManagerProxy> quota_manager_proxy, | 
|  | base::WeakPtr<storage::BlobStorageContext> blob_context); | 
|  |  | 
|  | // Async operations in progress will cancel and not run their callbacks. | 
|  | virtual ~CacheStorageCache(); | 
|  |  | 
|  | // Returns true if the backend is ready to operate. | 
|  | bool LazyInitialize(); | 
|  |  | 
|  | // Returns all entries in this cache. | 
|  | void OpenAllEntries(const OpenAllEntriesCallback& callback); | 
|  | void DidOpenNextEntry(std::unique_ptr<OpenAllEntriesContext> entries_context, | 
|  | const OpenAllEntriesCallback& callback, | 
|  | int rv); | 
|  |  | 
|  | // Match callbacks | 
|  | void MatchImpl(std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const ResponseCallback& callback); | 
|  | void MatchDidOpenEntry(std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const ResponseCallback& callback, | 
|  | std::unique_ptr<disk_cache::Entry*> entry_ptr, | 
|  | int rv); | 
|  | void MatchDidReadMetadata(std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const ResponseCallback& callback, | 
|  | disk_cache::ScopedEntryPtr entry, | 
|  | std::unique_ptr<CacheMetadata> headers); | 
|  |  | 
|  | // MatchAll callbacks | 
|  | void MatchAllImpl(std::unique_ptr<MatchAllContext> context); | 
|  | void MatchAllDidOpenAllEntries( | 
|  | std::unique_ptr<MatchAllContext> context, | 
|  | std::unique_ptr<OpenAllEntriesContext> entries_context, | 
|  | CacheStorageError error); | 
|  | void MatchAllProcessNextEntry(std::unique_ptr<MatchAllContext> context, | 
|  | const Entries::iterator& iter); | 
|  | void MatchAllDidReadMetadata(std::unique_ptr<MatchAllContext> context, | 
|  | const Entries::iterator& iter, | 
|  | std::unique_ptr<CacheMetadata> metadata); | 
|  |  | 
|  | // WriteSideData callbacks | 
|  | void WriteSideDataImpl(const ErrorCallback& callback, | 
|  | const GURL& url, | 
|  | base::Time expected_response_time, | 
|  | scoped_refptr<net::IOBuffer> buffer, | 
|  | int buf_len); | 
|  | void WriteSideDataDidGetUsageAndQuota(const ErrorCallback& callback, | 
|  | const GURL& url, | 
|  | base::Time expected_response_time, | 
|  | scoped_refptr<net::IOBuffer> buffer, | 
|  | int buf_len, | 
|  | storage::QuotaStatusCode status_code, | 
|  | int64_t usage, | 
|  | int64_t quota); | 
|  | void WriteSideDataDidOpenEntry(const ErrorCallback& callback, | 
|  | base::Time expected_response_time, | 
|  | scoped_refptr<net::IOBuffer> buffer, | 
|  | int buf_len, | 
|  | std::unique_ptr<disk_cache::Entry*> entry_ptr, | 
|  | int rv); | 
|  | void WriteSideDataDidReadMetaData(const ErrorCallback& callback, | 
|  | base::Time expected_response_time, | 
|  | scoped_refptr<net::IOBuffer> buffer, | 
|  | int buf_len, | 
|  | disk_cache::ScopedEntryPtr entry, | 
|  | std::unique_ptr<CacheMetadata> headers); | 
|  | void WriteSideDataDidWrite(const ErrorCallback& callback, | 
|  | disk_cache::ScopedEntryPtr entry, | 
|  | int expected_bytes, | 
|  | int rv); | 
|  |  | 
|  | // Puts the request and response object in the cache. The response body (if | 
|  | // present) is stored in the cache, but not the request body. Returns OK on | 
|  | // success. | 
|  | void Put(const CacheStorageBatchOperation& operation, | 
|  | const ErrorCallback& callback); | 
|  | void PutImpl(std::unique_ptr<PutContext> put_context); | 
|  | void PutDidDelete(std::unique_ptr<PutContext> put_context, | 
|  | CacheStorageError delete_error); | 
|  | void PutDidGetUsageAndQuota(std::unique_ptr<PutContext> put_context, | 
|  | storage::QuotaStatusCode status_code, | 
|  | int64_t usage, | 
|  | int64_t quota); | 
|  | void PutDidCreateEntry(std::unique_ptr<disk_cache::Entry*> entry_ptr, | 
|  | std::unique_ptr<PutContext> put_context, | 
|  | int rv); | 
|  | void PutDidWriteHeaders(std::unique_ptr<PutContext> put_context, | 
|  | int expected_bytes, | 
|  | int rv); | 
|  | void PutDidWriteBlobToCache(std::unique_ptr<PutContext> put_context, | 
|  | BlobToDiskCacheIDMap::KeyType blob_to_cache_key, | 
|  | disk_cache::ScopedEntryPtr entry, | 
|  | bool success); | 
|  |  | 
|  | // Asynchronously calculates the current cache size, notifies the quota | 
|  | // manager of any change from the last report, and sets cache_size_ to the new | 
|  | // size. Runs |callback| once complete. | 
|  | void UpdateCacheSize(); | 
|  | void UpdateCacheSizeGotSize(int current_cache_size); | 
|  |  | 
|  | // Returns ERROR_NOT_FOUND if not found. Otherwise deletes and returns OK. | 
|  | void Delete(const CacheStorageBatchOperation& operation, | 
|  | const ErrorCallback& callback); | 
|  | void DeleteImpl(std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const CacheStorageCacheQueryParams& match_params, | 
|  | const ErrorCallback& callback); | 
|  | void DeleteDidOpenAllEntries( | 
|  | std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const ErrorCallback& callback, | 
|  | std::unique_ptr<OpenAllEntriesContext> entries_context, | 
|  | CacheStorageError error); | 
|  | void DeleteDidOpenEntry(const GURL& origin, | 
|  | std::unique_ptr<ServiceWorkerFetchRequest> request, | 
|  | const CacheStorageCache::ErrorCallback& callback, | 
|  | std::unique_ptr<disk_cache::Entry*> entryptr, | 
|  | int rv); | 
|  |  | 
|  | // Keys callbacks. | 
|  | void KeysImpl(const RequestsCallback& callback); | 
|  | void KeysDidOpenAllEntries( | 
|  | const RequestsCallback& callback, | 
|  | std::unique_ptr<OpenAllEntriesContext> entries_context, | 
|  | CacheStorageError error); | 
|  | void KeysProcessNextEntry(std::unique_ptr<KeysContext> keys_context, | 
|  | const Entries::iterator& iter); | 
|  | void KeysDidReadMetadata(std::unique_ptr<KeysContext> keys_context, | 
|  | const Entries::iterator& iter, | 
|  | std::unique_ptr<CacheMetadata> metadata); | 
|  |  | 
|  | void CloseImpl(const base::Closure& callback); | 
|  |  | 
|  | void SizeImpl(const SizeCallback& callback); | 
|  |  | 
|  | void GetSizeThenCloseDidGetSize(const SizeCallback& callback, | 
|  | int64_t cache_size); | 
|  |  | 
|  | // Loads the backend and calls the callback with the result (true for | 
|  | // success). The callback will always be called. Virtual for tests. | 
|  | virtual void CreateBackend(const ErrorCallback& callback); | 
|  | void CreateBackendDidCreate(const CacheStorageCache::ErrorCallback& callback, | 
|  | std::unique_ptr<ScopedBackendPtr> backend_ptr, | 
|  | int rv); | 
|  |  | 
|  | void InitBackend(); | 
|  | void InitDidCreateBackend(CacheStorageError cache_create_error); | 
|  | void InitGotCacheSize(CacheStorageError cache_create_error, int cache_size); | 
|  |  | 
|  | void PendingClosure(const base::Closure& callback); | 
|  | void PendingErrorCallback(const ErrorCallback& callback, | 
|  | CacheStorageError error); | 
|  | void PendingResponseCallback( | 
|  | const ResponseCallback& callback, | 
|  | CacheStorageError error, | 
|  | std::unique_ptr<ServiceWorkerResponse> response, | 
|  | std::unique_ptr<storage::BlobDataHandle> blob_data_handle); | 
|  | void PendingResponsesCallback( | 
|  | const ResponsesCallback& callback, | 
|  | CacheStorageError error, | 
|  | std::unique_ptr<Responses> responses, | 
|  | std::unique_ptr<BlobDataHandles> blob_data_handles); | 
|  | void PendingRequestsCallback(const RequestsCallback& callback, | 
|  | CacheStorageError error, | 
|  | std::unique_ptr<Requests> requests); | 
|  | void PendingSizeCallback(const SizeCallback& callback, int64_t size); | 
|  |  | 
|  | void PopulateResponseMetadata(const CacheMetadata& metadata, | 
|  | ServiceWorkerResponse* response); | 
|  | std::unique_ptr<storage::BlobDataHandle> PopulateResponseBody( | 
|  | disk_cache::ScopedEntryPtr entry, | 
|  | ServiceWorkerResponse* response); | 
|  |  | 
|  | // Be sure to check |backend_state_| before use. | 
|  | std::unique_ptr<disk_cache::Backend> backend_; | 
|  |  | 
|  | GURL origin_; | 
|  | const std::string cache_name_; | 
|  | base::FilePath path_; | 
|  | scoped_refptr<net::URLRequestContextGetter> request_context_getter_; | 
|  | scoped_refptr<storage::QuotaManagerProxy> quota_manager_proxy_; | 
|  | base::WeakPtr<storage::BlobStorageContext> blob_storage_context_; | 
|  | BackendState backend_state_ = BACKEND_UNINITIALIZED; | 
|  | std::unique_ptr<CacheStorageScheduler> scheduler_; | 
|  | std::vector<SizeCallback> pending_size_callbacks_; | 
|  | bool initializing_ = false; | 
|  | int64_t cache_size_ = 0; | 
|  |  | 
|  | // Owns the elements of the list | 
|  | BlobToDiskCacheIDMap active_blob_to_disk_cache_writers_; | 
|  |  | 
|  | // Whether or not to store data in disk or memory. | 
|  | bool memory_only_; | 
|  |  | 
|  | base::WeakPtrFactory<CacheStorageCache> weak_ptr_factory_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(CacheStorageCache); | 
|  | }; | 
|  |  | 
|  | }  // namespace content | 
|  |  | 
|  | #endif  // CONTENT_BROWSER_CACHE_STORAGE_CACHE_STORAGE_CACHE_H_ |