124 lines
3.7 KiB
JavaScript
124 lines
3.7 KiB
JavaScript
|
"use strict";
|
||
|
|
||
|
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
||
|
|
||
|
Object.defineProperty(exports, "__esModule", {
|
||
|
value: true
|
||
|
});
|
||
|
exports.maxArgumentCount = maxArgumentCount;
|
||
|
exports.memoize = memoize;
|
||
|
exports.memoizeCompare = memoizeCompare;
|
||
|
|
||
|
var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof"));
|
||
|
|
||
|
var _lruQueue = require("./lruQueue.js");
|
||
|
|
||
|
// function utils
|
||
|
|
||
|
/**
|
||
|
* Memoize a given function by caching the computed result.
|
||
|
* The cache of a memoized function can be cleared by deleting the `cache`
|
||
|
* property of the function.
|
||
|
*
|
||
|
* @param {function} fn The function to be memoized.
|
||
|
* Must be a pure function.
|
||
|
* @param {Object} [options]
|
||
|
* @param {function(args: Array): string} [options.hasher]
|
||
|
* A custom hash builder. Is JSON.stringify by default.
|
||
|
* @param {number | undefined} [options.limit]
|
||
|
* Maximum number of values that may be cached. Undefined indicates
|
||
|
* unlimited (default)
|
||
|
* @return {function} Returns the memoized function
|
||
|
*/
|
||
|
function memoize(fn) {
|
||
|
var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {},
|
||
|
hasher = _ref.hasher,
|
||
|
limit = _ref.limit;
|
||
|
|
||
|
limit = limit == null ? Number.POSITIVE_INFINITY : limit;
|
||
|
hasher = hasher == null ? JSON.stringify : hasher;
|
||
|
return function memoize() {
|
||
|
if ((0, _typeof2.default)(memoize.cache) !== 'object') {
|
||
|
memoize.cache = {
|
||
|
values: new Map(),
|
||
|
lru: (0, _lruQueue.lruQueue)(limit || Number.POSITIVE_INFINITY)
|
||
|
};
|
||
|
}
|
||
|
|
||
|
var args = [];
|
||
|
|
||
|
for (var i = 0; i < arguments.length; i++) {
|
||
|
args[i] = arguments[i];
|
||
|
}
|
||
|
|
||
|
var hash = hasher(args);
|
||
|
|
||
|
if (memoize.cache.values.has(hash)) {
|
||
|
memoize.cache.lru.hit(hash);
|
||
|
return memoize.cache.values.get(hash);
|
||
|
}
|
||
|
|
||
|
var newVal = fn.apply(fn, args);
|
||
|
memoize.cache.values.set(hash, newVal);
|
||
|
memoize.cache.values.delete(memoize.cache.lru.hit(hash));
|
||
|
return newVal;
|
||
|
};
|
||
|
}
|
||
|
/**
|
||
|
* Memoize a given function by caching all results and the arguments,
|
||
|
* and comparing against the arguments of previous results before
|
||
|
* executing again.
|
||
|
* This is less performant than `memoize` which calculates a hash,
|
||
|
* which is very fast to compare. Use `memoizeCompare` only when it is
|
||
|
* not possible to create a unique serializable hash from the function
|
||
|
* arguments.
|
||
|
* The isEqual function must compare two sets of arguments
|
||
|
* and return true when equal (can be a deep equality check for example).
|
||
|
* @param {function} fn
|
||
|
* @param {function(a: *, b: *) : boolean} isEqual
|
||
|
* @returns {function}
|
||
|
*/
|
||
|
|
||
|
|
||
|
function memoizeCompare(fn, isEqual) {
|
||
|
var memoize = function memoize() {
|
||
|
var args = [];
|
||
|
|
||
|
for (var i = 0; i < arguments.length; i++) {
|
||
|
args[i] = arguments[i];
|
||
|
}
|
||
|
|
||
|
for (var c = 0; c < memoize.cache.length; c++) {
|
||
|
var cached = memoize.cache[c];
|
||
|
|
||
|
if (isEqual(args, cached.args)) {
|
||
|
// TODO: move this cache entry to the top so recently used entries move up?
|
||
|
return cached.res;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var res = fn.apply(fn, args);
|
||
|
memoize.cache.unshift({
|
||
|
args: args,
|
||
|
res: res
|
||
|
});
|
||
|
return res;
|
||
|
};
|
||
|
|
||
|
memoize.cache = [];
|
||
|
return memoize;
|
||
|
}
|
||
|
/**
|
||
|
* Find the maximum number of arguments expected by a typed function.
|
||
|
* @param {function} fn A typed function
|
||
|
* @return {number} Returns the maximum number of expected arguments.
|
||
|
* Returns -1 when no signatures where found on the function.
|
||
|
*/
|
||
|
|
||
|
|
||
|
function maxArgumentCount(fn) {
|
||
|
return Object.keys(fn.signatures || {}).reduce(function (args, signature) {
|
||
|
var count = (signature.match(/,/g) || []).length + 1;
|
||
|
return Math.max(args, count);
|
||
|
}, -1);
|
||
|
}
|