2022-09-19 10:49:13 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2022 Google Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2022-06-22 13:25:44 +00:00
|
|
|
import {Protocol} from 'devtools-protocol';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2023-09-13 19:57:26 +00:00
|
|
|
import {CdpHTTPRequest} from './HTTPRequest.js';
|
2021-12-07 15:08:45 +00:00
|
|
|
|
2022-06-27 07:24:23 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2023-07-17 08:52:54 +00:00
|
|
|
export interface QueuedEventGroup {
|
2021-12-09 10:02:24 +00:00
|
|
|
responseReceivedEvent: Protocol.Network.ResponseReceivedEvent;
|
|
|
|
loadingFinishedEvent?: Protocol.Network.LoadingFinishedEvent;
|
|
|
|
loadingFailedEvent?: Protocol.Network.LoadingFailedEvent;
|
2023-07-17 08:52:54 +00:00
|
|
|
}
|
2021-12-07 15:08:45 +00:00
|
|
|
|
2022-06-27 07:24:23 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2021-12-07 15:08:45 +00:00
|
|
|
export type FetchRequestId = string;
|
|
|
|
|
2022-06-27 07:24:23 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2023-07-17 08:52:54 +00:00
|
|
|
export interface RedirectInfo {
|
2021-12-07 15:08:45 +00:00
|
|
|
event: Protocol.Network.RequestWillBeSentEvent;
|
|
|
|
fetchRequestId?: FetchRequestId;
|
2023-07-17 08:52:54 +00:00
|
|
|
}
|
2022-06-27 07:24:23 +00:00
|
|
|
type RedirectInfoList = RedirectInfo[];
|
2021-12-07 15:08:45 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
2022-06-27 07:24:23 +00:00
|
|
|
*/
|
|
|
|
export type NetworkRequestId = string;
|
|
|
|
|
|
|
|
/**
|
2021-12-07 15:08:45 +00:00
|
|
|
* Helper class to track network events by request ID
|
2022-06-27 07:24:23 +00:00
|
|
|
*
|
|
|
|
* @internal
|
2021-12-07 15:08:45 +00:00
|
|
|
*/
|
|
|
|
export class NetworkEventManager {
|
2022-08-12 12:15:26 +00:00
|
|
|
/**
|
2021-12-07 15:08:45 +00:00
|
|
|
* There are four possible orders of events:
|
2022-08-12 12:15:26 +00:00
|
|
|
* A. `_onRequestWillBeSent`
|
|
|
|
* B. `_onRequestWillBeSent`, `_onRequestPaused`
|
|
|
|
* C. `_onRequestPaused`, `_onRequestWillBeSent`
|
|
|
|
* D. `_onRequestPaused`, `_onRequestWillBeSent`, `_onRequestPaused`,
|
|
|
|
* `_onRequestWillBeSent`, `_onRequestPaused`, `_onRequestPaused`
|
|
|
|
* (see crbug.com/1196004)
|
2021-12-07 15:08:45 +00:00
|
|
|
*
|
|
|
|
* For `_onRequest` we need the event from `_onRequestWillBeSent` and
|
|
|
|
* optionally the `interceptionId` from `_onRequestPaused`.
|
|
|
|
*
|
|
|
|
* If request interception is disabled, call `_onRequest` once per call to
|
|
|
|
* `_onRequestWillBeSent`.
|
|
|
|
* If request interception is enabled, call `_onRequest` once per call to
|
|
|
|
* `_onRequestPaused` (once per `interceptionId`).
|
|
|
|
*
|
|
|
|
* Events are stored to allow for subsequent events to call `_onRequest`.
|
|
|
|
*
|
|
|
|
* Note that (chains of) redirect requests have the same `requestId` (!) as
|
|
|
|
* the original request. We have to anticipate series of events like these:
|
2022-08-12 12:15:26 +00:00
|
|
|
* A. `_onRequestWillBeSent`,
|
|
|
|
* `_onRequestWillBeSent`, ...
|
|
|
|
* B. `_onRequestWillBeSent`, `_onRequestPaused`,
|
|
|
|
* `_onRequestWillBeSent`, `_onRequestPaused`, ...
|
|
|
|
* C. `_onRequestWillBeSent`, `_onRequestPaused`,
|
|
|
|
* `_onRequestPaused`, `_onRequestWillBeSent`, ...
|
|
|
|
* D. `_onRequestPaused`, `_onRequestWillBeSent`,
|
|
|
|
* `_onRequestPaused`, `_onRequestWillBeSent`, `_onRequestPaused`,
|
|
|
|
* `_onRequestWillBeSent`, `_onRequestPaused`, `_onRequestPaused`, ...
|
|
|
|
* (see crbug.com/1196004)
|
2021-12-07 15:08:45 +00:00
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
#requestWillBeSentMap = new Map<
|
2021-12-07 15:08:45 +00:00
|
|
|
NetworkRequestId,
|
|
|
|
Protocol.Network.RequestWillBeSentEvent
|
|
|
|
>();
|
2022-06-13 09:16:25 +00:00
|
|
|
#requestPausedMap = new Map<
|
2021-12-07 15:08:45 +00:00
|
|
|
NetworkRequestId,
|
|
|
|
Protocol.Fetch.RequestPausedEvent
|
|
|
|
>();
|
2023-09-13 19:57:26 +00:00
|
|
|
#httpRequestsMap = new Map<NetworkRequestId, CdpHTTPRequest>();
|
2021-12-07 15:08:45 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The below maps are used to reconcile Network.responseReceivedExtraInfo
|
|
|
|
* events with their corresponding request. Each response and redirect
|
|
|
|
* response gets an ExtraInfo event, and we don't know which will come first.
|
|
|
|
* This means that we have to store a Response or an ExtraInfo for each
|
|
|
|
* response, and emit the event when we get both of them. In addition, to
|
|
|
|
* handle redirects, we have to make them Arrays to represent the chain of
|
|
|
|
* events.
|
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
#responseReceivedExtraInfoMap = new Map<
|
2021-12-07 15:08:45 +00:00
|
|
|
NetworkRequestId,
|
|
|
|
Protocol.Network.ResponseReceivedExtraInfoEvent[]
|
|
|
|
>();
|
2022-06-13 09:16:25 +00:00
|
|
|
#queuedRedirectInfoMap = new Map<NetworkRequestId, RedirectInfoList>();
|
|
|
|
#queuedEventGroupMap = new Map<NetworkRequestId, QueuedEventGroup>();
|
2021-12-07 15:08:45 +00:00
|
|
|
|
|
|
|
forget(networkRequestId: NetworkRequestId): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#requestWillBeSentMap.delete(networkRequestId);
|
|
|
|
this.#requestPausedMap.delete(networkRequestId);
|
|
|
|
this.#queuedEventGroupMap.delete(networkRequestId);
|
|
|
|
this.#queuedRedirectInfoMap.delete(networkRequestId);
|
|
|
|
this.#responseReceivedExtraInfoMap.delete(networkRequestId);
|
2021-12-07 15:08:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
responseExtraInfo(
|
|
|
|
networkRequestId: NetworkRequestId
|
|
|
|
): Protocol.Network.ResponseReceivedExtraInfoEvent[] {
|
2022-06-13 09:16:25 +00:00
|
|
|
if (!this.#responseReceivedExtraInfoMap.has(networkRequestId)) {
|
|
|
|
this.#responseReceivedExtraInfoMap.set(networkRequestId, []);
|
2021-12-07 15:08:45 +00:00
|
|
|
}
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#responseReceivedExtraInfoMap.get(
|
2022-05-23 15:41:11 +00:00
|
|
|
networkRequestId
|
|
|
|
) as Protocol.Network.ResponseReceivedExtraInfoEvent[];
|
2021-12-07 15:08:45 +00:00
|
|
|
}
|
|
|
|
|
2021-12-09 10:02:24 +00:00
|
|
|
private queuedRedirectInfo(fetchRequestId: FetchRequestId): RedirectInfoList {
|
2022-06-13 09:16:25 +00:00
|
|
|
if (!this.#queuedRedirectInfoMap.has(fetchRequestId)) {
|
|
|
|
this.#queuedRedirectInfoMap.set(fetchRequestId, []);
|
2021-12-07 15:08:45 +00:00
|
|
|
}
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#queuedRedirectInfoMap.get(fetchRequestId) as RedirectInfoList;
|
2021-12-07 15:08:45 +00:00
|
|
|
}
|
2021-12-09 10:02:24 +00:00
|
|
|
|
|
|
|
queueRedirectInfo(
|
|
|
|
fetchRequestId: FetchRequestId,
|
|
|
|
redirectInfo: RedirectInfo
|
|
|
|
): void {
|
|
|
|
this.queuedRedirectInfo(fetchRequestId).push(redirectInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
takeQueuedRedirectInfo(
|
|
|
|
fetchRequestId: FetchRequestId
|
|
|
|
): RedirectInfo | undefined {
|
|
|
|
return this.queuedRedirectInfo(fetchRequestId).shift();
|
|
|
|
}
|
|
|
|
|
2023-05-30 11:07:55 +00:00
|
|
|
inFlightRequestsCount(): number {
|
2023-05-31 07:32:16 +00:00
|
|
|
let inFlightRequestCounter = 0;
|
|
|
|
for (const request of this.#httpRequestsMap.values()) {
|
2023-05-30 11:07:55 +00:00
|
|
|
if (!request.response()) {
|
2023-05-31 07:32:16 +00:00
|
|
|
inFlightRequestCounter++;
|
2023-05-30 11:07:55 +00:00
|
|
|
}
|
|
|
|
}
|
2023-05-31 07:32:16 +00:00
|
|
|
return inFlightRequestCounter;
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
storeRequestWillBeSent(
|
|
|
|
networkRequestId: NetworkRequestId,
|
|
|
|
event: Protocol.Network.RequestWillBeSentEvent
|
|
|
|
): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#requestWillBeSentMap.set(networkRequestId, event);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getRequestWillBeSent(
|
|
|
|
networkRequestId: NetworkRequestId
|
|
|
|
): Protocol.Network.RequestWillBeSentEvent | undefined {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#requestWillBeSentMap.get(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
forgetRequestWillBeSent(networkRequestId: NetworkRequestId): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#requestWillBeSentMap.delete(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getRequestPaused(
|
|
|
|
networkRequestId: NetworkRequestId
|
|
|
|
): Protocol.Fetch.RequestPausedEvent | undefined {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#requestPausedMap.get(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
forgetRequestPaused(networkRequestId: NetworkRequestId): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#requestPausedMap.delete(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
storeRequestPaused(
|
|
|
|
networkRequestId: NetworkRequestId,
|
|
|
|
event: Protocol.Fetch.RequestPausedEvent
|
|
|
|
): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#requestPausedMap.set(networkRequestId, event);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
2023-09-13 19:57:26 +00:00
|
|
|
getRequest(networkRequestId: NetworkRequestId): CdpHTTPRequest | undefined {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#httpRequestsMap.get(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
2023-09-13 13:47:55 +00:00
|
|
|
storeRequest(
|
|
|
|
networkRequestId: NetworkRequestId,
|
2023-09-13 19:57:26 +00:00
|
|
|
request: CdpHTTPRequest
|
2023-09-13 13:47:55 +00:00
|
|
|
): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#httpRequestsMap.set(networkRequestId, request);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
forgetRequest(networkRequestId: NetworkRequestId): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#httpRequestsMap.delete(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getQueuedEventGroup(
|
|
|
|
networkRequestId: NetworkRequestId
|
|
|
|
): QueuedEventGroup | undefined {
|
2022-06-13 09:16:25 +00:00
|
|
|
return this.#queuedEventGroupMap.get(networkRequestId);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
queueEventGroup(
|
|
|
|
networkRequestId: NetworkRequestId,
|
|
|
|
event: QueuedEventGroup
|
|
|
|
): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#queuedEventGroupMap.set(networkRequestId, event);
|
2021-12-09 10:02:24 +00:00
|
|
|
}
|
2022-04-19 09:28:44 +00:00
|
|
|
|
|
|
|
forgetQueuedEventGroup(networkRequestId: NetworkRequestId): void {
|
2022-06-13 09:16:25 +00:00
|
|
|
this.#queuedEventGroupMap.delete(networkRequestId);
|
2022-04-19 09:28:44 +00:00
|
|
|
}
|
2021-12-07 15:08:45 +00:00
|
|
|
}
|