/data-capsule

A pluggable capsule for storing key/value data for your application

Primary LanguageJavaScript

data-capsule

A pluggable capsule for storing key/value data for your application.

Following plugins support is built in:

  1. LocalStorage - works with browser's native local storage, plugin automatically manages expiration and auto-cleans to never exceed quota.
  2. WixStorage - reference implementation (just for reference, do not try to use) of storage plugin that communicates with wix server to store the data.
  3. CachedStorage - works with two plugins, will use one of them for persistnet storage and the second for local cache in order to do less requests to more expensive persistent storage. this can be used for example with WixStorage as persistent storage on remote servers combined with LocalStorage for local cache.
  4. FrameStorage - a nice way to offload storage request from frame to parent window. parent window can use LocalStorage, CachedStorage or any other plugin you can think of. this is extremely helpful for working around 3rd party site data issues.

installation

$ npm install --save data-capsule

DataCapsule

DataCapsule is the one responsible for your needs. When you create a capsule you must provide a strategy (see below) for how data will be saved. In addition, you can optionally provide a namespace and scope for your capsule. If you didn't provide namespace in costructor, you must do so for each operation you do on the capsule. However, scope is optional. The idea is that namespace distinguishes your data from other applications who are using the same strategy (for example in case of localstorage, other applications that run in the same origin), and scope distinguishes your data from data the same application might write when in different context (for example, user id is often used as scope). Even if you provided namespace and scope in constructor, you may override it for specific operation by passing different namespace or scope when passing options to one of the capsule's methods.

constructor(options) {
  // options: {
  //   strategy: Any class which implements BaseStorage (see below)
  //   namespace: String (optional)
  //   scope: Any serializable type (optional)
  // }
}

After instantiating a DataCapsule with your chosen strategy (see some strategies below), you can start using the capsule. The capsule provides only four methods:

  1. setItem(key, value, options): Promise<void> - Just pass a key (string) and value (any serializable type) and it will be saved. As described above, the options object may contain namespace and scope if you didn't do so in constructur or would like to override the values you passed in constructor. Only in setItems, you can also pass expiration (in seconds) in options. If you didn't pass expiration, the storage strategy you provided may decide on a default expiration. The returned promise indicates if operation was successful.
  2. getItem(key, options): Promise<value> - Just pass a key (string) and returned promise will get resolved with the value that you previously saved. The options object may contain namespace and scope if you didn't do so in constructur or would like to override the values you passed in constructor. In case no such key exists in storage, promise will get rejected with exception NOT_FOUND (you can import {NOT_FOUND} from 'data-capsule and test if err === NOT_FOUND if you want to distinguish this case from other errors).
  3. removeItem(key, options): Promise<void> - Just pass a key (string) and returned promise will get resolved with the value that you previously saved. The options object may contain namespace and scope if you didn't do so in constructur or would like to override the values you passed in constructor. The returned promise indicates if operation was successful.
  4. getAllItems(options): Promise<key/value> - Returned promise will get resolved with a key/value map containing all keys that are stored in this namespace/scope and their values. Keys lookup will be done according to the namespace and scope you passed in constructor and in options. As usual: If you didn't provide namespace in costructor, you must do so for each operation you do on the capsule. However, scope is optional.

LocalStorage

No parameters needed in LocalStorage constructor. Note the LocalStorageCapsule short form below.

import {DataCapsule, LocalStorageStrategy} from 'data-capsule';

const capsule = new DataCapsule({
  strategy: new LocalStorageStrategy(),
  namespace: 'wix'
});
await capsule.setItem('shahata', 123);
console.log(await capsule.getItem('shahata')); // logs 123

And shorter alternative:

import {LocalStorageCapsule} from 'data-capsule';

const capsule = new LocalStorageCapsule({namespace: 'wix'});
await capsule.setItem('shahata', 123);
console.log(await capsule.getItem('shahata')); // logs 123

CachedStorage

Constructor accepts remoteStrategy and localStrategy. Get operations are first tried against localStrategy and if no local cache exists, we then fallback to remoteStrategy ans eventually cache to localStrategy. Set operations are cached to localStrategy as well. Note that localStrategy cache is always set with expiration period of one hour in order to avoid stale cache issues. The localStrategy is new LocalStorageStrategy() bye default, so you don't have to pass it, but anyway you are better off using the short form below:

import {DataCapsule, CachedStorageStrategy, LocalStorageStrategy, WixStorageStrategy} from 'data-capsule';

