|  | // Copyright 2021 The Chromium Authors | 
|  | // 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_OP_SCHEDULER_H_ | 
|  | #define CONTENT_BROWSER_BACKGROUND_SYNC_BACKGROUND_SYNC_OP_SCHEDULER_H_ | 
|  |  | 
|  | #include <map> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/containers/queue.h" | 
|  | #include "base/feature_list.h" | 
|  | #include "base/functional/bind.h" | 
|  | #include "base/functional/callback.h" | 
|  | #include "base/memory/weak_ptr.h" | 
|  | #include "base/sequence_checker.h" | 
|  | #include "base/task/sequenced_task_runner.h" | 
|  | #include "content/common/content_export.h" | 
|  |  | 
|  | namespace content { | 
|  |  | 
|  | // BackgroundSyncOpScheduler runs the scheduled callbacks sequentially. Add an | 
|  | // operation by calling ScheduleOperation() with your callback. Once your | 
|  | // operation is done be sure to call CompleteOperationAndRunNext() to schedule | 
|  | // the next operation.  This is a simplified version of CacheStorageScheduler. | 
|  | class CONTENT_EXPORT BackgroundSyncOpScheduler { | 
|  | public: | 
|  | explicit BackgroundSyncOpScheduler( | 
|  | scoped_refptr<base::SequencedTaskRunner> task_runner); | 
|  |  | 
|  | BackgroundSyncOpScheduler(const BackgroundSyncOpScheduler&) = delete; | 
|  | BackgroundSyncOpScheduler& operator=(const BackgroundSyncOpScheduler&) = | 
|  | delete; | 
|  |  | 
|  | virtual ~BackgroundSyncOpScheduler(); | 
|  |  | 
|  | // Adds the operation to the tail of the queue and starts it if possible. | 
|  | void ScheduleOperation(base::OnceClosure closure); | 
|  |  | 
|  | // Call this after each operation completes. It cleans up the operation | 
|  | // associated with the given id.  If may also start the next set of | 
|  | // operations. | 
|  | void CompleteOperationAndRunNext(); | 
|  |  | 
|  | // Returns true if there are any running or pending operations. | 
|  | bool ScheduledOperations() const; | 
|  |  | 
|  | // Wraps |callback| to also call CompleteOperationAndRunNext. | 
|  | template <typename... Args> | 
|  | base::OnceCallback<void(Args...)> WrapCallbackToRunNext( | 
|  | base::OnceCallback<void(Args...)> callback) { | 
|  | return base::BindOnce( | 
|  | &BackgroundSyncOpScheduler::RunNextContinuation<Args...>, | 
|  | weak_ptr_factory_.GetWeakPtr(), std::move(callback)); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | // virtual for testing | 
|  | virtual void DoneStartingAvailableOperations() {} | 
|  |  | 
|  | private: | 
|  | class Operation; | 
|  |  | 
|  | // Maybe start running the next operation depending on the current | 
|  | // set of running operations and the mode of the next operation. | 
|  | void MaybeRunOperation(); | 
|  |  | 
|  | template <typename... Args> | 
|  | void RunNextContinuation(base::OnceCallback<void(Args...)> callback, | 
|  | Args... args) { | 
|  | // Grab a weak ptr to guard against the scheduler being deleted during the | 
|  | // callback. | 
|  | base::WeakPtr<BackgroundSyncOpScheduler> scheduler = | 
|  | weak_ptr_factory_.GetWeakPtr(); | 
|  |  | 
|  | std::move(callback).Run(std::forward<Args>(args)...); | 
|  | if (scheduler) | 
|  | CompleteOperationAndRunNext(); | 
|  | } | 
|  |  | 
|  | scoped_refptr<base::SequencedTaskRunner> task_runner_; | 
|  |  | 
|  | base::queue<std::unique_ptr<Operation>> pending_operations_; | 
|  |  | 
|  | std::unique_ptr<Operation> running_operation_; | 
|  |  | 
|  | SEQUENCE_CHECKER(sequence_checker_); | 
|  | base::WeakPtrFactory<BackgroundSyncOpScheduler> weak_ptr_factory_{this}; | 
|  | }; | 
|  |  | 
|  | }  // namespace content | 
|  |  | 
|  | #endif  // CONTENT_BROWSER_BACKGROUND_SYNC_BACKGROUND_SYNC_OP_SCHEDULER_H_ |