diff --git a/.eslintrc b/.eslintrc index a149d140..13ed0463 100644 --- a/.eslintrc +++ b/.eslintrc @@ -1,5 +1,8 @@ { "root": true, + + "parser": "babel-eslint", + "parserOptions": { "ecmaVersion": 6, }, @@ -429,7 +432,7 @@ "spaced-comment": [2, "always"], // Require "use strict" to be defined globally in the script. - "strict": [2, "global"], + "strict": [0, "global"], // Allow vars to be declared anywhere in the scope. "vars-on-top": 0, diff --git a/.travis.yml b/.travis.yml index af9f2801..19920241 100644 --- a/.travis.yml +++ b/.travis.yml @@ -11,6 +11,10 @@ script: - echo "RE-RUN tests on the webpack and browserify bundled files" && npm install -g browserify webpack && ./test/run-module-bundlers-smoketests.sh +- export DISPLAY=:99.0 +- sh -e /etc/init.d/xvfb start +- echo "RUN integration tests on chrome" && + ./test/run-chrome-smoketests.sh after_script: npm run publish-coverage diff --git a/package.json b/package.json index 616e3c57..5db72286 100644 --- a/package.json +++ b/package.json @@ -17,10 +17,14 @@ }, "homepage": "https://github.com/mozilla/webextension-polyfill", "devDependencies": { + "async-wait-until": "^1.1.5", + "babel-eslint": "^8.0.1", "babel-plugin-transform-es2015-modules-umd": "^6.24.1", - "babel-preset-babili": "0.0.10", + "babel-preset-babili": "^0.0.10", + "babel-preset-es2017": "^6.24.1", "chai": "^3.5.0", - "eslint": "3.9.1", + "eslint": "^3.9.1", + "finalhandler": "^1.1.0", "grunt": "^1.0.1", "grunt-babel": "^6.0.0", "grunt-contrib-concat": "^1.0.1", @@ -31,6 +35,8 @@ "jsdom": "^9.6.0", "mocha": "^3.1.0", "nyc": "^8.3.1", + "puppeteer": "^0.10.2", + "serve-static": "^1.13.1", "sinon": "^1.17.6" }, "nyc": { @@ -47,6 +53,7 @@ "publish-coverage": "grunt coveralls", "test": "mocha", "test-coverage": "COVERAGE=y nyc mocha", - "test-minified": "TEST_MINIFIED_POLYFILL=1 mocha" + "test-minified": "TEST_MINIFIED_POLYFILL=1 mocha", + "test-integration": "mocha -r test/mocha-babel test/integration/test-*" } } diff --git a/test/fixtures/index.html b/test/fixtures/index.html new file mode 100644 index 00000000..151a140a --- /dev/null +++ b/test/fixtures/index.html @@ -0,0 +1,10 @@ + + + + Browser Polyfill Test Page + + + +

Browser Polyfill Test Page

+ + diff --git a/test/fixtures/runtime-messaging-extension/background.js b/test/fixtures/runtime-messaging-extension/background.js new file mode 100644 index 00000000..db8d4d23 --- /dev/null +++ b/test/fixtures/runtime-messaging-extension/background.js @@ -0,0 +1,11 @@ +const {name} = browser.runtime.getManifest(); + +console.log(name, "background page loaded"); + +browser.runtime.onMessage.addListener((msg, sender) => { + console.log(name, "background received msg", {msg, sender}); + + return Promise.resolve("background page reply"); +}); + +console.log(name, "background page ready to receive a content script message..."); diff --git a/test/fixtures/runtime-messaging-extension/content.js b/test/fixtures/runtime-messaging-extension/content.js new file mode 100644 index 00000000..435d9316 --- /dev/null +++ b/test/fixtures/runtime-messaging-extension/content.js @@ -0,0 +1,9 @@ +const {name} = browser.runtime.getManifest(); + +console.log(name, "content script loaded"); + +browser.runtime.sendMessage("content script message").then(reply => { + console.log(name, "content script received reply", {reply}); +}); + +console.log(name, "content script message sent"); diff --git a/test/fixtures/runtime-messaging-extension/manifest.json b/test/fixtures/runtime-messaging-extension/manifest.json new file mode 100644 index 00000000..084a4230 --- /dev/null +++ b/test/fixtures/runtime-messaging-extension/manifest.json @@ -0,0 +1,24 @@ +{ + "manifest_version": 2, + "name": "test-extension-runtime-messaging", + "version": "0.1", + "description": "test-extension-runtime-messaging", + "content_scripts": [ + { + "matches": [ + "http://localhost/*" + ], + "js": [ + "browser-polyfill.js", + "content.js" + ] + } + ], + "permissions": [], + "background": { + "scripts": [ + "browser-polyfill.js", + "background.js" + ] + } +} diff --git a/test/integration/setup.js b/test/integration/setup.js new file mode 100644 index 00000000..a6de495a --- /dev/null +++ b/test/integration/setup.js @@ -0,0 +1,21 @@ +const finalhandler = require("finalhandler"); +const http = require("http"); +const serveStatic = require("serve-static"); + +exports.createHTTPServer = async (path) => { + var serve = serveStatic(path); + + var server = http.createServer((req, res) => { + serve(req, res, finalhandler(req, res)); + }); + + return new Promise((resolve, reject) => { + server.listen((err) => { + if (err) { + reject(err); + } else { + resolve(server); + } + }); + }); +}; diff --git a/test/integration/test-runtime-messaging-on-chrome.js b/test/integration/test-runtime-messaging-on-chrome.js new file mode 100644 index 00000000..a470de99 --- /dev/null +++ b/test/integration/test-runtime-messaging-on-chrome.js @@ -0,0 +1,92 @@ +"use strict"; + +const path = require("path"); + +const waitUntil = require("async-wait-until"); +const {deepEqual} = require("chai").assert; +const puppeteer = require("puppeteer"); + +const {createHTTPServer} = require("./setup"); + +const fixtureExtensionDirName = "runtime-messaging-extension"; + +const extensionName = require(`../fixtures/${fixtureExtensionDirName}/manifest.json`).name; + +describe("browser.runtime.onMessage/sendMessage", function() { + this.timeout(10000); + + it("works as expected on Chrome", async () => { + const server = await createHTTPServer(path.join(__dirname, "..", "fixtures")); + + const url = `http://localhost:${server.address().port}`; + + const browser = await puppeteer.launch({ + // Chrome Extensions are not currently supported in headless mode. + headless: false, + + // Custom chrome arguments. + args: [ + `--load-extension=${process.env.TEST_EXTENSIONS_PATH}/${fixtureExtensionDirName}`, + ], + }); + + const page = await browser.newPage(); + + const pageConsoleMessages = []; + const pageErrors = []; + + page.on("console", (...args) => { + pageConsoleMessages.push(args); + }); + + page.on("error", (error) => { + pageErrors.push(error); + }); + + await page.goto(url); + + const expectedConsoleMessages = [ + [extensionName, "content script loaded"], + [extensionName, "content script message sent"], + [extensionName, "content script received reply", {"reply": "background page reply"}], + ]; + + const lastExpectedMessage = expectedConsoleMessages.slice(-1).pop(); + + let unexpectedException; + + try { + // Wait until the last expected message has been received. + await waitUntil(() => { + return pageConsoleMessages.filter((msg) => { + return msg[0] === lastExpectedMessage[0] && msg[1] === lastExpectedMessage[1]; + }).length > 0; + }, 5000); + } catch (error) { + // Collect any unexpected exception (e.g. a timeout error raised by waitUntil), + // it will be part of the deepEqual assertion of the results. + unexpectedException = error; + } + + let actualResults = { + consoleMessages: pageConsoleMessages, + unexpectedException, + }; + + let expectedResults = { + consoleMessages: expectedConsoleMessages, + unexpectedException: undefined, + }; + + try { + deepEqual(actualResults, expectedResults, "Got the expected results"); + } finally { + // ensure that we close the browser and the test HTTP server before exiting + // the test, even when the assertions fails. + await Promise.all([ + browser.close(), + new Promise(resolve => server.close(resolve)), + ]); + } + }); +}); diff --git a/test/mocha-babel.js b/test/mocha-babel.js new file mode 100644 index 00000000..a8c29691 --- /dev/null +++ b/test/mocha-babel.js @@ -0,0 +1,3 @@ +require("babel-core/register")({ + presets: ["es2017"], +}); diff --git a/test/run-chrome-smoketests.sh b/test/run-chrome-smoketests.sh new file mode 100755 index 00000000..f91553ed --- /dev/null +++ b/test/run-chrome-smoketests.sh @@ -0,0 +1,21 @@ +echo "\nTest webextension-polyfill from an extension running on chrome" +echo "===============================================" + +export TEST_EXTENSIONS_PATH=/tmp/browser-polyfill-chrome-smoketests + +MARKER_FILE=$TEST_EXTENSIONS_PATH/.created-for-run-chrome-smoketests + +# Check if the marker file exists and then remove the directory. +if [ -f $MARKER_FILE ]; then + rm -fr $TEST_EXTENSIONS_PATH +fi + +## Exits immediately if the directory already exists (which can only happen in a local +## development environment, while this test will usually run on travis). +mkdir $TEST_EXTENSIONS_PATH || exit 1 +touch $MARKER_FILE + +cp -rf test/fixtures/runtime-messaging-extension $TEST_EXTENSIONS_PATH +cp -rf dist/browser-polyfill.js* $TEST_EXTENSIONS_PATH/runtime-messaging-extension/ + +npm run test-integration