From c9ee6667d4f7edbeb838ac0149cf15bc228fad89 Mon Sep 17 00:00:00 2001 From: Ronan Jouchet Date: Sun, 15 Mar 2020 16:50:01 -0400 Subject: [PATCH] Revamp and move to TypeScript (#898) ## Breaking changes - Require **Node >= 8.10.0 and npm 5.6.0** - Move to **Electron 8.1.1**. - That's it. Lots of care went into breaking CLI & programmatic behavior as little as possible. **Please report regressions**. - Known issue: build may fail behind a proxy. Get in touch if you use one: https://github.com/jiahaog/nativefier/issues/907#issuecomment-596144768 ## Changes summary Nativefier didn't get much love recently, to the point that it's becoming hard to run on recent Node, due to old dependencies. Also, some past practices now seem weird, as better expressible by modern JS/TS, discouraging contributions including mine. Addressing this, and one thing leading to another, came a bigger-than-expected revamp, aiming at making Nativefier more **lean, stable, future-proof, user-friendly and dev-friendly**, while **not changing the CLI/programmatic interfaces**. Highlights: - **Require Node>=8**, as imposed by many of our dependencies. Node 8 is twice LTS, and easily available even in conservative Linux distros. No reason not to demand it. - **Default to Electron 8**. - **Bump** all dependencies to latest version, including electron-packager. - **Move to TS**. TS is great. As of today, I see no reason not to use it, and fight interface bugs at runtime rather than at compile time. With that, get rid of everything Babel/Webpack. - **Move away from Gulp**. Gulp's selling point is perf via streaming, but for small builds like Nativefier, npm tasks are plenty good and less dependency bloat. Gulp was the driver for this PR: broken on Node 12, and I didn't feel like just upgrading and keeping it. - Add tons of **verbose logs** everywhere it makes sense, to have a fine & clear trace of the program flow. This will be helpful to debug user-reported issues, and already helped me fix a few bugs. - With better simple logging, get rid of the quirky and buggy progress bar based on package `progress`. Nice logging (minimal by default, the verbose logging mentioned above is only used when passing `--verbose`) is better and one less dependency. - **Dump `async` package**, a relic from old callback-hell early Node. Also dump a few other micro-packages unnecessary now. - A first pass of code **cleanup** thanks to modern JS/TS features: fixes, simplifications, jsdoc type annotations to types, etc. - **Remove GitHub integrations Hound & CodeClimate**, which are more exotic than good'ol'linters, and whose signal-to-noise ratio is too low. - Quality: **Add tests** and add **Windows + macOS CI builds**. Also, add a **manual test script**, helping to quickly verify the hard-to-programatically-test stuff before releases, and limit regressions. - **Fix a very small number of existing bugs**. The goal of this PR was *not* to fix bugs, but to get Nativefier in better shape to do so. Bugfixes will come later. Still, these got addressed: - Add common `Alt`+`Left`/`Right` for previous/next navigation. - Improve #379: fix zoom with `Ctrl` + numpad `+`/`-` - Fix pinch-to-zoom (see https://github.com/jiahaog/nativefier/issues/379#issuecomment-598612128 ) --- .codeclimate.yml | 18 -- .eslintrc.js | 23 ++ .eslintrc.yml | 15 - .gitignore | 5 +- .hound.yml | 7 - .npmignore | 20 +- .travis.yml | 26 +- docs/changelog.md => CHANGELOG.md | 0 README.md | 59 ++-- app/.eslintrc.js | 24 ++ app/.eslintrc.yml | 2 - app/package.json | 33 +- .../contextMenu.js => contextMenu.ts} | 6 +- .../{login/loginWindow.js => loginWindow.ts} | 11 +- .../mainWindow.js => mainWindow.ts} | 289 +++++++++--------- ...pers.test.js => mainWindowHelpers.test.ts} | 4 +- ...nWindowHelpers.js => mainWindowHelpers.ts} | 14 +- app/src/components/{menu/menu.js => menu.ts} | 89 ++---- .../{trayIcon/trayIcon.js => trayIcon.ts} | 23 +- app/src/helpers/helpers.js | 87 ------ .../{helpers.test.js => helpers.test.ts} | 4 +- app/src/helpers/helpers.ts | 78 +++++ .../helpers/{inferFlash.js => inferFlash.ts} | 54 ++-- app/src/{main.js => main.ts} | 48 ++- app/src/{static/preload.js => preload.ts} | 29 +- app/src/static/{login => }/login.css | 0 app/src/static/{login => }/login.html | 0 app/src/static/login.js | 10 + app/src/static/login/login.js | 12 - app/tsconfig.json | 19 ++ docs/development.md | 58 ++-- docs/dock.png | Bin 0 -> 25557 bytes scripts/changelog => docs/generate-changelog | 4 +- docs/manual-test | 61 ++++ docs/release.md | 29 +- {screenshots => docs}/walkthrough.gif | Bin e2e/index.test.js | 83 ----- gulp/build.js | 18 -- gulp/build/build-app.js | 12 - gulp/build/build-cli.js | 9 - gulp/build/build-static.js | 17 -- gulp/helpers/gulp-helpers.js | 29 -- gulp/helpers/src-paths.js | 22 -- gulp/release.js | 11 - gulp/watch.js | 13 - gulpfile.babel.js | 9 - {bin => icon-scripts}/convertToIcns | 0 {bin => icon-scripts}/convertToIco | 0 {bin => icon-scripts}/convertToIconset | 0 {bin => icon-scripts}/convertToPng | 4 +- jest.config.js | 3 - package.json | 146 +++++---- screenshots/dock.png | Bin 27764 -> 0 bytes src/build/buildApp.js | 163 ---------- src/build/buildIcon.ts | 96 ++++++ src/build/buildMain.js | 247 --------------- src/build/buildNativefierApp.ts | 125 ++++++++ src/build/iconBuild.js | 106 ------- src/build/prepareElectronApp.ts | 157 ++++++++++ src/{cli.js => cli.ts} | 160 +++++----- src/constants.js | 5 - src/constants.ts | 13 + src/helpers/convertToIcns.js | 65 ---- src/helpers/convertToIcns.test.js | 40 --- src/helpers/dishonestProgress.js | 70 ----- src/helpers/helpers.js | 107 ------- src/helpers/helpers.ts | 141 +++++++++ src/helpers/iconShellHelpers.js | 102 ------- src/helpers/iconShellHelpers.ts | 89 ++++++ src/helpers/packagerConsole.js | 31 -- src/index.js | 6 - src/infer/index.js | 4 - src/infer/inferIcon.js | 130 -------- src/infer/inferIcon.ts | 111 +++++++ src/infer/{inferOs.js => inferOs.ts} | 15 +- src/infer/inferTitle.js | 25 -- src/infer/inferTitle.test.js | 21 -- src/infer/inferTitle.test.ts | 19 ++ src/infer/inferTitle.ts | 23 ++ src/infer/inferUserAgent.js | 69 ----- src/infer/inferUserAgent.test.js | 37 --- src/infer/inferUserAgent.test.ts | 29 ++ src/infer/inferUserAgent.ts | 82 +++++ src/integration-test.ts | 57 ++++ src/jestSetupFiles.ts | 7 + src/main.ts | 20 ++ src/options/asyncConfig.js | 22 -- src/options/asyncConfig.test.js | 18 -- src/options/asyncConfig.ts | 12 + src/options/fields/fields.test.ts | 36 +++ src/options/fields/fields.ts | 29 ++ src/options/fields/icon.js | 14 - src/options/fields/icon.test.js | 43 --- src/options/fields/icon.test.ts | 60 ++++ src/options/fields/icon.ts | 28 ++ src/options/fields/index.js | 32 -- src/options/fields/index.test.js | 21 -- src/options/fields/name.js | 26 -- src/options/fields/name.test.js | 93 ------ src/options/fields/name.test.ts | 108 +++++++ src/options/fields/name.ts | 38 +++ src/options/fields/userAgent.js | 9 - src/options/fields/userAgent.test.js | 20 -- src/options/fields/userAgent.test.ts | 26 ++ src/options/fields/userAgent.ts | 22 ++ src/options/model.ts | 56 ++++ src/options/normalizeUrl.js | 26 -- src/options/normalizeUrl.test.js | 17 -- src/options/normalizeUrl.test.ts | 17 ++ src/options/normalizeUrl.ts | 31 ++ src/options/optionsMain.js | 133 -------- src/options/optionsMain.test.js | 17 -- src/options/optionsMain.test.ts | 25 ++ src/options/optionsMain.ts | 167 ++++++++++ src/utils/index.js | 3 - src/utils/sanitizeFilename.js | 16 - ...ename.test.js => sanitizeFilename.test.ts} | 12 +- src/utils/sanitizeFilename.ts | 24 ++ test-resources/test-injection.js | 2 +- tsconfig.json | 20 ++ webpack.config.js | 23 -- 121 files changed, 2423 insertions(+), 2732 deletions(-) delete mode 100644 .codeclimate.yml create mode 100644 .eslintrc.js delete mode 100644 .eslintrc.yml delete mode 100644 .hound.yml rename docs/changelog.md => CHANGELOG.md (100%) create mode 100644 app/.eslintrc.js delete mode 100644 app/.eslintrc.yml rename app/src/components/{contextMenu/contextMenu.js => contextMenu.ts} (86%) rename app/src/components/{login/loginWindow.js => loginWindow.ts} (65%) rename app/src/components/{mainWindow/mainWindow.js => mainWindow.ts} (53%) rename app/src/components/{mainWindow/mainWindowHelpers.test.js => mainWindowHelpers.test.ts} (97%) rename app/src/components/{mainWindow/mainWindowHelpers.js => mainWindowHelpers.ts} (79%) rename app/src/components/{menu/menu.js => menu.ts} (76%) rename app/src/components/{trayIcon/trayIcon.js => trayIcon.ts} (71%) delete mode 100644 app/src/helpers/helpers.js rename app/src/helpers/{helpers.test.js => helpers.test.ts} (94%) create mode 100644 app/src/helpers/helpers.ts rename app/src/helpers/{inferFlash.js => inferFlash.ts} (56%) rename app/src/{main.js => main.ts} (79%) rename app/src/{static/preload.js => preload.ts} (62%) rename app/src/static/{login => }/login.css (100%) rename app/src/static/{login => }/login.html (100%) create mode 100644 app/src/static/login.js delete mode 100644 app/src/static/login/login.js create mode 100644 app/tsconfig.json create mode 100644 docs/dock.png rename scripts/changelog => docs/generate-changelog (94%) create mode 100755 docs/manual-test rename {screenshots => docs}/walkthrough.gif (100%) delete mode 100644 e2e/index.test.js delete mode 100644 gulp/build.js delete mode 100644 gulp/build/build-app.js delete mode 100644 gulp/build/build-cli.js delete mode 100644 gulp/build/build-static.js delete mode 100644 gulp/helpers/gulp-helpers.js delete mode 100644 gulp/helpers/src-paths.js delete mode 100644 gulp/release.js delete mode 100644 gulp/watch.js delete mode 100644 gulpfile.babel.js rename {bin => icon-scripts}/convertToIcns (100%) rename {bin => icon-scripts}/convertToIco (100%) rename {bin => icon-scripts}/convertToIconset (100%) rename {bin => icon-scripts}/convertToPng (85%) delete mode 100644 jest.config.js delete mode 100644 screenshots/dock.png delete mode 100644 src/build/buildApp.js create mode 100644 src/build/buildIcon.ts delete mode 100644 src/build/buildMain.js create mode 100644 src/build/buildNativefierApp.ts delete mode 100644 src/build/iconBuild.js create mode 100644 src/build/prepareElectronApp.ts rename src/{cli.js => cli.ts} (50%) delete mode 100644 src/constants.js create mode 100644 src/constants.ts delete mode 100644 src/helpers/convertToIcns.js delete mode 100644 src/helpers/convertToIcns.test.js delete mode 100644 src/helpers/dishonestProgress.js delete mode 100644 src/helpers/helpers.js create mode 100644 src/helpers/helpers.ts delete mode 100644 src/helpers/iconShellHelpers.js create mode 100644 src/helpers/iconShellHelpers.ts delete mode 100644 src/helpers/packagerConsole.js delete mode 100644 src/index.js delete mode 100644 src/infer/index.js delete mode 100644 src/infer/inferIcon.js create mode 100644 src/infer/inferIcon.ts rename src/infer/{inferOs.js => inferOs.ts} (63%) delete mode 100644 src/infer/inferTitle.js delete mode 100644 src/infer/inferTitle.test.js create mode 100644 src/infer/inferTitle.test.ts create mode 100644 src/infer/inferTitle.ts delete mode 100644 src/infer/inferUserAgent.js delete mode 100644 src/infer/inferUserAgent.test.js create mode 100644 src/infer/inferUserAgent.test.ts create mode 100644 src/infer/inferUserAgent.ts create mode 100644 src/integration-test.ts create mode 100644 src/jestSetupFiles.ts create mode 100644 src/main.ts delete mode 100644 src/options/asyncConfig.js delete mode 100644 src/options/asyncConfig.test.js create mode 100644 src/options/asyncConfig.ts create mode 100644 src/options/fields/fields.test.ts create mode 100644 src/options/fields/fields.ts delete mode 100644 src/options/fields/icon.js delete mode 100644 src/options/fields/icon.test.js create mode 100644 src/options/fields/icon.test.ts create mode 100644 src/options/fields/icon.ts delete mode 100644 src/options/fields/index.js delete mode 100644 src/options/fields/index.test.js delete mode 100644 src/options/fields/name.js delete mode 100644 src/options/fields/name.test.js create mode 100644 src/options/fields/name.test.ts create mode 100644 src/options/fields/name.ts delete mode 100644 src/options/fields/userAgent.js delete mode 100644 src/options/fields/userAgent.test.js create mode 100644 src/options/fields/userAgent.test.ts create mode 100644 src/options/fields/userAgent.ts create mode 100644 src/options/model.ts delete mode 100644 src/options/normalizeUrl.js delete mode 100644 src/options/normalizeUrl.test.js create mode 100644 src/options/normalizeUrl.test.ts create mode 100644 src/options/normalizeUrl.ts delete mode 100644 src/options/optionsMain.js delete mode 100644 src/options/optionsMain.test.js create mode 100644 src/options/optionsMain.test.ts create mode 100644 src/options/optionsMain.ts delete mode 100644 src/utils/index.js delete mode 100644 src/utils/sanitizeFilename.js rename src/utils/{sanitizeFilename.test.js => sanitizeFilename.test.ts} (68%) create mode 100644 src/utils/sanitizeFilename.ts create mode 100644 tsconfig.json delete mode 100644 webpack.config.js diff --git a/.codeclimate.yml b/.codeclimate.yml deleted file mode 100644 index 64f4747a02..0000000000 --- a/.codeclimate.yml +++ /dev/null @@ -1,18 +0,0 @@ ---- -engines: - csslint: - enabled: true - duplication: - enabled: true - config: - languages: - - javascript - eslint: - enabled: false - fixme: - enabled: true -ratings: - paths: - - "**.js" -exclude_paths: -- test/ diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 0000000000..503225ee81 --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,23 @@ +// # https://github.com/typescript-eslint/typescript-eslint/blob/master/docs/getting-started/linting/README.md +module.exports = { + root: true, + parser: '@typescript-eslint/parser', + parserOptions: { + tsconfigRootDir: __dirname, + project: ['./tsconfig.json'], + }, + plugins: ['@typescript-eslint', 'prettier'], + extends: [ + 'eslint:recommended', + 'prettier', + 'plugin:@typescript-eslint/eslint-recommended', + 'plugin:@typescript-eslint/recommended', + 'plugin:@typescript-eslint/recommended-requiring-type-checking', + ], + rules: { + 'prettier/prettier': 'error', + // TODO remove when done killing anys and making tsc strict + '@typescript-eslint/no-explicit-any': 'off', + '@typescript-eslint/ban-ts-ignore': 'off', + }, +}; diff --git a/.eslintrc.yml b/.eslintrc.yml deleted file mode 100644 index d81db00b75..0000000000 --- a/.eslintrc.yml +++ /dev/null @@ -1,15 +0,0 @@ -extends: - - airbnb-base - - prettier -env: - # TODO: find out how to turn this on only for src/**/*.test.js files - jest: true -plugins: - - import - - prettier -rules: - # TODO: Remove this when we have shifted away from the async package - no-shadow: 'warn' - # Gulpfiles and tests use dev dependencies - import/no-extraneous-dependencies: ['error', { devDependencies: ['gulpfile.babel.js', 'gulp/**/**.js', 'test/**/**.js']}] - prettier/prettier: "error" diff --git a/.gitignore b/.gitignore index d9a44d1022..d615764bb2 100644 --- a/.gitignore +++ b/.gitignore @@ -5,7 +5,7 @@ package-lock.json # ignore compiled lib files -lib/* +lib* app/lib/* built-tests @@ -48,3 +48,6 @@ node_modules *.iml out gen + +# Builds when testing npm pack +nativefier*.tgz diff --git a/.hound.yml b/.hound.yml deleted file mode 100644 index a2b4dcfea0..0000000000 --- a/.hound.yml +++ /dev/null @@ -1,7 +0,0 @@ -eslint: - enabled: true - config_file: .eslintrc.yml - ignore_file: .eslintignore - -jshint: - enabled: false diff --git a/.npmignore b/.npmignore index 90f595fa0b..f1bc61ff24 100644 --- a/.npmignore +++ b/.npmignore @@ -1,7 +1,19 @@ # OSX -.DS_Store - /* !lib/ -!app/lib -!bin +!app/lib/ +!icon-scripts +.DS_Store +.eslintrc.yml +src/ +app/src/ +app/node_modules +*tsconfig.tsbuildinfo +*package-lock.json +*tsconfig.json +*jestSetupFiles* +*-test.js +*-test.js.map +*.test.d.ts +*.test.js +*.test.js.map diff --git a/.travis.yml b/.travis.yml index c83800b9ff..c83c43ba5e 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,21 +1,19 @@ language: node_js -addons: - code_climate: - repo_token: CODE_CLIMATE_TOKEN +os: +- linux +- osx +- windows node_js: -- '11' -- '10' +- '13' # Changing this? Remind to adjust the linter condition below causing linter to run for only one version (for faster CI) +- '12' - '8' -- '7' -- '6' -before_install: -- npm install -g npm@5.8.x install: -- npm run dev-up +- npm install +- npm run build script: -- npm run ci -after_script: -- codeclimate-test-reporter < ./coverage/lcov.info +# Only run linter once, for faster CI +- if [ "$TRAVIS_OS_NAME" = "linux" ] && [ "$TRAVIS_NODE_VERSION" = "13" ]; then npm run lint; fi +- npm test deploy: provider: npm skip_cleanup: true @@ -25,4 +23,4 @@ deploy: on: tags: true repo: jiahaog/nativefier - node: '8' + node: '12' diff --git a/docs/changelog.md b/CHANGELOG.md similarity index 100% rename from docs/changelog.md rename to CHANGELOG.md diff --git a/README.md b/README.md index ef7fe652ec..3520a3e474 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,9 @@ # Nativefier -[![Build Status](https://travis-ci.org/jiahaog/nativefier.svg?branch=development)](https://travis-ci.org/jiahaog/nativefier) -[![Code Climate](https://codeclimate.com/github/jiahaog/nativefier/badges/gpa.svg)](https://codeclimate.com/github/jiahaog/nativefier) +[![Build Status](https://travis-ci.org/jiahaog/nativefier.svg)](https://travis-ci.org/jiahaog/nativefier) [![npm version](https://badge.fury.io/js/nativefier.svg)](https://www.npmjs.com/package/nativefier) -[![Dependency Status](https://david-dm.org/jiahaog/nativefier.svg)](https://david-dm.org/jiahaog/nativefier) -![Dock](screenshots/dock.png) +![Dock](dock.png) You want to make a native wrapper for WhatsApp Web (or any web page). @@ -13,7 +11,7 @@ You want to make a native wrapper for WhatsApp Web (or any web page). nativefier web.whatsapp.com ``` -![Walkthrough](screenshots/walkthrough.gif) +![Walkthrough animation](walkthrough.gif) You're done. @@ -21,33 +19,31 @@ You're done. - [Installation](#installation) - [Usage](#usage) - - [Optional dependencies](#optional-dependencies) - [How it works](#how-it-works) - [Development](docs/development.md) - [License](#license) ## Introduction -Nativefier is a command-line tool to easily create a desktop application for any web site with succinct and minimal configuration. Apps are wrapped by [Electron](http://electron.atom.io) in an OS executable (`.app`, `.exe`, etc.) for use on Windows, macOS and Linux. +Nativefier is a command-line tool to easily create a desktop application for any web site with succinct and minimal configuration. Apps are wrapped by [Electron](https://www.electronjs.org/) in an OS executable (`.app`, `.exe`, etc.) for use on Windows, macOS and Linux. -I did this because I was tired of having to `⌘-tab` or `alt-tab` to my browser and then search through the numerous open tabs when I was using [Facebook Messenger](http://messenger.com) or [Whatsapp Web](http://web.whatsapp.com) ([relevant Hacker News thread](https://news.ycombinator.com/item?id=10930718)). +I did this because I was tired of having to `⌘-tab` or `alt-tab` to my browser and then search through the numerous open tabs when I was using [Facebook Messenger](https://messenger.com) or [Whatsapp Web](https://web.whatsapp.com) ([relevant Hacker News thread](https://news.ycombinator.com/item?id=10930718)). -[Changelog](https://github.com/jiahaog/nativefier/blob/master/docs/changelog.md). [Developer docs](https://github.com/jiahaog/nativefier/blob/master/docs/development.md). +[Changelog](https://github.com/jiahaog/nativefier/blob/master/CHANGELOG.md). [Developer docs](https://github.com/jiahaog/nativefier/blob/master/docs/development.md). -### Features +Features: - Automatically retrieves the correct icon and app name. - JavaScript and CSS injection. -- Flash Support (with [`--flash`](docs/api.md#flash) flag). - Many more, see the [API docs](docs/api.md) or `nativefier --help` ## Installation -### Requirements - - macOS 10.9+ / Windows / Linux -- [Node.js](https://nodejs.org/) `>=6` (4.x may work but is no longer tested, please upgrade) -- See [optional dependencies](#optional-dependencies) for more. +- [Node.js](https://nodejs.org/) `>=8` +- Optional dependencies: + - [ImageMagick](http://www.imagemagick.org/) to convert icons. Make sure `convert` and `identify` are in your `$PATH`. + - [Wine](https://www.winehq.org/) to package Windows apps under non-Windows platforms. Make sure `wine` is in your `$PATH`. ```bash npm install nativefier -g @@ -55,42 +51,23 @@ npm install nativefier -g ## Usage -Creating a native desktop app for [medium.com](http://medium.com): +Creating a native desktop app for [medium.com](https://medium.com): ```bash -nativefier "http://medium.com" +nativefier "medium.com" ``` -Nativefier will intelligently attempt to determine the app name, your OS and processor architecture, among other options. If desired, the app name or other options can be overwritten by specifying the `--name "Medium"` as part of the command line options: +Nativefier will attempt to determine the app name, your OS and processor architecture, among other options. If desired, the app name or other options can be overwritten by specifying the `--name "Medium"` as part of the command line options: ```bash -nativefier --name "Some Awesome App" "http://medium.com" +nativefier --name "Some Awesome App" "medium.com" ``` -Read the [API documentation](docs/api.md) (or `nativefier --help`) for other command line flags and options that can be used to configure the packaged app. - -If you would like high resolution icons to be used, please contribute to the [icon repository](https://github.com/jiahaog/nativefier-icons)! - -**Windows Users:** Take note that the application menu is automatically hidden by default, you can press `alt` on your keyboard to access it. - -**Linux Users:** Do not put spaces if you define the app name yourself with `--name`, as this will cause problems when pinning a packaged app to the launcher. - -## Optional dependencies - -### Icons for Windows apps packaged under non-Windows platforms - -You need [Wine](https://www.winehq.org/) installed; make sure that `wine` is in your `$PATH`. - -### Icon conversion for macOS - -To support conversion of a `.png` or `.ico` into a `.icns` for a packaged macOS app icon (currently only supported on macOS), you need the following dependencies. -* [iconutil](https://developer.apple.com/library/mac/documentation/GraphicsAnimation/Conceptual/HighResolutionOSX/Optimizing/Optimizing.html) (comes with [Xcode](https://developer.apple.com/xcode/)). -* [imagemagick](http://www.imagemagick.org/script/index.php). Make sure `convert` and `identify` are in your `$PATH`. -* If the tools are not found, then Nativefier will fall back to the built-in macOS tool `sips` to perform the conversion, which is more limited. +Read the [API documentation](docs/api.md) (or `nativefier --help`) for other command-line flags that can be used to configure the packaged app. -### Flash +To have high-resolution icons used by default for an app/domain, please contribute to the [icon repository](https://github.com/jiahaog/nativefier-icons)! -[Google Chrome](https://www.google.com/chrome/) is required for flash to be supported; you should pass the path to its embedded Flash plugin to the `--flash` flag. See the [API docs](docs/api.md) for more details. +Note that the application menu is hidden by default for a minimal UI. You can press the `alt` keyboard key to access it. ## How it works diff --git a/app/.eslintrc.js b/app/.eslintrc.js new file mode 100644 index 0000000000..e474023916 --- /dev/null +++ b/app/.eslintrc.js @@ -0,0 +1,24 @@ +// # https://github.com/typescript-eslint/typescript-eslint/blob/master/docs/getting-started/linting/README.md +module.exports = { + parser: '@typescript-eslint/parser', + parserOptions: { + tsconfigRootDir: __dirname, + project: ['./tsconfig.json'], + }, + plugins: ['@typescript-eslint'], + extends: [ + 'eslint:recommended', + 'prettier', + 'plugin:@typescript-eslint/eslint-recommended', + 'plugin:@typescript-eslint/recommended', + 'plugin:@typescript-eslint/recommended-requiring-type-checking', + ], + rules: { + 'prettier/prettier': 'error', + // TODO remove when done killing anys and making tsc strict + '@typescript-eslint/no-explicit-any': 'off', + '@typescript-eslint/ban-ts-ignore': 'off', + '@typescript-eslint/no-use-before-define': 'off', + '@typescript-eslint/explicit-function-return-type': 'off', + }, +}; diff --git a/app/.eslintrc.yml b/app/.eslintrc.yml deleted file mode 100644 index d4c13bf624..0000000000 --- a/app/.eslintrc.yml +++ /dev/null @@ -1,2 +0,0 @@ -settings: - import/core-modules: [ electron ] diff --git a/app/package.json b/app/package.json index 543d35e20d..cb351a1a62 100644 --- a/app/package.json +++ b/app/package.json @@ -3,23 +3,24 @@ "version": "1.0.0", "description": "Placeholder for the nativefier cli to override with a target url", "main": "lib/main.js", - "dependencies": { - "electron-context-menu": "^0.10.0", - "electron-dl": "^1.10.0", - "electron-squirrel-startup": "^1.0.0", - "electron-window-state": "^4.1.1", - "loglevel": "^1.5.1", - "source-map-support": "^0.5.0", - "wurl": "^2.5.2" - }, - "devDependencies": {}, - "scripts": { - "test": "echo \"Error: no test specified\" && exit 1" - }, + "author": "Jia Hao", + "license": "MIT", "keywords": [ "desktop", - "electron" + "electron", + "placeholder" ], - "author": "Jia Hao", - "license": "MIT" + "scripts": {}, + "dependencies": { + "electron-context-menu": "0.x", + "electron-dl": "3.x", + "electron-squirrel-startup": "1.x", + "electron-window-state": "5.x", + "loglevel": "1.x", + "source-map-support": "0.x", + "wurl": "2.x" + }, + "devDependencies": { + "electron": "8.x" + } } diff --git a/app/src/components/contextMenu/contextMenu.js b/app/src/components/contextMenu.ts similarity index 86% rename from app/src/components/contextMenu/contextMenu.js rename to app/src/components/contextMenu.ts index 6cc7758bd3..37eebf99ad 100644 --- a/app/src/components/contextMenu/contextMenu.js +++ b/app/src/components/contextMenu.ts @@ -1,9 +1,9 @@ import { shell } from 'electron'; import contextMenu from 'electron-context-menu'; -function initContextMenu(createNewWindow, createNewTab) { +export function initContextMenu(createNewWindow, createNewTab): void { contextMenu({ - prepend: (params) => { + prepend: (actions, params) => { const items = []; if (params.linkURL) { items.push({ @@ -31,5 +31,3 @@ function initContextMenu(createNewWindow, createNewTab) { }, }); } - -export default initContextMenu; diff --git a/app/src/components/login/loginWindow.js b/app/src/components/loginWindow.ts similarity index 65% rename from app/src/components/login/loginWindow.js rename to app/src/components/loginWindow.ts index ba653c16ac..beb25fa5bc 100644 --- a/app/src/components/login/loginWindow.js +++ b/app/src/components/loginWindow.ts @@ -1,18 +1,19 @@ +import * as path from 'path'; + import { BrowserWindow, ipcMain } from 'electron'; -import path from 'path'; -function createLoginWindow(loginCallback) { +export function createLoginWindow(loginCallback): BrowserWindow { const loginWindow = new BrowserWindow({ width: 300, height: 400, frame: false, resizable: false, webPreferences: { - nodeIntegration: true, + nodeIntegration: true, // TODO work around this; insecure }, }); loginWindow.loadURL( - `file://${path.join(__dirname, '/static/login/login.html')}`, + `file://${path.join(__dirname, '..', 'static/login.html')}`, ); ipcMain.once('login-message', (event, usernameAndPassword) => { @@ -21,5 +22,3 @@ function createLoginWindow(loginCallback) { }); return loginWindow; } - -export default createLoginWindow; diff --git a/app/src/components/mainWindow/mainWindow.js b/app/src/components/mainWindow.ts similarity index 53% rename from app/src/components/mainWindow/mainWindow.js rename to app/src/components/mainWindow.ts index dcff24138c..d25b2bd962 100644 --- a/app/src/components/mainWindow/mainWindow.js +++ b/app/src/components/mainWindow.ts @@ -1,13 +1,10 @@ -import fs from 'fs'; -import path from 'path'; -import { BrowserWindow, shell, ipcMain, dialog } from 'electron'; +import * as fs from 'fs'; +import * as path from 'path'; + +import { BrowserWindow, shell, ipcMain, dialog, Event } from 'electron'; import windowStateKeeper from 'electron-window-state'; -import mainWindowHelpers from './mainWindowHelpers'; -import helpers from '../../helpers/helpers'; -import createMenu from '../menu/menu'; -import initContextMenu from '../contextMenu/contextMenu'; -const { +import { isOSX, linkIsInternal, getCssToInject, @@ -15,13 +12,19 @@ const { getAppIcon, nativeTabsSupported, getCounterValue, -} = helpers; - -const { onNewWindowHelper } = mainWindowHelpers; +} from '../helpers/helpers'; +import { initContextMenu } from './contextMenu'; +import { onNewWindowHelper } from './mainWindowHelpers'; +import { createMenu } from './menu'; const ZOOM_INTERVAL = 0.1; -function maybeHideWindow(window, event, fastQuit, tray) { +function hideWindow( + window: BrowserWindow, + event: Event, + fastQuit: boolean, + tray, +): void { if (isOSX() && !fastQuit) { // this is called when exiting from clicking the cross button on the window event.preventDefault(); @@ -33,66 +36,51 @@ function maybeHideWindow(window, event, fastQuit, tray) { // will close the window on other platforms } -function maybeInjectCss(browserWindow) { +function injectCss(browserWindow: BrowserWindow): void { if (!shouldInjectCss()) { return; } const cssToInject = getCssToInject(); - const injectCss = () => { - browserWindow.webContents.insertCSS(cssToInject); - }; - const onHeadersReceived = (details, callback) => { - injectCss(); - callback({ cancel: false, responseHeaders: details.responseHeaders }); - }; - - browserWindow.webContents.on('did-finish-load', () => { - // remove the injection of css the moment the page is loaded - browserWindow.webContents.session.webRequest.onHeadersReceived(null); - }); - - // on every page navigation inject the css browserWindow.webContents.on('did-navigate', () => { - // we have to inject the css in onHeadersReceived so they're early enough - // will run multiple times, so did-finish-load will remove this handler + // We must inject css early enough; so onHeadersReceived is a good place. + // Will run multiple times, see `did-finish-load` below that unsets this handler. browserWindow.webContents.session.webRequest.onHeadersReceived( { urls: [] }, // Pass an empty filter list; null will not match _any_ urls - onHeadersReceived, + (details, callback) => { + browserWindow.webContents.insertCSS(cssToInject); + callback({ cancel: false, responseHeaders: details.responseHeaders }); + }, ); }); } -function clearCache(browserWindow, targetUrl = null) { +async function clearCache(browserWindow: BrowserWindow): Promise { const { session } = browserWindow.webContents; - session.clearStorageData(() => { - session.clearCache(() => { - if (targetUrl) { - browserWindow.loadURL(targetUrl); - } - }); - }); + await session.clearStorageData(); + await session.clearCache(); } -function setProxyRules(browserWindow, proxyRules) { - browserWindow.webContents.session.setProxy( - { - proxyRules, - }, - () => {}, - ); +function setProxyRules(browserWindow: BrowserWindow, proxyRules): void { + browserWindow.webContents.session.setProxy({ + proxyRules, + pacScript: '', + proxyBypassRules: '', + }); } /** - * - * @param {{}} inpOptions AppArgs from nativefier.json + * @param {{}} nativefierOptions AppArgs from nativefier.json * @param {function} onAppQuit * @param {function} setDockBadge - * @returns {electron.BrowserWindow} */ -function createMainWindow(inpOptions, onAppQuit, setDockBadge) { - const options = Object.assign({}, inpOptions); +export function createMainWindow( + nativefierOptions, + onAppQuit, + setDockBadge, +): BrowserWindow { + const options = { ...nativefierOptions }; const mainWindowState = windowStateKeeper({ defaultWidth: options.width || 1280, defaultHeight: options.height || 800, @@ -105,43 +93,37 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { webPreferences: { javascript: true, plugins: true, - // node globals causes problems with sites like messenger.com - nodeIntegration: false, + nodeIntegration: false, // `true` is *insecure*, and cause trouble with messenger.com webSecurity: !options.insecure, - preload: path.join(__dirname, 'static', 'preload.js'), + preload: path.join(__dirname, '..', 'preload.js'), zoomFactor: options.zoom, }, }; - const browserwindowOptions = Object.assign({}, options.browserwindowOptions); - - const mainWindow = new BrowserWindow( - Object.assign( - { - frame: !options.hideWindowFrame, - width: mainWindowState.width, - height: mainWindowState.height, - minWidth: options.minWidth, - minHeight: options.minHeight, - maxWidth: options.maxWidth, - maxHeight: options.maxHeight, - x: options.x, - y: options.y, - autoHideMenuBar: !options.showMenuBar, - // after webpack path here should reference `resources/app/` - icon: getAppIcon(), - // set to undefined and not false because explicitly setting to false will disable full screen - fullscreen: options.fullScreen || undefined, - // Whether the window should always stay on top of other windows. Default is false. - alwaysOnTop: options.alwaysOnTop, - titleBarStyle: options.titleBarStyle, - show: options.tray !== 'start-in-tray', - backgroundColor: options.backgroundColor, - }, - DEFAULT_WINDOW_OPTIONS, - browserwindowOptions, - ), - ); + const browserwindowOptions = { ...options.browserwindowOptions }; + + const mainWindow = new BrowserWindow({ + frame: !options.hideWindowFrame, + width: mainWindowState.width, + height: mainWindowState.height, + minWidth: options.minWidth, + minHeight: options.minHeight, + maxWidth: options.maxWidth, + maxHeight: options.maxHeight, + x: options.x, + y: options.y, + autoHideMenuBar: !options.showMenuBar, + icon: getAppIcon(), + // set to undefined and not false because explicitly setting to false will disable full screen + fullscreen: options.fullScreen || undefined, + // Whether the window should always stay on top of other windows. Default is false. + alwaysOnTop: options.alwaysOnTop, + titleBarStyle: options.titleBarStyle, + show: options.tray !== 'start-in-tray', + backgroundColor: options.backgroundColor, + ...DEFAULT_WINDOW_OPTIONS, + ...browserwindowOptions, + }); mainWindowState.manage(mainWindow); @@ -151,7 +133,7 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { options.maximize = undefined; try { fs.writeFileSync( - path.join(__dirname, '..', 'nativefier.json'), + path.join(__dirname, '../..', 'nativefier.json'), JSON.stringify(options), ); } catch (exception) { @@ -160,7 +142,7 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { } } - const withFocusedWindow = (block) => { + const withFocusedWindow = (block: (window: BrowserWindow) => void): void => { const focusedWindow = BrowserWindow.getFocusedWindow(); if (focusedWindow) { return block(focusedWindow); @@ -168,75 +150,85 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { return undefined; }; - const adjustWindowZoom = (window, adjustment) => { - window.webContents.getZoomFactor((zoomFactor) => { - window.webContents.setZoomFactor(zoomFactor + adjustment); - }); + const adjustWindowZoom = (window: BrowserWindow, adjustment): void => { + window.webContents.zoomFactor = window.webContents.zoomFactor + adjustment; }; - const onZoomIn = () => { - withFocusedWindow((focusedWindow) => + const onZoomIn = (): void => { + withFocusedWindow((focusedWindow: BrowserWindow) => adjustWindowZoom(focusedWindow, ZOOM_INTERVAL), ); }; - const onZoomOut = () => { - withFocusedWindow((focusedWindow) => + const onZoomOut = (): void => { + withFocusedWindow((focusedWindow: BrowserWindow) => adjustWindowZoom(focusedWindow, -ZOOM_INTERVAL), ); }; - const onZoomReset = () => { - withFocusedWindow((focusedWindow) => { - focusedWindow.webContents.setZoomFactor(options.zoom); + const onZoomReset = (): void => { + withFocusedWindow((focusedWindow: BrowserWindow) => { + focusedWindow.webContents.zoomFactor = options.zoom; }); }; - const clearAppData = () => { - dialog.showMessageBox( - mainWindow, - { - type: 'warning', - buttons: ['Yes', 'Cancel'], - defaultId: 1, - title: 'Clear cache confirmation', - message: - 'This will clear all data (cookies, local storage etc) from this app. Are you sure you wish to proceed?', - }, - (response) => { - if (response !== 0) { - return; - } - clearCache(mainWindow, options.targetUrl); - }, - ); + const clearAppData = async (): Promise => { + const response = await dialog.showMessageBox(mainWindow, { + type: 'warning', + buttons: ['Yes', 'Cancel'], + defaultId: 1, + title: 'Clear cache confirmation', + message: + 'This will clear all data (cookies, local storage etc) from this app. Are you sure you wish to proceed?', + }); + + if (response.response !== 0) { + return; + } + await clearCache(mainWindow); }; - const onGoBack = () => { + const onGoBack = (): void => { withFocusedWindow((focusedWindow) => { focusedWindow.webContents.goBack(); }); }; - const onGoForward = () => { + const onGoForward = (): void => { withFocusedWindow((focusedWindow) => { focusedWindow.webContents.goForward(); }); }; - const getCurrentUrl = () => + const getCurrentUrl = (): void => withFocusedWindow((focusedWindow) => focusedWindow.webContents.getURL()); - const onWillNavigate = (event, urlToGo) => { + const onWillNavigate = (event: Event, urlToGo: string): void => { if (!linkIsInternal(options.targetUrl, urlToGo, options.internalUrls)) { event.preventDefault(); shell.openExternal(urlToGo); } }; - let createNewWindow; + const createNewWindow: (url: string) => BrowserWindow = (url: string) => { + const window = new BrowserWindow(DEFAULT_WINDOW_OPTIONS); + if (options.userAgent) { + window.webContents.userAgent = options.userAgent; + } - const createNewTab = (url, foreground) => { + if (options.proxyRules) { + setProxyRules(window, options.proxyRules); + } + + injectCss(window); + sendParamsOnDidFinishLoad(window); + window.webContents.on('new-window', onNewWindow); + window.webContents.on('will-navigate', onWillNavigate); + window.loadURL(url); + return window; + }; + + const createNewTab = (url: string, foreground: boolean): BrowserWindow => { withFocusedWindow((focusedWindow) => { const newTab = createNewWindow(url); focusedWindow.addTabbedWindow(newTab); @@ -248,7 +240,7 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { return undefined; }; - const createAboutBlankWindow = () => { + const createAboutBlankWindow = (): BrowserWindow => { const window = createNewWindow('about:blank'); window.hide(); window.webContents.once('did-stop-loading', () => { @@ -261,11 +253,15 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { return window; }; - const onNewWindow = (event, urlToGo, _, disposition) => { - const preventDefault = (newGuest) => { + const onNewWindow = ( + event: Event & { newGuest?: any }, + urlToGo: string, + frameName: string, + disposition, + ): void => { + const preventDefault = (newGuest: any): void => { event.preventDefault(); if (newGuest) { - // eslint-disable-next-line no-param-reassign event.newGuest = newGuest; } }; @@ -275,37 +271,24 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { options.targetUrl, options.internalUrls, preventDefault, - shell.openExternal, + shell.openExternal.bind(this), createAboutBlankWindow, nativeTabsSupported, createNewTab, ); }; - const sendParamsOnDidFinishLoad = (window) => { + const sendParamsOnDidFinishLoad = (window: BrowserWindow): void => { window.webContents.on('did-finish-load', () => { + // In children windows too: Restore pinch-to-zoom, disabled by default in recent Electron. + // See https://github.com/jiahaog/nativefier/issues/379#issuecomment-598612128 + // and https://github.com/electron/electron/pull/12679 + window.webContents.setVisualZoomLevelLimits(1, 3); + window.webContents.send('params', JSON.stringify(options)); }); }; - createNewWindow = (url) => { - const window = new BrowserWindow(DEFAULT_WINDOW_OPTIONS); - if (options.userAgent) { - window.webContents.setUserAgent(options.userAgent); - } - - if (options.proxyRules) { - setProxyRules(window, options.proxyRules); - } - - maybeInjectCss(window); - sendParamsOnDidFinishLoad(window); - window.webContents.on('new-window', onNewWindow); - window.webContents.on('will-navigate', onWillNavigate); - window.loadURL(url); - return window; - }; - const menuOptions = { nativefierVersion: options.nativefierVersion, appQuit: onAppQuit, @@ -329,14 +312,14 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { } if (options.userAgent) { - mainWindow.webContents.setUserAgent(options.userAgent); + mainWindow.webContents.userAgent = options.userAgent; } if (options.proxyRules) { setProxyRules(mainWindow, options.proxyRules); } - maybeInjectCss(mainWindow); + injectCss(mainWindow); sendParamsOnDidFinishLoad(mainWindow); if (options.counter) { @@ -366,6 +349,15 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { mainWindow.webContents.on('new-window', onNewWindow); mainWindow.webContents.on('will-navigate', onWillNavigate); + mainWindow.webContents.on('did-finish-load', () => { + // Restore pinch-to-zoom, disabled by default in recent Electron. + // See https://github.com/jiahaog/nativefier/issues/379#issuecomment-598309817 + // and https://github.com/electron/electron/pull/12679 + mainWindow.webContents.setVisualZoomLevelLimits(1, 3); + + // Remove potential css injection code set in `did-navigate`) (see injectCss code) + mainWindow.webContents.session.webRequest.onHeadersReceived(null); + }); if (options.clearCache) { clearCache(mainWindow); @@ -373,6 +365,7 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { mainWindow.loadURL(options.targetUrl); + // @ts-ignore mainWindow.on('new-tab', () => createNewTab(options.targetUrl, true)); mainWindow.on('close', (event) => { @@ -383,10 +376,10 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { mainWindow.setFullScreen(false); mainWindow.once( 'leave-full-screen', - maybeHideWindow.bind(this, mainWindow, event, options.fastQuit), + hideWindow.bind(this, mainWindow, event, options.fastQuit), ); } - maybeHideWindow(mainWindow, event, options.fastQuit, options.tray); + hideWindow(mainWindow, event, options.fastQuit, options.tray); if (options.clearCache) { clearCache(mainWindow); @@ -395,5 +388,3 @@ function createMainWindow(inpOptions, onAppQuit, setDockBadge) { return mainWindow; } - -export default createMainWindow; diff --git a/app/src/components/mainWindow/mainWindowHelpers.test.js b/app/src/components/mainWindowHelpers.test.ts similarity index 97% rename from app/src/components/mainWindow/mainWindowHelpers.test.js rename to app/src/components/mainWindowHelpers.test.ts index d1a3fe3f7b..eb69702271 100644 --- a/app/src/components/mainWindow/mainWindowHelpers.test.js +++ b/app/src/components/mainWindowHelpers.test.ts @@ -1,6 +1,4 @@ -import mainWindowHelpers from './mainWindowHelpers'; - -const { onNewWindowHelper } = mainWindowHelpers; +import { onNewWindowHelper } from './mainWindowHelpers'; const originalUrl = 'https://medium.com/'; const internalUrl = 'https://medium.com/topics/technology'; diff --git a/app/src/components/mainWindow/mainWindowHelpers.js b/app/src/components/mainWindowHelpers.ts similarity index 79% rename from app/src/components/mainWindow/mainWindowHelpers.js rename to app/src/components/mainWindowHelpers.ts index dee75f4be3..ddabd814a6 100644 --- a/app/src/components/mainWindow/mainWindowHelpers.js +++ b/app/src/components/mainWindowHelpers.ts @@ -1,18 +1,16 @@ -import helpers from '../../helpers/helpers'; +import { linkIsInternal } from '../helpers/helpers'; -const { linkIsInternal } = helpers; - -function onNewWindowHelper( - urlToGo, +export function onNewWindowHelper( + urlToGo: string, disposition, - targetUrl, + targetUrl: string, internalUrls, preventDefault, openExternal, createAboutBlankWindow, nativeTabsSupported, createNewTab, -) { +): void { if (!linkIsInternal(targetUrl, urlToGo, internalUrls)) { openExternal(urlToGo); preventDefault(); @@ -29,5 +27,3 @@ function onNewWindowHelper( } } } - -export default { onNewWindowHelper }; diff --git a/app/src/components/menu/menu.js b/app/src/components/menu.ts similarity index 76% rename from app/src/components/menu/menu.js rename to app/src/components/menu.ts index 0a61551e23..37a95195c8 100644 --- a/app/src/components/menu/menu.js +++ b/app/src/components/menu.ts @@ -1,19 +1,6 @@ -import { Menu, shell, clipboard } from 'electron'; +import { Menu, clipboard, globalShortcut, shell } from 'electron'; -/** - * @param nativefierVersion - * @param appQuit - * @param zoomIn - * @param zoomOut - * @param zoomReset - * @param zoomBuildTimeValue - * @param goBack - * @param goForward - * @param getCurrentUrl - * @param clearAppData - * @param disableDevTools - */ -function createMenu({ +export function createMenu({ nativefierVersion, appQuit, zoomIn, @@ -25,13 +12,13 @@ function createMenu({ getCurrentUrl, clearAppData, disableDevTools, -}) { +}): void { const zoomResetLabel = zoomBuildTimeValue === 1.0 ? 'Reset Zoom' : `Reset Zoom (to ${zoomBuildTimeValue * 100}%, set at build time)`; - const template = [ + const template: any[] = [ { label: '&Edit', submenu: [ @@ -83,9 +70,7 @@ function createMenu({ }, { label: 'Clear App Data', - click: () => { - clearAppData(); - }, + click: clearAppData, }, ], }, @@ -94,17 +79,19 @@ function createMenu({ submenu: [ { label: 'Back', - accelerator: 'CmdOrCtrl+[', - click: () => { - goBack(); - }, + accelerator: (() => { + globalShortcut.register('Alt+Left', goBack); + return 'CmdOrCtrl+['; + })(), + click: goBack, }, { label: 'Forward', - accelerator: 'CmdOrCtrl+]', - click: () => { - goForward(); - }, + accelerator: (() => { + globalShortcut.register('Alt+Right', goForward); + return 'CmdOrCtrl+]'; + })(), + click: goForward, }, { label: 'Reload', @@ -122,7 +109,7 @@ function createMenu({ label: 'Toggle Full Screen', accelerator: (() => { if (process.platform === 'darwin') { - return 'Ctrl+Command+F'; + return 'Ctrl+Cmd+F'; } return 'F11'; })(), @@ -135,44 +122,32 @@ function createMenu({ { label: 'Zoom In', accelerator: (() => { - if (process.platform === 'darwin') { - return 'Command+='; - } - return 'Ctrl+='; + globalShortcut.register('CmdOrCtrl+numadd', zoomIn); + return 'CmdOrCtrl+='; })(), - click: () => { - zoomIn(); - }, + click: zoomIn, }, { label: 'Zoom Out', accelerator: (() => { - if (process.platform === 'darwin') { - return 'Command+-'; - } - return 'Ctrl+-'; + globalShortcut.register('CmdOrCtrl+numsub', zoomOut); + return 'CmdOrCtrl+-'; })(), - click: () => { - zoomOut(); - }, + click: zoomOut, }, { label: zoomResetLabel, accelerator: (() => { - if (process.platform === 'darwin') { - return 'Command+0'; - } - return 'Ctrl+0'; + globalShortcut.register('CmdOrCtrl+num0', zoomReset); + return 'CmdOrCtrl+0'; })(), - click: () => { - zoomReset(); - }, + click: zoomReset, }, { label: 'Toggle Developer Tools', accelerator: (() => { if (process.platform === 'darwin') { - return 'Alt+Command+I'; + return 'Alt+Cmd+I'; } return 'Ctrl+Shift+I'; })(), @@ -240,12 +215,12 @@ function createMenu({ }, { label: 'Hide App', - accelerator: 'Command+H', + accelerator: 'Cmd+H', role: 'hide', }, { label: 'Hide Others', - accelerator: 'Command+Shift+H', + accelerator: 'Cmd+Shift+H', role: 'hideothers', }, { @@ -257,10 +232,8 @@ function createMenu({ }, { label: 'Quit', - accelerator: 'Command+Q', - click: () => { - appQuit(); - }, + accelerator: 'Cmd+Q', + click: appQuit, }, ], }); @@ -278,5 +251,3 @@ function createMenu({ const menu = Menu.buildFromTemplate(template); Menu.setApplicationMenu(menu); } - -export default createMenu; diff --git a/app/src/components/trayIcon/trayIcon.js b/app/src/components/trayIcon.ts similarity index 71% rename from app/src/components/trayIcon/trayIcon.js rename to app/src/components/trayIcon.ts index 79aff5e4ff..778e3c0013 100644 --- a/app/src/components/trayIcon/trayIcon.js +++ b/app/src/components/trayIcon.ts @@ -1,17 +1,12 @@ -import helpers from '../../helpers/helpers'; +import { app, Tray, Menu, ipcMain, nativeImage, BrowserWindow } from 'electron'; -const { app, Tray, Menu, ipcMain, nativeImage } = require('electron'); +import { getAppIcon, getCounterValue } from '../helpers/helpers'; -const { getAppIcon, getCounterValue } = helpers; - -/** - * - * @param {{}} inpOptions AppArgs from nativefier.json - * @param {electron.BrowserWindow} mainWindow MainWindow created from main.js - * @returns {electron.Tray} - */ -function createTrayIcon(inpOptions, mainWindow) { - const options = Object.assign({}, inpOptions); +export function createTrayIcon( + nativefierOptions, + mainWindow: BrowserWindow, +): Tray { + const options = { ...nativefierOptions }; if (options.tray) { const iconPath = getAppIcon(); @@ -33,7 +28,7 @@ function createTrayIcon(inpOptions, mainWindow) { }, { label: 'Quit', - click: app.exit, + click: app.exit.bind(this), }, ]); @@ -69,5 +64,3 @@ function createTrayIcon(inpOptions, mainWindow) { return null; } - -export default createTrayIcon; diff --git a/app/src/helpers/helpers.js b/app/src/helpers/helpers.js deleted file mode 100644 index 259b20fc40..0000000000 --- a/app/src/helpers/helpers.js +++ /dev/null @@ -1,87 +0,0 @@ -import wurl from 'wurl'; -import os from 'os'; -import fs from 'fs'; -import path from 'path'; - -const INJECT_CSS_PATH = path.join(__dirname, '..', 'inject/inject.css'); -const log = require('loglevel'); - -function isOSX() { - return os.platform() === 'darwin'; -} - -function isLinux() { - return os.platform() === 'linux'; -} - -function isWindows() { - return os.platform() === 'win32'; -} - -function linkIsInternal(currentUrl, newUrl, internalUrlRegex) { - if (newUrl === 'about:blank') { - return true; - } - - if (internalUrlRegex) { - const regex = RegExp(internalUrlRegex); - return regex.test(newUrl); - } - - const currentDomain = wurl('domain', currentUrl); - const newDomain = wurl('domain', newUrl); - return currentDomain === newDomain; -} - -function shouldInjectCss() { - try { - fs.accessSync(INJECT_CSS_PATH, fs.F_OK); - return true; - } catch (e) { - return false; - } -} - -function getCssToInject() { - return fs.readFileSync(INJECT_CSS_PATH).toString(); -} - -/** - * Helper method to print debug messages from the main process in the browser window - * @param {BrowserWindow} browserWindow - * @param message - */ -function debugLog(browserWindow, message) { - // need the timeout as it takes time for the preload javascript to be loaded in the window - setTimeout(() => { - browserWindow.webContents.send('debug', message); - }, 3000); - log.info(message); -} - -function getAppIcon() { - return path.join(__dirname, '../', `/icon.${isWindows() ? 'ico' : 'png'}`); -} - -function nativeTabsSupported() { - return isOSX(); -} - -function getCounterValue(title) { - const itemCountRegex = /[([{]([\d.,]*)\+?[}\])]/; - const match = itemCountRegex.exec(title); - return match ? match[1] : undefined; -} - -export default { - isOSX, - isLinux, - isWindows, - linkIsInternal, - getCssToInject, - debugLog, - shouldInjectCss, - getAppIcon, - nativeTabsSupported, - getCounterValue, -}; diff --git a/app/src/helpers/helpers.test.js b/app/src/helpers/helpers.test.ts similarity index 94% rename from app/src/helpers/helpers.test.js rename to app/src/helpers/helpers.test.ts index 3164c4da25..c7792d9bdb 100644 --- a/app/src/helpers/helpers.test.js +++ b/app/src/helpers/helpers.test.ts @@ -1,6 +1,4 @@ -import helpers from './helpers'; - -const { linkIsInternal, getCounterValue } = helpers; +import { linkIsInternal, getCounterValue } from './helpers'; const internalUrl = 'https://medium.com/'; const internalUrlSubPath = 'topic/technology'; diff --git a/app/src/helpers/helpers.ts b/app/src/helpers/helpers.ts new file mode 100644 index 0000000000..d7a59b9bc8 --- /dev/null +++ b/app/src/helpers/helpers.ts @@ -0,0 +1,78 @@ +import * as fs from 'fs'; +import * as os from 'os'; +import * as path from 'path'; + +import { BrowserWindow } from 'electron'; +import * as log from 'loglevel'; +import wurl from 'wurl'; + +const INJECT_CSS_PATH = path.join(__dirname, '../..', 'inject/inject.css'); + +export function isOSX(): boolean { + return os.platform() === 'darwin'; +} + +export function isLinux(): boolean { + return os.platform() === 'linux'; +} + +export function isWindows(): boolean { + return os.platform() === 'win32'; +} + +export function linkIsInternal( + currentUrl: string, + newUrl: string, + internalUrlRegex: string | RegExp, +): boolean { + if (newUrl === 'about:blank') { + return true; + } + + if (internalUrlRegex) { + const regex = RegExp(internalUrlRegex); + return regex.test(newUrl); + } + + const currentDomain = wurl('domain', currentUrl); + const newDomain = wurl('domain', newUrl); + return currentDomain === newDomain; +} + +export function shouldInjectCss(): boolean { + try { + fs.accessSync(INJECT_CSS_PATH); + return true; + } catch (e) { + return false; + } +} + +export function getCssToInject(): string { + return fs.readFileSync(INJECT_CSS_PATH).toString(); +} + +/** + * Helper to print debug messages from the main process in the browser window + */ +export function debugLog(browserWindow: BrowserWindow, message: string): void { + // Need a delay, as it takes time for the preloaded js to be loaded by the window + setTimeout(() => { + browserWindow.webContents.send('debug', message); + }, 3000); + log.info(message); +} + +export function getAppIcon(): string { + return path.join(__dirname, `../../icon.${isWindows() ? 'ico' : 'png'}`); +} + +export function nativeTabsSupported(): boolean { + return isOSX(); +} + +export function getCounterValue(title: string): string { + const itemCountRegex = /[([{]([\d.,]*)\+?[}\])]/; + const match = itemCountRegex.exec(title); + return match ? match[1] : undefined; +} diff --git a/app/src/helpers/inferFlash.js b/app/src/helpers/inferFlash.ts similarity index 56% rename from app/src/helpers/inferFlash.js rename to app/src/helpers/inferFlash.ts index e674754979..091d8e2dcd 100644 --- a/app/src/helpers/inferFlash.js +++ b/app/src/helpers/inferFlash.ts @@ -1,31 +1,32 @@ -import fs from 'fs'; -import path from 'path'; -import helpers from './helpers'; +import * as fs from 'fs'; +import * as path from 'path'; + +import * as log from 'loglevel'; + +import { isOSX, isWindows, isLinux } from './helpers'; -const { isOSX, isWindows, isLinux } = helpers; -const log = require('loglevel'); /** - * Synchronously find a file or directory - * @param {RegExp} pattern regex - * @param {string} base path - * @param {boolean} [findDir] if true, search results will be limited to only directories - * @returns {Array} + * Find a file or directory */ -function findSync(pattern, basePath, findDir) { - const matches = []; +function findSync( + pattern: RegExp, + basePath: string, + limitSearchToDirectories = false, +): string[] { + const matches: string[] = []; (function findSyncRecurse(base) { - let children; + let children: string[]; try { children = fs.readdirSync(base); - } catch (exception) { - if (exception.code === 'ENOENT') { + } catch (err) { + if (err.code === 'ENOENT') { return; } - throw exception; + throw err; } - children.forEach((child) => { + for (const child of children) { const childPath = path.join(base, child); const childIsDirectory = fs.lstatSync(childPath).isDirectory(); const patternMatches = pattern.test(childPath); @@ -38,7 +39,7 @@ function findSync(pattern, basePath, findDir) { return; } - if (!findDir) { + if (!limitSearchToDirectories) { matches.push(childPath); return; } @@ -46,23 +47,23 @@ function findSync(pattern, basePath, findDir) { if (childIsDirectory) { matches.push(childPath); } - }); + } })(basePath); return matches; } -function linuxMatch() { +function findFlashOnLinux() { return findSync(/libpepflashplayer\.so/, '/opt/google/chrome')[0]; } -function windowsMatch() { +function findFlashOnWindows() { return findSync( /pepflashplayer\.dll/, 'C:\\Program Files (x86)\\Google\\Chrome', )[0]; } -function darwinMatch() { +function findFlashOnMac() { return findSync( /PepperFlashPlayer.plugin/, '/Applications/Google Chrome.app/', @@ -70,20 +71,19 @@ function darwinMatch() { )[0]; } -function inferFlash() { +export function inferFlashPath() { if (isOSX()) { - return darwinMatch(); + return findFlashOnMac(); } if (isWindows()) { - return windowsMatch(); + return findFlashOnWindows(); } if (isLinux()) { - return linuxMatch(); + return findFlashOnLinux(); } log.warn('Unable to determine OS to infer flash player'); return null; } -export default inferFlash; diff --git a/app/src/main.js b/app/src/main.ts similarity index 79% rename from app/src/main.js rename to app/src/main.ts index 343c9c2750..28dead498c 100644 --- a/app/src/main.js +++ b/app/src/main.ts @@ -1,29 +1,26 @@ import 'source-map-support/register'; + import fs from 'fs'; import path from 'path'; + import { app, crashReporter, globalShortcut } from 'electron'; import electronDownload from 'electron-dl'; -import createLoginWindow from './components/login/loginWindow'; -import createMainWindow from './components/mainWindow/mainWindow'; -import createTrayIcon from './components/trayIcon/trayIcon'; -import helpers from './helpers/helpers'; -import inferFlash from './helpers/inferFlash'; +import { createLoginWindow } from './components/loginWindow'; +import { createMainWindow } from './components/mainWindow'; +import { createTrayIcon } from './components/trayIcon'; +import { isOSX } from './helpers/helpers'; +import { inferFlashPath } from './helpers/inferFlash'; -const electronSquirrelStartup = require('electron-squirrel-startup'); - -// Entrypoint for electron-squirrel-startup. -// See https://github.com/jiahaog/nativefier/pull/744 for sample use case -if (electronSquirrelStartup) { +// Entrypoint for Squirrel, a windows update framework. See https://github.com/jiahaog/nativefier/pull/744 +if (require('electron-squirrel-startup')) { app.exit(); } -const { isOSX } = helpers; - const APP_ARGS_FILE_PATH = path.join(__dirname, '..', 'nativefier.json'); const appArgs = JSON.parse(fs.readFileSync(APP_ARGS_FILE_PATH, 'utf8')); -const fileDownloadOptions = Object.assign({}, appArgs.fileDownloadOptions); +const fileDownloadOptions = { ...appArgs.fileDownloadOptions }; electronDownload(fileDownloadOptions); if (appArgs.processEnvs) { @@ -38,7 +35,7 @@ let mainWindow; if (typeof appArgs.flashPluginDir === 'string') { app.commandLine.appendSwitch('ppapi-flash-path', appArgs.flashPluginDir); } else if (appArgs.flashPluginDir) { - const flashPath = inferFlash(); + const flashPath = inferFlashPath(); app.commandLine.appendSwitch('ppapi-flash-path', flashPath); } @@ -76,18 +73,15 @@ if (appArgs.basicAuthPassword) { ); } -// do nothing for setDockBadge if not OSX -let setDockBadge = () => {}; - -if (isOSX()) { - let currentBadgeCount = 0; - - setDockBadge = (count, bounce = false) => { - app.dock.setBadge(count); - if (bounce && count > currentBadgeCount) app.dock.bounce(); - currentBadgeCount = count; - }; -} +const isRunningMacos = isOSX(); +let currentBadgeCount = 0; +const setDockBadge = isRunningMacos + ? (count: number, bounce = false) => { + app.dock.setBadge(count.toString()); + if (bounce && count > currentBadgeCount) app.dock.bounce(); + currentBadgeCount = count; + } + : () => undefined; app.on('window-all-closed', () => { if (!isOSX() || appArgs.fastQuit) { @@ -147,7 +141,7 @@ if (shouldQuit) { }); app.on('ready', () => { - mainWindow = createMainWindow(appArgs, app.quit, setDockBadge); + mainWindow = createMainWindow(appArgs, app.quit.bind(this), setDockBadge); createTrayIcon(appArgs, mainWindow); // Register global shortcuts diff --git a/app/src/static/preload.js b/app/src/preload.ts similarity index 62% rename from app/src/static/preload.js rename to app/src/preload.ts index c409a4a417..3b12be9d45 100644 --- a/app/src/static/preload.js +++ b/app/src/preload.ts @@ -1,25 +1,19 @@ /** - Preload file that will be executed in the renderer process - */ - -/** - * Note: This needs to be attached prior to the imports, as the they will delay - * the attachment till after the event has been raised. + * Preload file that will be executed in the renderer process. + * Note: This needs to be attached **prior to imports**, as imports + * would delay the attachment till after the event has been raised. */ document.addEventListener('DOMContentLoaded', () => { - // Due to the early attachment, this triggers a linter error - // because it's not yet been defined. - // eslint-disable-next-line no-use-before-define - injectScripts(); + injectScripts(); // eslint-disable-line @typescript-eslint/no-use-before-define }); -// Disable imports being first due to the above event attachment -import { ipcRenderer } from 'electron'; // eslint-disable-line import/first -import path from 'path'; // eslint-disable-line import/first -import fs from 'fs'; // eslint-disable-line import/first +import * as fs from 'fs'; +import * as path from 'path'; + +import { ipcRenderer } from 'electron'; +import * as log from 'loglevel'; -const INJECT_JS_PATH = path.join(__dirname, '../../', 'inject/inject.js'); -const log = require('loglevel'); +const INJECT_JS_PATH = path.join(__dirname, '..', 'inject/inject.js'); /** * Patches window.Notification to: * - set a callback on a new Notification @@ -40,6 +34,7 @@ function setNotificationCallback(createCallback, clickCallback) { get: () => OldNotify.permission, }); + // @ts-ignore window.Notification = newNotify; } @@ -49,7 +44,6 @@ function injectScripts() { return; } // Dynamically require scripts - // eslint-disable-next-line global-require, import/no-dynamic-require require(INJECT_JS_PATH); } @@ -68,6 +62,5 @@ ipcRenderer.on('params', (event, message) => { }); ipcRenderer.on('debug', (event, message) => { - // eslint-disable-next-line no-console log.info('debug:', message); }); diff --git a/app/src/static/login/login.css b/app/src/static/login.css similarity index 100% rename from app/src/static/login/login.css rename to app/src/static/login.css diff --git a/app/src/static/login/login.html b/app/src/static/login.html similarity index 100% rename from app/src/static/login/login.html rename to app/src/static/login.html diff --git a/app/src/static/login.js b/app/src/static/login.js new file mode 100644 index 0000000000..65617dc4dd --- /dev/null +++ b/app/src/static/login.js @@ -0,0 +1,10 @@ +const { ipcRenderer } = require('electron'); + +document.getElementById('login-form').addEventListener('submit', (event) => { + event.preventDefault(); + const usernameInput = document.getElementById('username-input'); + const username = usernameInput.nodeValue || usernameInput.value; + const passwordInput = document.getElementById('password-input'); + const password = passwordInput.nodeValue || passwordInput.value; + ipcRenderer.send('login-message', [username, password]); +}); diff --git a/app/src/static/login/login.js b/app/src/static/login/login.js deleted file mode 100644 index 2e35dfc78c..0000000000 --- a/app/src/static/login/login.js +++ /dev/null @@ -1,12 +0,0 @@ -import electron from 'electron'; - -const { ipcRenderer } = electron; - -const form = document.getElementById('login-form'); - -form.addEventListener('submit', (event) => { - event.preventDefault(); - const username = document.getElementById('username-input').value; - const password = document.getElementById('password-input').value; - ipcRenderer.send('login-message', [username, password]); -}); diff --git a/app/tsconfig.json b/app/tsconfig.json new file mode 100644 index 0000000000..516a9c1ee8 --- /dev/null +++ b/app/tsconfig.json @@ -0,0 +1,19 @@ +{ + "compilerOptions": { + "allowJs": true, + "declaration": false, + "esModuleInterop": true, + "incremental": true, + "module": "commonjs", + "moduleResolution": "node", + "outDir": "./lib", + "resolveJsonModule": true, + "skipLibCheck": true, + "sourceMap": true, + "target": "es2017", + "lib": ["es2017", "dom"] + }, + "include": [ + "./src/**/*" + ] +} diff --git a/docs/development.md b/docs/development.md index 7992090653..c5b28381f9 100644 --- a/docs/development.md +++ b/docs/development.md @@ -1,6 +1,6 @@ -# Development +# Development Guide -## Environment Setup +## Setup First, clone the project @@ -9,49 +9,59 @@ git clone https://github.com/jiahaog/nativefier.git cd nativefier ``` -Install dependencies and build: +Install dependencies: ```bash -# macOS and Linux -npm run dev-up - -# Windows -npm run dev-up-win +npm install ``` -If dependencies are installed and you just want to re-build, +Build nativefier: ```bash npm run build ``` -You can set up a symbolic link so that running `nativefier` invokes your development version including your changes: +Set up a symbolic link so that running `nativefier` calls your dev version with your changes: ```bash npm link +which nativefier +# -> Should return a path, e.g. /home/youruser/.node_modules/lib/node_modules/nativefier +# If not, be sure your `npm_config_prefix` env var is set and in your `PATH` ``` -After doing so (and not forgetting to build with `npm run build`), you can run Nativefier with your test parameters: +After doing so, you can run Nativefier with your test parameters: ```bash -nativefier <--your-awesome-new-flag> +nativefier --your-awesome-new-flag 'https://your-test-site.com' ``` -Or you can automatically watch the files for changes with: - +Then run your nativefier app *through the command line too* (to see logs & errors): ```bash -npm run watch +# Under Linux +./your-test-site-linux-x64/your-test-site + +# Under Windows +your-test-site-win32-x64/your-test-site.exe + +# Under macOS +open -a YourTestSite.app ``` -## Tests +## Linting & formatting -```bash -# To run all tests (unit, end-to-end), -npm test +Nativefier uses [Prettier](https://prettier.io/), which will shout at you for +not formatting code exactly like it expects. This guarantees a homogenous style, +but is painful to do manually. Do yourself a favor and install a +[Prettier plugin for your editor](https://prettier.io/docs/en/editors.html). -# To run only unit tests, -npm run jest +## Tests -# To run only end-to-end tests, -npm run e2e -``` +- To run all tests, `npm t` +- To run only unit tests, `npm run test:unit` +- To run only integration tests, `npm run test:integration` +- Logging is suppressed by default in tests, to avoid polluting Jest output. + To get debug logs, `npm run test:withlog` or set the `LOGLEVEL` env. var. +- For a good live experience, open two terminal panes/tabs running code/tests watchers: + 2. Run a TSC watcher: `npm run build:watch` + 3. Run a Jest unit tests watcher: `npm run test:watch` diff --git a/docs/dock.png b/docs/dock.png new file mode 100644 index 0000000000000000000000000000000000000000..a81ca6de40b70f45262d2481177790faa759ecb2 GIT binary patch literal 25557 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
  • -OSK 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 literal 0 HcmV?d00001 diff --git a/scripts/changelog b/docs/generate-changelog similarity index 94% rename from scripts/changelog rename to docs/generate-changelog index 8b170d8d42..f1d26104e3 100755 --- a/scripts/changelog +++ b/docs/generate-changelog @@ -43,9 +43,9 @@ mv package.json.tmp package.json # Unset the editor so that git changelog does not open a editor EDITOR=: -git changelog docs/changelog.md --tag "$VERSION" +git changelog CHANGELOG.md --tag "$VERSION" # Commit these changes -git add docs/changelog.md +git add CHANGELOG.md git add package.json git commit -m "Update changelog for \`v$VERSION\`" diff --git a/docs/manual-test b/docs/manual-test new file mode 100755 index 0000000000..ef3f1a4fe8 --- /dev/null +++ b/docs/manual-test @@ -0,0 +1,61 @@ +#!/usr/bin/env bash +# Manual test to validate some hard-to-programmatically-test features work. + +set -eo pipefail + +missingDeps=false +if ! command -v mktemp > /dev/null; then echo "Missing mktemp"; missingDeps=true; fi +if ! command -v uname > /dev/null; then echo "Missing uname"; missingDeps=true; fi +if ! command -v node > /dev/null; then echo "Missing node"; missingDeps=true; fi +if [ "$missingDeps" = true ]; then exit 1; fi + + +script_dir="$( cd "$( dirname "${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )" +nativefier_dir="$script_dir/.." +pushd "$nativefier_dir" + +printf "\n***** Creating test dirs & resources *****\n" +tmp_dir=$(mktemp -d -t nativefier-manual-test-XXXXX) +resources_dir="$tmp_dir/resources" +mkdir "$resources_dir" +injected_css="$resources_dir/inject.css" +injected_js="$resources_dir/inject.js" +echo '* { background-color: blue; }' > "$injected_css" +echo 'alert("hello world from inject");' > "$injected_js" + +printf "\n***** Building test app *****\n" +node ./lib/cli.js 'https://npmjs.com/' \ + --inject "$injected_css" \ + --inject "$injected_js" \ + --name "app" \ + "$tmp_dir" + +printf "\n***** Test checklist ***** +- Injected js: should show an alert saying hello +- Injected css: should make npmjs all blue +- Internal links open internally +- External links open in browser +- Keyboard shortcuts: {back, forward, zoom in/out/zero} work +- Console: no Electron runtime deprecation warnings/error logged +" + +printf "\n***** Running app *****\n" +if [ "$(uname -s)" = "Darwin" ]; then + open -a 'app-darwin-x64/app.app' +else + "$tmp_dir/app-linux-x64/app" +fi + +printf "\nDid everything work as expected? [yN] " +read -r response +if [ "$response" != 'y' ]; then + echo "Back to fixing" + exit 1 +else + echo "Yayyyyyyyyyyy" +fi + +if [ -n "$tmp_dir" ]; then + printf "\n***** Deleting test dir %s *****\n" "$tmp_dir" + rm -rf "$tmp_dir" +fi diff --git a/docs/release.md b/docs/release.md index fb372b53c7..be4ae49926 100644 --- a/docs/release.md +++ b/docs/release.md @@ -1,17 +1,28 @@ # Release -Releases are automatically deployed to NPM on Travis, when they are tagged. However, we have to make sure that the version in the `package.json`, and the changelog is updated. +Releases are automatically deployed to npm from Travis, when they are tagged. +However, we have to make sure that the version in the `package.json`, +and the changelog is updated. -## Dependencies -- [Git Extras](https://github.com/tj/git-extras/blob/master/Installation.md) -- [jq](https://stedolan.github.io/jq/download/) +## Tests -## How to Release `$VERSION` +Before anything, run a little manual smoke test of some of our +hard-to-programatically-test features: + +```bash +npm run test:manual +``` + +## How to release + +With [Git Extras](https://github.com/tj/git-extras/blob/master/Installation.md) +and [jq](https://stedolan.github.io/jq/download/) installed. While on `master`, with no uncommitted changes, ```bash npm run changelog -- $VERSION +# For example, npm run changelog -- 7.7.1 ``` This command does 3 things: @@ -22,15 +33,17 @@ This command does 3 things: Now we may want to cleanup the changelog: ```bash -vim docs/changelog.md - +vim CHANGELOG.md git commit --amend ``` Once we are satisfied, + ```bash git push origin master ``` -On [GitHub Releases](https://github.com/jiahaog/nativefier/releases), draft and publish a new release with title `Nativefier vX.X.X`. +On [GitHub Releases](https://github.com/jiahaog/nativefier/releases), +draft and publish a new release with title `Nativefier vX.X.X` (yes, with a `v`). +The new version will be visible on npm within a few minutes/hours. diff --git a/screenshots/walkthrough.gif b/docs/walkthrough.gif similarity index 100% rename from screenshots/walkthrough.gif rename to docs/walkthrough.gif diff --git a/e2e/index.test.js b/e2e/index.test.js deleted file mode 100644 index d37555b441..0000000000 --- a/e2e/index.test.js +++ /dev/null @@ -1,83 +0,0 @@ -import tmp from 'tmp'; -import fs from 'fs'; -import path from 'path'; -import async from 'async'; - -import nativefier from '../src'; - -const PLATFORMS = ['darwin', 'linux']; -tmp.setGracefulCleanup(); - -function checkApp(appPath, inputOptions, callback) { - try { - let relPathToConfig; - - switch (inputOptions.platform) { - case 'darwin': - relPathToConfig = path.join( - 'google-test-app.app', - 'Contents/Resources/app', - ); - break; - case 'linux': - relPathToConfig = 'resources/app'; - break; - case 'win32': - relPathToConfig = 'resources/app'; - break; - default: - throw new Error('Unknown app platform'); - } - - const nativefierConfigPath = path.join( - appPath, - relPathToConfig, - 'nativefier.json', - ); - const nativefierConfig = JSON.parse(fs.readFileSync(nativefierConfigPath)); - - expect(inputOptions.targetUrl).toBe(nativefierConfig.targetUrl); - // app name is not consistent for linux - // assert.strictEqual(inputOptions.appName, nativefierConfig.name, - // 'Packaged app must have the same name as the input parameters'); - callback(); - } catch (exception) { - callback(exception); - } -} - -describe('Nativefier Module', () => { - jest.setTimeout(240000); - test('Can build an app from a target url', (done) => { - async.eachSeries( - PLATFORMS, - (platform, callback) => { - const tmpObj = tmp.dirSync({ unsafeCleanup: true }); - - const tmpPath = tmpObj.name; - const options = { - name: 'google-test-app', - targetUrl: 'http://google.com', - out: tmpPath, - overwrite: true, - platform: null, - }; - - options.platform = platform; - nativefier(options, (error, appPath) => { - if (error) { - callback(error); - return; - } - - checkApp(appPath, options, (err) => { - callback(err); - }); - }); - }, - (error) => { - done(error); - }, - ); - }); -}); diff --git a/gulp/build.js b/gulp/build.js deleted file mode 100644 index 74bad1f8a2..0000000000 --- a/gulp/build.js +++ /dev/null @@ -1,18 +0,0 @@ -import gulp from 'gulp'; -import del from 'del'; -import runSequence from 'run-sequence'; -import PATHS from './helpers/src-paths'; - -gulp.task('build', (callback) => { - runSequence('clean', ['build-cli', 'build-app'], callback); -}); - -gulp.task('clean', (callback) => { - del(PATHS.CLI_DEST).then(() => { - del(PATHS.APP_DEST).then(() => { - del(PATHS.TEST_DEST).then(() => { - callback(); - }); - }); - }); -}); diff --git a/gulp/build/build-app.js b/gulp/build/build-app.js deleted file mode 100644 index aaa5dc998e..0000000000 --- a/gulp/build/build-app.js +++ /dev/null @@ -1,12 +0,0 @@ -import gulp from 'gulp'; -import webpack from 'webpack-stream'; -import PATHS from '../helpers/src-paths'; - -const webpackConfig = require('./../../webpack.config.js'); - -gulp.task('build-app', ['build-static'], () => - gulp - .src(PATHS.APP_MAIN_JS) - .pipe(webpack(webpackConfig)) - .pipe(gulp.dest(PATHS.APP_DEST)), -); diff --git a/gulp/build/build-cli.js b/gulp/build/build-cli.js deleted file mode 100644 index 991c7f03c1..0000000000 --- a/gulp/build/build-cli.js +++ /dev/null @@ -1,9 +0,0 @@ -import gulp from 'gulp'; -import PATHS from '../helpers/src-paths'; -import helpers from '../helpers/gulp-helpers'; - -const { buildES6 } = helpers; - -gulp.task('build-cli', (done) => - buildES6(PATHS.CLI_SRC_JS, PATHS.CLI_DEST, done), -); diff --git a/gulp/build/build-static.js b/gulp/build/build-static.js deleted file mode 100644 index 7d4cd3f7a0..0000000000 --- a/gulp/build/build-static.js +++ /dev/null @@ -1,17 +0,0 @@ -import gulp from 'gulp'; -import PATHS from '../helpers/src-paths'; -import helpers from '../helpers/gulp-helpers'; - -const { buildES6 } = helpers; - -gulp.task('build-static-not-js', () => - gulp - .src([PATHS.APP_STATIC_ALL, '!**/*.js']) - .pipe(gulp.dest(PATHS.APP_STATIC_DEST)), -); - -gulp.task('build-static-js', (done) => - buildES6(PATHS.APP_STATIC_JS, PATHS.APP_STATIC_DEST, done), -); - -gulp.task('build-static', ['build-static-js', 'build-static-not-js']); diff --git a/gulp/helpers/gulp-helpers.js b/gulp/helpers/gulp-helpers.js deleted file mode 100644 index fa3109d803..0000000000 --- a/gulp/helpers/gulp-helpers.js +++ /dev/null @@ -1,29 +0,0 @@ -import gulp from 'gulp'; -import shellJs from 'shelljs'; -import sourcemaps from 'gulp-sourcemaps'; -import babel from 'gulp-babel'; - -function shellExec(cmd, silent, callback) { - shellJs.exec(cmd, { silent }, (code, stdout, stderr) => { - if (code) { - callback(JSON.stringify({ code, stdout, stderr })); - return; - } - callback(); - }); -} - -function buildES6(src, dest, callback) { - return gulp - .src(src) - .pipe(sourcemaps.init()) - .pipe(babel()) - .on('error', callback) - .pipe(sourcemaps.write('.')) - .pipe(gulp.dest(dest)); -} - -export default { - shellExec, - buildES6, -}; diff --git a/gulp/helpers/src-paths.js b/gulp/helpers/src-paths.js deleted file mode 100644 index 512f64fbb7..0000000000 --- a/gulp/helpers/src-paths.js +++ /dev/null @@ -1,22 +0,0 @@ -import path from 'path'; - -const paths = { - APP_SRC: 'app/src', - APP_DEST: 'app/lib', - CLI_SRC: 'src', - CLI_DEST: 'lib', - TEST_SRC: 'test', - TEST_DEST: 'built-tests', -}; - -paths.APP_MAIN_JS = path.join(paths.APP_SRC, '/main.js'); -paths.APP_ALL = `${paths.APP_SRC}/**/*`; -paths.APP_STATIC_ALL = `${path.join(paths.APP_SRC, 'static')}/**/*`; -paths.APP_STATIC_JS = `${path.join(paths.APP_SRC, 'static')}/**/*.js`; -paths.APP_STATIC_DEST = path.join(paths.APP_DEST, 'static'); -paths.CLI_SRC_JS = `${paths.CLI_SRC}/**/*.js`; -paths.CLI_DEST_JS = `${paths.CLI_DEST}/**/*.js`; -paths.TEST_SRC_JS = `${paths.TEST_SRC}/**/*.js`; -paths.TEST_DEST_JS = `${paths.TEST_DEST}/**/*.js`; - -export default paths; diff --git a/gulp/release.js b/gulp/release.js deleted file mode 100644 index d22f344bb7..0000000000 --- a/gulp/release.js +++ /dev/null @@ -1,11 +0,0 @@ -import gulp from 'gulp'; -import runSequence from 'run-sequence'; -import helpers from './helpers/gulp-helpers'; - -const { shellExec } = helpers; - -gulp.task('publish', (done) => { - shellExec('npm publish', false, done); -}); - -gulp.task('release', (callback) => runSequence('build', 'publish', callback)); diff --git a/gulp/watch.js b/gulp/watch.js deleted file mode 100644 index df9611833d..0000000000 --- a/gulp/watch.js +++ /dev/null @@ -1,13 +0,0 @@ -import gulp from 'gulp'; -import PATHS from './helpers/src-paths'; - -const log = require('loglevel'); - -gulp.task('watch', ['build'], () => { - const handleError = function watch(error) { - log.error(error); - }; - gulp.watch(PATHS.APP_ALL, ['build-app']).on('error', handleError); - - gulp.watch(PATHS.CLI_SRC_JS, ['build-cli']).on('error', handleError); -}); diff --git a/gulpfile.babel.js b/gulpfile.babel.js deleted file mode 100644 index 0cfb1c656b..0000000000 --- a/gulpfile.babel.js +++ /dev/null @@ -1,9 +0,0 @@ -import gulp from 'gulp'; -import requireDir from 'require-dir'; - -requireDir('./gulp', { - recurse: true, - duplicates: true, -}); - -gulp.task('default', ['build']); diff --git a/bin/convertToIcns b/icon-scripts/convertToIcns similarity index 100% rename from bin/convertToIcns rename to icon-scripts/convertToIcns diff --git a/bin/convertToIco b/icon-scripts/convertToIco similarity index 100% rename from bin/convertToIco rename to icon-scripts/convertToIco diff --git a/bin/convertToIconset b/icon-scripts/convertToIconset similarity index 100% rename from bin/convertToIconset rename to icon-scripts/convertToIconset diff --git a/bin/convertToPng b/icon-scripts/convertToPng similarity index 85% rename from bin/convertToPng rename to icon-scripts/convertToPng index 60b9ac6b3c..ff77284d88 100755 --- a/bin/convertToPng +++ b/icon-scripts/convertToPng @@ -8,8 +8,8 @@ set -e -type convert >/dev/null 2>&1 || { echo >&2 "Cannot find required ImageMagick Convert executable"; exit 1; } -type identify >/dev/null 2>&1 || { echo >&2 "Cannot find required ImageMagick Identify executable"; exit 1; } +type convert >/dev/null 2>&1 || { echo >&2 "Cannot find required ImageMagick 'convert' executable, please install it and make sure it is in your PATH"; exit 1; } +type identify >/dev/null 2>&1 || { echo >&2 "Cannot find required ImageMagick 'identify' executable, please install it and make sure it is in your PATH"; exit 1; } # Parameters SOURCE="$1" diff --git a/jest.config.js b/jest.config.js deleted file mode 100644 index 25c9bac510..0000000000 --- a/jest.config.js +++ /dev/null @@ -1,3 +0,0 @@ -module.exports = { - testEnvironment: 'node', -}; diff --git a/package.json b/package.json index 3ace3763cb..f545a03100 100644 --- a/package.json +++ b/package.json @@ -2,6 +2,12 @@ "name": "nativefier", "version": "7.7.1", "description": "Wrap web apps natively", + "license": "MIT", + "author": "Goh Jia Hao", + "engines": { + "node": ">= 8.10.0", + "npm": ">= 5.6.0" + }, "keywords": [ "desktop", "electron", @@ -9,97 +15,85 @@ "native", "wrapper" ], - "main": "lib/index.js", - "scripts": { - "dev-up": "npm install && (cd ./app && npm install) && npm run build", - "dev-up-win": "npm install & cd app & npm install & cd .. & npm run build", - "test": "jest src", - "guard": "jest --watch src", - "e2e": "jest e2e", - "tdd": "gulp tdd", - "lint": "eslint .", - "lint:fix": "eslint . --fix", - "ci": "npm run lint && npm test && npm run e2e", - "clean": "gulp clean", - "build": "gulp build", - "watch": "while true ; do gulp watch ; done", - "package-placeholder": "npm run build && node lib/cli.js http://www.bennish.net/web-notifications.html ~/Desktop --overwrite --name notification-test --icon ./test-resources/iconSampleGrey.png --inject ./test-resources/test-injection.js --inject ./test-resources/test-injection.css && open ~/Desktop/notification-test-darwin-x64/notification-test.app", - "start-placeholder": "npm run build && electron app", - "changelog": "./scripts/changelog", - "format": "prettier --write '{gulp,src}/**/*.js' 'app/src/**/*.js'" - }, + "main": "lib/main.js", "bin": { "nativefier": "lib/cli.js" }, + "homepage": "https://github.com/jiahaog/nativefier", "repository": { "type": "git", "url": "git+https://github.com/jiahaog/nativefier.git" }, - "author": "Goh Jia Hao", - "license": "MIT", "bugs": { "url": "https://github.com/jiahaog/nativefier/issues" }, - "homepage": "https://github.com/jiahaog/nativefier#readme", + "scripts": { + "build-app-static": "ncp app/src/static/ app/lib/static/", + "build": "npm run clean && tsc --build . app && npm run build-app-static", + "build:watch": "tsc --build . app --watch", + "changelog": "./docs/generate-changelog", + "ci": "npm run lint && npm test", + "clean": "rimraf lib/ app/lib/", + "clean:full": "rimraf lib/ app/lib/ node_modules/ app/node_modules/", + "lint:fix": "eslint . --fix", + "lint:format": "prettier --write 'src/**/*.js' 'app/src/**/*.js'", + "lint": "eslint . --ext .ts", + "list-outdated-deps": "npm out; cd app && npm out; true", + "postinstall": "cd app && yarn install --no-lockfile --no-progress --silent", + "test:integration": "jest --testRegex '.*integration-test.js'", + "test:manual": "npm run build && ./docs/manual-test", + "test:unit": "jest", + "test:watch": "jest --watch", + "test:withlog": "LOGLEVEL=trace npm run test", + "test": "jest --testRegex '[-.]test\\.js$'" + }, "dependencies": { - "async": "^2.6.0", - "axios": "^0.18.0", - "babel-polyfill": "^6.26.0", - "cheerio": "^1.0.0-rc.2", - "commander": "^2.14.0", - "electron-packager": "^12.2.0", - "gitcloud": "^0.1.0", - "hasbin": "^1.2.3", - "lodash": "^4.17.5", - "loglevel": "^1.6.1", - "ncp": "^2.0.0", - "page-icon": "^0.3.0", - "progress": "^2.0.0", - "sanitize-filename": "^1.6.1", - "shelljs": "^0.8.1", - "source-map-support": "^0.5.3", - "tmp": "0.0.33", - "validator": "^10.2.0" + "@types/cheerio": "0.x", + "@types/electron-packager": "14.x", + "@types/lodash": "4.x", + "@types/ncp": "2.x", + "@types/node": "8.x", + "@types/page-icon": "0.x", + "@types/shelljs": "0.x", + "@types/tmp": "0.x", + "axios": "0.x", + "cheerio": "^1.0.0-rc.3", + "commander": "4.x", + "electron-packager": "14.x", + "gitcloud": "0.x", + "hasbin": "1.x", + "lodash": "4.x", + "loglevel": "1.x", + "ncp": "2.x", + "page-icon": "0.x", + "sanitize-filename": "1.x", + "shelljs": "0.x", + "source-map-support": "0.x", + "tmp": "0.x", + "yarn": "1.x" }, "devDependencies": { - "babel-core": "^6.26.0", - "babel-jest": "^23.4.0", - "babel-loader": "^7.1.2", - "babel-plugin-transform-object-rest-spread": "^6.26.0", - "babel-preset-env": "^1.6.1", - "babel-register": "^6.26.0", - "chai": "^4.1.2", - "del": "^3.0.0", - "eslint": "^5.2.0", - "eslint-config-airbnb-base": "^13.0.0", - "eslint-config-prettier": "^4.0.0", - "eslint-plugin-import": "^2.8.0", - "eslint-plugin-prettier": "^3.0.0", - "gulp": "^3.9.1", - "gulp-babel": "^7.0.1", - "gulp-sourcemaps": "^2.6.4", - "jest": "^23.4.1", - "prettier": "^1.12.1", - "require-dir": "^1.0.0", - "run-sequence": "^2.2.1", - "webpack-stream": "^5.0.0" - }, - "engines": { - "node": ">= 4.0" + "@types/jest": "25.x", + "@typescript-eslint/eslint-plugin": "2.x", + "@typescript-eslint/parser": "2.x", + "eslint": "6.x", + "eslint-config-prettier": "6.x", + "eslint-plugin-prettier": "3.x", + "jest": "25.x", + "prettier": "1.x", + "rimraf": "3.x", + "typescript": "3.x" }, - "babel": { - "plugins": [ - "transform-object-rest-spread" + "jest": { + "collectCoverage": true, + "setupFiles": [ + "./lib/jestSetupFiles" ], - "presets": [ - [ - "env", - { - "targets": { - "node": "4.0.0" - } - } - ] + "testEnvironment": "node", + "testPathIgnorePatterns": [ + "/node_modules/", + "/app/src.*", + "/src.*" ] } } diff --git a/screenshots/dock.png b/screenshots/dock.png deleted file mode 100644 index 9c8e961f5eaae67cb728bde1bf863425be9adb51..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27764 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 { - resolve(); - }); - } - const promises = srcs.map( - (src) => - new Promise((resolve, reject) => { - if (!fs.existsSync(src)) { - reject(new Error('Error copying injection files: file not found')); - return; - } - - let destFileName; - if (path.extname(src) === '.js') { - destFileName = 'inject.js'; - } else if (path.extname(src) === '.css') { - destFileName = 'inject.css'; - } else { - resolve(); - return; - } - - copy(src, path.join(dest, 'inject', destFileName), (error) => { - if (error) { - reject(new Error(`Error Copying injection files: ${error}`)); - return; - } - resolve(); - }); - }), - ); - - return new Promise((resolve, reject) => { - Promise.all(promises) - .then(() => { - resolve(); - }) - .catch((error) => { - reject(error); - }); - }); -} - -function normalizeAppName(appName, url) { - // use a simple 3 byte random string to prevent collision - const hash = crypto.createHash('md5'); - hash.update(url); - const postFixHash = hash.digest('hex').substring(0, 6); - const normalized = _.kebabCase(appName.toLowerCase()); - return `${normalized}-nativefier-${postFixHash}`; -} - -function changeAppPackageJsonName(appPath, name, url) { - const packageJsonPath = path.join(appPath, '/package.json'); - const packageJson = JSON.parse(fs.readFileSync(packageJsonPath)); - packageJson.name = normalizeAppName(name, url); - fs.writeFileSync(packageJsonPath, JSON.stringify(packageJson)); -} - -/** - * Creates a temporary directory and copies the './app folder' inside, - * and adds a text file with the configuration for the single page app. - * - * @param {string} src - * @param {string} dest - * @param {{}} options - * @param callback - */ -function buildApp(src, dest, options, callback) { - const appArgs = selectAppArgs(options); - - copy(src, dest, (error) => { - if (error) { - callback(`Error Copying temporary directory: ${error}`); - return; - } - - fs.writeFileSync( - path.join(dest, '/nativefier.json'), - JSON.stringify(appArgs), - ); - - maybeCopyScripts(options.inject, dest) - .catch((err) => { - log.warn(err); - }) - .then(() => { - changeAppPackageJsonName(dest, appArgs.name, appArgs.targetUrl); - callback(); - }); - }); -} - -export default buildApp; diff --git a/src/build/buildIcon.ts b/src/build/buildIcon.ts new file mode 100644 index 0000000000..7e4490a43b --- /dev/null +++ b/src/build/buildIcon.ts @@ -0,0 +1,96 @@ +import * as path from 'path'; + +import * as log from 'loglevel'; + +import { isOSX } from '../helpers/helpers'; +import { + convertToPng, + convertToIco, + convertToIcns, +} from '../helpers/iconShellHelpers'; +import { AppOptions } from '../options/model'; + +function iconIsIco(iconPath: string): boolean { + return path.extname(iconPath) === '.ico'; +} + +function iconIsPng(iconPath: string): boolean { + return path.extname(iconPath) === '.png'; +} + +function iconIsIcns(iconPath: string): boolean { + return path.extname(iconPath) === '.icns'; +} + +/** + * Will convert a `.png` icon to the appropriate arch format (if necessary), + * and return adjusted options + */ +export async function convertIconIfNecessary( + options: AppOptions, +): Promise { + if (!options.packager.icon) { + log.debug('Option "icon" not set, skipping icon conversion.'); + return; + } + + if (options.packager.platform === 'win32') { + if (iconIsIco(options.packager.icon)) { + log.debug( + 'Building for Windows and icon is already a .ico, no conversion needed', + ); + return; + } + + try { + const iconPath = await convertToIco(options.packager.icon); + options.packager.icon = iconPath; + return; + } catch (error) { + log.warn('Failed to convert icon to .ico, skipping.', error); + return; + } + } + + if (options.packager.platform === 'linux') { + if (iconIsPng(options.packager.icon)) { + log.debug( + 'Building for Linux and icon is already a .png, no conversion needed', + ); + return; + } + + try { + const iconPath = await convertToPng(options.packager.icon); + options.packager.icon = iconPath; + return; + } catch (error) { + log.warn('Failed to convert icon to .png, skipping.', error); + return; + } + } + + if (iconIsIcns(options.packager.icon)) { + log.debug( + 'Building for macOS and icon is already a .icns, no conversion needed', + ); + return; + } + + if (!isOSX()) { + log.warn( + 'Skipping icon conversion to .icns, conversion is only supported on macOS', + ); + return; + } + + try { + const iconPath = await convertToIcns(options.packager.icon); + options.packager.icon = iconPath; + return; + } catch (error) { + log.warn('Failed to convert icon to .icns, skipping.', error); + options.packager.icon = undefined; + return; + } +} diff --git a/src/build/buildMain.js b/src/build/buildMain.js deleted file mode 100644 index bde1e3b63d..0000000000 --- a/src/build/buildMain.js +++ /dev/null @@ -1,247 +0,0 @@ -import path from 'path'; -import packager from 'electron-packager'; -import tmp from 'tmp'; -import ncp from 'ncp'; -import async from 'async'; -import hasBinary from 'hasbin'; -import log from 'loglevel'; -import DishonestProgress from '../helpers/dishonestProgress'; -import optionsFactory from '../options/optionsMain'; -import iconBuild from './iconBuild'; -import helpers from '../helpers/helpers'; -import PackagerConsole from '../helpers/packagerConsole'; -import buildApp from './buildApp'; - -const copy = ncp.ncp; -const { isWindows } = helpers; - -/** - * Checks the app path array to determine if the packaging was completed successfully - * @param appPathArray Result from electron-packager - * @returns {*} - */ -function getAppPath(appPathArray) { - if (appPathArray.length === 0) { - // directory already exists, --overwrite is not set - // exit here - return null; - } - - if (appPathArray.length > 1) { - log.warn( - 'Warning: This should not be happening, packaged app path contains more than one element:', - appPathArray, - ); - } - - return appPathArray[0]; -} - -/** - * Removes the `icon` parameter from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function maybeNoIconOption(options) { - const packageOptions = JSON.parse(JSON.stringify(options)); - if (options.platform === 'win32' && !isWindows()) { - if (!hasBinary.sync('wine')) { - log.warn( - 'Wine is required to set the icon for a Windows app when packaging on non-windows platforms', - ); - packageOptions.icon = null; - } - } - return packageOptions; -} - -/** - * For windows and linux, we have to copy over the icon to the resources/app folder, which the - * BrowserWindow is hard coded to read the icon from - * @param {{}} options - * @param {string} appPath - * @param callback - */ -function maybeCopyIcons(options, appPath, callback) { - if (!options.icon) { - callback(); - return; - } - - if (options.platform === 'darwin' || options.platform === 'mas') { - callback(); - return; - } - - // windows & linux - // put the icon file into the app - const destIconPath = path.join(appPath, 'resources/app'); - const destFileName = `icon${path.extname(options.icon)}`; - copy(options.icon, path.join(destIconPath, destFileName), (error) => { - callback(error); - }); -} - -/** - * Removes invalid parameters from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function removeInvalidOptions(options, param) { - const packageOptions = JSON.parse(JSON.stringify(options)); - if (options.platform === 'win32' && !isWindows()) { - if (!hasBinary.sync('wine')) { - log.warn( - `Wine is required to use "${param}" option for a Windows app when packaging on non-windows platforms`, - ); - packageOptions[param] = null; - } - } - return packageOptions; -} - -/** - * Removes the `appCopyright` parameter from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function maybeNoAppCopyrightOption(options) { - return removeInvalidOptions(options, 'appCopyright'); -} - -/** - * Removes the `buildVersion` parameter from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function maybeNoBuildVersionOption(options) { - return removeInvalidOptions(options, 'buildVersion'); -} - -/** - * Removes the `appVersion` parameter from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function maybeNoAppVersionOption(options) { - return removeInvalidOptions(options, 'appVersion'); -} - -/** - * Removes the `versionString` parameter from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function maybeNoVersionStringOption(options) { - return removeInvalidOptions(options, 'versionString'); -} - -/** - * Removes the `win32metadata` parameter from options if building for Windows while not on Windows - * and Wine is not installed - * @param options - */ -function maybeNoWin32metadataOption(options) { - return removeInvalidOptions(options, 'win32metadata'); -} - -/** - * @callback buildAppCallback - * @param error - * @param {string} appPath - */ - -/** - * - * @param {{}} inpOptions - * @param {buildAppCallback} callback - */ -function buildMain(inpOptions, callback) { - const options = Object.assign({}, inpOptions); - - // pre process app - const tmpObj = tmp.dirSync({ mode: '0755', unsafeCleanup: true }); - const tmpPath = tmpObj.name; - - // todo check if this is still needed on later version of packager - const packagerConsole = new PackagerConsole(); - - const progress = new DishonestProgress(5); - - async.waterfall( - [ - (cb) => { - progress.tick('inferring'); - optionsFactory(options) - .then((result) => { - cb(null, result); - }) - .catch((error) => { - cb(error); - }); - }, - (opts, cb) => { - progress.tick('copying'); - buildApp(opts.dir, tmpPath, opts, (error) => { - if (error) { - cb(error); - return; - } - // Change the reference file for the Electron app to be the temporary path - const newOptions = Object.assign({}, opts, { - dir: tmpPath, - }); - cb(null, newOptions); - }); - }, - (opts, cb) => { - progress.tick('icons'); - iconBuild(opts, (error, optionsWithIcon) => { - cb(null, optionsWithIcon); - }); - }, - (opts, cb) => { - progress.tick('packaging'); - // maybe skip passing icon parameter to electron packager - let packageOptions = maybeNoIconOption(opts); - // maybe skip passing other parameters to electron packager - packageOptions = maybeNoAppCopyrightOption(packageOptions); - packageOptions = maybeNoAppVersionOption(packageOptions); - packageOptions = maybeNoBuildVersionOption(packageOptions); - packageOptions = maybeNoVersionStringOption(packageOptions); - packageOptions = maybeNoWin32metadataOption(packageOptions); - - packagerConsole.override(); - - packager(packageOptions) - .then((appPathArray) => { - packagerConsole.restore(); // restore console.error - cb(null, opts, appPathArray); // options still contain the icon to waterfall - }) - .catch((error) => { - packagerConsole.restore(); // restore console.error - cb(error, opts); // options still contain the icon to waterfall - }); - }, - (opts, appPathArray, cb) => { - progress.tick('finalizing'); - // somehow appPathArray is a 1 element array - const appPath = getAppPath(appPathArray); - if (!appPath) { - cb(); - return; - } - - maybeCopyIcons(opts, appPath, (error) => { - cb(error, appPath); - }); - }, - ], - (error, appPath) => { - packagerConsole.playback(); - callback(error, appPath); - }, - ); -} - -export default buildMain; diff --git a/src/build/buildNativefierApp.ts b/src/build/buildNativefierApp.ts new file mode 100644 index 0000000000..7598d3abf4 --- /dev/null +++ b/src/build/buildNativefierApp.ts @@ -0,0 +1,125 @@ +import * as path from 'path'; + +import * as electronGet from '@electron/get'; +import * as electronPackager from 'electron-packager'; +import * as hasbin from 'hasbin'; +import * as log from 'loglevel'; + +import { isWindows, getTempDir, copyFileOrDir } from '../helpers/helpers'; +import { getOptions } from '../options/optionsMain'; +import { prepareElectronApp } from './prepareElectronApp'; +import { convertIconIfNecessary } from './buildIcon'; +import { AppOptions } from '../options/model'; + +const OPTIONS_REQUIRING_WINDOWS_FOR_WINDOWS_BUILD = [ + 'icon', + 'appCopyright', + 'appVersion', + 'buildVersion', + 'versionString', + 'win32metadata', +]; + +/** + * Checks the app path array to determine if packaging completed successfully + */ +function getAppPath(appPath: string | string[]): string { + if (!Array.isArray(appPath)) { + return appPath; + } + + if (appPath.length === 0) { + return null; // directory already exists and `--overwrite` not set + } + + if (appPath.length > 1) { + log.warn( + 'Warning: This should not be happening, packaged app path contains more than one element:', + appPath, + ); + } + + return appPath[0]; +} + +/** + * For Windows & Linux, we have to copy over the icon to the resources/app + * folder, which the BrowserWindow is hard-coded to read the icon from + */ +async function copyIconsIfNecessary( + options: AppOptions, + appPath: string, +): Promise { + log.debug('Copying icons if necessary'); + if (!options.packager.icon) { + log.debug('No icon specified in options; aborting'); + return; + } + + if ( + options.packager.platform === 'darwin' || + options.packager.platform === 'mas' + ) { + log.debug('No copying necessary on macOS; aborting'); + return; + } + + // windows & linux: put the icon file into the app + const destAppPath = path.join(appPath, 'resources/app'); + const destFileName = `icon${path.extname(options.packager.icon)}`; + const destIconPath = path.join(destAppPath, destFileName); + + log.debug(`Copying icon ${options.packager.icon} to`, destIconPath); + await copyFileOrDir(options.packager.icon, destIconPath); +} + +function trimUnprocessableOptions(options: AppOptions): void { + if ( + options.packager.platform === 'win32' && + !isWindows() && + !hasbin.sync('wine') + ) { + const optionsPresent = Object.entries(options) + .filter( + ([key, value]) => + OPTIONS_REQUIRING_WINDOWS_FOR_WINDOWS_BUILD.includes(key) && !!value, + ) + .map(([key]) => key); + if (optionsPresent.length === 0) { + return; + } + log.warn( + `*Not* setting [${optionsPresent.join(', ')}], as couldn't find Wine.`, + 'Wine is required when packaging a Windows app under on non-Windows platforms.', + ); + for (const keyToUnset of optionsPresent) { + options[keyToUnset] = null; + } + } +} + +export async function buildNativefierApp(rawOptions: any): Promise { + log.info('Processing options...'); + const options = await getOptions(rawOptions); + + log.info('\nPreparing Electron app...'); + const tmpPath = getTempDir('app', 0o755); + await prepareElectronApp(options.packager.dir, tmpPath, options); + + log.info('\nConverting icons...'); + options.packager.dir = tmpPath; // const optionsWithTmpPath = { ...options, dir: tmpPath }; + await convertIconIfNecessary(options); + + log.info( + "\nPackaging... This will take a few seconds, maybe minutes if the requested Electron isn't cached yet...", + ); + trimUnprocessableOptions(options); + electronGet.initializeProxy(); // https://github.com/electron/get#proxies + const appPathArray = await electronPackager(options.packager); + + log.info('\nFinalizing build...'); + const appPath = getAppPath(appPathArray); + await copyIconsIfNecessary(options, appPath); + + return appPath; +} diff --git a/src/build/iconBuild.js b/src/build/iconBuild.js deleted file mode 100644 index 4a1798fb09..0000000000 --- a/src/build/iconBuild.js +++ /dev/null @@ -1,106 +0,0 @@ -import path from 'path'; -import log from 'loglevel'; -import helpers from '../helpers/helpers'; -import iconShellHelpers from '../helpers/iconShellHelpers'; - -const { isOSX } = helpers; -const { convertToPng, convertToIco, convertToIcns } = iconShellHelpers; - -function iconIsIco(iconPath) { - return path.extname(iconPath) === '.ico'; -} - -function iconIsPng(iconPath) { - return path.extname(iconPath) === '.png'; -} - -function iconIsIcns(iconPath) { - return path.extname(iconPath) === '.icns'; -} - -/** - * @callback augmentIconsCallback - * @param error - * @param options - */ - -/** - * Will check and convert a `.png` to `.icns` if necessary and augment - * options.icon with the result - * - * @param inpOptions will need options.platform and options.icon - * @param {augmentIconsCallback} callback - */ -function iconBuild(inpOptions, callback) { - const options = Object.assign({}, inpOptions); - const returnCallback = () => { - callback(null, options); - }; - - if (!options.icon) { - returnCallback(); - return; - } - - if (options.platform === 'win32') { - if (iconIsIco(options.icon)) { - returnCallback(); - return; - } - - convertToIco(options.icon) - .then((outPath) => { - options.icon = outPath; - returnCallback(); - }) - .catch((error) => { - log.warn('Skipping icon conversion to .ico', error); - returnCallback(); - }); - return; - } - - if (options.platform === 'linux') { - if (iconIsPng(options.icon)) { - returnCallback(); - return; - } - - convertToPng(options.icon) - .then((outPath) => { - options.icon = outPath; - returnCallback(); - }) - .catch((error) => { - log.warn('Skipping icon conversion to .png', error); - returnCallback(); - }); - return; - } - - if (iconIsIcns(options.icon)) { - returnCallback(); - return; - } - - if (!isOSX()) { - log.warn( - 'Skipping icon conversion to .icns, conversion is only supported on OSX', - ); - returnCallback(); - return; - } - - convertToIcns(options.icon) - .then((outPath) => { - options.icon = outPath; - returnCallback(); - }) - .catch((error) => { - log.warn('Skipping icon conversion to .icns', error); - options.icon = undefined; - returnCallback(); - }); -} - -export default iconBuild; diff --git a/src/build/prepareElectronApp.ts b/src/build/prepareElectronApp.ts new file mode 100644 index 0000000000..7d2cb989fc --- /dev/null +++ b/src/build/prepareElectronApp.ts @@ -0,0 +1,157 @@ +import * as fs from 'fs'; +import * as crypto from 'crypto'; +import * as path from 'path'; +import { promisify } from 'util'; + +import { kebabCase } from 'lodash'; +import * as log from 'loglevel'; + +import { copyFileOrDir } from '../helpers/helpers'; +import { AppOptions } from '../options/model'; + +const writeFileAsync = promisify(fs.writeFile); + +/** + * Only picks certain app args to pass to nativefier.json + */ +function pickElectronAppArgs(options: AppOptions): any { + return { + alwaysOnTop: options.nativefier.alwaysOnTop, + appCopyright: options.packager.appCopyright, + appVersion: options.packager.appVersion, + backgroundColor: options.nativefier.backgroundColor, + basicAuthPassword: options.nativefier.basicAuthPassword, + basicAuthUsername: options.nativefier.basicAuthUsername, + bounce: options.nativefier.bounce, + browserwindowOptions: options.nativefier.browserwindowOptions, + buildVersion: options.packager.buildVersion, + clearCache: options.nativefier.clearCache, + counter: options.nativefier.counter, + crashReporter: options.nativefier.crashReporter, + darwinDarkModeSupport: options.packager.darwinDarkModeSupport, + disableContextMenu: options.nativefier.disableContextMenu, + disableDevTools: options.nativefier.disableDevTools, + disableGpu: options.nativefier.disableGpu, + diskCacheSize: options.nativefier.diskCacheSize, + enableEs3Apis: options.nativefier.enableEs3Apis, + fastQuit: options.nativefier.fastQuit, + fileDownloadOptions: options.nativefier.fileDownloadOptions, + flashPluginDir: options.nativefier.flashPluginDir, + fullScreen: options.nativefier.fullScreen, + globalShortcuts: options.nativefier.globalShortcuts, + height: options.nativefier.height, + hideWindowFrame: options.nativefier.hideWindowFrame, + ignoreCertificate: options.nativefier.ignoreCertificate, + ignoreGpuBlacklist: options.nativefier.ignoreGpuBlacklist, + insecure: options.nativefier.insecure, + internalUrls: options.nativefier.internalUrls, + maxHeight: options.nativefier.maxHeight, + maximize: options.nativefier.maximize, + maxWidth: options.nativefier.maxWidth, + minHeight: options.nativefier.minHeight, + minWidth: options.nativefier.minWidth, + name: options.packager.name, + nativefierVersion: options.nativefier.nativefierVersion, + processEnvs: options.nativefier.processEnvs, + proxyRules: options.nativefier.proxyRules, + showMenuBar: options.nativefier.showMenuBar, + singleInstance: options.nativefier.singleInstance, + targetUrl: options.packager.targetUrl, + titleBarStyle: options.nativefier.titleBarStyle, + tray: options.nativefier.tray, + userAgent: options.nativefier.userAgent, + versionString: options.nativefier.versionString, + width: options.nativefier.width, + win32metadata: options.packager.win32metadata, + x: options.nativefier.x, + y: options.nativefier.y, + zoom: options.nativefier.zoom, + }; +} + +async function maybeCopyScripts(srcs: string[], dest: string): Promise { + if (!srcs || srcs.length === 0) { + log.debug('No files to inject, skipping copy.'); + return; + } + + log.debug(`Copying ${srcs.length} files to inject in app.`); + for (const src of srcs) { + if (!fs.existsSync(src)) { + throw new Error( + `File ${src} not found. Note that Nativefier expects *local* files, not URLs.`, + ); + } + + let destFileName: string; + if (path.extname(src) === '.js') { + destFileName = 'inject.js'; + } else if (path.extname(src) === '.css') { + destFileName = 'inject.css'; + } else { + return; + } + + const destPath = path.join(dest, 'inject', destFileName); + log.debug(`Copying injection file "${src}" to "${destPath}"`); + await copyFileOrDir(src, destPath); + } +} + +function normalizeAppName(appName: string, url: string): string { + // use a simple 3 byte random string to prevent collision + const hash = crypto.createHash('md5'); + hash.update(url); + const postFixHash = hash.digest('hex').substring(0, 6); + const normalized = kebabCase(appName.toLowerCase()); + return `${normalized}-nativefier-${postFixHash}`; +} + +function changeAppPackageJsonName( + appPath: string, + name: string, + url: string, +): void { + const packageJsonPath = path.join(appPath, '/package.json'); + const packageJson = JSON.parse(fs.readFileSync(packageJsonPath).toString()); + const normalizedAppName = normalizeAppName(name, url); + packageJson.name = normalizedAppName; + log.debug(`Updating ${packageJsonPath} 'name' field to ${normalizedAppName}`); + + fs.writeFileSync(packageJsonPath, JSON.stringify(packageJson)); +} + +/** + * Creates a temporary directory, copies the './app folder' inside, + * and adds a text file with the app configuration. + */ +export async function prepareElectronApp( + src: string, + dest: string, + options: AppOptions, +): Promise { + log.debug(`Copying electron app from ${src} to ${dest}`); + try { + await copyFileOrDir(src, dest); + } catch (err) { + throw `Error copying electron app from ${src} to temp dir ${dest}. Error: ${err}`; + } + + const appJsonPath = path.join(dest, '/nativefier.json'); + log.debug(`Writing app config to ${appJsonPath}`); + await writeFileAsync( + appJsonPath, + JSON.stringify(pickElectronAppArgs(options)), + ); + + try { + await maybeCopyScripts(options.nativefier.inject, dest); + } catch (err) { + log.error('Error copying injection files.', err); + } + changeAppPackageJsonName( + dest, + options.packager.name, + options.packager.targetUrl, + ); +} diff --git a/src/cli.js b/src/cli.ts similarity index 50% rename from src/cli.js rename to src/cli.ts index f2355d30d6..20cd41b89f 100755 --- a/src/cli.js +++ b/src/cli.ts @@ -1,19 +1,22 @@ -#! /usr/bin/env node - +#!/usr/bin/env node import 'source-map-support/register'; -import program from 'commander'; -import nativefier from './index'; -const dns = require('dns'); -const log = require('loglevel'); -const packageJson = require('./../package'); +import * as commander from 'commander'; +import * as dns from 'dns'; +import * as log from 'loglevel'; + +import { buildNativefierApp } from './main'; -function collect(val, memo) { +// package.json is `require`d to let tsc strip the `src` folder by determining +// baseUrl=src. A static import would prevent that and cause an ugly extra "src" folder +const packageJson = require('../package.json'); // eslint-disable-line @typescript-eslint/no-var-requires + +function collect(val: any, memo: any[]): any[] { memo.push(val); return memo; } -function parseMaybeBoolString(val) { +function parseBooleanOrString(val: string): boolean | string { switch (val) { case 'true': return true; @@ -24,19 +27,19 @@ function parseMaybeBoolString(val) { } } -function parseJson(val) { +function parseJson(val: string): any { if (!val) return {}; return JSON.parse(val); } -function getProcessEnvs(val) { - if (!val) return {}; - const pEnv = {}; - pEnv.processEnvs = parseJson(val); - return pEnv; +function getProcessEnvs(val: string): any { + if (!val) { + return {}; + } + return { processEnvs: parseJson(val) }; } -function checkInternet() { +function checkInternet(): void { dns.lookup('npmjs.com', (err) => { if (err && err.code === 'ENOTFOUND') { log.warn( @@ -63,31 +66,36 @@ if (require.main === module) { sanitizedArgs.push(arg); }); - program + const positionalOptions = { + targetUrl: '', + out: '', + }; + commander + .name('nativefier') .version(packageJson.version, '-v, --version') .arguments(' [dest]') - .action((targetUrl, appDir) => { - program.targetUrl = targetUrl; - program.out = appDir; + .action((url, outputDirectory) => { + positionalOptions.targetUrl = url; + positionalOptions.out = outputDirectory; }) .option('-n, --name ', 'app name') - .option('-p, --platform ', "'osx', 'mas', 'linux' or 'windows'") + .option('-p, --platform ', "'mac', 'mas', 'linux' or 'windows'") .option('-a, --arch ', "'ia32' or 'x64' or 'armv7l'") .option( '--app-version ', - 'The release version of the application. Maps to the `ProductVersion` metadata property on Windows, and `CFBundleShortVersionString` on OS X.', + '(macOS, windows only) the version of the app. Maps to the `ProductVersion` metadata property on Windows, and `CFBundleShortVersionString` on macOS.', ) .option( '--build-version ', - 'The build version of the application. Maps to the `FileVersion` metadata property on Windows, and `CFBundleVersion` on OS X.', + '(macOS, windows only) The build version of the app. Maps to `FileVersion` metadata property on Windows, and `CFBundleVersion` on macOS', ) .option( '--app-copyright ', - 'The human-readable copyright line for the app. Maps to the `LegalCopyright` metadata property on Windows, and `NSHumanReadableCopyright` on OS X', + '(macOS, windows only) a human-readable copyright line for the app. Maps to `LegalCopyright` metadata property on Windows, and `NSHumanReadableCopyright` on macOS', ) .option( '--win32metadata ', - 'a JSON string of key/value pairs of application metadata (ProductName, InternalName, FileDescription) to embed into the executable (Windows only).', + '(windows only) a JSON string of key/value pairs (ProductName, InternalName, FileDescription) to embed as executable metadata', parseJson, ) .option( @@ -96,19 +104,19 @@ if (require.main === module) { ) .option( '--no-overwrite', - 'do not override output directory if it already exists, defaults to false', + 'do not override output directory if it already exists; defaults to false', ) .option( '-c, --conceal', - 'packages the source code within your app into an archive, defaults to false, see https://electronjs.org/docs/tutorial/application-packaging', + 'packages the app source code into an asar archive; defaults to false', ) .option( '--counter', - 'if the target app should use a persistent counter badge in the dock (macOS only), defaults to false', + '(macOS only) set a dock count badge, determined by looking for a number in the window title; defaults to false', ) .option( '--bounce', - 'if the the dock icon should bounce when counter increases (macOS only), defaults to false', + '(macOS only) make he dock icon bounce when the counter increases; defaults to false', ) .option( '-i, --icon ', @@ -116,61 +124,61 @@ if (require.main === module) { ) .option( '--width ', - 'set window default width, defaults to 1280px', + 'set window default width; defaults to 1280px', parseInt, ) .option( '--height ', - 'set window default height, defaults to 800px', + 'set window default height; defaults to 800px', parseInt, ) .option( '--min-width ', - 'set window minimum width, defaults to 0px', + 'set window minimum width; defaults to 0px', parseInt, ) .option( '--min-height ', - 'set window minimum height, defaults to 0px', + 'set window minimum height; defaults to 0px', parseInt, ) .option( '--max-width ', - 'set window maximum width, default is no limit', + 'set window maximum width; default is unlimited', parseInt, ) .option( '--max-height ', - 'set window maximum height, default is no limit', + 'set window maximum height; default is unlimited', parseInt, ) .option('--x ', 'set window x location', parseInt) .option('--y ', 'set window y location', parseInt) - .option('-m, --show-menu-bar', 'set menu bar visible, defaults to false') + .option('-m, --show-menu-bar', 'set menu bar visible; defaults to false') .option( '-f, --fast-quit', - 'quit app after window close (macOS only), defaults to false', + '(macOS only) quit app on window close; defaults to false', ) - .option('-u, --user-agent ', 'set the user agent string for the app') + .option('-u, --user-agent ', 'set the app user agent string') .option( '--honest', - 'prevent the nativefied app from changing the user agent string to masquerade as a regular chrome browser', + 'prevent the normal changing of the user agent string to appear as a regular Chrome browser', ) - .option('--ignore-certificate', 'ignore certificate related errors') + .option('--ignore-certificate', 'ignore certificate-related errors') .option('--disable-gpu', 'disable hardware acceleration') .option( '--ignore-gpu-blacklist', - 'allow WebGl apps to work on non supported graphics cards', + 'force WebGL apps to work on unsupported GPUs', ) - .option('--enable-es3-apis', 'force activation of WebGl 2.0') + .option('--enable-es3-apis', 'force activation of WebGL 2.0') .option( '--insecure', - 'enable loading of insecure content, defaults to false', + 'enable loading of insecure content; defaults to false', ) - .option('--flash', 'if flash should be enabled') + .option('--flash', 'enables Adobe Flash; defaults to false') .option( '--flash-path ', - 'path to Chrome flash plugin, find it in `Chrome://plugins`', + 'path to Chrome flash plugin; find it in `chrome://plugins`', ) .option( '--disk-cache-size ', @@ -178,31 +186,31 @@ if (require.main === module) { ) .option( '--inject ', - 'path to a CSS/JS file to be injected', + 'path to a CSS/JS file to be injected. Pass multiple times to inject multiple files.', collect, [], ) + .option('--full-screen', 'always start the app full screen') + .option('--maximize', 'always start the app maximized') + .option('--hide-window-frame', 'disable window frame and controls') + .option('--verbose', 'enable verbose/debug/troubleshooting logs') + .option('--disable-context-menu', 'disable the context menu (right click)') .option( - '--full-screen', - 'if the app should always be started in full screen', + '--disable-dev-tools', + 'disable developer tools (Ctrl+Shift+I / F12)', ) - .option('--maximize', 'if the app should always be started maximized') - .option('--hide-window-frame', 'disable window frame and controls') - .option('--verbose', 'if verbose logs should be displayed') - .option('--disable-context-menu', 'disable the context menu') - .option('--disable-dev-tools', 'disable developer tools') .option( '--zoom ', - 'default zoom factor to use when the app is opened, defaults to 1.0', + 'default zoom factor to use when the app is opened; defaults to 1.0', parseFloat, ) .option( '--internal-urls ', - 'regular expression of URLs to consider "internal"; all other URLs will be opened in an external browser. (default: URLs on same second-level domain as app)', + 'regex of URLs to consider "internal"; all other URLs will be opened in an external browser. Default: URLs on same second-level domain as app', ) .option( '--proxy-rules ', - 'proxy rules. See https://electronjs.org/docs/api/session?q=proxy#sessetproxyconfig-callback', + 'proxy rules; see https://www.electronjs.org/docs/api/session#sessetproxyconfig', ) .option( '--crash-reporter ', @@ -218,29 +226,29 @@ if (require.main === module) { ) .option( '--processEnvs ', - 'a JSON string of key/value pairs to be set as environment variables before any browser windows are opened.', + 'a JSON string of key/value pairs to be set as environment variables before any browser windows are opened', getProcessEnvs, ) .option( '--file-download-options ', - 'a JSON string of key/value pairs to be set as file download options. See https://github.com/sindresorhus/electron-dl for available options.', + 'a JSON string of key/value pairs to be set as file download options. See https://github.com/sindresorhus/electron-dl for available options.', parseJson, ) .option( '--tray [start-in-tray]', - "Allow app to stay in system tray. If 'start-in-tray' is given as argument, don't show main window on first start", - parseMaybeBoolString, + "Allow app to stay in system tray. If 'start-in-tray' is set as argument, don't show main window on first start", + parseBooleanOrString, ) .option('--basic-auth-username ', 'basic http(s) auth username') .option('--basic-auth-password ', 'basic http(s) auth password') .option('--always-on-top', 'enable always on top window') .option( '--title-bar-style ', - "(macOS only) set title bar style ('hidden', 'hiddenInset'). Consider injecting custom CSS (via --inject) for better integration.", + "(macOS only) set title bar style ('hidden', 'hiddenInset'). Consider injecting custom CSS (via --inject) for better integration", ) .option( '--global-shortcuts ', - 'JSON file with global shortcut configuration. See https://github.com/jiahaog/nativefier/blob/master/docs/api.md#global-shortcuts', + 'JSON file defining global shortcuts. See https://github.com/jiahaog/nativefier/blob/master/docs/api.md#global-shortcuts', ) .option( '--browserwindow-options ', @@ -249,7 +257,7 @@ if (require.main === module) { ) .option( '--background-color ', - "Sets the background color (for seamless experience while the app is loading). Example value: '#2e2c29'", + "sets the app background color, for better integration while the app is loading. Example value: '#2e2c29'", ) .option( '--darwin-dark-mode-support', @@ -258,19 +266,19 @@ if (require.main === module) { .parse(sanitizedArgs); if (!process.argv.slice(2).length) { - program.help(); + commander.help(); } checkInternet(); - nativefier(program, (error, appPath) => { - if (error) { - log.error(error); - return; - } - - if (!appPath) { - // app exists and --overwrite is not passed - return; - } - log.info(`App built to ${appPath}`); - }); + const options = { ...positionalOptions, ...commander.opts() }; + buildNativefierApp(options) + .then((appPath) => { + if (!appPath) { + log.info(`App *not* built to ${appPath}`); + return; + } + log.info(`App built to ${appPath}`); + }) + .catch((error) => { + log.error('Error during build. Run with --verbose for details.', error); + }); } diff --git a/src/constants.js b/src/constants.js deleted file mode 100644 index b1873f0935..0000000000 --- a/src/constants.js +++ /dev/null @@ -1,5 +0,0 @@ -import path from 'path'; - -export const DEFAULT_APP_NAME = 'APP'; -export const ELECTRON_VERSION = '5.0.13'; -export const PLACEHOLDER_APP_DIR = path.join(__dirname, './../', 'app'); diff --git a/src/constants.ts b/src/constants.ts new file mode 100644 index 0000000000..25f2a45d15 --- /dev/null +++ b/src/constants.ts @@ -0,0 +1,13 @@ +import * as path from 'path'; + +export const DEFAULT_APP_NAME = 'APP'; + +// Update both together +export const DEFAULT_ELECTRON_VERSION = '8.1.1'; +export const DEFAULT_CHROME_VERSION = '80.0.3987.141'; + +export const ELECTRON_MAJOR_VERSION = parseInt( + DEFAULT_ELECTRON_VERSION.split('.')[0], + 10, +); +export const PLACEHOLDER_APP_DIR = path.join(__dirname, './../', 'app'); diff --git a/src/helpers/convertToIcns.js b/src/helpers/convertToIcns.js deleted file mode 100644 index ea0b9fa762..0000000000 --- a/src/helpers/convertToIcns.js +++ /dev/null @@ -1,65 +0,0 @@ -import shell from 'shelljs'; -import path from 'path'; -import tmp from 'tmp'; -import helpers from './helpers'; - -const { isOSX } = helpers; -tmp.setGracefulCleanup(); - -const PNG_TO_ICNS_BIN_PATH = path.join(__dirname, '../..', 'bin/convertToIcns'); - -/** - * @callback pngToIcnsCallback - * @param error - * @param {string} icnsDest If error, will return the original png src - */ - -/** - * - * @param {string} pngSrc - * @param {string} icnsDest - * @param {pngToIcnsCallback} callback - */ -function convertToIcns(pngSrc, icnsDest, callback) { - if (!isOSX()) { - callback('OSX is required to convert .png to .icns icon', pngSrc); - return; - } - - shell.exec( - `"${PNG_TO_ICNS_BIN_PATH}" "${pngSrc}" "${icnsDest}"`, - { silent: true }, - (exitCode, stdOut, stdError) => { - if (stdOut.includes('icon.iconset:error') || exitCode) { - if (exitCode) { - callback( - { - stdOut, - stdError, - }, - pngSrc, - ); - return; - } - - callback(stdOut, pngSrc); - return; - } - - callback(null, icnsDest); - }, - ); -} - -/** - * Converts the png to a temporary directory which will be cleaned up on process exit - * @param {string} pngSrc - * @param {pngToIcnsCallback} callback - */ -function convertToIcnsTmp(pngSrc, callback) { - const tempIconDirObj = tmp.dirSync({ unsafeCleanup: true }); - const tempIconDirPath = tempIconDirObj.name; - convertToIcns(pngSrc, `${tempIconDirPath}/icon.icns`, callback); -} - -export default convertToIcnsTmp; diff --git a/src/helpers/convertToIcns.test.js b/src/helpers/convertToIcns.test.js deleted file mode 100644 index 275cc27432..0000000000 --- a/src/helpers/convertToIcns.test.js +++ /dev/null @@ -1,40 +0,0 @@ -import fs from 'fs'; -import os from 'os'; -import path from 'path'; -import convertToIcns from './convertToIcns'; - -// Prerequisite for test: to use OSX with sips, iconutil and imagemagick convert - -function testConvertPng(pngName) { - if (os.platform() !== 'darwin') { - // Skip png conversion tests, OSX is required - return Promise.resolve(); - } - - return new Promise((resolve, reject) => - convertToIcns( - path.join(__dirname, '../../', 'test-resources', pngName), - (error, icnsPath) => { - if (error) { - reject(error); - return; - } - - const stat = fs.statSync(icnsPath); - - expect(stat.isFile()).toBe(true); - resolve(); - }, - ), - ); -} - -describe('Get Icon Module', () => { - test('Can convert a rgb png to icns', async () => { - await testConvertPng('iconSample.png'); - }); - - test('Can convert a grey png to icns', async () => { - await testConvertPng('iconSampleGrey.png'); - }); -}); diff --git a/src/helpers/dishonestProgress.js b/src/helpers/dishonestProgress.js deleted file mode 100644 index beac6fee07..0000000000 --- a/src/helpers/dishonestProgress.js +++ /dev/null @@ -1,70 +0,0 @@ -import ProgressBar from 'progress'; - -class DishonestProgress { - constructor(total) { - this.tickParts = total * 10; - - this.bar = new ProgressBar(' :task [:bar] :percent', { - complete: '=', - incomplete: ' ', - total: total * this.tickParts, - width: 50, - clear: true, - }); - - this.tickingPrevious = { - message: '', - remainder: 0, - interval: null, - }; - } - - tick(message) { - const { - remainder: prevRemainder, - message: prevMessage, - interval: prevInterval, - } = this.tickingPrevious; - - if (prevRemainder) { - this.bar.tick(prevRemainder, { - task: prevMessage, - }); - clearInterval(prevInterval); - } - - const realRemainder = this.bar.total - this.bar.curr; - if (realRemainder === this.tickParts) { - this.bar.tick(this.tickParts, { - task: message, - }); - return; - } - - this.bar.tick({ - task: message, - }); - - this.tickingPrevious = { - message, - remainder: this.tickParts, - interval: null, - }; - - this.tickingPrevious.remainder -= 1; - - this.tickingPrevious.interval = setInterval(() => { - if (this.tickingPrevious.remainder === 1) { - clearInterval(this.tickingPrevious.interval); - return; - } - - this.bar.tick({ - task: message, - }); - this.tickingPrevious.remainder -= 1; - }, 200); - } -} - -export default DishonestProgress; diff --git a/src/helpers/helpers.js b/src/helpers/helpers.js deleted file mode 100644 index ba525b85bc..0000000000 --- a/src/helpers/helpers.js +++ /dev/null @@ -1,107 +0,0 @@ -import os from 'os'; -import axios from 'axios'; -import hasBinary from 'hasbin'; -import path from 'path'; - -function isOSX() { - return os.platform() === 'darwin'; -} - -function isWindows() { - return os.platform() === 'win32'; -} - -function downloadFile(fileUrl) { - return axios - .get(fileUrl, { - responseType: 'arraybuffer', - }) - .then((response) => { - if (!response.data) { - return null; - } - return { - data: response.data, - ext: path.extname(fileUrl), - }; - }); -} - -function allowedIconFormats(platform) { - const hasIdentify = hasBinary.sync('identify'); - const hasConvert = hasBinary.sync('convert'); - const hasIconUtil = hasBinary.sync('iconutil'); - - const pngToIcns = hasConvert && hasIconUtil; - const pngToIco = hasConvert; - const icoToIcns = pngToIcns && hasIdentify; - const icoToPng = hasConvert; - - // todo scripts for the following - const icnsToPng = false; - const icnsToIco = false; - - const formats = []; - - // todo shell scripting is not supported on windows, temporary override - if (isWindows()) { - switch (platform) { - case 'darwin': - formats.push('.icns'); - break; - case 'linux': - formats.push('.png'); - break; - case 'win32': - formats.push('.ico'); - break; - default: - throw new Error( - `function allowedIconFormats error: Unknown platform ${platform}`, - ); - } - return formats; - } - - switch (platform) { - case 'darwin': - formats.push('.icns'); - if (pngToIcns) { - formats.push('.png'); - } - if (icoToIcns) { - formats.push('.ico'); - } - break; - case 'linux': - formats.push('.png'); - if (icoToPng) { - formats.push('.ico'); - } - if (icnsToPng) { - formats.push('.icns'); - } - break; - case 'win32': - formats.push('.ico'); - if (pngToIco) { - formats.push('.png'); - } - if (icnsToIco) { - formats.push('.icns'); - } - break; - default: - throw new Error( - `function allowedIconFormats error: Unknown platform ${platform}`, - ); - } - return formats; -} - -export default { - isOSX, - isWindows, - downloadFile, - allowedIconFormats, -}; diff --git a/src/helpers/helpers.ts b/src/helpers/helpers.ts new file mode 100644 index 0000000000..3f7df8a991 --- /dev/null +++ b/src/helpers/helpers.ts @@ -0,0 +1,141 @@ +import * as os from 'os'; +import * as path from 'path'; + +import axios from 'axios'; +import * as hasbin from 'hasbin'; +import { ncp } from 'ncp'; +import * as log from 'loglevel'; +import * as tmp from 'tmp'; +tmp.setGracefulCleanup(); // cleanup temp dirs even when an uncaught exception occurs + +const now = new Date(); +const TMP_TIME = `${now.getHours()}-${now.getMinutes()}-${now.getSeconds()}`; + +type DownloadResult = { + data: Buffer; + ext: string; +}; + +export function isOSX(): boolean { + return os.platform() === 'darwin'; +} + +export function isWindows(): boolean { + return os.platform() === 'win32'; +} + +/** + * Create a temp directory with a debug-friendly name, and return its path. + * Will be automatically deleted on exit. + */ +export function getTempDir(prefix: string, mode?: number): string { + return tmp.dirSync({ + mode, + unsafeCleanup: true, // recursively remove tmp dir on exit, even if not empty. + prefix: `nativefier-${TMP_TIME}-${prefix}-`, + }).name; +} + +export async function copyFileOrDir( + sourceFileOrDir: string, + dest: string, +): Promise { + return new Promise((resolve, reject) => { + ncp(sourceFileOrDir, dest, (error: any) => { + if (error) { + reject(error); + } + resolve(); + }); + }); +} + +export async function downloadFile(fileUrl: string): Promise { + log.debug(`Downloading ${fileUrl}`); + return axios + .get(fileUrl, { + responseType: 'arraybuffer', + }) + .then((response) => { + if (!response.data) { + return null; + } + return { + data: response.data, + ext: path.extname(fileUrl), + }; + }); +} + +export function getAllowedIconFormats(platform: string): string[] { + const hasIdentify = hasbin.sync('identify'); + const hasConvert = hasbin.sync('convert'); + const hasIconUtil = hasbin.sync('iconutil'); + + const pngToIcns = hasConvert && hasIconUtil; + const pngToIco = hasConvert; + const icoToIcns = pngToIcns && hasIdentify; + const icoToPng = hasConvert; + + // Unsupported + const icnsToPng = false; + const icnsToIco = false; + + const formats = []; + + // Shell scripting is not supported on windows, temporary override + if (isWindows()) { + switch (platform) { + case 'darwin': + formats.push('.icns'); + break; + case 'linux': + formats.push('.png'); + break; + case 'win32': + formats.push('.ico'); + break; + default: + throw new Error(`Unknown platform ${platform}`); + } + log.debug( + `Allowed icon formats when building for ${platform} (limited on Windows):`, + formats, + ); + return formats; + } + + switch (platform) { + case 'darwin': + formats.push('.icns'); + if (pngToIcns) { + formats.push('.png'); + } + if (icoToIcns) { + formats.push('.ico'); + } + break; + case 'linux': + formats.push('.png'); + if (icoToPng) { + formats.push('.ico'); + } + if (icnsToPng) { + formats.push('.icns'); + } + break; + case 'win32': + formats.push('.ico'); + if (pngToIco) { + formats.push('.png'); + } + if (icnsToIco) { + formats.push('.icns'); + } + break; + default: + throw new Error(`Unknown platform ${platform}`); + } + log.debug(`Allowed icon formats when building for ${platform}:`, formats); + return formats; +} diff --git a/src/helpers/iconShellHelpers.js b/src/helpers/iconShellHelpers.js deleted file mode 100644 index 4259d72046..0000000000 --- a/src/helpers/iconShellHelpers.js +++ /dev/null @@ -1,102 +0,0 @@ -import shell from 'shelljs'; -import path from 'path'; -import tmp from 'tmp'; -import helpers from './helpers'; - -const { isWindows, isOSX } = helpers; - -tmp.setGracefulCleanup(); - -const SCRIPT_PATHS = { - singleIco: path.join(__dirname, '../..', 'bin/singleIco'), - convertToPng: path.join(__dirname, '../..', 'bin/convertToPng'), - convertToIco: path.join(__dirname, '../..', 'bin/convertToIco'), - convertToIcns: path.join(__dirname, '../..', 'bin/convertToIcns'), -}; - -/** - * Executes a shell script with the form "./pathToScript param1 param2" - * @param {string} shellScriptPath - * @param {string} icoSrc input .ico - * @param {string} dest has to be a .ico path - */ -function iconShellHelper(shellScriptPath, icoSrc, dest) { - return new Promise((resolve, reject) => { - if (isWindows()) { - reject(new Error('OSX or Linux is required')); - return; - } - - shell.exec( - `"${shellScriptPath}" "${icoSrc}" "${dest}"`, - { silent: true }, - (exitCode, stdOut, stdError) => { - if (exitCode) { - // eslint-disable-next-line prefer-promise-reject-errors - reject({ - stdOut, - stdError, - }); - return; - } - - resolve(dest); - }, - ); - }); -} - -function getTmpDirPath() { - const tempIconDirObj = tmp.dirSync({ unsafeCleanup: true }); - return tempIconDirObj.name; -} - -/** - * Converts the ico to a temporary directory which will be cleaned up on process exit - * @param {string} icoSrc path to a .ico file - * @return {Promise} - */ - -function singleIco(icoSrc) { - return iconShellHelper( - SCRIPT_PATHS.singleIco, - icoSrc, - `${getTmpDirPath()}/icon.ico`, - ); -} - -function convertToPng(icoSrc) { - return iconShellHelper( - SCRIPT_PATHS.convertToPng, - icoSrc, - `${getTmpDirPath()}/icon.png`, - ); -} - -function convertToIco(icoSrc) { - return iconShellHelper( - SCRIPT_PATHS.convertToIco, - icoSrc, - `${getTmpDirPath()}/icon.ico`, - ); -} - -function convertToIcns(icoSrc) { - if (!isOSX()) { - return new Promise((resolve, reject) => - reject(new Error('OSX is required to convert to a .icns icon')), - ); - } - return iconShellHelper( - SCRIPT_PATHS.convertToIcns, - icoSrc, - `${getTmpDirPath()}/icon.icns`, - ); -} - -export default { - singleIco, - convertToPng, - convertToIco, - convertToIcns, -}; diff --git a/src/helpers/iconShellHelpers.ts b/src/helpers/iconShellHelpers.ts new file mode 100644 index 0000000000..059aca9378 --- /dev/null +++ b/src/helpers/iconShellHelpers.ts @@ -0,0 +1,89 @@ +import * as path from 'path'; + +import * as shell from 'shelljs'; + +import { isWindows, isOSX, getTempDir } from './helpers'; +import * as log from 'loglevel'; + +const SCRIPT_PATHS = { + singleIco: path.join(__dirname, '../..', 'icon-scripts/singleIco'), + convertToPng: path.join(__dirname, '../..', 'icon-scripts/convertToPng'), + convertToIco: path.join(__dirname, '../..', 'icon-scripts/convertToIco'), + convertToIcns: path.join(__dirname, '../..', 'icon-scripts/convertToIcns'), +}; + +/** + * Executes a shell script with the form "./pathToScript param1 param2" + */ +async function iconShellHelper( + shellScriptPath: string, + icoSource: string, + icoDestination: string, +): Promise { + return new Promise((resolve, reject) => { + if (isWindows()) { + reject( + new Error( + 'Icon conversion only supported on macOS or Linux. ' + + 'If building for Windows, download/create a .ico and pass it with --icon favicon.ico . ' + + 'If building for macOS/Linux, do it from macOS/Linux', + ), + ); + return; + } + + const shellCommand = `"${shellScriptPath}" "${icoSource}" "${icoDestination}"`; + log.debug( + `Converting icon ${icoSource} to ${icoDestination}.`, + `Calling: ${shellCommand}`, + ); + shell.exec(shellCommand, { silent: true }, (exitCode, stdOut, stdError) => { + if (exitCode) { + reject({ + stdOut, + stdError, + }); + return; + } + + log.debug(`Conversion succeeded and produced icon at ${icoDestination}`); + resolve(icoDestination); + }); + }); +} + +export function singleIco(icoSrc: string): Promise { + return iconShellHelper( + SCRIPT_PATHS.singleIco, + icoSrc, + `${getTempDir('iconconv')}/icon.ico`, + ); +} + +export async function convertToPng(icoSrc: string): Promise { + return iconShellHelper( + SCRIPT_PATHS.convertToPng, + icoSrc, + `${getTempDir('iconconv')}/icon.png`, + ); +} + +export async function convertToIco(icoSrc: string): Promise { + return iconShellHelper( + SCRIPT_PATHS.convertToIco, + icoSrc, + `${getTempDir('iconconv')}/icon.ico`, + ); +} + +export async function convertToIcns(icoSrc: string): Promise { + if (!isOSX()) { + throw new Error('macOS is required to convert to a .icns icon'); + } + + return iconShellHelper( + SCRIPT_PATHS.convertToIcns, + icoSrc, + `${getTempDir('iconconv')}/icon.icns`, + ); +} diff --git a/src/helpers/packagerConsole.js b/src/helpers/packagerConsole.js deleted file mode 100644 index 72006f1bd0..0000000000 --- a/src/helpers/packagerConsole.js +++ /dev/null @@ -1,31 +0,0 @@ -// TODO: remove this file and use quiet mode of new version of electron packager -const log = require('loglevel'); - -class PackagerConsole { - constructor() { - this.logs = []; - } - - // eslint-disable-next-line no-underscore-dangle - _log(...messages) { - this.logs.push(...messages); - } - - override() { - this.consoleError = log.error; - - // need to bind because somehow when _log() is called this refers to console - // eslint-disable-next-line no-underscore-dangle - log.error = this._log.bind(this); - } - - restore() { - log.error = this.consoleError; - } - - playback() { - log.log(this.logs.join(' ')); - } -} - -export default PackagerConsole; diff --git a/src/index.js b/src/index.js deleted file mode 100644 index c7cb271392..0000000000 --- a/src/index.js +++ /dev/null @@ -1,6 +0,0 @@ -import 'source-map-support/register'; -import 'babel-polyfill'; - -import buildApp from './build/buildMain'; - -export default buildApp; diff --git a/src/infer/index.js b/src/infer/index.js deleted file mode 100644 index 0495ef07f4..0000000000 --- a/src/infer/index.js +++ /dev/null @@ -1,4 +0,0 @@ -export { default as inferIcon } from './inferIcon'; -export { default as inferOs } from './inferOs'; -export { default as inferTitle } from './inferTitle'; -export { default as inferUserAgent } from './inferUserAgent'; diff --git a/src/infer/inferIcon.js b/src/infer/inferIcon.js deleted file mode 100644 index fa5f86b990..0000000000 --- a/src/infer/inferIcon.js +++ /dev/null @@ -1,130 +0,0 @@ -import pageIcon from 'page-icon'; -import path from 'path'; -import fs from 'fs'; -import tmp from 'tmp'; -import gitCloud from 'gitcloud'; -import helpers from '../helpers/helpers'; - -const { downloadFile, allowedIconFormats } = helpers; -tmp.setGracefulCleanup(); - -const GITCLOUD_SPACE_DELIMITER = '-'; - -function getMaxMatchScore(iconWithScores) { - return iconWithScores.reduce((maxScore, currentIcon) => { - const currentScore = currentIcon.score; - if (currentScore > maxScore) { - return currentScore; - } - return maxScore; - }, 0); -} - -/** - * also maps ext to icon object - */ -function getMatchingIcons(iconsWithScores, maxScore) { - return iconsWithScores - .filter((item) => item.score === maxScore) - .map((item) => Object.assign({}, item, { ext: path.extname(item.url) })); -} - -function mapIconWithMatchScore(fileIndex, targetUrl) { - const normalisedTargetUrl = targetUrl.toLowerCase(); - return fileIndex.map((item) => { - const itemWords = item.name.split(GITCLOUD_SPACE_DELIMITER); - const score = itemWords.reduce((currentScore, word) => { - if (normalisedTargetUrl.includes(word)) { - return currentScore + 1; - } - return currentScore; - }, 0); - - return Object.assign({}, item, { score }); - }); -} - -function inferIconFromStore(targetUrl, platform) { - const allowedFormats = new Set(allowedIconFormats(platform)); - - return gitCloud('https://jiahaog.github.io/nativefier-icons/').then( - (fileIndex) => { - const iconWithScores = mapIconWithMatchScore(fileIndex, targetUrl); - const maxScore = getMaxMatchScore(iconWithScores); - - if (maxScore === 0) { - return null; - } - - const iconsMatchingScore = getMatchingIcons(iconWithScores, maxScore); - const iconsMatchingExt = iconsMatchingScore.filter((icon) => - allowedFormats.has(icon.ext), - ); - const matchingIcon = iconsMatchingExt[0]; - const iconUrl = matchingIcon && matchingIcon.url; - - if (!iconUrl) { - return null; - } - return downloadFile(iconUrl); - }, - ); -} - -function writeFilePromise(outPath, data) { - return new Promise((resolve, reject) => { - fs.writeFile(outPath, data, (error) => { - if (error) { - reject(error); - return; - } - resolve(outPath); - }); - }); -} - -function inferFromPage(targetUrl, platform, outDir) { - let preferredExt = '.png'; - if (platform === 'win32') { - preferredExt = '.ico'; - } - - // todo might want to pass list of preferences instead - return pageIcon(targetUrl, { ext: preferredExt }).then((icon) => { - if (!icon) { - return null; - } - - const outfilePath = path.join(outDir, `/icon${icon.ext}`); - return writeFilePromise(outfilePath, icon.data); - }); -} - -/** - * - * @param {string} targetUrl - * @param {string} platform - * @param {string} outDir - */ -function inferIconFromUrlToPath(targetUrl, platform, outDir) { - return inferIconFromStore(targetUrl, platform).then((icon) => { - if (!icon) { - return inferFromPage(targetUrl, platform, outDir); - } - - const outfilePath = path.join(outDir, `/icon${icon.ext}`); - return writeFilePromise(outfilePath, icon.data); - }); -} - -/** - * @param {string} targetUrl - * @param {string} platform - */ -function inferIcon(targetUrl, platform) { - const tmpObj = tmp.dirSync({ unsafeCleanup: true }); - const tmpPath = tmpObj.name; - return inferIconFromUrlToPath(targetUrl, platform, tmpPath); -} - -export default inferIcon; diff --git a/src/infer/inferIcon.ts b/src/infer/inferIcon.ts new file mode 100644 index 0000000000..9dee947960 --- /dev/null +++ b/src/infer/inferIcon.ts @@ -0,0 +1,111 @@ +import * as path from 'path'; +import { writeFile } from 'fs'; +import { promisify } from 'util'; + +import * as gitCloud from 'gitcloud'; +import * as pageIcon from 'page-icon'; + +import { + downloadFile, + getAllowedIconFormats, + getTempDir, +} from '../helpers/helpers'; +import * as log from 'loglevel'; + +const writeFileAsync = promisify(writeFile); + +const GITCLOUD_SPACE_DELIMITER = '-'; +const GITCLOUD_URL = 'https://jiahaog.github.io/nativefier-icons/'; + +function getMaxMatchScore(iconWithScores: any[]): number { + const score = iconWithScores.reduce((maxScore, currentIcon) => { + const currentScore = currentIcon.score; + if (currentScore > maxScore) { + return currentScore; + } + return maxScore; + }, 0); + log.debug('Max icon match score:', score); + return score; +} + +function getMatchingIcons(iconsWithScores: any[], maxScore: number): any[] { + return iconsWithScores + .filter((item) => item.score === maxScore) + .map((item) => ({ ...item, ext: path.extname(item.url) })); +} + +function mapIconWithMatchScore(cloudIcons: any[], targetUrl: string): any { + const normalisedTargetUrl = targetUrl.toLowerCase(); + return cloudIcons.map((item) => { + const itemWords = item.name.split(GITCLOUD_SPACE_DELIMITER); + const score = itemWords.reduce((currentScore, word) => { + if (normalisedTargetUrl.includes(word)) { + return currentScore + 1; + } + return currentScore; + }, 0); + + return { ...item, score }; + }); +} + +async function inferIconFromStore( + targetUrl: string, + platform: string, +): Promise { + log.debug(`Inferring icon from store for ${targetUrl} on ${platform}`); + const allowedFormats = new Set(getAllowedIconFormats(platform)); + + const cloudIcons = await gitCloud(GITCLOUD_URL); + log.debug(`Got ${cloudIcons.length} icons from gitcloud`); + const iconWithScores = mapIconWithMatchScore(cloudIcons, targetUrl); + const maxScore = getMaxMatchScore(iconWithScores); + + if (maxScore === 0) { + log.debug('No relevant icon in store.'); + return null; + } + + const iconsMatchingScore = getMatchingIcons(iconWithScores, maxScore); + const iconsMatchingExt = iconsMatchingScore.filter((icon) => + allowedFormats.has(icon.ext), + ); + const matchingIcon = iconsMatchingExt[0]; + const iconUrl = matchingIcon && matchingIcon.url; + + if (!iconUrl) { + log.debug('Could not infer icon from store'); + return null; + } + return downloadFile(iconUrl); +} + +export async function inferIcon( + targetUrl: string, + platform: string, +): Promise { + log.debug(`Inferring icon for ${targetUrl} on ${platform}`); + const tmpDirPath = getTempDir('iconinfer'); + + let icon: { ext: string; data: Buffer } = await inferIconFromStore( + targetUrl, + platform, + ); + if (!icon) { + const ext = platform === 'win32' ? '.ico' : '.png'; + log.debug(`Trying to extract a ${ext} icon from the page.`); + icon = await pageIcon(targetUrl, { ext }); + } + if (!icon) { + return null; + } + log.debug(`Got an icon from the page.`); + + const iconPath = path.join(tmpDirPath, `/icon${icon.ext}`); + log.debug( + `Writing ${(icon.data.length / 1024).toFixed(1)} kb icon to ${iconPath}`, + ); + await writeFileAsync(iconPath, icon.data); + return iconPath; +} diff --git a/src/infer/inferOs.js b/src/infer/inferOs.ts similarity index 63% rename from src/infer/inferOs.js rename to src/infer/inferOs.ts index 7f639da67b..7bd903850d 100644 --- a/src/infer/inferOs.js +++ b/src/infer/inferOs.ts @@ -1,28 +1,27 @@ -import os from 'os'; +import * as os from 'os'; +import * as log from 'loglevel'; -function inferPlatform() { +export function inferPlatform(): string { const platform = os.platform(); if ( platform === 'darwin' || + // @ts-ignore platform === 'mas' || platform === 'win32' || platform === 'linux' ) { + log.debug('Inferred platform', platform); return platform; } throw new Error(`Untested platform ${platform} detected`); } -function inferArch() { +export function inferArch(): string { const arch = os.arch(); if (arch !== 'ia32' && arch !== 'x64' && arch !== 'arm') { throw new Error(`Incompatible architecture ${arch} detected`); } + log.debug('Inferred arch', arch); return arch; } - -export default { - inferPlatform, - inferArch, -}; diff --git a/src/infer/inferTitle.js b/src/infer/inferTitle.js deleted file mode 100644 index 7636f930a5..0000000000 --- a/src/infer/inferTitle.js +++ /dev/null @@ -1,25 +0,0 @@ -import axios from 'axios'; -import cheerio from 'cheerio'; - -const USER_AGENT = - 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2227.1 Safari/537.36'; - -function inferTitle(url) { - const options = { - method: 'get', - url, - headers: { - // fake a user agent because pages like http://messenger.com will throw 404 error - 'User-Agent': USER_AGENT, - }, - }; - - return axios(options).then(({ data }) => { - const $ = cheerio.load(data); - return $('title') - .first() - .text(); - }); -} - -export default inferTitle; diff --git a/src/infer/inferTitle.test.js b/src/infer/inferTitle.test.js deleted file mode 100644 index 700cc7793f..0000000000 --- a/src/infer/inferTitle.test.js +++ /dev/null @@ -1,21 +0,0 @@ -import axios from 'axios'; -import inferTitle from './inferTitle'; - -jest.mock('axios', () => - jest.fn(() => - Promise.resolve({ - data: ` - - - TEST_TITLE - - `, - }), - ), -); - -test('it returns the correct title', async () => { - const result = await inferTitle('someurl'); - expect(axios).toHaveBeenCalledTimes(1); - expect(result).toBe('TEST_TITLE'); -}); diff --git a/src/infer/inferTitle.test.ts b/src/infer/inferTitle.test.ts new file mode 100644 index 0000000000..c7153b7b9d --- /dev/null +++ b/src/infer/inferTitle.test.ts @@ -0,0 +1,19 @@ +import axios from 'axios'; + +import { inferTitle } from './inferTitle'; + +test('it returns the correct title', async () => { + const axiosGetMock = jest.spyOn(axios, 'get'); + axiosGetMock.mockResolvedValue({ + data: ` + + + TEST_TITLE + + `, + }); + const result = await inferTitle('someurl'); + + expect(axiosGetMock).toHaveBeenCalledTimes(1); + expect(result).toBe('TEST_TITLE'); +}); diff --git a/src/infer/inferTitle.ts b/src/infer/inferTitle.ts new file mode 100644 index 0000000000..1eee360212 --- /dev/null +++ b/src/infer/inferTitle.ts @@ -0,0 +1,23 @@ +import axios from 'axios'; +import * as cheerio from 'cheerio'; +import * as log from 'loglevel'; + +const USER_AGENT = + 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2227.1 Safari/537.36'; + +export async function inferTitle(url: string): Promise { + const { data } = await axios.get(url, { + headers: { + // Fake user agent for pages like http://messenger.com + 'User-Agent': USER_AGENT, + }, + }); + log.debug(`Fetched ${(data.length / 1024).toFixed(1)} kb page at`, url); + const $ = cheerio.load(data); + const inferredTitle = $('title') + .first() + .text(); + + log.debug('Inferred title:', inferredTitle); + return inferredTitle; +} diff --git a/src/infer/inferUserAgent.js b/src/infer/inferUserAgent.js deleted file mode 100644 index 44f78b6f05..0000000000 --- a/src/infer/inferUserAgent.js +++ /dev/null @@ -1,69 +0,0 @@ -import axios from 'axios'; -import _ from 'lodash'; -import log from 'loglevel'; - -const ELECTRON_VERSIONS_URL = 'https://atom.io/download/atom-shell/index.json'; -const DEFAULT_CHROME_VERSION = '61.0.3163.100'; - -function getChromeVersionForElectronVersion( - electronVersion, - url = ELECTRON_VERSIONS_URL, -) { - return axios.get(url, { timeout: 5000 }).then((response) => { - if (response.status !== 200) { - throw new Error(`Bad request: Status code ${response.status}`); - } - - const { data } = response; - const electronVersionToChromeVersion = _.zipObject( - data.map((d) => d.version), - data.map((d) => d.chrome), - ); - - if (!(electronVersion in electronVersionToChromeVersion)) { - throw new Error( - `Electron version '${electronVersion}' not found in retrieved version list!`, - ); - } - - return electronVersionToChromeVersion[electronVersion]; - }); -} - -export function getUserAgentString(chromeVersion, platform) { - let userAgent; - switch (platform) { - case 'darwin': - case 'mas': - userAgent = `Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion} Safari/537.36`; - break; - case 'win32': - userAgent = `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion} Safari/537.36`; - break; - case 'linux': - userAgent = `Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion} Safari/537.36`; - break; - default: - throw new Error( - 'Error invalid platform specified to getUserAgentString()', - ); - } - return userAgent; -} - -function inferUserAgent( - electronVersion, - platform, - url = ELECTRON_VERSIONS_URL, -) { - return getChromeVersionForElectronVersion(electronVersion, url) - .then((chromeVersion) => getUserAgentString(chromeVersion, platform)) - .catch(() => { - log.warn( - `Unable to infer chrome version for user agent, using ${DEFAULT_CHROME_VERSION}`, - ); - return getUserAgentString(DEFAULT_CHROME_VERSION, platform); - }); -} - -export default inferUserAgent; diff --git a/src/infer/inferUserAgent.test.js b/src/infer/inferUserAgent.test.js deleted file mode 100644 index 8ae0f2b080..0000000000 --- a/src/infer/inferUserAgent.test.js +++ /dev/null @@ -1,37 +0,0 @@ -import _ from 'lodash'; -import inferUserAgent from './inferUserAgent'; - -const TEST_RESULT = { - darwin: - 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.75 Safari/537.36', - mas: - 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.75 Safari/537.36', - win32: - 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.75 Safari/537.36', - linux: - 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.75 Safari/537.36', -}; - -function testPlatform(platform) { - return expect(inferUserAgent('0.37.1', platform)).resolves.toBe( - TEST_RESULT[platform], - ); -} - -describe('Infer User Agent', () => { - test('Can infer userAgent for all platforms', async () => { - const testPromises = _.keys(TEST_RESULT).map((platform) => - testPlatform(platform), - ); - await Promise.all(testPromises); - }); - - test('Connection error will still get a user agent', async () => { - jest.setTimeout(6000); - - const TIMEOUT_URL = 'http://www.google.com:81/'; - await expect(inferUserAgent('1.6.7', 'darwin', TIMEOUT_URL)).resolves.toBe( - 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/61.0.3163.100 Safari/537.36', - ); - }); -}); diff --git a/src/infer/inferUserAgent.test.ts b/src/infer/inferUserAgent.test.ts new file mode 100644 index 0000000000..8ac9d29cc8 --- /dev/null +++ b/src/infer/inferUserAgent.test.ts @@ -0,0 +1,29 @@ +import { inferUserAgent } from './inferUserAgent'; +import { DEFAULT_ELECTRON_VERSION, DEFAULT_CHROME_VERSION } from '../constants'; + +const EXPECTED_USERAGENTS = { + darwin: `Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${DEFAULT_CHROME_VERSION} Safari/537.36`, + mas: `Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${DEFAULT_CHROME_VERSION} Safari/537.36`, + win32: `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${DEFAULT_CHROME_VERSION} Safari/537.36`, + linux: `Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${DEFAULT_CHROME_VERSION} Safari/537.36`, +}; + +describe('Infer User Agent', () => { + test('Can infer userAgent for all platforms', async () => { + jest.setTimeout(10000); + for (const [arch, archUa] of Object.entries(EXPECTED_USERAGENTS)) { + const ua = await inferUserAgent(DEFAULT_ELECTRON_VERSION, arch); + expect(ua).toBe(archUa); + } + }); + + // TODO make fast by mocking timeout, and un-skip + test.skip('Connection error will still get a user agent', async () => { + jest.setTimeout(6000); + + const TIMEOUT_URL = 'http://www.google.com:81/'; + await expect(inferUserAgent('1.6.7', 'darwin', TIMEOUT_URL)).resolves.toBe( + 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/61.0.3163.100 Safari/537.36', + ); + }); +}); diff --git a/src/infer/inferUserAgent.ts b/src/infer/inferUserAgent.ts new file mode 100644 index 0000000000..4213e899ba --- /dev/null +++ b/src/infer/inferUserAgent.ts @@ -0,0 +1,82 @@ +import * as _ from 'lodash'; +import axios from 'axios'; +import * as log from 'loglevel'; +import { DEFAULT_CHROME_VERSION } from '../constants'; + +const ELECTRON_VERSIONS_URL = 'https://atom.io/download/atom-shell/index.json'; + +async function getChromeVersionForElectronVersion( + electronVersion: string, + url = ELECTRON_VERSIONS_URL, +): Promise { + log.debug('Grabbing electron<->chrome versions file from', url); + const response = await axios.get(url, { timeout: 5000 }); + if (response.status !== 200) { + throw new Error(`Bad request: Status code ${response.status}`); + } + const { data } = response; + const electronVersionToChromeVersion: _.Dictionary = _.zipObject( + data.map((d) => d.version), + data.map((d) => d.chrome), + ); + if (!(electronVersion in electronVersionToChromeVersion)) { + throw new Error( + `Electron version '${electronVersion}' not found in retrieved version list!`, + ); + } + const chromeVersion = electronVersionToChromeVersion[electronVersion]; + log.debug( + `Associated electron v${electronVersion} to chrome v${chromeVersion}`, + ); + return chromeVersion; +} + +export function getUserAgentString( + chromeVersion: string, + platform: string, +): string { + let userAgent: string; + switch (platform) { + case 'darwin': + case 'mas': + userAgent = `Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion} Safari/537.36`; + break; + case 'win32': + userAgent = `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion} Safari/537.36`; + break; + case 'linux': + userAgent = `Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion} Safari/537.36`; + break; + default: + throw new Error( + 'Error invalid platform specified to getUserAgentString()', + ); + } + log.debug( + `Given chrome ${chromeVersion} on ${platform},`, + `using user agent: ${userAgent}`, + ); + return userAgent; +} + +export async function inferUserAgent( + electronVersion: string, + platform: string, + url = ELECTRON_VERSIONS_URL, +): Promise { + log.debug( + `Inferring user agent for electron ${electronVersion} / ${platform}`, + ); + try { + const chromeVersion = await getChromeVersionForElectronVersion( + electronVersion, + url, + ); + return getUserAgentString(chromeVersion, platform); + } catch (e) { + log.warn( + `Unable to infer chrome version for user agent, using ${DEFAULT_CHROME_VERSION}`, + ); + return getUserAgentString(DEFAULT_CHROME_VERSION, platform); + } +} diff --git a/src/integration-test.ts b/src/integration-test.ts new file mode 100644 index 0000000000..25eb164333 --- /dev/null +++ b/src/integration-test.ts @@ -0,0 +1,57 @@ +import * as fs from 'fs'; +import * as path from 'path'; + +import { getTempDir } from './helpers/helpers'; +import { buildNativefierApp } from './main'; + +function checkApp(appRoot: string, inputOptions: any): void { + let relativeAppFolder: string; + + switch (inputOptions.platform) { + case 'darwin': + relativeAppFolder = path.join('Google.app', 'Contents/Resources/app'); + break; + case 'linux': + relativeAppFolder = 'resources/app'; + break; + case 'win32': + relativeAppFolder = 'resources/app'; + break; + default: + throw new Error('Unknown app platform'); + } + + const appPath = path.join(appRoot, relativeAppFolder); + + const configPath = path.join(appPath, 'nativefier.json'); + const nativefierConfig = JSON.parse(fs.readFileSync(configPath).toString()); + expect(inputOptions.targetUrl).toBe(nativefierConfig.targetUrl); + + // Test name inferring + expect(nativefierConfig.name).toBe('Google'); + + // Test icon writing + const iconFile = + inputOptions.platform === 'darwin' ? '../electron.icns' : 'icon.png'; + const iconPath = path.join(appPath, iconFile); + expect(fs.existsSync(iconPath)).toBe(true); + expect(fs.statSync(iconPath).size).toBeGreaterThan(1000); +} + +describe('Nativefier', () => { + jest.setTimeout(300000); + + test('builds a Nativefier app for several platforms', async () => { + for (const platform of ['darwin', 'linux']) { + const tempDirectory = getTempDir('integtest'); + const options = { + targetUrl: 'https://google.com/', + out: tempDirectory, + overwrite: true, + platform, + }; + const appPath = await buildNativefierApp(options); + checkApp(appPath, options); + } + }); +}); diff --git a/src/jestSetupFiles.ts b/src/jestSetupFiles.ts new file mode 100644 index 0000000000..c2cb9fdf05 --- /dev/null +++ b/src/jestSetupFiles.ts @@ -0,0 +1,7 @@ +import * as log from 'loglevel'; + +if (process.env.LOGLEVEL) { + log.setLevel(process.env.LOGLEVEL as log.LogLevelDesc); +} else { + log.disableAll(); +} diff --git a/src/main.ts b/src/main.ts new file mode 100644 index 0000000000..db5b1b2a8d --- /dev/null +++ b/src/main.ts @@ -0,0 +1,20 @@ +import 'source-map-support/register'; + +import { buildNativefierApp } from './build/buildNativefierApp'; + +export { buildNativefierApp }; + +/** + * Only for compatibility with Nativefier <= 7.7.1 ! + * Use the better, modern async `buildNativefierApp` instead if you can! + */ +function buildNativefierAppOldCallbackStyle( + options: any, + callback: (err: any, result?: any) => void, +): void { + buildNativefierApp(options) + .then((result) => callback(null, result)) + .catch((err) => callback(err)); +} + +export default buildNativefierAppOldCallbackStyle; diff --git a/src/options/asyncConfig.js b/src/options/asyncConfig.js deleted file mode 100644 index 597ea84274..0000000000 --- a/src/options/asyncConfig.js +++ /dev/null @@ -1,22 +0,0 @@ -import fields from './fields'; - -function resultArrayToObject(fieldResults) { - return fieldResults.reduce( - (accumulator, value) => Object.assign({}, accumulator, value), - {}, - ); -} - -function inferredOptions(oldOptions, fieldResults) { - const newOptions = resultArrayToObject(fieldResults); - return Object.assign({}, oldOptions, newOptions); -} - -// Takes the options object and infers new values -// which may need async work -export default function(options) { - const tasks = fields(options); - return Promise.all(tasks).then((fieldResults) => - inferredOptions(options, fieldResults), - ); -} diff --git a/src/options/asyncConfig.test.js b/src/options/asyncConfig.test.js deleted file mode 100644 index 11b64f4bea..0000000000 --- a/src/options/asyncConfig.test.js +++ /dev/null @@ -1,18 +0,0 @@ -import asyncConfig from './asyncConfig'; -import fields from './fields'; - -jest.mock('./fields'); - -fields.mockImplementation(() => [ - Promise.resolve({ - someField: 'newValue', - }), -]); - -test('it should merge the result of the promise', async () => { - const param = { another: 'field', someField: 'oldValue' }; - const expected = { another: 'field', someField: 'newValue' }; - - const result = await asyncConfig(param); - expect(result).toEqual(expected); -}); diff --git a/src/options/asyncConfig.ts b/src/options/asyncConfig.ts new file mode 100644 index 0000000000..7e4edec2d3 --- /dev/null +++ b/src/options/asyncConfig.ts @@ -0,0 +1,12 @@ +import * as log from 'loglevel'; + +import { processOptions } from './fields/fields'; +import { AppOptions } from './model'; + +/** + * Takes the options object and infers new values needing async work + */ +export async function asyncConfig(options: AppOptions): Promise { + log.debug('\nPerforming async options post-processing.'); + await processOptions(options); +} diff --git a/src/options/fields/fields.test.ts b/src/options/fields/fields.test.ts new file mode 100644 index 0000000000..7ac935717f --- /dev/null +++ b/src/options/fields/fields.test.ts @@ -0,0 +1,36 @@ +import { processOptions } from './fields'; + +test('fully-defined async options are returned as-is', async () => { + const options = { + packager: { + icon: '/my/icon.png', + name: 'my beautiful app ', + targetUrl: 'https://myurl.com', + dir: '/tmp/myapp', + }, + nativefier: { userAgent: 'random user agent' }, + }; + // @ts-ignore + await processOptions(options); + + expect(options.packager.icon).toEqual('/my/icon.png'); + expect(options.packager.name).toEqual('my beautiful app'); + expect(options.nativefier.userAgent).toEqual('random user agent'); +}); + +test('user agent is inferred if not passed', async () => { + const options = { + packager: { + icon: '/my/icon.png', + name: 'my beautiful app ', + targetUrl: 'https://myurl.com', + dir: '/tmp/myapp', + platform: 'linux', + }, + nativefier: { userAgent: undefined }, + }; + // @ts-ignore + await processOptions(options); + + expect(options.nativefier.userAgent).toMatch(/Linux.*Chrome/); +}); diff --git a/src/options/fields/fields.ts b/src/options/fields/fields.ts new file mode 100644 index 0000000000..0ff7718b61 --- /dev/null +++ b/src/options/fields/fields.ts @@ -0,0 +1,29 @@ +import { icon } from './icon'; +import { userAgent } from './userAgent'; +import { AppOptions } from '../model'; +import { name } from './name'; + +const OPTION_POSTPROCESSORS = [ + { namespace: 'nativefier', option: 'userAgent', processor: userAgent }, + { namespace: 'packager', option: 'icon', processor: icon }, + { namespace: 'packager', option: 'name', processor: name }, +]; + +export async function processOptions(options: AppOptions): Promise { + const processedOptions = await Promise.all( + OPTION_POSTPROCESSORS.map(async ({ namespace, option, processor }) => { + const result = await processor(options); + return { + namespace, + option, + result, + }; + }), + ); + + for (const { namespace, option, result } of processedOptions) { + if (result !== null) { + options[namespace][option] = result; + } + } +} diff --git a/src/options/fields/icon.js b/src/options/fields/icon.js deleted file mode 100644 index 9303685297..0000000000 --- a/src/options/fields/icon.js +++ /dev/null @@ -1,14 +0,0 @@ -import log from 'loglevel'; -import { inferIcon } from '../../infer'; - -export default function({ icon, targetUrl, platform }) { - // Icon is the path to the icon - if (icon) { - return Promise.resolve(icon); - } - - return inferIcon(targetUrl, platform).catch((error) => { - log.warn('Cannot automatically retrieve the app icon:', error); - return null; - }); -} diff --git a/src/options/fields/icon.test.js b/src/options/fields/icon.test.js deleted file mode 100644 index 35f1c07b8d..0000000000 --- a/src/options/fields/icon.test.js +++ /dev/null @@ -1,43 +0,0 @@ -import log from 'loglevel'; -import icon from './icon'; -import { inferIcon } from '../../infer'; - -jest.mock('./../../infer/inferIcon'); -jest.mock('loglevel'); - -const mockedResult = 'icon path'; - -describe('when the icon parameter is passed', () => { - test('it should return the icon parameter', async () => { - expect(inferIcon).toHaveBeenCalledTimes(0); - - const params = { icon: './icon.png' }; - await expect(icon(params)).resolves.toBe(params.icon); - }); -}); - -describe('when the icon parameter is not passed', () => { - test('it should call inferIcon', async () => { - inferIcon.mockImplementationOnce(() => Promise.resolve(mockedResult)); - const params = { targetUrl: 'some url', platform: 'mac' }; - - const result = await icon(params); - - expect(result).toBe(mockedResult); - expect(inferIcon).toHaveBeenCalledWith(params.targetUrl, params.platform); - }); - - describe('when inferIcon resolves with an error', () => { - test('it should handle the error', async () => { - inferIcon.mockImplementationOnce(() => - Promise.reject(new Error('some error')), - ); - const params = { targetUrl: 'some url', platform: 'mac' }; - - const result = await icon(params); - expect(result).toBe(null); - expect(inferIcon).toHaveBeenCalledWith(params.targetUrl, params.platform); - expect(log.warn).toHaveBeenCalledTimes(1); - }); - }); -}); diff --git a/src/options/fields/icon.test.ts b/src/options/fields/icon.test.ts new file mode 100644 index 0000000000..ca1f300eb8 --- /dev/null +++ b/src/options/fields/icon.test.ts @@ -0,0 +1,60 @@ +import * as log from 'loglevel'; + +import { icon } from './icon'; +import { inferIcon } from '../../infer/inferIcon'; + +jest.mock('./../../infer/inferIcon'); +jest.mock('loglevel'); + +const mockedResult = 'icon path'; +const ICON_PARAMS_PROVIDED = { + packager: { + icon: './icon.png', + targetUrl: 'https://google.com', + platform: 'mac', + }, +}; +const ICON_PARAMS_NEEDS_INFER = { + packager: { + targetUrl: 'https://google.com', + platform: 'mac', + }, +}; + +describe('when the icon parameter is passed', () => { + test('it should return the icon parameter', async () => { + expect(inferIcon).toHaveBeenCalledTimes(0); + await expect(icon(ICON_PARAMS_PROVIDED)).resolves.toBe(null); + }); +}); + +describe('when the icon parameter is not passed', () => { + test('it should call inferIcon', async () => { + (inferIcon as jest.Mock).mockImplementationOnce(() => + Promise.resolve(mockedResult), + ); + const result = await icon(ICON_PARAMS_NEEDS_INFER); + + expect(result).toBe(mockedResult); + expect(inferIcon).toHaveBeenCalledWith( + ICON_PARAMS_NEEDS_INFER.packager.targetUrl, + ICON_PARAMS_NEEDS_INFER.packager.platform, + ); + }); + + describe('when inferIcon resolves with an error', () => { + test('it should handle the error', async () => { + (inferIcon as jest.Mock).mockImplementationOnce(() => + Promise.reject(new Error('some error')), + ); + const result = await icon(ICON_PARAMS_NEEDS_INFER); + + expect(result).toBe(null); + expect(inferIcon).toHaveBeenCalledWith( + ICON_PARAMS_NEEDS_INFER.packager.targetUrl, + ICON_PARAMS_NEEDS_INFER.packager.platform, + ); + expect(log.warn).toHaveBeenCalledTimes(1); // eslint-disable-line @typescript-eslint/unbound-method + }); + }); +}); diff --git a/src/options/fields/icon.ts b/src/options/fields/icon.ts new file mode 100644 index 0000000000..1a568d4d52 --- /dev/null +++ b/src/options/fields/icon.ts @@ -0,0 +1,28 @@ +import * as log from 'loglevel'; + +import { inferIcon } from '../../infer/inferIcon'; + +type IconParams = { + packager: { + icon?: string; + targetUrl: string; + platform?: string; + }; +}; + +export async function icon(options: IconParams): Promise { + if (options.packager.icon) { + log.debug('Got icon from options. Using it, no inferring needed'); + return null; + } + + try { + return await inferIcon( + options.packager.targetUrl, + options.packager.platform, + ); + } catch (error) { + log.warn('Cannot automatically retrieve the app icon:', error); + return null; + } +} diff --git a/src/options/fields/index.js b/src/options/fields/index.js deleted file mode 100644 index e8cd312f2e..0000000000 --- a/src/options/fields/index.js +++ /dev/null @@ -1,32 +0,0 @@ -import icon from './icon'; -import userAgent from './userAgent'; -import name from './name'; - -const fields = [ - { - field: 'userAgent', - task: userAgent, - }, - { - field: 'icon', - task: icon, - }, - { - field: 'name', - task: name, - }, -]; - -// Modifies the result of each promise from a scalar -// value to a object containing its fieldname -function wrap(fieldName, promise, args) { - return promise(args).then((result) => ({ - [fieldName]: result, - })); -} - -// Returns a list of promises which will all resolve -// with the following result: {[fieldName]: fieldvalue} -export default function(options) { - return fields.map(({ field, task }) => wrap(field, task, options)); -} diff --git a/src/options/fields/index.test.js b/src/options/fields/index.test.js deleted file mode 100644 index 045857e0f5..0000000000 --- a/src/options/fields/index.test.js +++ /dev/null @@ -1,21 +0,0 @@ -import fields from './index'; -import icon from './icon'; -import userAgent from './userAgent'; -import name from './name'; - -jest.mock('./icon'); -jest.mock('./name'); -jest.mock('./userAgent'); - -const modules = [icon, userAgent, name]; -modules.forEach((module) => { - module.mockImplementation(() => Promise.resolve()); -}); - -test('it should return a list of promises', () => { - const result = fields({}); - expect(result).toHaveLength(3); - result.forEach((value) => { - expect(value).toBeInstanceOf(Promise); - }); -}); diff --git a/src/options/fields/name.js b/src/options/fields/name.js deleted file mode 100644 index 62c8c2146c..0000000000 --- a/src/options/fields/name.js +++ /dev/null @@ -1,26 +0,0 @@ -import log from 'loglevel'; -import { sanitizeFilename } from '../../utils'; -import { inferTitle } from '../../infer'; -import { DEFAULT_APP_NAME } from '../../constants'; - -function tryToInferName({ name, targetUrl }) { - // .length also checks if its the commanderJS function or a string - if (name && name.length > 0) { - return Promise.resolve(name); - } - - return inferTitle(targetUrl) - .then((pageTitle) => pageTitle || DEFAULT_APP_NAME) - .catch((error) => { - log.warn( - `Unable to automatically determine app name, falling back to '${DEFAULT_APP_NAME}'. Reason: ${error}`, - ); - return DEFAULT_APP_NAME; - }); -} - -export default function({ platform, name, targetUrl }) { - return tryToInferName({ name, targetUrl }).then((result) => - sanitizeFilename(platform, result), - ); -} diff --git a/src/options/fields/name.test.js b/src/options/fields/name.test.js deleted file mode 100644 index 12673ecb02..0000000000 --- a/src/options/fields/name.test.js +++ /dev/null @@ -1,93 +0,0 @@ -import log from 'loglevel'; -import name from './name'; -import { DEFAULT_APP_NAME } from '../../constants'; -import { inferTitle } from '../../infer'; -import { sanitizeFilename } from '../../utils'; - -jest.mock('./../../infer/inferTitle'); -jest.mock('./../../utils/sanitizeFilename'); -jest.mock('loglevel'); - -sanitizeFilename.mockImplementation((_, filename) => filename); - -const mockedResult = 'mock name'; - -describe('well formed name parameters', () => { - const params = { name: 'appname', platform: 'something' }; - test('it should not call inferTitle', async () => { - const result = await name(params); - - expect(inferTitle).toHaveBeenCalledTimes(0); - expect(result).toBe(params.name); - }); - - test('it should call sanitize filename', async () => { - const result = await name(params); - expect(sanitizeFilename).toHaveBeenCalledWith(params.platform, result); - }); -}); - -describe('bad name parameters', () => { - beforeEach(() => { - inferTitle.mockImplementationOnce(() => Promise.resolve(mockedResult)); - }); - - const params = { targetUrl: 'some url' }; - describe('when the name is undefined', () => { - test('it should call inferTitle', async () => { - await name(params); - expect(inferTitle).toHaveBeenCalledWith(params.targetUrl); - }); - }); - - describe('when the name is an empty string', () => { - test('it should call inferTitle', async () => { - const testParams = { - ...params, - name: '', - }; - - await name(testParams); - expect(inferTitle).toHaveBeenCalledWith(params.targetUrl); - }); - }); - - test('it should call sanitize filename', () => - name(params).then((result) => { - expect(sanitizeFilename).toHaveBeenCalledWith(params.platform, result); - })); -}); - -describe('handling inferTitle results', () => { - const params = { targetUrl: 'some url', name: '', platform: 'something' }; - test('it should return the result from inferTitle', async () => { - inferTitle.mockImplementationOnce(() => Promise.resolve(mockedResult)); - - const result = await name(params); - expect(result).toBe(mockedResult); - expect(inferTitle).toHaveBeenCalledWith(params.targetUrl); - }); - - describe('when the returned pageTitle is falsey', () => { - test('it should return the default app name', async () => { - inferTitle.mockImplementationOnce(() => Promise.resolve(null)); - - const result = await name(params); - expect(result).toBe(DEFAULT_APP_NAME); - expect(inferTitle).toHaveBeenCalledWith(params.targetUrl); - }); - }); - - describe('when inferTitle resolves with an error', () => { - test('it should return the default app name', async () => { - inferTitle.mockImplementationOnce(() => - Promise.reject(new Error('some error')), - ); - - const result = await name(params); - expect(result).toBe(DEFAULT_APP_NAME); - expect(inferTitle).toHaveBeenCalledWith(params.targetUrl); - expect(log.warn).toHaveBeenCalledTimes(1); - }); - }); -}); diff --git a/src/options/fields/name.test.ts b/src/options/fields/name.test.ts new file mode 100644 index 0000000000..c7758c4db9 --- /dev/null +++ b/src/options/fields/name.test.ts @@ -0,0 +1,108 @@ +import * as log from 'loglevel'; + +import { name } from './name'; +import { DEFAULT_APP_NAME } from '../../constants'; +import { inferTitle } from '../../infer/inferTitle'; +import { sanitizeFilename } from '../../utils/sanitizeFilename'; + +jest.mock('./../../infer/inferTitle'); +jest.mock('./../../utils/sanitizeFilename'); +jest.mock('loglevel'); + +const inferTitleMockedResult = 'mock name'; +const NAME_PARAMS_PROVIDED = { + packager: { + name: 'appname', + targetUrl: 'https://google.com', + platform: 'linux', + }, +}; +const NAME_PARAMS_NEEDS_INFER = { + packager: { + targetUrl: 'https://google.com', + platform: 'mac', + }, +}; +beforeAll(() => { + (sanitizeFilename as jest.Mock).mockImplementation((_, filename) => filename); +}); + +describe('well formed name parameters', () => { + test('it should not call inferTitle', async () => { + const result = await name(NAME_PARAMS_PROVIDED); + + expect(inferTitle).toHaveBeenCalledTimes(0); + expect(result).toBe(NAME_PARAMS_PROVIDED.packager.name); + }); + + test('it should call sanitize filename', async () => { + const result = await name(NAME_PARAMS_PROVIDED); + + expect(sanitizeFilename).toHaveBeenCalledWith( + NAME_PARAMS_PROVIDED.packager.platform, + result, + ); + }); +}); + +describe('bad name parameters', () => { + beforeEach(() => { + (inferTitle as jest.Mock).mockResolvedValue(inferTitleMockedResult); + }); + + const params = { packager: { targetUrl: 'some url', platform: 'whatever' } }; + test('it should call inferTitle when the name is undefined', async () => { + await name(params); + expect(inferTitle).toHaveBeenCalledWith(params.packager.targetUrl); + }); + + test('it should call inferTitle when the name is an empty string', async () => { + const testParams = { + ...params, + name: '', + }; + + await name(testParams); + expect(inferTitle).toHaveBeenCalledWith(params.packager.targetUrl); + }); + + test('it should call sanitize filename', async () => { + const result = await name(params); + expect(sanitizeFilename).toHaveBeenCalledWith( + params.packager.platform, + result, + ); + }); +}); + +describe('handling inferTitle results', () => { + test('it should return the result from inferTitle', async () => { + const result = await name(NAME_PARAMS_NEEDS_INFER); + + expect(result).toEqual(inferTitleMockedResult); + expect(inferTitle).toHaveBeenCalledWith( + NAME_PARAMS_NEEDS_INFER.packager.targetUrl, + ); + }); + + test('it should return the default app name when the returned pageTitle is falsey', async () => { + (inferTitle as jest.Mock).mockResolvedValue(null); + const result = await name(NAME_PARAMS_NEEDS_INFER); + + expect(result).toEqual(DEFAULT_APP_NAME); + expect(inferTitle).toHaveBeenCalledWith( + NAME_PARAMS_NEEDS_INFER.packager.targetUrl, + ); + }); + + test('it should return the default app name when inferTitle rejects', async () => { + (inferTitle as jest.Mock).mockRejectedValue('some error'); + const result = await name(NAME_PARAMS_NEEDS_INFER); + + expect(result).toEqual(DEFAULT_APP_NAME); + expect(inferTitle).toHaveBeenCalledWith( + NAME_PARAMS_NEEDS_INFER.packager.targetUrl, + ); + expect(log.warn).toHaveBeenCalledTimes(1); // eslint-disable-line @typescript-eslint/unbound-method + }); +}); diff --git a/src/options/fields/name.ts b/src/options/fields/name.ts new file mode 100644 index 0000000000..d63208f3ef --- /dev/null +++ b/src/options/fields/name.ts @@ -0,0 +1,38 @@ +import * as log from 'loglevel'; + +import { sanitizeFilename } from '../../utils/sanitizeFilename'; +import { inferTitle } from '../../infer/inferTitle'; +import { DEFAULT_APP_NAME } from '../../constants'; + +type NameParams = { + packager: { + name?: string; + platform?: string; + targetUrl: string; + }; +}; + +async function tryToInferName(targetUrl: string): Promise { + try { + log.debug('Inferring name for', targetUrl); + const pageTitle = await inferTitle(targetUrl); + return pageTitle || DEFAULT_APP_NAME; + } catch (error) { + log.warn( + `Unable to automatically determine app name, falling back to '${DEFAULT_APP_NAME}'. Reason: ${error}`, + ); + return DEFAULT_APP_NAME; + } +} + +export async function name(options: NameParams): Promise { + if (options.packager.name) { + log.debug( + `Got name ${options.packager.name} from options. No inferring needed`, + ); + return sanitizeFilename(options.packager.platform, options.packager.name); + } + + const inferredName = await tryToInferName(options.packager.targetUrl); + return sanitizeFilename(options.packager.platform, inferredName); +} diff --git a/src/options/fields/userAgent.js b/src/options/fields/userAgent.js deleted file mode 100644 index 510d4b030b..0000000000 --- a/src/options/fields/userAgent.js +++ /dev/null @@ -1,9 +0,0 @@ -import { inferUserAgent } from '../../infer'; - -export default function({ userAgent, electronVersion, platform }) { - if (userAgent) { - return Promise.resolve(userAgent); - } - - return inferUserAgent(electronVersion, platform); -} diff --git a/src/options/fields/userAgent.test.js b/src/options/fields/userAgent.test.js deleted file mode 100644 index eed6c62fd7..0000000000 --- a/src/options/fields/userAgent.test.js +++ /dev/null @@ -1,20 +0,0 @@ -import userAgent from './userAgent'; -import { inferUserAgent } from '../../infer'; - -jest.mock('./../../infer/inferUserAgent'); - -test('when a userAgent parameter is passed', async () => { - expect(inferUserAgent).toHaveBeenCalledTimes(0); - - const params = { userAgent: 'valid user agent' }; - await expect(userAgent(params)).resolves.toBe(params.userAgent); -}); - -test('no userAgent parameter is passed', async () => { - const params = { electronVersion: '123', platform: 'mac' }; - await userAgent(params); - expect(inferUserAgent).toHaveBeenCalledWith( - params.electronVersion, - params.platform, - ); -}); diff --git a/src/options/fields/userAgent.test.ts b/src/options/fields/userAgent.test.ts new file mode 100644 index 0000000000..fdc8b81047 --- /dev/null +++ b/src/options/fields/userAgent.test.ts @@ -0,0 +1,26 @@ +import { userAgent } from './userAgent'; +import { inferUserAgent } from '../../infer/inferUserAgent'; + +jest.mock('./../../infer/inferUserAgent'); + +test('when a userAgent parameter is passed', async () => { + expect(inferUserAgent).toHaveBeenCalledTimes(0); + + const params = { + packager: {}, + nativefier: { userAgent: 'valid user agent' }, + }; + await expect(userAgent(params)).resolves.toBe(null); +}); + +test('no userAgent parameter is passed', async () => { + const params = { + packager: { electronVersion: '123', platform: 'mac' }, + nativefier: {}, + }; + await userAgent(params); + expect(inferUserAgent).toHaveBeenCalledWith( + params.packager.electronVersion, + params.packager.platform, + ); +}); diff --git a/src/options/fields/userAgent.ts b/src/options/fields/userAgent.ts new file mode 100644 index 0000000000..1fd5f6527f --- /dev/null +++ b/src/options/fields/userAgent.ts @@ -0,0 +1,22 @@ +import { inferUserAgent } from '../../infer/inferUserAgent'; + +type UserAgentOpts = { + packager: { + electronVersion?: string; + platform?: string; + }; + nativefier: { + userAgent?: string; + }; +}; + +export async function userAgent(options: UserAgentOpts): Promise { + if (options.nativefier.userAgent) { + return null; + } + + return inferUserAgent( + options.packager.electronVersion, + options.packager.platform, + ); +} diff --git a/src/options/model.ts b/src/options/model.ts new file mode 100644 index 0000000000..4d3ea4eae6 --- /dev/null +++ b/src/options/model.ts @@ -0,0 +1,56 @@ +import * as electronPackager from 'electron-packager'; + +export interface ElectronPackagerOptions extends electronPackager.Options { + targetUrl: string; +} + +export interface AppOptions { + packager: ElectronPackagerOptions; + nativefier: { + alwaysOnTop: boolean; + backgroundColor: string; + basicAuthPassword: string; + basicAuthUsername: string; + bounce: boolean; + browserwindowOptions: any; + clearCache: boolean; + counter: boolean; + crashReporter: string; + disableContextMenu: boolean; + disableDevTools: boolean; + disableGpu: boolean; + diskCacheSize: number; + enableEs3Apis: boolean; + fastQuit: boolean; + fileDownloadOptions: any; + flashPluginDir: string; + fullScreen: boolean; + globalShortcuts: any; + hideWindowFrame: boolean; + ignoreCertificate: boolean; + ignoreGpuBlacklist: boolean; + inject: string[]; + insecure: boolean; + internalUrls: string; + maximize: boolean; + nativefierVersion: string; + processEnvs: string; + proxyRules: string; + showMenuBar: boolean; + singleInstance: boolean; + titleBarStyle: string; + tray: string | boolean; + userAgent: string; + verbose: boolean; + versionString: string; + width: number; + height: number; + minWidth: number; + minHeight: number; + maxWidth: number; + maxHeight: number; + x: number; + y: number; + zoom: number; + }; +} diff --git a/src/options/normalizeUrl.js b/src/options/normalizeUrl.js deleted file mode 100644 index d4d487cacf..0000000000 --- a/src/options/normalizeUrl.js +++ /dev/null @@ -1,26 +0,0 @@ -import url from 'url'; -import validator from 'validator'; - -function appendProtocol(testUrl) { - const parsed = url.parse(testUrl); - if (!parsed.protocol) { - return `http://${testUrl}`; - } - return testUrl; -} - -function normalizeUrl(testUrl) { - const urlWithProtocol = appendProtocol(testUrl); - - const validatorOptions = { - require_protocol: true, - require_tld: false, - allow_trailing_dot: true, // mDNS addresses, https://github.com/jiahaog/nativefier/issues/308 - }; - if (!validator.isURL(urlWithProtocol, validatorOptions)) { - throw new Error(`Your Url: "${urlWithProtocol}" is invalid!`); - } - return urlWithProtocol; -} - -export default normalizeUrl; diff --git a/src/options/normalizeUrl.test.js b/src/options/normalizeUrl.test.js deleted file mode 100644 index 7c363564a6..0000000000 --- a/src/options/normalizeUrl.test.js +++ /dev/null @@ -1,17 +0,0 @@ -import normalizeUrl from './normalizeUrl'; - -test("a proper URL shouldn't be mangled", () => { - expect(normalizeUrl('http://www.google.com')).toEqual( - 'http://www.google.com', - ); -}); - -test('missing protocol should default to http', () => { - expect(normalizeUrl('www.google.com')).toEqual('http://www.google.com'); -}); - -test("a proper URL shouldn't be mangled", () => { - expect(() => { - normalizeUrl('http://ssddfoo bar'); - }).toThrow('Your Url: "http://ssddfoo bar" is invalid!'); -}); diff --git a/src/options/normalizeUrl.test.ts b/src/options/normalizeUrl.test.ts new file mode 100644 index 0000000000..2788b3d74f --- /dev/null +++ b/src/options/normalizeUrl.test.ts @@ -0,0 +1,17 @@ +import { normalizeUrl } from './normalizeUrl'; + +test("a proper URL shouldn't be mangled", () => { + expect(normalizeUrl('http://www.google.com')).toEqual( + 'http://www.google.com/', + ); +}); + +test('missing protocol should default to https', () => { + expect(normalizeUrl('www.google.com')).toEqual('https://www.google.com/'); +}); + +test("a proper URL shouldn't be mangled", () => { + expect(() => { + normalizeUrl('http://ssddfoo bar'); + }).toThrow('Your url "http://ssddfoo bar" is invalid'); +}); diff --git a/src/options/normalizeUrl.ts b/src/options/normalizeUrl.ts new file mode 100644 index 0000000000..e513b1934f --- /dev/null +++ b/src/options/normalizeUrl.ts @@ -0,0 +1,31 @@ +import * as url from 'url'; + +import * as log from 'loglevel'; + +function appendProtocol(inputUrl: string): string { + const parsed = url.parse(inputUrl); + if (!parsed.protocol) { + const urlWithProtocol = `https://${inputUrl}`; + log.warn( + `URL "${inputUrl}" lacks a protocol.`, + `Will try to parse it as HTTPS: "${urlWithProtocol}".`, + `Please pass "http://${inputUrl}" if this is what you meant.`, + ); + return urlWithProtocol; + } + return inputUrl; +} + +export function normalizeUrl(urlToNormalize: string): string { + const urlWithProtocol = appendProtocol(urlToNormalize); + + let parsedUrl: url.URL; + try { + parsedUrl = new url.URL(urlWithProtocol); + } catch (err) { + throw `Your url "${urlWithProtocol}" is invalid`; + } + const normalizedUrl = parsedUrl.toString(); + log.debug(`Normalized URL ${urlToNormalize} to:`, normalizedUrl); + return normalizedUrl; +} diff --git a/src/options/optionsMain.js b/src/options/optionsMain.js deleted file mode 100644 index 5cf883db44..0000000000 --- a/src/options/optionsMain.js +++ /dev/null @@ -1,133 +0,0 @@ -import fs from 'fs'; -import log from 'loglevel'; - -import inferOs from '../infer/inferOs'; -import normalizeUrl from './normalizeUrl'; -import packageJson from '../../package.json'; -import { ELECTRON_VERSION, PLACEHOLDER_APP_DIR } from '../constants'; -import asyncConfig from './asyncConfig'; - -const { inferPlatform, inferArch } = inferOs; - -/** - * Extracts only desired keys from inpOptions and augments it with defaults - * @param {Object} inpOptions - * @returns {Promise} - */ -export default function(inpOptions) { - const options = { - dir: PLACEHOLDER_APP_DIR, - name: inpOptions.name, - targetUrl: normalizeUrl(inpOptions.targetUrl), - platform: inpOptions.platform || inferPlatform(), - arch: inpOptions.arch || inferArch(), - electronVersion: inpOptions.electronVersion || ELECTRON_VERSION, - nativefierVersion: packageJson.version, - out: inpOptions.out || process.cwd(), - overwrite: inpOptions.overwrite, - asar: inpOptions.conceal || false, - icon: inpOptions.icon, - counter: inpOptions.counter || false, - bounce: inpOptions.bounce || false, - width: inpOptions.width || 1280, - height: inpOptions.height || 800, - minWidth: inpOptions.minWidth, - minHeight: inpOptions.minHeight, - maxWidth: inpOptions.maxWidth, - maxHeight: inpOptions.maxHeight, - showMenuBar: inpOptions.showMenuBar || false, - fastQuit: inpOptions.fastQuit || false, - userAgent: inpOptions.userAgent, - ignoreCertificate: inpOptions.ignoreCertificate || false, - disableGpu: inpOptions.disableGpu || false, - ignoreGpuBlacklist: inpOptions.ignoreGpuBlacklist || false, - enableEs3Apis: inpOptions.enableEs3Apis || false, - insecure: inpOptions.insecure || false, - flashPluginDir: inpOptions.flashPath || inpOptions.flash || null, - diskCacheSize: inpOptions.diskCacheSize || null, - inject: inpOptions.inject || null, - ignore: 'src', - fullScreen: inpOptions.fullScreen || false, - maximize: inpOptions.maximize || false, - hideWindowFrame: inpOptions.hideWindowFrame, - verbose: inpOptions.verbose, - disableContextMenu: inpOptions.disableContextMenu, - disableDevTools: inpOptions.disableDevTools, - crashReporter: inpOptions.crashReporter, - // workaround for electron-packager#375 - tmpdir: false, - zoom: inpOptions.zoom || 1.0, - internalUrls: inpOptions.internalUrls || null, - proxyRules: inpOptions.proxyRules || null, - singleInstance: inpOptions.singleInstance || false, - clearCache: inpOptions.clearCache || false, - appVersion: inpOptions.appVersion, - buildVersion: inpOptions.buildVersion, - appCopyright: inpOptions.appCopyright, - versionString: inpOptions.versionString, - win32metadata: inpOptions.win32metadata || { - ProductName: inpOptions.name, - InternalName: inpOptions.name, - FileDescription: inpOptions.name, - }, - processEnvs: inpOptions.processEnvs, - fileDownloadOptions: inpOptions.fileDownloadOptions, - tray: inpOptions.tray || false, - basicAuthUsername: inpOptions.basicAuthUsername || null, - basicAuthPassword: inpOptions.basicAuthPassword || null, - alwaysOnTop: inpOptions.alwaysOnTop || false, - titleBarStyle: inpOptions.titleBarStyle || null, - globalShortcuts: inpOptions.globalShortcuts || null, - browserwindowOptions: inpOptions.browserwindowOptions, - backgroundColor: inpOptions.backgroundColor || null, - darwinDarkModeSupport: inpOptions.darwinDarkModeSupport || false, - }; - - if (options.verbose) { - log.setLevel('trace'); - } else { - log.setLevel('error'); - } - - if (options.flashPluginDir) { - options.insecure = true; - } - - if (inpOptions.honest) { - options.userAgent = null; - } - - if (options.platform.toLowerCase() === 'windows') { - options.platform = 'win32'; - } - - if ( - options.platform.toLowerCase() === 'osx' || - options.platform.toLowerCase() === 'mac' - ) { - options.platform = 'darwin'; - } - - if (options.width > options.maxWidth) { - options.width = options.maxWidth; - } - - if (options.height > options.maxHeight) { - options.height = options.maxHeight; - } - - if (typeof inpOptions.x !== 'undefined') { - options.x = inpOptions.x; - } - - if (typeof inpOptions.y !== 'undefined') { - options.y = inpOptions.y; - } - - if (options.globalShortcuts) { - const globalShortcutsFileContent = fs.readFileSync(options.globalShortcuts); - options.globalShortcuts = JSON.parse(globalShortcutsFileContent); - } - - return asyncConfig(options); -} diff --git a/src/options/optionsMain.test.js b/src/options/optionsMain.test.js deleted file mode 100644 index 345fd416f5..0000000000 --- a/src/options/optionsMain.test.js +++ /dev/null @@ -1,17 +0,0 @@ -import optionsMain from './optionsMain'; -import asyncConfig from './asyncConfig'; - -jest.mock('./asyncConfig'); -const mockedAsyncConfig = { some: 'options' }; -asyncConfig.mockImplementation(() => Promise.resolve(mockedAsyncConfig)); - -test('it should call the async config', async () => { - const params = { - targetUrl: 'http://example.com', - }; - const result = await optionsMain(params); - expect(asyncConfig).toHaveBeenCalledWith(expect.objectContaining(params)); - expect(result).toEqual(mockedAsyncConfig); -}); - -// TODO add more tests diff --git a/src/options/optionsMain.test.ts b/src/options/optionsMain.test.ts new file mode 100644 index 0000000000..2e1b2292aa --- /dev/null +++ b/src/options/optionsMain.test.ts @@ -0,0 +1,25 @@ +import { getOptions } from './optionsMain'; +import * as asyncConfig from './asyncConfig'; + +const mockedAsyncConfig = { some: 'options' }; +let asyncConfigMock: jasmine.Spy; + +beforeAll(() => { + asyncConfigMock = spyOn(asyncConfig, 'asyncConfig').and.returnValue( + mockedAsyncConfig, + ); +}); + +test('it should call the async config', async () => { + const params = { + targetUrl: 'https://example.com/', + }; + const result = await getOptions(params); + expect(asyncConfigMock).toHaveBeenCalledWith( + expect.objectContaining({ + packager: expect.anything(), + nativefier: expect.anything(), + }), + ); + expect(result.packager.targetUrl).toEqual(params.targetUrl); +}); diff --git a/src/options/optionsMain.ts b/src/options/optionsMain.ts new file mode 100644 index 0000000000..3987b9a23f --- /dev/null +++ b/src/options/optionsMain.ts @@ -0,0 +1,167 @@ +import * as fs from 'fs'; + +import * as log from 'loglevel'; + +// package.json is `require`d to let tsc strip the `src` folder by determining +// baseUrl=src. A static import would prevent that and cause an ugly extra `src` folder in `lib` +const packageJson = require('../../package.json'); // eslint-disable-line @typescript-eslint/no-var-requires +import { + DEFAULT_ELECTRON_VERSION, + PLACEHOLDER_APP_DIR, + ELECTRON_MAJOR_VERSION, +} from '../constants'; +import { inferPlatform, inferArch } from '../infer/inferOs'; +import { asyncConfig } from './asyncConfig'; +import { AppOptions } from './model'; +import { normalizeUrl } from './normalizeUrl'; + +const SEMVER_VERSION_NUMBER_REGEX = /\d+\.\d+\.\d+[-_\w\d.]*/; + +/** + * Process and validate raw user arguments + */ +export async function getOptions(rawOptions: any): Promise { + const options: AppOptions = { + packager: { + appCopyright: rawOptions.appCopyright, + appVersion: rawOptions.appVersion, + arch: rawOptions.arch || inferArch(), + asar: rawOptions.conceal || false, + buildVersion: rawOptions.buildVersion, + darwinDarkModeSupport: rawOptions.darwinDarkModeSupport || false, + dir: PLACEHOLDER_APP_DIR, + electronVersion: rawOptions.electronVersion || DEFAULT_ELECTRON_VERSION, + icon: rawOptions.icon, + name: typeof rawOptions.name === 'string' ? rawOptions.name : '', + out: rawOptions.out || process.cwd(), + overwrite: rawOptions.overwrite, + platform: rawOptions.platform || inferPlatform(), + targetUrl: normalizeUrl(rawOptions.targetUrl), + tmpdir: false, // workaround for electron-packager#375 + win32metadata: rawOptions.win32metadata || { + ProductName: rawOptions.name, + InternalName: rawOptions.name, + FileDescription: rawOptions.name, + }, + }, + nativefier: { + alwaysOnTop: rawOptions.alwaysOnTop || false, + backgroundColor: rawOptions.backgroundColor || null, + basicAuthPassword: rawOptions.basicAuthPassword || null, + basicAuthUsername: rawOptions.basicAuthUsername || null, + bounce: rawOptions.bounce || false, + browserwindowOptions: rawOptions.browserwindowOptions, + clearCache: rawOptions.clearCache || false, + counter: rawOptions.counter || false, + crashReporter: rawOptions.crashReporter, + disableContextMenu: rawOptions.disableContextMenu, + disableDevTools: rawOptions.disableDevTools, + disableGpu: rawOptions.disableGpu || false, + diskCacheSize: rawOptions.diskCacheSize || null, + enableEs3Apis: rawOptions.enableEs3Apis || false, + fastQuit: rawOptions.fastQuit || false, + fileDownloadOptions: rawOptions.fileDownloadOptions, + flashPluginDir: rawOptions.flashPath || rawOptions.flash || null, + fullScreen: rawOptions.fullScreen || false, + globalShortcuts: null, + hideWindowFrame: rawOptions.hideWindowFrame, + ignoreCertificate: rawOptions.ignoreCertificate || false, + ignoreGpuBlacklist: rawOptions.ignoreGpuBlacklist || false, + inject: rawOptions.inject || [], + insecure: rawOptions.insecure || false, + internalUrls: rawOptions.internalUrls || null, + maximize: rawOptions.maximize || false, + nativefierVersion: packageJson.version, + processEnvs: rawOptions.processEnvs, + proxyRules: rawOptions.proxyRules || null, + showMenuBar: rawOptions.showMenuBar || false, + singleInstance: rawOptions.singleInstance || false, + titleBarStyle: rawOptions.titleBarStyle || null, + tray: rawOptions.tray || false, + userAgent: rawOptions.userAgent, + verbose: rawOptions.verbose, + versionString: rawOptions.versionString, + width: rawOptions.width || 1280, + height: rawOptions.height || 800, + minWidth: rawOptions.minWidth, + minHeight: rawOptions.minHeight, + maxWidth: rawOptions.maxWidth, + maxHeight: rawOptions.maxHeight, + x: rawOptions.x, + y: rawOptions.y, + zoom: rawOptions.zoom || 1.0, + }, + }; + + if (options.nativefier.verbose) { + log.setLevel('trace'); + try { + require('debug').enable('electron-packager'); + } catch (err) { + log.debug( + 'Failed to enable electron-packager debug output. This should not happen,', + 'and suggests their internals changed. Please report an issue.', + ); + } + + log.debug( + 'Running in verbose mode! This will produce a mountain of logs and', + 'is recommended only for troubleshooting or if you like Shakespeare.', + ); + } else { + log.setLevel('info'); + } + + if (rawOptions.electronVersion) { + const requestedVersion: string = rawOptions.electronVersion; + if (!SEMVER_VERSION_NUMBER_REGEX.exec(requestedVersion)) { + throw `Invalid Electron version number "${requestedVersion}". Aborting.`; + } + const requestedMajorVersion = parseInt(requestedVersion.split('.')[0], 10); + if (requestedMajorVersion < ELECTRON_MAJOR_VERSION) { + log.warn( + `\nATTENTION: Using **old** Electron version ${requestedVersion} as requested.`, + "\nIt's untested, bugs and horror will happen, you're on your own.", + `\nSimply abort & re-run without passing the version flag to default to ${DEFAULT_ELECTRON_VERSION}`, + ); + } + } + + if (options.nativefier.flashPluginDir) { + options.nativefier.insecure = true; + } + + if (rawOptions.honest) { + options.nativefier.userAgent = null; + } + + if (options.packager.platform.toLowerCase() === 'windows') { + options.packager.platform = 'win32'; + } + + if ( + ['osx', 'mac', 'macos'].includes(options.packager.platform.toLowerCase()) + ) { + options.packager.platform = 'darwin'; + } + + if (options.nativefier.width > options.nativefier.maxWidth) { + options.nativefier.width = options.nativefier.maxWidth; + } + + if (options.nativefier.height > options.nativefier.maxHeight) { + options.nativefier.height = options.nativefier.maxHeight; + } + + if (rawOptions.globalShortcuts) { + log.debug('Use global shortcuts file at', rawOptions.globalShortcuts); + const globalShortcuts = JSON.parse( + fs.readFileSync(rawOptions.globalShortcuts).toString(), + ); + options.nativefier.globalShortcuts = globalShortcuts; + } + + await asyncConfig(options); + + return options; +} diff --git a/src/utils/index.js b/src/utils/index.js deleted file mode 100644 index e36cf7226e..0000000000 --- a/src/utils/index.js +++ /dev/null @@ -1,3 +0,0 @@ -// TODO remove the eslint disable when we have more than one -// eslint-disable-next-line import/prefer-default-export -export { default as sanitizeFilename } from './sanitizeFilename'; diff --git a/src/utils/sanitizeFilename.js b/src/utils/sanitizeFilename.js deleted file mode 100644 index 3da71642e7..0000000000 --- a/src/utils/sanitizeFilename.js +++ /dev/null @@ -1,16 +0,0 @@ -import sanitizeFilenameLib from 'sanitize-filename'; -import { DEFAULT_APP_NAME } from '../constants'; - -export default function(platform, str) { - let result = sanitizeFilenameLib(str); - - // remove all non ascii or use default app name - // eslint-disable-next-line no-control-regex - result = result.replace(/[^\x00-\x7F]/g, '') || DEFAULT_APP_NAME; - - // spaces will cause problems with Ubuntu when pinned to the dock - if (platform === 'linux') { - return result.replace(/ /g, ''); - } - return result; -} diff --git a/src/utils/sanitizeFilename.test.js b/src/utils/sanitizeFilename.test.ts similarity index 68% rename from src/utils/sanitizeFilename.test.js rename to src/utils/sanitizeFilename.test.ts index 772b1bdb1d..6bce56065d 100644 --- a/src/utils/sanitizeFilename.test.js +++ b/src/utils/sanitizeFilename.test.ts @@ -1,16 +1,6 @@ -import sanitizeFilenameLib from 'sanitize-filename'; -import sanitizeFilename from './sanitizeFilename'; +import { sanitizeFilename } from './sanitizeFilename'; import { DEFAULT_APP_NAME } from '../constants'; -jest.mock('sanitize-filename'); -sanitizeFilenameLib.mockImplementation((str) => str); - -test('it should call the sanitize-filename npm module', () => { - const param = 'abc'; - sanitizeFilename('', param); - expect(sanitizeFilenameLib).toHaveBeenCalledWith(param); -}); - describe('replacing non ascii characters', () => { const nonAscii = '�'; test('it should return a result without non ascii characters', () => { diff --git a/src/utils/sanitizeFilename.ts b/src/utils/sanitizeFilename.ts new file mode 100644 index 0000000000..7e16fc7016 --- /dev/null +++ b/src/utils/sanitizeFilename.ts @@ -0,0 +1,24 @@ +import * as log from 'loglevel'; + +import { DEFAULT_APP_NAME } from '../constants'; + +// eslint-disable-next-line @typescript-eslint/no-var-requires +const sanitize = require('sanitize-filename'); + +export function sanitizeFilename( + platform: string, + filenameToSanitize: string, +): string { + let result = sanitize(filenameToSanitize); + + // remove all non ascii or use default app name + // eslint-disable-next-line no-control-regex + result = result.replace(/[^\x00-\x7F]/g, '') || DEFAULT_APP_NAME; + + // spaces will cause problems with Ubuntu when pinned to the dock + if (platform === 'linux') { + result = result.replace(/ /g, ''); + } + log.debug(`Sanitized filename for ${filenameToSanitize} : ${result}`); + return result; +} diff --git a/test-resources/test-injection.js b/test-resources/test-injection.js index 1a14e3529c..b535a98e17 100644 --- a/test-resources/test-injection.js +++ b/test-resources/test-injection.js @@ -1,3 +1,3 @@ -const log = require('loglevel'); +const log = require('loglevel'); // eslint-disable-line @typescript-eslint/no-var-requires log.info('This is a test injecton script'); diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000000..2f93f018c3 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,20 @@ +{ + "compilerOptions": { + "allowJs": false, + "declaration": false, + "incremental": true, + "module": "commonjs", + "moduleResolution": "node", + "outDir": "./lib", + "resolveJsonModule": true, + "skipLibCheck": true, + "sourceMap": true, + // https://stackoverflow.com/questions/48378495/recommended-typescript-config-for-node-8 + // and 'dom' to tell tsc it's okay to use the URL object (which is in Node >= 7) + "target": "es2017", + "lib": ["es2017", "dom"] + }, + "include": [ + "./src/**/*" + ] +} diff --git a/webpack.config.js b/webpack.config.js deleted file mode 100644 index ad47a311c1..0000000000 --- a/webpack.config.js +++ /dev/null @@ -1,23 +0,0 @@ -const electronPublicApi = ['electron']; - -const nodeModules = {}; -electronPublicApi.forEach((apiString) => { - nodeModules[apiString] = `commonjs ${apiString}`; -}); - -module.exports = { - target: 'node', - output: { - filename: 'main.js', - }, - node: { - global: false, - __dirname: false, - }, - externals: nodeModules, - module: { - rules: [{ test: /\.js$/, exclude: /node_modules/, loader: 'babel-loader' }], - }, - devtool: 'source-map', - mode: 'none', -};