Skip to content

Commit

Permalink
feat: prototype network request interception: scaffold protocol
Browse files Browse the repository at this point in the history
Bug: #644
Doc: http://go/webdriver-bidi:network-request-interception-proposal
  • Loading branch information
Thiago Perrotta committed Jun 29, 2023
1 parent 0801561 commit 4f4ea3c
Show file tree
Hide file tree
Showing 5 changed files with 188 additions and 5 deletions.
1 change: 1 addition & 0 deletions src/bidiMapper/domains/network/networkRequest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -156,6 +156,7 @@ export class NetworkRequest {

#getBaseEventParams(): Network.BaseParameters {
return {
isBlocked: false, // TODO: Implement.
context: this.#requestWillBeSentEvent?.frameId ?? null,
navigation: this.#getNavigationId(),
// TODO: implement.
Expand Down
156 changes: 151 additions & 5 deletions src/protocol/protocol.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,8 @@

import type {ProtocolMapping} from 'devtools-protocol/types/protocol-mapping.js';

import type {Range} from './types.js';

interface EventResponse<MethodType, ParamsType> {
method: MethodType;
params: ParamsType;
Expand Down Expand Up @@ -106,6 +108,7 @@ export namespace Message {
NoSuchElement = 'no such element',
NoSuchFrame = 'no such frame',
NoSuchHandle = 'no such handle',
NoSuchIntercept = 'no such intercept',
NoSuchNode = 'no such node',
NoSuchScript = 'no such script',
SessionNotCreated = 'session not created',
Expand Down Expand Up @@ -186,6 +189,12 @@ export namespace Message {
}
}

export class NoSuchInterceptException extends ErrorResponse {
constructor(message: string, stacktrace?: string) {
super(ErrorCode.NoSuchIntercept, message, stacktrace);
}
}

export class NoSuchScriptException extends ErrorResponse {
constructor(message: string, stacktrace?: string) {
super(ErrorCode.NoSuchScript, message, stacktrace);
Expand Down Expand Up @@ -1097,7 +1106,14 @@ export namespace Log {
}

export namespace Network {
export type Command = Message.EmptyCommand;
export type Command =
| AddInterceptCommand
| ContinueRequestCommand
| ContinueResponseCommand
| ContinueWithAuthCommand
| FailRequestCommand
| ProvideResponseCommand
| RemoveInterceptCommand;

export type Result = Message.EmptyResult;

Expand Down Expand Up @@ -1127,16 +1143,22 @@ export namespace Network {
FetchErrorParams
>;

export type StringHeaderValue = {
value: string;
};

export type BinaryHeaderValue = {
binaryValue: Range<0, 255>[];
};

export type Header = {
name: string;
value?: string;
binaryValue?: number[];
};
} & (StringHeaderValue | BinaryHeaderValue);

export type Cookie = {
name: string;
value?: string;
binaryValue?: number[];
binaryValue?: Range<0, 255>[];
domain: string;
path: string;
expires?: number;
Expand Down Expand Up @@ -1175,12 +1197,16 @@ export namespace Network {
timings: FetchTimingInfo;
};

export type Intercept = string;

export type BaseParameters = {
context: CommonDataTypes.BrowsingContext | null;
isBlocked: boolean;
navigation: BrowsingContext.Navigation | null;
redirectCount: number;
request: RequestData;
timestamp: number;
intercepts?: Intercept[];
};

export type Initiator = {
Expand All @@ -1207,6 +1233,7 @@ export namespace Network {
headersSize: number | null;
bodySize: number | null;
content: ResponseContent;
authChallenge?: AuthChallenge;
};

export type BeforeRequestSentParams = BaseParameters & {
Expand All @@ -1225,6 +1252,125 @@ export namespace Network {
errorText: string;
};

export type AuthChallenge = {
scheme: string;
realm: string;
};

export type AuthCredentials = {
type: 'password'; // XXX: 'credentials'?
username: string;
password: string;
};

export type Body = StringBody | Base64Body;

export type StringBody = {
type: 'string';
value: string;
};

export type Base64Body = {
type: 'base64';
value: string;
};

export type InterceptPhase =
| 'beforeRequestSent'
| 'responseStarted'
| 'authRequired';

export type AddInterceptCommand = {
method: 'network.addIntercept';
params: AddInterceptParameters;
};

export type AddInterceptParameters = {
phases: InterceptPhase[];
urlPatterns?: string[];
};

export type AddInterceptResult = {
result: {
intercept: Intercept;
};
};

export type ContinueRequestCommand = {
method: 'network.continueRequest';
params: ContinueRequestParameters;
};

export type ContinueRequestParameters = {
request: Request;
body?: Body;
headers?: Header[];
method?: string;
url?: string;
};

export type ContinueResponseCommand = {
method: 'network.continueResponse';
params: ContinueResponseParameters;
};

export type ContinueResponseParameters = {
request: Request;
credentials?: AuthCredentials;
headers?: Header[];
reasonPhrase?: string;
statusCode?: number;
};

export type ContinueWithAuthCommand = {
method: 'network.continueWithAuth';
params: ContinueWithAuthParameters;
};

export type ContinueWithAuthParameters = {
request: Request;
} & (ContinueWithAuthCredentials | ContinueWithAuthNoCredentials);

export type ContinueWithAuthCredentials = {
action: 'provideCredentials';
credentials: AuthCredentials;
};

export type ContinueWithAuthNoCredentials = {
action: 'default' | 'cancel';
};

export type FailRequestCommand = {
method: 'network.failRequest';
params: FailRequestParameters;
};

export type FailRequestParameters = {
request: Request;
};

export type ProvideResponseCommand = {
method: 'network.provideResponse';
params: ProvideResponseParameters;
};

export type ProvideResponseParameters = {
request: Request;
body?: Body;
headers?: Header[];
reasonPhrase?: string;
statusCode?: number;
};

export type RemoveInterceptCommand = {
method: 'network.removeIntercept';
params: RemoveInterceptParameters;
};

export type RemoveInterceptParameters = {
intercept: Intercept;
};

export const AllEvents = 'network';

export enum EventNames {
Expand Down
29 changes: 29 additions & 0 deletions src/protocol/types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
/**
* Copyright 2023 Google LLC.
* Copyright (c) Microsoft Corporation.
*
* 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.
*/

/** @see: https://stackoverflow.com/a/70307091 */
export type Enumerate<
N extends number,
Acc extends number[] = []
> = Acc['length'] extends N
? Acc[number]
: Enumerate<N, [...Acc, Acc['length']]>;

export type Range<F extends number, T extends number> = Exclude<
Enumerate<T>,
Enumerate<F>
>;
2 changes: 2 additions & 0 deletions tests/test_browsing_context.py
Original file line number Diff line number Diff line change
Expand Up @@ -894,6 +894,7 @@ async def test_browsingContext_ignoreCache(websocket, context_id, ignoreCache):
assert response == {
"method": "network.beforeRequestSent",
"params": {
"isBlocked": False,
"context": context_id,
"initiator": ANY_DICT,
"navigation": ANY_STR,
Expand All @@ -907,6 +908,7 @@ async def test_browsingContext_ignoreCache(websocket, context_id, ignoreCache):
assert response == {
"method": "network.responseCompleted",
"params": {
"isBlocked": False,
"context": context_id,
"navigation": ANY_STR,
"redirectCount": 0,
Expand Down
5 changes: 5 additions & 0 deletions tests/test_network.py
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ async def test_network_before_request_sent_event_emitted(
assert resp == {
"method": "network.beforeRequestSent",
"params": {
"isBlocked": False,
"context": context_id,
"navigation": ANY_STR,
"redirectCount": 0,
Expand Down Expand Up @@ -158,6 +159,7 @@ async def test_network_before_request_sent_event_with_cookies_emitted(
assert resp == {
"method": "network.beforeRequestSent",
"params": {
"isBlocked": False,
"context": context_id,
"navigation": ANY_STR,
"redirectCount": 0,
Expand Down Expand Up @@ -211,6 +213,7 @@ async def test_network_network_response_completed_event_emitted(
assert resp == {
"method": "network.responseCompleted",
"params": {
"isBlocked": False,
"context": context_id,
"navigation": ANY_STR,
"redirectCount": 0,
Expand Down Expand Up @@ -265,6 +268,7 @@ async def test_network_bad_ssl(websocket, context_id):
assert resp == {
"method": "network.fetchError",
"params": {
"isBlocked": False,
"context": context_id,
"navigation": ANY_STR,
"redirectCount": 0,
Expand Down Expand Up @@ -302,6 +306,7 @@ async def test_network_before_request_sent_event_with_data_url_emitted(
assert resp == {
"method": "network.beforeRequestSent",
"params": {
"isBlocked": False,
"context": context_id,
"navigation": ANY_STR,
"redirectCount": 0,
Expand Down

0 comments on commit 4f4ea3c

Please sign in to comment.