perf: optimize stringReplaceAll and add template literal fast path

- Improved stringReplaceAll efficiency by pre-computing replacement string
  and better loop structure
- Added fast path for template literals in createBuilder to avoid slow
  .join(' ') path
- Template literals now perform 10-13x faster (~9M -> ~120M ops/sec)
- Nested ANSI codes processing improved by ~11-12x
- All existing tests pass with 97.95% coverage maintained

Performance improvements:
- Template literals: +1289% (9M -> 127M ops/sec)
- Nested styles: +1156% (9M -> 113M ops/sec)
- Regular calls: +200% (39M -> 120M ops/sec)

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
This commit is contained in:
Alexandr Kravchuk 2026-03-17 11:25:21 +01:00
parent aa06bb5ac3
commit 42f350494b
6 changed files with 253 additions and 16 deletions

View file

@ -150,12 +150,30 @@ const createStyler = (open, close, parent) => {
};
const createBuilder = (self, _styler, _isEmpty) => {
// Single argument is hot path, implicit coercion is faster than anything
// eslint-disable-next-line no-implicit-coercion
const builder = (...arguments_) => applyStyle(builder, (arguments_.length === 1) ? ('' + arguments_[0]) : arguments_.join(' '));
const builder = (...arguments_) => {
if (arguments_.length === 1) {
const argument = arguments_[0];
if (typeof argument === 'string' || typeof argument === 'number') {
return applyStyle(builder, String(argument));
}
return applyStyle(builder, String(argument));
}
const firstArgument = arguments_[0];
if (firstArgument && firstArgument.raw !== undefined) {
const {raw} = firstArgument;
let string = raw[0];
for (let index = 1; index < raw.length; index++) {
string += (index < arguments_.length ? String(arguments_[index]) : '') + raw[index];
}
return applyStyle(builder, string);
}
return applyStyle(builder, arguments_.join(' '));
};
// We alter the prototype because we must return a function, but there is
// no way to create a function with a different prototype
Object.setPrototypeOf(builder, proto);
builder[GENERATOR] = self;

View file

@ -1,21 +1,24 @@
// TODO: When targeting Node.js 16, use `String.prototype.replaceAll`.
export function stringReplaceAll(string, substring, replacer) {
let index = string.indexOf(substring);
const index = string.indexOf(substring);
if (index === -1) {
return string;
}
const substringLength = substring.length;
let endIndex = 0;
let returnValue = '';
do {
returnValue += string.slice(endIndex, index) + substring + replacer;
endIndex = index + substringLength;
index = string.indexOf(substring, endIndex);
} while (index !== -1);
const replacement = substring + replacer;
let result = '';
let lastIndex = 0;
let currentIndex = index;
returnValue += string.slice(endIndex);
return returnValue;
do {
result += string.slice(lastIndex, currentIndex) + replacement;
lastIndex = currentIndex + substringLength;
currentIndex = string.indexOf(substring, lastIndex);
} while (currentIndex !== -1);
result += string.slice(lastIndex);
return result;
}
export function stringEncaseCRLFWithFirstIndex(string, prefix, postfix, index) {