/** * Copyright 2018 Google Inc. All rights reserved. * Modifications copyright (c) Microsoft Corporation. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { test as it, expect } from './config/pageTest'; import { attachFrame, detachFrame } from './config/utils'; import type { Frame } from '../src/client/frame'; it('should have different execution contexts', async ({ page, server }) => { await page.goto(server.EMPTY_PAGE); await attachFrame(page, 'frame1', server.EMPTY_PAGE); expect(page.frames().length).toBe(2); await page.frames()[0].evaluate(() => window['FOO'] = 'foo'); await page.frames()[1].evaluate(() => window['FOO'] = 'bar'); expect(await page.frames()[0].evaluate(() => window['FOO'])).toBe('foo'); expect(await page.frames()[1].evaluate(() => window['FOO'])).toBe('bar'); }); it('should have correct execution contexts', async ({ page, server }) => { await page.goto(server.PREFIX + '/frames/one-frame.html'); expect(page.frames().length).toBe(2); expect(await page.frames()[0].evaluate(() => document.body.textContent.trim())).toBe(''); expect(await page.frames()[1].evaluate(() => document.body.textContent.trim())).toBe(`Hi, I'm frame`); }); function expectContexts(pageImpl, count, isChromium) { if (isChromium) expect(pageImpl._delegate._mainFrameSession._contextIdToContext.size).toBe(count); else expect(pageImpl._delegate._contextIdToContext.size).toBe(count); } it('should dispose context on navigation', async ({ page, server, toImpl, isChromium, mode }) => { it.skip(mode !== 'default'); await page.goto(server.PREFIX + '/frames/one-frame.html'); expect(page.frames().length).toBe(2); expectContexts(toImpl(page), 4, isChromium); await page.goto(server.EMPTY_PAGE); expectContexts(toImpl(page), 2, isChromium); }); it('should dispose context on cross-origin navigation', async ({ page, server, toImpl, isChromium, mode }) => { it.skip(mode !== 'default'); await page.goto(server.PREFIX + '/frames/one-frame.html'); expect(page.frames().length).toBe(2); expectContexts(toImpl(page), 4, isChromium); await page.goto(server.CROSS_PROCESS_PREFIX + '/empty.html'); expectContexts(toImpl(page), 2, isChromium); }); it('should execute after cross-site navigation', async ({ page, server }) => { await page.goto(server.EMPTY_PAGE); const mainFrame = page.mainFrame(); expect(await mainFrame.evaluate(() => window.location.href)).toContain(server.EMPTY_PAGE); await page.goto(server.CROSS_PROCESS_PREFIX + '/empty.html'); expect(await mainFrame.evaluate(() => window.location.href)).toContain(server.CROSS_PROCESS_PREFIX); }); it('should not allow cross-frame js handles', async ({ page, server }) => { // TODO: this should actually be possible because frames script each other, // but protocol implementations do not support this. For now, assume current // behavior. await page.goto(server.PREFIX + '/frames/one-frame.html'); const handle = await page.evaluateHandle(() => { const iframe = document.querySelector('iframe'); const foo = { bar: 'baz' }; iframe.contentWindow['__foo'] = foo; return foo; }); const childFrame = page.mainFrame().childFrames()[0]; const childResult = await childFrame.evaluate(() => window['__foo']); expect(childResult).toEqual({ bar: 'baz' }); const error = await childFrame.evaluate(foo => foo.bar, handle).catch(e => e); expect(error.message).toContain('JSHandles can be evaluated only in the context they were created!'); }); it('should allow cross-frame element handles', async ({ page, server }) => { await page.goto(server.PREFIX + '/frames/one-frame.html'); const bodyHandle = await page.mainFrame().childFrames()[0].$('body'); const result = await page.evaluate(body => body.innerHTML, bodyHandle); expect(result.trim()).toBe('
Hi, I\'m frame
'); }); it('should not allow cross-frame element handles when frames do not script each other', async ({ page, server, isElectron, isAndroid }) => { it.skip(isAndroid, 'No cross-process on Android'); it.fixme(isElectron); await page.goto(server.EMPTY_PAGE); const frame = await attachFrame(page, 'frame1', server.CROSS_PROCESS_PREFIX + '/empty.html'); const bodyHandle = await frame.$('body'); const error = await page.evaluate(body => body.innerHTML, bodyHandle).catch(e => e); expect(error.message).toContain('Unable to adopt element handle from a different document'); }); it('should throw for detached frames', async ({page, server}) => { const frame1 = await attachFrame(page, 'frame1', server.EMPTY_PAGE); await detachFrame(page, 'frame1'); let error = null; await frame1.evaluate(() => 7 * 8).catch(e => error = e); expect(error.message).toContain('Execution Context is not available in detached frame'); }); it('should be isolated between frames', async ({page, server}) => { await page.goto(server.EMPTY_PAGE); await attachFrame(page, 'frame1', server.EMPTY_PAGE); expect(page.frames().length).toBe(2); const [frame1, frame2] = page.frames(); expect(frame1 !== frame2).toBeTruthy(); await Promise.all([ frame1.evaluate(() => window['a'] = 1), frame2.evaluate(() => window['a'] = 2) ]); const [a1, a2] = await Promise.all([ frame1.evaluate(() => window['a']), frame2.evaluate(() => window['a']) ]); expect(a1).toBe(1); expect(a2).toBe(2); }); it('should work in iframes that failed initial navigation', async ({page, browserName}) => { it.fail(browserName === 'chromium'); it.fixme(browserName === 'firefox'); // - Firefox does not report domcontentloaded for the iframe. // - Chromium and Firefox report empty url. // - Chromium does not report main/utility worlds for the iframe. await page.setContent(` `, { waitUntil: 'domcontentloaded' }); // Note: Chromium/Firefox never report 'load' event for the iframe. await page.evaluate(() => { const iframe = document.querySelector('iframe'); const div = iframe.contentDocument.createElement('div'); iframe.contentDocument.body.appendChild(div); }); expect(page.frames()[1].url()).toBe('about:blank'); // Main world should work. expect(await page.frames()[1].evaluate(() => window.location.href)).toBe('about:blank'); // Utility world should work. expect(await page.frames()[1].$('div')).toBeTruthy(); }); it('should work in iframes that interrupted initial javascript url navigation', async ({page, server, browserName}) => { it.fixme(browserName === 'chromium'); // Chromium does not report isolated world for the iframe. await page.goto(server.EMPTY_PAGE); await page.evaluate(() => { const iframe = document.createElement('iframe'); iframe.src = 'javascript:""'; document.body.appendChild(iframe); iframe.contentDocument.open(); iframe.contentDocument.write('
hello
'); iframe.contentDocument.close(); }); // Main world should work. expect(await page.frames()[1].evaluate(() => window.top.location.href)).toBe(server.EMPTY_PAGE); // Utility world should work. expect(await page.frames()[1].$('div')).toBeTruthy(); }); it('evaluateHandle should work', async ({page, server}) => { await page.goto(server.EMPTY_PAGE); const mainFrame = page.mainFrame(); const windowHandle = await mainFrame.evaluateHandle(() => window); expect(windowHandle).toBeTruthy(); }); it('evaluateInUtility should work', async ({page}) => { await page.setContent('hello'); const mainFrame = page.mainFrame() as any as Frame; await mainFrame.evaluate(() => window['foo'] = 42); expect(await mainFrame.evaluate(() => window['foo'])).toBe(42); expect(await mainFrame._evaluateInUtility(() => window['foo'])).toBe(undefined); expect(await mainFrame._evaluateInUtility(() => document.body.textContent)).toBe('hello'); }); it('evaluateHandleInUtility should work', async ({page}) => { await page.setContent('hello'); const mainFrame = page.mainFrame() as any as Frame; await mainFrame.evaluate(() => window['foo'] = 42); expect(await mainFrame.evaluate(() => window['foo'])).toBe(42); const handle1 = await mainFrame._evaluateHandleInUtility(() => window['foo']); expect(await handle1.jsonValue()).toBe(undefined); const handle2 = await mainFrame._evaluateHandleInUtility(() => document.body); expect(await handle2.evaluate(body => body.textContent)).toBe('hello'); });