IndexedDB: Refactor cursor iteration to remove duplicate code
authorcommit-queue@webkit.org <commit-queue@webkit.org@268f45cc-cd09-0410-ab3c-d52691b4dbfc>
Thu, 12 Apr 2012 22:19:15 +0000 (22:19 +0000)
committercommit-queue@webkit.org <commit-queue@webkit.org@268f45cc-cd09-0410-ab3c-d52691b4dbfc>
Thu, 12 Apr 2012 22:19:15 +0000 (22:19 +0000)
https://bugs.webkit.org/show_bug.cgi?id=83302

Patch by Alec Flett <alecflett@chromium.org> on 2012-04-12
Reviewed by Ojan Vafai.

No new tests, no behavior changes.

* Modules/indexeddb/IDBBackingStore.h:
(Cursor):
* Modules/indexeddb/IDBLevelDBBackingStore.cpp:
(WebCore):

git-svn-id: http://svn.webkit.org/repository/webkit/trunk@114041 268f45cc-cd09-0410-ab3c-d52691b4dbfc

Source/WebCore/ChangeLog
Source/WebCore/Modules/indexeddb/IDBBackingStore.h
Source/WebCore/Modules/indexeddb/IDBLevelDBBackingStore.cpp

index 554060a..43f57b1 100644 (file)
@@ -1,3 +1,17 @@
+2012-04-12  Alec Flett  <alecflett@chromium.org>
+
+        IndexedDB: Refactor cursor iteration to remove duplicate code
+        https://bugs.webkit.org/show_bug.cgi?id=83302
+
+        Reviewed by Ojan Vafai.
+
+        No new tests, no behavior changes.
+
+        * Modules/indexeddb/IDBBackingStore.h:
+        (Cursor):
+        * Modules/indexeddb/IDBLevelDBBackingStore.cpp:
+        (WebCore):
+
 2012-04-12  Erik Arvidsson  <arv@chromium.org>
 
         Add support for [ArrayClass] and use that on NodeList
index b5fcb3a..551b826 100644 (file)
@@ -88,8 +88,13 @@ public:
 
     class Cursor : public RefCounted<Cursor> {
     public:
+        enum IteratorState {
+            Ready = 0,
+            Seek
+        };
+
         virtual PassRefPtr<Cursor> clone() = 0;
-        virtual bool continueFunction(const IDBKey* = 0) = 0;
+        virtual bool continueFunction(const IDBKey* = 0, IteratorState nextState = Seek) = 0;
         virtual PassRefPtr<IDBKey> key() = 0;
         virtual PassRefPtr<IDBKey> primaryKey() = 0;
         virtual String value() = 0;
index f1f6ed5..8d57022 100644 (file)
@@ -980,7 +980,7 @@ struct CursorOptions {
 class CursorImplCommon : public IDBBackingStore::Cursor {
 public:
     // IDBBackingStore::Cursor
-    virtual bool continueFunction(const IDBKey*);
+    virtual bool continueFunction(const IDBKey*, IteratorState);
     virtual PassRefPtr<IDBKey> key() { return m_currentKey; }
     virtual PassRefPtr<IDBKey> primaryKey() { return m_currentKey; }
     virtual String value() = 0;
@@ -1015,6 +1015,9 @@ protected:
 
     virtual ~CursorImplCommon() {}
 
+    bool isPastBounds() const;
+    bool haveEnteredRange() const;
+
     LevelDBTransaction* m_transaction;
     CursorOptions m_cursorOptions;
     OwnPtr<LevelDBIterator> m_iterator;
@@ -1024,62 +1027,26 @@ protected:
 bool CursorImplCommon::firstSeek()
 {
     m_iterator = m_transaction->createIterator();
-
     if (m_cursorOptions.forward)
         m_iterator->seek(m_cursorOptions.lowKey);
     else
         m_iterator->seek(m_cursorOptions.highKey);
 
-    for (;;) {
-        if (!m_iterator->isValid())
-            return false;
-
-        if (m_cursorOptions.forward && m_cursorOptions.highOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) >= 0)
-            return false;
-        if (m_cursorOptions.forward && !m_cursorOptions.highOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) > 0)
-            return false;
-        if (!m_cursorOptions.forward && m_cursorOptions.lowOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) <= 0)
-            return false;
-        if (!m_cursorOptions.forward && !m_cursorOptions.lowOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) < 0)
-            return false;
-
-        if (m_cursorOptions.forward && m_cursorOptions.lowOpen) {
-            // lowKey not included in the range.
-            if (compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) <= 0) {
-                m_iterator->next();
-                continue;
-            }
-        }
-        if (!m_cursorOptions.forward && m_cursorOptions.highOpen) {
-            // highKey not included in the range.
-            if (compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) >= 0) {
-                m_iterator->prev();
-                continue;
-            }
-        }
-
-        if (!loadCurrentRow()) {
-            if (m_cursorOptions.forward)
-                m_iterator->next();
-            else
-                m_iterator->prev();
-            continue;
-        }
-
-        return true;
-    }
+    return continueFunction(0, Ready);
 }
 
