Skip to content

Latest commit

 

History

History
400 lines (188 loc) · 11.3 KB

README.md

File metadata and controls

400 lines (188 loc) · 11.3 KB

Light Weight promised based IndexedDB wrapper

As we already know that indexedDB is an event-driven system. Everything in indexedDB is an event that makes it very difficult to work with as well as very difficult to write production-ready code using native indexedDB because the event-driven system is very error-prone.

On the other side, indexed-PDB makes it very easy to use the indexedDB as it wraps the native code to promise. Which makes it very easy to use and understand and also ready to use in the larger codebase.

Code comparision

Below is the indexedDB code of reading all rows from the objectStore using the cursor.

var db;
var request = indexedDB.open("MyTestDatabase");

request.onerror = function (event) {
    console.log("Why didn't you allow my web app to use IndexedDB?!");
};

request.onsuccess = function (event) {
    db = event.target.result;
    // Handling Errors
    db.onerror = function (event) {
        // Generic error handler for all errors targeted at this database's
        // requests!
        console.error("Database error: " + event.target.errorCode);
    };

    // open the transaction
    var transaction = db.transaction(["customers"], "readwrite");
    // Do something when all the data is added to the database.
    transaction.oncomplete = function (event) {
        console.log("All done!");
    };

    transaction.onerror = function (event) {
        // Don't forget to handle errors!
    };

    var objectStore = transaction.objectStore("customers");

    var customers = [];

    objectStore.openCursor().onsuccess = function (event) {
        var cursor = event.target.result;
        if (cursor) {
            customers.push(cursor.value);
            cursor.continue();
        }
        else {
            console.log("Got all customers: " + customers);
        }
    };
};

And below is the indexed-PDB code of reading all rows from the objectStore using the cursor.

import { openDB } from 'indexed-pdb'

(async () => {
    try {
        const db = await openDB('MyTestDatabase')
        var objectStore = db.transaction(["customers"], "readwrite").objectStore("customers");
        var customers = [];

        await objectStore.openCursor().then(function push_item(cursor) {
            if (!cursor) {
                return
            }
            customers.push(cursor.value)
            return cursor.continue().then(push_item)
        })

        console.log("Got all customers: " + customers);
        
    } catch (error) {
        console.log(error, 'any error during the process');
    }

})()

As you can see indexed-PDB is easy to understand and also there are very few lines of code.

Install

Clearly you can't use indexedDB in the node environment. But if you want to emulate the behavior in the NodeJs environment then you can use fake-indexeddb

We currently do not support the emulation in the NodeJs environment and I don't think you should use fake-indexeddb in nodeJs env.

Setting up in WebApp

Using indexed-pdb in the web app ( Angular, React, or Vue.js ) is pretty straight forward.

npm i indexed-pdb

After installing it from the npm registry just import it into your working project

import { openDB } from 'indexed-pdb'

Using Directly In a browser

Using the modules method directly via unpkg:

<script type="module">
  import { openDB} from 'https://unpkg.com/indexed-pdb?module';

  async function doDatabaseStuff() {
    const db = await openDB();
  }
</script>

Using external script reference

<script src="https://unpkg.com/indexed-pdb@1.1.1/build/iife/index-min.js"></script>
<script>
  async function doDatabaseStuff() {
    const db = await ipdb.openDB();
  }
</script>

API

openDB

This method opens a database, and returns a promise for an enhanced IDBDatabase.

(
    async function () {
        const database = openDB(db_name, version, function upgradeCallback (upgradeDB) {
            // create new object store here...
        })
    }
)()
  • name : Name of the database
  • version : Schema version, or undefined to open the current version.
  • upgradeCallback (optional) : Called if this version of the database has never been opened before. Use it to specify the schema for the database. This is similar to the upgradeneeded event in plain IndexedDB.
  • database : An enhanced IDBDatabase. ( IDBDatabaseWrapper)
  • upgradeDB : An enhanced IDBDatabase. Use this to create new object store on the upgraded database.

IDBDatabaseWrapper

A object returned from the openDB.

Properties

IDBDatabaseWrapper.name - (Read only)

A DOMString that contains the name of the connected database.

IDBDatabaseWrapper.version - (Read only)

A 64-bit integer that contains the version of the connected database. When a database is first created, this attribute is an empty string.

IDBDatabaseWrapper.objectStoreNames - ( Read only)

A DOMStringList that contains a list of the names of the object stores currently in the connected database.

Methods

IDBDatabaseWrapper.close() @return - void

Returns immediately and closes the connection to a database in a separate thread.The connection is not actually closed until all transactions created using this connection are complete. No new transactions can be created for this connection once this method is called. Methods that create transactions throw an exception if a closing operation is pending.


