|  | <!DOCTYPE html> | 
|  | <script src="/resources/testharness.js"></script> | 
|  | <script src="/resources/testharnessreport.js"></script> | 
|  | <script src="/common/get-host-info.sub.js"></script> | 
|  | <body> | 
|  | <script> | 
|  | host_info = get_host_info(); | 
|  |  | 
|  | function verifyNumberOfDownloads(url, number) { | 
|  | var numDownloads = 0; | 
|  | let absoluteURL = new URL(url, location.href).href; | 
|  | performance.getEntriesByName(absoluteURL).forEach(entry => { | 
|  | if (entry.transferSize > 0) { | 
|  | numDownloads++; | 
|  | } | 
|  | }); | 
|  | assert_equals(numDownloads, number, url); | 
|  | } | 
|  |  | 
|  | function attachAndWaitForLoad(element) { | 
|  | return new Promise((resolve, reject) => { | 
|  | element.onload = resolve; | 
|  | element.onerror = reject; | 
|  | document.body.appendChild(element); | 
|  | }); | 
|  | } | 
|  |  | 
|  | function attachAndWaitForError(element) { | 
|  | return new Promise((resolve, reject) => { | 
|  | element.onload = reject; | 
|  | element.onerror = resolve; | 
|  | document.body.appendChild(element); | 
|  | }); | 
|  | } | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/dummy.js'; | 
|  | return attachAndWaitForLoad(link).then(() => { | 
|  | verifyNumberOfDownloads('resources/dummy.js', 1); | 
|  |  | 
|  | // Verify that <script> doesn't fetch the module again. | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.src = 'resources/dummy.js'; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads('resources/dummy.js', 1); | 
|  | }); | 
|  | }, 'link rel=modulepreload'); | 
|  |  | 
|  | /** | 
|  | * Begin tests to ensure crossorigin value behaves the same on | 
|  | * link rel=modulepreload as it does script elements. | 
|  | */ | 
|  | promise_test(function(t) { | 
|  | document.cookie = 'same=1'; | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.crossorigin = 'anonymous'; | 
|  | link.href = 'resources/dummy.js?sameOriginAnonymous'; | 
|  | return attachAndWaitForLoad(link).then(() => { | 
|  | verifyNumberOfDownloads('resources/dummy.js?sameOriginAnonymous', 1); | 
|  |  | 
|  | // Verify that <script> doesn't fetch the module again. | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.crossorigin = 'anonymous'; | 
|  | script.src = 'resources/dummy.js?sameOriginAnonymous'; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads('resources/dummy.js?sameOriginAnonymous', 1); | 
|  | }); | 
|  | }, 'same-origin link rel=modulepreload crossorigin=anonymous'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.crossorigin = 'use-credentials'; | 
|  | link.href = 'resources/dummy.js?sameOriginUseCredentials'; | 
|  | return attachAndWaitForLoad(link).then(() => { | 
|  | verifyNumberOfDownloads('resources/dummy.js?sameOriginUseCredentials', 1); | 
|  |  | 
|  | // Verify that <script> doesn't fetch the module again. | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.crossorigin = 'use-credentials'; | 
|  | script.src = 'resources/dummy.js?sameOriginUseCredentials'; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads('resources/dummy.js?sameOriginUseCredentials', 1); | 
|  | }); | 
|  | }, 'same-origin link rel=modulepreload crossorigin=use-credentials'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | const setCookiePromise = fetch( | 
|  | `${host_info.HTTP_REMOTE_ORIGIN}/cookies/resources/set-cookie.py?name=cross&path=/preload/`, | 
|  | { | 
|  | mode: 'no-cors', | 
|  | credentials: 'include', | 
|  | }); | 
|  |  | 
|  | return setCookiePromise.then(() => { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = `${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginNone`; | 
|  | return attachAndWaitForLoad(link); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads(`${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginNone`, 1); | 
|  |  | 
|  | // Verify that <script> doesn't fetch the module again. | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.src = `${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginNone`; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads(`${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginNone`, 1); | 
|  | }); | 
|  | }, 'cross-origin link rel=modulepreload'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.crossorigin = 'anonymous'; | 
|  | link.href = `${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginAnonymous`; | 
|  | return attachAndWaitForLoad(link).then(() => { | 
|  | verifyNumberOfDownloads(`${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginAnonymous`, 1); | 
|  |  | 
|  | // Verify that <script> doesn't fetch the module again. | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.crossorigin = 'anonymous'; | 
|  | script.src = `${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginAnonymous`; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads(`${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginAnonymous`, 1); | 
|  | }); | 
|  | }, 'cross-origin link rel=modulepreload crossorigin=anonymous'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.crossorigin = 'use-credentials'; | 
|  | link.href = `${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginUseCredentials`; | 
|  | return attachAndWaitForLoad(link).then(() => { | 
|  | verifyNumberOfDownloads(`${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginUseCredentials`, 1); | 
|  |  | 
|  | // Verify that <script> doesn't fetch the module again. | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.crossorigin = 'use-credentials'; | 
|  | script.src = `${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginUseCredentials`; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads(`${host_info.HTTP_REMOTE_ORIGIN}/preload/resources/cross-origin-module.py?crossOriginUseCredentials`, 1); | 
|  | }); | 
|  | }, 'cross-origin link rel=modulepreload crossorigin=use-credentials'); | 
|  | /** | 
|  | * End link rel=modulepreload crossorigin attribute tests. | 
|  | */ | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/module1.js'; | 
|  | return attachAndWaitForLoad(link).then(() => { | 
|  | verifyNumberOfDownloads('resources/module1.js', 1); | 
|  | // The load event fires before (optional) submodules fetch. | 
|  | verifyNumberOfDownloads('resources/module2.js', 0); | 
|  |  | 
|  | var script = document.createElement('script'); | 
|  | script.type = 'module'; | 
|  | script.src = 'resources/module1.js'; | 
|  | return attachAndWaitForLoad(script); | 
|  | }).then(() => { | 
|  | verifyNumberOfDownloads('resources/module1.js', 1); | 
|  | verifyNumberOfDownloads('resources/module2.js', 1); | 
|  | }); | 
|  | }, 'link rel=modulepreload with submodules'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/syntax-error.js'; | 
|  | return attachAndWaitForLoad(link); | 
|  | }, 'link rel=modulepreload for a module with syntax error'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/not-exist.js'; | 
|  | return attachAndWaitForError(link); | 
|  | }, 'link rel=modulepreload for a module with network error'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = null; | 
|  | return attachAndWaitForError(link); | 
|  | }, 'link rel=modulepreload with bad href attribute'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/module1.js?as-script'; | 
|  | link.as = 'script' | 
|  | return attachAndWaitForLoad(link); | 
|  | }, 'link rel=modulepreload as=script'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/module1.js?as-image'; | 
|  | link.as = 'image' | 
|  | return attachAndWaitForError(link); | 
|  | }, 'link rel=modulepreload with invalid as= value'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/module1.js?integrity-match'; | 
|  | link.integrity = 'sha256-ZPBZ+J9CiHzZXaBBluSeCpjzuTUkT+rSWIdXUV3AtVo=' | 
|  | return attachAndWaitForLoad(link); | 
|  | }, 'link rel=modulepreload with integrity match'); | 
|  |  | 
|  | promise_test(function(t) { | 
|  | var link = document.createElement('link'); | 
|  | link.rel = 'modulepreload'; | 
|  | link.href = 'resources/module1.js?integrity-doesnotmatch'; | 
|  | link.integrity = 'sha384-doesnotmatch' | 
|  | return attachAndWaitForError(link); | 
|  | }, 'link rel=modulepreload with integrity mismatch'); | 
|  |  | 
|  | </script> | 
|  | </body> |