-bool CursorImplCommon::continueFunction(const IDBKey* key)
+bool CursorImplCommon::continueFunction(const IDBKey* key, IteratorState nextState)
 {
-    // FIXME: This shares a lot of code with firstSeek.
     RefPtr<IDBKey> previousKey = m_currentKey;
 
     for (;;) {
-        if (m_cursorOptions.forward)
-            m_iterator->next();
-        else
-            m_iterator->prev();
+        if (nextState == Seek) {
+            if (m_cursorOptions.forward)
+                m_iterator->next();
+            else
+                m_iterator->prev();
+        } else
+            nextState = Seek; // for subsequent iterations
 
         if (!m_iterator->isValid())
             return false;
@@ -1088,15 +1055,14 @@ bool CursorImplCommon::continueFunction(const IDBKey* key)
         if (!m_transaction->get(m_iterator->key(), trash))
              continue;
 
-        if (m_cursorOptions.forward && m_cursorOptions.highOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) >= 0) // high key not included in range
-            return false;
-        if (m_cursorOptions.forward && !m_cursorOptions.highOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) > 0)
-            return false;
-        if (!m_cursorOptions.forward && m_cursorOptions.lowOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) <= 0) // low key not included in range
-            return false;
-        if (!m_cursorOptions.forward && !m_cursorOptions.lowOpen && compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) < 0)
+        if (isPastBounds())
             return false;
 
+        if (!haveEnteredRange())
+            continue;
+
+        // The row may not load because there's a stale entry in the
+        // index. This is not fatal.
         if (!loadCurrentRow())
             continue;
 
@@ -1119,6 +1085,33 @@ bool CursorImplCommon::continueFunction(const IDBKey* key)
     return true;
 }
 
+bool CursorImplCommon::haveEnteredRange() const
+{
+    if (m_cursorOptions.forward) {
+        if (m_cursorOptions.lowOpen)
+            return compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) > 0;
+
+        return compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) >= 0;
+    }
+    if (m_cursorOptions.highOpen)
+        return compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) < 0;
+
+    return compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) <= 0;
+}
+
+bool CursorImplCommon::isPastBounds() const
+{
+    if (m_cursorOptions.forward) {
+        if (m_cursorOptions.highOpen)
+            return compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) >= 0;
+        return compareIndexKeys(m_iterator->key(), m_cursorOptions.highKey) > 0;
+    }
+
+    if (m_cursorOptions.lowOpen)
+        return compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) <= 0;
+    return compareIndexKeys(m_iterator->key(), m_cursorOptions.lowKey) < 0;
+}
+
 class ObjectStoreCursorImpl : public CursorImplCommon {
 public:
     static PassRefPtr<ObjectStoreCursorImpl> create(LevelDBTransaction* transaction, const CursorOptions& cursorOptions)