A clerk is a white-collar worker who conducts general office tasks. The promise-clerk package conducts general Promise-related tasks to help with fault tolerance.
Table of Contents generated with DocToc
npm install --save github:tether/promise-clerk#v1.0.0
PromiseChainErrorCatcher
: catch and log errors (with context) in a Promise chainassertIsPromise
: converts non-promise objects into rejected Promises, returns Promise objects as-isQuitter
: convenient way to break a Promise chain, especially when the chain was constructed with a loopReallyDeterminedPropertyGetter
: if you can't trust a data source, provide one or more fallbacksTransaction
: Resume multi-step operations at the failed step
PromiseChainErrorCatcher can be used to collect and then summarize errors. It is designed to simplify debugging complex Promise chains
import { PromiseChainErrorCatcher } from 'promise-clerk'
const errorCatcher = new PromiseChainErrorCatcher('Your Module')
const promiseUnderTest = riskyFooBar()
.catch(errorCatcher.nameError('Foo Bar step'))
.catch(error => {
errorCatcher.push(error)
return riskyQuux()
.then(errorCatcher.recordSuccess('Quux step'))
.catch(errorCatcher.nameError('Quux step'))
})
.then(value => {
return doNotCareIfThisFailsOrSucceeds()
.catch(errorCatcher.nameError('Unimportant step'))
.catch(errorCatcher.catchError)
.then(() => doSomethingThatFails())
.catch(errorCatcher.nameError('Failing step'))
})
.catch(errorCatcher.handleFinalError())
This will reject with:
Your Module has failed.
- Foo Bar step failed because <error message from riskyFooBar>
- Quux step resolved with "value from riskyQuux"
- Unimportant step failed because <error message from doNotCareIfThisFailsOrSucceeds>
- Failing step failed because <error message from doSomethingThatFails>
Note: this was recently extracted from the ReallyDeterminedPropertyGetter and may not be fully generalized yet.
import { assertIsPromise } from 'promise-clerk'
assertIsPromise(probablyReturnsPromise())
.then(doSuffAfterPromise())
assertIsPromise always returns a Promise. If it was not given a Promise, it returns a rejected Promise, otherwise it returns the original Promise. It considers the argument a Promise if it's truthy and has a truthy property named then
.
Note, this was recently extracted from ReallyDeterminedPropertyGetter and PromiseChainErrorCatcher, and may not be fully generalized yet.
Quitter is a convenient way to break a Promise chain, especially when the chain was constructed with a loop
import { Quitter } from 'promise-clerk'
const quitter = new Quitter()
const promise = Promise.resolve()
listOfEndpoints.forEach(endpoint => {
promise.then(value => {
quitter.maybeQuit(new Error('not continuing'))
return endpoint.getData()
.then(data => {
quitter.quitOnCondition(data.isReallyBad)
return data
})
})
})
In this example, for each endpoint in the list, you getData
-- but if any endpoint returns data that { isReallyBad: true }
, then you don't fetch the data from any more endpoints.
You could also use the quit
method instead of quitOnCondition
if there is no condition.
ReallyDeterminedPropertyGetter provides a way to define an external value (e.g. accessible via some API) with one or more secondary data sources to fall back to in case the primary one is unavailable
import { ReallyDeterminedPropertyGetter } from 'promise-clerk'
const movieListing = await new ReallyDeterminedPropertyGetter()
.verify(movies => movies.length === movieTitles.length)
.primarySource(() => mainAPI.getMovieListingsPromise(movieTitles))
.secondarySource(() => Promise.all(movieTitles.map(title => otherAPI.getMovie(title))))
.secondarySource(() => new Promise((resolve, reject) =>
oldSchoolHttpGet('https://movies.com?titles=' + movieTitles,join(','), (err, results) => err ? reject(err) : resolve(results))
))
.synchronizeWithPrimarySource(movies => mainAPI.updateMovieListings(movies))
.ignoreSynchronizationErrors() // Only if you don't care whether mainApi.updateMovieListings() succeeds or fails
.get()
The basic algorithm is:
- if no
verify
method is provided, consider all values verified. Otherwise, a value is considered verified if theverify
method returns true when provided the value - try getting the value from a primary source, return it if found and verified
- try getting value from a secondary source, return it if found and verified
- repeat until the value is found and verified or we're out of sources
Available methods are:
primarySource(getter: Function<Promise>)
// Required, may only be called once. Register a primary sourcesecondarySource(getter: Function<Promise>)
// Optional, may be called any number of times. Register a secondary source (will be attempted in the order added)verify(verify: (value) => boolean)
// Optional, may only be called once. Will be called for each found value. Values for whichverify
returns false will be ignoredsynchronizeWithPrimarySource((value) => Promise)
// Optional, may be called any number of times. Registers a callback which is called if the primary source fails but a secondary source succeedsignoreSynchronizationErrors()
// Optional. If it has been called, then any errors produced by a primarySourceSynchronizer function are ignored instead of causing the mainget
method to rejectget()
// Returns a Promise which resolves with the result, if available. May be called repeatedly as long as the returned Promise resolves before callingget
again
Note: this class uses the Builder Pattern (read more: https://en.wikipedia.org/wiki/Builder_pattern) to avoid having a long list of constructor arguments, some being optional, others required, etc..
Transaction class provides a robust way to run a series of asynchronous functions in sequence
import { Transaction } from 'promise-clerk'
const transaction = new Transaction([
() => doThingOne(),
() => doThingTwo()
() => doThingThree()
]).then(() => tellUserThatOperationSucceeded())
transaction.onResume(() => showLoadingIndicator())
transaction.onSpecificError('404', () => specialHandlingOf404Error())
transaction.onSpecificError(/5\d\d/, () => specialHandlingOf5xxError())
transaction.onError(error => {
tellUserThatSomethingWentWrong(error.message)
.then(userAnswer => {
if(userAnswer === 'try again') {
transaction.resume() // Repeats the failed step and continues with subsequent steps
}
})
})
For support, bug reports and or feature requests please make sure to read our community guidelines and use the issue list of this repo and make sure it's not present yet in our reporting checklist.
The open source community is very important to us. If you want to participate to this repository, please make sure to read our guidelines before making any pull request. If you have any related project, please let everyone know in our wiki.:1
Need inspiration for naming or design? Consider Netflix's Hystrix fault-tolerance library
- Each utility should have a unit test named
utility-name.test.js
in the test folder.
- We use Standard JS -- please run
npm run standard
and deal with any issues before committing.
For support, bug reports and or feature requests please make sure to read our community guidelines and use the issue list of this repo and make sure it's not present yet in our reporting checklist.
The MIT License (MIT)
Copyright (c) 2018 Petrofeed Inc
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.