Security News
New Python Packaging Proposal Aims to Solve Phantom Dependency Problem with SBOMs
PEP 770 proposes adding SBOM support to Python packages to improve transparency and catch hidden non-Python dependencies that security tools often miss.
cache-service
Advanced tools
A tiered caching solution for JavaScript.
Upgrading from an older version? Please see the Breaking Change History section.
Do you use superagent? Check out superagent-cache to get superagent queries with cache-service built right in.
cache-service allows you to create redundant, cache-agnostic caching configurations. I've supplied a redis wrapper and node-cache wrapper as separate npm modules, but you can add any cache you want as long as you follow the same interface.
Require and instantiate cache-service as follows:
var cs = require('cache-service');
var cacheService = new cs();
This gives you the default configuration. Now you can cache like normal with the benefit of a tiered solution:
function getData(key, cb){
cacheService.get(key, function (err, response){
if (err) { // err is truthy if an actual error occurred
return cb(err);
}
if(response) { // response is null when there's no cache response
return cb(err, response);
}
performQuery(key, function (err, response){
var value = response.body.user;
cacheService.set(key, value);
return cb(err, value);
});
});
}
npm install cache-service --save
npm test
By following the Basic Usage example above, cache-service will:
All caches will have a defaultExpiration of 900 seconds unless specified otherwise.
To use a custom configuraiton, take advantage of the the two optional params you can hand to cache-service
's constructor as follows:
//Require the desired modules (make sure to add them to your package.json)
var nodeCacheModule = require('cache-service-node-cache');
var redisModule = require('cache-service-redis');
var cs = require('cache-service');
//Instantiate your cache modules
var nodeCache = new nodeCacheModule({defaultExpiration: 500});
var redisCache = new redisModule({redisEnv: 'REDISCLOUD_URL'});
//Setup your cache-service constructor params
var cacheServiceConfig = {verbose: true};
var cacheModules = [nodeCache, redisCache];
//Instantiate cache-service
var cacheService = new cs(cacheServiceConfig, cacheModules);
This code block will result in a tiered cache-service
instance that uses a primary in-memory cache and a fallback redis cache. For more information on what cache modules are available, see the Available Cache Modules section.
cache-service's constructor takes two parameters in the following order: cacheServiceConfig and cacheModules:
var cacheService = new cs(cacheServiceConfig, cacheModules);
This is where you set cache-service-level config options. Here are all the available options:
A namespace to be applied to all keys generated for this instance of cache-service.
When false, cache-service will log only errors. When true, cache-service will log all activity (useful for testing and debugging).
Let's say you have an instance of cache-service with two caches inside of it: cacheA and cacheB. If cacheA has a shorter defaultExpiration than cacheB and cacheA does not have the key for which you're looking, cache-service will then look in cacheB. If cache-service finds the desired key in cacheB and writeToVolatileCaches
is true
, cache-service will then write that key to cacheA.
This is particularly useful if you want to have a short-term in-memory cache with the most used queries and a longer-term redis cache with all of the cached data.
This is where you give cache-service the pre-instantiated caches you want it to use. Here's an example cacheModules
array:
//Instantiate some cache modules
var nodeCacheInstance = new nodeCacheModule({defaultExpiration: 500});
var redisCacheInstance = new redisCacheModule({redisEnv: 'REDISCLOUD_URL'});
//Place the new cache modules into the cacheModules array
var cacheModules = [
nodeCacheInstance
redisCacheInstance
]
This cacheModules
array will provide a primary node-cache instance with a fallback redis cache. The node-cache instance would have a 500-second defaultExpiration and the redis instance would have a 15-minute default expiraiton.
As you can see in the Cache Modules Array example above, each cache module constructor takes an object. This object is a cacheModuleConfig
object. I've added a more explicit example below for clarity:
var cacheModuleConfig = {defaultExpiration: 500, readOnly: true};
var cacheModule = new nodeCacheModule(cacheModuleConfig);
var cacheModules = [cacheModule];
Every cache module, regardless of the type of cache its wrapping, accepts a cacheModuleConfig
in its constructor. While the properties accepted by a given cache module's cacheModuleConfig
may vary, the properties listed below should always be available in all cache modules. If you need to know what unique cacheModuleConfig
properties a specific cache module accepts, visit that cache module's documentation.
An arbitrary identifier you can assign so you know which cache is responsible for logs and errors.
The expiration to include when executing cache set commands. Can be overridden via .set()
's optional expiraiton param.
By default, if two subsequent caches have the same defaultExpiraiton
, the second of the two caches will be checked in the event that the first cache does not have a key. If checkOnPreviousEmpty
is false
, cache-service will not check subsequent caches with the same defaultExpiration
.
Whether a cache should not be written to. Useful if you're sharing a redis cache with another team and your contract with them is that you will not alter their data.
Retrieve a value by a given key. All available cache modules attempt to JSON.parse
all values returned from .get()
and JSON.stringify
all values passed to .set()
.
Retrieve the values belonging to a series of keys. If a key is not found, it will not be in response
.
See the Using Background Refresh section for more about the
refresh
andcallback
params.
Set a value by a given key. All available cache modules attempt to JSON.stringify
all values passed to .set()
and JSON.parse
all values returned from .get()
.
IMPORTANT: The
callback
param takes precedence over therefresh
param. This means that, when only four arguments are passed and the last param is a function,cache-service
will assume the last param iscallback
. Similarly, if three params are passed and the third is a function,cache-service
will assume the third param iscallback
rather thanrefresh
. This is done to maintain backwards compatibility with versions released before thebackground refresh
feature was added.
Set multiple values to multiple keys
This function exposes a heirarchy of expiration values as follows:
expiration
property of a key that also contains a cacheValue
property will override all other expirations. (This means that, if you are caching an object, the string 'cacheValue' is a reserved property name within that object.)cacheValue
and expiration
as properties is not present, the expiration
provided to the .mset()
argument list will be used.defaultExpiration
will be applied.Delete a key or an array of keys and their associated values.
Flush all keys and values from an instance of cache-service.
IMPORTANT: While this property is available for viewing, modifying it at run time will almost certainly break cache-service.
This is the cacheModules
array you passed as the second param to the constructor.
With a typical cache setup, you're left to find the perfect compromise between having a long expiration so that users don't have to suffer through the worst case load time, and a short expiration so data doesn't get stale. cache-service
eliminates the need to worry about users suffering through the longest wait time by automatically refreshing keys for you.
By default, background refresh is off. It will turn itself on the first time you pass a refresh
param to .set()
.
cache-service
relies on the background refresh feature of the final cache you pass in the cacheModules
array. When you call .set()
and pass refresh
and callback
params, cache-service
routes the provided refresh
param to ONLY the final cache in the cacheModules
array. This means that you almost certainly want cache-service
's writeToVolatileCaches
property set to true
(it defaults to true
) so that the refresh written to the final cache will propogate forward to earlier caches
If desired, configure the following properties in the final cache in the cacheModules
array:
backgroundRefreshInterval
backgroundRefreshMinTtl
backgroundRefreshIntervalCheck
Background refresh is exposed via the .set()
command as follows:
cacheModule.set('key', 'value', 300, refresh, cb);
If you want to pass refresh
, you must also pass cb
because if only four params are passed, cache-service-node-cache
will assume the fourth param is cb
.
The refresh
param MUST be a function that accepts key
and a callback function that accepts err
and response
as follows:
var refresh = function(key, cb){
var response = goGetData();
cb(null, response);
}
A redis wrapper for cache-service or standalone use. Available on NPM.
An in-memory cache wrapper for cache-service or standalone use. Available on NPM.
A super-light in-memory cache for cache-service or standalone use. (This module is bundled with cache-service
and provided in the default configuration if you do not provide a cacheModules
constructor param.) Available on NPM.
cache-service allows you to inject any cache type that matches the Cache Module Interface. Any such modules should be stored in a separate NPM package so that you can include exactly what you need in your app and nothing more. I've provided a redis wrapper as an NPM module, so let's use that one for the purposes of these examples. To install a cache module, locate the cache-service-compatible package you want to use (we'll use cache-service-redis
), then install it as follows:
npm install cache-service-redis --save
Now that you've installed the cache modules you want, go ahead and require them in your project:
var redisCacheModule = require('cache-service-redis');
To instantiate it, simply pass a Cache Module Configuration object:
var redisCacheInstance = new nodeCacheModule({
redisEnv: 'REDISCLOUD_URL',
defaultExpiration: 60,
cacheWhenEmpty: false
});
Now let's pass our manually instantiated redisCacheInstance into our cache-service constructor:
var cacheService = new cs({}, [
redisCacheInstance
]);
Any cache can be used with cache-service as long as it follows the cacheModule interface. Whether you'd like to wrap your own cache for your app or create a new NPM module so everyone can benefit from your work, your cache wrapper should follow the instructions below.
Your cache wrapper must define the following top-level properties. Detailed descriptions for each of these can be found in the Cache Module Configuration Object documentation.
Your cache wrapper must define any of the following top-level functions you plan to use. Detailed descriptions for each of these can be found in the API documentation. (If you're making an open-source package rather than just using your custom cache wrapper on your own, please include all functionality.)
Once your cache meets the requirements listed above, you can follow the Install, Require, Instantiate, and Inject instructions to use it directly.
If this explanation doesn't cut it for you, have a look at cache-service-cache-module, cache-service-node-cache, and cache-service-redis to see how I'm doing it.
If you decide to create an open-source cache module for use with cache-service, please start your module name with 'cache-service-' so that people can find it easily.
To make a pull request to this repo, please
.flush()
:
1.1.0
). My goal was to accomidate two user groups:
cache-service
's complete API. Unminified and with comments, the module's JavaScript file is 165 lines. It will be injected into cacheService.caches
if no cacheModules
constructor param is provided.cacheModuleConfig
object as the second param in its constructor. That object has been replaced with the cacheModules
array which is an array of pre-instantiated cache modules. These changes make it so that your app need not include cache modules you will not use. Thanks @nickdaugherty for the suggestion.cacheModules
array or cache-service will throw an exception.cacheService.cacheCollection
is now cacheService.caches
.cache-service stores its caches in a top-level property called cacheCollection
. Older versions of cacheCollection
contained two arrays (preApi
and postApi
). In version 1.0.0, cacheCollection
has been simplified by eliminating its preApi
and postApi
properties. This means that if you have any advanced references such as cacheService.cacheCollection.preApi[0]
, you can simplify them to cacheService.cacheCollection[0]
.
Prior to this version, a complete .mget()
miss (meaning that zero of the passed keys were found) returned null
. Starting with this version, the same scenario returns an empty object.
FAQs
A tiered caching solution for node.
The npm package cache-service receives a total of 109 weekly downloads. As such, cache-service popularity was classified as not popular.
We found that cache-service demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
PEP 770 proposes adding SBOM support to Python packages to improve transparency and catch hidden non-Python dependencies that security tools often miss.
Security News
Socket CEO Feross Aboukhadijeh discusses open source security challenges, including zero-day attacks and supply chain risks, on the Cyber Security Council podcast.
Security News
Research
Socket researchers uncover how threat actors weaponize Out-of-Band Application Security Testing (OAST) techniques across the npm, PyPI, and RubyGems ecosystems to exfiltrate sensitive data.