From 2fe88478d838dfa593fe69a2c66c50d6c9c3db39 Mon Sep 17 00:00:00 2001 From: Connor Clark Date: Wed, 5 Jun 2024 13:22:22 -0700 Subject: [PATCH] core(network): align headers end time with send when no data received (#16044) --- core/lib/lantern/types/lantern.d.ts | 6 +++++- core/lib/network-request.js | 20 ++++++++++++-------- core/test/lib/network-recorder-test.js | 2 +- core/test/network-records-to-devtools-log.js | 7 ++++++- 4 files changed, 24 insertions(+), 11 deletions(-) diff --git a/core/lib/lantern/types/lantern.d.ts b/core/lib/lantern/types/lantern.d.ts index b52cc58c710b..e69afef87592 100644 --- a/core/lib/lantern/types/lantern.d.ts +++ b/core/lib/lantern/types/lantern.d.ts @@ -60,7 +60,11 @@ export class NetworkRequest { * HTTP cache or going to the network for DNS/connection setup, in milliseconds. */ networkRequestTime: number; - /** When the last byte of the response headers is received, in milliseconds. */ + /** + * When the last byte of the response headers is received, in milliseconds. + * Equal to networkRequestTime if no data is recieved over the + * network (ex: cached requests or data urls). + */ responseHeadersEndTime: number; /** When the last byte of the response body is received, in milliseconds. */ networkEndTime: number; diff --git a/core/lib/network-request.js b/core/lib/network-request.js index 77bf329a04fe..60f8f639d181 100644 --- a/core/lib/network-request.js +++ b/core/lib/network-request.js @@ -129,7 +129,11 @@ class NetworkRequest { * HTTP cache or going to the network for DNS/connection setup, in milliseconds. */ this.networkRequestTime = -1; - /** When the last byte of the response headers is received, in milliseconds. */ + /** + * When the last byte of the response headers is received, in milliseconds. + * Equal to networkRequestTime if no data is recieved over the + * network (ex: cached requests or data urls). + */ this.responseHeadersEndTime = -1; /** When the last byte of the response body is received, in milliseconds. */ this.networkEndTime = -1; @@ -222,8 +226,9 @@ class NetworkRequest { this.isSecure = UrlUtils.isSecureScheme(this.parsedURL.scheme); this.rendererStartTime = data.timestamp * 1000; - // Expected to be overridden with better value in `_recomputeTimesWithResourceTiming`. + // These are expected to be overridden with better value in `_recomputeTimesWithResourceTiming`. this.networkRequestTime = this.rendererStartTime; + this.responseHeadersEndTime = this.rendererStartTime; this.requestMethod = data.request.method; @@ -350,8 +355,7 @@ class NetworkRequest { if (response.protocol) this.protocol = response.protocol; - // This is updated in _recomputeTimesWithResourceTiming, if timings are present. - this.responseHeadersEndTime = timestamp * 1000; + this.responseTimestamp = timestamp * 1000; this.transferSize = response.encodedDataLength; this.responseHeadersTransferSize = response.encodedDataLength; @@ -381,7 +385,7 @@ class NetworkRequest { _recomputeTimesWithResourceTiming(timing) { // Don't recompute times if the data is invalid. RequestTime should always be a thread timestamp. // If we don't have receiveHeadersEnd, we really don't have more accurate data. - if (timing.requestTime === 0 || timing.receiveHeadersEnd === -1) return; + if (timing.requestTime === -1 || timing.receiveHeadersEnd === -1) return; // Take networkRequestTime and responseHeadersEndTime from timing data for better accuracy. // Before this, networkRequestTime and responseHeadersEndTime were set to bogus values based on @@ -392,15 +396,15 @@ class NetworkRequest { // See https://raw.githubusercontent.com/GoogleChrome/lighthouse/main/docs/Network-Timings.svg this.networkRequestTime = timing.requestTime * 1000; const headersReceivedTime = this.networkRequestTime + timing.receiveHeadersEnd; - // This was set in `_onResponse` as that event's timestamp. - const responseTimestamp = this.responseHeadersEndTime; // Update this.responseHeadersEndTime. All timing values from the netstack (timing) are well-ordered, and // so are the timestamps from CDP (which this.responseHeadersEndTime belongs to). It shouldn't be possible // that this timing from the netstack is greater than the onResponse timestamp, but just to ensure proper order // is maintained we bound the new timing by the network request time and the response timestamp. this.responseHeadersEndTime = headersReceivedTime; - this.responseHeadersEndTime = Math.min(this.responseHeadersEndTime, responseTimestamp); + if (this.responseTimestamp !== undefined) { + this.responseHeadersEndTime = Math.min(this.responseHeadersEndTime, this.responseTimestamp); + } this.responseHeadersEndTime = Math.max(this.responseHeadersEndTime, this.networkRequestTime); // We're only at responseReceived (_onResponse) at this point. diff --git a/core/test/lib/network-recorder-test.js b/core/test/lib/network-recorder-test.js index 3eba3bc9d464..3811cb5cf73f 100644 --- a/core/test/lib/network-recorder-test.js +++ b/core/test/lib/network-recorder-test.js @@ -587,7 +587,7 @@ describe('network recorder', function() { localizedFailDescription: 'net::ERR_ABORTED', rendererStartTime: 500, networkRequestTime: 500, - responseHeadersEndTime: -1, + responseHeadersEndTime: 500, networkEndTime: 501, transferSize: 0, resourceSize: 0, diff --git a/core/test/network-records-to-devtools-log.js b/core/test/network-records-to-devtools-log.js index 8c6ef1e1ef16..abfb7acf024b 100644 --- a/core/test/network-records-to-devtools-log.js +++ b/core/test/network-records-to-devtools-log.js @@ -96,6 +96,7 @@ function extractPartialTiming(networkRecord) { responseHeadersEndTime, networkEndTime, redirectResponseTimestamp, // Generated timestamp added in addRedirectResponseIfNeeded; only used as backup start time. + responseTimestamp, // Time of response event from CDP, not from the netstack. } = networkRecord; const requestTime = timeDefined(requestTimeS) ? requestTimeS * 1000 : undefined; @@ -130,6 +131,7 @@ function extractPartialTiming(networkRecord) { return { redirectResponseTimestamp, + responseTimestamp, rendererStartTime, networkRequestTime, requestTime, @@ -199,11 +201,14 @@ function getNormalizedRequestTiming(networkRecord) { const networkEndTime = extractedTimes.networkEndTime ?? (responseHeadersEndTime + defaultTimingOffset); + const responseTimestamp = extractedTimes.responseTimestamp ?? requestTime + receiveHeadersEnd; + return { rendererStartTime, networkRequestTime, responseHeadersEndTime, networkEndTime, + responseTimestamp, timing: { // TODO: other `timing` properties could have default values. ...networkRecord.timing, @@ -277,7 +282,7 @@ function getResponseReceivedEvent(networkRecord, index, normalizedTiming) { method: 'Network.responseReceived', params: { requestId: getBaseRequestId(networkRecord) || `${idBase}.${index}`, - timestamp: normalizedTiming.responseHeadersEndTime / 1000, + timestamp: normalizedTiming.responseTimestamp / 1000, type: networkRecord.resourceType || undefined, response: { url: networkRecord.url || exampleUrl,