summaryrefslogtreecommitdiffstats
path: root/content/browser/background_sync/background_sync_manager.h
blob: a2cad0d4d3afae1383de7eec01c886ccf5b202b0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
// Copyright 2015 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_BACKGROUND_SYNC_BACKGROUND_SYNC_MANAGER_H_
#define CONTENT_BROWSER_BACKGROUND_SYNC_BACKGROUND_SYNC_MANAGER_H_

#include <list>
#include <map>

#include "base/callback_forward.h"
#include "base/memory/scoped_ptr.h"
#include "base/memory/weak_ptr.h"
#include "content/browser/background_sync/background_sync.pb.h"
#include "content/browser/cache_storage/cache_storage_scheduler.h"
#include "content/browser/service_worker/service_worker_context_observer.h"
#include "content/browser/service_worker/service_worker_storage.h"
#include "content/common/content_export.h"
#include "content/common/service_worker/service_worker_status_code.h"
#include "url/gurl.h"

namespace content {

class ServiceWorkerContextWrapper;

// BackgroundSyncManager manages and stores the set of background sync
// registrations across all registered service workers for a profile.
// Registrations are stored along with their associated Service Worker
// registration in ServiceWorkerStorage. If the ServiceWorker is unregistered,
// the sync registrations are removed. This class expects to be run on the IO
// thread. The asynchronous methods are executed sequentially.

// TODO(jkarlin): Check permissions when registering, scheduling, and firing
// background sync. In the meantime, --enable-service-worker-sync is required to
// fire a sync event.
// TODO(jkarlin): Unregister syncs when permission is revoked.
// TODO(jkarlin): Create a background sync scheduler to actually run the
// registered events.
// TODO(jkarlin): Keep the browser alive if "Let Google Chrome Run in the
// Background" is true and a sync is registered.
class CONTENT_EXPORT BackgroundSyncManager
    : NON_EXPORTED_BASE(public ServiceWorkerContextObserver) {
 public:
  enum ErrorType {
    ERROR_TYPE_OK = 0,
    ERROR_TYPE_STORAGE,
    ERROR_TYPE_NOT_FOUND
  };

  // TODO(jkarlin): Remove this and use the struct from IPC messages once it
  // lands.
  struct CONTENT_EXPORT BackgroundSyncRegistration {
    using RegistrationId = int64;
    static const RegistrationId kInvalidRegistrationId;
    BackgroundSyncRegistration() {}

    bool Equals(const BackgroundSyncRegistration& other) {
      return this->name == other.name && this->fire_once == other.fire_once &&
             this->min_period == other.min_period &&
             network_state == other.network_state &&
             power_state == other.power_state;
    }

    RegistrationId id = kInvalidRegistrationId;
    std::string name;
    bool fire_once = true;
    int64 min_period = 0;
    SyncNetworkState network_state = NETWORK_STATE_ONLINE;
    SyncPowerState power_state = POWER_STATE_AVOID_DRAINING;
  };

  struct CONTENT_EXPORT BackgroundSyncRegistrations {
    using NameToRegistrationMap =
        std::map<std::string, BackgroundSyncRegistration>;
    static const BackgroundSyncRegistration::RegistrationId kInitialId;

    BackgroundSyncRegistrations();
    explicit BackgroundSyncRegistrations(
        BackgroundSyncRegistration::RegistrationId next_id);
    ~BackgroundSyncRegistrations();

    NameToRegistrationMap name_to_registration_map;
    BackgroundSyncRegistration::RegistrationId next_id;
  };

  using StatusCallback = base::Callback<void(ErrorType)>;
  using StatusAndRegistrationCallback =
      base::Callback<void(ErrorType, const BackgroundSyncRegistration&)>;

  static scoped_ptr<BackgroundSyncManager> Create(
      const scoped_refptr<ServiceWorkerContextWrapper>& service_worker_context);
  ~BackgroundSyncManager() override;

  // Stores the given background sync registration and adds it to the scheduling
  // queue. Overwrites any existing registration with the same name but
  // different parameters (other than the id). Calls |callback| with ErrorTypeOK
  // and the accepted registration on success. The accepted registration will
  // have a unique id. It may also have altered parameters if the user or UA
  // chose different parameters than those supplied.
  void Register(const GURL& origin,
                int64 sw_registration_id,
                const BackgroundSyncRegistration& sync_registration,
                const StatusAndRegistrationCallback& callback);

  // Removes the background sync registration with |sync_registration_name| if
  // the |sync_registration_id| matches. |sync_registration_id| will not match
  // if, for instance, a new registration with the same name has replaced it.
  // Calls |callback| with ErrorTypeNotFound if no match is found. Calls
  // |callback| with ErrorTypeOK on success.
  void Unregister(
      const GURL& origin,
      int64 sw_registration_id,
      const std::string& sync_registration_name,
      BackgroundSyncRegistration::RegistrationId sync_registration_id,
      const StatusCallback& callback);

  // Finds the background sync registration associated with
  // |sw_registration_id|. Calls |callback| with ErrorTypeNotFound if it doesn't
  // exist. Calls |callback| with ErrorTypeOK on success.
  void GetRegistration(const GURL& origin,
                       int64 sw_registration_id,
                       const std::string sync_registration_name,
                       const StatusAndRegistrationCallback& callback);

  // ServiceWorkerContextObserver overrides.
  void OnRegistrationDeleted(int64 registration_id,
                             const GURL& pattern) override;
  void OnStorageWiped() override;

 protected:
  explicit BackgroundSyncManager(
      const scoped_refptr<ServiceWorkerContextWrapper>& context);

  // Init must be called before any public member function. Only call it once.
  void Init();

  // The following methods are virtual for testing.
  virtual void StoreDataInBackend(
      int64 sw_registration_id,
      const GURL& origin,
      const std::string& key,
      const std::string& data,
      const ServiceWorkerStorage::StatusCallback& callback);
  virtual void GetDataFromBackend(
      const std::string& key,
      const ServiceWorkerStorage::GetUserDataForAllRegistrationsCallback&
          callback);

 private:
  using PermissionStatusCallback = base::Callback<void(bool)>;
  using SWIdToRegistrationsMap = std::map<int64, BackgroundSyncRegistrations>;

  // Disable the manager. Already queued operations will abort once they start
  // to run (in their impl methods). Future operations will not queue. Any
  // registrations are cleared from memory and the backend (if it's still
  // functioning). The manager will reenable itself once it receives the
  // OnStorageWiped message or on browser restart.
  void DisableAndClearManager(const base::Closure& callback);
  void DisableAndClearDidGetRegistrations(
      const base::Closure& callback,
      const std::vector<std::pair<int64, std::string>>& user_data,
      ServiceWorkerStatusCode status);
  void DisableAndClearManagerClearedOne(const base::Closure& barrier_closure,
                                        ServiceWorkerStatusCode status);

  // Returns the existing registration in |existing_registration| if it is not
  // null.
  bool LookupRegistration(int64 sw_registration_id,
                          const std::string& sync_registration_name,
                          BackgroundSyncRegistration* existing_registration);

  // Store all registrations for a given |sw_registration_id|.
  void StoreRegistrations(const GURL& origin,
                          int64 sw_registration_id,
                          const ServiceWorkerStorage::StatusCallback& callback);

  // If the registration is in the map, removes it and returns the removed
  // registration in |old_registration|. |old_registration| may be null.
  void RemoveRegistrationFromMap(int64 sw_registration_id,
                                 const std::string& sync_registration_name,
                                 BackgroundSyncRegistration* old_registration);

  void AddRegistrationToMap(
      int64 sw_registration_id,
      const BackgroundSyncRegistration& sync_registration);

  void InitImpl(const base::Closure& callback);
  void InitDidGetDataFromBackend(
      const base::Closure& callback,
      const std::vector<std::pair<int64, std::string>>& user_data,
      ServiceWorkerStatusCode status);

  // Register callbacks
  void RegisterImpl(const GURL& origin,
                    int64 sw_registration_id,
                    const BackgroundSyncRegistration& sync_registration,
                    const StatusAndRegistrationCallback& callback);
  void RegisterDidStore(int64 sw_registration_id,
                        const BackgroundSyncRegistration& sync_registration,
                        const StatusAndRegistrationCallback& callback,
                        ServiceWorkerStatusCode status);

  // Unregister callbacks
  void UnregisterImpl(
      const GURL& origin,
      int64 sw_registration_id,
      const std::string& sync_registration_name,
      BackgroundSyncRegistration::RegistrationId sync_registration_id,
      const StatusCallback& callback);
  void UnregisterDidStore(
      int64 sw_registration_id,
      const BackgroundSyncRegistration& old_sync_registration,
      const StatusCallback& callback,
      ServiceWorkerStatusCode status);

  // GetRegistration callbacks
  void GetRegistrationImpl(const GURL& origin,
                           int64 sw_registration_id,
                           const std::string sync_registration_name,
                           const StatusAndRegistrationCallback& callback);

  // OnRegistrationDeleted callbacks
  void OnRegistrationDeletedImpl(int64 registration_id,
                                 const base::Closure& callback);

  // OnStorageWiped callbacks
  void OnStorageWipedImpl(const base::Closure& callback);

  // Operation Scheduling callbacks
  void PendingStatusAndRegistrationCallback(
      const StatusAndRegistrationCallback& callback,
      ErrorType error,
      const BackgroundSyncRegistration& sync_registration);
  void PendingStatusCallback(const StatusCallback& callback, ErrorType error);
  void PendingClosure(const base::Closure& closure);

  StatusAndRegistrationCallback MakeStatusAndRegistrationCompletion(
      const StatusAndRegistrationCallback& callback);
  BackgroundSyncManager::StatusCallback MakeStatusCompletion(
      const StatusCallback& callback);
  base::Closure MakeEmptyCompletion();

  SWIdToRegistrationsMap sw_to_registrations_map_;
  CacheStorageScheduler op_scheduler_;
  scoped_refptr<ServiceWorkerContextWrapper> service_worker_context_;
  bool disabled_;

  base::WeakPtrFactory<BackgroundSyncManager> weak_ptr_factory_;

  DISALLOW_COPY_AND_ASSIGN(BackgroundSyncManager);
};

}  // namespace content

#endif  // CONTENT_BROWSER_BACKGROUND_SYNC_BACKGROUND_SYNC_MANAGER_H_