These are guidelines for what AMP cache implementations should look like. Some items are required for overall security of the platform while others are suggestions for performance improvements. All modifications are made to both AMP and AMP4ADS documents except where noted.
For example, given a recent version of everything.amp.html, the output after modifications will be this version.
The AMP Cache parses and re-serializes all documents to remove any ambiguities in parsing the document which might result in subtly different parses in different browsers.
example
before | after |
---|---|
<foo><!-- comment --></foo> |
<foo></foo> |
example
before | after |
---|---|
<P DATA-FOO=BAR> |
<p data-foo=BAR> |
example
before | after |
---|---|
<p data-foo='< >'> |
<p data-foo="< >"> |
Void elements are tags that have no end tag and also no contents. All other tags are closed.
example
before | after |
---|---|
<foo><bar></foo> <br/> |
<foo><bar></bar></foo> <br> |
example
before | after |
---|---|
<p data-foo=bar > |
<p data-foo=bar> |
example
before | after |
---|---|
3 < 4 |
3 < 4 |
example
before | after |
---|---|
|
"\u00A0" |
a |
a |
' |
' |
example
before | after |
---|---|
<body></body><div>foo</div>text |
<body><div>foo</div>text</body> |
The AMP Cache rewrites URLs found in the AMP HTML for two purposes. One is to rebase relative URLs found in the document so that the URL remains the same when loaded from the AMP Cache. The other reason is to improve performance by selecting a different equivalent resource. This includes rewriting image and font URLs to use a cached copy and rewriting AMP javascript URLs to use a copy with longer cache lifetimes.
All relative href
, src
, bookend-config-src
, data-iframe-src
and data-no-service-worker-fallback-shell-url
URLs are rewritten as absolute URLs
bookend-confg-src
is part of <amp-story>
spec
data-iframe-src
and data-no-service-worker-fallback-shell-url
are part of <amp-install-serviceworker>
spec
example
before | after |
---|---|
<a href=foo.html target=_top>Lorem ipsum</a> |
<a href=https://example.com/foo.html target=_top>Lorem ipsum</a> |
<amp-list src="list.json" ...>...</amp-list> |
<amp-list src="https://example.com/list.json" ...>...</amp-list> |
<amp-install-serviceworker data-iframe-src="sw.html"...></amp-install-serviceworker> |
<amp-install-serviceworker data-iframe-src="https://example.com/sw.html"...></amp-install-serviceworker> |
example
before | after |
---|---|
<form action=/subscribe>...</form> |
<form action=https://example.com/subscribe>...</form> |
example
before | after |
---|---|
@font-face { font-family: Foo; src: url(font.woff); } |
@font-face { font-family: Foo; src: url(https://example.com/font.woff); } |
example
before | after |
---|---|
<amp-img src=https://example.com/foo.png></amp-img> |
<amp-img src=/i/s/example.com/foo.png></amp-img> |
<amp-img srcset="https://example.com/bar.png 1080w, https://example.com/bar-400.png 400w"> |
<amp-img src="/i/s/example.com/bar.png 1080w, /i/s/example.com/bar-400.png 400w"> |
<amp-anim src=foo.gif></amp-anim> |
<amp-anim src=/i/s/example.com/foo.gif></amp-anim> |
<amp-video poster=bar.png> |
<amp-video poster=/i/s/example.com/bar.png> |
<svg class="icon" xmlns:xlink="http://www.w3.org/1999/xlink"><use xlink:href="https://example.com/icon.svg#icon"></use></svg> |
<svg class=icon xmlns:xlink="http://www.w3.org/1999/xlink"><use xlink:href="/i/s/example.com/icon.svg#icon"></use></svg> |
Condition:
If <a>
tag does not have attribute target=_blank
or target=_top
then add a target=
. This added target=
will be either be target=_blank
or target=_top
. If the document has <base target=...>
of either _top
or _blank
then use that value. Otherwise all other target
values are rewritten to _top
.
example
before | after |
---|---|
<a href=https://example.com/foo.html>Lorem ipsum</a> |
<a href=https://example.com/foo.html target=_top>Lorem ipsum</a> |
<a href=https://example.com/bar.html target=_blank>Lorem ipsum</a> |
<a href=https://example.com/bar.html target=_blank>Lorem ipsum</a> |
<a href=https://example.com/baz.html target=window>Lorem ipsum</a> |
<a href=https://example.com/baz.html target=_top>Lorem ipsum</a> |
<head> ... <base target=_blank> ... </head> <body> ... <a href=https://example.com/foo.html>Lorem ipsum</a> ... </body> |
<head> ... <base target=_blank> ... </head> <body> ... <a href=https://example.com/foo.html target=_blank>Lorem ipsum</a> ... </body> |
Before the AMP Runtime script
tag, insert a link
tag that tells the browser the AMP Runtime script
tag is high priority despite being an async script
tag.
example
before | after |
---|---|
<head> ... <script async src=https://cdn.ampproject.org/v0.js></script> ... </head> |
<head> ... <link as=script href=https://cdn.ampproject.org/v0.js rel=preload> <script async src=https://cdn.ampproject.org/v0.js></script> ... </head> |
When a given AMP document does not have a favicon present, insert one. Inserted tag is of the form <link href={document_protocol}://{document_domain}/favicon.ico rel=icon>
.
Condition:
No <link>
tag present with attribute rel
equal to any of the following: icon
, icon shortcut
, shortcut icon
.
example
before | after |
---|---|
<head> ... </head> |
<head> ... <link href=https://example.com/favicon.ico rel=icon> </head> |
<head> ... <link href=https://example.com/favicon.ico rel="icon shortcut"> ... </head> |
<head> ... <link href=https://example.com/favicon.ico rel="icon shortcut"> ... </head> |
Condition:
<link rel=manifest>
tag present in the document.
example
before | after |
---|---|
<head> ... <link rel=manifest> ... </head> |
<head> ... <link rel=origin-manifest> ... </head> |
If the document was fetched from HTTP origins and does not have a meta referrer tag then insert one.
Condition:
No <meta name=referrer ...>
tag present and document was fetched from HTTP and not HTTPS.
example
before | after |
---|---|
<head> ... </head> |
<head> ... <meta content=always name=referrer> </head> |
AMP Cache pages should not show up in search result pages. The cache also uses robots.txt
to enforce this.
example
before | after |
---|---|
<head> ... </head> |
<head> ... <meta content=noindex name=robots> </head> |
Some <meta>
tags are used by AMP Components and providing them before the AMP Component's script has loaded is important. As a result we move all <meta>
tags to be the first children of <head>
. An example of one of these tags is <meta name="amp-experiments-opt-in" ...>
.
example
before | after |
---|---|
<head> <meta charset=utf-8> ... <script async src=https://cdn.ampproject.org/v0.js></script> <meta name="amp-experiments-opt-in" content="experiment-a,experiment-b"> </head> |
<head> <meta charset=utf-8> <meta name="amp-experiments-opt-in" content="experiment-a,experiment-b"> <script async src=https://cdn.ampproject.org/v0.js></script> ... </head> |
The AMP Cache removes any resource hints in the original document.
Condition
Any <link>
tag present with attribute rel
equal to any of the following:
dns-prefetch
preconnect
prefetch
preload
prerender
example
before | after |
---|---|
<head> ... <link rel=dns-prefetch href=https://example.com> ... </head> |
<head> ... </head> |
Condition:
Remove any <meta>
tags except for those that:
- have attribute
charset
- do not have attributes
content
,itemprop
,name
andproperty
- have attribute
http-equiv
where attribute value is not one of:content-security-policy
refresh
x-dns-prefetch-control
- have attribute
name
with case-insensitive prefixamp-
- have attribute
name
with case-insensitive prefixamp4ads-
- have attribute
name
with case-insensitive prefixdc.
- have attribute
name
with case-insensitive prefixi-amphtml-
- have attribute
name
with case-insensitive prefixtwitter:
- have attribute
name=apple-itunes-app
- have attribute
name=referrer
[note: this may be inserted by AMP Cache] - have attribute
name=robots
[note: this is inserted by AMP Cache] - have attribute
name=viewport
- have attribute
property
with case-insensitive prefix "al:" - have attribute
property
with case-insensitive prefix "fb:" - have attribute
property
with case-insensitive prefix "og:"
example
before | after |
---|---|
<meta charset=utf-8> <meta http-equiv=content-language content=en> <meta name=description content="An example AMP page"> <meta name=twitter:title content="AMP Example"> |
<meta charset=utf-8> <meta http-equiv=content-language content=en> <meta name=twitter:title content="AMP Example"> |
<meta charset=utf-8> <meta http-equiv=x-dns-prefetch-control content=on> |
<meta charset=utf-8> |
This is discussed in detail at Server side filtering for amp-live-list
Condition:
Remove nonce
attribute from every tag.
example
before | after |
---|---|
<script async custom-element=amp-youtube nonce=cryptohash src=https://cdn.ampproject.org/v0/amp-youtube-0.1.js></script> |
<script async custom-element=amp-youtube src=https://cdn.ampproject.org/v0/amp-youtube-0.1.js></script> |
These are modifications that either reduce the byte size of the document or decreases the time to render. An AMP cache is not required to implement these.
If possible, rewrite to use the stable version. Otherwise use the unversioned path. The stable version takes the form <script async src=https://cdn.ampproject.org/rtv/{version}/v0.js></script>
.
example
before | after |
---|---|
<script async src=https://cdn.ampproject.org/v0.js></script> |
<script async src=https://cdn.ampproject.org/rtv/031485231782273/v0.js></script> |
The AMP Cache adds prefetch hint tags for browsers to assist in loading resources earlier and thus speed up page loads.
Condition:
Has a stylesheet of the form: <link href=https://fonts.googleapis.com/... rel=stylesheet>
.
example
before | after |
---|---|
<head> ... <link href=https://fonts.googleapis.com/css?family=Lato rel=stylesheet> ... </head> |
<head> ... <link href=https://fonts.googleapis.com/css?family=Lato rel=stylesheet> <link href=https://fonts.gstatic.com rel="dns-prefetch preconnect" crossorigin> ... </head> |
The AMP Cache places the AMP engine javascript as the second child of <head>
right after <meta charset=utf-8>
. It then emits any other render blocking custom-element script tags followed by the remaining custom-element <script>
tags in the document. Render blocking custom-element <script>
tags are listed in SERVICES at render-delaying-services.js.
example
before | after |
---|---|
<head> ... <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async src=https://cdn.ampproject.org/v0.js></script> <meta charset=utf-8> ... </head> |
<head> <meta charset=utf-8> <script async src=https://cdn.ampproject.org/v0.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> |
If a custom-element <script>
tag is included more than once, the AMP Cache removes all but one.
example
before | after |
---|---|
<head> ... <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> ... </head> |
<head> ... <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> |
This is currently a work in progress.
If a custom-element <script>
tag is included in <head>
but not used in <body>
then remove it. There are several exceptions to this listed under Condition.
Condition: Remove unused custom-element extensions with the following exceptions:
- Do not remove any custom-element extensions if
<amp-live-list>
is present within the document - Do not remove any of the following custom-element extensions:
- amp-access
- amp-access-laterpay
- amp-analytics
- amp-auto-ads
- amp-dynamic-css-classes
- amp-form
example
before | after |
---|---|
<head> ... <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-analytics src=https://cdn.ampproject.org/v0/amp-analytics-0.1.js></script> <script async custom-element=amp-youtube src=https://cdn.ampproject.org/v0/amp-youtube-0.1.js></script> ... </head> <body> ... <amp-youtube ...></amp-youtube> ... </body> |
<head> ... <script async custom-element=amp-analytics src=https://cdn.ampproject.org/v0/amp-analytics-0.1.js></script> <script async custom-element=amp-youtube src=https://cdn.ampproject.org/v0/amp-youtube-0.1.js></script> ... </head> <body> ... <amp-youtube ...></amp-youtube> ... </body> |
Remove whitespace in <head>
except for tags that should preserve whitespace.
Condition: Remove whitespace except for within these tags:
<script>
<style>
example
before | after |
---|---|
<head> ... <meta charset=utf-8> <style amp-custom> body { background-color: white; } </style> ... </head> |
<head>...<meta charset=utf-8><style amp-custom> body { background-color: white; } </style>...</head> |
This is currently a work in progress for AMP documents and implemented for AMP4ADS documents.
Remove quotes from around an attribute’s value unless the attribute’s value has an ASCII character in the set { 0x20(space), 0x22("), 0x27('), 0x3E(>), 0x60(`) }.
example
before | after |
---|---|
<head> ... <meta charset="utf-8"> <script async src="https://cdn.ampproject.org/v0.js"></script> <link rel="icon shortcut" href="https://example.com/favicon.ico"> ... </head> |
<head> ... <meta charset=utf-8> <script async src=https://cdn.ampproject.org/v0.js></script> <link rel="icon shortcut" href=https://example.com/favicon.ico> ... </head> |
These are AMP4ADS specific modifications and not implemented for AMP documents.
The AMP Cache places the AMP4ADS engine javascript as the second child of <head>
right after <meta charset=utf-8>
.
example
before | after |
---|---|
<head> ... <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async src=https://cdn.ampproject.org/amp4ads-v0.js></script> <meta charset=utf-8> ... </head> |
<head> <meta charset=utf-8> <script async src=https://cdn.ampproject.org/amp4ads-v0.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> |
AMP4ADS requires providing the start and end position of the block of <script>
tags that represent the AMP4ADS engine and custom-element extensions. These
string offsets are in UTF-16 encoding lengths. This data is provided in the
amp-ad-metadata
JSON as ampRuntimeUtf16CharOffsets
. The amp-ad-metadata
JSON is appended to the end of the <body>
.
example
before | after |
---|---|
<head> <meta charset=utf-8> <script async src=https://cdn.ampproject.org/amp4ads-v0.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> <body> ... </body> |
<head> <meta charset=utf-8> <script async src=https://cdn.ampproject.org/amp4ads-v0.js></script> <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> <body> ... <script type=application/json amp-ad-metadata> { "ampRuntimeUtf16CharOffsets" : [ 55, 337 ] } </script> </body> |
AMP4ADS requires providing the start and end position of amp-access
and
amp-analytics
JSON. These string offsets are in UTF-16 encoding lengths. This
data is provided in the amp-ad-metadata
JSON as jsonUtf16CharOffsets
. The
amp-ad-metadata
JSON is appended to the end of the <body>
.
example
before | after |
---|---|
<head> ... <script id=amp-access type=application/json> { ... } </script> ... </head> <body> ... <amp-analytics> <script type=application/json> { ... } </script> </amp-analytics> ... </body> |
<head> ... <script id=amp-access type=application/json> { ... } </script> ... </head> <body> ... <amp-analytics> <script type=application/json> { ... } </script> </amp-analytics> ... <script type=application/json amp-ad-metadata> { "jsonUtf16CharOffsets" : { "amp-access": [ 12, 92 ], "amp-analytics": [ 105, 175], } } </script> </body> |
AMP4ADS requires providing the start and end position of CSS body
selectors in
<style amp-custom>
. These string offsets are in UTF-16 encoding lengths. This
data is provided in the amp-ad-metadata
JSON as cssReplacementRanges
. The
amp-ad-metadata
JSON is appended to the end of the <body>
.
example
before | after |
---|---|
<head> ... <style amp-custom> body { background-color: #eee; } ... @media screen and (min-width:480px) { body { color: #ccc; } } </style> ... </head> <body> ... </body> |
<head> ... <style amp-custom> body { background-color: #eee; } ... @media screen and (min-width:480px) { body { color: #ccc; } } </style> ... </head> <body> ... <script type=application/json amp-ad-metadata> { "cssReplacementRanges" : [ [ 59, 63 ], [ 139, 143 ], } </script> </body> |
AMP4ADS requires providing additional metadata of what custom-element extensions
are included in the AMP4ADS document. This data is provided in the
amp-ad-metadata
JSON as customElementExtensions
and extensions
. The
amp-ad-metadata
JSON is appended to the end of the <body>
.
customElementExtensions
is just a list of the custom-element extension names
in the document. This is now deprecated but still used.
extensions
is a list of objects containing custom-element
and src
attributes, which are the name of the custom-element and the location of the
extension respectively.
example
before | after |
---|---|
<head> ... <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> <body> ... </body> |
<head> ... <script async custom-element=amp-accordion src=https://cdn.ampproject.org/v0/amp-accordion-0.1.js></script> <script async custom-element=amp-instagram src=https://cdn.ampproject.org/v0/amp-instagram-0.1.js></script> ... </head> <body> ... <script type=application/json amp-ad-metadata> { "customElementExtensions" : [ "amp-accordion", "amp-instagram" ], "extensions" : [ { "custom-element" : "amp-accordion", "src" : "https://cdn.ampproject.org/v0/amp-accordion-0.1.js" }, { "custom-element" : "amp-instagram", "src" : "https://cdn.ampproject.org/v0/amp-instagram-0.1.js" }] } </script> </body> |
AMP4ADS requires providing additional metadata of what custom fonts (href
and
if present media
) were included in the AMP4ADS document. This data is provided
in the amp-ad-metadata
JSON as customStylesheets
. The amp-ad-metadata
JSON
is appended to the end of the <body>
.
example
before | after |
---|---|
<head> ... <link href=https://fonts.googleapis.com/css?family=Foo rel=stylesheet> <link href=https://fonts.googleapis.com/css?family=Bar rel=stylesheet media=print> ... </head> <body> ... </body> |
<head> ... <link href=https://fonts.googleapis.com/css?family=Foo rel=stylesheet> <link href=https://fonts.googleapis.com/css?family=Bar rel=stylesheet media=print> ... </head> <body> ... <script type=application/json amp-ad-metadata> { "customStylesheets" : [ { "href" : "https://fonts.googleapis.com/css?family=Foo" }, { "href" : "https://fonts.googleapis.com/css?family=Bar", "media" : "print" } } </script> </body> |