Description
- Version: 12.16.1
- Platform: Linux; OSX
- Subsystem: V8
Once we upgraded a service to node 12, we noticed a huge spike in CPU utilization. A client library was doing many inserts/deletes on an object over time.
Big thanks to @mmarchini who root caused and wrote the reproducible code.
What steps will reproduce the bug?
Inserting and deleting many properties on an Object ({}
) will cause the issue.
The follow script reproduces the issue:
'use strict';
const foo = {};
let ms = 0;
const C = 8380000;
for (let i=0; i < C; i++) {
const key = `foo-${i}`;
const start = new Date();
foo[key] = {};
ms += new Date() - start;
if (i % 175)
delete foo[key];
}
console.log(`Inserted ${C} in ${ms}ms, average ${ms/C} per insert`);
const STEP = 5000;
for (let j=0; j < 3; j++) {
ms = 0;
for (let i=0; i < STEP; i++) {
const key = `${j}-${i}`;
const start = new Date();
foo[key] = {};
ms += new Date() - start;
delete foo[key];
}
console.log(`Inserted ${STEP} in ${ms}ms, average ${ms/STEP}ms per insert`);
}
On node 12.x it results something like:
Inserted 8380000 in 8684ms, average 0.0010362768496420048 per insert
Inserted 5000 in 4ms, average 0.0008ms per insert
Inserted 5000 in 7483ms, average 1.4966ms per insert
Inserted 5000 in 25055ms, average 5.011ms per insert
While on Node 10.x it results something like:
Inserted 8380000 in 7436ms, average 0.0008873508353221957 per insert
Inserted 5000 in 1ms, average 0.0002ms per insert
Inserted 5000 in 14ms, average 0.0028ms per insert
Inserted 5000 in 6ms, average 0.0012ms per insert
(results from Darwin nfml-ghermetoX5J 18.7.0 Darwin Kernel Version 18.7.0: Thu Jun 20 18:42:21 PDT 2019; root:xnu-4903.270.47~4/RELEASE_X86_64 x86_64
)
Additional information
After a lengthy investigation we could verify that V8 version 7.8.279.23
which ships with Node.js version 12.16.x has a regression that was introduced on V8 7.x. The issue is fixed on V8 8.x and was caused by a bitfield overflow after many deletes/inserts in an object.
The issue doesn't manifest when using a Map instead of a plain Object.
Activity