Skip to content

Dexie.vip()

David Fahlander edited this page Jun 11, 2014 · 11 revisions

Sample

db.on('ready', function() {
    return new Dexie.Promise(function (resolve, reject) {
        // At this point we are still VIP:ed, but if we use an asyncronic
        // api without encapsulating it in another Dexie.Promise, we will
        // loose our VIP status. One example is when using setTimeout():
        setTimeout(function(){
            // Here we are no longer auto-VIP:et
            // If we don't VIP ourselves here, database will be blocked for
            // us because it is waiting for ourselves to finish! Deadlock will occur!
            db.vip(function () {
                // Make db calls here will ignore the blocked state of db:
                db.friends.put({name: "Urban"}).then(resolve).catch(reject);
            });
        }, 0);
    });
});

Description

Method to be used by subscribers to the on('ready') event. This will let caller through to access DB even when it is blocked while the db.ready() subscribers are firing.

Sequence of Database Open:

1. Dexie.open() is called
2. backend IDB opens DB
3. backend IDB.onsuccess is fired
4. on('ready') is firing
5. Waiting for returned Promise from on('ready') subscriber
6. All on('ready') subscribers' Promises are resolved
7. The 'blocked' state is cleared and any pending requests are resumed.

Note that this method is only useful for on('ready') subscribers that is returning a Promise from the event. If not using vip() the database could deadlock since it wont open until the returned Promise is resolved, and any non-VIPed operation started by the caller will not resolve until database is opened.

The on('ready') subscriber is already VIP:ed when executed. Any Promise that it creates directly or indirectly will also derive the VIP status. So will also any then() callback from the promise do. This means that it is normally not needed to use the vip() method.

Once the executing Promise is VIP:ed, any then() callbacks of that Promise will also become vipped. This does not mean that the VIP status is global - it is stored as a property in the Promise Specific Data structure, Promise.PSD which works the same way for promises as Thread-Specific-Data (TSD) does for threads.

Solving Above Sample Without vip()

The following sample solves the deadlock issue without using the vip() method. Note that it does the same thing as the sample on the top of this page but since it surrounds the external setTimeout() call with a Promise, the VIP status will remain in the then() method of the Promise.

db.on('ready', function() {
    return new Dexie.Promise(function (resolve, reject) {
        // At this point we are still VIP:ed, but if we use an asyncronic
        // api without encapsulating it in another Dexie.Promise, we will
        // loose our VIP status. One example is when using setTimeout():
        setTimeout(function(){
            resolve();
        }, 0);
    }).then(function() {
        // Since we are executing in the then() clause of an already VIP:ed
        // Promise, we dont need to VIP ourselves here.
        return db.friends.put({name: "Urban"});
    });
});

Situations when vip() is required

As shown in the above example, it was not required to call vip() in the setTimeout() sample since we could workaround it by letting the async callback call resolve() and then follow it up in the then() method instead. But there might be more complex situations when working with various asyncronic API:s where vip() could be handy.

Clone this wiki locally