123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848 |
- <!doctype html><!-- This is a valid HTML5 document. -->
- <!-- Screen readers, SEO, extensions and so on. -->
- <html lang=fr>
- <!-- Has to be within the first 1024 bytes, hence before the <title>
- See: https://www.w3.org/TR/2012/CR-html5-20121217/document-metadata.html#charset -->
- <meta charset=utf-8>
- <!-- Why no `X-UA-Compatible` meta: https://stackoverflow.com/a/6771584 -->
- <!-- The viewport meta is quite crowded and we are responsible for that.
- See: https://codepen.io/tigt/post/meta-viewport-for-2015 -->
- <meta name=viewport content="width=device-width,minimum-scale=1,initial-scale=1,shrink-to-fit=no">
- <!-- Required to make a valid HTML5 document. -->
- <title>Self-Host Your Static Assets (archive) — David Larlet</title>
- <!-- Generated from https://realfavicongenerator.net/ such a mess. -->
- <link rel="apple-touch-icon" sizes="180x180" href="/static/david/icons/apple-touch-icon.png">
- <link rel="icon" type="image/png" sizes="32x32" href="/static/david/icons/favicon-32x32.png">
- <link rel="icon" type="image/png" sizes="16x16" href="/static/david/icons/favicon-16x16.png">
- <link rel="manifest" href="/manifest.json">
- <link rel="mask-icon" href="/static/david/icons/safari-pinned-tab.svg" color="#5bbad5">
- <link rel="shortcut icon" href="/static/david/icons/favicon.ico">
- <meta name="apple-mobile-web-app-title" content="David Larlet">
- <meta name="application-name" content="David Larlet">
- <meta name="msapplication-TileColor" content="#da532c">
- <meta name="msapplication-config" content="/static/david/icons/browserconfig.xml">
- <meta name="theme-color" content="#f0f0ea">
- <!-- That good ol' feed, subscribe :p. -->
- <link rel=alternate type="application/atom+xml" title=Feed href="/david/log/">
-
- <meta name="robots" content="noindex, nofollow">
- <meta content="origin-when-cross-origin" name="referrer">
- <!-- Canonical URL for SEO purposes -->
- <link rel="canonical" href="https://csswizardry.com/2019/05/self-host-your-static-assets/">
-
- <style>
- /* http://meyerweb.com/eric/tools/css/reset/ */
- html, body, div, span,
- h1, h2, h3, h4, h5, h6, p, blockquote, pre,
- a, abbr, address, big, cite, code,
- del, dfn, em, img, ins,
- small, strike, strong, tt, var,
- dl, dt, dd, ol, ul, li,
- fieldset, form, label, legend,
- table, caption, tbody, tfoot, thead, tr, th, td,
- article, aside, canvas, details, embed,
- figure, figcaption, footer, header, hgroup,
- menu, nav, output, ruby, section, summary,
- time, mark, audio, video {
- margin: 0;
- padding: 0;
- border: 0;
- font-size: 100%;
- font: inherit;
- vertical-align: baseline;
- }
- /* HTML5 display-role reset for older browsers */
- article, aside, details, figcaption, figure,
- footer, header, hgroup, menu, nav, section { display: block; }
- body { line-height: 1; }
- blockquote, q { quotes: none; }
- blockquote:before, blockquote:after,
- q:before, q:after {
- content: '';
- content: none;
- }
- table {
- border-collapse: collapse;
- border-spacing: 0;
- }
-
- /* http://practicaltypography.com/equity.html */
- /* https://calendar.perfplanet.com/2016/no-font-face-bulletproof-syntax/ */
- /* https://www.filamentgroup.com/lab/js-web-fonts.html */
- @font-face {
- font-family: 'EquityTextB';
- src: url('/static/david/css/fonts/Equity-Text-B-Regular-webfont.woff2') format('woff2'),
- url('/static/david/css/fonts/Equity-Text-B-Regular-webfont.woff') format('woff');
- font-weight: 300;
- font-style: normal;
- font-display: swap;
- }
- @font-face {
- font-family: 'EquityTextB';
- src: url('/static/david/css/fonts/Equity-Text-B-Italic-webfont.woff2') format('woff2'),
- url('/static/david/css/fonts/Equity-Text-B-Italic-webfont.woff') format('woff');
- font-weight: 300;
- font-style: italic;
- font-display: swap;
- }
- @font-face {
- font-family: 'EquityTextB';
- src: url('/static/david/css/fonts/Equity-Text-B-Bold-webfont.woff2') format('woff2'),
- url('/static/david/css/fonts/Equity-Text-B-Bold-webfont.woff') format('woff');
- font-weight: 700;
- font-style: normal;
- font-display: swap;
- }
-
- @font-face {
- font-family: 'ConcourseT3';
- src: url('/static/david/css/fonts/concourse_t3_regular-webfont-20190806.woff2') format('woff2'),
- url('/static/david/css/fonts/concourse_t3_regular-webfont-20190806.woff') format('woff');
- font-weight: 300;
- font-style: normal;
- font-display: swap;
- }
-
-
- /* http://practice.typekit.com/lesson/caring-about-opentype-features/ */
- body {
- /* http://www.cssfontstack.com/ Palatino 99% Win 86% Mac */
- font-family: "EquityTextB", Palatino, serif;
- background-color: #f0f0ea;
- color: #07486c;
- font-kerning: normal;
- -moz-osx-font-smoothing: grayscale;
- -webkit-font-smoothing: subpixel-antialiased;
- text-rendering: optimizeLegibility;
- font-variant-ligatures: common-ligatures contextual;
- font-feature-settings: "kern", "liga", "clig", "calt";
- }
- pre, code, kbd, samp, var, tt {
- font-family: 'TriplicateT4c', monospace;
- }
- em {
- font-style: italic;
- color: #323a45;
- }
- strong {
- font-weight: bold;
- color: black;
- }
- nav {
- background-color: #323a45;
- color: #f0f0ea;
- display: flex;
- justify-content: space-around;
- padding: 1rem .5rem;
- }
- nav:last-child {
- border-bottom: 1vh solid #2d7474;
- }
- nav a {
- color: #f0f0ea;
- }
- nav abbr {
- border-bottom: 1px dotted white;
- }
-
- h1 {
- border-top: 1vh solid #2d7474;
- border-bottom: .2vh dotted #2d7474;
- background-color: #e3e1e1;
- color: #323a45;
- text-align: center;
- padding: 5rem 0 4rem 0;
- width: 100%;
- font-family: 'ConcourseT3';
- display: flex;
- flex-direction: column;
- }
- h1.single {
- padding-bottom: 10rem;
- }
- h1 span {
- position: absolute;
- top: 1vh;
- left: 20%;
- line-height: 0;
- }
- h1 span a {
- line-height: 1.7;
- padding: 1rem 1.2rem .6rem 1.2rem;
- border-radius: 0 0 6% 6%;
- background: #2d7474;
- font-size: 1.3rem;
- color: white;
- text-decoration: none;
- }
- h2 {
- margin: 4rem 0 1rem;
- border-top: .2vh solid #2d7474;
- padding-top: 1vh;
- }
- h3 {
- text-align: center;
- margin: 3rem 0 .75em;
- }
- hr {
- height: .4rem;
- width: .4rem;
- border-radius: .4rem;
- background: #07486c;
- margin: 2.5rem auto;
- }
- time {
- display: bloc;
- margin-left: 0 !important;
- }
- ul, ol {
- margin: 2rem;
- }
- ul {
- list-style-type: square;
- }
- a {
- text-decoration-skip-ink: auto;
- text-decoration-thickness: 0.05em;
- text-underline-offset: 0.09em;
- }
- article {
- max-width: 50rem;
- display: flex;
- flex-direction: column;
- margin: 2rem auto;
- }
- article.single {
- border-top: .2vh dotted #2d7474;
- margin: -6rem auto 1rem auto;
- background: #f0f0ea;
- padding: 2rem;
- }
- article p:last-child {
- margin-bottom: 1rem;
- }
- p {
- padding: 0 .5rem;
- margin-left: 3rem;
- }
- p + p,
- figure + p {
- margin-top: 2rem;
- }
-
- blockquote {
- background-color: #e3e1e1;
- border-left: .5vw solid #2d7474;
- display: flex;
- flex-direction: column;
- align-items: center;
- padding: 1rem;
- margin: 1.5rem;
- }
- blockquote cite {
- font-style: italic;
- }
- blockquote p {
- margin-left: 0;
- }
-
- figure {
- border-top: .2vh solid #2d7474;
- background-color: #e3e1e1;
- text-align: center;
- padding: 1.5rem 0;
- margin: 1rem 0 0;
- font-size: 1.5rem;
- width: 100%;
- }
- figure img {
- max-width: 250px;
- max-height: 250px;
- border: .5vw solid #323a45;
- padding: 1px;
- }
- figcaption {
- padding: 1rem;
- line-height: 1.4;
- }
- aside {
- display: flex;
- flex-direction: column;
- background-color: #e3e1e1;
- padding: 1rem 0;
- border-bottom: .2vh solid #07486c;
- }
- aside p {
- max-width: 50rem;
- margin: 0 auto;
- }
-
- /* https://fvsch.com/code/css-locks/ */
- p, li, pre, code, kbd, samp, var, tt, time, details, figcaption {
- font-size: 1rem;
- line-height: calc( 1.5em + 0.2 * 1rem );
- }
- h1 {
- font-size: 1.9rem;
- line-height: calc( 1.2em + 0.2 * 1rem );
- }
- h2 {
- font-size: 1.6rem;
- line-height: calc( 1.3em + 0.2 * 1rem );
- }
- h3 {
- font-size: 1.35rem;
- line-height: calc( 1.4em + 0.2 * 1rem );
- }
- @media (min-width: 20em) {
- /* The (100vw - 20rem) / (50 - 20) part
- resolves to 0-1rem, depending on the
- viewport width (between 20em and 50em). */
- p, li, pre, code, kbd, samp, var, tt, time, details, figcaption {
- font-size: calc( 1rem + .6 * (100vw - 20rem) / (50 - 20) );
- line-height: calc( 1.5em + 0.2 * (100vw - 50rem) / (20 - 50) );
- margin-left: 0;
- }
- h1 {
- font-size: calc( 1.9rem + 1.5 * (100vw - 20rem) / (50 - 20) );
- line-height: calc( 1.2em + 0.2 * (100vw - 50rem) / (20 - 50) );
- }
- h2 {
- font-size: calc( 1.5rem + 1.5 * (100vw - 20rem) / (50 - 20) );
- line-height: calc( 1.3em + 0.2 * (100vw - 50rem) / (20 - 50) );
- }
- h3 {
- font-size: calc( 1.35rem + 1.5 * (100vw - 20rem) / (50 - 20) );
- line-height: calc( 1.4em + 0.2 * (100vw - 50rem) / (20 - 50) );
- }
- }
- @media (min-width: 50em) {
- /* The right part of the addition *must* be a
- rem value. In this example we *could* change
- the whole declaration to font-size:2.5rem,
- but if our baseline value was not expressed
- in rem we would have to use calc. */
- p, li, pre, code, kbd, samp, var, tt, time, details, figcaption {
- font-size: calc( 1rem + .6 * 1rem );
- line-height: 1.5em;
- }
- p, li, pre, details {
- margin-left: 3rem;
- }
- h1 {
- font-size: calc( 1.9rem + 1.5 * 1rem );
- line-height: 1.2em;
- }
- h2 {
- font-size: calc( 1.5rem + 1.5 * 1rem );
- line-height: 1.3em;
- }
- h3 {
- font-size: calc( 1.35rem + 1.5 * 1rem );
- line-height: 1.4em;
- }
- figure img {
- max-width: 500px;
- max-height: 500px;
- }
- }
-
- figure.unsquared {
- margin-bottom: 1.5rem;
- }
- figure.unsquared img {
- height: inherit;
- }
-
-
-
- @media print {
- body { font-size: 100%; }
- a:after { content: " (" attr(href) ")"; }
- a, a:link, a:visited, a:after {
- text-decoration: underline;
- text-shadow: none !important;
- background-image: none !important;
- background: white;
- color: black;
- }
- abbr[title] { border-bottom: 0; }
- abbr[title]:after { content: " (" attr(title) ")"; }
- img { page-break-inside: avoid; }
- @page { margin: 2cm .5cm; }
- h1, h2, h3 { page-break-after: avoid; }
- p3 { orphans: 3; widows: 3; }
- img {
- max-width: 250px !important;
- max-height: 250px !important;
- }
- nav, aside { display: none; }
- }
-
- ul.with_columns {
- column-count: 1;
- }
- @media (min-width: 20em) {
- ul.with_columns {
- column-count: 2;
- }
- }
- @media (min-width: 50em) {
- ul.with_columns {
- column-count: 3;
- }
- }
- ul.with_two_columns {
- column-count: 1;
- }
- @media (min-width: 20em) {
- ul.with_two_columns {
- column-count: 1;
- }
- }
- @media (min-width: 50em) {
- ul.with_two_columns {
- column-count: 2;
- }
- }
-
- .gallery {
- display: flex;
- flex-wrap: wrap;
- justify-content: space-around;
- }
- .gallery figure img {
- margin-left: 1rem;
- margin-right: 1rem;
- }
- .gallery figure figcaption {
- font-family: 'ConcourseT3'
- }
-
- footer {
- font-family: 'ConcourseT3';
- display: flex;
- flex-direction: column;
- border-top: 3px solid white;
- padding: 4rem 0;
- background-color: #07486c;
- color: white;
- }
- footer > * {
- max-width: 50rem;
- margin: 0 auto;
- }
- footer a {
- color: #f1c40f;
- }
- footer .avatar {
- width: 200px;
- height: 200px;
- border-radius: 50%;
- float: left;
- -webkit-shape-outside: circle();
- shape-outside: circle();
- margin-right: 2rem;
- padding: 2px 5px 5px 2px;
- background: white;
- border-left: 1px solid #f1c40f;
- border-top: 1px solid #f1c40f;
- border-right: 5px solid #f1c40f;
- border-bottom: 5px solid #f1c40f;
- }
- </style>
-
- <h1>
- <span><a id="jumper" href="#jumpto" title="Un peu perdu ?">?</a></span>
- Self-Host Your Static Assets (archive)
- <time>Pour la pérennité des contenus liés. Non-indexé, retrait sur simple email.</time>
- </h1>
- <section>
- <article>
- <h3><a href="https://csswizardry.com/2019/05/self-host-your-static-assets/">Source originale du contenu</a></h3>
- <details>
- <summary>Table of Contents</summary>
- <ol>
- <li><a href="#what-am-i-talking-about">What Am I Talking About?</a></li>
- <li><a href="#risk-slowdowns-and-outages">Risk: Slowdowns and Outages</a></li>
- <li><a href="#risk-service-shutdowns">Risk: Service Shutdowns</a></li>
- <li><a href="#risk-security-vulnerabilities">Risk: Security Vulnerabilities</a>
- <ol>
- <li><a href="#mitigation-subresource-integrity">Mitigation: Subresource Integrity</a></li>
- </ol>
- </li>
- <li><a href="#penalty-network-negotiation">Penalty: Network Negotiation</a>
- <ol>
- <li><a href="#mitigation-preconnect">Mitigation: <code class="highlighter-rouge">preconnect</code></a></li>
- </ol>
- </li>
- <li><a href="#penalty-loss-of-prioritisation">Penalty: Loss of Prioritisation</a></li>
- <li><a href="#penalty-caching">Penalty: Caching</a></li>
- <li><a href="#myth-cross-domain-caching">Myth: Cross-Domain Caching</a></li>
- <li><a href="#myth-access-to-a-cdn">Myth: Access to a CDN</a></li>
- <li><a href="#self-host-your-static-assets">Self-Host Your Static Assets</a></li>
- </ol>
- </details>
-
- <p>One of the quickest wins—and one of the first things I recommend my clients
- do—to make websites faster can at first seem counter-intuitive: you should
- self-host all of your static assets, forgoing others’ CDNs/infrastructure. In
- this short and hopefully very straightforward post, I want to outline the
- disadvantages of hosting your static assets ‘off-site’, and the overwhelming
- benefits of hosting them on your own origin.</p>
-
- <h2 id="what-am-i-talking-about">What Am I Talking About?</h2>
-
- <p>It’s not uncommon for developers to link to static assets such as libraries or
- plugins that are hosted at a public/CDN URL. A classic example is jQuery, that
- we might link to like so:</p>
-
- <div class="highlighter-rouge"><div class="highlight"><pre class="highlight"><code><script src="https://code.jquery.com/jquery-3.3.1.slim.min.js"></script>
- </code></pre></div></div>
-
- <p>There are a number of perceived benefits to doing this, but my aim later in this
- article is to either debunk these claims, or show how other costs vastly
- outweigh them.</p>
-
- <ul>
- <li><strong>It’s convenient.</strong> It requires very little effort or brainpower to include
- files like this. Copy and paste a line of HTML and you’re done. Easy.</li>
- <li><strong>We get access to a CDN.</strong> <code class="highlighter-rouge">code.jquery.com</code> is served by
- <a href="https://www.stackpath.com/products/cdn/">StackPath</a>, a CDN. By linking to
- assets on this origin, we get CDN-quality delivery, free!</li>
- <li><strong>Users might already have the file cached.</strong> If <code class="highlighter-rouge">website-a.com</code> links to
- <code class="highlighter-rouge">https://code.jquery.com/jquery-3.3.1.slim.min.js</code>, and a user goes from there
- to <code class="highlighter-rouge">website-b.com</code> who also links to
- <code class="highlighter-rouge">https://code.jquery.com/jquery-3.3.1.slim.min.js</code>, then the user will already
- have that file in their cache.</li>
- </ul>
-
- <h2 id="risk-slowdowns-and-outages">Risk: Slowdowns and Outages</h2>
-
- <p>I won’t go into too much detail in this post, because I have a <a href="https://csswizardry.com/2017/07/performance-and-resilience-stress-testing-third-parties/">whole
- article</a>
- on the subject of third party resilience and the risks associated with slowdowns
- and outages. Suffice to say, if you have any critical assets served by third
- party providers, and that provider is suffering slowdowns or, heaven forbid,
- outages, it’s pretty bleak news for you. You’re going to suffer, too.</p>
-
- <p>If you have any render-blocking CSS or synchronous JS hosted on third party
- domains, go and bring it onto your own infrastructure <em>right now</em>. Critical
- assets are far too valuable to leave on someone else’s servers.</p>
-
- <h2 id="risk-service-shutdowns">Risk: Service Shutdowns</h2>
-
- <p>A far less common occurrence, but what happens if a provider decides they need
- to shut down the service? This is exactly what <a href="https://rawgit.com">Rawgit</a> did
- in October 2018, yet (at the time of writing) a crude GitHub code search still
- yielded <a href="https://github.com/search?q=rawgit&type=Code">over a million
- references</a> to the now-sunset
- service, and almost 20,000 live sites are still linking to it!</p>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/big-query-rawgit.jpg" alt=""/>
- <figcaption>Many thanks to <a href="https://twitter.com/paulcalvano">Paul
- Calvano</a> who very kindly <a href="https://bigquery.cloud.google.com/savedquery/226352634162:7c27aa5bac804a6687f58db792c021ee">queried
- the HTTPArchive</a> for me.</figcaption>
- </figure>
-
- <h2 id="risk-security-vulnerabilities">Risk: Security Vulnerabilities</h2>
-
- <p>Another thing to take into consideration is the simple question of trust. If
- we’re bringing content from external sources onto our page, we have to hope that
- the assets that arrive are the ones we were expecting them to be, and that
- they’re doing only what we expected them to do.</p>
-
- <p>Imagine the damage that would be caused if someone managed to take control of
- a provider such as <code class="highlighter-rouge">code.jquery.com</code> and began serving compromised or malicious
- payloads. It doesn’t bear thinking about!</p>
-
- <h3 id="mitigation-subresource-integrity">Mitigation: Subresource Integrity</h3>
-
- <p>To the credit of all of the providers referenced so far in this article, they do
- all make use of <a href="https://developer.mozilla.org/en-US/docs/Web/Security/Subresource_Integrity">Subresource
- Integrity</a>
- (SRI). SRI is a mechanism by which the provider supplies a hash (technically,
- a hash that is then Base64 encoded) of the exact file that you both expect and
- intend to use. The browser can then check that the file you received is indeed
- the one you requested.</p>
-
- <div class="highlighter-rouge"><div class="highlight"><pre class="highlight"><code><script src="https://code.jquery.com/jquery-3.4.1.slim.min.js"
- integrity="sha256-pasqAKBDmFT4eHoN2ndd6lN370kFiGUFyTiUHWhU7k8="
- crossorigin="anonymous"></script>
- </code></pre></div></div>
-
- <p>Again, if you absolutely must link to an externally hosted static asset, make
- sure it’s SRI-enabled. You can add SRI yourself using <a href="https://www.srihash.org/">this handy
- generator</a>.</p>
-
- <h2 id="penalty-network-negotiation">Penalty: Network Negotiation</h2>
-
- <p>One of the biggest and most immediate penalties we pay is the cost of opening
- new TCP connections. Every new origin we need to visit needs a connection
- opening, and that can be very costly: DNS resolution, TCP handshakes, and TLS
- negotiation all add up, and the story gets worse the higher the latency of the
- connection is.</p>
-
- <p>I’m going to use an example taken straight from Bootstrap’s own <a href="https://getbootstrap.com/docs/4.3/getting-started/introduction/">Getting
- Started</a>. They
- instruct users to include these following four files:</p>
-
- <div class="highlighter-rouge"><div class="highlight"><pre class="highlight"><code><link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css" integrity="..." crossorigin="anonymous">
- <script src="https://code.jquery.com/jquery-3.3.1.slim.min.js" integrity="..." crossorigin="anonymous"></script>
- <script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.14.7/umd/popper.min.js" integrity="..." crossorigin="anonymous"></script>
- <script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js" integrity="..." crossorigin="anonymous"></script>
- </code></pre></div></div>
-
- <p>These four files are hosted across three different origins, so we’re going to
- need to open three TCP connections. How much does that cost?</p>
-
- <p>Well, on a reasonably fast connection, hosting these static assets off-site is
- 311ms, or 1.65×, slower than hosting them ourselves.</p>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/wpt-off-site-cable.png" alt=""/>
- <figcaption>By linking to three different origins in order to serve static
- assets, we cumulatively lose a needless 805ms to network negotiation. <a href="https://www.webpagetest.org/result/190531_FY_618f9076491312ef625cf2b1a51167ae/3/details/">Full
- test.</a></figcaption>
- </figure>
-
- <p>Okay, so not exactly terrifying, but Trainline, a client of mine, found that by
- reducing latency by 300ms, <a href="https://wpostats.com/2016/05/04/trainline-spending.html">customers spent an extra £8m
- a year</a>. This is
- a pretty quick way to make eight mill.</p>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/wpt-self-hosted-cable.png" alt=""/>
- <figcaption>By simply moving our assets onto the host domain, we completely
- remove any extra connection overhead. <a href="https://www.webpagetest.org/result/190531_FX_f7d7b8ae511b02aabc7fa0bbef0e37bc/3/details/">Full
- test.</a></figcaption>
- </figure>
-
- <p>On a slower, higher-latency connection, the story is much, much worse. Over 3G,
- the externally-hosted version comes in at an eye-watering <strong>1.765s slower</strong>.
- I thought this was meant to make our site faster?!</p>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/wpt-off-site-3g.png" alt=""/>
- <figcaption>On a high latency connection, network overhead totals a whopping
- 5.037s. All completely avoidable. <a href="https://www.webpagetest.org/result/190531_XE_a95eebddd2346f8bb572cecf4a8dae68/3/details/">Full
- test.</a></figcaption>
- </figure>
-
- <p>Moving the assets onto our own infrastructure brings load times down from around
- 5.4s to just 3.6s.</p>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/wpt-self-hosted-3g.png" alt=""/>
- <figcaption>By self-hosting our static assets, we don’t need to open any more
- connections. <a href="https://www.webpagetest.org/result/190531_ZF_4d76740567ec1eba1e6ec67acfd57627/1/details/">Full
- test.</a></figcaption>
- </figure>
-
- <p>If this isn’t already a compelling enough reason to self-host your static
- assets, I’m not sure what is!</p>
-
- <h3 id="mitigation-preconnect">Mitigation: <code class="highlighter-rouge">preconnect</code></h3>
-
- <p>Naturally, my whole point here is that you should not host any static assets
- off-site if you’re otherwise able to self-host them. However, if your hands are
- somehow tied, then you can use <a href="https://speakerdeck.com/csswizardry/more-than-you-ever-wanted-to-know-about-resource-hints?slide=28">a <code class="highlighter-rouge">preconnect</code> Resource
- Hint</a>
- to preemptively open a TCP connection to the specified origin(s):</p>
-
- <div class="highlighter-rouge"><div class="highlight"><pre class="highlight"><code><head>
-
- ...
-
- <link rel="preconnect" href="https://code.jquery.com" />
-
- ...
-
- </head>
- </code></pre></div></div>
-
- <p>For bonus points, deploying these as <a href="https://andydavies.me/blog/2019/03/22/improving-perceived-performance-with-a-link-rel-equals-preconnect-http-header/">HTTP
- headers</a>
- will be even faster.</p>
-
- <p><strong>N.B.</strong> Even if you do implement <code class="highlighter-rouge">preconnect</code>, you’re still only going to make
- a small dent in your lost time: you still need to open the relevant connections,
- and, especially on high latency connections, it’s unlikely that you’re ever
- going to fully pay off the overhead upfront.</p>
-
- <h2 id="penalty-loss-of-prioritisation">Penalty: Loss of Prioritisation</h2>
-
- <p>The second penalty comes in the form of a protocol-level optimisation that we
- miss out on the moment we split content across domains. If you’re running over
- HTTP/2—which, by now, you should be—you get access to prioritisation. All
- streams (ergo, resources) within the same TCP connection carry a priority, and
- the browser and server work in tandem to build a dependency tree of all of these
- prioritised streams so that we can return critical assets sooner, and perhaps
- delay the delivery of less important ones.</p>
-
- <p><small><strong>N.B.</strong> Technically, owing to H/2’s <a href="https://daniel.haxx.se/blog/2016/08/18/http2-connection-coalescing/">connection
- coalescence</a>,
- requests can be prioritised against each other over different domains as long as
- they share the same IP address.</small></p>
-
- <p>If we split our assets across multiple domains, we have to open up several
- unique TCP connections. We cannot cross-reference any of the priorities within
- these connections, so we lose the ability to deliver assets in a considered and
- well designed manner.</p>
-
- <p>Compare the two HTTP/2 dependency trees for both the off-site and self-hosted
- versions respectively:</p>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/wpt-dep-tree-off-site.png" alt=""/>
- <figcaption>Notice how we need to build new dependency trees per
- origin? Stream IDs 1 and 3 keep reoccurring.</figcaption>
- </figure>
-
- <figure>
- <img src="/wp-content/uploads/2019/05/wpt-dep-tree-self-hosted.png" alt=""/>
- <figcaption>By hosting all content under the same origin, we can build one, more
- complete dependency tree. Every stream has a unique ID as they’re all in the
- same tree.</figcaption>
- </figure>
-
- <p><small>Fun fact: Stream IDs with an odd number were initiated by the client;
- those with an even number were initiated by the server. I honestly don’t think
- I’ve ever seen an even-numbered ID in the wild.</small></p>
-
- <p>If we serve as much content as possible from one domain, we can let H/2 do its
- thing and prioritise assets more completely in the hopes of better-timed
- responses.</p>
-
- <h2 id="penalty-caching">Penalty: Caching</h2>
-
- <p>By and large, static asset hosts seem to do pretty well at establishing
- long-lived <code class="highlighter-rouge">max-age</code> directives. This makes sense, as static assets at versioned
- URLs (as above) will never change. This makes it very safe and sensible to
- enforce a reasonably aggressive cache policy.</p>
-
- <p>That said, this isn’t always the case, and by self-hosting your assets you can
- design <a href="https://csswizardry.com/2019/03/cache-control-for-civilians/">much more bespoke caching
- strategies</a>.</p>
-
- <h2 id="myth-cross-domain-caching">Myth: Cross-Domain Caching</h2>
-
- <p>A more interesting take is the power of cross-domain caching of assets. That is
- to say, if lots and lots of sites link to the same CDN-hosted version of, say,
- jQuery, then surely users are likely to already have that exact file on their
- machine already? Kinda like peer-to-peer resource sharing. This is one of the
- most common arguments I hear in favour of using a third-party static asset
- provider.</p>
-
- <p>Unfortunately, there seems to be no published evidence that backs up these
- claims: there is nothing to suggest that this is indeed the case. Conversely,
- <a href="https://discuss.httparchive.org/t/analyzing-resource-age-by-content-type/1659">recent
- research</a>
- by <a href="https://twitter.com/paulcalvano">Paul Calvano</a> hints that the opposite might
- be the case:</p>
-
- <blockquote>
- <p>There is a significant gap in the 1st vs 3rd party resource age of CSS and web
- fonts. 95% of first party fonts are older than 1 week compared to 50% of 3rd
- party fonts which are less than 1 week old! This makes a strong case for self
- hosting web fonts!</p>
- </blockquote>
-
- <p>In general, third party content seems to be less-well cached than first party
- content.</p>
-
- <p>Even more importantly, <a href="https://andydavies.me/blog/2018/09/06/safari-caching-and-3rd-party-resources/">Safari has completely disabled this
- feature</a>
- for fear of abuse where privacy is concerned, so the shared cache technique
- cannot work for, at the time of writing, <a href="http://gs.statcounter.com/">16% of users
- worldwide</a>.</p>
-
- <p>In short, although nice in theory, there is no evidence that cross-domain
- caching is in any way effective.</p>
-
- <h2 id="myth-access-to-a-cdn">Myth: Access to a CDN</h2>
-
- <p>Another commonly touted benefit of using a static asset provider is that they’re
- likely to be running beefy infrastructure with CDN capabilities: globally
- distributed, scalable, low-latency, high availability.</p>
-
- <p>While this is absolutely true, if you care about performance, you should be
- running your own content from a CDN already. With the price of modern hosting
- solutions being what they are (this site is fronted by Cloudflare which is
- free), there’s very little excuse for not serving your own assets from one.</p>
-
- <p>Put another way: if you think you need a CDN for your jQuery, you’ll need a CDN
- for everything. Go and get one.</p>
-
- <h2 id="self-host-your-static-assets">Self-Host Your Static Assets</h2>
-
- <p>There really is very little reason to leave your static assets on anyone else’s
- infrastructure. The perceived benefits are often a myth, and even if they
- weren’t, the trade-offs simply aren’t worth it. Loading assets from multiple
- origins is demonstrably slower. Take ten minutes over the next few days to audit
- your projects, and fetch any off-site static assets under your own control.</p>
- </article>
- </section>
-
-
- <nav id="jumpto">
- <p>
- <a href="/david/blog/">Accueil du blog</a> |
- <a href="https://csswizardry.com/2019/05/self-host-your-static-assets/">Source originale</a> |
- <a href="/david/stream/2019/">Accueil du flux</a>
- </p>
- </nav>
-
- <footer>
- <div>
- <img src="/static/david/david-larlet-avatar.jpg" loading="lazy" class="avatar" width="200" height="200">
- <p>
- Bonjour/Hi!
- Je suis <a href="/david/" title="Profil public">David Larlet</a>, je vis actuellement à Montréal et j’alimente cet espace depuis 15 ans. <br>
- Si tu as apprécié cette lecture, n’hésite pas à poursuivre ton exploration. Par exemple via les <a href="/david/blog/" title="Expériences bienveillantes">réflexions bimestrielles</a>, la <a href="/david/stream/2019/" title="Pensées (dés)articulées">veille hebdomadaire</a> ou en t’abonnant au <a href="/david/log/" title="S’abonner aux publications via RSS">flux RSS</a> (<a href="/david/blog/2019/flux-rss/" title="Tiens c’est quoi un flux RSS ?">so 2005</a>).
- </p>
- <p>
- Je m’intéresse à la place que je peux avoir dans ce monde. En tant qu’humain, en tant que membre d’une famille et en tant qu’associé d’une coopérative. De temps en temps, je fais aussi des <a href="https://github.com/davidbgk" title="Principalement sur Github mais aussi ailleurs">trucs techniques</a>. Et encore plus rarement, <a href="/david/talks/" title="En ce moment je laisse plutôt la place aux autres">j’en parle</a>.
- </p>
-
- <p>
- Voici quelques articles choisis :
- <a href="/david/blog/2019/faire-equipe/" title="Accéder à l’article complet">Faire équipe</a>,
- <a href="/david/blog/2018/bivouac-automnal/" title="Accéder à l’article complet">Bivouac automnal</a>,
- <a href="/david/blog/2018/commodite-effondrement/" title="Accéder à l’article complet">Commodité et effondrement</a>,
- <a href="/david/blog/2017/donnees-communs/" title="Accéder à l’article complet">Des données aux communs</a>,
- <a href="/david/blog/2016/accompagner-enfant/" title="Accéder à l’article complet">Accompagner un enfant</a>,
- <a href="/david/blog/2016/senior-developer/" title="Accéder à l’article complet">Senior developer</a>,
- <a href="/david/blog/2016/illusion-sociale/" title="Accéder à l’article complet">L’illusion sociale</a>,
- <a href="/david/blog/2016/instantane-scopyleft/" title="Accéder à l’article complet">Instantané Scopyleft</a>,
- <a href="/david/blog/2016/enseigner-web/" title="Accéder à l’article complet">Enseigner le Web</a>,
- <a href="/david/blog/2016/simplicite-defaut/" title="Accéder à l’article complet">Simplicité par défaut</a>,
- <a href="/david/blog/2016/minimalisme-esthetique/" title="Accéder à l’article complet">Minimalisme et esthétique</a>,
- <a href="/david/blog/2014/un-web-omni-present/" title="Accéder à l’article complet">Un web omni-présent</a>,
- <a href="/david/blog/2014/manifeste-developpeur/" title="Accéder à l’article complet">Manifeste de développeur</a>,
- <a href="/david/blog/2013/confort-convivialite/" title="Accéder à l’article complet">Confort et convivialité</a>,
- <a href="/david/blog/2013/testament-numerique/" title="Accéder à l’article complet">Testament numérique</a>,
- et <a href="/david/blog/" title="Accéder aux archives">bien d’autres…</a>
- </p>
- <p>
- On peut <a href="mailto:david%40larlet.fr" title="Envoyer un courriel">échanger par courriel</a>. Si éventuellement tu souhaites que l’on travaille ensemble, tu devrais commencer par consulter le <a href="http://larlet.com">profil dédié à mon activité professionnelle</a> et/ou contacter directement <a href="http://scopyleft.fr/">scopyleft</a>, la <abbr title="Société coopérative et participative">SCOP</abbr> dont je fais partie depuis six ans. Je recommande au préalable de lire <a href="/david/blog/2018/cout-site/" title="Attention ce qui va suivre peut vous choquer">combien coûte un site</a> et pourquoi je suis plutôt favorable à une <a href="/david/pro/devis/" title="Discutons-en !">non-demande de devis</a>.
- </p>
- <p>
- Je ne traque pas ta navigation mais mon
- <abbr title="Alwaysdata, 62 rue Tiquetonne 75002 Paris, +33.184162340">hébergeur</abbr>
- conserve des logs d’accès.
- </p>
- </div>
- </footer>
- <script type="text/javascript">
- ;(_ => {
- const jumper = document.getElementById('jumper')
- jumper.addEventListener('click', e => {
- e.preventDefault()
- const anchor = e.target.getAttribute('href')
- const targetEl = document.getElementById(anchor.substring(1))
- targetEl.scrollIntoView({behavior: 'smooth'})
- })
- })()
- </script>
|