const capsule = new DataCapsule({
  strategy: new CachedStorageStrategy(new WixStorageStrategy(), new LocalStorageStrategy()),
  namespace: 'wix'
});
await capsule.setItem('shahata', 123); //send setItem request to server
console.log(await capsule.getItem('shahata')); // logs 123
// ^ does not send getItem request to server since value is cached

And shorter alternative:

import {LocalStorageCachedCapsule, WixStorageStrategy} from 'data-capsule';

const capsule = new LocalStorageCachedCapsule({
  remoteStrategy: new WixStorageStrategy(),
  namespace: 'wix'
});
await capsule.setItem('shahata', 123); //send setItem request to server
console.log(await capsule.getItem('shahata')); // logs 123
// ^ does not send getItem request to server since value is cached

FrameStorage

This allows frame windows to send commands via postMessage while host window eventually invokes command on actual storage strategy. In host window, all you have to do is create a FrameStorageListener and start it and finally stop it if and when you are done. Note:

  1. FrameStorageListener optionally gets a storage strategy as argument. By default, this is new LocalStorageStrategy(), but you can decide to override this default with any other strategy.
  2. FrameStorageListener.start receives mandatory callback as arguement. This callback is invoked for each valid message host window gets. The callback needs to return a boolean that will indicate whether this source (the window who sent the message), origin (its origin) and token (an optional token passed to FrameStorageStrategy below) can be trusted. If callback returns false, message is ignored.
import {FrameStorageListener} from 'data-capsule';

const listener = new FrameStorageListener();
listener.start((source, origin, token) => token === 'secret');
//...
listener.stop();

In frame window you create a DataCapsule with a FrameStorageStrategy strategy. Note FrameStorageStrategy constructor requires the follwing arguments:

  1. target - The window which runs the listener that will work with the real storage on our behalf. Typically this will be window.top since the whole idea of this strategy is to use 1st party data storage instead of problematic 3rd party.
  2. origin - The origin that target window must be in for the events to be dispatched. This value can be '*' but this is widely unrecommended since it is dangerous and opens your application for malicious attacks.
  3. token - This token is later passed to the start callback in order to help verify that the messages from this frame can be trusted. What this token represents, how it gets verified in callback and how it is initially passed to frame is up to you.
import {DataCapsule, FrameStorageStrategy} from 'data-capsule';

const capsule = new DataCapsule({
  strategy: new FrameStorageStrategy(window.top, '*', 'secret'),
  namespace: 'wix'
});
await capsule.setItem('shahata', 123);
console.log(await capsule.getItem('shahata')); // logs 123

BaseStorage

Following you can see the interface for BaseStorage. The DataCapsule and all storage strategies implement this interface and defer only in constructor.

Don't forget that you can also implement your own storage strategy! All you need to do is implement something similar to the example below and then you can pass it as strategy to DataCapsule.

import {BaseStorage} from 'data-capsule';

class MyStorageStrategy extends BaseStorage {
  setItem(key, value, options) {
    // key: String (mandatory)
    // value: Any serializable type (mandatory)
    // options: {
    //   namespace: String (mandatory)
    //   scope: Any serializable type (optional)
    //   expiration: Number of seconds (optional)
    //   ...: Any additional custom options are passed to you
    // }
    // returns a promise that resolves on success and rejects on failure
    return Promise.resolve();
  }

  getItem(key, options) {
    // key: String (mandatory)
    // options: {
    //   namespace: String (mandatory)
    //   scope: Any serializable type (optional)
    //   ...: Any additional custom options are passed to you
    // }
    // returns a promise that resolves on success and rejects on failure
    // promise resolves with value which is assigned to key in this namespace/scope
    return Promise.resolve(123);
  }

  removeItem(key, options) {
    // key: String (mandatory)
    // options: {
    //   namespace: String (mandatory)
    //   scope: Any serializable type (optional)
    //   ...: Any additional custom options are passed to you
    // }
    // returns a promise that resolves on success and rejects on failure
    return Promise.resolve();
  }

  getAllItems(options) {
    // options: {
    //   namespace: String (mandatory)
    //   scope: Any serializable type (optional)
    //   ...: Any additional custom options are passed to you
    // }
    // returns a promise that resolves on success and rejects on failure
    // promise resolves with key/value map of all keys in this namespace/scope
    return Promise.resolve({key1: 123, key2: 456});
  }
}