UNPKG

@lodestar/beacon-node

Version:

A Typescript implementation of the beacon chain

88 lines 3.49 kB
import fs from "node:fs"; import path from "node:path"; import { sleep } from "@lodestar/utils"; export { ProfileThread }; var ProfileThread; (function (ProfileThread) { ProfileThread["MAIN"] = "main"; ProfileThread["NETWORK"] = "network"; ProfileThread["DISC5"] = "discv5"; })(ProfileThread || (ProfileThread = {})); /** * The time to take a Bun profile. * If we increase this time it'll potentiall cause the app to crash. * If we decrease this time, profile recorded will be fragmented and hard to analyze. */ const BUN_PROFILE_MS = 3 * 1000; export async function profileThread(thread, durationMs, dirpath) { return globalThis.Bun ? profileBun(thread, durationMs) : profileNodeJS(thread, durationMs, dirpath); } /** * Take `durationMs` profile of the current thread and return the persisted file path. */ async function profileNodeJS(thread, durationMs, dirpath) { const inspector = await import("node:inspector"); // due to some typing issues, not able to use promisify here const profile = await new Promise((resolve, reject) => { // Start the inspector and connect to it const session = new inspector.Session(); session.connect(); session.post("Profiler.enable", () => { session.post("Profiler.start", async () => { await sleep(durationMs); session.post("Profiler.stop", (err, { profile }) => { if (!err) { resolve(JSON.stringify(profile)); } else { reject(err); } // Detach from the inspector and close the session session.post("Profiler.disable"); session.disconnect(); }); }); }); }); const filePath = path.join(dirpath, `${thread}_thread_${new Date().toISOString()}.cpuprofile`); fs.writeFileSync(filePath, profile); return filePath; } /** * Unlike NodeJS, Bun console.profile() api flush data to the inspector, * so this api returns ms taken of this profile instead of file path. */ async function profileBun(thread, durationMs) { const start = Date.now(); let now = Date.now(); while (now - start < durationMs) { // biome-ignore lint/suspicious/noConsole: need to use console api to profile in Bun console.profile(String(now)); await sleep(BUN_PROFILE_MS); // biome-ignore lint/suspicious/noConsole: need to use console api to profile in Bun console.profileEnd(String(now)); now = Date.now(); } return `Successfully take Bun ${thread} thread profile in ${now - start}ms. Check your inspector to see the profile.`; } /** * Write heap snapshot of the current thread to the specified file. */ export async function writeHeapSnapshot(prefix, dirpath) { // Lazily import NodeJS only modules const fs = await import("node:fs"); const v8 = await import("node:v8"); const snapshotStream = v8.getHeapSnapshot(); const filepath = `${dirpath}/${prefix}_${new Date().toISOString()}.heapsnapshot`; const fileStream = fs.createWriteStream(filepath); return new Promise((resolve, reject) => { fileStream.on("error", (err) => { reject(err); }); snapshotStream.pipe(fileStream); snapshotStream.on("end", () => { resolve(filepath); }); }); } //# sourceMappingURL=profile.js.map