Create-react-app: Disable inline images for images that are less than 10,000 bytes

Created on 10 Nov 2017  路  14Comments  路  Source: facebook/create-react-app

Thanks for this awesome project.

I recently get a warning during build telling me:

The bundle size is significantly larger than recommended.
Consider reducing it with code splitting: https://goo.gl/9VhYWB
You can also analyze the project dependencies: https://goo.gl/LeUzfb

Analyzing the project dependencies, I discovered that the easiest and most significant improvement would be to get rid of these small images under 10,000 bytes that are imported as Data URI by webpack (it's 25% of the size of my app).

I don't see an option to disable the feature. Does it exists? If no I have currently 2 options that are:

  • moving the images in the public folder
  • ejecting the app

Both options are a big loss so an option to change the current behaviour would be nice.

Thanks for your help

proposal

Most helpful comment

Another point to consider when using url-loader's inlining: small changes in either CRA's chosen threshold, or asset sizes in an app can mean a previously working Content-Security-Policy ends up blocking newly-inlined assets due to lack of data: in the img-src CSP policy directive.

For sites that have correct long-lived cache control headers (especially those remembering to use the immutable directive), and aren't as concerned about the initial cold cache case, IMO never inlining is preferable.

All 14 comments

Intriguing use case.
Even if this shaves off 25%, that means your bundle is still huge. If you split the application, the inlined images should also be split and distributed across the bundles.

Let's see what others have to say, but the third option is what I just mentioned.

edit: this option currently cannot be modified or disabled

Another point to consider when using url-loader's inlining: small changes in either CRA's chosen threshold, or asset sizes in an app can mean a previously working Content-Security-Policy ends up blocking newly-inlined assets due to lack of data: in the img-src CSP policy directive.

For sites that have correct long-lived cache control headers (especially those remembering to use the immutable directive), and aren't as concerned about the initial cold cache case, IMO never inlining is preferable.

Here's another attempt to solve this: https://github.com/facebook/create-react-app/pull/6060

It's similar to the idea of INLINE_RUNTIME_CHUNK. Now, running IMAGE_INLINE_SIZE_LIMIT=0 yarn run build will mean that every image in (src/*.(bmp|gif|jp?eg|png)) will be considered too big and not base64 into the JS.

Any progress on this proposal?

My PR is conflicting now but I'm not sure how to ping/pester it for review.

Adding my name to this because I don't want my images inlined!

Websites dealing with CSP will likely add data: in the CSP header because of the inline images (something like img-src 'self' data:;). MDN states that this is not recommanded and can be insecure (emphasis is mine):

data: Allows data: URIs to be used as a content source. This is insecure; an attacker can also inject arbitrary data: URIs. Use this sparingly and definitely not for scripts.


The alternative for CSP is using a nonce-based approach, but it has the following drawbacks:

  • way more complex to put in place, as it requires server-side modification of some static files (plus the documentation is very scarce);
  • the nonce passed to __webpack_nonce__ is not applied to img tags (this may be a bug);
  • browsers without CSP version 2 support will fallback to CSP version 1, so the data: will need to be present in the CSP header anyways for CSP version 1;
  • nonce-based approach are not well-known yet, which means less documentation, more error-prone, and users are likely to use the other, simplest, solution.

All in all, it would be way simpler to just have an option to disable inline images. #6060 gets my +1 :+1:

Messing with CSP just to bend-over-backward for this "flaw" in CRA isn't the right direction. I still think my pr should get reviewed and landed. And perhaps it should be changed to make the default minimum 0 bytes, meaning you have to deliberately set an env var if you want base64 inlining.

Hey! If you're here and also don't want base64 inlining you can...

  1. For images in the src/*.js don't do this:
import myImage from './foo.png';

<img src={myImage}>

Instead, do this:

const myImage = process.env.PUBLIC_URL + '/foo.png';

<img src={myImage}>
  1. In your src/App.js don't do this:
div.page { background: url(./bullet.png) no-repeat left center;

do this instead:

div.page { background: url(/static/bullet.png) no-repeat left center;

You'll also need to git mv src/foo.png public/static/images/ or whatever is equivalent in your project.

But I want the images loaded, checksummed, optimized and whatnot. I just don't want them inlined. In my use case long strings are problematic (using React to render multipart emails).

Hopefully #6060 gets the stamp of approval

Pile on reason: I realized a .png was being added as "data:" instead of using a path. At first I thought it was the file type because the same image as a .jpg worked. I don't want to have to add "data:" to my CSP as it's not recommended for security reasons. I'd like to be able to basically disable this feature all together and always use the image path.

Adding my name to this because I don't want my images inlined!

We're in the exact same position as @Kizmar, images are being converted to data: with base64, which our CSP is blocking. It's not advised to allow data: URIs as content source, as MDN says:

data: Allows data: URIs to be used as a content source. This is insecure; an attacker can also inject arbitrary data: URIs. Use this sparingly and definitely not for scripts.

Ideally there would be a solution that enables us to keep our sites secure whilst still utilising the standard image processing (just without the base64 step).

Was this page helpful?
0 / 5 - 0 ratings

Related issues

ap13p picture ap13p  路  3Comments

wereHamster picture wereHamster  路  3Comments

alleroux picture alleroux  路  3Comments

adrice727 picture adrice727  路  3Comments

fson picture fson  路  3Comments