-
Notifications
You must be signed in to change notification settings - Fork 9.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
DBW: Responsive Images Audit #1497
Conversation
Waiting to add tests for feedback on the audit mechanism. Right now as described in the fileoverview...
This gives the somewhat confusing result though that a page can easily pass with mobile emulation on and fail on desktop even though the mobile violations are more impactful to the user. |
I know there's also been advice that you don't necessarily want a 1:1 image pixel to display pixel ratio on mobile devices as it's just a waste, and really you want some notion of angular resolution of those pixels, but I have no idea of what the state of the art is on that. |
e.g. ratio of 1 may be too low, 2 could work well, but 2.6 or whatever would be wasted pixels as the average visitor won't hold the phone up close to look at most pictures (however you want to measure that) |
Yeah I'd just generally prefer Lighthouse to stay out of these judgement calls when possible. IMO if there's a legitimate reason for acting a particular way that isn't very unpredictable LH should try not to fail you for it but just make sure you're aware of it. This does get into the area of something I'd really like to see across the board though: severity of audits rather than pass/fail. Feels like another conversation though :) |
well we do have support for 0-100 scores, we just haven't used them in a while :) Defining how severe is severe is hard, of course |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looking good. While we figure out scoring and what constitutes waste, some initial comments, mostly around style
const usedRatio = usedPixels / actualPixels; | ||
const usedRatioFullDPR = usedPixelsFullDPR / actualPixels; | ||
|
||
if (!usedRatio || usedRatio >= 1) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
what is !usedRatio
protecting against here? NaN
? May want to use !isFinite(usedRatio)
instead so it handles Infinity
as well. Both seem like an error condition, though (as opposed to usedRatio >= 1
), so it may warrant some kind of error string instead?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah will better handle the edge cases, originally this was for an image that was never shown (0 used pixels) but that is now handled by checking for the network record
|
||
const size = image.networkRecord.resourceSize; | ||
const transferTimeInMs = 1000 * (image.networkRecord.endTime - | ||
image.networkRecord.startTime); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
better to use responseReceivedTime
instead of startTime
since connection overhead won't change much compared to transmission time alone
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this was actually meant to be a question; I don't actually know that it's better to do this :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yeah that's a good point! also should decide what to do about handling data URI images since the "response time" will be nothing but they just impacted another assets response time
const size = image.networkRecord.resourceSize; | ||
const transferTimeInMs = 1000 * (image.networkRecord.endTime - | ||
image.networkRecord.startTime); | ||
const wastedBytes = Math.round(size * (1 - usedRatio)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
maybe introduce a unusedRatio
or wastedRatio
intermediate variable for 1 - usedRatio
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
const wastedBytes = Math.round(size * (1 - usedRatio)); | ||
const wastedTime = Math.round(transferTimeInMs * (1 - usedRatio)); | ||
const percentSavings = Math.round(100 * (1 - usedRatio)); | ||
const label = `${Math.round(size / KB_IN_BYTES)}KB total, ${percentSavings}% savings`; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this is a clever solution, though it's obviously an estimate. I would imagine it's a better estimate for a block-based image format vs something like PNG? Have you considered your canvas solution (from the image optimization audit) instead of/in addition to this?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah just an estimate, but the impact of the differences is likely to be much more pronounced at the smaller size end of the spectrum where it's less likely to matter anyway. That seemed like overkill for the first pass, but I'm open to sharing that little gem of logic :)
|
||
/** | ||
* @param {!Object} image | ||
* @param {number} DPR |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
maybe @param {number} DPR devicePixelRatio
to make this param clearer on first read?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
|
||
/** | ||
* @param {!{src: string}} tag | ||
* @return {Promise<!Object>} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
need !
before Promise
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
* @param {!{src: string}} tag | ||
* @return {Promise<!Object>} | ||
*/ | ||
fetchTagWithSizeInformation(tag) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
element
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
* @return {Promise<!Object>} | ||
*/ | ||
fetchTagWithSizeInformation(tag) { | ||
const url = JSON.stringify(tag.src); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
stringify for escaped encoding or...? better to parse with new URL()
and reject bad ones?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
just as replacement for '${element.src}'
, but will change. they all should be valid urls since they came from img.currentSrc
// fill in natural size information if we can | ||
return tag.isPicture && tag.networkRecord ? | ||
this.fetchTagWithSizeInformation(tag) : | ||
Promise.resolve(tag); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
can just return tag
(without the promise wrapper) if using Promise.all
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yeah will do
|
||
return driver.evaluateAsync(`(${collectImageTagInfo.toString()})()`) | ||
.then(tags => { | ||
return Promise.all(tags.map(tag => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Promise.all
makes me nervous when interacting with the debugger protocol. Right now driver.evaluateAsync
is just the one command internally, but if we change that at some point, Promise.all
could end up interleaving commands. Usually better to go serial. OTOH, that sets the network requests off one at a time, much slower than necessary. Maybe ship the array of images to investigate off to the browser in one evaluateAsync
call and put the loop over them in there?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
will serialize, there shouldn't be many network requests happening here and if there end up being many we should revisit
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
there shouldn't be many network requests happening here and if there end up being many we should revisit
yeah, for the requests I'm mostly worried about the cache behavior on devices with less storage. It's possible that for an image on a page the decoded pixel data is still present but the image file has been evicted from the network cache, so img.src = url
may force another download. We'll have to investigate to find out if that is possible.
category: 'Images', | ||
name: 'uses-responsive-images', | ||
description: 'Site uses appropriate image sizes', | ||
helpText: 'Image sizes served should be based on the display size to save network bytes. ' + |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm wrestling with display size. "viewport size"?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well not just viewport size but how large the image is when displayed after taking into account DPR. i.e. if the viewport is huge but the image fills 1/4 of the width of the viewport and DPR is 2, target 1/2 of the viewport size.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"device display size"?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
sgtm
*/ | ||
static computeWaste(image, DPR) { | ||
const url = URL.getDisplayName(image.src); | ||
const actualPixels = image.naturalWidth * image.naturalHeight; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
TIL about naturalWidth
and naturalHeight
:)
const transferTimeInMs = 1000 * (image.networkRecord.endTime - | ||
image.networkRecord.responseReceivedTime); | ||
const wastedBytes = Math.round(size * wastedRatio); | ||
const wastedTime = Math.round(transferTimeInMs * wastedRatio); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
how confident are we that this relationship of wastedBytes
-> wastedTime
is correct?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
More confident than the wastedPixels
-> wastedBytes
relationship! Do you have anything in particular you're thinking of? This somewhat assumes keep-alive and the congestion window is already flooded by the time we hit these images.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nothing in particular. Just thinking out loud. Maybe the final label should be clear that this is an estimate? https://github.com/GoogleChrome/lighthouse/pull/1497/files/b4fe416f62b776844a9a743864921256f01dc86a#diff-31529c3403e44b3ff053b53ab95cca7dR118
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done!
const entries = srcset.split(','); | ||
const relativeUrls = entries.map(entry => entry.trim().split(' ')[0]); | ||
return relativeUrls.map(url => { | ||
const a = document.createElement('a'); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice trick to get full urls. Any value in usingnew URL()
with the 2nd base param instead of creating dom? I guess what you have also works if there's <base>
tag on the page.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Didn't know about the 2nd base param! I'll switch to that.
}); | ||
} | ||
|
||
function getElementInfo(element) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
doc element as a HTMLImageElement
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
will doc
function getElementInfo(element) { | ||
return { | ||
tagName: element.tagName, | ||
src: element.currentSrc, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
can you add comment for later that currentSrc
is used b/c it gives you the chosen media src that the browser selected based on the display?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
return Object.assign(getElementInfo(element), {isPicture: false}); | ||
} | ||
|
||
const imgElementInfo = getElementInfo(element); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you could move imgElementInfo
above and reuse it in the first if
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
const wastedBytes = Math.round(size * wastedRatio); | ||
const wastedTime = Math.round(transferTimeInMs * wastedRatio); | ||
const percentSavings = Math.round(100 * wastedRatio); | ||
const label = `${Math.round(size / KB_IN_BYTES)}KB total, ${percentSavings}% savings`; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I found x% savings
kinda confusing in the screenshot. Maybe % waste
would be a better thing to report to the user?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Or somehow communicating that this is the % savings if the correct size was used...?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How about "potential savings" as in the previous audits? I was trying to find ways to make these seem more encouraging to users rather than scolding since they can be a bit harsh. If you've done everything imaginable in the realm of responsive images and you're over by like 10%, it felt mean to say 10% waste
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yea, potential savings SGTM
Complex stuff. Seems like a nice approach. #978 also mention css |
@ebidel yeah I struggled more with those since it's much less obvious how to get their actual size used and they're much more likely to be used as sprites which will intentionally be too big for their display size. I agree on postponing and we can just leave that issue open for now. |
a.href = url; | ||
return a.href; | ||
try { | ||
return new URL(url, window.location.href).href; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since you're running this on the page, if the site uses a <base href="http://somethingelse.com/">
, new URL(url, window.location.href).href;
might not be correct. e.g. http://jsbin.com/yilakeqeyi/edit?html,output
return new URL(url, document.baseURI || window.location.href).href;
should do it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
document.baseURI || window.location.href
drive-by nit: if there isn't a <base>
element then these should be equal, and document.baseURI
will always be defined anyways, so return new URL(url, document.baseURI).href
should be sufficient
3272942
to
4f77f2d
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
looking good. A few last things.
Should also have a smoke test for this. If you don't want to add it to the DBW test, the offline-ready test runs the full default config so it runs all the audits. The offline-ready page has an image in it, so it'll be able to assert something about it (could also test the unoptimized images audit output in there as well).
return null; | ||
} | ||
|
||
// TODO(phulce): use an average transfer time for data URI images |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
issue for this?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
add the issue number to the TODO
so we'll have the cross reference
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
if (!processed) { | ||
return results; | ||
} else if (processed instanceof Error) { | ||
debugString = processed.message; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
purposefully only using the last error (overwriting each time)?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah that seemed to be the pattern followed a few other places
const wastedRatioFullDPR = 1 - (usedPixelsFullDPR / actualPixels); | ||
|
||
if (!Number.isFinite(wastedRatio)) { | ||
return new Error(`Invalid image sizing information ${url}`); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
worth including size info for help debugging? could only be naturalWidth
, naturalHeight
, clientWidth
, or clientHeight
as the culprits.
OTOH this may never really happen
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah this theoretically should never happen and was just for catching a bug I had lower down.
return Promise.resolve({ | ||
scrollWidth: window.innerWidth, | ||
viewportWidth: window.outerWidth | ||
viewportWidth: window.outerWidth, | ||
devicePixelRatio: window.devicePixelRatio, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
no, sounds good for now. If we add another measure here we can change it :)
return Object.assign(imgElementInfo, { | ||
isPicture: true, | ||
// nested chain is too deep for DevTools to handle so stringify | ||
sources: JSON.stringify(sources), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yeah, maybe remove if we don't use it
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
Would you mind adding a checklist or whatever of possible improvements (and the TODO you added in this PR and anything else you've thought about while writing this) at the top of that issue so we have a central place for tracking? |
Done but split out the TODO into a new issue since it isn't really responsive image specific (applies to all byte efficiency audits) |
@brendankenny PTAL :) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
two last things :)
@@ -113,7 +113,16 @@ | |||
</template> | |||
|
|||
<template id="unoptimized-images-tmpl"> | |||
<img src=" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
does this warning show up in the extendedInfo
and so can be checked?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes it's the 2nd in length: 2
return null; | ||
} | ||
|
||
// TODO(phulce): use an average transfer time for data URI images |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
add the issue number to the TODO
so we'll have the cross reference
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🖼
R: all
addresses #978 #876