Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 1 | // Copyright 2018 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 | |
| 5 | /** |
| 6 | * @implements {SDK.SDKModelObserver<!SDK.ServiceWorkerManager>} |
| 7 | * @unrestricted |
| 8 | */ |
| 9 | Audits2.AuditController = class extends Common.Object { |
| 10 | constructor(protocolService) { |
| 11 | super(); |
| 12 | |
| 13 | protocolService.registerStatusCallback( |
| 14 | message => this.dispatchEventToListeners(Audits2.Events.AuditProgressChanged, {message})); |
| 15 | |
| 16 | for (const preset of Audits2.Presets) |
| 17 | preset.setting.addChangeListener(this.recomputePageAuditability.bind(this)); |
| 18 | |
| 19 | SDK.targetManager.observeModels(SDK.ServiceWorkerManager, this); |
| 20 | SDK.targetManager.addEventListener( |
| 21 | SDK.TargetManager.Events.InspectedURLChanged, this.recomputePageAuditability, this); |
| 22 | } |
| 23 | |
| 24 | /** |
| 25 | * @override |
| 26 | * @param {!SDK.ServiceWorkerManager} serviceWorkerManager |
| 27 | */ |
| 28 | modelAdded(serviceWorkerManager) { |
| 29 | if (this._manager) |
| 30 | return; |
| 31 | |
| 32 | this._manager = serviceWorkerManager; |
| 33 | this._serviceWorkerListeners = [ |
| 34 | this._manager.addEventListener( |
| 35 | SDK.ServiceWorkerManager.Events.RegistrationUpdated, this.recomputePageAuditability, this), |
| 36 | this._manager.addEventListener( |
| 37 | SDK.ServiceWorkerManager.Events.RegistrationDeleted, this.recomputePageAuditability, this), |
| 38 | ]; |
| 39 | |
| 40 | this.recomputePageAuditability(); |
| 41 | } |
| 42 | |
| 43 | /** |
| 44 | * @override |
| 45 | * @param {!SDK.ServiceWorkerManager} serviceWorkerManager |
| 46 | */ |
| 47 | modelRemoved(serviceWorkerManager) { |
| 48 | if (this._manager !== serviceWorkerManager) |
| 49 | return; |
| 50 | |
| 51 | Common.EventTarget.removeEventListeners(this._serviceWorkerListeners); |
| 52 | this._manager = null; |
| 53 | this.recomputePageAuditability(); |
| 54 | } |
| 55 | |
| 56 | /** |
| 57 | * @return {boolean} |
| 58 | */ |
| 59 | _hasActiveServiceWorker() { |
| 60 | if (!this._manager) |
| 61 | return false; |
| 62 | |
| 63 | const mainTarget = this._manager.target(); |
| 64 | if (!mainTarget) |
| 65 | return false; |
| 66 | |
| 67 | const inspectedURL = mainTarget.inspectedURL().asParsedURL(); |
| 68 | const inspectedOrigin = inspectedURL && inspectedURL.securityOrigin(); |
| 69 | for (const registration of this._manager.registrations().values()) { |
| 70 | if (registration.securityOrigin !== inspectedOrigin) |
| 71 | continue; |
| 72 | |
| 73 | for (const version of registration.versions.values()) { |
| 74 | if (version.controlledClients.length > 1) |
| 75 | return true; |
| 76 | } |
| 77 | } |
| 78 | |
| 79 | return false; |
| 80 | } |
| 81 | |
| 82 | /** |
| 83 | * @return {boolean} |
| 84 | */ |
| 85 | _hasAtLeastOneCategory() { |
| 86 | return Audits2.Presets.some(preset => preset.setting.get()); |
| 87 | } |
| 88 | |
| 89 | /** |
| 90 | * @return {?string} |
| 91 | */ |
| 92 | _unauditablePageMessage() { |
| 93 | if (!this._manager) |
| 94 | return null; |
| 95 | |
| 96 | const mainTarget = this._manager.target(); |
| 97 | const inspectedURL = mainTarget && mainTarget.inspectedURL(); |
| 98 | if (inspectedURL && !/^(http|chrome-extension)/.test(inspectedURL)) { |
| 99 | return Common.UIString( |
Lorne Mitchell | 7aa2c6c | 2019-04-03 03:50:10 +0000 | [diff] [blame] | 100 | 'Can only audit HTTP/HTTPS pages and Chrome extensions. Navigate to a different page to start an audit.'); |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 101 | } |
| 102 | |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 103 | return null; |
| 104 | } |
| 105 | |
| 106 | /** |
| 107 | * @return {!Promise<string>} |
| 108 | */ |
| 109 | async _evaluateInspectedURL() { |
| 110 | const mainTarget = this._manager.target(); |
| 111 | const runtimeModel = mainTarget.model(SDK.RuntimeModel); |
| 112 | const executionContext = runtimeModel && runtimeModel.defaultExecutionContext(); |
| 113 | let inspectedURL = mainTarget.inspectedURL(); |
| 114 | if (!executionContext) |
| 115 | return inspectedURL; |
| 116 | |
| 117 | // Evaluate location.href for a more specific URL than inspectedURL provides so that SPA hash navigation routes |
| 118 | // will be respected and audited. |
| 119 | try { |
| 120 | const result = await executionContext.evaluate( |
| 121 | { |
| 122 | expression: 'window.location.href', |
| 123 | objectGroup: 'audits', |
| 124 | includeCommandLineAPI: false, |
| 125 | silent: false, |
| 126 | returnByValue: true, |
| 127 | generatePreview: false |
| 128 | }, |
| 129 | /* userGesture */ false, /* awaitPromise */ false); |
| 130 | if (!result.exceptionDetails && result.object) { |
| 131 | inspectedURL = result.object.value; |
| 132 | result.object.release(); |
| 133 | } |
| 134 | } catch (err) { |
| 135 | console.error(err); |
| 136 | } |
| 137 | |
| 138 | return inspectedURL; |
| 139 | } |
| 140 | |
| 141 | /** |
| 142 | * @return {!Object} |
| 143 | */ |
| 144 | getFlags() { |
| 145 | const flags = {}; |
| 146 | for (const runtimeSetting of Audits2.RuntimeSettings) |
| 147 | runtimeSetting.setFlags(flags, runtimeSetting.setting.get()); |
| 148 | return flags; |
| 149 | } |
| 150 | |
| 151 | /** |
| 152 | * @return {!Array<string>} |
| 153 | */ |
| 154 | getCategoryIDs() { |
| 155 | const categoryIDs = []; |
| 156 | for (const preset of Audits2.Presets) { |
| 157 | if (preset.setting.get()) |
| 158 | categoryIDs.push(preset.configID); |
| 159 | } |
| 160 | return categoryIDs; |
| 161 | } |
| 162 | |
| 163 | /** |
| 164 | * @param {{force: boolean}=} options |
| 165 | * @return {!Promise<string>} |
| 166 | */ |
| 167 | async getInspectedURL(options) { |
| 168 | if (options && options.force || !this._inspectedURL) |
| 169 | this._inspectedURL = await this._evaluateInspectedURL(); |
| 170 | return this._inspectedURL; |
| 171 | } |
| 172 | |
| 173 | recomputePageAuditability() { |
| 174 | const hasActiveServiceWorker = this._hasActiveServiceWorker(); |
| 175 | const hasAtLeastOneCategory = this._hasAtLeastOneCategory(); |
| 176 | const unauditablePageMessage = this._unauditablePageMessage(); |
| 177 | |
| 178 | let helpText = ''; |
| 179 | if (hasActiveServiceWorker) { |
| 180 | helpText = Common.UIString( |
Lorne Mitchell | 7aa2c6c | 2019-04-03 03:50:10 +0000 | [diff] [blame] | 181 | 'Multiple tabs are being controlled by the same service worker. Close your other tabs on the same origin to audit this page.'); |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 182 | } else if (!hasAtLeastOneCategory) { |
| 183 | helpText = Common.UIString('At least one category must be selected.'); |
| 184 | } else if (unauditablePageMessage) { |
| 185 | helpText = unauditablePageMessage; |
| 186 | } |
| 187 | |
| 188 | this.dispatchEventToListeners(Audits2.Events.PageAuditabilityChanged, {helpText}); |
| 189 | } |
| 190 | }; |
| 191 | |
| 192 | |
Patrick Hulce | 05c18ce | 2018-05-24 00:34:56 +0000 | [diff] [blame] | 193 | /** @typedef {{setting: !Common.Setting, configID: string, title: string, description: string}} */ |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 194 | Audits2.Preset; |
| 195 | |
| 196 | /** @type {!Array.<!Audits2.Preset>} */ |
| 197 | Audits2.Presets = [ |
| 198 | // configID maps to Lighthouse's Object.keys(config.categories)[0] value |
| 199 | { |
| 200 | setting: Common.settings.createSetting('audits2.cat_perf', true), |
| 201 | configID: 'performance', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 202 | title: ls`Performance`, |
| 203 | description: ls`How long does this app take to show content and become usable` |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 204 | }, |
| 205 | { |
| 206 | setting: Common.settings.createSetting('audits2.cat_pwa', true), |
| 207 | configID: 'pwa', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 208 | title: ls`Progressive Web App`, |
| 209 | description: ls`Does this page meet the standard of a Progressive Web App` |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 210 | }, |
| 211 | { |
| 212 | setting: Common.settings.createSetting('audits2.cat_best_practices', true), |
| 213 | configID: 'best-practices', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 214 | title: ls`Best practices`, |
| 215 | description: ls`Does this page follow best practices for modern web development` |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 216 | }, |
| 217 | { |
| 218 | setting: Common.settings.createSetting('audits2.cat_a11y', true), |
| 219 | configID: 'accessibility', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 220 | title: ls`Accessibility`, |
| 221 | description: ls`Is this page usable by people with disabilities or impairments` |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 222 | }, |
| 223 | { |
| 224 | setting: Common.settings.createSetting('audits2.cat_seo', true), |
| 225 | configID: 'seo', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 226 | title: ls`SEO`, |
| 227 | description: ls`Is this page optimized for search engine results ranking` |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 228 | }, |
| 229 | ]; |
| 230 | |
Patrick Hulce | 05c18ce | 2018-05-24 00:34:56 +0000 | [diff] [blame] | 231 | /** @typedef {{setting: !Common.Setting, description: string, setFlags: function(!Object, string), options: (!Array|undefined), title: (string|undefined)}} */ |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 232 | Audits2.RuntimeSetting; |
| 233 | |
| 234 | /** @type {!Array.<!Audits2.RuntimeSetting>} */ |
| 235 | Audits2.RuntimeSettings = [ |
| 236 | { |
| 237 | setting: Common.settings.createSetting('audits2.device_type', 'mobile'), |
Patrick Hulce | 05c18ce | 2018-05-24 00:34:56 +0000 | [diff] [blame] | 238 | description: ls`Apply mobile emulation during auditing`, |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 239 | setFlags: (flags, value) => { |
| 240 | flags.disableDeviceEmulation = value === 'desktop'; |
| 241 | }, |
| 242 | options: [ |
Patrick Hulce | 05c18ce | 2018-05-24 00:34:56 +0000 | [diff] [blame] | 243 | {label: ls`Mobile`, value: 'mobile'}, |
| 244 | {label: ls`Desktop`, value: 'desktop'}, |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 245 | ], |
| 246 | }, |
| 247 | { |
| 248 | setting: Common.settings.createSetting('audits2.throttling', 'default'), |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 249 | setFlags: (flags, value) => { |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 250 | switch (value) { |
| 251 | case 'devtools': |
| 252 | flags.throttlingMethod = 'devtools'; |
| 253 | break; |
| 254 | case 'off': |
| 255 | flags.throttlingMethod = 'provided'; |
| 256 | break; |
| 257 | default: |
| 258 | flags.throttlingMethod = 'simulate'; |
| 259 | } |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 260 | }, |
| 261 | options: [ |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 262 | { |
cjamcl@google.com | bbf3f9b | 2019-04-19 19:25:18 +0000 | [diff] [blame] | 263 | label: ls`Simulated Slow 4G, 4x CPU Slowdown`, |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 264 | value: 'default', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 265 | title: ls`Throttling is simulated, resulting in faster audit runs with similar measurement accuracy` |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 266 | }, |
| 267 | { |
cjamcl@google.com | bbf3f9b | 2019-04-19 19:25:18 +0000 | [diff] [blame] | 268 | label: ls`Applied Slow 4G, 4x CPU Slowdown`, |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 269 | value: 'devtools', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 270 | title: ls`Typical DevTools throttling, with actual traffic shaping and CPU slowdown applied` |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 271 | }, |
| 272 | { |
| 273 | label: ls`No throttling`, |
| 274 | value: 'off', |
Christy Chen | 38dccb5 | 2019-05-08 22:32:15 +0000 | [diff] [blame^] | 275 | title: ls`No network or CPU throttling used. (Useful when not evaluating performance)` |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 276 | }, |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 277 | ], |
| 278 | }, |
| 279 | { |
Patrick Hulce | 05c18ce | 2018-05-24 00:34:56 +0000 | [diff] [blame] | 280 | setting: Common.settings.createSetting('audits2.clear_storage', true), |
| 281 | title: ls`Clear storage`, |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 282 | description: ls`Reset storage (localStorage, IndexedDB, etc) before auditing. (Good for performance & PWA testing)`, |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 283 | setFlags: (flags, value) => { |
Patrick Hulce | 05c18ce | 2018-05-24 00:34:56 +0000 | [diff] [blame] | 284 | flags.disableStorageReset = !value; |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 285 | }, |
Patrick Hulce | a087f62 | 2018-05-18 00:37:53 +0000 | [diff] [blame] | 286 | }, |
| 287 | ]; |
| 288 | |
Trent Apted | ba184a6 | 2018-05-25 02:13:48 +0000 | [diff] [blame] | 289 | Audits2.Events = { |
| 290 | PageAuditabilityChanged: Symbol('PageAuditabilityChanged'), |
| 291 | AuditProgressChanged: Symbol('AuditProgressChanged'), |
| 292 | RequestAuditStart: Symbol('RequestAuditStart'), |
| 293 | RequestAuditCancel: Symbol('RequestAuditCancel'), |
Paul Irish | 8f1e33d | 2018-05-31 02:29:50 +0000 | [diff] [blame] | 294 | }; |