Security News
Node.js EOL Versions CVE Dubbed the "Worst CVE of the Year" by Security Experts
Critics call the Node.js EOL CVE a misuse of the system, sparking debate over CVE standards and the growing noise in vulnerability databases.
inline-loops.macro
Advanced tools
Iteration helpers that inline to native loops for performance
inline-loops.macro
is a babel macro that will inline calls to the iteration methods provided, replacing them with for
loops (or for-in
in the case of objects). While this adds more code, it is also considerably more performant than the native versions of these methods. When working in non-JIT environments this is also faster than equivalent runtime helpers, as it avoids function calls and inlines operations when possible.
This is inspired by the work done on babel-plugin-loop-optimizer, but aims to be both more targeted and more full-featured. Rather than globally replace all native calls, the use of macros allow a controlled, opt-in usage. This macro also supports decrementing array and object iteration, as well as nested usage.
You can use it for everything, only for hotpaths, as a replacement for lodash
with legacy support, whatever you see fit for your project. The support should be the same as the support for babel-plugin-macros
.
import { map, reduce, someObject } from 'inline-loops.macro';
function contrivedExample(array) {
const doubled = map(array, (value) => value * 2);
const doubleObject = reduce(doubled, (object, value) => ({
...object,
[value]: value
}, {});
if (someObject(doubleObject, (value) => value > 100)) {
console.log('I am large!');
}
}
every
(MDN documentation)
everyRight
=> same as every
, but iterating in reverseeveryObject
=> same as every
but iterating over objects intead of arraysfilter
(MDN documentation)
filterRight
=> same as filter
, but iterating in reversefilterObject
=> same as filter
but iterating over objects intead of arraysfind
(MDN documentation)
findLast
=> same as find
, but iterating in reverse (MDN documentation)findObject
=> same as find
but iterating over objects intead of arraysfindIndex
(MDN documentation)
findLastIndex
=> same as findIndex
, but iterating in reverse (MDN documentation)findKey
=> same as findIndex
but iterating over objects intead of arraysflatMap
(MDN documentation)
flatMapRight
=> same as flatMap
, but iterating in reverseforEach
(MDN documentation)
forEachRight
=> same as forEach
, but iterating in reverseforEachObject
=> same as forEach
but iterating over objects intead of arraysmap
(MDN documentation)
mapRight
=> same as map
, but iterating in reversemapObject
=> same as map
but iterating over objects intead of arraysreduce
(MDN documentation)
reduceRight
=> same as reduce
, but iterating in reverse (MDN documentation)reduceObject
=> same as reduce
but iterating over objects intead of arrayssome
(MDN documentation)
someRight
=> same as some
, but iterating in reversesomeObject
=> same as some
but iterating over objects intead of arraysInternally Babel will transform these calls to their respective loop-driven alternatives. Example
// this
const foo = map(array, fn);
// becomes this
const _length = array.length;
const _results = Array(_length);
for (let _key = 0, _value; _key < _length; ++_key) {
_value = array[_key];
_results[_key] = fn(_value, _key, array);
}
const foo = _results;
If you are passing uncached values as the array or the handler, it will store those values as local variables and execute the same loop based on those variables.
One extra performance boost is that inline-loops
will try to inline the callback operations when possible. For example:
// this
const doubled = map(array, (value) => value * 2);
// becomes this
const _length = array.length;
const _results = Array(_length);
for (let _key = 0, _value; _key < _length; ++_key) {
_value = array[_key];
_results[_key] = _value * 2;
}
const doubled = _results;
Notice that there is no reference to the original function, because it used the return directly. This even works with nested calls!
// this
const isAllTuples = every(array, (tuple) =>
every(tuple, (value) => Array.isArray(value) && value.length === 2),
);
// becomes this
let _determination = true;
for (
let _key = 0, _length = array.length, _tuple, _result;
_key < _length;
++_key
) {
_tuple = array[_key];
let _determination2 = true;
for (
let _key2 = 0, _length2 = _tuple.length, _value, _result2;
_key2 < _length2;
++_key2
) {
_value = _tuple[_key2];
_result2 = Array.isArray(_value) && _value.length === 2;
if (!_result2) {
_determination2 = false;
break;
}
}
_result = _determination2;
if (!_result) {
_determination = false;
break;
}
}
const isAllTuples = _determination;
Inevitably not everything can be inlined, so there are known bailout scenarios:
return
statements (as there is no scope to return from, the conversion of the logic would be highly complex)return
statement is not top-level (same reason as with multiple return
s)this
keyword is used (closure must be maintained to guarantee correct value)That means if you are cranking every last ounce of performance out of this macro, you may want to get cozy with ternaries.
import { map } from 'inline-loops.macro';
// this will bail out to storing the function and calling it in the loop
const deopted = map(array, (value) => {
if (value % 2 === 0) {
return 'even';
}
return 'odd';
});
// this will inline the operation and avoid function calls
const inlined = map(array, (value) => (value % 2 === 0 ? 'even' : 'odd'));
Some aspects of implementing this macro that you should be aware of:
*Object
methods do not perform hasOwnProperty
checkThe object methods will do operations in for-in
loop, but will not guard via a hasOwnProperty
check. For example:
// this
const doubled = mapObject(object, (value) => value * 2);
// becomes this
let _result = {};
let _value;
for (let _key in object) {
_value = object[_key];
_result[key] = _value * 2;
}
const doubled = _result;
This works in a vast majority of cases, as the need for hasOwnProperty
checks are often an edge case; it only matters when using objects created via a custom constructor, iterating over static properties on functions, or other non-standard operations. hasOwnProperty
is a slowdown, but can be especially expensive in legacy browsers or non-JIT environments.
If you need to incorporate this, you can just filter prior to the operation:
const filtered = filterObject(object, (_, key) => Object.hasOwn(object, key));
const doubled = mapObject(filtered, (value) => value * 2);
find*
methodsMost of the operations follow the same naming conventions:
{method}
(incrementing array){method}Right
(decrementing array){method}Object
(object)The exception to this is the collection of find
-related methods:
find
findLast
findObject
findIndex
findLastIndex
findKey
The reason for findLast
/ findLastIndex
instead of findRight
/ findIndexRight
is because unlike all the other right-direction methods, those are part of the ES spec. Additionally, the reason for findIndex
vs findKey
is semantic, as objects have keys and arrays have indices.
Standard stuff, clone the repo and npm install
dependencies. The npm scripts available:
build
=> runs babel to transform the macro for legacy NodeJS supportclean
=> remove any files from dist
lint
=> runs ESLint against all files in the src
folderlint:fix
=> runs lint
, fixing any errors if possibleprepublishOnly
=> run lint
, typecheck
, test
, clean
, and
dist`release
=> release new versionrelease:beta
=> release new beta versiontest
=> run jest teststest:watch
=> run test
, but with persistent watchertypecheck
=> run tsc
against the codebaseFAQs
Macros that will inline loops for arrays and objects
We found that inline-loops.macro 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
Critics call the Node.js EOL CVE a misuse of the system, sparking debate over CVE standards and the growing noise in vulnerability databases.
Security News
cURL and Go security teams are publicly rejecting CVSS as flawed for assessing vulnerabilities and are calling for more accurate, context-aware approaches.
Security News
Bun 1.2 enhances its JavaScript runtime with 90% Node.js compatibility, built-in S3 and Postgres support, HTML Imports, and faster, cloud-first performance.