Home | History | Annotate | Download | only in extensions
      1 // Copyright (c) 2013 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 #ifndef CHROME_BROWSER_CHROMEOS_EXTENSIONS_EXTERNAL_CACHE_H_
      6 #define CHROME_BROWSER_CHROMEOS_EXTENSIONS_EXTERNAL_CACHE_H_
      7 
      8 #include <string>
      9 
     10 #include "base/basictypes.h"
     11 #include "base/callback_forward.h"
     12 #include "base/files/file_path.h"
     13 #include "base/gtest_prod_util.h"
     14 #include "base/memory/ref_counted.h"
     15 #include "base/memory/scoped_ptr.h"
     16 #include "base/memory/weak_ptr.h"
     17 #include "base/sequenced_task_runner.h"
     18 #include "chrome/browser/extensions/updater/extension_downloader_delegate.h"
     19 #include "chrome/browser/extensions/updater/local_extension_cache.h"
     20 #include "content/public/browser/notification_observer.h"
     21 #include "content/public/browser/notification_registrar.h"
     22 
     23 namespace base {
     24 class DictionaryValue;
     25 }
     26 
     27 namespace extensions {
     28 class ExtensionDownloader;
     29 }
     30 
     31 namespace net {
     32 class URLRequestContextGetter;
     33 }
     34 
     35 namespace chromeos {
     36 
     37 // The ExternalCache manages a cache for external extensions.
     38 class ExternalCache : public content::NotificationObserver,
     39                       public extensions::ExtensionDownloaderDelegate {
     40  public:
     41   typedef base::Callback<void(const std::string& id, bool success)>
     42       PutExternalExtensionCallback;
     43 
     44   class Delegate {
     45    public:
     46     virtual ~Delegate() {}
     47     // Caller owns |prefs|.
     48     virtual void OnExtensionListsUpdated(
     49         const base::DictionaryValue* prefs) = 0;
     50     // Called after extension with |id| is loaded in cache.
     51     virtual void OnExtensionLoadedInCache(const std::string& id) {}
     52     // Called when extension with |id| is failed with downloading for |error|.
     53     virtual void OnExtensionDownloadFailed(
     54         const std::string& id,
     55         extensions::ExtensionDownloaderDelegate::Error error) {}
     56 
     57     // Cache needs to provide already installed extensions otherwise they
     58     // will be removed. Cache calls this function to get version of installed
     59     // extension or empty string if not installed.
     60     virtual std::string GetInstalledExtensionVersion(const std::string& id);
     61   };
     62 
     63   // The |request_context| is used for update checks. All file I/O is done via
     64   // the |backend_task_runner|. If |always_check_updates| is |false|, update
     65   // checks are performed for extensions that have an |external_update_url|
     66   // only. If |wait_for_cache_initialization| is |true|, the cache contents will
     67   // not be read until a flag file appears in the cache directory, signaling
     68   // that the cache is ready.
     69   ExternalCache(const base::FilePath& cache_dir,
     70                 net::URLRequestContextGetter* request_context,
     71                 const scoped_refptr<base::SequencedTaskRunner>&
     72                     backend_task_runner,
     73                 Delegate* delegate,
     74                 bool always_check_updates,
     75                 bool wait_for_cache_initialization);
     76   virtual ~ExternalCache();
     77 
     78   // Returns already cached extensions.
     79   const base::DictionaryValue* cached_extensions() {
     80     return cached_extensions_.get();
     81   }
     82 
     83   // Implementation of content::NotificationObserver:
     84   virtual void Observe(int type,
     85                        const content::NotificationSource& source,
     86                        const content::NotificationDetails& details) OVERRIDE;
     87 
     88   // Implementation of ExtensionDownloaderDelegate:
     89   virtual void OnExtensionDownloadFailed(
     90       const std::string& id,
     91       Error error,
     92       const PingResult& ping_result,
     93       const std::set<int>& request_ids) OVERRIDE;
     94 
     95   virtual void OnExtensionDownloadFinished(
     96       const std::string& id,
     97       const base::FilePath& path,
     98       bool file_ownership_passed,
     99       const GURL& download_url,
    100       const std::string& version,
    101       const PingResult& ping_result,
    102       const std::set<int>& request_ids) OVERRIDE;
    103 
    104   virtual bool IsExtensionPending(const std::string& id) OVERRIDE;
    105 
    106   virtual bool GetExtensionExistingVersion(const std::string& id,
    107                                            std::string* version) OVERRIDE;
    108 
    109   // Shut down the cache. The |callback| will be invoked when the cache has shut
    110   // down completely and there are no more pending file I/O operations.
    111   void Shutdown(const base::Closure& callback);
    112 
    113   // Replace the list of extensions to cache with |prefs| and perform update
    114   // checks for these.
    115   void UpdateExtensionsList(scoped_ptr<base::DictionaryValue> prefs);
    116 
    117   // If a user of one of the ExternalCache's extensions detects that
    118   // the extension is damaged then this method can be used to remove it from
    119   // the cache and retry to download it after a restart.
    120   void OnDamagedFileDetected(const base::FilePath& path);
    121 
    122   // Removes extensions listed in |ids| from external cache, corresponding crx
    123   // files will be removed from disk too.
    124   void RemoveExtensions(const std::vector<std::string>& ids);
    125 
    126   // If extension with |id| exists in the cache, returns |true|, |file_path| and
    127   // |version| for the extension. Extension will be marked as used with current
    128   // timestamp.
    129   bool GetExtension(const std::string& id,
    130                     base::FilePath* file_path,
    131                     std::string* version);
    132 
    133   // Puts the external |crx_file_path| into |local_cache_| for extension with
    134   // |id|.
    135   void PutExternalExtension(const std::string& id,
    136                             const base::FilePath& crx_file_path,
    137                             const std::string& version,
    138                             const PutExternalExtensionCallback& callback);
    139 
    140  private:
    141   // Notifies the that the cache has been updated, providing
    142   // extensions loader with an updated list of extensions.
    143   void UpdateExtensionLoader();
    144 
    145   // Checks the cache contents and initiate download if needed.
    146   void CheckCache();
    147 
    148   // Invoked on the UI thread when a new entry has been installed in the cache.
    149   void OnPutExtension(const std::string& id,
    150                       const base::FilePath& file_path,
    151                       bool file_ownership_passed);
    152 
    153   // Invoked on the UI thread when the external extension has been installed
    154   // in the local cache by calling PutExternalExtension.
    155   void OnPutExternalExtension(const std::string& id,
    156                               const PutExternalExtensionCallback& callback,
    157                               const base::FilePath& file_path,
    158                               bool file_ownership_passed);
    159 
    160   extensions::LocalExtensionCache local_cache_;
    161 
    162   // Request context used by the |downloader_|.
    163   scoped_refptr<net::URLRequestContextGetter> request_context_;
    164 
    165   // Task runner for executing file I/O tasks.
    166   const scoped_refptr<base::SequencedTaskRunner> backend_task_runner_;
    167 
    168   // Delegate that would like to get notifications about cache updates.
    169   Delegate* delegate_;
    170 
    171   // Updates needs to be check for the extensions with external_crx too.
    172   bool always_check_updates_;
    173 
    174   // Set to true if cache should wait for initialization flag file.
    175   bool wait_for_cache_initialization_;
    176 
    177   // This is the list of extensions currently configured.
    178   scoped_ptr<base::DictionaryValue> extensions_;
    179 
    180   // This contains extensions that are both currently configured
    181   // and that have a valid crx in the cache.
    182   scoped_ptr<base::DictionaryValue> cached_extensions_;
    183 
    184   // Used to download the extensions and to check for updates.
    185   scoped_ptr<extensions::ExtensionDownloader> downloader_;
    186 
    187   // Observes failures to install CRX files.
    188   content::NotificationRegistrar notification_registrar_;
    189 
    190   // Weak factory for callbacks.
    191   base::WeakPtrFactory<ExternalCache> weak_ptr_factory_;
    192 
    193   DISALLOW_COPY_AND_ASSIGN(ExternalCache);
    194 };
    195 
    196 }  // namespace chromeos
    197 
    198 #endif  // CHROME_BROWSER_CHROMEOS_EXTENSIONS_EXTERNAL_CACHE_H_
    199