phantomas
Version:
Headless Chromium-based web performance metrics collector and monitoring tool
389 lines (325 loc) • 11.3 kB
JavaScript
/**
* Simple HTTP requests monitor and analyzer
*/
;
const assert = require("assert"),
debug = require("debug")("phantomas:modules:requestsMonitor");
/**
* Given key-value set of HTTP headers returns the set with lowercased header names
*
* @param {object} headers
* @returns {object}
*/
function lowerCaseHeaders(headers) {
var res = {};
Object.keys(headers).forEach((headerName) => {
res[headerName.toLowerCase()] = headers[headerName];
});
return res;
}
// parse given URL to get protocol and domain
function parseEntryUrl(entry) {
var parsed;
// asset type
entry.type = "other";
if (entry.url.indexOf("data:") === 0) {
// base64 encoded data
entry.domain = false;
entry.protocol = false;
entry.isBase64 = true;
} else if (entry.url.indexOf("blob:") === 0) {
// blob image or video
entry.domain = false;
entry.protocol = false;
entry.isBlob = true;
} else {
parsed = new URL(entry.url);
entry.protocol = parsed.protocol.replace(":", ""); // e.g. "http:"
entry.domain = parsed.hostname;
entry.query = parsed.search.substring(1);
if (entry.protocol === "https") {
entry.isSSL = true;
}
}
return entry;
}
/**
* Detect response content type using "Content-Type header value"
*
* @param {string} headerValue
* @param {object} entry
*/
function addContentType(headerValue, entry) {
var value = headerValue.split(";").shift().toLowerCase();
entry.contentType = value;
switch (value) {
case "text/html":
entry.type = "html";
entry.isHTML = true;
break;
case "text/xml":
entry.type = "xml";
entry.isXML = true;
break;
case "text/css":
entry.type = "css";
entry.isCSS = true;
break;
case "application/x-javascript":
case "application/javascript":
case "text/javascript":
entry.type = "js";
entry.isJS = true;
break;
case "application/json":
entry.type = "json";
entry.isJSON = true;
break;
case "image/png":
case "image/jpeg":
case "image/gif":
case "image/svg+xml":
case "image/webp":
case "image/avif":
entry.type = "image";
entry.isImage = true;
if (value === "image/svg+xml") {
entry.isSVG = true;
}
break;
case "video/webm":
case "video/mp4":
entry.type = "video";
entry.isVideo = true;
break;
// @see http://stackoverflow.com/questions/2871655/proper-mime-type-for-fonts
case "application/font-wof":
case "application/font-woff":
case "application/font-woff2":
case "application/vnd.ms-fontobject":
case "application/x-font-opentype":
case "application/x-font-truetype":
case "application/x-font-ttf":
case "application/x-font-woff":
case "font/opentype":
case "font/ttf":
case "font/woff":
case "font/woff2":
entry.type = "webfont";
entry.isWebFont = true;
if (/ttf|truetype$/.test(value)) {
entry.isTTF = true;
}
break;
case "application/octet-stream":
var ext = (entry.url || "").split(".").pop();
switch (ext) {
// @see http://stackoverflow.com/questions/2871655/proper-mime-type-for-fonts#comment-8077637
case "otf":
entry.type = "webfont";
entry.isWebFont = true;
break;
}
break;
case "image/x-icon":
case "image/vnd.microsoft.icon":
entry.type = "favicon";
entry.isFavicon = true;
break;
default:
debug(
"Unknown content type found: " + value + " for <" + entry.url + ">"
);
}
return entry;
}
module.exports = function (phantomas) {
// imports
var HTTP_STATUS_CODES = require("http").STATUS_CODES;
// register metric
phantomas.setMetric("requests"); // @desc total number of HTTP requests made
phantomas.setMetric("gzipRequests"); // @desc number of gzipped HTTP responses @unreliable
phantomas.setMetric("postRequests"); // @desc number of POST requests
phantomas.setMetric("httpsRequests"); // @desc number of HTTPS requests
phantomas.setMetric("notFound"); // @desc number of HTTP 404 responses
phantomas.setMetric("bodySize"); // @desc size of the uncompressed content of all responses
phantomas.setMetric("contentLength"); // @desc size of the compressed content of all responses, i.e. what was transfered in packets
phantomas.setMetric("httpTrafficCompleted"); // @desc time it took to receive the last byte of the last HTTP response
var requests = {};
phantomas.on(
"request",
/** @param {Request} request **/ (request) => {
const resId = request._requestId;
requests[resId] = request;
// request data
// https://github.com/GoogleChrome/puppeteer/blob/v1.11.0/docs/api.md#class-request
phantomas.emit("send", request); // @desc request has been sent
}
);
phantomas.on(
"response",
/** @param {Response} resp **/ (resp) => {
const resId = resp._requestId,
request = requests[resId];
if (resp.fromDiskCache === true) {
phantomas.log("response from disk cache ignored: %j", resp);
return;
}
var entry = {
id: resId,
url: resp.url,
method: request.method,
headers: lowerCaseHeaders(resp.headers), // All header names are lower-case
bodySize: resp.dataLength,
transferedSize: resp.encodedDataLength,
};
// this is set to zero when requests interception is enabled
// use Content-Length response header instead
if (entry.transferedSize == 0) {
entry.transferedSize = parseInt(
entry.headers["content-length"] || "0",
10
);
}
// that's the response size as reported by Chrome's dev tools (headers + compressed body)
// note: base64-encoded resources do not have "resp.headersText" set
entry.responseSize = entry.transferedSize;
phantomas.log("headers: %j", resp);
entry = parseEntryUrl(entry);
/**
* Time to First Byte is the amount of time it takes for the browser
* to receive the first byte of data from the server
* after the browser makes the request.
*
* https://developers.google.com/web/tools/chrome-devtools/network-performance/reference#timing-explanation
* https://www.w3.org/TR/navigation-timing/#performancetiming
* https://chromedevtools.github.io/devtools-protocol/tot/Network#type-ResourceTiming
*
* "Throughout this work, time is measured in milliseconds"
*/
if (!entry.isBase64 && !entry.isBlob) {
// resp.timing is empty when handling %3D%3D
assert(
typeof resp.timing !== "undefined",
"resp.timing is empty when handling " + resp.url
);
// how long a given request stalled waiting for DNS, proxy, connection, SSL negotation, etc.
entry.stalled = resp.timing.sendStart;
// how it took to receive a first byte of the response after making a request
entry.timeToFirstByte =
resp.timing.receiveHeadersEnd - resp.timing.sendEnd;
// difference between when the request was sent and when it was received
entry.timeToLastByte = resp._timestamp - request._timestamp;
}
// POST requests
if (entry.method === "POST") {
phantomas.incrMetric("postRequests");
phantomas.addOffender("postRequests", entry.url);
}
entry.headersSize = 0;
// response content type
// https://chromedevtools.github.io/devtools-protocol/tot/Network#type-ResourceType
Object.keys(entry.headers).forEach((headerName) => {
const headerValue = entry.headers[headerName];
entry.headersSize +=
headerName.length +
headerValue.length +
2 /* ": " */ +
2; /* line break - CR+LF */
switch (headerName) {
// detect content type
case "content-type":
entry = addContentType(headerValue, entry);
break;
// content compression
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Encoding#Directives
case "content-encoding":
if (headerValue === "gzip" || headerValue === "br") {
entry.gzip = true;
phantomas.log(
"Response compressed with %s, %f kB -> %f kB (x%f)",
headerValue,
entry.bodySize / 1024,
entry.transferedSize / 1024,
entry.bodySize / entry.transferedSize
);
}
// A format using the Brotli algorithm.
if (headerValue === "br") {
entry.brotli = true;
}
break;
// detect cookies (issue #92)
case "set-cookie":
entry.hasCookies = true;
break;
}
});
// HTTP code
entry.status = resp.status || 200; // for base64 data
entry.statusText = HTTP_STATUS_CODES[entry.status];
switch (entry.status) {
case 301: // Moved Permanently
case 302: // Found
case 303: // See Other
entry.isRedirect = true;
break;
case 404: // Not Found
phantomas.incrMetric("notFound");
phantomas.addOffender("notFound", entry.url);
break;
}
// HTTP and TLS protocols version
entry.httpVersion = resp.protocol;
if (resp.securityDetails) {
entry.tlsVersion = resp.securityDetails.protocol;
}
// requests stats
if (!entry.isBase64 && !entry.isBlob) {
phantomas.incrMetric("requests");
phantomas.addOffender("requests", {
url: entry.url,
type: entry.type,
size: entry.responseSize,
});
phantomas.incrMetric("bodySize", entry.bodySize);
phantomas.incrMetric("contentLength", entry.transferedSize);
}
if (entry.gzip) {
phantomas.incrMetric("gzipRequests");
phantomas.addOffender("gzipRequests", {
url: entry.url,
transferedSize: entry.transferedSize,
bodySize: entry.bodySize,
});
}
if (entry.isSSL) {
phantomas.incrMetric("httpsRequests");
phantomas.addOffender("httpsRequests", entry.url);
}
if (entry.isBase64) {
phantomas.emit("base64recv", entry, resp); // @desc base64-encoded "response" has been received
} else if (entry.isBlob) {
// Do nothing
} else {
phantomas.log(
"recv: HTTP %d <%s> [%s]",
entry.status,
entry.url,
entry.contentType
);
phantomas.emit("recv", entry, resp); // @desc response has been received
}
phantomas.log("Response metadata: %j", entry);
}
);
// completion of the last HTTP request
var loadStartedTime;
phantomas.on("loadStarted", () => (loadStartedTime = Date.now())); // when the monitoring started?
phantomas.on("recv", (entry) =>
phantomas.setMetric(
"httpTrafficCompleted",
entry.recvEndTime - loadStartedTime
)
);
};