Efficient JavaScript client side cache. When data is fetched from a backend then it can be cached locally, for example for an App that's running on a mobile device.
- Cache data is locally stored in an in-memory cache (as a plain JavaScript object)
- Values in the cache may be anything you can store in a JavaScript variable: Strings, Arrays or JSON Objects.
- Every value you put into the cache can have a limited time to life. If that TTL expires, then the value will be refetched from the backend.
- Populating-Cache is not just a simple key=value store. Values can be stored under a path.
- A object in the cache may reference another object in the cache via its path, e.g.
posts/ID4711.createdBy
may reference auser/ID42
entity. These database references (DBrefs) can automatically be populated when getting values from the cache. - Populating cache is 100% tested and highly configurable.
- No dependencies!
Install the npm dependency in your project: npm install populating-cache
import PopulatingCache from 'populating-cache' // the module exports a class
// Create a new cache intance (More details on "fetchFunc" later.)
let cache = new PopulatingCache({fetchFunc: (path) => Promise.resolve("valueFetchedFromBackend")})
// PUT values into the cache and store them for later
cache.put("key1", "Just any value")
cache.put("key2", {foo: "bar"} )
// GET a value back from the cache:
let value = await cache.get("key1") // value == "Just any value"
// PUT an array into the cache
cache.put("myArray", [0,1,2,3])
cache.put("myArray[1]", 111) // replace one array element
cache.put("myArray[]", 4) // append to array
let arrayItem = await cache.get("myArray[2]") // 2
let fullArray = await cache.get("myArray") // [0,111,2,3,4]
// PUT an object with an _id into the cache
let aPost = { _id:4711, title: "Blog post title"}
cache.put("posts/4711", aPost)
let cachedPost = await cache.get("posts/4711") // cachedPost === aPost
When you try to get
a value that is either not yet in the cache or already expired, then it is fetched from the backend. Populating-cache will
- call your
fetchFunc(path)
to receive the current value from the backend put
the returned value into the cache- update its TTL
- and then
get(path)
returns the value as returned by yourfetchFunc()
/**
* When a value needs to be fetched from the backend
* then populating-cache will call this function that you must provide.
* @param {Array} path to the value that needs to be fetched,
* e.g. [{posts: 4711}, "comments"]
* @return {Promise} value that you fetched from your backend
*/
let fetchFunc = function(path) {
// [...] call backend, make REST request, etc.
return Promise.resolve(valueFetchedFromBackend)
}
// Configuration of your cache instance. See below for defaults.
const cacheConfig = {
fetchFunc: fetchFunc,
ttl: 60 * 1000 // time to live for elements in the cache
}
let cache = new PopulatingCache(cacheConfig)
You may simply store values under String keys. But Populating-Cache
is much more powerfull. It can also store values in a tree structure. The cached elements in this tree are identified by the path from the root of the tree to the element.
Values in the cache (normally) sit at the leaves of this tree.
Example: Cache the tags of a post with _id="af3d-e3ff"
. If a post with that _id
does not yet exist, then it will automatically be created in the cache.
cache.put(["post/af3d-e3ff", "tags"], ["tag1", "tag2"])
When you get
something from the cache, you will receive that cache element and everything under it.
A path
defines where a value will be stored in the cache. It is an array of path elements from the root of the cache to the position of value in the cache. When you call put(path, value)
then the algorithm walks along path
and and then stores value
at the end of this path. All intermediate elements along the path (objects & arrays) will automatically be created if necessary.
Normalized form | Shortcut | Description |
---|---|---|
["myKey"] | "myKey" | Key at root level in cache |
["abc", "def", ghi"] | abc.def.ghi | deep path into cache |
["myArray[3]"] | "myArray[3]" | the n-th element of an array |
[{object: id}] | "object/id" | the object element of the array that has that _id |
["abc", {foo: "ID-4711}, "bar"] | abc.foo/ID-4711.bar | all types of elements can appear in one path |
You may pass the shortcut form to PUT and GET. Populating-cache will always call your fetch function with the normalized form.
// Store value under a path
cache.put(["parentKey", "childArray[3]"], value) // cache.parentKey.childArray[3] = value
// Populting-Cache can automatically find array items by their _id and store values under this item.
// Creator of comment with id "ef37d" in post with id "id42"
// (Here we store a user object in the cache.)
cache.put(["posts/af3d-e3ff", "comments/4ccf-ff33", "createdBy"], {_id:"14a3-2e2f", name: "John Doe", email: "john@doe.com"})
// Path elements can also be objects. This results in the same path as the example above.
cache.put([{posts:"af3d-e3ff"}, {comments:"4ccf-ff33"}, "createdBy"], user)
Each value in the cache can have a time to life after which it expires. When you try to get
an expired value, then it is refetched from the backend by calling your fetchFunc()
.
For simple keys this is simple. But since populating-cache is a tree structure, this can become more complex.
When you fetch a deep value at the end of a long path
, then all intermediate elements in the cache are also checked.
If an intermediate element is expired, then the element at this subpath is fetched.
Example:
// put a user object into the cache
cache.put("data.user", { name: "username", email: "john.doe@domain.com")
// ... some time passes, until the user entity expires in the cache ...
let email = cache.get("data.user.email")
Since the user
element is already expired, this will call fetchFunc
. The path
parameter will be the parsed sub path to the expired user
element ["data", "user"]
:
let parsedSubPath = ["data", "user"]
let valueFromBackend = this.fetchFunc(parsedSubPath)
this.put(parsedSubPath, valueFromBackend) // update TTL
return parsedSubPath
The data returned by the backend my contain references to other entities. These references are inspired by MongoDB DBrefs and the populate()
function of the awesome Mongoose lib.
When GET-ing a value from the cache, then DBrefs can automatically be resolved from within the cache content. Let's use the following content of a populating-cache
as an example:
cacheMetadata = {
"posts": [
{
_id: 4711,
text: "This is an example post"
comments: [
{
_id: 101,
commentText: "This is a comment by user1",
createdBy: {
$refPath: "users/901" // referenced `path` to other entity in cache
}
}
]
}
],
"users": [
{ // This is the referenced user
_id: 901,
email: "user1@domain.com"
},
]
}
A simple list of posts
. Each post can have comments
. And each comment is createdBy
a user
.
But the createdBy-user is not stored again and again for every comment. Instead, users are cached seperately under the top-level key users
. Each comment then references one user. The $ref
attribute is the name of the top-level key. And $id
is the _id of the user under that key that created the comment.
When a path in a call to get(path)
spans a DBref, then this ref is automatically resolved:
let path = [{posts:4711}, {comments: 101}, "createdBy", "email"]
let result = await cache.get(path)
// result is now "user1@domain.com"
When a DBref is resolved, then of course the TTL of the referenced target entity (in our example user/901
) is taken into account. If that user's data in the cache is expired, then a query for that user will be sent to the backend. The response of the backend will be used to update the user data in the cache. The TTL in the metadata will be refreshed. And this updated data will be used to populate the DBref.
You can subscribe to changes in the cache. Your listener will be notified, when an element is PUT into the cache.
const cache = new PopulatingChache({fetchFunc: fetchFunc})
const onPutListern = (path, value) => console.log(value + "was PUT into cache at path "+path)
cache.subscribe("foo", onPutListener)
cache.put("foo.some.path", "dummyValue")
// onPutListener("foo.some.path", "dummyValue") has now been called.
By default listeners are called when a value is put at or below their path. So by default its actually a path prefix.
You can listen exextly to changes of on element by passing a third argument (true) to the subcribe function: cache.subsribe("some.path", exactListener, true)
exactListener will only be called, when exactly this element at this path changes.
When you want to be notified about all changes in the cache, then you can register a listener at root level: cache.subscribe("", rootListener)
put("key", { foo: "bar"})
put("key", { baz: "boo"}, { merge: true}) // Will throw an error if cache[key] is not an object.
// Cache an array
put("array", ["one", "two", "three"])
// Replace the third element of array with "item2"
// This will create the array if necessary.
// and throw an error if "array" is not an array.
put("array[2]", "item2")
// replace the third element of array with an object
put("array[2]", { foo: "bar"})
// merge property "baz" into the third element of array.
put("array[2]", { baz: "boo"}, { merge: true})
// Array now looks like this:
assert.equals(await cache.get("array"), ["one", "two", {
foo: "bar",
baz: "boo"
}])
// Append "item" to array.
// Will create a new array if necessary.
// Will throw an error if "array" is an object.
put("array[]", "item" )
// Be carefull, don't forget the brackets! This would store the String value "item"
// under the key "array" and replace your array. This is most probably not what you want.
put("array", "item" )
// You can also pass append: true as an option
put("array[]", "item", { append: true} )
put("array", "item", { append: true} ) // this is ok, but not recommended
// Example: User created a new comment under a blog post
let newComment = {_id:42, text:"This is a comment"}
// Append newComment to array of comments of post with _id = 4711
put(["posts/4711", "comments[]"], newComment)
// This will throw an id-mismatch error, because newComment has _id=42 and not 9999
put(["posts/4711", "comments/9999"], newComment)
// Will implicitly add _id:42 to the passed value.
put(["posts/4711", "comments/42"], {text: "just something"})
// Merge property "baz" into existing comment with _id = 42
put(["posts/4711", "comments/42"], {baz:"boo"}, {merge:true})
Object IDs can be numeric or alphanumeric UUIDs. The name of the _id
property can be configured.
Populating-cache
is heavily tested. Have a look at the JEST test cases. There you can also learn a lot about how to use populating-cache.
You can run all tests easily: npm run test
Thanks to the creators of node-cache for coffeescript inspiration.
Populating Cache
A lightweight client side cache that can store values in a tree structure. https://github.com/doogiemuc/populating-cache
fetchFunc
Function async function that will be called to fetch elements from the backend. One param: pathconfig
Object configuration parameters that may overwrite the DEFFAULT_CONFIG
Cache value
under the given path
in this cache. The value may then be retrieved back with a call to get(path)
.
All intermediate objects along the path will be created in the cache if necessary.
Then value
is stored as the leaf at the end of path with the given ttl
or the configured defaultTTLms
.
path
Array path under which thevalue
shall be stored in the cachevalue
Any The value to store in the cache.ttl
Number time to live / how long value can be stored in the cache. Defaults to config.defaultTTLms (optional, defaultthis.config.defaultTTLms
)merge
(optional, defaultfalse
)
Returns Object the cache instance, so that calls to put() can be chained: myCache.put("foo", "bar").put("foo2", "baz2")
Fetch a value from the cache. If the value isn't in the cache or if it is expired,
then the backend will be queried for the value under path
.
When the backend is called, then the returned value will again be stored in the cache and its TTL will be updated.
path
(String | Array) array that forms the path to the value that shall be fetched. Each array element can be one of three formats: - a plain "attributeName" for object properties in the cache. - { arrayAttr: "abde3f"} for the array element with that id. - "array[index]" for array elements. Index must be a positive. For example: [{posts:5}, {comments:42}] is the comment with id 42 of the post with id 5 For a REST backend this can be translated to the REST resource at /posts/5/comments/42callBackend
(optional, defaultthis.CALL_BACKEND_WHEN_EXPIRED
)populate
Boolean Automatically populate DBrefs from this cache if possible. (optional, defaultthis.config.populate
)force
Boolean Force calls to backend, even when cache element is not yet expired
Returns any (A Promise that resolves to) the fetched value. Either directly from the cache or from the backend.
This method decides if the backend needs to be called to fetch a given value. The backend will be called, IF
- cacheElem is null, ie. not yet in the cache
- cacheElem is expired, because its metadata.ttl is in the past.
- or when callBackend === FORCE_BACKEND_CALL
When the backend is call, then the returned value is PUT() back into the cache with and updated TTL.
path
Array The full path to cacheElemcacheElem
Any the leaf element at the end of path or null if not in the cache yetmetadata
Object metadata for cacheElemcallBackend
Number always call backend if true
Returns Promise element either directly from the cache or fetched from the backend
Delete an element from the cache. It's value will be set to undefined. If path points to an array element, then that array element will be set to undefined, so that the length of the array does not change.
path
Array path to the element that shall be deleted
Check if a path points to a defined and not yet expired value in the cache.
If the value is expired, the backend will not be called.
If path points to an undefined
value, then isInCache will return false.
path
Array path to an element in the cachepopulate
Boolean wether to populate DBrefs along path (optional, defaultthis.config.populate
)
Returns Boolean true if there is a value in the cache and it is not yet expired.
Get (a direct reference!) to all the data in the cache.
The cacheData is exactly as returned by fetchFunc(path)
While you put
values into the cache, Populating-Cache automatically
creates a second parallel tree of metadata next to the cacheData
.
The metadata contains the time to life (TTL) of the value in the cache.
_ttl
is the number of milliseconds since the epoc, when the value expires.
path
Array fetch metadata of a specific element. If null or undefind, that all metadata of the whole cache is returned.
Returns Object metadata of cached value under path, e.g. { _ttl: 55235325000, _type: "Integer"}
Completely empty out the cache. This also clears out all metadata.
Recursively walk through the cacheData and delete all expired elements. You MAY call this from time to time to optimize the cache's memory consumption
Parse a path into an array of { key, id, index } objects.
-
path
(String | Array) plain string or array of path elements -
Throws any an Error when path or a path element is invalid
Returns Array Array of parsed objects { key, id, index }. id
or index
may be undefind in each path element.
Convert the given path to a REST URL path.
path
Array a populating-cache path array
Default configuration for a cache. You can overwrite these when creating a new PopulatingCache instance. Each cache instance can have its own configuration.