Skip to content

Commit ae9b13e

Browse files
sebastianbenzijjkTimer
authored
Migrate to AMP Optimizer 2.0 (#10535)
* Migrate to AMP Optimizer 2.0 Most notable changes: * Automatically import all missing AMP component scripts. * Automatically add any missing mandatary AMP tags. * 40% faster I've updated the docs to mention component auto import and added a corresponding test case. * change validator tests which now pass validation * Improve wording Co-Authored-By: JJ Kasper <[email protected]> * Update adding-amp-components.md Co-authored-by: JJ Kasper <[email protected]> Co-authored-by: Joe Haddad <[email protected]>
1 parent 1c247b5 commit ae9b13e

File tree

8 files changed

+132
-54
lines changed

8 files changed

+132
-54
lines changed

docs/advanced-features/amp-support/adding-amp-components.md

+29-3
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,35 @@ description: Add components from the AMP community to AMP pages, and make your p
44

55
# Adding AMP Components
66

7-
The AMP community provide [many components](https://amp.dev/documentation/components/) to make AMP pages more interactive. You can add these components to your page by using `next/head`, as in the following example:
7+
The AMP community provides [many components](https://amp.dev/documentation/components/) to make AMP pages more interactive. Next.js will automatically import all components used on a page and there is no need to manually import AMP component scripts:
8+
9+
```jsx
10+
export const config = { amp: true }
11+
12+
function MyAmpPage() {
13+
const date = new Date()
14+
15+
return (
16+
<div>
17+
<p>Some time: {date.toJSON()}</p>
18+
<amp-timeago
19+
width="0"
20+
height="15"
21+
datetime={date.toJSON()}
22+
layout="responsive"
23+
>
24+
.
25+
</amp-timeago>
26+
</div>
27+
)
28+
}
29+
30+
export default MyAmpPage
31+
```
32+
33+
The above example uses the [`amp-timeago`](https://amp.dev/documentation/components/amp-timeago/?format=websites) component.
34+
35+
By default, the latest version of a component is always imported. If you want to customize the version, you can use `next/head`, as in the following example:
836

937
```jsx
1038
import Head from 'next/head'
@@ -40,5 +68,3 @@ function MyAmpPage() {
4068

4169
export default MyAmpPage
4270
```
43-
44-
The above example uses the [`amp-timeago`](https://amp.dev/documentation/components/amp-timeago/?format=websites) component.

packages/next/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@
5757
]
5858
},
5959
"dependencies": {
60-
"@ampproject/toolbox-optimizer": "1.1.1",
60+
"@ampproject/toolbox-optimizer": "2.0.0",
6161
"@babel/core": "7.7.2",
6262
"@babel/plugin-proposal-class-properties": "7.7.0",
6363
"@babel/plugin-proposal-nullish-coalescing-operator": "7.7.4",

test/integration/amp-export-validation/pages/cat.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ export const config = { amp: true }
22

33
export default () => (
44
<div>
5-
{/* I show a warning since the amp-video script isn't added */}
6-
<amp-video src="/cats.mp4" height={400} width={800} />
5+
{/* I show a warning since the width and height attribute is missing */}
6+
<amp-video src="/cats.mp4" layout="responsive" />
77
</div>
88
)

test/integration/amp-export-validation/pages/dog-cat.js

-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,5 @@ export default () => (
44
<div>
55
{/* I throw an error since <amp-img/> should be used instead */}
66
<img src="/dog.gif" height={400} width={800} />
7-
{/* I show a warning since the amp-video script isn't added */}
8-
<amp-video src="/cats.mp4" height={400} width={800} />
97
</div>
108
)

test/integration/amp-export-validation/test/index.test.js

+2-5
Original file line numberDiff line numberDiff line change
@@ -32,9 +32,6 @@ describe('AMP Validation on Export', () => {
3232
expect(buildOutput).toMatch(
3333
/error.*The parent tag of tag 'IMG-I-AMPHTML-INTRINSIC-SIZER' is 'div', but it can only be 'i-amphtml-sizer-intrinsic'/
3434
)
35-
expect(buildOutput).toMatch(
36-
/warn.*The tag 'amp-video extension .js script' is missing or incorrect, but required by 'amp-video'/
37-
)
3835
})
3936

4037
it('should export AMP pages', async () => {
@@ -63,7 +60,7 @@ describe('AMP Validation on Export', () => {
6360
stderr: true,
6461
})
6562
expect(stdout).toMatch(
66-
/warn.*The tag 'amp-video extension \.js script' is missing/
63+
/error.*The mandatory attribute 'height' is missing in tag 'amp-video'\./
6764
)
6865
await expect(access(join(outDir, 'cat.html'))).resolves.toBe(undefined)
6966
await expect(stderr).not.toMatch(
@@ -117,7 +114,7 @@ describe('AMP Validation on Export', () => {
117114
stderr: true,
118115
})
119116
expect(stdout).toMatch(
120-
/warn.*The tag 'amp-video extension .js script' is missing or incorrect, but required by 'amp-video'/
117+
/error.*The parent tag of tag 'IMG-I-AMPHTML-INTRINSIC-SIZER' is 'div', but it can only be 'i-amphtml-sizer-intrinsic'/
121118
)
122119
expect(stdout).toMatch(
123120
/error.*The parent tag of tag 'IMG-I-AMPHTML-INTRINSIC-SIZER' is 'div', but it can only be 'i-amphtml-sizer-intrinsic'/
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
export const config = { amp: true }
2+
3+
export default () => (
4+
<amp-twitter
5+
width="390"
6+
height="330"
7+
layout="responsive"
8+
data-momentid="1018806709412876288"
9+
></amp-twitter>
10+
)

test/integration/amphtml/test/index.test.js

+5
Original file line numberDiff line numberDiff line change
@@ -112,6 +112,11 @@ describe('AMP Usage', () => {
112112
const html = await renderViaHTTP(appPort, '/only-amp')
113113
await validateAMP(html)
114114
})
115+
116+
it('should auto import extensions', async () => {
117+
const html = await renderViaHTTP(appPort, '/auto-import')
118+
await validateAMP(html)
119+
})
115120
})
116121

117122
describe('With AMP context', () => {

yarn.lock

+83-41
Original file line numberDiff line numberDiff line change
@@ -2,36 +2,47 @@
22
# yarn lockfile v1
33

44

5-
"@ampproject/toolbox-core@^1.1.1":
6-
version "1.1.1"
7-
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-core/-/toolbox-core-1.1.1.tgz#540c8f3ab0f5d1faa1ba35282cd5f5f3f0e16a76"
8-
integrity sha512-jcuVJUnGDRUEJgMYO6QVdf1dBy/oLZX3NjN2hYG48biFcPCvXevuv4xYFZMJsnsHSvXKg8y0qB8rANNyhTUN/A==
5+
"@ampproject/toolbox-core@^2.0.0":
6+
version "2.0.0"
7+
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-core/-/toolbox-core-2.0.0.tgz#3329a01eec8f767614385cc56cee8be32ccf7f32"
8+
integrity sha512-xAJOmh6MPS2mdHNsK8mj1t8TLh6mlehirh0fOBsRhKCNCJXgg4Gfd2u5igy8VFq9sYnuWP/npFyjGX36qpXW5Q==
99
dependencies:
10-
node-fetch "2.6.0"
10+
cross-fetch "3.0.4"
1111

12-
"@ampproject/toolbox-optimizer@1.1.1":
13-
version "1.1.1"
14-
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-optimizer/-/toolbox-optimizer-1.1.1.tgz#be66245c966ba9b0f5e3020109f87fea90ea377d"
15-
integrity sha512-LTtTM5FSOrWuTJ6mOwPfZmpUDI6polrNz3tX2EmDmDkjDK+43vSpq1OHtukivIFHafdixJuoeki5dF3PC/ZoWw==
12+
"@ampproject/toolbox-optimizer@2.0.0":
13+
version "2.0.0"
14+
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-optimizer/-/toolbox-optimizer-2.0.0.tgz#96ea3d26bc7a9f19718a27c721e021a121003601"
15+
integrity sha512-ZYRi7vB4ALC8DnTHQLchAeHMGsFml/Zr4fNWBlTiMGfvWGL0XTV9YyP4s24IDwAlunEgynmz0FTrGMJdRpNf2Q==
1616
dependencies:
17-
"@ampproject/toolbox-core" "^1.1.1"
18-
"@ampproject/toolbox-runtime-version" "^1.1.1"
19-
"@ampproject/toolbox-script-csp" "^1.1.1"
17+
"@ampproject/toolbox-core" "^2.0.0"
18+
"@ampproject/toolbox-runtime-version" "^2.0.0"
19+
"@ampproject/toolbox-script-csp" "^2.0.0"
20+
"@ampproject/toolbox-validator-rules" "^2.0.0"
2021
css "2.2.4"
21-
parse5 "5.1.0"
22-
parse5-htmlparser2-tree-adapter "5.1.0"
22+
domhandler "3.0.0"
23+
domutils "2.0.0"
24+
htmlparser2 "4.0.0"
25+
normalize-html-whitespace "1.0.0"
26+
terser "4.6.3"
2327

24-
"@ampproject/toolbox-runtime-version@^1.1.1":
25-
version "1.1.1"
26-
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-runtime-version/-/toolbox-runtime-version-1.1.1.tgz#628fe5091db4f90b68960620e22ad64f9f2563bd"
27-
integrity sha512-ibmw5p+0Sz+wingbX/Dyboe8a0+XDkMfFGSM7KFE0h2z3Op9MADup8ZPLeHT54Z7cYKmB6ob60FVHtQQDhEXNw==
28+
"@ampproject/toolbox-runtime-version@^2.0.0":
29+
version "2.0.0"
30+
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-runtime-version/-/toolbox-runtime-version-2.0.0.tgz#01202be490ce8baf8654127de35dff5f93e922a0"
31+
integrity sha512-7XDlo7l4Ozpc/XWmJeYNF0ZfXCy7Vdh07spN+xEOst8gF99yaavZRNVkdgyTxLR3BpY9RIqqhSs6OxfkOKlRZQ==
2832
dependencies:
29-
"@ampproject/toolbox-core" "^1.1.1"
33+
"@ampproject/toolbox-core" "^2.0.0"
3034

31-
"@ampproject/toolbox-script-csp@^1.1.1":
32-
version "1.1.1"
33-
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-script-csp/-/toolbox-script-csp-1.1.1.tgz#0b049a1c86c99f300162a10e1b9ce83c6e354a45"
34-
integrity sha512-gACGfsVKinCy/977FSrlVgo6jxTZ0lcTCvCnRlNwvSOcxJVm+jJR3sP7/F43fpak9Gsq/EwFaatfnNMbunPc+w==
35+
"@ampproject/toolbox-script-csp@^2.0.0":
36+
version "2.0.0"
37+
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-script-csp/-/toolbox-script-csp-2.0.0.tgz#7aa55fb693749e657b63985ebe49e1d3489a8b79"
38+
integrity sha512-9mW3yiKwjORi0ViuayphVZii9MwiPhneZGZWy+kN44xr3SpN7iQC52/WWWTBOZX9z1zaUh8DqGc//VFY5ILSAw==
39+
40+
"@ampproject/toolbox-validator-rules@^2.0.0":
41+
version "2.0.0"
42+
resolved "https://registry.yarnpkg.com/@ampproject/toolbox-validator-rules/-/toolbox-validator-rules-2.0.0.tgz#eefdce2f7f773e4675d72c7a1068c70105a647c3"
43+
integrity sha512-5f2IvT1m/zNtqfNXFE9V4ZtKofIttST65QL1wf4lKjBhJsbZgAAeR/u4DgOvftRVRmgn6IpUiWY1pt9xWaI5yA==
44+
dependencies:
45+
cross-fetch "3.0.4"
3546

3647
"@babel/code-frame@^7.0.0", "@babel/code-frame@^7.5.5":
3748
version "7.5.5"
@@ -5352,6 +5363,14 @@ [email protected]:
53525363
dependencies:
53535364
cross-spawn "^7.0.0"
53545365

5366+
5367+
version "3.0.4"
5368+
resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.0.4.tgz#7bef7020207e684a7638ef5f2f698e24d9eb283c"
5369+
integrity sha512-MSHgpjQqgbT/94D4CyADeNoYh52zMkCX4pcJvPP5WqPsLFMKjr2TCMg381ox5qI0ii2dPwaLx/00477knXqXVw==
5370+
dependencies:
5371+
node-fetch "2.6.0"
5372+
whatwg-fetch "3.0.0"
5373+
53555374
[email protected], cross-spawn@^6.0.0, cross-spawn@^6.0.5:
53565375
version "6.0.5"
53575376
resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.5.tgz#4a5ec7c64dfae22c3a14124dbacdee846d80cbc4"
@@ -6084,7 +6103,7 @@ doctrine@^3.0.0:
60846103
dependencies:
60856104
esutils "^2.0.2"
60866105

6087-
dom-serializer@0:
6106+
dom-serializer@0, dom-serializer@^0.2.1:
60886107
version "0.2.2"
60896108
resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.2.2.tgz#1afb81f533717175d478655debc5e332d9f9bb51"
60906109
integrity sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==
@@ -6127,6 +6146,13 @@ domexception@^1.0.1:
61276146
dependencies:
61286147
webidl-conversions "^4.0.2"
61296148

6149+
[email protected], domhandler@^3.0.0:
6150+
version "3.0.0"
6151+
resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-3.0.0.tgz#51cd13efca31da95bbb0c5bee3a48300e333b3e9"
6152+
integrity sha512-eKLdI5v9m67kbXQbJSNn1zjh0SDzvzWVWtX+qEI3eMjZw8daH9k8rlj1FZY9memPwjiskQFbe7vHVVJIAqoEhw==
6153+
dependencies:
6154+
domelementtype "^2.0.1"
6155+
61306156
domhandler@^2.3.0:
61316157
version "2.4.2"
61326158
resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-2.4.2.tgz#8805097e933d65e85546f726d60f5eb88b44f803"
@@ -6142,6 +6168,15 @@ [email protected]:
61426168
dom-serializer "0"
61436169
domelementtype "1"
61446170

6171+
[email protected], domutils@^2.0.0:
6172+
version "2.0.0"
6173+
resolved "https://registry.yarnpkg.com/domutils/-/domutils-2.0.0.tgz#15b8278e37bfa8468d157478c58c367718133c08"
6174+
integrity sha512-n5SelJ1axbO636c2yUtOGia/IcJtVtlhQbFiVDBZHKV5ReJO1ViX7sFEemtuyoAnBxk5meNSYgA8V4s0271efg==
6175+
dependencies:
6176+
dom-serializer "^0.2.1"
6177+
domelementtype "^2.0.1"
6178+
domhandler "^3.0.0"
6179+
61456180
domutils@^1.5.1, domutils@^1.7.0:
61466181
version "1.7.0"
61476182
resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.7.0.tgz#56ea341e834e06e6748af7a1cb25da67ea9f8c2a"
@@ -8093,6 +8128,16 @@ html-entities@^1.2.0:
80938128
resolved "https://registry.yarnpkg.com/html-entities/-/html-entities-1.2.1.tgz#0df29351f0721163515dfb9e5543e5f6eed5162f"
80948129
integrity sha1-DfKTUfByEWNRXfueVUPl9u7VFi8=
80958130

8131+
8132+
version "4.0.0"
8133+
resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-4.0.0.tgz#6034658db65b7713a572a9ebf79f650832dceec8"
8134+
integrity sha512-cChwXn5Vam57fyXajDtPXL1wTYc8JtLbr2TN76FYu05itVVVealxLowe2B3IEznJG4p9HAYn/0tJaRlGuEglFQ==
8135+
dependencies:
8136+
domelementtype "^2.0.1"
8137+
domhandler "^3.0.0"
8138+
domutils "^2.0.0"
8139+
entities "^2.0.0"
8140+
80968141
htmlparser2@^3.9.1:
80978142
version "3.10.1"
80988143
resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-3.10.1.tgz#bd679dc3f59897b6a34bb10749c855bb53a9392f"
@@ -11172,6 +11217,11 @@ nopt@^4.0.1:
1117211217
abbrev "1"
1117311218
osenv "^0.1.4"
1117411219

11220+
11221+
version "1.0.0"
11222+
resolved "https://registry.yarnpkg.com/normalize-html-whitespace/-/normalize-html-whitespace-1.0.0.tgz#5e3c8e192f1b06c3b9eee4b7e7f28854c7601e34"
11223+
integrity sha512-9ui7CGtOOlehQu0t/OhhlmDyc71mKVlv+4vF+me4iZLPrNtRL2xoquEdfZxasC/bdQi/Hr3iTrpyRKIG+ocabA==
11224+
1117511225
normalize-package-data@^2.0.0, normalize-package-data@^2.3.0, normalize-package-data@^2.3.2, normalize-package-data@^2.3.4, normalize-package-data@^2.3.5, normalize-package-data@^2.4.0:
1117611226
version "2.5.0"
1117711227
resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz#e66db1838b200c1dfc233225d12cb36520e234a8"
@@ -11949,28 +11999,11 @@ parse-url@^5.0.0:
1194911999
parse-path "^4.0.0"
1195012000
protocols "^1.4.0"
1195112001

11952-
11953-
version "5.1.0"
11954-
resolved "https://registry.yarnpkg.com/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-5.1.0.tgz#a8244ee12bbd6b8937ad2a16ea43fe348aebcc86"
11955-
integrity sha512-OrI4DNmghGcwDB3XN8FKKN7g5vBmau91uqj+VYuwuj/r6GhFBMBNymsM+Z9z+Z1p4HHgI0UuQirQRgh3W5d88g==
11956-
dependencies:
11957-
parse5 "^5.1.0"
11958-
1195912002
1196012003
version "4.0.0"
1196112004
resolved "https://registry.yarnpkg.com/parse5/-/parse5-4.0.0.tgz#6d78656e3da8d78b4ec0b906f7c08ef1dfe3f608"
1196212005
integrity sha512-VrZ7eOd3T1Fk4XWNXMgiGBK/z0MG48BWG2uQNU4I72fkQuKUTZpl+u9k+CxEG0twMVzSmXEEz12z5Fnw1jIQFA==
1196312006

11964-
11965-
version "5.1.0"
11966-
resolved "https://registry.yarnpkg.com/parse5/-/parse5-5.1.0.tgz#c59341c9723f414c452975564c7c00a68d58acd2"
11967-
integrity sha512-fxNG2sQjHvlVAYmzBZS9YlDp6PTSSDwa98vkD4QgVDDCAo84z5X1t5XyJQ62ImdLXx5NdIIfihey6xpum9/gRQ==
11968-
11969-
parse5@^5.1.0:
11970-
version "5.1.1"
11971-
resolved "https://registry.yarnpkg.com/parse5/-/parse5-5.1.1.tgz#f68e4e5ba1852ac2cadc00f4555fff6c2abb6178"
11972-
integrity sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug==
11973-
1197412007
parseurl@~1.3.3:
1197512008
version "1.3.3"
1197612009
resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4"
@@ -15859,6 +15892,15 @@ [email protected]:
1585915892
source-map "~0.6.1"
1586015893
source-map-support "~0.5.12"
1586115894

15895+
15896+
version "4.6.3"
15897+
resolved "https://registry.yarnpkg.com/terser/-/terser-4.6.3.tgz#e33aa42461ced5238d352d2df2a67f21921f8d87"
15898+
integrity sha512-Lw+ieAXmY69d09IIc/yqeBqXpEQIpDGZqT34ui1QWXIUpR2RjbqEkT8X7Lgex19hslSqcWM5iMN2kM11eMsESQ==
15899+
dependencies:
15900+
commander "^2.20.0"
15901+
source-map "~0.6.1"
15902+
source-map-support "~0.5.12"
15903+
1586215904
terser@^3.8.2:
1586315905
version "3.17.0"
1586415906
resolved "https://registry.yarnpkg.com/terser/-/terser-3.17.0.tgz#f88ffbeda0deb5637f9d24b0da66f4e15ab10cb2"

0 commit comments

Comments
 (0)