Files
gh-rafaelcalleja-claude-mar…/skills/chrome-devtools/scripts/performance.js
2025-11-30 08:48:52 +08:00

146 lines
3.8 KiB
JavaScript

#!/usr/bin/env node
/**
* Measure performance metrics and record trace
* Usage: node performance.js --url https://example.com [--trace trace.json] [--metrics]
*/
import { getBrowser, getPage, closeBrowser, parseArgs, outputJSON, outputError } from './lib/browser.js';
import fs from 'fs/promises';
async function measurePerformance() {
const args = parseArgs(process.argv.slice(2));
if (!args.url) {
outputError(new Error('--url is required'));
return;
}
try {
const browser = await getBrowser({
headless: args.headless !== 'false'
});
const page = await getPage(browser);
// Start tracing if requested
if (args.trace) {
await page.tracing.start({
path: args.trace,
categories: [
'devtools.timeline',
'disabled-by-default-devtools.timeline',
'disabled-by-default-devtools.timeline.frame'
]
});
}
// Navigate
await page.goto(args.url, {
waitUntil: 'networkidle2'
});
// Stop tracing
if (args.trace) {
await page.tracing.stop();
}
// Get performance metrics
const metrics = await page.metrics();
// Get Core Web Vitals
const vitals = await page.evaluate(() => {
return new Promise((resolve) => {
const vitals = {
LCP: null,
FID: null,
CLS: 0,
FCP: null,
TTFB: null
};
// LCP
try {
new PerformanceObserver((list) => {
const entries = list.getEntries();
if (entries.length > 0) {
const lastEntry = entries[entries.length - 1];
vitals.LCP = lastEntry.renderTime || lastEntry.loadTime;
}
}).observe({ entryTypes: ['largest-contentful-paint'], buffered: true });
} catch (e) {}
// CLS
try {
new PerformanceObserver((list) => {
list.getEntries().forEach((entry) => {
if (!entry.hadRecentInput) {
vitals.CLS += entry.value;
}
});
}).observe({ entryTypes: ['layout-shift'], buffered: true });
} catch (e) {}
// FCP
try {
const paintEntries = performance.getEntriesByType('paint');
const fcpEntry = paintEntries.find(e => e.name === 'first-contentful-paint');
if (fcpEntry) {
vitals.FCP = fcpEntry.startTime;
}
} catch (e) {}
// TTFB
try {
const [navigationEntry] = performance.getEntriesByType('navigation');
if (navigationEntry) {
vitals.TTFB = navigationEntry.responseStart - navigationEntry.requestStart;
}
} catch (e) {}
// Wait a bit for metrics to stabilize
setTimeout(() => resolve(vitals), 1000);
});
});
// Get resource timing
const resources = await page.evaluate(() => {
return performance.getEntriesByType('resource').map(r => ({
name: r.name,
type: r.initiatorType,
duration: r.duration,
size: r.transferSize,
startTime: r.startTime
}));
});
const result = {
success: true,
url: page.url(),
metrics: {
...metrics,
JSHeapUsedSizeMB: (metrics.JSHeapUsedSize / 1024 / 1024).toFixed(2),
JSHeapTotalSizeMB: (metrics.JSHeapTotalSize / 1024 / 1024).toFixed(2)
},
vitals: vitals,
resources: {
count: resources.length,
totalDuration: resources.reduce((sum, r) => sum + r.duration, 0),
items: args.resources === 'true' ? resources : undefined
}
};
if (args.trace) {
result.trace = args.trace;
}
outputJSON(result);
if (args.close !== 'false') {
await closeBrowser();
}
} catch (error) {
outputError(error);
}
}
measurePerformance();