blob: 65c0f10680666d32efb55eccf73bf530ca308898 [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';
Danil Somsikov5875b3b2023-05-17 21:41:3618import * as LegacyWrapper from '../../../ui/components/legacy_wrapper/legacy_wrapper.js';
Wolfgang Beyer40530b682022-05-17 13:02:0119import * as UI from '../../../ui/legacy/legacy.js';
20import * as LitHtml from '../../../ui/lit-html/lit-html.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:2521import * as Sources from '../../sources/sources.js';
Wolfgang Beyerd2286472022-09-02 18:32:3022
Wolfgang Beyer65a82282022-09-05 08:22:3723import {type RequestHeaderSectionData, RequestHeaderSection} from './RequestHeaderSection.js';
Wolfgang Beyerb9f7d152022-10-10 16:10:1424import {
25 type ResponseHeaderSectionData,
26 ResponseHeaderSection,
27 RESPONSE_HEADER_SECTION_DATA_KEY,
28} from './ResponseHeaderSection.js';
Wolfgang Beyer40530b682022-05-17 13:02:0129
30import requestHeadersViewStyles from './RequestHeadersView.css.js';
31
Wolfgang Beyerd1dd7962022-05-24 15:48:4832const RAW_HEADER_CUTOFF = 3000;
Wolfgang Beyer40530b682022-05-17 13:02:0133const {render, html} = LitHtml;
34
35const UIStrings = {
36 /**
Jack Franklinfd72c072022-12-21 11:45:0137 *@description Text in Request Headers View of the Network panel
38 */
Wolfgang Beyer37fcd282022-06-29 09:34:1739 fromDiskCache: '(from disk cache)',
40 /**
Jack Franklinfd72c072022-12-21 11:45:0141 *@description Text in Request Headers View of the Network panel
42 */
Wolfgang Beyer40530b682022-05-17 13:02:0143 fromMemoryCache: '(from memory cache)',
44 /**
Jack Franklinfd72c072022-12-21 11:45:0145 *@description Text in Request Headers View of the Network panel
46 */
Wolfgang Beyer37fcd282022-06-29 09:34:1747 fromPrefetchCache: '(from prefetch cache)',
48 /**
Jack Franklinfd72c072022-12-21 11:45:0149 *@description Text in Request Headers View of the Network panel
50 */
Wolfgang Beyer40530b682022-05-17 13:02:0151 fromServiceWorker: '(from `service worker`)',
52 /**
Jack Franklinfd72c072022-12-21 11:45:0153 *@description Text in Request Headers View of the Network panel
54 */
Wolfgang Beyer40530b682022-05-17 13:02:0155 fromSignedexchange: '(from signed-exchange)',
56 /**
Jack Franklinfd72c072022-12-21 11:45:0157 *@description Text in Request Headers View of the Network panel
58 */
Wolfgang Beyer40530b682022-05-17 13:02:0159 fromWebBundle: '(from Web Bundle)',
60 /**
Jack Franklinfd72c072022-12-21 11:45:0161 *@description Section header for a list of the main aspects of a http request
62 */
Wolfgang Beyer40530b682022-05-17 13:02:0163 general: 'General',
64 /**
Jack Franklinfd72c072022-12-21 11:45:0165 *@description Label for a link from the network panel's headers view to the file in which
66 * header overrides are defined in the sources panel.
67 */
Wolfgang Beyer185fb1f2022-08-01 13:48:2568 headerOverrides: 'Header overrides',
69 /**
Jack Franklinfd72c072022-12-21 11:45:0170 *@description Label for a checkbox to switch between raw and parsed headers
71 */
Wolfgang Beyer235544c2022-05-24 08:07:4572 raw: 'Raw',
73 /**
Jack Franklinfd72c072022-12-21 11:45:0174 *@description Text in Request Headers View of the Network panel
75 */
Wolfgang Beyer37fcd282022-06-29 09:34:1776 referrerPolicy: 'Referrer Policy',
Wolfgang Beyer235544c2022-05-24 08:07:4577 /**
Jack Franklinfd72c072022-12-21 11:45:0178 *@description Text in Network Log View Columns of the Network panel
79 */
Wolfgang Beyer37fcd282022-06-29 09:34:1780 remoteAddress: 'Remote Address',
81 /**
Jack Franklinfd72c072022-12-21 11:45:0182 *@description Text in Request Headers View of the Network panel
83 */
Wolfgang Beyer37fcd282022-06-29 09:34:1784 requestHeaders: 'Request Headers',
Wolfgang Beyer40530b682022-05-17 13:02:0185 /**
Jack Franklinfd72c072022-12-21 11:45:0186 *@description The HTTP method of a request
87 */
Wolfgang Beyer40530b682022-05-17 13:02:0188 requestMethod: 'Request Method',
89 /**
Jack Franklinfd72c072022-12-21 11:45:0190 *@description The URL of a request
91 */
Wolfgang Beyer37fcd282022-06-29 09:34:1792 requestUrl: 'Request URL',
93 /**
Jack Franklinfd72c072022-12-21 11:45:0194 *@description A context menu item in the Network Log View Columns of the Network panel
95 */
Wolfgang Beyer235544c2022-05-24 08:07:4596 responseHeaders: 'Response Headers',
97 /**
Jack Franklinfd72c072022-12-21 11:45:0198 *@description Title text for a link to the Sources panel to the file containing the header override definitions
99 */
Wolfgang Beyerc8f09372022-09-19 14:27:27100 revealHeaderOverrides: 'Reveal header override definitions',
101 /**
Jack Franklinfd72c072022-12-21 11:45:01102 *@description Text to show more content
103 */
Wolfgang Beyerd1dd7962022-05-24 15:48:48104 showMore: 'Show more',
105 /**
Jack Franklinfd72c072022-12-21 11:45:01106 *@description HTTP response code
107 */
Wolfgang Beyer40530b682022-05-17 13:02:01108 statusCode: 'Status Code',
Wolfgang Beyer40530b682022-05-17 13:02:01109};
110const str_ = i18n.i18n.registerUIStrings('panels/network/components/RequestHeadersView.ts', UIStrings);
111const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
112
Danil Somsikov5875b3b2023-05-17 21:41:36113export class RequestHeadersView extends LegacyWrapper.LegacyWrapper.WrappableComponent {
114 #request: Readonly<SDK.NetworkRequest.NetworkRequest>;
115 static readonly litTagName = LitHtml.literal`devtools-request-headers`;
116 readonly #shadow = this.attachShadow({mode: 'open'});
117 #showResponseHeadersText = false;
118 #showRequestHeadersText = false;
119 #showResponseHeadersTextFull = false;
120 #showRequestHeadersTextFull = false;
121 #toReveal?: {section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string} = undefined;
122 readonly #workspace = Workspace.Workspace.WorkspaceImpl.instance();
Wolfgang Beyer40530b682022-05-17 13:02:01123
124 constructor(request: SDK.NetworkRequest.NetworkRequest) {
125 super();
126 this.#request = request;
Wolfgang Beyer40530b682022-05-17 13:02:01127 }
128
Randolf Jungffd14242023-04-19 00:32:25129 override wasShown(): void {
Wolfgang Beyer40530b682022-05-17 13:02:01130 this.#request.addEventListener(SDK.NetworkRequest.Events.RemoteAddressChanged, this.#refreshHeadersView, this);
131 this.#request.addEventListener(SDK.NetworkRequest.Events.FinishedLoading, this.#refreshHeadersView, this);
Wolfgang Beyer87150062022-07-29 10:45:47132 this.#request.addEventListener(SDK.NetworkRequest.Events.RequestHeadersChanged, this.#refreshHeadersView, this);
Wolfgang Beyerb9f7d152022-10-10 16:10:14133 this.#request.addEventListener(
134 SDK.NetworkRequest.Events.ResponseHeadersChanged, this.#resetAndRefreshHeadersView, this);
Wolfgang Beyer40530b682022-05-17 13:02:01135 this.#refreshHeadersView();
136 }
137
Randolf Jungffd14242023-04-19 00:32:25138 override willHide(): void {
Wolfgang Beyer40530b682022-05-17 13:02:01139 this.#request.removeEventListener(SDK.NetworkRequest.Events.RemoteAddressChanged, this.#refreshHeadersView, this);
140 this.#request.removeEventListener(SDK.NetworkRequest.Events.FinishedLoading, this.#refreshHeadersView, this);
Wolfgang Beyer87150062022-07-29 10:45:47141 this.#request.removeEventListener(SDK.NetworkRequest.Events.RequestHeadersChanged, this.#refreshHeadersView, this);
Wolfgang Beyerb9f7d152022-10-10 16:10:14142 this.#request.removeEventListener(
143 SDK.NetworkRequest.Events.ResponseHeadersChanged, this.#resetAndRefreshHeadersView, this);
144 }
145
146 #resetAndRefreshHeadersView(): void {
147 this.#request.deleteAssociatedData(RESPONSE_HEADER_SECTION_DATA_KEY);
Danil Somsikov5875b3b2023-05-17 21:41:36148 void this.render();
Wolfgang Beyer40530b682022-05-17 13:02:01149 }
150
151 #refreshHeadersView(): void {
Danil Somsikov5875b3b2023-05-17 21:41:36152 void this.render();
Wolfgang Beyer40530b682022-05-17 13:02:01153 }
Wolfgang Beyer40426d82022-08-03 11:08:07154
155 revealHeader(section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string): void {
Danil Somsikov5875b3b2023-05-17 21:41:36156 this.#toReveal = {section, header};
157 void this.render();
Wolfgang Beyer40530b682022-05-17 13:02:01158 }
159
160 connectedCallback(): void {
161 this.#shadow.adoptedStyleSheets = [requestHeadersViewStyles];
Wolfgang Beyer185fb1f2022-08-01 13:48:25162 this.#workspace.addEventListener(
163 Workspace.Workspace.Events.UISourceCodeAdded, this.#uiSourceCodeAddedOrRemoved, this);
164 this.#workspace.addEventListener(
165 Workspace.Workspace.Events.UISourceCodeRemoved, this.#uiSourceCodeAddedOrRemoved, this);
Wolfgang Beyer1e62b692022-11-07 13:52:08166 Common.Settings.Settings.instance()
167 .moduleSetting('persistenceNetworkOverridesEnabled')
Danil Somsikov5875b3b2023-05-17 21:41:36168 .addChangeListener(this.render, this);
Wolfgang Beyer185fb1f2022-08-01 13:48:25169 }
170
171 disconnectedCallback(): void {
172 this.#workspace.removeEventListener(
173 Workspace.Workspace.Events.UISourceCodeAdded, this.#uiSourceCodeAddedOrRemoved, this);
174 this.#workspace.removeEventListener(
175 Workspace.Workspace.Events.UISourceCodeRemoved, this.#uiSourceCodeAddedOrRemoved, this);
Wolfgang Beyer1e62b692022-11-07 13:52:08176 Common.Settings.Settings.instance()
177 .moduleSetting('persistenceNetworkOverridesEnabled')
Danil Somsikov5875b3b2023-05-17 21:41:36178 .removeChangeListener(this.render, this);
Wolfgang Beyer185fb1f2022-08-01 13:48:25179 }
180
181 #uiSourceCodeAddedOrRemoved(event: Common.EventTarget.EventTargetEvent<Workspace.UISourceCode.UISourceCode>): void {
182 if (this.#getHeaderOverridesFileUrl() === event.data.url()) {
Danil Somsikov5875b3b2023-05-17 21:41:36183 void this.render();
Wolfgang Beyer185fb1f2022-08-01 13:48:25184 }
Wolfgang Beyer40530b682022-05-17 13:02:01185 }
186
Danil Somsikov5875b3b2023-05-17 21:41:36187 override async render(): Promise<void> {
Wolfgang Beyer3e62bd72022-07-28 14:26:48188 if (!this.#request) {
189 return;
190 }
Wolfgang Beyer40530b682022-05-17 13:02:01191
192 // Disabled until https://crbug.com/1079231 is fixed.
193 // clang-format off
194 render(html`
195 ${this.#renderGeneralSection()}
Wolfgang Beyer235544c2022-05-24 08:07:45196 ${this.#renderResponseHeaders()}
197 ${this.#renderRequestHeaders()}
Wolfgang Beyer40530b682022-05-17 13:02:01198 `, this.#shadow, {host: this});
199 // clang-format on
200 }
201
Wolfgang Beyer3e62bd72022-07-28 14:26:48202 #renderResponseHeaders(): LitHtml.LitTemplate {
203 if (!this.#request) {
204 return LitHtml.nothing;
205 }
Wolfgang Beyer235544c2022-05-24 08:07:45206
207 const toggleShowRaw = (): void => {
208 this.#showResponseHeadersText = !this.#showResponseHeadersText;
Danil Somsikov5875b3b2023-05-17 21:41:36209 void this.render();
Wolfgang Beyer235544c2022-05-24 08:07:45210 };
211
212 // Disabled until https://crbug.com/1079231 is fixed.
213 // clang-format off
214 return html`
215 <${Category.litTagName}
216 @togglerawevent=${toggleShowRaw}
217 .data=${{
218 name: 'responseHeaders',
219 title: i18nString(UIStrings.responseHeaders),
220 headerCount: this.#request.sortedResponseHeaders.length,
221 checked: this.#request.responseHeadersText ? this.#showResponseHeadersText : undefined,
Wolfgang Beyer185fb1f2022-08-01 13:48:25222 additionalContent: this.#renderHeaderOverridesLink(),
Wolfgang Beyer40426d82022-08-03 11:08:07223 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.Response,
Wolfgang Beyer235544c2022-05-24 08:07:45224 } as CategoryData}
225 aria-label=${i18nString(UIStrings.responseHeaders)}
226 >
Wolfgang Beyerd1dd7962022-05-24 15:48:48227 ${this.#showResponseHeadersText ?
228 this.#renderRawHeaders(this.#request.responseHeadersText, true) : html`
Wolfgang Beyerd2286472022-09-02 18:32:30229 <${ResponseHeaderSection.litTagName} .data=${{
230 request: this.#request,
231 toReveal: this.#toReveal,
232 } as ResponseHeaderSectionData}></${ResponseHeaderSection.litTagName}>
Wolfgang Beyer235544c2022-05-24 08:07:45233 `}
234 </${Category.litTagName}>
235 `;
Wolfgang Beyerbcf4a832022-07-25 12:34:47236 // clang-format on
Wolfgang Beyer235544c2022-05-24 08:07:45237 }
238
Wolfgang Beyer185fb1f2022-08-01 13:48:25239 #renderHeaderOverridesLink(): LitHtml.LitTemplate {
240 const overrideable = Root.Runtime.experiments.isEnabled(Root.Runtime.ExperimentName.HEADER_OVERRIDES);
241 if (!overrideable || !this.#workspace.uiSourceCodeForURL(this.#getHeaderOverridesFileUrl())) {
242 return LitHtml.nothing;
243 }
244
245 const overridesSetting: Common.Settings.Setting<boolean> =
246 Common.Settings.Settings.instance().moduleSetting('persistenceNetworkOverridesEnabled');
247 // Disabled until https://crbug.com/1079231 is fixed.
248 // clang-format off
Kateryna Prokopenko49cec5c2023-04-24 14:22:03249 const fileIcon = html`
250 <${IconButton.Icon.Icon.litTagName} class=${overridesSetting.get() ? 'inline-icon dot purple': 'inline-icon'} .data=${{
Andrés Olivaresd5f066d2023-04-11 08:49:03251 iconName: 'document',
252 color: 'var(--icon-default)',
Benedikt Meurer79574fe2023-04-24 07:50:42253 width: '16px',
254 height: '16px',
Wolfgang Beyer185fb1f2022-08-01 13:48:25255 } as IconButton.Icon.IconData}>
256 </${IconButton.Icon.Icon.litTagName}>`;
257 // clang-format on
258
259 const revealHeadersFile = (event: Event): void => {
260 event.preventDefault();
261 const uiSourceCode = this.#workspace.uiSourceCodeForURL(this.#getHeaderOverridesFileUrl());
262 if (uiSourceCode) {
263 Sources.SourcesPanel.SourcesPanel.instance().showUISourceCode(uiSourceCode);
Wolfgang Beyer24043612023-02-16 13:22:10264 Sources.SourcesPanel.SourcesPanel.instance().revealInNavigator(uiSourceCode);
Wolfgang Beyer185fb1f2022-08-01 13:48:25265 }
266 };
267
Wolfgang Beyer1c924582023-03-22 15:28:04268 // Disabled until https://crbug.com/1079231 is fixed.
269 // clang-format off
Wolfgang Beyer185fb1f2022-08-01 13:48:25270 return html`
Wolfgang Beyer1c924582023-03-22 15:28:04271 <x-link href="https://goo.gle/devtools-override" class="link devtools-link">
272 <${IconButton.Icon.Icon.litTagName} class="inline-icon" .data=${{
Danil Somsikovc1bbaf82023-04-11 16:02:49273 iconName: 'help',
274 color: 'var(--icon-link)',
Wolfgang Beyer1c924582023-03-22 15:28:04275 width: '16px',
276 height: '16px',
277 } as IconButton.Icon.IconData}>
278 </${IconButton.Icon.Icon.litTagName}
279 ></x-link>
Wolfgang Beyerc8f09372022-09-19 14:27:27280 <x-link @click=${revealHeadersFile} class="link devtools-link" title=${UIStrings.revealHeaderOverrides}>
Wolfgang Beyer185fb1f2022-08-01 13:48:25281 ${fileIcon}${i18nString(UIStrings.headerOverrides)}
282 </x-link>
283 `;
Wolfgang Beyer1c924582023-03-22 15:28:04284 // clang-format on
Wolfgang Beyer185fb1f2022-08-01 13:48:25285 }
286
287 #getHeaderOverridesFileUrl(): Platform.DevToolsPath.UrlString {
288 if (!this.#request) {
289 return Platform.DevToolsPath.EmptyUrlString;
290 }
291 const fileUrl = Persistence.NetworkPersistenceManager.NetworkPersistenceManager.instance().fileUrlFromNetworkUrl(
292 this.#request.url(), /* ignoreInactive */ true);
293 return fileUrl.substring(0, fileUrl.lastIndexOf('/')) + '/' +
294 Persistence.NetworkPersistenceManager.HEADERS_FILENAME as Platform.DevToolsPath.UrlString;
295 }
296
Wolfgang Beyer3e62bd72022-07-28 14:26:48297 #renderRequestHeaders(): LitHtml.LitTemplate {
298 if (!this.#request) {
299 return LitHtml.nothing;
300 }
Wolfgang Beyer7f495ee2022-07-08 13:45:13301 const requestHeadersText = this.#request.requestHeadersText();
302
Wolfgang Beyer235544c2022-05-24 08:07:45303 const toggleShowRaw = (): void => {
304 this.#showRequestHeadersText = !this.#showRequestHeadersText;
Danil Somsikov5875b3b2023-05-17 21:41:36305 void this.render();
Wolfgang Beyer235544c2022-05-24 08:07:45306 };
307
Wolfgang Beyer235544c2022-05-24 08:07:45308 // Disabled until https://crbug.com/1079231 is fixed.
309 // clang-format off
310 return html`
311 <${Category.litTagName}
312 @togglerawevent=${toggleShowRaw}
313 .data=${{
314 name: 'requestHeaders',
315 title: i18nString(UIStrings.requestHeaders),
316 headerCount: this.#request.requestHeaders().length,
317 checked: requestHeadersText? this.#showRequestHeadersText : undefined,
Wolfgang Beyer40426d82022-08-03 11:08:07318 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.Request,
Wolfgang Beyer235544c2022-05-24 08:07:45319 } as CategoryData}
320 aria-label=${i18nString(UIStrings.requestHeaders)}
321 >
Wolfgang Beyerd1dd7962022-05-24 15:48:48322 ${(this.#showRequestHeadersText && requestHeadersText) ?
323 this.#renderRawHeaders(requestHeadersText, false) : html`
Wolfgang Beyer65a82282022-09-05 08:22:37324 <${RequestHeaderSection.litTagName} .data=${{
325 request: this.#request,
326 toReveal: this.#toReveal,
327 } as RequestHeaderSectionData}></${RequestHeaderSection.litTagName}>
Wolfgang Beyer235544c2022-05-24 08:07:45328 `}
329 </${Category.litTagName}>
330 `;
Wolfgang Beyerbcf4a832022-07-25 12:34:47331 // clang-format on
Wolfgang Beyer235544c2022-05-24 08:07:45332 }
333
Wolfgang Beyerd1dd7962022-05-24 15:48:48334 #renderRawHeaders(rawHeadersText: string, forResponseHeaders: boolean): LitHtml.TemplateResult {
335 const trimmed = rawHeadersText.trim();
336 const showFull = forResponseHeaders ? this.#showResponseHeadersTextFull : this.#showRequestHeadersTextFull;
337 const isShortened = !showFull && trimmed.length > RAW_HEADER_CUTOFF;
338
339 const showMore = ():void => {
340 if (forResponseHeaders) {
341 this.#showResponseHeadersTextFull = true;
342 } else {
343 this.#showRequestHeadersTextFull = true;
344 }
Danil Somsikov5875b3b2023-05-17 21:41:36345 void this.render();
Wolfgang Beyerd1dd7962022-05-24 15:48:48346 };
347
348 const onContextMenuOpen = (event: Event): void => {
349 const showFull = forResponseHeaders ? this.#showResponseHeadersTextFull : this.#showRequestHeadersTextFull;
350 if (!showFull) {
351 const contextMenu = new UI.ContextMenu.ContextMenu(event);
352 const section = contextMenu.newSection();
353 section.appendItem(i18nString(UIStrings.showMore), showMore);
354 void contextMenu.show();
355 }
356 };
357
358 const addContextMenuListener = (el: Element):void => {
359 if (isShortened) {
360 el.addEventListener('contextmenu', onContextMenuOpen);
361 }
362 };
363
364 return html`
365 <div class="row raw-headers-row" on-render=${ComponentHelpers.Directives.nodeRenderedCallback(addContextMenuListener)}>
366 <div class="raw-headers">${isShortened ? trimmed.substring(0, RAW_HEADER_CUTOFF) : trimmed}</div>
367 ${isShortened ? html`
368 <${Buttons.Button.Button.litTagName}
369 .size=${Buttons.Button.Size.SMALL}
370 .variant=${Buttons.Button.Variant.SECONDARY}
371 @click=${showMore}
372 >${i18nString(UIStrings.showMore)}</${Buttons.Button.Button.litTagName}>
373 ` : LitHtml.nothing}
374 </div>
375 `;
376 }
377
Wolfgang Beyer3e62bd72022-07-28 14:26:48378 #renderGeneralSection(): LitHtml.LitTemplate {
379 if (!this.#request) {
380 return LitHtml.nothing;
381 }
Wolfgang Beyer40530b682022-05-17 13:02:01382
Wolfgang Beyer40426d82022-08-03 11:08:07383 const statusClasses = [];
Wolfgang Beyer40530b682022-05-17 13:02:01384 if (this.#request.statusCode < 300 || this.#request.statusCode === 304) {
Wolfgang Beyer40426d82022-08-03 11:08:07385 statusClasses.push('green-circle');
Wolfgang Beyer40530b682022-05-17 13:02:01386 } else if (this.#request.statusCode < 400) {
Wolfgang Beyer40426d82022-08-03 11:08:07387 statusClasses.push('yellow-circle');
388 } else {
389 statusClasses.push('red-circle');
Wolfgang Beyer40530b682022-05-17 13:02:01390 }
391
392 let statusText = this.#request.statusCode + ' ' + this.#request.statusText;
Wolfgang Beyer40530b682022-05-17 13:02:01393 if (this.#request.cachedInMemory()) {
394 statusText += ' ' + i18nString(UIStrings.fromMemoryCache);
Wolfgang Beyer40426d82022-08-03 11:08:07395 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01396 } else if (this.#request.fetchedViaServiceWorker) {
397 statusText += ' ' + i18nString(UIStrings.fromServiceWorker);
Wolfgang Beyer40426d82022-08-03 11:08:07398 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01399 } else if (this.#request.redirectSourceSignedExchangeInfoHasNoErrors()) {
400 statusText += ' ' + i18nString(UIStrings.fromSignedexchange);
Wolfgang Beyer40426d82022-08-03 11:08:07401 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01402 } else if (this.#request.webBundleInnerRequestInfo()) {
403 statusText += ' ' + i18nString(UIStrings.fromWebBundle);
Wolfgang Beyer40426d82022-08-03 11:08:07404 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01405 } else if (this.#request.fromPrefetchCache()) {
406 statusText += ' ' + i18nString(UIStrings.fromPrefetchCache);
Wolfgang Beyer40426d82022-08-03 11:08:07407 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01408 } else if (this.#request.cached()) {
409 statusText += ' ' + i18nString(UIStrings.fromDiskCache);
Wolfgang Beyer40426d82022-08-03 11:08:07410 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01411 }
412
413 // Disabled until https://crbug.com/1079231 is fixed.
414 // clang-format off
415 return html`
Wolfgang Beyer235544c2022-05-24 08:07:45416 <${Category.litTagName}
Wolfgang Beyer40426d82022-08-03 11:08:07417 .data=${{
418 name: 'general',
419 title: i18nString(UIStrings.general),
420 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.General,
421 } as CategoryData}
Wolfgang Beyer235544c2022-05-24 08:07:45422 aria-label=${i18nString(UIStrings.general)}
423 >
Wolfgang Beyer40426d82022-08-03 11:08:07424 ${this.#renderGeneralRow(i18nString(UIStrings.requestUrl), this.#request.url())}
425 ${this.#request.statusCode? this.#renderGeneralRow(i18nString(UIStrings.requestMethod), this.#request.requestMethod) : LitHtml.nothing}
426 ${this.#request.statusCode? this.#renderGeneralRow(i18nString(UIStrings.statusCode), statusText, statusClasses) : LitHtml.nothing}
427 ${this.#request.remoteAddress()? this.#renderGeneralRow(i18nString(UIStrings.remoteAddress), this.#request.remoteAddress()) : LitHtml.nothing}
428 ${this.#request.referrerPolicy()? this.#renderGeneralRow(i18nString(UIStrings.referrerPolicy), String(this.#request.referrerPolicy())) : LitHtml.nothing}
Wolfgang Beyer40530b682022-05-17 13:02:01429 </${Category.litTagName}>
430 `;
431 // clang-format on
432 }
Wolfgang Beyer40426d82022-08-03 11:08:07433
434 #renderGeneralRow(name: Common.UIString.LocalizedString, value: string, classNames?: string[]): LitHtml.LitTemplate {
435 const isHighlighted = this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.General &&
Wolfgang Beyerfd504ec2022-09-06 09:48:11436 name.toLowerCase() === this.#toReveal?.header?.toLowerCase();
Wolfgang Beyer40426d82022-08-03 11:08:07437 return html`
438 <div class="row ${isHighlighted ? 'header-highlight' : ''}">
439 <div class="header-name">${name}:</div>
440 <div
441 class="header-value ${classNames?.join(' ')}"
442 @copy=${(): void => Host.userMetrics.actionTaken(Host.UserMetrics.Action.NetworkPanelCopyValue)}
443 >${value}</div>
444 </div>
445 `;
446 }
Wolfgang Beyer40530b682022-05-17 13:02:01447}
448
Wolfgang Beyer235544c2022-05-24 08:07:45449export class ToggleRawHeadersEvent extends Event {
450 static readonly eventName = 'togglerawevent';
451
452 constructor() {
453 super(ToggleRawHeadersEvent.eventName, {});
454 }
455}
456
Wolfgang Beyer40530b682022-05-17 13:02:01457export interface CategoryData {
458 name: string;
459 title: Common.UIString.LocalizedString;
Wolfgang Beyer235544c2022-05-24 08:07:45460 headerCount?: number;
461 checked?: boolean;
Wolfgang Beyer185fb1f2022-08-01 13:48:25462 additionalContent?: LitHtml.LitTemplate;
Wolfgang Beyer40426d82022-08-03 11:08:07463 forceOpen?: boolean;
Wolfgang Beyer40530b682022-05-17 13:02:01464}
465
466export class Category extends HTMLElement {
467 static readonly litTagName = LitHtml.literal`devtools-request-headers-category`;
468 readonly #shadow = this.attachShadow({mode: 'open'});
469 #expandedSetting?: Common.Settings.Setting<boolean>;
470 #title: Common.UIString.LocalizedString = Common.UIString.LocalizedEmptyString;
Wolfgang Beyer235544c2022-05-24 08:07:45471 #headerCount?: number = undefined;
472 #checked: boolean|undefined = undefined;
Wolfgang Beyer185fb1f2022-08-01 13:48:25473 #additionalContent: LitHtml.LitTemplate|undefined = undefined;
Wolfgang Beyer40426d82022-08-03 11:08:07474 #forceOpen: boolean|undefined = undefined;
Wolfgang Beyer40530b682022-05-17 13:02:01475
476 connectedCallback(): void {
Jack Franklin1f19d522022-07-06 14:00:18477 this.#shadow.adoptedStyleSheets = [requestHeadersViewStyles, Input.checkboxStyles];
Wolfgang Beyer40530b682022-05-17 13:02:01478 }
479
480 set data(data: CategoryData) {
481 this.#title = data.title;
482 this.#expandedSetting =
483 Common.Settings.Settings.instance().createSetting('request-info-' + data.name + '-category-expanded', true);
Wolfgang Beyer235544c2022-05-24 08:07:45484 this.#headerCount = data.headerCount;
485 this.#checked = data.checked;
Wolfgang Beyer185fb1f2022-08-01 13:48:25486 this.#additionalContent = data.additionalContent;
Wolfgang Beyer40426d82022-08-03 11:08:07487 this.#forceOpen = data.forceOpen;
Wolfgang Beyer40530b682022-05-17 13:02:01488 this.#render();
489 }
490
Wolfgang Beyer235544c2022-05-24 08:07:45491 #onCheckboxToggle(): void {
492 this.dispatchEvent(new ToggleRawHeadersEvent());
493 }
494
Wolfgang Beyer40530b682022-05-17 13:02:01495 #render(): void {
Wolfgang Beyer40426d82022-08-03 11:08:07496 const isOpen = (this.#expandedSetting ? this.#expandedSetting.get() : true) || this.#forceOpen;
Wolfgang Beyer40530b682022-05-17 13:02:01497 // Disabled until https://crbug.com/1079231 is fixed.
498 // clang-format off
499 render(html`
Wolfgang Beyer235544c2022-05-24 08:07:45500 <details ?open=${isOpen} @toggle=${this.#onToggle}>
501 <summary class="header" @keydown=${this.#onSummaryKeyDown}>
Wolfgang Beyer185fb1f2022-08-01 13:48:25502 <div class="header-grid-container">
503 <div>
Wolfgang Beyerd2ea5672022-09-05 07:37:28504 ${this.#title}${this.#headerCount !== undefined ?
Wolfgang Beyer185fb1f2022-08-01 13:48:25505 html`<span class="header-count"> (${this.#headerCount})</span>` :
506 LitHtml.nothing
507 }
508 </div>
509 <div class="hide-when-closed">
510 ${this.#checked !== undefined ? html`
511 <label><input type="checkbox" .checked=${this.#checked} @change=${this.#onCheckboxToggle} />${i18nString(UIStrings.raw)}</label>
512 ` : LitHtml.nothing}
513 </div>
514 <div class="hide-when-closed">${this.#additionalContent}</div>
Wolfgang Beyer235544c2022-05-24 08:07:45515 </summary>
Wolfgang Beyer40530b682022-05-17 13:02:01516 <slot></slot>
517 </details>
518 `, this.#shadow, {host: this});
519 // clang-format on
520 }
521
522 #onSummaryKeyDown(event: KeyboardEvent): void {
523 if (!event.target) {
524 return;
525 }
526 const summaryElement = event.target as HTMLElement;
527 const detailsElement = summaryElement.parentElement as HTMLDetailsElement;
528 if (!detailsElement) {
529 throw new Error('<details> element is not found for a <summary> element');
530 }
531 switch (event.key) {
532 case 'ArrowLeft':
533 detailsElement.open = false;
534 break;
535 case 'ArrowRight':
536 detailsElement.open = true;
537 break;
538 }
539 }
540
541 #onToggle(event: Event): void {
542 this.#expandedSetting?.set((event.target as HTMLDetailsElement).open);
543 }
544}
545
Danil Somsikov5875b3b2023-05-17 21:41:36546ComponentHelpers.CustomElements.defineComponent('devtools-request-headers', RequestHeadersView);
Wolfgang Beyer40530b682022-05-17 13:02:01547ComponentHelpers.CustomElements.defineComponent('devtools-request-headers-category', Category);
548
549declare global {
550 // eslint-disable-next-line @typescript-eslint/no-unused-vars
551 interface HTMLElementTagNameMap {
Danil Somsikov5875b3b2023-05-17 21:41:36552 'devtools-request-headers': RequestHeadersView;
Wolfgang Beyer40530b682022-05-17 13:02:01553 'devtools-request-headers-category': Category;
554 }
555}