blob: 4b02c8155e7d7b6daccffb693f7a25ca020d117a [file] [log] [blame]
Wolfgang Beyer40530b682022-05-17 13:02:011// Copyright 2022 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5import * as Common from '../../../core/common/common.js';
Wolfgang Beyer7f495ee2022-07-08 13:45:136import * as Host from '../../../core/host/host.js';
Wolfgang Beyer40530b682022-05-17 13:02:017import * as i18n from '../../../core/i18n/i18n.js';
Wolfgang Beyer7f495ee2022-07-08 13:45:138import * as Platform from '../../../core/platform/platform.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:259import * as Root from '../../../core/root/root.js';
Wolfgang Beyer40530b682022-05-17 13:02:0110import * as SDK from '../../../core/sdk/sdk.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:2511import * as Persistence from '../../../models/persistence/persistence.js';
12import * as Workspace from '../../../models/workspace/workspace.js';
Wolfgang Beyer40426d82022-08-03 11:08:0713import * as NetworkForward from '../../../panels/network/forward/forward.js';
Wolfgang Beyerd1dd7962022-05-24 15:48:4814import * as Buttons from '../../../ui/components/buttons/buttons.js';
Wolfgang Beyer40530b682022-05-17 13:02:0115import * as ComponentHelpers from '../../../ui/components/helpers/helpers.js';
Wolfgang Beyer7f495ee2022-07-08 13:45:1316import * as IconButton from '../../../ui/components/icon_button/icon_button.js';
Jack Franklin1f19d522022-07-06 14:00:1817import * as Input from '../../../ui/components/input/input.js';
Wolfgang Beyer40530b682022-05-17 13:02:0118import * as UI from '../../../ui/legacy/legacy.js';
19import * as LitHtml from '../../../ui/lit-html/lit-html.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:2520import * as Sources from '../../sources/sources.js';
Wolfgang Beyerd2286472022-09-02 18:32:3021
Wolfgang Beyer65a82282022-09-05 08:22:3722import {type RequestHeaderSectionData, RequestHeaderSection} from './RequestHeaderSection.js';
Wolfgang Beyerb9f7d152022-10-10 16:10:1423import {
24 type ResponseHeaderSectionData,
25 ResponseHeaderSection,
26 RESPONSE_HEADER_SECTION_DATA_KEY,
27} from './ResponseHeaderSection.js';
Wolfgang Beyer40530b682022-05-17 13:02:0128
29import requestHeadersViewStyles from './RequestHeadersView.css.js';
30
Wolfgang Beyerd1dd7962022-05-24 15:48:4831const RAW_HEADER_CUTOFF = 3000;
Wolfgang Beyer40530b682022-05-17 13:02:0132const {render, html} = LitHtml;
33
34const UIStrings = {
35 /**
Jack Franklinfd72c072022-12-21 11:45:0136 *@description Text in Request Headers View of the Network panel
37 */
Wolfgang Beyer37fcd282022-06-29 09:34:1738 fromDiskCache: '(from disk cache)',
39 /**
Jack Franklinfd72c072022-12-21 11:45:0140 *@description Text in Request Headers View of the Network panel
41 */
Wolfgang Beyer40530b682022-05-17 13:02:0142 fromMemoryCache: '(from memory cache)',
43 /**
Jack Franklinfd72c072022-12-21 11:45:0144 *@description Text in Request Headers View of the Network panel
45 */
Wolfgang Beyer37fcd282022-06-29 09:34:1746 fromPrefetchCache: '(from prefetch cache)',
47 /**
Jack Franklinfd72c072022-12-21 11:45:0148 *@description Text in Request Headers View of the Network panel
49 */
Wolfgang Beyer40530b682022-05-17 13:02:0150 fromServiceWorker: '(from `service worker`)',
51 /**
Jack Franklinfd72c072022-12-21 11:45:0152 *@description Text in Request Headers View of the Network panel
53 */
Wolfgang Beyer40530b682022-05-17 13:02:0154 fromSignedexchange: '(from signed-exchange)',
55 /**
Jack Franklinfd72c072022-12-21 11:45:0156 *@description Text in Request Headers View of the Network panel
57 */
Wolfgang Beyer40530b682022-05-17 13:02:0158 fromWebBundle: '(from Web Bundle)',
59 /**
Jack Franklinfd72c072022-12-21 11:45:0160 *@description Section header for a list of the main aspects of a http request
61 */
Wolfgang Beyer40530b682022-05-17 13:02:0162 general: 'General',
63 /**
Jack Franklinfd72c072022-12-21 11:45:0164 *@description Label for a link from the network panel's headers view to the file in which
65 * header overrides are defined in the sources panel.
66 */
Wolfgang Beyer185fb1f2022-08-01 13:48:2567 headerOverrides: 'Header overrides',
68 /**
Jack Franklinfd72c072022-12-21 11:45:0169 *@description Label for a checkbox to switch between raw and parsed headers
70 */
Wolfgang Beyer235544c2022-05-24 08:07:4571 raw: 'Raw',
72 /**
Jack Franklinfd72c072022-12-21 11:45:0173 *@description Text in Request Headers View of the Network panel
74 */
Wolfgang Beyer37fcd282022-06-29 09:34:1775 referrerPolicy: 'Referrer Policy',
Wolfgang Beyer235544c2022-05-24 08:07:4576 /**
Jack Franklinfd72c072022-12-21 11:45:0177 *@description Text in Network Log View Columns of the Network panel
78 */
Wolfgang Beyer37fcd282022-06-29 09:34:1779 remoteAddress: 'Remote Address',
80 /**
Jack Franklinfd72c072022-12-21 11:45:0181 *@description Text in Request Headers View of the Network panel
82 */
Wolfgang Beyer37fcd282022-06-29 09:34:1783 requestHeaders: 'Request Headers',
Wolfgang Beyer40530b682022-05-17 13:02:0184 /**
Jack Franklinfd72c072022-12-21 11:45:0185 *@description The HTTP method of a request
86 */
Wolfgang Beyer40530b682022-05-17 13:02:0187 requestMethod: 'Request Method',
88 /**
Jack Franklinfd72c072022-12-21 11:45:0189 *@description The URL of a request
90 */
Wolfgang Beyer37fcd282022-06-29 09:34:1791 requestUrl: 'Request URL',
92 /**
Jack Franklinfd72c072022-12-21 11:45:0193 *@description A context menu item in the Network Log View Columns of the Network panel
94 */
Wolfgang Beyer235544c2022-05-24 08:07:4595 responseHeaders: 'Response Headers',
96 /**
Jack Franklinfd72c072022-12-21 11:45:0197 *@description Title text for a link to the Sources panel to the file containing the header override definitions
98 */
Wolfgang Beyerc8f09372022-09-19 14:27:2799 revealHeaderOverrides: 'Reveal header override definitions',
100 /**
Jack Franklinfd72c072022-12-21 11:45:01101 *@description Text to show more content
102 */
Wolfgang Beyerd1dd7962022-05-24 15:48:48103 showMore: 'Show more',
104 /**
Jack Franklinfd72c072022-12-21 11:45:01105 *@description HTTP response code
106 */
Wolfgang Beyer40530b682022-05-17 13:02:01107 statusCode: 'Status Code',
Wolfgang Beyer40530b682022-05-17 13:02:01108};
109const str_ = i18n.i18n.registerUIStrings('panels/network/components/RequestHeadersView.ts', UIStrings);
110const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
111
112export class RequestHeadersView extends UI.Widget.VBox {
Wolfgang Beyerd2286472022-09-02 18:32:30113 readonly #requestHeadersComponent = new RequestHeadersComponent();
Wolfgang Beyer40530b682022-05-17 13:02:01114 readonly #request: SDK.NetworkRequest.NetworkRequest;
115
116 constructor(request: SDK.NetworkRequest.NetworkRequest) {
117 super();
118 this.#request = request;
Wolfgang Beyerd2286472022-09-02 18:32:30119 this.contentElement.appendChild(this.#requestHeadersComponent);
Wolfgang Beyer40530b682022-05-17 13:02:01120 }
121
Randolf Jungffd14242023-04-19 00:32:25122 override wasShown(): void {
Wolfgang Beyer40530b682022-05-17 13:02:01123 this.#request.addEventListener(SDK.NetworkRequest.Events.RemoteAddressChanged, this.#refreshHeadersView, this);
124 this.#request.addEventListener(SDK.NetworkRequest.Events.FinishedLoading, this.#refreshHeadersView, this);
Wolfgang Beyer87150062022-07-29 10:45:47125 this.#request.addEventListener(SDK.NetworkRequest.Events.RequestHeadersChanged, this.#refreshHeadersView, this);
Wolfgang Beyerb9f7d152022-10-10 16:10:14126 this.#request.addEventListener(
127 SDK.NetworkRequest.Events.ResponseHeadersChanged, this.#resetAndRefreshHeadersView, this);
Wolfgang Beyer40530b682022-05-17 13:02:01128 this.#refreshHeadersView();
129 }
130
Randolf Jungffd14242023-04-19 00:32:25131 override willHide(): void {
Wolfgang Beyer40530b682022-05-17 13:02:01132 this.#request.removeEventListener(SDK.NetworkRequest.Events.RemoteAddressChanged, this.#refreshHeadersView, this);
133 this.#request.removeEventListener(SDK.NetworkRequest.Events.FinishedLoading, this.#refreshHeadersView, this);
Wolfgang Beyer87150062022-07-29 10:45:47134 this.#request.removeEventListener(SDK.NetworkRequest.Events.RequestHeadersChanged, this.#refreshHeadersView, this);
Wolfgang Beyerb9f7d152022-10-10 16:10:14135 this.#request.removeEventListener(
136 SDK.NetworkRequest.Events.ResponseHeadersChanged, this.#resetAndRefreshHeadersView, this);
137 }
138
139 #resetAndRefreshHeadersView(): void {
140 this.#request.deleteAssociatedData(RESPONSE_HEADER_SECTION_DATA_KEY);
141 this.#requestHeadersComponent.data = {
142 request: this.#request,
143 };
Wolfgang Beyer40530b682022-05-17 13:02:01144 }
145
146 #refreshHeadersView(): void {
Wolfgang Beyerd2286472022-09-02 18:32:30147 this.#requestHeadersComponent.data = {
Wolfgang Beyer40530b682022-05-17 13:02:01148 request: this.#request,
149 };
150 }
Wolfgang Beyer40426d82022-08-03 11:08:07151
152 revealHeader(section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string): void {
Wolfgang Beyerd2286472022-09-02 18:32:30153 this.#requestHeadersComponent.data = {
Wolfgang Beyer40426d82022-08-03 11:08:07154 request: this.#request,
155 toReveal: {section, header: header},
156 };
157 }
Wolfgang Beyer40530b682022-05-17 13:02:01158}
159
160export interface RequestHeadersComponentData {
161 request: SDK.NetworkRequest.NetworkRequest;
Wolfgang Beyer40426d82022-08-03 11:08:07162 toReveal?: {section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string};
Wolfgang Beyer40530b682022-05-17 13:02:01163}
164
165export class RequestHeadersComponent extends HTMLElement {
166 static readonly litTagName = LitHtml.literal`devtools-request-headers`;
167 readonly #shadow = this.attachShadow({mode: 'open'});
168 #request?: Readonly<SDK.NetworkRequest.NetworkRequest>;
Wolfgang Beyer235544c2022-05-24 08:07:45169 #showResponseHeadersText = false;
170 #showRequestHeadersText = false;
Wolfgang Beyerd1dd7962022-05-24 15:48:48171 #showResponseHeadersTextFull = false;
172 #showRequestHeadersTextFull = false;
Wolfgang Beyer40426d82022-08-03 11:08:07173 #toReveal?: {section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string} = undefined;
Wolfgang Beyer185fb1f2022-08-01 13:48:25174 readonly #workspace = Workspace.Workspace.WorkspaceImpl.instance();
Wolfgang Beyer40530b682022-05-17 13:02:01175
176 set data(data: RequestHeadersComponentData) {
177 this.#request = data.request;
Wolfgang Beyer40426d82022-08-03 11:08:07178 this.#toReveal = data.toReveal;
Wolfgang Beyer40530b682022-05-17 13:02:01179 this.#render();
180 }
181
182 connectedCallback(): void {
183 this.#shadow.adoptedStyleSheets = [requestHeadersViewStyles];
Wolfgang Beyer185fb1f2022-08-01 13:48:25184 this.#workspace.addEventListener(
185 Workspace.Workspace.Events.UISourceCodeAdded, this.#uiSourceCodeAddedOrRemoved, this);
186 this.#workspace.addEventListener(
187 Workspace.Workspace.Events.UISourceCodeRemoved, this.#uiSourceCodeAddedOrRemoved, this);
Wolfgang Beyer1e62b692022-11-07 13:52:08188 Common.Settings.Settings.instance()
189 .moduleSetting('persistenceNetworkOverridesEnabled')
190 .addChangeListener(this.#render, this);
Wolfgang Beyer185fb1f2022-08-01 13:48:25191 }
192
193 disconnectedCallback(): void {
194 this.#workspace.removeEventListener(
195 Workspace.Workspace.Events.UISourceCodeAdded, this.#uiSourceCodeAddedOrRemoved, this);
196 this.#workspace.removeEventListener(
197 Workspace.Workspace.Events.UISourceCodeRemoved, this.#uiSourceCodeAddedOrRemoved, this);
Wolfgang Beyer1e62b692022-11-07 13:52:08198 Common.Settings.Settings.instance()
199 .moduleSetting('persistenceNetworkOverridesEnabled')
200 .removeChangeListener(this.#render, this);
Wolfgang Beyer185fb1f2022-08-01 13:48:25201 }
202
203 #uiSourceCodeAddedOrRemoved(event: Common.EventTarget.EventTargetEvent<Workspace.UISourceCode.UISourceCode>): void {
204 if (this.#getHeaderOverridesFileUrl() === event.data.url()) {
205 this.#render();
206 }
Wolfgang Beyer40530b682022-05-17 13:02:01207 }
208
209 #render(): void {
Wolfgang Beyer3e62bd72022-07-28 14:26:48210 if (!this.#request) {
211 return;
212 }
Wolfgang Beyer40530b682022-05-17 13:02:01213
214 // Disabled until https://crbug.com/1079231 is fixed.
215 // clang-format off
216 render(html`
217 ${this.#renderGeneralSection()}
Wolfgang Beyer235544c2022-05-24 08:07:45218 ${this.#renderResponseHeaders()}
219 ${this.#renderRequestHeaders()}
Wolfgang Beyer40530b682022-05-17 13:02:01220 `, this.#shadow, {host: this});
221 // clang-format on
222 }
223
Wolfgang Beyer3e62bd72022-07-28 14:26:48224 #renderResponseHeaders(): LitHtml.LitTemplate {
225 if (!this.#request) {
226 return LitHtml.nothing;
227 }
Wolfgang Beyer235544c2022-05-24 08:07:45228
229 const toggleShowRaw = (): void => {
230 this.#showResponseHeadersText = !this.#showResponseHeadersText;
231 this.#render();
232 };
233
234 // Disabled until https://crbug.com/1079231 is fixed.
235 // clang-format off
236 return html`
237 <${Category.litTagName}
238 @togglerawevent=${toggleShowRaw}
239 .data=${{
240 name: 'responseHeaders',
241 title: i18nString(UIStrings.responseHeaders),
242 headerCount: this.#request.sortedResponseHeaders.length,
243 checked: this.#request.responseHeadersText ? this.#showResponseHeadersText : undefined,
Wolfgang Beyer185fb1f2022-08-01 13:48:25244 additionalContent: this.#renderHeaderOverridesLink(),
Wolfgang Beyer40426d82022-08-03 11:08:07245 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.Response,
Wolfgang Beyer235544c2022-05-24 08:07:45246 } as CategoryData}
247 aria-label=${i18nString(UIStrings.responseHeaders)}
248 >
Wolfgang Beyerd1dd7962022-05-24 15:48:48249 ${this.#showResponseHeadersText ?
250 this.#renderRawHeaders(this.#request.responseHeadersText, true) : html`
Wolfgang Beyerd2286472022-09-02 18:32:30251 <${ResponseHeaderSection.litTagName} .data=${{
252 request: this.#request,
253 toReveal: this.#toReveal,
254 } as ResponseHeaderSectionData}></${ResponseHeaderSection.litTagName}>
Wolfgang Beyer235544c2022-05-24 08:07:45255 `}
256 </${Category.litTagName}>
257 `;
Wolfgang Beyerbcf4a832022-07-25 12:34:47258 // clang-format on
Wolfgang Beyer235544c2022-05-24 08:07:45259 }
260
Wolfgang Beyer185fb1f2022-08-01 13:48:25261 #renderHeaderOverridesLink(): LitHtml.LitTemplate {
262 const overrideable = Root.Runtime.experiments.isEnabled(Root.Runtime.ExperimentName.HEADER_OVERRIDES);
263 if (!overrideable || !this.#workspace.uiSourceCodeForURL(this.#getHeaderOverridesFileUrl())) {
264 return LitHtml.nothing;
265 }
266
267 const overridesSetting: Common.Settings.Setting<boolean> =
268 Common.Settings.Settings.instance().moduleSetting('persistenceNetworkOverridesEnabled');
269 // Disabled until https://crbug.com/1079231 is fixed.
270 // clang-format off
271 const fileIcon = overridesSetting.get() ? html`
272 <${IconButton.Icon.Icon.litTagName} class="inline-icon purple-dot" .data=${{
Benedikt Meurer79574fe2023-04-24 07:50:42273 iconName: 'document',
274 color: 'var(--icon-default)',
275 width: '16px',
276 height: '16px',
Wolfgang Beyer185fb1f2022-08-01 13:48:25277 } as IconButton.Icon.IconData}>
278 </${IconButton.Icon.Icon.litTagName}>` : html`
279 <${IconButton.Icon.Icon.litTagName} class="inline-icon" .data=${{
Andrés Olivaresd5f066d2023-04-11 08:49:03280 iconName: 'document',
281 color: 'var(--icon-default)',
Benedikt Meurer79574fe2023-04-24 07:50:42282 width: '16px',
283 height: '16px',
Wolfgang Beyer185fb1f2022-08-01 13:48:25284 } as IconButton.Icon.IconData}>
285 </${IconButton.Icon.Icon.litTagName}>`;
286 // clang-format on
287
288 const revealHeadersFile = (event: Event): void => {
289 event.preventDefault();
290 const uiSourceCode = this.#workspace.uiSourceCodeForURL(this.#getHeaderOverridesFileUrl());
291 if (uiSourceCode) {
292 Sources.SourcesPanel.SourcesPanel.instance().showUISourceCode(uiSourceCode);
Wolfgang Beyer24043612023-02-16 13:22:10293 Sources.SourcesPanel.SourcesPanel.instance().revealInNavigator(uiSourceCode);
Wolfgang Beyer185fb1f2022-08-01 13:48:25294 }
295 };
296
Wolfgang Beyer1c924582023-03-22 15:28:04297 // Disabled until https://crbug.com/1079231 is fixed.
298 // clang-format off
Wolfgang Beyer185fb1f2022-08-01 13:48:25299 return html`
Wolfgang Beyer1c924582023-03-22 15:28:04300 <x-link href="https://goo.gle/devtools-override" class="link devtools-link">
301 <${IconButton.Icon.Icon.litTagName} class="inline-icon" .data=${{
Danil Somsikovc1bbaf82023-04-11 16:02:49302 iconName: 'help',
303 color: 'var(--icon-link)',
Wolfgang Beyer1c924582023-03-22 15:28:04304 width: '16px',
305 height: '16px',
306 } as IconButton.Icon.IconData}>
307 </${IconButton.Icon.Icon.litTagName}
308 ></x-link>
Wolfgang Beyerc8f09372022-09-19 14:27:27309 <x-link @click=${revealHeadersFile} class="link devtools-link" title=${UIStrings.revealHeaderOverrides}>
Wolfgang Beyer185fb1f2022-08-01 13:48:25310 ${fileIcon}${i18nString(UIStrings.headerOverrides)}
311 </x-link>
312 `;
Wolfgang Beyer1c924582023-03-22 15:28:04313 // clang-format on
Wolfgang Beyer185fb1f2022-08-01 13:48:25314 }
315
316 #getHeaderOverridesFileUrl(): Platform.DevToolsPath.UrlString {
317 if (!this.#request) {
318 return Platform.DevToolsPath.EmptyUrlString;
319 }
320 const fileUrl = Persistence.NetworkPersistenceManager.NetworkPersistenceManager.instance().fileUrlFromNetworkUrl(
321 this.#request.url(), /* ignoreInactive */ true);
322 return fileUrl.substring(0, fileUrl.lastIndexOf('/')) + '/' +
323 Persistence.NetworkPersistenceManager.HEADERS_FILENAME as Platform.DevToolsPath.UrlString;
324 }
325
Wolfgang Beyer3e62bd72022-07-28 14:26:48326 #renderRequestHeaders(): LitHtml.LitTemplate {
327 if (!this.#request) {
328 return LitHtml.nothing;
329 }
Wolfgang Beyer7f495ee2022-07-08 13:45:13330 const requestHeadersText = this.#request.requestHeadersText();
331
Wolfgang Beyer235544c2022-05-24 08:07:45332 const toggleShowRaw = (): void => {
333 this.#showRequestHeadersText = !this.#showRequestHeadersText;
334 this.#render();
335 };
336
Wolfgang Beyer235544c2022-05-24 08:07:45337 // Disabled until https://crbug.com/1079231 is fixed.
338 // clang-format off
339 return html`
340 <${Category.litTagName}
341 @togglerawevent=${toggleShowRaw}
342 .data=${{
343 name: 'requestHeaders',
344 title: i18nString(UIStrings.requestHeaders),
345 headerCount: this.#request.requestHeaders().length,
346 checked: requestHeadersText? this.#showRequestHeadersText : undefined,
Wolfgang Beyer40426d82022-08-03 11:08:07347 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.Request,
Wolfgang Beyer235544c2022-05-24 08:07:45348 } as CategoryData}
349 aria-label=${i18nString(UIStrings.requestHeaders)}
350 >
Wolfgang Beyerd1dd7962022-05-24 15:48:48351 ${(this.#showRequestHeadersText && requestHeadersText) ?
352 this.#renderRawHeaders(requestHeadersText, false) : html`
Wolfgang Beyer65a82282022-09-05 08:22:37353 <${RequestHeaderSection.litTagName} .data=${{
354 request: this.#request,
355 toReveal: this.#toReveal,
356 } as RequestHeaderSectionData}></${RequestHeaderSection.litTagName}>
Wolfgang Beyer235544c2022-05-24 08:07:45357 `}
358 </${Category.litTagName}>
359 `;
Wolfgang Beyerbcf4a832022-07-25 12:34:47360 // clang-format on
Wolfgang Beyer235544c2022-05-24 08:07:45361 }
362
Wolfgang Beyerd1dd7962022-05-24 15:48:48363 #renderRawHeaders(rawHeadersText: string, forResponseHeaders: boolean): LitHtml.TemplateResult {
364 const trimmed = rawHeadersText.trim();
365 const showFull = forResponseHeaders ? this.#showResponseHeadersTextFull : this.#showRequestHeadersTextFull;
366 const isShortened = !showFull && trimmed.length > RAW_HEADER_CUTOFF;
367
368 const showMore = ():void => {
369 if (forResponseHeaders) {
370 this.#showResponseHeadersTextFull = true;
371 } else {
372 this.#showRequestHeadersTextFull = true;
373 }
374 this.#render();
375 };
376
377 const onContextMenuOpen = (event: Event): void => {
378 const showFull = forResponseHeaders ? this.#showResponseHeadersTextFull : this.#showRequestHeadersTextFull;
379 if (!showFull) {
380 const contextMenu = new UI.ContextMenu.ContextMenu(event);
381 const section = contextMenu.newSection();
382 section.appendItem(i18nString(UIStrings.showMore), showMore);
383 void contextMenu.show();
384 }
385 };
386
387 const addContextMenuListener = (el: Element):void => {
388 if (isShortened) {
389 el.addEventListener('contextmenu', onContextMenuOpen);
390 }
391 };
392
393 return html`
394 <div class="row raw-headers-row" on-render=${ComponentHelpers.Directives.nodeRenderedCallback(addContextMenuListener)}>
395 <div class="raw-headers">${isShortened ? trimmed.substring(0, RAW_HEADER_CUTOFF) : trimmed}</div>
396 ${isShortened ? html`
397 <${Buttons.Button.Button.litTagName}
398 .size=${Buttons.Button.Size.SMALL}
399 .variant=${Buttons.Button.Variant.SECONDARY}
400 @click=${showMore}
401 >${i18nString(UIStrings.showMore)}</${Buttons.Button.Button.litTagName}>
402 ` : LitHtml.nothing}
403 </div>
404 `;
405 }
406
Wolfgang Beyer3e62bd72022-07-28 14:26:48407 #renderGeneralSection(): LitHtml.LitTemplate {
408 if (!this.#request) {
409 return LitHtml.nothing;
410 }
Wolfgang Beyer40530b682022-05-17 13:02:01411
Wolfgang Beyer40426d82022-08-03 11:08:07412 const statusClasses = [];
Wolfgang Beyer40530b682022-05-17 13:02:01413 if (this.#request.statusCode < 300 || this.#request.statusCode === 304) {
Wolfgang Beyer40426d82022-08-03 11:08:07414 statusClasses.push('green-circle');
Wolfgang Beyer40530b682022-05-17 13:02:01415 } else if (this.#request.statusCode < 400) {
Wolfgang Beyer40426d82022-08-03 11:08:07416 statusClasses.push('yellow-circle');
417 } else {
418 statusClasses.push('red-circle');
Wolfgang Beyer40530b682022-05-17 13:02:01419 }
420
421 let statusText = this.#request.statusCode + ' ' + this.#request.statusText;
Wolfgang Beyer40530b682022-05-17 13:02:01422 if (this.#request.cachedInMemory()) {
423 statusText += ' ' + i18nString(UIStrings.fromMemoryCache);
Wolfgang Beyer40426d82022-08-03 11:08:07424 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01425 } else if (this.#request.fetchedViaServiceWorker) {
426 statusText += ' ' + i18nString(UIStrings.fromServiceWorker);
Wolfgang Beyer40426d82022-08-03 11:08:07427 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01428 } else if (this.#request.redirectSourceSignedExchangeInfoHasNoErrors()) {
429 statusText += ' ' + i18nString(UIStrings.fromSignedexchange);
Wolfgang Beyer40426d82022-08-03 11:08:07430 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01431 } else if (this.#request.webBundleInnerRequestInfo()) {
432 statusText += ' ' + i18nString(UIStrings.fromWebBundle);
Wolfgang Beyer40426d82022-08-03 11:08:07433 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01434 } else if (this.#request.fromPrefetchCache()) {
435 statusText += ' ' + i18nString(UIStrings.fromPrefetchCache);
Wolfgang Beyer40426d82022-08-03 11:08:07436 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01437 } else if (this.#request.cached()) {
438 statusText += ' ' + i18nString(UIStrings.fromDiskCache);
Wolfgang Beyer40426d82022-08-03 11:08:07439 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01440 }
441
442 // Disabled until https://crbug.com/1079231 is fixed.
443 // clang-format off
444 return html`
Wolfgang Beyer235544c2022-05-24 08:07:45445 <${Category.litTagName}
Wolfgang Beyer40426d82022-08-03 11:08:07446 .data=${{
447 name: 'general',
448 title: i18nString(UIStrings.general),
449 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.General,
450 } as CategoryData}
Wolfgang Beyer235544c2022-05-24 08:07:45451 aria-label=${i18nString(UIStrings.general)}
452 >
Wolfgang Beyer40426d82022-08-03 11:08:07453 ${this.#renderGeneralRow(i18nString(UIStrings.requestUrl), this.#request.url())}
454 ${this.#request.statusCode? this.#renderGeneralRow(i18nString(UIStrings.requestMethod), this.#request.requestMethod) : LitHtml.nothing}
455 ${this.#request.statusCode? this.#renderGeneralRow(i18nString(UIStrings.statusCode), statusText, statusClasses) : LitHtml.nothing}
456 ${this.#request.remoteAddress()? this.#renderGeneralRow(i18nString(UIStrings.remoteAddress), this.#request.remoteAddress()) : LitHtml.nothing}
457 ${this.#request.referrerPolicy()? this.#renderGeneralRow(i18nString(UIStrings.referrerPolicy), String(this.#request.referrerPolicy())) : LitHtml.nothing}
Wolfgang Beyer40530b682022-05-17 13:02:01458 </${Category.litTagName}>
459 `;
460 // clang-format on
461 }
Wolfgang Beyer40426d82022-08-03 11:08:07462
463 #renderGeneralRow(name: Common.UIString.LocalizedString, value: string, classNames?: string[]): LitHtml.LitTemplate {
464 const isHighlighted = this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.General &&
Wolfgang Beyerfd504ec2022-09-06 09:48:11465 name.toLowerCase() === this.#toReveal?.header?.toLowerCase();
Wolfgang Beyer40426d82022-08-03 11:08:07466 return html`
467 <div class="row ${isHighlighted ? 'header-highlight' : ''}">
468 <div class="header-name">${name}:</div>
469 <div
470 class="header-value ${classNames?.join(' ')}"
471 @copy=${(): void => Host.userMetrics.actionTaken(Host.UserMetrics.Action.NetworkPanelCopyValue)}
472 >${value}</div>
473 </div>
474 `;
475 }
Wolfgang Beyer40530b682022-05-17 13:02:01476}
477
Wolfgang Beyer235544c2022-05-24 08:07:45478export class ToggleRawHeadersEvent extends Event {
479 static readonly eventName = 'togglerawevent';
480
481 constructor() {
482 super(ToggleRawHeadersEvent.eventName, {});
483 }
484}
485
Wolfgang Beyer40530b682022-05-17 13:02:01486export interface CategoryData {
487 name: string;
488 title: Common.UIString.LocalizedString;
Wolfgang Beyer235544c2022-05-24 08:07:45489 headerCount?: number;
490 checked?: boolean;
Wolfgang Beyer185fb1f2022-08-01 13:48:25491 additionalContent?: LitHtml.LitTemplate;
Wolfgang Beyer40426d82022-08-03 11:08:07492 forceOpen?: boolean;
Wolfgang Beyer40530b682022-05-17 13:02:01493}
494
495export class Category extends HTMLElement {
496 static readonly litTagName = LitHtml.literal`devtools-request-headers-category`;
497 readonly #shadow = this.attachShadow({mode: 'open'});
498 #expandedSetting?: Common.Settings.Setting<boolean>;
499 #title: Common.UIString.LocalizedString = Common.UIString.LocalizedEmptyString;
Wolfgang Beyer235544c2022-05-24 08:07:45500 #headerCount?: number = undefined;
501 #checked: boolean|undefined = undefined;
Wolfgang Beyer185fb1f2022-08-01 13:48:25502 #additionalContent: LitHtml.LitTemplate|undefined = undefined;
Wolfgang Beyer40426d82022-08-03 11:08:07503 #forceOpen: boolean|undefined = undefined;
Wolfgang Beyer40530b682022-05-17 13:02:01504
505 connectedCallback(): void {
Jack Franklin1f19d522022-07-06 14:00:18506 this.#shadow.adoptedStyleSheets = [requestHeadersViewStyles, Input.checkboxStyles];
Wolfgang Beyer40530b682022-05-17 13:02:01507 }
508
509 set data(data: CategoryData) {
510 this.#title = data.title;
511 this.#expandedSetting =
512 Common.Settings.Settings.instance().createSetting('request-info-' + data.name + '-category-expanded', true);
Wolfgang Beyer235544c2022-05-24 08:07:45513 this.#headerCount = data.headerCount;
514 this.#checked = data.checked;
Wolfgang Beyer185fb1f2022-08-01 13:48:25515 this.#additionalContent = data.additionalContent;
Wolfgang Beyer40426d82022-08-03 11:08:07516 this.#forceOpen = data.forceOpen;
Wolfgang Beyer40530b682022-05-17 13:02:01517 this.#render();
518 }
519
Wolfgang Beyer235544c2022-05-24 08:07:45520 #onCheckboxToggle(): void {
521 this.dispatchEvent(new ToggleRawHeadersEvent());
522 }
523
Wolfgang Beyer40530b682022-05-17 13:02:01524 #render(): void {
Wolfgang Beyer40426d82022-08-03 11:08:07525 const isOpen = (this.#expandedSetting ? this.#expandedSetting.get() : true) || this.#forceOpen;
Wolfgang Beyer40530b682022-05-17 13:02:01526 // Disabled until https://crbug.com/1079231 is fixed.
527 // clang-format off
528 render(html`
Wolfgang Beyer235544c2022-05-24 08:07:45529 <details ?open=${isOpen} @toggle=${this.#onToggle}>
530 <summary class="header" @keydown=${this.#onSummaryKeyDown}>
Wolfgang Beyer185fb1f2022-08-01 13:48:25531 <div class="header-grid-container">
532 <div>
Wolfgang Beyerd2ea5672022-09-05 07:37:28533 ${this.#title}${this.#headerCount !== undefined ?
Wolfgang Beyer185fb1f2022-08-01 13:48:25534 html`<span class="header-count"> (${this.#headerCount})</span>` :
535 LitHtml.nothing
536 }
537 </div>
538 <div class="hide-when-closed">
539 ${this.#checked !== undefined ? html`
540 <label><input type="checkbox" .checked=${this.#checked} @change=${this.#onCheckboxToggle} />${i18nString(UIStrings.raw)}</label>
541 ` : LitHtml.nothing}
542 </div>
543 <div class="hide-when-closed">${this.#additionalContent}</div>
Wolfgang Beyer235544c2022-05-24 08:07:45544 </summary>
Wolfgang Beyer40530b682022-05-17 13:02:01545 <slot></slot>
546 </details>
547 `, this.#shadow, {host: this});
548 // clang-format on
549 }
550
551 #onSummaryKeyDown(event: KeyboardEvent): void {
552 if (!event.target) {
553 return;
554 }
555 const summaryElement = event.target as HTMLElement;
556 const detailsElement = summaryElement.parentElement as HTMLDetailsElement;
557 if (!detailsElement) {
558 throw new Error('<details> element is not found for a <summary> element');
559 }
560 switch (event.key) {
561 case 'ArrowLeft':
562 detailsElement.open = false;
563 break;
564 case 'ArrowRight':
565 detailsElement.open = true;
566 break;
567 }
568 }
569
570 #onToggle(event: Event): void {
571 this.#expandedSetting?.set((event.target as HTMLDetailsElement).open);
572 }
573}
574
575ComponentHelpers.CustomElements.defineComponent('devtools-request-headers', RequestHeadersComponent);
576ComponentHelpers.CustomElements.defineComponent('devtools-request-headers-category', Category);
577
578declare global {
579 // eslint-disable-next-line @typescript-eslint/no-unused-vars
580 interface HTMLElementTagNameMap {
581 'devtools-request-headers': RequestHeadersComponent;
582 'devtools-request-headers-category': Category;
583 }
584}