blob: b05819f87f97b1ef367e7c2046e5c954e00ec407 [file] [log] [blame]
// 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/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_