Home | History | Annotate | Download | only in indexed_db
      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 CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_TRANSACTION_H_
      6 #define CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_TRANSACTION_H_
      7 
      8 #include <queue>
      9 #include <set>
     10 #include <stack>
     11 
     12 #include "base/basictypes.h"
     13 #include "base/memory/ref_counted.h"
     14 #include "base/memory/scoped_ptr.h"
     15 #include "base/time/time.h"
     16 #include "base/timer/timer.h"
     17 #include "content/browser/indexed_db/indexed_db_backing_store.h"
     18 #include "content/browser/indexed_db/indexed_db_database.h"
     19 #include "content/browser/indexed_db/indexed_db_database_error.h"
     20 #include "third_party/WebKit/public/platform/WebIDBTypes.h"
     21 
     22 namespace content {
     23 
     24 class BlobWriteCallbackImpl;
     25 class IndexedDBCursor;
     26 class IndexedDBDatabaseCallbacks;
     27 
     28 class CONTENT_EXPORT IndexedDBTransaction
     29     : public NON_EXPORTED_BASE(base::RefCounted<IndexedDBTransaction>) {
     30  public:
     31   typedef base::Callback<void(IndexedDBTransaction*)> Operation;
     32 
     33   enum State {
     34     CREATED,     // Created, but not yet started by coordinator.
     35     STARTED,     // Started by the coordinator.
     36     COMMITTING,  // In the process of committing, possibly waiting for blobs
     37                  // to be written.
     38     FINISHED,    // Either aborted or committed.
     39   };
     40 
     41   IndexedDBTransaction(
     42       int64 id,
     43       scoped_refptr<IndexedDBDatabaseCallbacks> callbacks,
     44       const std::set<int64>& object_store_ids,
     45       blink::WebIDBTransactionMode,
     46       IndexedDBDatabase* db,
     47       IndexedDBBackingStore::Transaction* backing_store_transaction);
     48 
     49   virtual void Abort();
     50   leveldb::Status Commit();
     51   void Abort(const IndexedDBDatabaseError& error);
     52 
     53   // Called by the transaction coordinator when this transaction is unblocked.
     54   void Start();
     55 
     56   blink::WebIDBTransactionMode mode() const { return mode_; }
     57   const std::set<int64>& scope() const { return object_store_ids_; }
     58 
     59   void ScheduleTask(Operation task) {
     60     ScheduleTask(blink::WebIDBTaskTypeNormal, task);
     61   }
     62   void ScheduleTask(blink::WebIDBTaskType, Operation task);
     63   void ScheduleAbortTask(Operation abort_task);
     64   void RegisterOpenCursor(IndexedDBCursor* cursor);
     65   void UnregisterOpenCursor(IndexedDBCursor* cursor);
     66   void AddPreemptiveEvent() { pending_preemptive_events_++; }
     67   void DidCompletePreemptiveEvent() {
     68     pending_preemptive_events_--;
     69     DCHECK_GE(pending_preemptive_events_, 0);
     70   }
     71   IndexedDBBackingStore::Transaction* BackingStoreTransaction() {
     72     return transaction_.get();
     73   }
     74   int64 id() const { return id_; }
     75 
     76   IndexedDBDatabase* database() const { return database_.get(); }
     77   IndexedDBDatabaseCallbacks* connection() const { return callbacks_.get(); }
     78 
     79   State state() const { return state_; }
     80   bool IsTimeoutTimerRunning() const { return timeout_timer_.IsRunning(); }
     81 
     82   struct Diagnostics {
     83     base::Time creation_time;
     84     base::Time start_time;
     85     int tasks_scheduled;
     86     int tasks_completed;
     87   };
     88 
     89   const Diagnostics& diagnostics() const { return diagnostics_; }
     90 
     91  private:
     92   friend class BlobWriteCallbackImpl;
     93 
     94   FRIEND_TEST_ALL_PREFIXES(IndexedDBTransactionTestMode, AbortPreemptive);
     95   FRIEND_TEST_ALL_PREFIXES(IndexedDBTransactionTest, Timeout);
     96   FRIEND_TEST_ALL_PREFIXES(IndexedDBTransactionTest,
     97                            SchedulePreemptiveTask);
     98   FRIEND_TEST_ALL_PREFIXES(IndexedDBTransactionTestMode,
     99                            ScheduleNormalTask);
    100 
    101   friend class base::RefCounted<IndexedDBTransaction>;
    102   virtual ~IndexedDBTransaction();
    103 
    104   void RunTasksIfStarted();
    105 
    106   bool IsTaskQueueEmpty() const;
    107   bool HasPendingTasks() const;
    108 
    109   void BlobWriteComplete(bool success);
    110   void ProcessTaskQueue();
    111   void CloseOpenCursors();
    112   leveldb::Status CommitPhaseTwo();
    113   void Timeout();
    114 
    115   const int64 id_;
    116   const std::set<int64> object_store_ids_;
    117   const blink::WebIDBTransactionMode mode_;
    118 
    119   bool used_;
    120   State state_;
    121   bool commit_pending_;
    122   scoped_refptr<IndexedDBDatabaseCallbacks> callbacks_;
    123   scoped_refptr<IndexedDBDatabase> database_;
    124 
    125   class TaskQueue {
    126    public:
    127     TaskQueue();
    128     ~TaskQueue();
    129     bool empty() const { return queue_.empty(); }
    130     void push(Operation task) { queue_.push(task); }
    131     Operation pop();
    132     void clear();
    133 
    134    private:
    135     std::queue<Operation> queue_;
    136 
    137     DISALLOW_COPY_AND_ASSIGN(TaskQueue);
    138   };
    139 
    140   class TaskStack {
    141    public:
    142     TaskStack();
    143     ~TaskStack();
    144     bool empty() const { return stack_.empty(); }
    145     void push(Operation task) { stack_.push(task); }
    146     Operation pop();
    147     void clear();
    148 
    149    private:
    150     std::stack<Operation> stack_;
    151 
    152     DISALLOW_COPY_AND_ASSIGN(TaskStack);
    153   };
    154 
    155   TaskQueue task_queue_;
    156   TaskQueue preemptive_task_queue_;
    157   TaskStack abort_task_stack_;
    158 
    159   scoped_ptr<IndexedDBBackingStore::Transaction> transaction_;
    160   bool backing_store_transaction_begun_;
    161 
    162   bool should_process_queue_;
    163   int pending_preemptive_events_;
    164 
    165   std::set<IndexedDBCursor*> open_cursors_;
    166 
    167   // This timer is started after requests have been processed. If no subsequent
    168   // requests are processed before the timer fires, assume the script is
    169   // unresponsive and abort to unblock the transaction queue.
    170   base::OneShotTimer<IndexedDBTransaction> timeout_timer_;
    171 
    172   Diagnostics diagnostics_;
    173 };
    174 
    175 }  // namespace content
    176 
    177 #endif  // CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_TRANSACTION_H_
    178