IDBDatabaseWrapper.createObjectStore() @return IDBObjectStoreWrapper

Creates a new object store with the given name and options and returns a new IDBObjectStoreWrapper. Throws a InvalidStateError DOMException if not called within an upgrade transaction.


IDBDatabaseWrapper.deleteObjectStore(name) @return void

  • name : Name of the object store to delete.

Deletes the object store with the given name. Throws a "InvalidStateError" DOMException if not called within an upgrade transaction.


IDBDatabaseWrapper.transaction(storeNames, mode) @return IDBTransactionWrapper

  • storeNames : Array of store names to open the transaction on.

  • mode ( optional) : Provide one of the value from "readonly" | "readwrite" | "versionchange". If you want to just read data from objectStore then use "readonly" , If you want to insert new data to objectStore then use "readwrite" and if you want to add new index or change index on the objectStore then use "versionchange".

Immediately returns a transaction object (IDBTransactionWrapper) containing the IDBTransactionWrapper.objectStore method, which you can use to access your object store. Runs in a separate thread.


IDBObjectStoreWrapper

The IDBObjectStoreWrapper interface API represents an object store in a database. Records within an object store are sorted according to their keys. This sorting enables fast insertion, look-up, and ordered retrieval.

Properties

IDBObjectStoreWrapper.indexNames - ( Read only )

A list of the names of indexes on objects in object store.

IDBObjectStoreWrapper.keyPath - ( Read only )

The key path of object store. If this attribute is null, the application must provide a key for each modification operation.

IDBObjectStoreWrapper.name - ( Read only )

The name of object store.

IDBObjectStoreWrapper.transaction - ( Read only )

The IDBTransactionWrapper object to which this object store belongs.

IDBObjectStoreWrapper.autoIncrement - ( Read only )

Returns true if the store has a key generator, and false otherwise.

Methods

IDBObjectStoreWrapper.add(value, key, transactionCallback(transaction){}) @return Promise<IDBValidKey>

  • value : Value to add to object store

  • key (optional) : Unique key for the value

  • transactionCallback (optional) : A callback to get the IDBTransactionWrapper

    • transaction : The IDBTransactionWrapper object to which this object store belongs.

Adds or updates a record in store with the given value and key. If the store uses in-line keys and key is specified a "DataError" DOMException will be thrown.

If put() is used, any existing record with the key will be replaced. If add() is used, and if a record with the key already exists the request will fail, with request's error set to a "ConstraintError" DOMException.If successful, request's result will be the record's key.


IDBObjectStoreWrapper.addAll(value) @return Promise<any[]>

  • value : Array of the value to add to object store.

This method will only work if object store uses in-line keys. This method will add all the values of the array with single call. If successful, request's result will be the record's keys


IDBObjectStoreWrapper.clear() @return Promise<"DONE">

Use this method to delete all the records of the object store. If successful, request's result will be "DONE" string.


IDBObjectStoreWrapper.count(key) @return Promise<number>

  • key : key or IDBKeyRange of the value to retrive.

Returns the total number of records that match the provided key or IDBKeyRange. If no arguments are provided, it returns the total number of records in the store.


IDBObjectStoreWrapper.createIndex(name, keyPath, options) @return IDBIndexWrapper

  • name : Name of index

  • keyPath : property or array of properties of the object on which to create the index.

  • options (optional) : An object { multiEntry, unique } that tells how to create the index on the object store. If your value can be duplicate then use { multiEntry: true } or if you are sure that your value will be unique then use { unique: true }

Creates a new index in store with the given name, keyPath and options and returns a new IDBIndexWrapper. If the keyPath and options define constraints that cannot be satisfied with the data already in store the upgrade transaction will abort with a "ConstraintError" DOMException.

Throws an "InvalidStateError" DOMException if not called within an upgrade transaction.


IDBObjectStoreWrapper.delete(key) @return Promise<"OK">

  • key: key or range of key on which to delete the values

Deletes records in store with the given key or in the given key range in query. If successful, request's result will be "OK" string.


IDBObjectStoreWrapper.deleteIndex(name) @return void

  • name: Name of index to delete.

Deletes the index in store with the given name. Throws an "InvalidStateError" DOMException if not called within an upgrade transaction.


IDBObjectStoreWrapper.get(query) @return Promise<any>

  • query : Key or range of the key

Retrieves the value of the first record matching the given key or key range in query. If successful, result will be the value, or undefined if there was no matching record.


IDBObjectStoreWrapper.getKey(query) @return Promise<string | number | Date | ArrayBuffer | IDBArrayKey | ArrayBufferView | undefined>

  • query : Key or range of the key

Retrieves the key of the first record matching the given key or key range in query. If successful, result will be the key, or undefined if there was no matching record. Use this to check the existence of the key in object store.