Skip to content
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

Merged
merged 11 commits into from
Jan 25, 2017
Merged

DBW: Responsive Images Audit #1497

merged 11 commits into from
Jan 25, 2017

Conversation

patrickhulce
Copy link
Collaborator

R: all

addresses #978 #876

@patrickhulce
Copy link
Collaborator Author

Waiting to add tests for feedback on the audit mechanism. Right now as described in the fileoverview...

  *   Checks to see if the images used on the page are larger than
  *   their display sizes. The audit will list all images that are larger than
  *   their display size regardless of DPR (a 1000px wide image displayed as a
  *   500px high-res image on a Retina display will show up as 75% unused);
  *   however, the audit will only fail pages that use images that have waste
  *   when computed with DPR taken into account.

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.

Example Mobile Result
image

Example Desktop Result of Same Site
image

@brendankenny
Copy link
Member

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.

@brendankenny
Copy link
Member

you don't necessarily want a 1:1 image pixel to display pixel ratio on mobile devices as it's just a waste

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)

@patrickhulce
Copy link
Collaborator Author

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 :)

@brendankenny
Copy link
Member

severity of audits rather than pass/fail

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

Copy link
Member

@brendankenny brendankenny left a 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) {
Copy link
Member

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?

Copy link
Collaborator Author

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);
Copy link
Member

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

Copy link
Member

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 :)

Copy link
Collaborator Author

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));
Copy link
Member

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

Copy link
Collaborator Author

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`;
Copy link
Member

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?

Copy link
Collaborator Author

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
Copy link
Member

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?

Copy link
Collaborator Author

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>}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

need ! before Promise

Copy link
Collaborator Author

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) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

element

Copy link
Collaborator Author

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);
Copy link
Member

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?

Copy link
Collaborator Author

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);
Copy link
Member

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

Copy link
Collaborator Author

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 => {
Copy link
Member

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?

Copy link
Collaborator Author

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

Copy link
Member

@brendankenny brendankenny Jan 20, 2017

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. ' +
Copy link
Contributor

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"?

Copy link
Collaborator Author

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.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"device display size"?

Copy link
Collaborator Author

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;
Copy link
Contributor

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);
Copy link
Contributor

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?

Copy link
Collaborator Author

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.

Copy link
Contributor

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

Copy link
Collaborator Author

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');
Copy link
Contributor

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.

Copy link
Collaborator Author

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) {
Copy link
Contributor

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

Copy link
Collaborator Author

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,
Copy link
Contributor

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?

Copy link
Collaborator Author

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);
Copy link
Contributor

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

Copy link
Collaborator Author

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`;
Copy link
Contributor

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?

Copy link
Contributor

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...?

Copy link
Collaborator Author

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.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yea, potential savings SGTM

@ebidel
Copy link
Contributor

ebidel commented Jan 21, 2017

Complex stuff. Seems like a nice approach.

#978 also mention css image-set() and background images. I feel that adding CSS stuff to the audit might be a bit much. WDYT?

@patrickhulce
Copy link
Collaborator Author

patrickhulce commented Jan 23, 2017

#978 also mention css image-set() and background images. I feel that adding CSS stuff to the audit might be a bit much. WDYT?

@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;
Copy link
Contributor

@ebidel ebidel Jan 23, 2017

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.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

Copy link
Member

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

Copy link
Member

@brendankenny brendankenny left a 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
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

issue for this?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Member

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

Copy link
Collaborator Author

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;
Copy link
Member

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)?

Copy link
Collaborator Author

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}`);
Copy link
Member

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

Copy link
Collaborator Author

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,
Copy link
Member

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),
Copy link
Member

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

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

@brendankenny
Copy link
Member

I agree on postponing and we can just leave that issue open for now.

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?

@ebidel ebidel mentioned this pull request Jan 24, 2017
2 tasks
@patrickhulce
Copy link
Collaborator Author

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)

@patrickhulce
Copy link
Collaborator Author

@brendankenny PTAL :)

Copy link
Member

@brendankenny brendankenny left a 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="
Copy link
Member

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?

Copy link
Collaborator Author

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
Copy link
Member

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

Copy link
Member

@brendankenny brendankenny left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🖼↕️ LGTM!
↔️

@brendankenny brendankenny merged commit bf964de into master Jan 25, 2017
@brendankenny brendankenny deleted the image_usage branch January 25, 2017 00:15
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants