Compare commits

..

16 Commits

Author SHA1 Message Date
Lorenzo Alberto Maria Ambrosi
f6ce9a217d Merge branch 'save-url-image-2' of github.com:balena-io/etcher into save-url-image-2 2020-10-19 12:54:34 +02:00
Lorenzo Alberto Maria Ambrosi
fce2d94df7 Rework system & large drives handling logic
Change-type: patch
Changelog-entry: Rework system & large drives handling logic
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-19 12:22:17 +02:00
Lorenzo Alberto Maria Ambrosi
3feb22ee66 Add primary colors to default flow
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-14 13:12:11 +02:00
Lorenzo Alberto Maria Ambrosi
b80a6b2feb Add UI option to save images flashed from URLs
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-14 13:12:11 +02:00
Lorenzo Alberto Maria Ambrosi
b4e6970119 Rework system & large drives handling logic
Change-type: patch
Changelog-entry: Rework system & large drives handling logic
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-14 13:12:11 +02:00
Lorenzo Alberto Maria Ambrosi
2e3978b3c9 Add more typings & refactor code accordingly
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-14 13:04:35 +02:00
Lorenzo Alberto Maria Ambrosi
c6cd421f17 Fix URL not being selected with custom protocol
Change-type: patch
Changelog-entry: Fix URL not being selected with custom protocol
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-14 12:30:55 +02:00
Lorenzo Alberto Maria Ambrosi
c3296eed54 Add dash on table when selecting only some rows
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-10-01 14:52:42 +02:00
Lorenzo Alberto Maria Ambrosi
153e37b9dc Fix settings spacing
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-25 11:34:06 +02:00
Lorenzo Alberto Maria Ambrosi
78aca6a19f Use drive-selector's table for flash errors table
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-25 11:34:06 +02:00
Lorenzo Alberto Maria Ambrosi
27695babfd Update rendition to v18.8.3
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-18 10:50:44 +02:00
Lorenzo Alberto Maria Ambrosi
06a96db72d Fix zoomFactor in webviews
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-18 09:45:31 +02:00
Lorenzo Alberto Maria Ambrosi
6584cef774 Add retry button to the errors modal in success screen
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-18 09:45:31 +02:00
Lorenzo Alberto Maria Ambrosi
3c77800b1d Cleanup after child-process is terminated
Change-type: patch
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-18 09:45:31 +02:00
Lorenzo Alberto Maria Ambrosi
74a78076cf Add skip function to validation
Change-type: patch
Changelog-entry: Add skip function to validation
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-18 09:45:31 +02:00
Lorenzo Alberto Maria Ambrosi
8ff8b02f37 Rework success screen
Change-type: patch
Changelog-entry: Rework success screen
Signed-off-by: Lorenzo Alberto Maria Ambrosi <lorenzothunder.ambrosi@gmail.com>
2020-09-18 09:45:31 +02:00
83 changed files with 8571 additions and 21777 deletions

View File

@@ -7,6 +7,7 @@ indent_size = 2
end_of_line = lf end_of_line = lf
charset = utf-8 charset = utf-8
trim_trailing_whitespace = true trim_trailing_whitespace = true
insert_final_newline = true
[*.md] [*.md]
trim_trailing_whitespace = false trim_trailing_whitespace = false

1
.gitattributes vendored
View File

@@ -27,7 +27,6 @@ Makefile text
*.yml text *.yml text
*.patch text *.patch text
*.txt text *.txt text
*.tpl text
CODEOWNERS text CODEOWNERS text
*.plist text *.plist text

View File

@@ -1,11 +1,6 @@
- **Etcher version:** - **Etcher version:**
- **Operating system and architecture:** - **Operating system and architecture:**
- **Image flashed:** - **Image flashed:**
- **What do you think should have happened:** <!-- or a step by step reproduction process -->
- **What happened:**
- **Do you see any meaningful error information in the DevTools?** - **Do you see any meaningful error information in the DevTools?**
<!-- You can open DevTools by pressing `Ctrl+Shift+I` (`Ctrl+Alt+I` for Etcher before v1.3.x), or `Cmd+Opt+I` if you're on macOS. -->
<!-- issues with missing information will be labeled as not-enough-info and closed shortly --> <!-- You can open DevTools by pressing `Ctrl+Shift+I` (`Ctrl+Alt+I` for Etcher before v1.3.x), or `Cmd+Opt+I` if you're on macOS. -->
<!-- please try to include as many influencing elements as possible are you root, does any other process block the device, etc. -->
<!-- if you find a solution in the meantime thank you for sharing the fix and not just closing / abandoning your issue -->

View File

@@ -15,7 +15,7 @@
}, },
"builder": { "builder": {
"appId": "io.balena.etcher", "appId": "io.balena.etcher",
"copyright": "Copyright 2016-2021 Balena Ltd", "copyright": "Copyright 2016-2020 Balena Ltd",
"productName": "balenaEtcher", "productName": "balenaEtcher",
"nodeGypRebuild": false, "nodeGypRebuild": false,
"afterPack": "./afterPack.js", "afterPack": "./afterPack.js",
@@ -24,13 +24,13 @@
"generated", "generated",
"lib/shared/catalina-sudo/sudo-askpass.osascript.js" "lib/shared/catalina-sudo/sudo-askpass.osascript.js"
], ],
"beforeBuild": "./beforeBuild.js",
"afterSign": "./afterSignHook.js", "afterSign": "./afterSignHook.js",
"mac": { "mac": {
"category": "public.app-category.developer-tools", "category": "public.app-category.developer-tools",
"hardenedRuntime": true, "hardenedRuntime": true,
"entitlements": "entitlements.mac.plist", "entitlements": "entitlements.mac.plist",
"entitlementsInherit": "entitlements.mac.plist", "entitlementsInherit": "entitlements.mac.plist"
"artifactName": "${productName}-${version}.${ext}"
}, },
"dmg": { "dmg": {
"iconSize": 110, "iconSize": 110,

File diff suppressed because it is too large Load Diff

View File

@@ -3,809 +3,6 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
This project adheres to [Semantic Versioning](http://semver.org/). This project adheres to [Semantic Versioning](http://semver.org/).
# v1.7.9
## (2022-04-22)
* patch: update allowed extensions to include deb afterinstall in build [mcraa]
* patch: add update notification [Peter Makra]
* patch: fix usb-device-boot link in README [Andrew Scheller]
* Fix application directory for Debian postinst script [Ken Bannister]
# v1.7.8
## (2022-03-18)
* patch: complete suse uninstall readme [Peter Makra]
* patch: completed suse instructions [Peter Makra]
* patch: order rpm instrictions [Peter Makra]
* patch: enabled update notification for version 1.7.8 [Peter Makra]
* patch: updated title to balenaEtcher [Peter Makra]
* patch: cleanup and organize readme [Peter Makra]
* patch: extend cloudsmith attribution in readme [Peter Makra]
* Update macOS Icon to Big Sur Style [Logicer]
# v1.7.7
## (2022-02-22)
* patch: clarified update check [Peter Makra]
* patch: autoupdate stagingPercentage check, include default [Peter Makra]
# v1.7.6
## (2022-02-21)
* patch: version number notification [Peter Makra]
* patch: fixed typos in template [Peter Makra]
* patch: add requirements and help to issue template [mcraa]
* patch: add requirements and help to issue template [mcraa]
# v1.7.5
## (2022-02-21)
* patch: fix flashing from URL when using basic auth [Marco Füllemann]
# v1.7.4
## (2022-02-21)
* patch: set version update notification 1.7.3 [Peter Makra]
* patch: updated electron to 12.2.3 [Peter Makra]
* patch: updated electron to 12.2.3 [Peter Makra]
# v1.7.3
## (2021-12-29)
* patch: fix mesage of null [Peter Makra]
# v1.7.2
## (2021-12-21)
* patch: fixed open from browser on windows [Peter Makra]
# v1.7.1
## (2021-11-22)
* patch: Revert back to electron-rebuild [Lorenzo Alberto Maria Ambrosi]
* patch: Disallow TS in JS [Lorenzo Alberto Maria Ambrosi]
* patch: Remove esInterop TS flag [Lorenzo Alberto Maria Ambrosi]
* patch: Use @balena/sudo-prompt [Lorenzo Alberto Maria Ambrosi]
* patch: Update rpiboot guide link [Lorenzo Alberto Maria Ambrosi]
* patch: Improve webpack build time [Lorenzo Alberto Maria Ambrosi]
# v1.7.0
## (2021-11-09)
* patch: Add missing @types/react@16.8.5 [Lorenzo Alberto Maria Ambrosi]
* patch: Use npm ci in Makefile [Lorenzo Alberto Maria Ambrosi]
* patch: Add draft info boxes for system information [Lorenzo Alberto Maria Ambrosi]
* patch: Remove electron-rebuild package [Lorenzo Alberto Maria Ambrosi]
* patch: Make electron a dev. dependency [Lorenzo Alberto Maria Ambrosi]
* patch: Remove electron-rebuild package [Lorenzo Alberto Maria Ambrosi]
* patch: Use exact modules versions [Lorenzo Alberto Maria Ambrosi]
* patch: Update etcher-sdk from v6.2.5 to v6.3.0 [Lorenzo Alberto Maria Ambrosi]
* Fix write step for Http file process [JSReds]
* patch: Fix linting errors [Lorenzo Alberto Maria Ambrosi]
* minor: Refactor dependencies installation to avoid custom scripts [Lorenzo Alberto Maria Ambrosi]
* patch: Fix LEDs init error [Lorenzo Alberto Maria Ambrosi]
# v1.6.0
## (2021-09-20)
* Add support for basic auth when downloading images from URL. [Marco Füllemann]
* patch: Update etcher-sdk from v6.2.1 to v6.2.5 [Lorenzo Alberto Maria Ambrosi]
* Update Makefile to Apple M1 info [David Gaspar]
* Add LED settings for potentially different hardware [Lorenzo Alberto Maria Ambrosi]
# v1.5.122
## (2021-09-02)
* Restore image file selection LED-drive pathing [Lorenzo Alberto Maria Ambrosi]
* Update scripts submodule [Lorenzo Alberto Maria Ambrosi]
* Change LEDs colours [Lorenzo Alberto Maria Ambrosi]
* Windows images now show the proper warning again [Lorenzo Alberto Maria Ambrosi]
* Fix Update and install with DNF instructions [Mohamed Salah]
* Add possibile authorization as a query param [JSReds]
* update the windows part [Xtraim]
* Update SUPPORT.md [thambu1710]
* replace make webpack with npm run webpack [Seth Falco]
* Add loader on image select [JSReds]
* add pnp-webpack-plugin [Zane Hitchcox]
* Remove redundant codespell dependency/tests [Lorenzo Alberto Maria Ambrosi]
# v1.5.121
## (2021-07-05)
* patch: Delete Codeowners [Vipul Gupta]
* Add source maps for devtools [Lorenzo Alberto Maria Ambrosi]
* Clone submodules when initializing modules [Lorenzo Alberto Maria Ambrosi]
* patch: Select drive on list interaction rather than modal closing [Lorenzo Alberto Maria Ambrosi]
# v1.5.120
## (2021-05-11)
* Update README to reference Cloudsmith [Lorenzo Alberto Maria Ambrosi]
# v1.5.119
## (2021-04-30)
* Update readme for new PPA provider [Lorenzo Alberto Maria Ambrosi]
# v1.5.118
## (2021-04-27)
* patch: development environment [Zane Hitchcox]
* patch: watch files for electron [Zane Hitchcox]
# v1.5.117
## (2021-04-02)
* Rename mac releases (keep old naming) [Alexis Svinartchouk]
* Disable spectron tests on macOS [Alexis Svinartchouk]
* Update electron to v12.0.2 [Alexis Svinartchouk]
<details>
<summary> Update etcher-sdk from 6.1.1 to 6.2.1 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-6.2.1
> ### (2021-03-26)
>
>
> <details>
> <summary> Update node-raspberrypi-usbboot from 0.2.11 to 0.3.0 [Alexis Svinartchouk] </summary>
>
>> ### node-raspberrypi-usbboot-0.3.0
>> #### (2021-03-26)
>>
>> * Add support for compute module 4 [Alexis Svinartchouk]
>> * Fix size endianness of boot_message_t message [Alexis Svinartchouk]
>>
> </details>
>
>
> ## etcher-sdk-6.2.0
> ### (2021-02-18)
>
> * Added BeagleBone USB Boot example [Parthiban Gandhi]
> * Added BeagleBone USB Boot support [Parthiban Gandhi]
>
</details>
* Fix getAppPath() returning an asar file on macOS [Alexis Svinartchouk]
* Grammar fix [Andrew Scheller]
* (docs) update README.md [vlad doster]
* Update copyright year in electron-builder.yml [Andrew Scheller]
* Update copyright year in .resinci.json [Andrew Scheller]
* Separate the Yum and DNF instructions. [Dugan Chen]
* Set msvs_version to 2019 when rebuilding [Alexis Svinartchouk]
* Use moduleIds: 'natural' in webpack config to keep js files in arm64 and x64 mac builds identical [Alexis Svinartchouk]
* Update electron-builder to 22.10.5 [Alexis Svinartchouk]
* Update spectron to v13 [Alexis Svinartchouk]
* Update dependencies, use aws4-axios@2.2.1 to avoid adding more dependiencies [Alexis Svinartchouk]
* Update scripts to build universal mac dmgs on the ci [Alexis Svinartchouk]
* Fix beforeBuild.js script to also work on mac [Alexis Svinartchouk]
* Support building universal dmgs (x64 and arm64) for mac [Alexis Svinartchouk]
* Update electron-builder to 22.10.4 [Alexis Svinartchouk]
* Fix titlebar z-index [Alexis Svinartchouk]
* Explicitly set contextIsolation to false [Alexis Svinartchouk]
* Update electron from 9.4.1 to 11.2.3 [Alexis Svinartchouk]
<details>
<summary> Update etcher-sdk from 6.1.0 to 6.1.1 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-6.1.1
> ### (2021-02-10)
>
>
> <details>
> <summary> Update node-raspberrypi-usbboot from 0.2.10 to 0.2.11 [Alexis Svinartchouk] </summary>
>
>> ### node-raspberrypi-usbboot-0.2.11
>> #### (2021-02-10)
>>
>> * Update @balena.io/usb from 1.3.12 to 1.3.14 [Alexis Svinartchouk]
>>
> </details>
>
>
</details>
# v1.5.116
## (2021-02-03)
* Only cleanup temporary decompressed files in child-writer [Alexis Svinartchouk]
* Add .versionbot/CHANGELOG.yml [Alexis Svinartchouk]
* Stop using node-tmp, use withTmpFile from etcher-sdk instead [Alexis Svinartchouk]
<details>
<summary> Update etcher-sdk from 5.2.2 to 6.1.0 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-6.1.0
> ### (2021-02-03)
>
> * Prefix temporary decompressed images filenames [Alexis Svinartchouk]
>
> ## etcher-sdk-6.0.1
> ### (2021-02-02)
>
> * Ignore ENOENT errors on unlink in withTmpFile [Alexis Svinartchouk]
>
> ## etcher-sdk-6.0.0
> ### (2021-02-01)
>
> * Export tmp and add prefix and postfix options [Alexis Svinartchouk]
>
> ## etcher-sdk-5.2.3
> ### (2021-01-26)
>
> * upgrade lint [Zane Hitchcox]
>
</details>
* Revert "Change some border colors to have higher contrast" [Alexis Svinartchouk]
* Update electron to v9.4.1 [Alexis Svinartchouk]
<details>
<summary> Update etcher-sdk from 5.2.1 to 5.2.2 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-5.2.2
> ### (2021-01-19)
>
>
> <details>
> <summary> Update drivelist from 9.2.2 to 9.2.4 [Alexis Svinartchouk] </summary>
>
>> ### drivelist-9.2.4
>> #### (2021-01-19)
>>
>> * Pass strings between methods as std::string instead of char * [Floris Bos]
>>
>> ### drivelist-9.2.3
>> #### (2021-01-19)
>>
>> * Support lsblk versions that do no support the pttype column [Alexis Svinartchouk]
>>
> </details>
>
>
</details>
# v1.5.115
## (2021-01-18)
<details>
<summary> Update etcher-sdk from 5.1.12 to 5.2.1 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-5.2.1
> ### (2021-01-15)
>
> * Only run one diskpart at a time [Alexis Svinartchouk]
> * Ignore diskpart VDS_E_DISK_IS_OFFLINE errors [Alexis Svinartchouk]
>
> ## etcher-sdk-5.2.0
> ### (2021-01-06)
>
> * Store progress on usbboot devices [Alexis Svinartchouk]
>
</details>
# v1.5.114
## (2021-01-12)
* Remove libappindicator1 debian dependency [Alexis Svinartchouk]
<details>
<summary> Update etcher-sdk from 5.1.11 to 5.1.12 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-5.1.12
> ### (2021-01-06)
>
> * Remove BlockDevice.mountpoints incorrect typing [Alexis Svinartchouk]
> * Update axios to 0.21.1 and aws4-axios to 2.0.1 [Alexis Svinartchouk]
>
</details>
<details>
<summary> Update rendition from 18.8.3 to 19.2.0 [Alexis Svinartchouk] </summary>
> ## rendition-19.2.0
> ### (2020-12-29)
>
> * Add truncate property to Txt component [JSReds]
>
> ## rendition-19.1.0
> ### (2020-12-29)
>
> * Add fallback image source to Img component [Stevche Radevski]
>
> ## rendition-19.0.0
> ### (2020-12-21)
>
> * Remove Arcslider component [Stevche Radevski]
>
> ## rendition-18.20.4
> ### (2020-12-17)
>
> * Upgrade rehype-raw to latest version [Kakhaber]
>
> ## rendition-18.20.3
> ### (2020-12-17)
>
> * Fix disabled button tooltip [JSReds]
>
> ## rendition-18.20.2
> ### (2020-12-16)
>
> * Turn keydown handler into an arrow function [Stevche Radevski]
>
> ## rendition-18.20.1
> ### (2020-12-14)
>
> * Fix form not getting the Enter key event when nested in a modal [Stevche Radevski]
>
> ## rendition-18.20.0
> ### (2020-12-14)
>
> * feat: Add new StatsBar component [Graham McCulloch]
>
> ## rendition-18.19.2
> ### (2020-12-14)
>
> * Update snapshots [Graham McCulloch]
> * Removed out-of-date documentation and template text [Graham McCulloch]
>
> ## rendition-18.19.1
> ### (2020-12-04)
>
> * Markdown: Fix line breaks [Kakhaber]
>
> ## rendition-18.19.0
> ### (2020-12-02)
>
> * Make card size responsive [Stevche Radevski]
>
> ## rendition-18.18.0
> ### (2020-12-02)
>
> * Allow passing responsive values to datagrid width props [Stevche Radevski]
>
> ## rendition-18.17.2
> ### (2020-12-01)
>
> * Update snapshots due to a Card change [JSReds]
>
> ## rendition-18.17.1
> ### (2020-12-01)
>
> * Card: make body to be full height [JSReds]
>
> ## rendition-18.17.0
> ### (2020-12-01)
>
> * Add star rating component [Kakhaber]
>
> ## rendition-18.16.0
> ### (2020-11-23)
>
> * Completely revamp the development setup for rendition [Stevche Radevski]
>
> ## rendition-18.15.1
> ### (2020-11-16)
>
> * Modal: Change the button margins to use the predefined spacing palette [Thodoris Greasidis]
>
> ## rendition-18.15.0
> ### (2020-11-16)
>
> * Modal: Move the cancel button first for dangerous & warning actions [Thodoris Greasidis]
>
> ## rendition-18.14.0
> ### (2020-11-16)
>
> * Allow passing checked items as a prop to Table [Stevche Radevski]
>
> ## rendition-18.13.4
> ### (2020-11-16)
>
> * Fix accidental complete lodash import [Thodoris Greasidis]
>
> ## rendition-18.13.3
> ### (2020-11-16)
>
> * Form: Remove the flaky Captcha sceenshot test [Thodoris Greasidis]
> * Update react-simplemde-editor & snapshots for upstream versions [Thodoris Greasidis]
>
> ## rendition-18.13.2
> ### (2020-10-29)
>
> * Updated snapshots [Graham McCulloch]
> * Fix: Confirm only depends on the files it needs [Graham McCulloch]
>
> ## rendition-18.13.1
> ### (2020-10-23)
>
> * Button: Preserve event during confirmation [Kakhaber]
>
> ## rendition-18.13.0
> ### (2020-10-22)
>
> * Button: Add confirmation property [Kakhaber]
>
> ## rendition-18.12.2
> ### (2020-10-21)
>
> * Tabs: changed interfaces and props [JSReds]
>
> ## rendition-18.12.1
> ### (2020-10-20)
>
> * Fix Tabs typings [Stevche Radevski]
>
> ## rendition-18.12.0
> ### (2020-10-19)
>
> * Add a Grid component [Stevche Radevski]
>
> ## rendition-18.11.3
> ### (2020-10-14)
>
> * Added more documentation for JsonSchemaRenderer [Graham McCulloch]
>
> ## rendition-18.11.2
> ### (2020-10-14)
>
> * fix: UI schema for JsonSchemaRenderer DropDownButton and ButtonGroup widgets [Graham McCulloch]
>
> ## rendition-18.11.1
> ### (2020-10-13)
>
> * Add dark mode to storybook [Stevche Radevski]
>
> ## rendition-18.11.0
> ### (2020-10-08)
>
> * Allow passing widget to extraFormats field [Stevche Radevski]
>
> ## rendition-18.10.2
> ### (2020-09-30)
>
> * Resolve module path not relying on node_moules dir [Kakhaber]
>
> ## rendition-18.10.1
> ### (2020-09-29)
>
> * Set tabpanel height so it stretches to full height [StefKors]
> * Specify tabs width to fix layout problems [StefKors]
>
> ## rendition-18.10.0
> ### (2020-09-24)
>
> * feat: Add ColorWidget for JsonSchemaRenderer [Graham McCulloch]
>
> ## rendition-18.9.2
> ### (2020-09-22)
>
> * Markdown: Ignore decorators inside a code block [Kakhaber]
>
> ## rendition-18.9.1
> ### (2020-09-21)
>
> * Add compact variation to tabs [StefKors]
>
> ## rendition-18.9.0
> ### (2020-09-18)
>
> * Improve spacing for Modal and Select components [Stevche Radevski]
>
> ## rendition-18.8.4
> ### (2020-09-17)
>
> * fix: Use widget's display name to reference the widget [Graham McCulloch]
>
</details>
* Update dependencies [Alexis Svinartchouk]
* Update @balena/lint to 5.3.0 [Alexis Svinartchouk]
* Update webpack to v5 [Alexis Svinartchouk]
* Fix typo in webpack.config.ts comment [Alexis Svinartchouk]
* docs: fix quote marks [Aaron Shaw]
* Disable screensaver while flashing (on balena-electron-env) [Alexis Svinartchouk]
# v1.5.113
## (2020-12-16)
* Show the first error for each drive (not the last) [Alexis Svinartchouk]
* Fix red leds not showing for failed devices [Alexis Svinartchouk]
* docs: add documentation links [Aaron Shaw]
* docs: update macOS version [Aaron Shaw]
* Improve hover message when the drive is too small [Alexis Svinartchouk]
* Update electron to v9.4.0 [Alexis Svinartchouk]
* Update npm to v6.14.8 [Giovanni Garufi]
* Update rgb leds colors [Alexis Svinartchouk]
* Remove unmountOnSuccess setting [Alexis Svinartchouk]
* Only show auto-updates setting on supported targets [Alexis Svinartchouk]
* Remove dead code in settings modal [Alexis Svinartchouk]
* Fix effective flashing speed calculation for compressed images [Alexis Svinartchouk]
* Change some border colors to have higher contrast [Lorenzo Alberto Maria Ambrosi]
<details>
<summary> Update etcher-sdk from 5.1.10 to 5.1.11 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-5.1.11
> ### (2020-12-07)
>
> * Don't use the O_SYNC flag for block devices, only O_DIRECT [Alexis Svinartchouk]
>
</details>
<details>
<summary> Update sys-class-rgb-led from 2.1.1 to 3.0.0 [Alexis Svinartchouk] </summary>
> ## sys-class-rgb-led-3.0.0
> ### (2020-12-03)
>
> * Add example etcher-pro rainbow animation [Alexis Svinartchouk]
> * Use one setInterval instead of a loop for each led, t in seconds [Alexis Svinartchouk]
>
</details>
# v1.5.112
## (2020-12-02)
* Add rendition and sys-class-rgb-led to repo.yml [Alexis Svinartchouk]
<details>
<summary> Update sys-class-rgb-led from 2.1.0 to 2.1.1 [Alexis Svinartchouk] </summary>
> ## sys-class-rgb-led-2.1.1
> ### (2020-12-01)
>
> * Replace resin-lint with @balena/lint [Alexis Svinartchouk]
> * Update typescript to v4.1.2 [Alexis Svinartchouk]
> * Add versionbot changelog [Alexis Svinartchouk]
>
</details>
* Fix layout when the featured project is not showing [Alexis Svinartchouk]
* Improve flashing error handling [Alexis Svinartchouk]
* Fix modal content height on Windows [Alexis Svinartchouk]
<details>
<summary> Update etcher-sdk from 5.1.5 to 5.1.10 [Alexis Svinartchouk] </summary>
> ## etcher-sdk-5.1.10
> ### (2020-12-02)
>
>
> <details>
> <summary> Update balena-image-fs from 7.0.5 to 7.0.6 [Alexis Svinartchouk] </summary>
>
>> ### balena-image-fs-7.0.6
>> #### (2020-12-02)
>>
>>
>> <details>
>> <summary> Update ext2fs from 3.0.4 to 3.0.5 [Alexis Svinartchouk] </summary>
>>
>>> #### node-ext2fs-3.0.5
>>> ##### (2020-12-02)
>>>
>>> * Fix reading and discarding with offsets > 32 bits [Alexis Svinartchouk]
>>>
>> </details>
>>
>>
> </details>
>
>
> ## etcher-sdk-5.1.9
> ### (2020-12-01)
>
> * Add repo.yml file [Alexis Svinartchouk]
> * Update @balena/udif from 1.1.0 to 1.1.1 [Alexis Svinartchouk]
>
> <details>
> <summary> Update zip-part-stream from 1.0.2 to 1.0.3 [Alexis Svinartchouk] </summary>
>
>> ### zip-part-stream-1.0.3
>> #### (2020-11-30)
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update node-raspberrypi-usbboot from 0.2.9 to 0.2.10 [Alexis Svinartchouk] </summary>
>
>> ### node-raspberrypi-usbboot-0.2.10
>> #### (2020-11-30)
>>
>> * Update typescript to v4.1.2 [Alexis Svinartchouk]
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update mountutils from 1.3.19 to 1.3.20 [Alexis Svinartchouk] </summary>
>
>> ### mountutils-1.3.20
>> #### (2020-11-30)
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update gzip-stream from 1.1.1 to 1.1.2 [Alexis Svinartchouk] </summary>
>
>> ### gzip-stream-1.1.2
>> #### (2020-11-30)
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update drivelist from 9.2.1 to 9.2.2 [Alexis Svinartchouk] </summary>
>
>> ### drivelist-9.2.2
>> #### (2020-11-30)
>>
>> * Update typescript to v4.1.2 [Alexis Svinartchouk]
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update blockmap from 4.0.2 to 4.0.3 [Alexis Svinartchouk] </summary>
>
>> ### blockmap-4.0.3
>> #### (2020-11-30)
>>
>> * Update typescript to v4.1.2 [Alexis Svinartchouk]
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update partitioninfo from 6.0.1 to 6.0.2 [Alexis Svinartchouk] </summary>
>
>> ### partitioninfo-6.0.2
>> #### (2020-11-27)
>>
>>
>> <details>
>> <summary> Update file-disk from 8.0.0 to 8.0.1 [Alexis Svinartchouk] </summary>
>>
>>> #### file-disk-8.0.1
>>> ##### (2020-11-26)
>>>
>>> * Add versionbot changelog [Alexis Svinartchouk]
>>>
>> </details>
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update file-disk from 8.0.0 to 8.0.1 [Alexis Svinartchouk] </summary>
>
>> ### file-disk-8.0.1
>> #### (2020-11-26)
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
>> ### file-disk-8.0.1
>> #### (2020-11-26)
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> <details>
> <summary> Update balena-image-fs from 7.0.4 to 7.0.5 [Alexis Svinartchouk] </summary>
>
>> ### balena-image-fs-7.0.5
>> #### (2020-11-27)
>>
>>
>> <details>
>> <summary> Update file-disk from 8.0.0 to 8.0.1 [Alexis Svinartchouk] </summary>
>>
>>> #### file-disk-8.0.1
>>> ##### (2020-11-26)
>>>
>>> * Add versionbot changelog [Alexis Svinartchouk]
>>>
>> </details>
>>
>>
>> <details>
>> <summary> Update ext2fs from 3.0.3 to 3.0.4 [Alexis Svinartchouk] </summary>
>>
>>> #### node-ext2fs-3.0.4
>>> ##### (2020-11-26)
>>>
>>> * Add versionbot changelog [Alexis Svinartchouk]
>>>
>> </details>
>>
>>
>> <details>
>> <summary> Update partitioninfo from 6.0.1 to 6.0.2 [Alexis Svinartchouk] </summary>
>>
>>> #### partitioninfo-6.0.2
>>> ##### (2020-11-27)
>>>
>>>
>>> <details>
>>> <summary> Update file-disk from 8.0.0 to 8.0.1 [Alexis Svinartchouk] </summary>
>>>
>>>> ##### file-disk-8.0.1
>>>> ###### (2020-11-26)
>>>>
>>>> * Add versionbot changelog [Alexis Svinartchouk]
>>>>
>>> </details>
>>>
>>> * Add versionbot changelog [Alexis Svinartchouk]
>>>
>> </details>
>>
>> * Add versionbot changelog [Alexis Svinartchouk]
>>
> </details>
>
>
> ## etcher-sdk-5.1.8
> ### (2020-11-26)
>
> * Add versionbot changelog [Alexis Svinartchouk]
>
> ## etcher-sdk-5.1.7
> ### (2020-11-25)
>
> * Don't start opening drives in advance to avoid unhandled rejections [Alexis Svinartchouk]
> * Update generated docs [Alexis Svinartchouk]
>
> ## etcher-sdk-5.1.6
> ### (2020-11-24)
>
> * Do not unmount source drives [Alexis Svinartchouk]
> * Factorize retrying transient errors [Alexis Svinartchouk]
> * Retry opening files & block devices on transient errors [Alexis Svinartchouk]
> * Update generated docs [Alexis Svinartchouk]
>
</details>
* Set useContentSize to true so the size is the same on all platforms [Alexis Svinartchouk]
# v1.5.111
## (2020-11-23)
* Warn when the source drive has no partition table [Alexis Svinartchouk]
* Use a different icon when no source drive is available [Alexis Svinartchouk]
* Allow selecting a locked SD card as the source drive [Alexis Svinartchouk]
* Remove "Validate write on success" setting. Validation is always enabled, press the "skip" button to skip it. [Alexis Svinartchouk]
* Update electron to v9.3.3 [Alexis Svinartchouk]
* Update etcher-sdk to 5.1.1, use WASM ext2fs module [Alexis Svinartchouk]
# v1.5.110
## (2020-11-04)
* Remove console.log in tests [Lorenzo Alberto Maria Ambrosi]
* Fix URL not being selected with custom protocol [Lorenzo Alberto Maria Ambrosi]
* Add skip function to validation [Lorenzo Alberto Maria Ambrosi]
* Rework success screen [Lorenzo Alberto Maria Ambrosi]
# v1.5.109 # v1.5.109
## (2020-09-14) ## (2020-09-14)

2
CODEOWNERS Normal file
View File

@@ -0,0 +1,2 @@
* @thundron @zvin @jviotti
/scripts @nazrhom

4
FAQ.md
View File

@@ -37,10 +37,10 @@ modules=xwayland.so
Sometimes, things might go wrong, and you end up with a half-flashed drive that is unusable by your operating systems, and common graphical tools might even refuse to get it back to a normal state. Sometimes, things might go wrong, and you end up with a half-flashed drive that is unusable by your operating systems, and common graphical tools might even refuse to get it back to a normal state.
To solve these kinds of problems, we've collected [a list of fail-proof methods](https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md#recovering-broken-drives) to completely erase your drive in major operating systems. To solve these kinds of problems, we've collected [a list of fail-proof methods](https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md#recovering-broken-drives) to completely erase your drive in major operating systems.
## I receive "No polkit authentication agent found" error in GNU/Linux ## I receive No polkit authentication agent found error in GNU/Linux
Etcher requires an available [polkit authentication agent](https://wiki.archlinux.org/index.php/Polkit#Authentication_agents) in your system in order to show a secure password prompt dialog to perform elevation. Make sure you have one installed for the desktop environment of your choice. Etcher requires an available [polkit authentication agent](https://wiki.archlinux.org/index.php/Polkit#Authentication_agents) in your system in order to show a secure password prompt dialog to perform elevation. Make sure you have one installed for the desktop environment of your choice.
## May I run Etcher in older macOS versions? ## May I run Etcher in older macOS versions?
Etcher GUI is based on the [Electron](http://electron.atom.io/) framework, [which only supports macOS 10.10 and newer versions](https://github.com/electron/electron/blob/master/docs/tutorial/support.md#supported-platforms). Etcher GUI is based on the [Electron](http://electron.atom.io/) framework, [which only supports macOS 10.9 and newer versions](https://github.com/electron/electron/blob/master/docs/tutorial/support.md#supported-platforms).

View File

@@ -3,7 +3,7 @@
# --------------------------------------------------------------------- # ---------------------------------------------------------------------
RESIN_SCRIPTS ?= ./scripts/resin RESIN_SCRIPTS ?= ./scripts/resin
export NPM_VERSION ?= 6.14.8 export NPM_VERSION ?= 6.14.5
S3_BUCKET = artifacts.ci.balena-cloud.com S3_BUCKET = artifacts.ci.balena-cloud.com
# This directory will be completely deleted by the `clean` rule # This directory will be completely deleted by the `clean` rule
@@ -66,9 +66,6 @@ else
ifeq ($(shell uname -m),x86_64) ifeq ($(shell uname -m),x86_64)
HOST_ARCH = x64 HOST_ARCH = x64
endif endif
ifeq ($(shell uname -m),arm64)
HOST_ARCH = aarch64
endif
endif endif
endif endif
@@ -89,9 +86,11 @@ TARGET_ARCH ?= $(HOST_ARCH)
# Electron # Electron
# --------------------------------------------------------------------- # ---------------------------------------------------------------------
electron-develop: electron-develop:
git submodule update --init && \ $(RESIN_SCRIPTS)/electron/install.sh \
npm ci && \ -b $(shell pwd) \
npm run webpack -r $(TARGET_ARCH) \
-s $(PLATFORM) \
-m $(NPM_VERSION)
electron-test: electron-test:
$(RESIN_SCRIPTS)/electron/test.sh \ $(RESIN_SCRIPTS)/electron/test.sh \

139
README.md
View File

@@ -5,15 +5,16 @@
Etcher is a powerful OS image flasher built with web technologies to ensure Etcher is a powerful OS image flasher built with web technologies to ensure
flashing an SDCard or USB drive is a pleasant and safe experience. It protects flashing an SDCard or USB drive is a pleasant and safe experience. It protects
you from accidentally writing to your hard-drives, ensures every byte of data you from accidentally writing to your hard-drives, ensures every byte of data
was written correctly, and much more. It can also directly flash Raspberry Pi devices that support [USB device boot mode](https://www.raspberrypi.com/documentation/computers/raspberry-pi.html#usb-device-boot-mode). was written correctly and much more. It can also flash directly Raspberry Pi devices that support the usbboot protocol
[![Current Release](https://img.shields.io/github/release/balena-io/etcher.svg?style=flat-square)](https://balena.io/etcher) [![Current Release](https://img.shields.io/github/release/balena-io/etcher.svg?style=flat-square)](https://balena.io/etcher)
[![License](https://img.shields.io/github/license/balena-io/etcher.svg?style=flat-square)](https://github.com/balena-io/etcher/blob/master/LICENSE) [![License](https://img.shields.io/github/license/balena-io/etcher.svg?style=flat-square)](https://github.com/balena-io/etcher/blob/master/LICENSE)
[![Dependency status](https://img.shields.io/david/balena-io/etcher.svg?style=flat-square)](https://david-dm.org/balena-io/etcher)
[![Balena.io Forums](https://img.shields.io/discourse/https/forums.balena.io/topics.svg?style=flat-square&label=balena.io%20forums)](https://forums.balena.io/c/etcher) [![Balena.io Forums](https://img.shields.io/discourse/https/forums.balena.io/topics.svg?style=flat-square&label=balena.io%20forums)](https://forums.balena.io/c/etcher)
--- ***
[**Download**][etcher] | [**Support**][support] | [**Documentation**][user-documentation] | [**Contributing**][contributing] | [**Roadmap**][milestones] [**Download**][etcher] | [**Support**][SUPPORT] | [**Documentation**][USER-DOCUMENTATION] | [**Contributing**][CONTRIBUTING] | [**Roadmap**][milestones]
## Supported Operating Systems ## Supported Operating Systems
@@ -21,7 +22,7 @@ was written correctly, and much more. It can also directly flash Raspberry Pi de
- macOS 10.10 (Yosemite) and later - macOS 10.10 (Yosemite) and later
- Microsoft Windows 7 and later - Microsoft Windows 7 and later
**Note**: Etcher will run on any platform officially supported by Note that Etcher will run on any platform officially supported by
[Electron][electron]. Read more in their [Electron][electron]. Read more in their
[documentation][electron-supported-platforms]. [documentation][electron-supported-platforms].
@@ -30,27 +31,21 @@ was written correctly, and much more. It can also directly flash Raspberry Pi de
Refer to the [downloads page][etcher] for the latest pre-made Refer to the [downloads page][etcher] for the latest pre-made
installers for all supported operating systems. installers for all supported operating systems.
## Packages
> [![Hosted By: Cloudsmith](https://img.shields.io/badge/OSS%20hosting%20by-cloudsmith-blue?logo=cloudsmith&style=for-the-badge)](https://cloudsmith.com) \
Package repository hosting is graciously provided by [Cloudsmith](https://cloudsmith.com).
Cloudsmith is the only fully hosted, cloud-native, universal package management solution, that
enables your organization to create, store and share packages in any format, to any place, with total
confidence.
#### Debian and Ubuntu based Package Repository (GNU/Linux x86/x64) #### Debian and Ubuntu based Package Repository (GNU/Linux x86/x64)
> Detailed or alternative steps in the [instructions by Cloudsmith](https://cloudsmith.io/~balena/repos/etcher/setup/#formats-deb) 1. Add Etcher debian repository:
1. Add Etcher Debian repository:
```sh ```sh
curl -1sLf \ echo "deb https://deb.etcher.io stable etcher" | sudo tee /etc/apt/sources.list.d/balena-etcher.list
'https://dl.cloudsmith.io/public/balena/etcher/setup.deb.sh' \
| sudo -E bash
``` ```
2. Update and install: 2. Trust Bintray.com's GPG key:
```sh
sudo apt-key adv --keyserver hkps://keyserver.ubuntu.com:443 --recv-keys 379CE192D401AB61
```
3. Update and install:
```sh ```sh
sudo apt-get update sudo apt-get update
@@ -61,48 +56,30 @@ confidence.
```sh ```sh
sudo apt-get remove balena-etcher-electron sudo apt-get remove balena-etcher-electron
rm /etc/apt/sources.list.d/balena-etcher.list sudo rm /etc/apt/sources.list.d/balena-etcher.list
apt-get clean sudo apt-get update
rm -rf /var/lib/apt/lists/*
apt-get update
``` ```
#### Redhat (RHEL) and Fedora-based Package Repository (GNU/Linux x86/x64) ##### OpenSUSE LEAP & Tumbleweed install
> Detailed or alternative steps in the [instructions by Cloudsmith](https://cloudsmith.io/~balena/repos/etcher/setup/#formats-rpm)
##### DNF
1. Add Etcher rpm repository:
```sh
curl -1sLf \
'https://dl.cloudsmith.io/public/balena/etcher/setup.rpm.sh' \
| sudo -E bash
```
2. Update and install:
```sh
sudo dnf install -y balena-etcher-electron
```
###### Uninstall
```sh ```sh
rm /etc/yum.repos.d/balena-etcher.repo sudo zypper ar https://balena.io/etcher/static/etcher-rpm.repo
rm /etc/yum.repos.d/balena-etcher-source.repo sudo zypper ref
sudo zypper in balena-etcher-electron
``` ```
##### Yum ##### Uninstall
```sh
sudo zypper rm balena-etcher-electron
```
#### Redhat (RHEL) and Fedora based Package Repository (GNU/Linux x86/x64)
1. Add Etcher rpm repository: 1. Add Etcher rpm repository:
```sh ```sh
curl -1sLf \ sudo wget https://balena.io/etcher/static/etcher-rpm.repo -O /etc/yum.repos.d/etcher-rpm.repo
'https://dl.cloudsmith.io/public/balena/etcher/setup.rpm.sh' \
| sudo -E bash
``` ```
2. Update and install: 2. Update and install:
@@ -110,38 +87,25 @@ rm /etc/yum.repos.d/balena-etcher-source.repo
```sh ```sh
sudo yum install -y balena-etcher-electron sudo yum install -y balena-etcher-electron
``` ```
or
###### Uninstall
```sh
sudo yum remove -y balena-etcher-electron
rm /etc/yum.repos.d/balena-etcher.repo
rm /etc/yum.repos.d/balena-etcher-source.repo
```
#### OpenSUSE LEAP & Tumbleweed install (zypper)
1. Add the repo
```sh ```sh
curl -1sLf \ sudo dnf install -y balena-etcher-electron
'https://dl.cloudsmith.io/public/balena/etcher/setup.rpm.sh' \
| sudo -E bash
```
2. Update and install
```sh
sudo zypper up
sudo zypper install balena-etcher-electron
``` ```
##### Uninstall ##### Uninstall
```sh ```sh
sudo zypper rm balena-etcher-electron sudo yum remove -y balena-etcher-electron
# remove the repo sudo rm /etc/yum.repos.d/etcher-rpm.repo
sudo zypper rr balena-etcher sudo yum clean all
sudo zypper rr balena-etcher-source sudo yum makecache fast
```
or
```sh
sudo dnf remove -y balena-etcher-electron
sudo rm /etc/yum.repos.d/etcher-rpm.repo
sudo dnf clean all
sudo dnf makecache
``` ```
#### Solus (GNU/Linux x64) #### Solus (GNU/Linux x64)
@@ -156,10 +120,11 @@ sudo eopkg it etcher
sudo eopkg rm etcher sudo eopkg rm etcher
``` ```
#### Arch/Manjaro Linux (GNU/Linux x64) #### Arch Linux / Manjaro (GNU/Linux x64)
Etcher is offered through the Arch User Repository and can be installed on both Manjaro and Arch systems. You can compile it from the source code in this repository using [`balena-etcher`](https://aur.archlinux.org/packages/balena-etcher/). The following example uses a common AUR helper to install the latest release: Etcher is offered through the Arch User Repository and can be installed on both Manjaro and Arch systems. You can compile it from the source code in this repository using [`balena-etcher`](https://aur.archlinux.org/packages/balena-etcher/). The following example uses a common AUR helper to install the latest release:
```sh ```sh
yay -S balena-etcher yay -S balena-etcher
``` ```
@@ -170,20 +135,20 @@ yay -S balena-etcher
yay -R balena-etcher yay -R balena-etcher
``` ```
#### Brew (macOS) #### Brew Cask (macOS)
**Note**: Etcher has to be updated manually to point to new versions, Note that the Etcher Cask has to be updated manually to point to new versions,
so it might not refer to the latest version immediately after an Etcher so it might not refer to the latest version immediately after an Etcher
release. release.
```sh ```sh
brew install balenaetcher brew cask install balenaetcher
``` ```
##### Uninstall ##### Uninstall
```sh ```sh
brew uninstall balenaetcher brew cask uninstall balenaetcher
``` ```
#### Chocolatey (Windows) #### Chocolatey (Windows)
@@ -203,20 +168,20 @@ choco uninstall etcher
## Support ## Support
If you're having any problem, please [raise an issue][newissue] on GitHub, and If you're having any problem, please [raise an issue][newissue] on GitHub and
the balena.io team will be happy to help. the balena.io team will be happy to help.
## License ## License
Etcher is free software and may be redistributed under the terms specified in Etcher is free software, and may be redistributed under the terms specified in
the [license]. the [license].
[etcher]: https://balena.io/etcher [etcher]: https://balena.io/etcher
[electron]: https://electronjs.org/ [electron]: https://electronjs.org/
[electron-supported-platforms]: https://electronjs.org/docs/tutorial/support#supported-platforms [electron-supported-platforms]: https://electronjs.org/docs/tutorial/support#supported-platforms
[support]: https://github.com/balena-io/etcher/blob/master/SUPPORT.md [SUPPORT]: https://github.com/balena-io/etcher/blob/master/SUPPORT.md
[contributing]: https://github.com/balena-io/etcher/blob/master/docs/CONTRIBUTING.md [CONTRIBUTING]: https://github.com/balena-io/etcher/blob/master/docs/CONTRIBUTING.md
[user-documentation]: https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md [USER-DOCUMENTATION]: https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md
[milestones]: https://github.com/balena-io/etcher/milestones [milestones]: https://github.com/balena-io/etcher/milestones
[newissue]: https://github.com/balena-io/etcher/issues/new [newissue]: https://github.com/balena-io/etcher/issues/new
[license]: https://github.com/balena-io/etcher/blob/master/LICENSE [license]: https://github.com/balena-io/etcher/blob/master/LICENSE

View File

@@ -1,16 +1,9 @@
Getting help with BalenaEtcher Getting help with Etcher
=============================== ========================
There are various ways to get support for Etcher if you experience an issue or There are various ways to get support for Etcher if you experience an issue or
have an idea you'd like to share with us. have an idea you'd like to share with us.
Documentation
------
We have answers to a variety of frequently asked questions in the [user
documentation][documentation] and also in the [FAQs][faq] on the Etcher website.
Forums Forums
------ ------
@@ -22,7 +15,7 @@ a look at the existing threads before opening a new one!
Make sure to mention the following information to help us provide better Make sure to mention the following information to help us provide better
support: support:
- The BalenaEtcher version you're running. - The Etcher version you're running.
- The operating system you're running Etcher in. - The operating system you're running Etcher in.
@@ -32,12 +25,10 @@ support:
GitHub GitHub
------ ------
If you encounter an issue or have a suggestion, head on over to BalenaEtcher's [issue If you encounter an issue or have a suggestion, head on over to Etcher's [issue
tracker][issues] and if there isn't a ticket covering it, [create tracker][issues] and if there isn't a ticket covering it, [create
one][new-issue]. one][new-issue].
[discourse]: https://forums.balena.io/c/etcher [discourse]: https://forums.balena.io/c/etcher
[issues]: https://github.com/balena-io/etcher/issues [issues]: https://github.com/balena-io/etcher/issues
[new-issue]: https://github.com/balena-io/etcher/issues/new [new-issue]: https://github.com/balena-io/etcher/issues/new
[documentation]: https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md
[faq]: https://etcher.io

View File

@@ -1,11 +0,0 @@
#!/bin/bash
# Link to the binary
# Must hardcode balenaEtcher directory; no variable available
ln -sf '/opt/balenaEtcher/${executable}' '/usr/bin/${executable}'
# SUID chrome-sandbox for Electron 5+
chmod 4755 '/opt/balenaEtcher/chrome-sandbox' || true
update-mime-database /usr/share/mime || true
update-desktop-database /usr/share/applications || true

Binary file not shown.

26
beforeBuild.js Normal file
View File

@@ -0,0 +1,26 @@
'use strict'
const cp = require('child_process');
const rimraf = require('rimraf');
const process = require('process');
// Rebuild native modules for ia32 and run webpack again for the ia32 part of windows packages
exports.default = function(context) {
if (context.platform.name === 'windows') {
cp.execFileSync(
'bash',
['./node_modules/.bin/electron-rebuild', '--types', 'dev', '--arch', context.arch],
);
rimraf.sync('generated');
cp.execFileSync(
'bash',
['./node_modules/.bin/webpack'],
{
env: {
...process.env,
npm_config_target_arch: context.arch,
},
},
);
}
}

View File

@@ -91,7 +91,7 @@ make electron-develop
```sh ```sh
# Build the GUI # Build the GUI
npm run webpack make webpack
# Start Electron # Start Electron
npm start npm start
``` ```

View File

@@ -160,18 +160,6 @@ pre-installed in all modern Windows versions.
- Run `clean`. This command will completely clean your drive by erasing any - Run `clean`. This command will completely clean your drive by erasing any
existent filesystem. existent filesystem.
- Run `create partition primary`. This command will create a new partition.
- Run `active`. This command will active the partition.
- Run `list partition`. This command will show available partition.
- Run `select partition N`, where `N` corresponds to the id of the newly available partition.
- Run `format override quick`. This command will format the partition. You can choose a specific formatting by adding `FS=xx` where `xx` could be `NTFS or FAT or FAT32` after `format`. Example : `format FS=NTFS override quick`
- Run `exit` to quit diskpart.
### OS X ### OS X
Run the following command in `Terminal.app`, replacing `N` by the corresponding Run the following command in `Terminal.app`, replacing `N` by the corresponding

View File

@@ -1,178 +0,0 @@
---
description: Getting started for etcherPro
slug: /
---
# EtcherPro User manual
## 1. Features
![EtcherPro-manual-01](img/etcher-pro-top-view.jpeg)
![EtcherPro-manual-02](img/etcher-pro-slots.jpeg)
![EtcherPro-manual-03](img/etcher-pro-rear-view.jpeg)
| Specifications | |
| --- | --- |
| Voltage in | 100 to 240 V AC - 10A - 50 to 60 Hz |
| Voltage out | 100 to 240 V AC - 10A - 50 to 60 Hz |
| Ports | 16 x USB 3.0, 16 X SD, 16 x mSD |
| Flashing capacity | 16 x targets when using an online image or, 15 x target drives and 1 x local source drive |
| Daisy-chaining | Up to 10 EtcherPro devices supported (160 x targets) |
| Language support | English |
| Software | balenaEtcher on balenaOS with auto-updates |
| Display | 7in RGB touch screen |
| Network connectivity | WiFi 2.4GHz, 5GHz |
| Working temperature | 5°C ~ 30°C |
| Certifications | CE, FCC |
## 2. Getting started
### 2.1 Setup
### Powering up the device
- EtcherPro is supplied with a mains power cable according to your region's plug standards
- On the back of the device, there are two groups of sockets, labelled as **IN** and **OUT**
- Plug the AC power cable to the socket labelled **POWER IN**, and then to the mains outlet
- The device should boot up automatically; wait until you see the Etcher interface show up
- [Warning] Please avoid using adaptors or extension leads, as this may damage the device or cause it to malfunction
### Connecting to WiFi
- The device will prompt you to connect to a local network the first time it boots (you can skip this step if you are not connecting to WiFi)
- Select the network to which you want to connect, type the password and select **OK**
- You can access the WiFi settings by selecting the WiFi icon at the top left corner of the screen
### 2.2 Etcher functions
### Flash from file
Using an image file as source to flash to one or multiple targets
- From the Etcher menu, select **Flash from file**
- Plug a drive that contains the image you would like to flash into the slot with the blue-colored blinking LED
- Select the image you want to flash from the file browser and select **OK**
- Plug at least one drive or device into an available slot. The plugged target(s) will be selected automatically and the LEDs will turn white
- Select **Flash**, to begin the flashing process
- The LED of each slot will first blink purple for flashing and then green for validating
- Once the flashing is complete, the LEDs will turn green for successfully flashed drives, and red for the failed ones
- You may safely unplug the drives when flashing is complete
### Flash from URL
Using an online image file as source to flash to one or multiple targets
- From the Etcher menu, select **Flash from URL**
- Enter the image URL of you would like to flash in the input field, and select **OK**
- Plug at least one drive or device into an available slot. The plugged target(s) will be selected automatically, and the LEDs will turn white
- Select **Flash**, to begin the flashing process
- The LED of each slot will first blink purple for flashing and then green for validating
- Once the flashing is complete, the LEDs will turn green for successfully flashed drives, and red for the failed ones
- You may safely unplug the drives when flashing is complete
### Clone drive
Using a drive as source, and cloning it to multiple drives
- From the Etcher menu, select **Clone drive**
- Plug a drive you would like to clone into the slot with the blue-colored blinking LED
- The drive will be selected automatically and the LED will stop blinking
- Plug at least one drive into an available slot. The plugged targets will be selected automatically, and the LEDs will turn white
- Select **Flash**, to begin the flashing process
- The LED of each slot will first blink purple for flashing and then green for validating
- Once the flashing is complete, the LEDs will turn green for successfully flashed drives, and red for the failed ones
- You may safely unplug the drives when flashing is complete
### Backup drive
Backing up one or multiple drives into another drive
- From the Etcher menu, select **more options**, then **Backup drive**
- Plug one or more drives you would like to backup into the slot(s) with the blue-colored blinking LED
- The drives will be autoselected and the LED will stop blinking
- Select **OK** to move on to the next step
- Plug the backup drive into the slot with the white-colored blinking LED. The plugged target will be automatically selected and the LED will stop blinking
- Select **Flash**, to begin the flashing process
- The LED of the target slot will first blink purple for flashing, and then green for validating
- Once the flashing is complete, the LED will turn green for a successful flash, and red for a failed one
- You may safely unplug the drive when flashing is complete
### Format drive
Formatting one or multiple drives
- From the Etcher menu, select **more options**, then **Format drive**
- Plug one or more drives you would like to format on the slots with the white-colored blinking LEDs
- The drives will be selected automatically, and the LED will stop blinking
- Select **Flash**, to begin the flashing process
- The LED of the target slot will first blink purple for flashing, and then green for validating
- Once the formatting is complete, the LEDs will turn green for successfully formatted drives, and red for the failed ones
- You may safely unplug the drive when formatting is complete
### 2.3 Daisy-chaining (power only)
Connecting up to 10 EtcherPro devices and power them from one socket.
- EtcherPro allows you to chain power (data chaining will be released later). To do this, you will need a male to female IEC C13/C14 power extension lead (min 10A rated) to connect one EtcherPro to another
- Plug the power extension lead to the 'POWER IN' side of the leading EtcherPro, and then to the 'POWER OUT' side of the successive EtcherPro
- The last EtcherPro of the stack should be plugged directly to the mains power socket
### 2.4 Sleep, wake and power off
- EtcherPro is set to automatically go into sleep mode after a few minutes
- You may change this setting by selecting the settings icon on the top right corner of the screen
- You may put the device to sleep manually by selecting the sleep button on the top left corner of the screen
- To wake up your device, just tap anywhere on the screen
- It is not necessary to power off your device, but if you would like to, you may simply unplug the power-in cable
### 3. Safety and handling
WARNING: Make sure you read and follow the safety and handling instructions before using EtcherPro in order to avoid the potential risk of causing damage to the device, electrical shock, fire, or damage to any other property. If EtcherPro gets physically damaged in any way, or you suspect liquid has leaked into the enclosure, unplug the power cable from the socket and avoid using the device before contacting support (pro@etcher.io).
Handling
It is important not to block the air vents on the back and bottom of the device. As such, we suggest setting up EtcherPro on a well supported desk, with plenty of surrounding space to ensure the device is properly ventilated while in use.
Liquid exposure
EtcherPros enclosure is not waterproof. It is important to keep liquids away from the device to avoid spillages. High humidity environments, rain or snow may also cause damage to the device.
Power
EtcherPro does not have an on/off switch. If you would like to power on the device, you need to plug the power cable into the mains socket. If you would like to power off the device, you need to unplug the power cable. Be sure to unplug the power cable from the socket if you suspect either the cable or the device is physically damaged in some way.
For your own protection and protection of the device, EtcherPro comes with a grounded AC power cable which only fits a grounded mains socket. If you dont have a grounded socket installed, you should contact a specialist who can safely install an appropriate grounded socket. Do not attempt to power on the device without a connected grounding wire or with a power cable that does not meet the original specifications.
Repairing
EtcherPro is not meant to be serviced or repaired by the user. If your device has any issues you should contact support (pro@etcher.io). Attempting to disassemble the device will void the warranty, and could also cause injury or harm.
Radio interference
EtcherPro contains components and radios that emit electromagnetic fields. These electromagnetic fields may interfere with medical devices, such as pacemakers and defibrillators. Consult your physician and medical device manufacturer for information specific to your medical device and whether you need to maintain a safe distance of separation between your medical device(s) and EtcherPro. If you suspect EtcherPro is interfering with your medical device, stop using EtcherPro immediately.
EtcherPro emits electromagnetic fields due to the usage of components and radios. These fields can interfere with other devices and potentially cause them to malfunction. If you suspect EtcherPro is interfering with another device, unplug EtcherPro from the power and contact support (pro@etcher.io).
This equipment is not suitable for use in locations where children are likely to be present.
Atmospheric conditions (dust and vapor)
The EtcherPro enclosure is not sealed. Using the device in an environment that has increased amounts of dust, powder, vapors, corrosive substances, or other contaminants can cause malfunction, injury, and/or fire.
### 4. Warranty
**Limited Product Warranty**
Balena warrants that, for a period of one (1) year after the date of shipment, the Products will be free from defects in materials and workmanship under normal use. As Balenas sole liability and Customers sole and exclusive remedy for any breach of the limited warranty set forth herein, Balena will, at its option and expense, repair or replace any Product returned to Customer during the warranty period that does not comply with such warranty, as confirmed by Balena. Replacement Products will be warranted for the remainder of the original warranty period or ninety (90) days, whichever is longer. All Products that are replaced become the property of Balena. Balena will have no obligation to the extent that any failure of a Product to comply with the limited warranty set forth in this limited product warranty results from or is otherwise attributable to: (i) negligence, misuse, or abuse of the Product; (ii) use of the Product other than in accordance with Balenas published specifications or user manual; (iii) modifications, alterations or repairs to the Product made by a party other than Balena or a party authorized by Balena; (iv) any failure by Customer or a third party to comply with environmental and storage requirements for the Product specified by Balena, including, but not limited to, temperature or humidity ranges; or (v) use of the Product in combination with any third-party devices or products that have not been provided or recommended by Balena. The parties agree that Balenas RMA Policy shall apply to Products returned pursuant to this limited product warranty for a breach of warranty.
THE LIMITED WARRANTY SET FORTH HEREIN IS IN LIEU OF, AND BALENA SPECIFICALLY DISCLAIMS, ANY AND ALL OTHER WARRANTIES AND CONDITIONS, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, ANY IMPLIED WARRANTIES OR CONDITIONS OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT, AND ANY WARRANTIES ARISING OUT OF COURSE OF DEALING OR USAGE OF TRADE. NO ADVICE OR INFORMATION, WHETHER ORAL OR WRITTEN, OBTAINED FROM BALENA OR ELSEWHERE, WILL CREATE ANY WARRANTY NOT EXPRESSLY STATED IN THESE TERMS.
The limited warranty does not apply to:
- Returned items that failed due to an accident, purchasers abuse, neglect or failure to operate in accordance with instructions provided in this refund policy.
- Returned items that failed due to incorrect voltage or improper wiring.
- Returned items that failed due to rain, excessive humidity, corrosive environments, or other contaminants.
- Any item damaged in shipment.
- Any product failure caused by installing or operating product under conditions not in accordance with installation and operation guidelines, or damaged by contact with tools or surroundings.
- Returned items with cosmetic defects that do not interfere with product functionality.
- Returned items that are incomplete or defaced.
- Returned items with a different serial number from what was authorized for return.
- Freight damaged items. If your shipment arrives damaged, you must note the damage on the carrier's delivery record in accordance with the carrier's policy, save the merchandise in the original box and packing it arrived in, and arrange for a carrier inspection of damaged merchandise.
**Initiating a warranty claim**
To initiate a warranty claim, please contact support (pro@etcher.io) to receive a copy of the RMA form. When filling the form, make sure you describe the issue as accurately as possible since it will be used as a basis for determining if the warranty claim is valid or not.
After Balenas evaluation of the return item, Warranty or Out-of-Warranty status will be determined. If the description of the problem is the same as listed on Page 1 of the RMA form, the product will be repaired or replaced under warranty at no charge and shipped back, prepaid, to the customer.
If the description of the problem is different from the problem listed on Page 1 of the RMA form we will contact the customer. At such time, the customer must issue a written confirmation to proceed with the repair(s), agree to cover the costs of the repair and return freight, or authorize the product to be shipped back as is, at the customers expense. Failure to obtain written confirmation within thirty (30) days of notification will result in the product being returned as is, at the customers expense.
If the product has no identifiable problem, we reserve the right to charge for testing and return shipping costs.
For any product returned to balena for reasons other than warranty, a 20% restocking fee and round-trip shipping costs will be deducted from the credit refund. All returned items must be in their original box or crating and must include all packing material, manuals, and accessories.

Binary file not shown.

Before

Width:  |  Height:  |  Size: 195 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 272 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 411 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.4 KiB

View File

@@ -1,9 +1,10 @@
appId: io.balena.etcher appId: io.balena.etcher
copyright: Copyright 2016-2021 Balena Ltd copyright: Copyright 2016-2020 Balena Ltd
productName: balenaEtcher productName: balenaEtcher
npmRebuild: true npmRebuild: true
nodeGypRebuild: false nodeGypRebuild: false
publish: null publish: null
beforeBuild: "./beforeBuild.js"
afterPack: "./afterPack.js" afterPack: "./afterPack.js"
asar: false asar: false
files: files:
@@ -15,7 +16,6 @@ mac:
hardenedRuntime: true hardenedRuntime: true
entitlements: "entitlements.mac.plist" entitlements: "entitlements.mac.plist"
entitlementsInherit: "entitlements.mac.plist" entitlementsInherit: "entitlements.mac.plist"
artifactName: "${productName}-${version}.${ext}"
dmg: dmg:
background: assets/dmg/background.tiff background: assets/dmg/background.tiff
icon: assets/icon.icns icon: assets/icon.icns
@@ -54,6 +54,7 @@ deb:
depends: depends:
- gconf2 - gconf2
- gconf-service - gconf-service
- libappindicator1
- libasound2 - libasound2
- libatk1.0-0 - libatk1.0-0
- libc6 - libc6
@@ -87,7 +88,6 @@ deb:
- libxss1 - libxss1
- libxtst6 - libxtst6
- polkit-1-auth-agent | policykit-1-gnome | polkit-kde-1 - polkit-1-auth-agent | policykit-1-gnome | polkit-kde-1
afterInstall: "./after-install.tpl"
rpm: rpm:
depends: depends:
- util-linux - util-linux

View File

@@ -23,12 +23,17 @@ import * as ReactDOM from 'react-dom';
import { v4 as uuidV4 } from 'uuid'; import { v4 as uuidV4 } from 'uuid';
import * as packageJSON from '../../../package.json'; import * as packageJSON from '../../../package.json';
import { DrivelistDrive, isSourceDrive } from '../../shared/drive-constraints'; import {
DrivelistDrive,
isDriveValid,
isSourceDrive,
} from '../../shared/drive-constraints';
import * as EXIT_CODES from '../../shared/exit-codes'; import * as EXIT_CODES from '../../shared/exit-codes';
import * as messages from '../../shared/messages'; import * as messages from '../../shared/messages';
import * as availableDrives from './models/available-drives'; import * as availableDrives from './models/available-drives';
import * as flashState from './models/flash-state'; import * as flashState from './models/flash-state';
import { deselectImage, getImage } from './models/selection-state'; import { init as ledsInit } from './models/leds';
import { deselectImage, getImage, selectDrive } from './models/selection-state';
import * as settings from './models/settings'; import * as settings from './models/settings';
import { Actions, observe, store } from './models/store'; import { Actions, observe, store } from './models/store';
import * as analytics from './modules/analytics'; import * as analytics from './modules/analytics';
@@ -37,7 +42,6 @@ import * as exceptionReporter from './modules/exception-reporter';
import * as osDialog from './os/dialog'; import * as osDialog from './os/dialog';
import * as windowProgress from './os/window-progress'; import * as windowProgress from './os/window-progress';
import MainPage from './pages/main/MainPage'; import MainPage from './pages/main/MainPage';
import './css/main.css';
window.addEventListener( window.addEventListener(
'unhandledrejection', 'unhandledrejection',
@@ -216,7 +220,8 @@ function prepareDrive(drive: Drive) {
disabled: true, disabled: true,
icon: 'warning', icon: 'warning',
size: null, size: null,
link: 'https://www.raspberrypi.com/documentation/computers/compute-module.html#flashing-the-compute-module-emmc', link:
'https://www.raspberrypi.org/documentation/hardware/computemodule/cm-emmc-flashing.md',
linkCTA: 'Install', linkCTA: 'Install',
linkTitle: 'Install missing drivers', linkTitle: 'Install missing drivers',
linkMessage: outdent` linkMessage: outdent`
@@ -246,6 +251,14 @@ async function addDrive(drive: Drive) {
const drives = getDrives(); const drives = getDrives();
drives[preparedDrive.device] = preparedDrive; drives[preparedDrive.device] = preparedDrive;
setDrives(drives); setDrives(drives);
if (
(await settings.get('autoSelectAllDrives')) &&
drive instanceof sdk.sourceDestination.BlockDevice &&
// @ts-ignore BlockDevice.drive is private
isDriveValid(drive.drive, getImage())
) {
selectDrive(drive.device);
}
} }
function removeDrive(drive: Drive) { function removeDrive(drive: Drive) {
@@ -333,19 +346,13 @@ window.addEventListener('beforeunload', async (event) => {
flashingWorkflowUuid, flashingWorkflowUuid,
}); });
popupExists = false; popupExists = false;
} catch (error: any) { } catch (error) {
exceptionReporter.report(error); exceptionReporter.report(error);
} }
}); });
export async function main() { async function main() {
try {
const { init: ledsInit } = require('./models/leds');
await ledsInit(); await ledsInit();
} catch (error: any) {
exceptionReporter.report(error);
}
ReactDOM.render( ReactDOM.render(
React.createElement(MainPage), React.createElement(MainPage),
document.getElementById('main'), document.getElementById('main'),
@@ -361,3 +368,5 @@ export async function main() {
}, },
); );
} }
main();

View File

@@ -42,8 +42,8 @@ import {
Table, Table,
} from '../../styled-components'; } from '../../styled-components';
import DriveSVGIcon from '../../../assets/tgt.svg';
import { SourceMetadata } from '../source-selector/source-selector'; import { SourceMetadata } from '../source-selector/source-selector';
import { middleEllipsis } from '../../utils/middle-ellipsis';
interface UsbbootDrive extends sourceDestination.UsbbootDrive { interface UsbbootDrive extends sourceDestination.UsbbootDrive {
progress: number; progress: number;
@@ -137,18 +137,15 @@ const InitProgress = styled(
`; `;
export interface DriveSelectorProps export interface DriveSelectorProps
extends Omit<ModalProps, 'done' | 'cancel' | 'onSelect'> { extends Omit<ModalProps, 'done' | 'cancel'> {
write: boolean;
multipleSelection: boolean; multipleSelection: boolean;
showWarnings?: boolean; showWarnings?: boolean;
cancel: (drives: DrivelistDrive[]) => void; cancel: () => void;
done: (drives: DrivelistDrive[]) => void; done: (drives: DrivelistDrive[]) => void;
titleLabel: string; titleLabel: string;
emptyListLabel: string; emptyListLabel: string;
emptyListIcon: JSX.Element;
selectedList?: DrivelistDrive[]; selectedList?: DrivelistDrive[];
updateSelectedList?: () => DrivelistDrive[]; updateSelectedList?: () => DrivelistDrive[];
onSelect?: (drive: DrivelistDrive) => void;
} }
interface DriveSelectorState { interface DriveSelectorState {
@@ -169,14 +166,12 @@ export class DriveSelector extends React.Component<
> { > {
private unsubscribe: (() => void) | undefined; private unsubscribe: (() => void) | undefined;
tableColumns: Array<TableColumn<Drive>>; tableColumns: Array<TableColumn<Drive>>;
originalList: DrivelistDrive[];
constructor(props: DriveSelectorProps) { constructor(props: DriveSelectorProps) {
super(props); super(props);
const defaultMissingDriversModalState: { drive?: DriverlessDrive } = {}; const defaultMissingDriversModalState: { drive?: DriverlessDrive } = {};
const selectedList = this.props.selectedList || []; const selectedList = this.props.selectedList || [];
this.originalList = [...(this.props.selectedList || [])];
this.state = { this.state = {
drives: getDrives(), drives: getDrives(),
@@ -203,9 +198,7 @@ export class DriveSelector extends React.Component<
fill={drive.isSystem ? '#fca321' : '#8f9297'} fill={drive.isSystem ? '#fca321' : '#8f9297'}
/> />
)} )}
<Txt ml={(hasWarnings && 8) || 0}> <Txt ml={(hasWarnings && 8) || 0}>{description}</Txt>
{middleEllipsis(description, 32)}
</Txt>
</Flex> </Flex>
); );
} }
@@ -265,8 +258,7 @@ export class DriveSelector extends React.Component<
return ( return (
isUsbbootDrive(drive) || isUsbbootDrive(drive) ||
isDriverlessDrive(drive) || isDriverlessDrive(drive) ||
!isDriveValid(drive, image, this.props.write) || !isDriveValid(drive, image)
(this.props.write && drive.isReadOnly)
); );
} }
@@ -309,9 +301,9 @@ export class DriveSelector extends React.Component<
case compatibility.system(): case compatibility.system():
return warning.systemDrive(); return warning.systemDrive();
case compatibility.tooSmall(): case compatibility.tooSmall():
const size = const recommendedDriveSize =
this.state.image?.recommendedDriveSize || this.state.image?.size || 0; this.state.image?.recommendedDriveSize || this.state.image?.size || 0;
return warning.tooSmall({ size }, drive); return warning.unrecommendedDriveSize({ recommendedDriveSize }, drive);
} }
} }
@@ -319,7 +311,6 @@ export class DriveSelector extends React.Component<
const statuses: DriveStatus[] = getDriveImageCompatibilityStatuses( const statuses: DriveStatus[] = getDriveImageCompatibilityStatuses(
drive, drive,
this.state.image, this.state.image,
this.props.write,
).slice(0, 2); ).slice(0, 2);
return ( return (
// the column render fn expects a single Element // the column render fn expects a single Element
@@ -354,6 +345,16 @@ export class DriveSelector extends React.Component<
} }
} }
private deselectingAll(rows: DrivelistDrive[]) {
return (
rows.length > 0 &&
rows.length === this.state.selectedList.length &&
this.state.selectedList.every(
(d) => rows.findIndex((r) => d.device === r.device) > -1,
)
);
}
componentDidMount() { componentDidMount() {
this.unsubscribe = store.subscribe(() => { this.unsubscribe = store.subscribe(() => {
const drives = getDrives(); const drives = getDrives();
@@ -379,8 +380,8 @@ export class DriveSelector extends React.Component<
const displayedDrives = this.getDisplayedDrives(drives); const displayedDrives = this.getDisplayedDrives(drives);
const disabledDrives = this.getDisabledDrives(drives, image); const disabledDrives = this.getDisabledDrives(drives, image);
const numberOfSystemDrives = drives.filter(isSystemDrive).length; const numberOfSystemDrives = drives.filter(isSystemDrive).length;
const numberOfDisplayedSystemDrives = const numberOfDisplayedSystemDrives = displayedDrives.filter(isSystemDrive)
displayedDrives.filter(isSystemDrive).length; .length;
const numberOfHiddenSystemDrives = const numberOfHiddenSystemDrives =
numberOfSystemDrives - numberOfDisplayedSystemDrives; numberOfSystemDrives - numberOfDisplayedSystemDrives;
const hasSystemDrives = selectedList.filter(isSystemDrive).length; const hasSystemDrives = selectedList.filter(isSystemDrive).length;
@@ -404,7 +405,7 @@ export class DriveSelector extends React.Component<
</Flex> </Flex>
} }
titleDetails={<Txt fontSize={11}>{getDrives().length} found</Txt>} titleDetails={<Txt fontSize={11}>{getDrives().length} found</Txt>}
cancel={() => cancel(this.originalList)} cancel={cancel}
done={() => done(selectedList)} done={() => done(selectedList)}
action={`Select (${selectedList.length})`} action={`Select (${selectedList.length})`}
primaryButtonProps={{ primaryButtonProps={{
@@ -421,7 +422,7 @@ export class DriveSelector extends React.Component<
alignItems="center" alignItems="center"
width="100%" width="100%"
> >
{this.props.emptyListIcon} <DriveSVGIcon width="40px" height="90px" />
<b>{this.props.emptyListLabel}</b> <b>{this.props.emptyListLabel}</b>
</Flex> </Flex>
) : ( ) : (
@@ -444,34 +445,14 @@ export class DriveSelector extends React.Component<
onCheck={(rows: Drive[]) => { onCheck={(rows: Drive[]) => {
let newSelection = rows.filter(isDrivelistDrive); let newSelection = rows.filter(isDrivelistDrive);
if (this.props.multipleSelection) { if (this.props.multipleSelection) {
if (rows.length === 0) { if (this.deselectingAll(newSelection)) {
newSelection = []; newSelection = [];
} }
const deselecting = selectedList.filter(
(selected) =>
newSelection.filter(
(row) => row.device === selected.device,
).length === 0,
);
const selecting = newSelection.filter(
(row) =>
selectedList.filter(
(selected) => row.device === selected.device,
).length === 0,
);
deselecting.concat(selecting).forEach((row) => {
if (this.props.onSelect) {
this.props.onSelect(row);
}
});
this.setState({ this.setState({
selectedList: newSelection, selectedList: newSelection,
}); });
return; return;
} }
if (this.props.onSelect) {
this.props.onSelect(newSelection[newSelection.length - 1]);
}
this.setState({ this.setState({
selectedList: newSelection.slice(newSelection.length - 1), selectedList: newSelection.slice(newSelection.length - 1),
}); });
@@ -483,9 +464,6 @@ export class DriveSelector extends React.Component<
) { ) {
return; return;
} }
if (this.props.onSelect) {
this.props.onSelect(row);
}
const index = selectedList.findIndex( const index = selectedList.findIndex(
(d) => d.device === row.device, (d) => d.device === row.device,
); );
@@ -534,7 +512,7 @@ export class DriveSelector extends React.Component<
if (missingDriversModal.drive !== undefined) { if (missingDriversModal.drive !== undefined) {
openExternal(missingDriversModal.drive.link); openExternal(missingDriversModal.drive.link);
} }
} catch (error: any) { } catch (error) {
logException(error); logException(error);
} finally { } finally {
this.setState({ missingDriversModal: {} }); this.setState({ missingDriversModal: {} });

View File

@@ -20,7 +20,6 @@ import { v4 as uuidV4 } from 'uuid';
import * as flashState from '../../models/flash-state'; import * as flashState from '../../models/flash-state';
import * as selectionState from '../../models/selection-state'; import * as selectionState from '../../models/selection-state';
import * as settings from '../../models/settings';
import { Actions, store } from '../../models/store'; import { Actions, store } from '../../models/store';
import * as analytics from '../../modules/analytics'; import * as analytics from '../../modules/analytics';
import { FlashAnother } from '../flash-another/flash-another'; import { FlashAnother } from '../flash-another/flash-another';
@@ -40,27 +39,24 @@ function restart(goToMain: () => void) {
goToMain(); goToMain();
} }
async function getSuccessBannerURL() {
return (
(await settings.get('successBannerURL')) ??
'https://www.balena.io/etcher/success-banner?borderTop=false&darkBackground=true'
);
}
function FinishPage({ goToMain }: { goToMain: () => void }) { function FinishPage({ goToMain }: { goToMain: () => void }) {
const [webviewShowing, setWebviewShowing] = React.useState(false); const [webviewShowing, setWebviewShowing] = React.useState(false);
const [successBannerURL, setSuccessBannerURL] = React.useState('');
(async () => {
setSuccessBannerURL(await getSuccessBannerURL());
})();
const flashResults = flashState.getFlashResults(); const flashResults = flashState.getFlashResults();
const errors: FlashError[] = ( let errors: FlashError[] = flashResults.results?.errors;
store.getState().toJS().failedDeviceErrors || [] if (errors === undefined) {
).map(([, error]: [string, FlashError]) => ({ errors = (store.getState().toJS().failedDevicePaths || []).map(
([, error]: [string, FlashError]) => ({
...error, ...error,
})); }),
const { averageSpeed, blockmappedSize, bytesWritten, failed, size } = );
flashState.getFlashState(); }
const {
averageSpeed,
blockmappedSize,
bytesWritten,
failed,
size,
} = flashState.getFlashState();
const { const {
skip, skip,
results = { results = {
@@ -89,7 +85,7 @@ function FinishPage({ goToMain }: { goToMain: () => void }) {
}} }}
> >
<FlashResults <FlashResults
image={selectionState.getImage()?.name} image={selectionState.getImageName()}
results={results} results={results}
skip={skip} skip={skip}
errors={errors} errors={errors}
@@ -103,9 +99,8 @@ function FinishPage({ goToMain }: { goToMain: () => void }) {
}} }}
/> />
</Flex> </Flex>
{successBannerURL.length && (
<SafeWebview <SafeWebview
src={successBannerURL} src="https://www.balena.io/etcher/success-banner?borderTop=false&darkBackground=true"
onWebviewShow={setWebviewShowing} onWebviewShow={setWebviewShowing}
style={{ style={{
display: webviewShowing ? 'flex' : 'none', display: webviewShowing ? 'flex' : 'none',
@@ -116,7 +111,6 @@ function FinishPage({ goToMain }: { goToMain: () => void }) {
height: '100vh', height: '100vh',
}} }}
/> />
)}
</Flex> </Flex>
); );
} }

View File

@@ -17,6 +17,7 @@
import CircleSvg from '@fortawesome/fontawesome-free/svgs/solid/circle.svg'; import CircleSvg from '@fortawesome/fontawesome-free/svgs/solid/circle.svg';
import CheckCircleSvg from '@fortawesome/fontawesome-free/svgs/solid/check-circle.svg'; import CheckCircleSvg from '@fortawesome/fontawesome-free/svgs/solid/check-circle.svg';
import TimesCircleSvg from '@fortawesome/fontawesome-free/svgs/solid/times-circle.svg'; import TimesCircleSvg from '@fortawesome/fontawesome-free/svgs/solid/times-circle.svg';
import * as _ from 'lodash';
import outdent from 'outdent'; import outdent from 'outdent';
import * as React from 'react'; import * as React from 'react';
import { Flex, FlexProps, Link, TableColumn, Txt } from 'rendition'; import { Flex, FlexProps, Link, TableColumn, Txt } from 'rendition';
@@ -26,17 +27,15 @@ import { progress } from '../../../../shared/messages';
import { bytesToMegabytes } from '../../../../shared/units'; import { bytesToMegabytes } from '../../../../shared/units';
import FlashSvg from '../../../assets/flash.svg'; import FlashSvg from '../../../assets/flash.svg';
import { getDrives } from '../../models/available-drives';
import { resetState } from '../../models/flash-state'; import { resetState } from '../../models/flash-state';
import * as selection from '../../models/selection-state'; import * as selection from '../../models/selection-state';
import { middleEllipsis } from '../../utils/middle-ellipsis'; import { middleEllipsis } from '../../utils/middle-ellipsis';
import { Modal, Table } from '../../styled-components'; import { Modal, Table } from '../../styled-components';
const ErrorsTable = styled((props) => <Table<FlashError> {...props} />)` const ErrorsTable = styled((props) => <Table<FlashError> {...props} />)`
&&& [data-display='table-head'], [data-display='table-head'],
&&& [data-display='table-body'] { [data-display='table-body'] {
> [data-display='table-row'] { [data-display='table-cell'] {
> [data-display='table-cell'] {
&:first-child { &:first-child {
width: 30%; width: 30%;
} }
@@ -49,24 +48,28 @@ const ErrorsTable = styled((props) => <Table<FlashError> {...props} />)`
width: 50%; width: 50%;
} }
} }
}
}
`; `;
const DoneIcon = (props: { const DoneIcon = (props: {
skipped: boolean; skipped: boolean;
color: string;
allFailed: boolean; allFailed: boolean;
someFailed: boolean;
}) => { }) => {
const { allFailed, someFailed } = props;
const someOrAllFailed = allFailed || someFailed;
const svgProps = { const svgProps = {
width: '28px', width: '24px',
fill: props.color, fill: someOrAllFailed ? '#c6c8c9' : '#1ac135',
style: { style: {
width: '28px',
height: '28px',
marginTop: '-25px', marginTop: '-25px',
marginLeft: '13px', marginLeft: '13px',
zIndex: 1, zIndex: 1,
color: someOrAllFailed ? '#c6c8c9' : '#1ac135',
}, },
}; };
return props.allFailed && !props.skipped ? ( return allFailed && !props.skipped ? (
<TimesCircleSvg {...svgProps} /> <TimesCircleSvg {...svgProps} />
) : ( ) : (
<CheckCircleSvg {...svgProps} /> <CheckCircleSvg {...svgProps} />
@@ -103,19 +106,6 @@ const columns: Array<TableColumn<FlashError>> = [
}, },
]; ];
function getEffectiveSpeed(results: {
sourceMetadata: {
size: number;
blockmappedSize?: number;
};
averageFlashingSpeed: number;
}) {
const flashedSize =
results.sourceMetadata.blockmappedSize ?? results.sourceMetadata.size;
const timeSpent = flashedSize / results.averageFlashingSpeed;
return results.sourceMetadata.size / timeSpent;
}
export function FlashResults({ export function FlashResults({
goToMain, goToMain,
image = '', image = '',
@@ -129,18 +119,22 @@ export function FlashResults({
errors: FlashError[]; errors: FlashError[];
skip: boolean; skip: boolean;
results: { results: {
bytesWritten: number;
sourceMetadata: { sourceMetadata: {
size: number; size: number;
blockmappedSize?: number; blockmappedSize: number;
}; };
averageFlashingSpeed: number; averageFlashingSpeed: number;
devices: { failed: number; successful: number }; devices: { failed: number; successful: number };
}; };
} & FlexProps) { } & FlexProps) {
const [showErrorsInfo, setShowErrorsInfo] = React.useState(false); const [showErrorsInfo, setShowErrorsInfo] = React.useState(false);
const allFailed = !skip && results.devices.successful === 0; const allFailed = results.devices.successful === 0;
const someFailed = results.devices.failed !== 0 || errors.length !== 0; const effectiveSpeed = _.round(
const effectiveSpeed = bytesToMegabytes(getEffectiveSpeed(results)).toFixed( bytesToMegabytes(
results.sourceMetadata.size /
(results.bytesWritten / results.averageFlashingSpeed),
),
1, 1,
); );
return ( return (
@@ -157,41 +151,42 @@ export function FlashResults({
<DoneIcon <DoneIcon
skipped={skip} skipped={skip}
allFailed={allFailed} allFailed={allFailed}
color={allFailed || someFailed ? '#c6c8c9' : '#1ac135'} someFailed={results.devices.failed !== 0}
/> />
<Txt>{middleEllipsis(image, 24)}</Txt> <Txt>{middleEllipsis(image, 24)}</Txt>
</Flex> </Flex>
<Txt fontSize={24} color="#fff" mb="17px"> <Txt fontSize={24} color="#fff" mb="17px">
Flash {allFailed ? 'Failed' : 'Complete'}! Flash Complete!
</Txt> </Txt>
{skip ? <Txt color="#7e8085">Validation has been skipped</Txt> : null} {skip ? <Flex color="#7e8085">Validation has been skipped</Flex> : null}
</Flex> </Flex>
<Flex flexDirection="column" color="#7e8085"> <Flex flexDirection="column" color="#7e8085">
{results.devices.successful !== 0 ? ( {Object.entries(results.devices).map(([type, quantity]) => {
const failedTargets = type === 'failed';
return quantity ? (
<Flex alignItems="center"> <Flex alignItems="center">
<CircleSvg width="14px" fill="#1ac135" /> <CircleSvg
width="14px"
fill={type === 'failed' ? '#ff4444' : '#1ac135'}
color={failedTargets ? '#ff4444' : '#1ac135'}
/>
<Txt ml="10px" color="#fff"> <Txt ml="10px" color="#fff">
{results.devices.successful} {quantity}
</Txt> </Txt>
<Txt ml="10px"> <Txt
{progress.successful(results.devices.successful)} ml="10px"
</Txt> tooltip={failedTargets ? formattedErrors(errors) : undefined}
</Flex> >
) : null} {progress[type](quantity)}
{errors.length !== 0 ? (
<Flex alignItems="center">
<CircleSvg width="14px" fill="#ff4444" />
<Txt ml="10px" color="#fff">
{errors.length}
</Txt>
<Txt ml="10px" tooltip={formattedErrors(errors)}>
{progress.failed(errors.length)}
</Txt> </Txt>
{failedTargets && (
<Link ml="10px" onClick={() => setShowErrorsInfo(true)}> <Link ml="10px" onClick={() => setShowErrorsInfo(true)}>
more info more info
</Link> </Link>
)}
</Flex> </Flex>
) : null} ) : null;
})}
{!allFailed && ( {!allFailed && (
<Txt <Txt
fontSize="10px" fontSize="10px"
@@ -223,15 +218,12 @@ export function FlashResults({
done={() => { done={() => {
setShowErrorsInfo(false); setShowErrorsInfo(false);
resetState(); resetState();
getDrives() selection
.map((drive) => { .getSelectedDrives()
selection.deselectDrive(drive.device); .filter((drive) =>
return drive.device; errors.every((error) => error.device !== drive.device),
})
.filter((driveDevice) =>
errors.some((error) => error.device === driveDevice),
) )
.forEach((driveDevice) => selection.selectDrive(driveDevice)); .forEach((drive) => selection.deselectDrive(drive.device));
goToMain(); goToMain();
}} }}
> >

View File

@@ -18,12 +18,12 @@ import * as React from 'react';
import { Flex, Button, ProgressBar, Txt } from 'rendition'; import { Flex, Button, ProgressBar, Txt } from 'rendition';
import { default as styled } from 'styled-components'; import { default as styled } from 'styled-components';
import { fromFlashState } from '../../modules/progress-status'; import { fromFlashState, FlashState } from '../../modules/progress-status';
import { StepButton } from '../../styled-components'; import { StepButton } from '../../styled-components';
const FlashProgressBar = styled(ProgressBar)` const FlashProgressBar = styled(ProgressBar)`
> div { > div {
width: 100%; width: 220px;
height: 12px; height: 12px;
color: white !important; color: white !important;
text-shadow: none !important; text-shadow: none !important;
@@ -33,7 +33,7 @@ const FlashProgressBar = styled(ProgressBar)`
} }
} }
width: 100%; width: 220px;
height: 12px; height: 12px;
margin-bottom: 6px; margin-bottom: 6px;
border-radius: 14px; border-radius: 14px;
@@ -44,7 +44,7 @@ const FlashProgressBar = styled(ProgressBar)`
`; `;
interface ProgressButtonProps { interface ProgressButtonProps {
type: 'decompressing' | 'flashing' | 'verifying'; type: FlashState['type'];
active: boolean; active: boolean;
percentage: number; percentage: number;
position: number; position: number;
@@ -58,6 +58,8 @@ const colors = {
decompressing: '#00aeef', decompressing: '#00aeef',
flashing: '#da60ff', flashing: '#da60ff',
verifying: '#1ac135', verifying: '#1ac135',
downloading: '#00aeef',
default: '#00aeef',
} as const; } as const;
const CancelButton = styled(({ type, onClick, ...props }) => { const CancelButton = styled(({ type, onClick, ...props }) => {
@@ -78,6 +80,7 @@ const CancelButton = styled(({ type, onClick, ...props }) => {
export class ProgressButton extends React.PureComponent<ProgressButtonProps> { export class ProgressButton extends React.PureComponent<ProgressButtonProps> {
public render() { public render() {
const type = this.props.type || 'default';
const percentage = this.props.percentage; const percentage = this.props.percentage;
const warning = this.props.warning; const warning = this.props.warning;
const { status, position } = fromFlashState({ const { status, position } = fromFlashState({
@@ -85,7 +88,6 @@ export class ProgressButton extends React.PureComponent<ProgressButtonProps> {
percentage, percentage,
position: this.props.position, position: this.props.position,
}); });
const type = this.props.type || 'default';
if (this.props.active) { if (this.props.active) {
return ( return (
<> <>

View File

@@ -31,7 +31,9 @@ interface ReducedFlashingInfosProps {
style?: React.CSSProperties; style?: React.CSSProperties;
} }
export class ReducedFlashingInfos extends React.Component<ReducedFlashingInfosProps> { export class ReducedFlashingInfos extends React.Component<
ReducedFlashingInfosProps
> {
constructor(props: ReducedFlashingInfosProps) { constructor(props: ReducedFlashingInfosProps) {
super(props); super(props);
this.state = {}; this.state = {};

View File

@@ -15,7 +15,6 @@
*/ */
import * as electron from 'electron'; import * as electron from 'electron';
import * as _ from 'lodash';
import * as React from 'react'; import * as React from 'react';
import * as packageJSON from '../../../../../package.json'; import * as packageJSON from '../../../../../package.json';
@@ -94,8 +93,8 @@ export class SafeWebview extends React.PureComponent<
); );
this.entryHref = url.href; this.entryHref = url.href;
// Events steal 'this' // Events steal 'this'
this.didFailLoad = _.bind(this.didFailLoad, this); this.didFailLoad = this.didFailLoad.bind(this);
this.didGetResponseDetails = _.bind(this.didGetResponseDetails, this); this.didGetResponseDetails = this.didGetResponseDetails.bind(this);
// Make a persistent electron session for the webview // Make a persistent electron session for the webview
this.session = electron.remote.session.fromPartition(ELECTRON_SESSION, { this.session = electron.remote.session.fromPartition(ELECTRON_SESSION, {
// Disable the cache for the session such that new content shows up when refreshing // Disable the cache for the session such that new content shows up when refreshing

View File

@@ -16,8 +16,9 @@
import GithubSvg from '@fortawesome/fontawesome-free/svgs/brands/github.svg'; import GithubSvg from '@fortawesome/fontawesome-free/svgs/brands/github.svg';
import * as _ from 'lodash'; import * as _ from 'lodash';
import * as os from 'os';
import * as React from 'react'; import * as React from 'react';
import { Box, Checkbox, Flex, TextWithCopy, Txt } from 'rendition'; import { Flex, Checkbox, Txt } from 'rendition';
import { version, packageType } from '../../../../../package.json'; import { version, packageType } from '../../../../../package.json';
import * as settings from '../../models/settings'; import * as settings from '../../models/settings';
@@ -25,39 +26,50 @@ import * as analytics from '../../modules/analytics';
import { open as openExternal } from '../../os/open-external/services/open-external'; import { open as openExternal } from '../../os/open-external/services/open-external';
import { Modal } from '../../styled-components'; import { Modal } from '../../styled-components';
const platform = os.platform();
interface Setting { interface Setting {
name: string; name: string;
label: string | JSX.Element; label: string | JSX.Element;
options?: {
description: string;
confirmLabel: string;
};
hide?: boolean;
} }
async function getSettingsList(): Promise<Setting[]> { async function getSettingsList(): Promise<Setting[]> {
const list: Setting[] = [ return [
{ {
name: 'errorReporting', name: 'errorReporting',
label: 'Anonymously report errors and usage statistics to balena.io', label: 'Anonymously report errors and usage statistics to balena.io',
}, },
]; {
if (['appimage', 'nsis', 'dmg'].includes(packageType)) { name: 'unmountOnSuccess',
list.push({ /**
* On Windows, "Unmounting" basically means "ejecting".
* On top of that, Windows users are usually not even
* familiar with the meaning of "unmount", which comes
* from the UNIX world.
*/
label: `${platform === 'win32' ? 'Eject' : 'Auto-unmount'} on success`,
},
{
name: 'validateWriteOnSuccess',
label: 'Validate write on success',
},
{
name: 'updatesEnabled', name: 'updatesEnabled',
label: 'Auto-updates enabled', label: 'Auto-updates enabled',
}); hide: ['rpm', 'deb'].includes(packageType),
} },
return list; ];
} }
interface SettingsModalProps { interface SettingsModalProps {
toggleModal: (value: boolean) => void; toggleModal: (value: boolean) => void;
} }
const UUID = process.env.BALENA_DEVICE_UUID;
const InfoBox = (props: any) => (
<Box fontSize={14}>
<Txt>{props.label}</Txt>
<TextWithCopy code text={props.value} copy={props.value} />
</Box>
);
export function SettingsModal({ toggleModal }: SettingsModalProps) { export function SettingsModal({ toggleModal }: SettingsModalProps) {
const [settingsList, setCurrentSettingsList] = React.useState<Setting[]>([]); const [settingsList, setCurrentSettingsList] = React.useState<Setting[]>([]);
React.useEffect(() => { React.useEffect(() => {
@@ -78,14 +90,25 @@ export function SettingsModal({ toggleModal }: SettingsModalProps) {
})(); })();
}); });
const toggleSetting = async (setting: string) => { const toggleSetting = async (
setting: string,
options?: Setting['options'],
) => {
const value = currentSettings[setting]; const value = currentSettings[setting];
analytics.logEvent('Toggle setting', { setting, value }); const dangerous = options !== undefined;
analytics.logEvent('Toggle setting', {
setting,
value,
dangerous,
});
await settings.set(setting, !value); await settings.set(setting, !value);
setCurrentSettings({ setCurrentSettings({
...currentSettings, ...currentSettings,
[setting]: !value, [setting]: !value,
}); });
return;
}; };
return ( return (
@@ -99,24 +122,18 @@ export function SettingsModal({ toggleModal }: SettingsModalProps) {
> >
<Flex flexDirection="column"> <Flex flexDirection="column">
{settingsList.map((setting: Setting, i: number) => { {settingsList.map((setting: Setting, i: number) => {
return ( return setting.hide ? null : (
<Flex key={setting.name} mb={14}> <Flex key={setting.name} mb={14}>
<Checkbox <Checkbox
toggle toggle
tabIndex={6 + i} tabIndex={6 + i}
label={setting.label} label={setting.label}
checked={currentSettings[setting.name]} checked={currentSettings[setting.name]}
onChange={() => toggleSetting(setting.name)} onChange={() => toggleSetting(setting.name, setting.options)}
/> />
</Flex> </Flex>
); );
})} })}
{UUID !== undefined && (
<Flex flexDirection="column">
<Txt fontSize={24}>System Information</Txt>
<InfoBox label="UUID" value={UUID.substr(0, 7)} />
</Flex>
)}
<Flex <Flex
mt={18} mt={18}
alignItems="center" alignItems="center"

View File

@@ -18,8 +18,6 @@ import CopySvg from '@fortawesome/fontawesome-free/svgs/solid/copy.svg';
import FileSvg from '@fortawesome/fontawesome-free/svgs/solid/file.svg'; import FileSvg from '@fortawesome/fontawesome-free/svgs/solid/file.svg';
import LinkSvg from '@fortawesome/fontawesome-free/svgs/solid/link.svg'; import LinkSvg from '@fortawesome/fontawesome-free/svgs/solid/link.svg';
import ExclamationTriangleSvg from '@fortawesome/fontawesome-free/svgs/solid/exclamation-triangle.svg'; import ExclamationTriangleSvg from '@fortawesome/fontawesome-free/svgs/solid/exclamation-triangle.svg';
import ChevronDownSvg from '@fortawesome/fontawesome-free/svgs/solid/chevron-down.svg';
import ChevronRightSvg from '@fortawesome/fontawesome-free/svgs/solid/chevron-right.svg';
import { sourceDestination } from 'etcher-sdk'; import { sourceDestination } from 'etcher-sdk';
import { ipcRenderer, IpcRendererEvent } from 'electron'; import { ipcRenderer, IpcRendererEvent } from 'electron';
import * as _ from 'lodash'; import * as _ from 'lodash';
@@ -27,16 +25,7 @@ import { GPTPartition, MBRPartition } from 'partitioninfo';
import * as path from 'path'; import * as path from 'path';
import * as prettyBytes from 'pretty-bytes'; import * as prettyBytes from 'pretty-bytes';
import * as React from 'react'; import * as React from 'react';
import { import { Flex, ButtonProps, Modal as SmallModal, Txt } from 'rendition';
Flex,
ButtonProps,
Modal as SmallModal,
Txt,
Card as BaseCard,
Input,
Spinner,
Link,
} from 'rendition';
import styled from 'styled-components'; import styled from 'styled-components';
import * as errors from '../../../../shared/errors'; import * as errors from '../../../../shared/errors';
@@ -51,65 +40,21 @@ import { replaceWindowsNetworkDriveLetter } from '../../os/windows-network-drive
import { import {
ChangeButton, ChangeButton,
DetailsText, DetailsText,
Modal,
StepButton, StepButton,
StepNameButton, StepNameButton,
ScrollableFlex,
} from '../../styled-components'; } from '../../styled-components';
import { colors } from '../../theme'; import { colors } from '../../theme';
import { middleEllipsis } from '../../utils/middle-ellipsis'; import { middleEllipsis } from '../../utils/middle-ellipsis';
import URLSelector from '../url-selector/url-selector';
import { SVGIcon } from '../svg-icon/svg-icon'; import { SVGIcon } from '../svg-icon/svg-icon';
import ImageSvg from '../../../assets/image.svg'; import ImageSvg from '../../../assets/image.svg';
import SrcSvg from '../../../assets/src.svg';
import { DriveSelector } from '../drive-selector/drive-selector'; import { DriveSelector } from '../drive-selector/drive-selector';
import { DrivelistDrive } from '../../../../shared/drive-constraints'; import { DrivelistDrive } from '../../../../shared/drive-constraints';
import axios, { AxiosRequestConfig } from 'axios';
import { isJson } from '../../../../shared/utils';
const recentUrlImagesKey = 'recentUrlImages';
function normalizeRecentUrlImages(urls: any[]): URL[] {
if (!Array.isArray(urls)) {
urls = [];
}
urls = urls
.map((url) => {
try {
return new URL(url);
} catch (error: any) {
// Invalid URL, skip
}
})
.filter((url) => url !== undefined);
urls = _.uniqBy(urls, (url) => url.href);
return urls.slice(urls.length - 5);
}
function getRecentUrlImages(): URL[] {
let urls = [];
try {
urls = JSON.parse(localStorage.getItem(recentUrlImagesKey) || '[]');
} catch {
// noop
}
return normalizeRecentUrlImages(urls);
}
function setRecentUrlImages(urls: URL[]) {
const normalized = normalizeRecentUrlImages(urls.map((url: URL) => url.href));
localStorage.setItem(recentUrlImagesKey, JSON.stringify(normalized));
}
const isURL = (imagePath: string) => const isURL = (imagePath: string) =>
imagePath.startsWith('https://') || imagePath.startsWith('http://'); imagePath.startsWith('https://') || imagePath.startsWith('http://');
const Card = styled(BaseCard)`
hr {
margin: 5px 0;
}
`;
// TODO move these styles to rendition // TODO move these styles to rendition
const ModalText = styled.p` const ModalText = styled.p`
a { a {
@@ -122,11 +67,10 @@ const ModalText = styled.p`
`; `;
function getState() { function getState() {
const image = selectionState.getImage();
return { return {
hasImage: selectionState.hasImage(), hasImage: selectionState.hasImage(),
imageName: image?.name, imageName: selectionState.getImageName(),
imageSize: image?.size, imageSize: selectionState.getImageSize(),
}; };
} }
@@ -134,132 +78,6 @@ function isString(value: any): value is string {
return typeof value === 'string'; return typeof value === 'string';
} }
const URLSelector = ({
done,
cancel,
}: {
done: (imageURL: string, auth?: Authentication) => void;
cancel: () => void;
}) => {
const [imageURL, setImageURL] = React.useState('');
const [recentImages, setRecentImages] = React.useState<URL[]>([]);
const [loading, setLoading] = React.useState(false);
const [showBasicAuth, setShowBasicAuth] = React.useState(false);
const [username, setUsername] = React.useState('');
const [password, setPassword] = React.useState('');
React.useEffect(() => {
const fetchRecentUrlImages = async () => {
const recentUrlImages: URL[] = await getRecentUrlImages();
setRecentImages(recentUrlImages);
};
fetchRecentUrlImages();
}, []);
return (
<Modal
cancel={cancel}
primaryButtonProps={{
disabled: loading || !imageURL,
}}
action={loading ? <Spinner /> : 'OK'}
done={async () => {
setLoading(true);
const urlStrings = recentImages.map((url: URL) => url.href);
const normalizedRecentUrls = normalizeRecentUrlImages([
...urlStrings,
imageURL,
]);
setRecentUrlImages(normalizedRecentUrls);
const auth = username ? { username, password } : undefined;
await done(imageURL, auth);
}}
>
<Flex flexDirection="column">
<Flex mb={15} style={{ width: '100%' }} flexDirection="column">
<Txt mb="10px" fontSize="24px">
Use Image URL
</Txt>
<Input
value={imageURL}
placeholder="Enter a valid URL"
type="text"
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
setImageURL(evt.target.value)
}
/>
<Link
mt={15}
mb={15}
fontSize="14px"
onClick={() => {
if (showBasicAuth) {
setUsername('');
setPassword('');
}
setShowBasicAuth(!showBasicAuth);
}}
>
<Flex alignItems="center">
{showBasicAuth && (
<ChevronDownSvg height="1em" fill="currentColor" />
)}
{!showBasicAuth && (
<ChevronRightSvg height="1em" fill="currentColor" />
)}
<Txt ml={8}>Authentication</Txt>
</Flex>
</Link>
{showBasicAuth && (
<React.Fragment>
<Input
mb={15}
value={username}
placeholder="Enter username"
type="text"
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
setUsername(evt.target.value)
}
/>
<Input
value={password}
placeholder="Enter password"
type="password"
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
setPassword(evt.target.value)
}
/>
</React.Fragment>
)}
</Flex>
{recentImages.length > 0 && (
<Flex flexDirection="column" height="78.6%">
<Txt fontSize={18}>Recent</Txt>
<ScrollableFlex flexDirection="column">
<Card
p="10px 15px"
rows={recentImages
.map((recent) => (
<Txt
key={recent.href}
onClick={() => {
setImageURL(recent.href);
}}
style={{
overflowWrap: 'break-word',
}}
>
{recent.pathname.split('/').pop()} - {recent.href}
</Txt>
))
.reverse()}
/>
</ScrollableFlex>
</Flex>
)}
</Flex>
</Modal>
);
};
interface Flow { interface Flow {
icon?: JSX.Element; icon?: JSX.Element;
onClick: (evt: React.MouseEvent) => void; onClick: (evt: React.MouseEvent) => void;
@@ -315,7 +133,6 @@ export interface SourceMetadata extends sourceDestination.Metadata {
drive?: DrivelistDrive; drive?: DrivelistDrive;
extension?: string; extension?: string;
archiveExtension?: string; archiveExtension?: string;
auth?: Authentication;
} }
interface SourceSelectorProps { interface SourceSelectorProps {
@@ -331,13 +148,6 @@ interface SourceSelectorState {
showURLSelector: boolean; showURLSelector: boolean;
showDriveSelector: boolean; showDriveSelector: boolean;
defaultFlowActive: boolean; defaultFlowActive: boolean;
imageSelectorOpen: boolean;
imageLoading: boolean;
}
interface Authentication {
username: string;
password: string;
} }
export class SourceSelector extends React.Component< export class SourceSelector extends React.Component<
@@ -355,8 +165,6 @@ export class SourceSelector extends React.Component<
showURLSelector: false, showURLSelector: false,
showDriveSelector: false, showDriveSelector: false,
defaultFlowActive: true, defaultFlowActive: true,
imageSelectorOpen: false,
imageLoading: false,
}; };
// Bind `this` since it's used in an event's callback // Bind `this` since it's used in an event's callback
@@ -377,52 +185,25 @@ export class SourceSelector extends React.Component<
} }
private async onSelectImage(_event: IpcRendererEvent, imagePath: string) { private async onSelectImage(_event: IpcRendererEvent, imagePath: string) {
this.setState({ imageLoading: true });
await this.selectSource( await this.selectSource(
imagePath, imagePath,
isURL(this.normalizeImagePath(imagePath)) isURL(imagePath) ? sourceDestination.Http : sourceDestination.File,
? sourceDestination.Http
: sourceDestination.File,
).promise; ).promise;
this.setState({ imageLoading: false });
} }
private async createSource( private async createSource(selected: string, SourceType: Source) {
selected: string,
SourceType: Source,
auth?: Authentication,
) {
try { try {
selected = await replaceWindowsNetworkDriveLetter(selected); selected = await replaceWindowsNetworkDriveLetter(selected);
} catch (error: any) { } catch (error) {
analytics.logException(error); analytics.logException(error);
} }
if (isJson(decodeURIComponent(selected))) {
const config: AxiosRequestConfig = JSON.parse(
decodeURIComponent(selected),
);
return new sourceDestination.Http({
url: config.url!,
axiosInstance: axios.create(_.omit(config, ['url'])),
});
}
if (SourceType === sourceDestination.File) { if (SourceType === sourceDestination.File) {
return new sourceDestination.File({ return new sourceDestination.File({
path: selected, path: selected,
}); });
} }
return new sourceDestination.Http({ url: selected });
return new sourceDestination.Http({ url: selected, auth });
}
public normalizeImagePath(imgPath: string) {
const decodedPath = decodeURIComponent(imgPath);
if (isJson(decodedPath)) {
return JSON.parse(decodedPath).url ?? decodedPath;
}
return decodedPath;
} }
private reselectSource() { private reselectSource() {
@@ -436,7 +217,6 @@ export class SourceSelector extends React.Component<
private selectSource( private selectSource(
selected: string | DrivelistDrive, selected: string | DrivelistDrive,
SourceType: Source, SourceType: Source,
auth?: Authentication,
): { promise: Promise<void>; cancel: () => void } { ): { promise: Promise<void>; cancel: () => void } {
let cancelled = false; let cancelled = false;
return { return {
@@ -448,10 +228,7 @@ export class SourceSelector extends React.Component<
let source; let source;
let metadata: SourceMetadata | undefined; let metadata: SourceMetadata | undefined;
if (isString(selected)) { if (isString(selected)) {
if ( if (SourceType === sourceDestination.Http && !isURL(selected)) {
SourceType === sourceDestination.Http &&
!isURL(this.normalizeImagePath(selected))
) {
this.handleError( this.handleError(
'Unsupported protocol', 'Unsupported protocol',
selected, selected,
@@ -469,7 +246,7 @@ export class SourceSelector extends React.Component<
}, },
}); });
} }
source = await this.createSource(selected, SourceType, auth); source = await this.createSource(selected, SourceType);
if (cancelled) { if (cancelled) {
return; return;
@@ -486,7 +263,7 @@ export class SourceSelector extends React.Component<
} }
metadata.SourceType = SourceType; metadata.SourceType = SourceType;
if (!metadata.hasMBR && this.state.warning === null) { if (!metadata.hasMBR) {
analytics.logEvent('Missing partition table', { metadata }); analytics.logEvent('Missing partition table', { metadata });
this.setState({ this.setState({
warning: { warning: {
@@ -495,7 +272,7 @@ export class SourceSelector extends React.Component<
}, },
}); });
} }
} catch (error: any) { } catch (error) {
this.handleError( this.handleError(
'Error opening source', 'Error opening source',
sourcePath, sourcePath,
@@ -505,20 +282,11 @@ export class SourceSelector extends React.Component<
} finally { } finally {
try { try {
await source.close(); await source.close();
} catch (error: any) { } catch (error) {
// Noop // Noop
} }
} }
} else { } else {
if (selected.partitionTableType === null) {
analytics.logEvent('Missing partition table', { selected });
this.setState({
warning: {
message: messages.warning.driveMissingPartitionTable(),
title: 'Missing partition table',
},
});
}
metadata = { metadata = {
path: selected.device, path: selected.device,
displayName: selected.displayName, displayName: selected.displayName,
@@ -530,7 +298,6 @@ export class SourceSelector extends React.Component<
} }
if (metadata !== undefined) { if (metadata !== undefined) {
metadata.auth = auth;
selectionState.selectSource(metadata); selectionState.selectSource(metadata);
analytics.logEvent('Select image', { analytics.logEvent('Select image', {
// An easy way so we can quickly identify if we're making use of // An easy way so we can quickly identify if we're making use of
@@ -585,7 +352,6 @@ export class SourceSelector extends React.Component<
private async openImageSelector() { private async openImageSelector() {
analytics.logEvent('Open image selector'); analytics.logEvent('Open image selector');
this.setState({ imageSelectorOpen: true });
try { try {
const imagePath = await osDialog.selectImage(); const imagePath = await osDialog.selectImage();
@@ -596,10 +362,8 @@ export class SourceSelector extends React.Component<
return; return;
} }
await this.selectSource(imagePath, sourceDestination.File).promise; await this.selectSource(imagePath, sourceDestination.File).promise;
} catch (error: any) { } catch (error) {
exceptionReporter.report(error); exceptionReporter.report(error);
} finally {
this.setState({ imageSelectorOpen: false });
} }
} }
@@ -638,7 +402,7 @@ export class SourceSelector extends React.Component<
private showSelectedImageDetails() { private showSelectedImageDetails() {
analytics.logEvent('Show selected image tooltip', { analytics.logEvent('Show selected image tooltip', {
imagePath: selectionState.getImage()?.path, imagePath: selectionState.getImagePath(),
}); });
this.setState({ this.setState({
@@ -650,21 +414,10 @@ export class SourceSelector extends React.Component<
this.setState({ defaultFlowActive }); this.setState({ defaultFlowActive });
} }
private closeModal() {
this.setState({
showDriveSelector: false,
});
}
// TODO add a visual change when dragging a file over the selector // TODO add a visual change when dragging a file over the selector
public render() { public render() {
const { flashing } = this.props; const { flashing } = this.props;
const { const { showImageDetails, showURLSelector, showDriveSelector } = this.state;
showImageDetails,
showURLSelector,
showDriveSelector,
imageLoading,
} = this.state;
const selectionImage = selectionState.getImage(); const selectionImage = selectionState.getImage();
let image: SourceMetadata | DrivelistDrive = let image: SourceMetadata | DrivelistDrive =
selectionImage !== undefined ? selectionImage : ({} as SourceMetadata); selectionImage !== undefined ? selectionImage : ({} as SourceMetadata);
@@ -702,18 +455,16 @@ export class SourceSelector extends React.Component<
}} }}
/> />
{selectionImage !== undefined || imageLoading ? ( {selectionImage !== undefined ? (
<> <>
<StepNameButton <StepNameButton
plain plain
onClick={() => this.showSelectedImageDetails()} onClick={() => this.showSelectedImageDetails()}
tooltip={imageName || imageBasename} tooltip={imageName || imageBasename}
> >
<Spinner show={imageLoading}>
{middleEllipsis(imageName || imageBasename, 20)} {middleEllipsis(imageName || imageBasename, 20)}
</Spinner>
</StepNameButton> </StepNameButton>
{!flashing && !imageLoading && ( {!flashing && (
<ChangeButton <ChangeButton
plain plain
mb={14} mb={14}
@@ -722,14 +473,13 @@ export class SourceSelector extends React.Component<
Remove Remove
</ChangeButton> </ChangeButton>
)} )}
{!_.isNil(imageSize) && !imageLoading && ( {!_.isNil(imageSize) && (
<DetailsText>{prettyBytes(imageSize)}</DetailsText> <DetailsText>{prettyBytes(imageSize)}</DetailsText>
)} )}
</> </>
) : ( ) : (
<> <>
<FlowSelector <FlowSelector
disabled={this.state.imageSelectorOpen}
primary={this.state.defaultFlowActive} primary={this.state.defaultFlowActive}
key="Flash from file" key="Flash from file"
flow={{ flow={{
@@ -766,9 +516,6 @@ export class SourceSelector extends React.Component<
{this.state.warning != null && ( {this.state.warning != null && (
<SmallModal <SmallModal
style={{
boxShadow: '0 3px 7px rgba(0, 0, 0, 0.3)',
}}
titleElement={ titleElement={
<span> <span>
<ExclamationTriangleSvg fill="#fca321" height="1em" />{' '} <ExclamationTriangleSvg fill="#fca321" height="1em" />{' '}
@@ -817,7 +564,7 @@ export class SourceSelector extends React.Component<
showURLSelector: false, showURLSelector: false,
}); });
}} }}
done={async (imageURL: string, auth?: Authentication) => { done={async (imageURL: string) => {
// Avoid analytics and selection state changes // Avoid analytics and selection state changes
// if no file was resolved from the dialog. // if no file was resolved from the dialog.
if (!imageURL) { if (!imageURL) {
@@ -827,7 +574,6 @@ export class SourceSelector extends React.Component<
({ promise, cancel: cancelURLSelection } = this.selectSource( ({ promise, cancel: cancelURLSelection } = this.selectSource(
imageURL, imageURL,
sourceDestination.Http, sourceDestination.Http,
auth,
)); ));
await promise; await promise;
} }
@@ -840,35 +586,24 @@ export class SourceSelector extends React.Component<
{showDriveSelector && ( {showDriveSelector && (
<DriveSelector <DriveSelector
write={false}
multipleSelection={false} multipleSelection={false}
titleLabel="Select source" titleLabel="Select source"
emptyListLabel="Plug a source drive" emptyListLabel="Plug a source"
emptyListIcon={<SrcSvg width="40px" />} cancel={() => {
cancel={(originalList) => { this.setState({
if (originalList.length) { showDriveSelector: false,
const originalSource = originalList[0]; });
if (selectionImage?.drive?.device !== originalSource.device) { }}
this.selectSource( done={async (drives: DrivelistDrive[]) => {
originalSource, if (drives.length) {
await this.selectSource(
drives[0],
sourceDestination.BlockDevice, sourceDestination.BlockDevice,
); );
} }
} else { this.setState({
selectionState.deselectImage(); showDriveSelector: false,
} });
this.closeModal();
}}
done={() => this.closeModal()}
onSelect={(drive) => {
if (drive) {
if (
selectionState.getImage()?.drive?.device === drive?.device
) {
return selectionState.deselectImage();
}
this.selectSource(drive, sourceDestination.BlockDevice);
}
}} }}
/> />
)} )}

View File

@@ -24,7 +24,7 @@ import {
} from '../../../../shared/drive-constraints'; } from '../../../../shared/drive-constraints';
import { compatibility, warning } from '../../../../shared/messages'; import { compatibility, warning } from '../../../../shared/messages';
import * as prettyBytes from 'pretty-bytes'; import * as prettyBytes from 'pretty-bytes';
import { getImage, getSelectedDrives } from '../../models/selection-state'; import { getSelectedDrives } from '../../models/selection-state';
import { import {
ChangeButton, ChangeButton,
DetailsText, DetailsText,
@@ -80,11 +80,9 @@ export function TargetSelectorButton(props: TargetSelectorProps) {
if (targets.length === 1) { if (targets.length === 1) {
const target = targets[0]; const target = targets[0];
const warnings = getDriveImageCompatibilityStatuses( const warnings = getDriveImageCompatibilityStatuses(target).map(
target, getDriveWarning,
getImage(), );
true,
).map(getDriveWarning);
return ( return (
<> <>
<StepNameButton plain tooltip={props.tooltip}> <StepNameButton plain tooltip={props.tooltip}>
@@ -108,11 +106,9 @@ export function TargetSelectorButton(props: TargetSelectorProps) {
if (targets.length > 1) { if (targets.length > 1) {
const targetsTemplate = []; const targetsTemplate = [];
for (const target of targets) { for (const target of targets) {
const warnings = getDriveImageCompatibilityStatuses( const warnings = getDriveImageCompatibilityStatuses(target).map(
target, getDriveWarning,
getImage(), );
true,
).map(getDriveWarning);
targetsTemplate.push( targetsTemplate.push(
<DetailsText <DetailsText
key={target.device} key={target.device}

View File

@@ -14,6 +14,7 @@
* limitations under the License. * limitations under the License.
*/ */
import { scanner } from 'etcher-sdk';
import * as React from 'react'; import * as React from 'react';
import { Flex, Txt } from 'rendition'; import { Flex, Txt } from 'rendition';
@@ -27,16 +28,14 @@ import {
getSelectedDrives, getSelectedDrives,
deselectDrive, deselectDrive,
selectDrive, selectDrive,
deselectAllDrives,
} from '../../models/selection-state'; } from '../../models/selection-state';
import * as settings from '../../models/settings';
import { observe } from '../../models/store'; import { observe } from '../../models/store';
import * as analytics from '../../modules/analytics'; import * as analytics from '../../modules/analytics';
import { TargetSelectorButton } from './target-selector-button'; import { TargetSelectorButton } from './target-selector-button';
import TgtSvg from '../../../assets/tgt.svg';
import DriveSvg from '../../../assets/drive.svg'; import DriveSvg from '../../../assets/drive.svg';
import { warning } from '../../../../shared/messages'; import { warning } from '../../../../shared/messages';
import { DrivelistDrive } from '../../../../shared/drive-constraints';
export const getDriveListLabel = () => { export const getDriveListLabel = () => {
return getSelectedDrives() return getSelectedDrives()
@@ -46,7 +45,12 @@ export const getDriveListLabel = () => {
.join('\n'); .join('\n');
}; };
const shouldShowDrivesButton = () => {
return !settings.getSync('disableExplicitDriveSelection');
};
const getDriveSelectionStateSlice = () => ({ const getDriveSelectionStateSlice = () => ({
showDrivesButton: shouldShowDrivesButton(),
driveListLabel: getDriveListLabel(), driveListLabel: getDriveListLabel(),
targets: getSelectedDrives(), targets: getSelectedDrives(),
image: getImage(), image: getImage(),
@@ -55,14 +59,13 @@ const getDriveSelectionStateSlice = () => ({
export const TargetSelectorModal = ( export const TargetSelectorModal = (
props: Omit< props: Omit<
DriveSelectorProps, DriveSelectorProps,
'titleLabel' | 'emptyListLabel' | 'multipleSelection' | 'emptyListIcon' 'titleLabel' | 'emptyListLabel' | 'multipleSelection'
>, >,
) => ( ) => (
<DriveSelector <DriveSelector
multipleSelection={true} multipleSelection={true}
titleLabel="Select target" titleLabel="Select target"
emptyListLabel="Plug a target drive" emptyListLabel="Plug a target drive"
emptyListIcon={<TgtSvg width="40px" />}
showWarnings={true} showWarnings={true}
selectedList={getSelectedDrives()} selectedList={getSelectedDrives()}
updateSelectedList={getSelectedDrives} updateSelectedList={getSelectedDrives}
@@ -70,7 +73,9 @@ export const TargetSelectorModal = (
/> />
); );
export const selectAllTargets = (modalTargets: DrivelistDrive[]) => { export const selectAllTargets = (
modalTargets: scanner.adapters.DrivelistDrive[],
) => {
const selectedDrivesFromState = getSelectedDrives(); const selectedDrivesFromState = getSelectedDrives();
const deselected = selectedDrivesFromState.filter( const deselected = selectedDrivesFromState.filter(
(drive) => (drive) =>
@@ -109,11 +114,13 @@ export const TargetSelector = ({
flashing, flashing,
}: TargetSelectorProps) => { }: TargetSelectorProps) => {
// TODO: inject these from redux-connector // TODO: inject these from redux-connector
const [{ driveListLabel, targets }, setStateSlice] = React.useState( const [
getDriveSelectionStateSlice(), { showDrivesButton, driveListLabel, targets },
setStateSlice,
] = React.useState(getDriveSelectionStateSlice());
const [showTargetSelectorModal, setShowTargetSelectorModal] = React.useState(
false,
); );
const [showTargetSelectorModal, setShowTargetSelectorModal] =
React.useState(false);
React.useEffect(() => { React.useEffect(() => {
return observe(() => { return observe(() => {
@@ -134,7 +141,7 @@ export const TargetSelector = ({
<TargetSelectorButton <TargetSelectorButton
disabled={disabled} disabled={disabled}
show={!hasDrive} show={!hasDrive && showDrivesButton}
tooltip={driveListLabel} tooltip={driveListLabel}
openDriveSelector={() => { openDriveSelector={() => {
setShowTargetSelectorModal(true); setShowTargetSelectorModal(true);
@@ -161,31 +168,11 @@ export const TargetSelector = ({
{showTargetSelectorModal && ( {showTargetSelectorModal && (
<TargetSelectorModal <TargetSelectorModal
write={true} cancel={() => setShowTargetSelectorModal(false)}
cancel={(originalList) => {
if (originalList.length) {
selectAllTargets(originalList);
} else {
deselectAllDrives();
}
setShowTargetSelectorModal(false);
}}
done={(modalTargets) => { done={(modalTargets) => {
if (modalTargets.length === 0) { selectAllTargets(modalTargets);
deselectAllDrives();
}
setShowTargetSelectorModal(false); setShowTargetSelectorModal(false);
}} }}
onSelect={(drive) => {
if (
getSelectedDrives().find(
(selectedDrive) => selectedDrive.device === drive.device,
)
) {
return deselectDrive(drive.device);
}
selectDrive(drive.device);
}}
/> />
)} )}
</Flex> </Flex>

View File

@@ -0,0 +1,167 @@
import { uniqBy } from 'lodash';
import * as React from 'react';
import Checkbox from 'rendition/dist_esm5/components/Checkbox';
import { Flex } from 'rendition/dist_esm5/components/Flex';
import Input from 'rendition/dist_esm5/components/Input';
import Link from 'rendition/dist_esm5/components/Link';
import RadioButton from 'rendition/dist_esm5/components/RadioButton';
import Txt from 'rendition/dist_esm5/components/Txt';
import * as settings from '../../models/settings';
import { Modal, ScrollableFlex } from '../../styled-components';
import { openDialog } from '../../os/dialog';
import { startEllipsis } from '../../utils/start-ellipsis';
const RECENT_URL_IMAGES_KEY = 'recentUrlImages';
const SAVE_IMAGE_AFTER_FLASH_KEY = 'saveUrlImage';
const SAVE_IMAGE_AFTER_FLASH_PATH_KEY = 'saveUrlImageTo';
function normalizeRecentUrlImages(urls: any[]): URL[] {
if (!Array.isArray(urls)) {
urls = [];
}
urls = urls
.map((url) => {
try {
return new URL(url);
} catch (error) {
// Invalid URL, skip
}
})
.filter((url) => url !== undefined);
urls = uniqBy(urls, (url) => url.href);
return urls.slice(-5);
}
function getRecentUrlImages(): URL[] {
let urls = [];
try {
urls = JSON.parse(localStorage.getItem(RECENT_URL_IMAGES_KEY) || '[]');
} catch {
// noop
}
return normalizeRecentUrlImages(urls);
}
function setRecentUrlImages(urls: string[]) {
localStorage.setItem(RECENT_URL_IMAGES_KEY, JSON.stringify(urls));
}
export const URLSelector = ({
done,
cancel,
}: {
done: (imageURL: string) => void;
cancel: () => void;
}) => {
const [imageURL, setImageURL] = React.useState('');
const [recentImages, setRecentImages] = React.useState<URL[]>([]);
const [loading, setLoading] = React.useState(false);
const [saveImage, setSaveImage] = React.useState(false);
const [saveImagePath, setSaveImagePath] = React.useState('');
React.useEffect(() => {
const fetchRecentUrlImages = async () => {
const recentUrlImages: URL[] = await getRecentUrlImages();
setRecentImages(recentUrlImages);
};
const getSaveImageSettings = async () => {
const saveUrlImage: boolean = await settings.get(
SAVE_IMAGE_AFTER_FLASH_KEY,
);
const saveUrlImageToPath: string = await settings.get(
SAVE_IMAGE_AFTER_FLASH_PATH_KEY,
);
setSaveImage(saveUrlImage);
setSaveImagePath(saveUrlImageToPath);
};
fetchRecentUrlImages();
getSaveImageSettings();
}, []);
return (
<Modal
title="Use Image URL"
cancel={cancel}
primaryButtonProps={{
className: loading || !imageURL ? 'disabled' : '',
}}
done={async () => {
setLoading(true);
const urlStrings = recentImages
.map((url: URL) => url.href)
.concat(imageURL);
setRecentUrlImages(urlStrings);
await done(imageURL);
}}
>
<Flex flexDirection="column">
<Flex mb="16px" width="100%" height="auto" flexDirection="column">
<Input
value={imageURL}
placeholder="Enter a valid URL"
type="text"
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
setImageURL(evt.target.value)
}
/>
<Flex alignItems="flex-end">
<Checkbox
mt="16px"
checked={saveImage}
onChange={(evt) => {
const value = evt.target.checked;
setSaveImage(value);
settings
.set(SAVE_IMAGE_AFTER_FLASH_KEY, value)
.then(() => setSaveImage(value));
}}
label={<>Save file to:&nbsp;</>}
/>
<Link
disabled={!saveImage}
onClick={async () => {
if (saveImage) {
const folder = await openDialog('openDirectory');
if (folder) {
await settings.set(SAVE_IMAGE_AFTER_FLASH_PATH_KEY, folder);
setSaveImagePath(folder);
}
}
}}
>
{startEllipsis(saveImagePath, 20)}
</Link>
</Flex>
</Flex>
{recentImages.length > 0 && (
<Flex flexDirection="column" height="58%">
<Txt fontSize={18} mb="10px">
Recent
</Txt>
<ScrollableFlex flexDirection="column" p="0">
{recentImages
.map((recent, i) => (
<RadioButton
mb={i !== 0 ? '6px' : '0'}
key={recent.href}
checked={imageURL === recent.href}
label={`${recent.pathname.split('/').pop()} - ${
recent.href
}`}
onChange={() => {
setImageURL(recent.href);
}}
style={{
overflowWrap: 'break-word',
}}
/>
))
.reverse()}
</ScrollableFlex>
</Flex>
)}
</Flex>
</Modal>
);
};
export default URLSelector;

View File

@@ -1,12 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>balenaEtcher</title>
<link rel="stylesheet" type="text/css" href="index.css">
</head>
<body>
<main id="main"></main>
<script src="http://localhost:3030/gui.js"></script>
</body>
</html>

View File

@@ -2,7 +2,7 @@
<html> <html>
<head> <head>
<meta charset="UTF-8"> <meta charset="UTF-8">
<title>balenaEtcher</title> <title>Etcher</title>
<link rel="stylesheet" type="text/css" href="index.css"> <link rel="stylesheet" type="text/css" href="index.css">
</head> </head>
<body> <body>

View File

@@ -14,10 +14,8 @@
* limitations under the License. * limitations under the License.
*/ */
import * as electron from 'electron';
import * as sdk from 'etcher-sdk'; import * as sdk from 'etcher-sdk';
import * as _ from 'lodash'; import * as _ from 'lodash';
import { DrivelistDrive } from '../../../shared/drive-constraints';
import { bytesToMegabytes } from '../../../shared/units'; import { bytesToMegabytes } from '../../../shared/units';
import { Actions, store } from './store'; import { Actions, store } from './store';
@@ -47,8 +45,6 @@ export function isFlashing(): boolean {
* start a flash process. * start a flash process.
*/ */
export function setFlashingFlag() { export function setFlashingFlag() {
// see https://github.com/balenablocks/balena-electron-env/blob/4fce9c461f294d4a768db8f247eea6f75d7b08b0/README.md#remote-methods
electron.ipcRenderer.send('disable-screensaver');
store.dispatch({ store.dispatch({
type: Actions.SET_FLASHING_FLAG, type: Actions.SET_FLASHING_FLAG,
data: {}, data: {},
@@ -70,8 +66,6 @@ export function unsetFlashingFlag(results: {
type: Actions.UNSET_FLASHING_FLAG, type: Actions.UNSET_FLASHING_FLAG,
data: results, data: results,
}); });
// see https://github.com/balenablocks/balena-electron-env/blob/4fce9c461f294d4a768db8f247eea6f75d7b08b0/README.md#remote-methods
electron.ipcRenderer.send('enable-screensaver');
} }
export function setDevicePaths(devicePaths: string[]) { export function setDevicePaths(devicePaths: string[]) {
@@ -81,29 +75,25 @@ export function setDevicePaths(devicePaths: string[]) {
}); });
} }
export function addFailedDeviceError({ export function addFailedDevicePath({
device, device,
error, error,
}: { }: {
device: DrivelistDrive; device: sdk.scanner.adapters.DrivelistDrive;
error: Error; error: Error;
}) { }) {
const failedDeviceErrorsMap = new Map( const failedDevicePathsMap = new Map(
store.getState().toJS().failedDeviceErrors, store.getState().toJS().failedDevicePaths,
); );
if (failedDeviceErrorsMap.has(device.device)) { failedDevicePathsMap.set(device.device, {
// Only store the first error
return;
}
failedDeviceErrorsMap.set(device.device, {
description: device.description, description: device.description,
device: device.device, device: device.device,
devicePath: device.devicePath, devicePath: device.devicePath,
...error, ...error,
}); });
store.dispatch({ store.dispatch({
type: Actions.SET_FAILED_DEVICE_ERRORS, type: Actions.SET_FAILED_DEVICE_PATHS,
data: Array.from(failedDeviceErrorsMap), data: Array.from(failedDevicePathsMap),
}); });
} }

View File

@@ -15,19 +15,40 @@
*/ */
import * as _ from 'lodash'; import * as _ from 'lodash';
import { Animator, AnimationFunction, Color, RGBLed } from 'sys-class-rgb-led'; import { AnimationFunction, Color, RGBLed } from 'sys-class-rgb-led';
import { import {
DrivelistDrive,
isSourceDrive, isSourceDrive,
DrivelistDrive,
} from '../../../shared/drive-constraints'; } from '../../../shared/drive-constraints';
import { getDrives } from './available-drives';
import { getSelectedDrives } from './selection-state';
import * as settings from './settings'; import * as settings from './settings';
import { observe, store } from './store'; import { DEFAULT_STATE, observe } from './store';
const leds: Map<string, RGBLed> = new Map(); const leds: Map<string, RGBLed> = new Map();
const animator = new Animator([], 10);
function setLeds(
drivesPaths: Set<string>,
colorOrAnimation: Color | AnimationFunction,
frequency?: number,
) {
for (const path of drivesPaths) {
const led = leds.get(path);
if (led) {
if (Array.isArray(colorOrAnimation)) {
led.setStaticColor(colorOrAnimation);
} else {
led.setAnimation(colorOrAnimation, frequency);
}
}
}
}
const red: Color = [1, 0, 0];
const green: Color = [0, 1, 0];
const blue: Color = [0, 0, 1];
const white: Color = [1, 1, 1];
const black: Color = [0, 0, 0];
const purple: Color = [0.5, 0, 0.5];
function createAnimationFunction( function createAnimationFunction(
intensityFunction: (t: number) => number, intensityFunction: (t: number) => number,
@@ -35,39 +56,21 @@ function createAnimationFunction(
): AnimationFunction { ): AnimationFunction {
return (t: number): Color => { return (t: number): Color => {
const intensity = intensityFunction(t); const intensity = intensityFunction(t);
return color.map((v: number) => v * intensity) as Color; return color.map((v) => v * intensity) as Color;
}; };
} }
function blink(t: number) { function blink(t: number) {
return Math.floor(t) % 2; return Math.floor(t / 1000) % 2;
} }
function one(_t: number) { function breathe(t: number) {
return 1; return (1 + Math.sin(t / 1000)) / 2;
} }
type LEDColors = { const breatheBlue = createAnimationFunction(breathe, blue);
green: Color; const blinkGreen = createAnimationFunction(blink, green);
purple: Color; const blinkPurple = createAnimationFunction(blink, purple);
red: Color;
blue: Color;
white: Color;
black: Color;
};
type LEDAnimationFunctions = {
blinkGreen: AnimationFunction;
blinkPurple: AnimationFunction;
staticRed: AnimationFunction;
staticGreen: AnimationFunction;
staticBlue: AnimationFunction;
staticWhite: AnimationFunction;
staticBlack: AnimationFunction;
};
let ledColors: LEDColors;
let ledAnimationFunctions: LEDAnimationFunctions;
interface LedsState { interface LedsState {
step: 'main' | 'flashing' | 'verifying' | 'finish'; step: 'main' | 'flashing' | 'verifying' | 'finish';
@@ -77,17 +80,6 @@ interface LedsState {
failedDrives: string[]; failedDrives: string[];
} }
function setLeds(animation: AnimationFunction, drivesPaths: Set<string>) {
const rgbLeds: RGBLed[] = [];
for (const path of drivesPaths) {
const led = leds.get(path);
if (led) {
rgbLeds.push(led);
}
}
return { animation, rgbLeds };
}
// Source slot (1st slot): behaves as a target unless it is chosen as source // Source slot (1st slot): behaves as a target unless it is chosen as source
// No drive: black // No drive: black
// Drive plugged: blue - on // Drive plugged: blue - on
@@ -118,7 +110,6 @@ export function updateLeds({
// Remove selected devices from plugged set // Remove selected devices from plugged set
for (const d of selectedOk) { for (const d of selectedOk) {
plugged.delete(d); plugged.delete(d);
unplugged.delete(d);
} }
// Remove plugged devices from unplugged set // Remove plugged devices from unplugged set
@@ -131,90 +122,74 @@ export function updateLeds({
selectedOk.delete(d); selectedOk.delete(d);
} }
const mapping: Array<{
animation: AnimationFunction;
rgbLeds: RGBLed[];
}> = [];
// Handle source slot // Handle source slot
if (sourceDrive !== undefined) { if (sourceDrive !== undefined) {
if (plugged.has(sourceDrive)) { if (unplugged.has(sourceDrive)) {
unplugged.delete(sourceDrive);
// TODO
setLeds(new Set([sourceDrive]), breatheBlue, 2);
} else if (plugged.has(sourceDrive)) {
plugged.delete(sourceDrive); plugged.delete(sourceDrive);
mapping.push( setLeds(new Set([sourceDrive]), blue);
setLeds(ledAnimationFunctions.staticBlue, new Set([sourceDrive])),
);
} }
} }
if (step === 'main') { if (step === 'main') {
mapping.push( setLeds(unplugged, black);
setLeds( setLeds(plugged, black);
ledAnimationFunctions.staticBlack, setLeds(selectedOk, white);
new Set([...unplugged, ...plugged]), setLeds(selectedFailed, white);
),
setLeds(
ledAnimationFunctions.staticWhite,
new Set([...selectedOk, ...selectedFailed]),
),
);
} else if (step === 'flashing') { } else if (step === 'flashing') {
mapping.push( setLeds(unplugged, black);
setLeds( setLeds(plugged, black);
ledAnimationFunctions.staticBlack, setLeds(selectedOk, blinkPurple, 2);
new Set([...unplugged, ...plugged]), setLeds(selectedFailed, red);
),
setLeds(ledAnimationFunctions.blinkPurple, selectedOk),
setLeds(ledAnimationFunctions.staticRed, selectedFailed),
);
} else if (step === 'verifying') { } else if (step === 'verifying') {
mapping.push( setLeds(unplugged, black);
setLeds( setLeds(plugged, black);
ledAnimationFunctions.staticBlack, setLeds(selectedOk, blinkGreen, 2);
new Set([...unplugged, ...plugged]), setLeds(selectedFailed, red);
),
setLeds(ledAnimationFunctions.blinkGreen, selectedOk),
setLeds(ledAnimationFunctions.staticRed, selectedFailed),
);
} else if (step === 'finish') { } else if (step === 'finish') {
mapping.push( setLeds(unplugged, black);
setLeds( setLeds(plugged, black);
ledAnimationFunctions.staticBlack, setLeds(selectedOk, green);
new Set([...unplugged, ...plugged]), setLeds(selectedFailed, red);
),
setLeds(ledAnimationFunctions.staticGreen, selectedOk),
setLeds(ledAnimationFunctions.staticRed, selectedFailed),
);
} }
animator.mapping = mapping; }
interface DeviceFromState {
devicePath?: string;
device: string;
} }
let ledsState: LedsState | undefined; let ledsState: LedsState | undefined;
function stateObserver() { function stateObserver(state: typeof DEFAULT_STATE) {
const s = store.getState().toJS(); const s = state.toJS();
let step: 'main' | 'flashing' | 'verifying' | 'finish'; let step: 'main' | 'flashing' | 'verifying' | 'finish';
if (s.isFlashing) { if (s.isFlashing) {
step = s.flashState.type; step = s.flashState.type;
} else { } else {
step = s.lastAverageFlashingSpeed == null ? 'main' : 'finish'; step = s.lastAverageFlashingSpeed == null ? 'main' : 'finish';
} }
const availableDrives = getDrives().filter( const availableDrives = s.availableDrives.filter(
(d: DrivelistDrive) => d.devicePath, (d: DeviceFromState) => d.devicePath,
); );
const sourceDrivePath = availableDrives.filter((d: DrivelistDrive) => const sourceDrivePath = availableDrives.filter((d: DrivelistDrive) =>
isSourceDrive(d, s.selection.image), isSourceDrive(d, s.selection.image),
)[0]?.devicePath; )[0]?.devicePath;
const availableDrivesPaths = availableDrives.map( const availableDrivesPaths = availableDrives.map(
(d: DrivelistDrive) => d.devicePath, (d: DeviceFromState) => d.devicePath,
); );
let selectedDrivesPaths: string[]; let selectedDrivesPaths: string[];
if (step === 'main') { if (step === 'main') {
selectedDrivesPaths = getSelectedDrives() selectedDrivesPaths = availableDrives
.filter((drive) => drive.devicePath !== null) .filter((d: DrivelistDrive) => s.selection.devices.includes(d.device))
.map((drive) => drive.devicePath) as string[]; .map((d: DrivelistDrive) => d.devicePath);
} else { } else {
selectedDrivesPaths = s.devicePaths; selectedDrivesPaths = s.devicePaths;
} }
const failedDevicePaths = s.failedDeviceErrors.map( const failedDevicePaths = s.failedDevicePaths.map(
([, { devicePath }]: [string, { devicePath: string }]) => devicePath, ([devicePath]: [string]) => devicePath,
); );
const newLedsState = { const newLedsState = {
step, step,
@@ -222,7 +197,7 @@ function stateObserver() {
availableDrives: availableDrivesPaths, availableDrives: availableDrivesPaths,
selectedDrives: selectedDrivesPaths, selectedDrives: selectedDrivesPaths,
failedDrives: failedDevicePaths, failedDrives: failedDevicePaths,
} as LedsState; };
if (!_.isEqual(newLedsState, ledsState)) { if (!_.isEqual(newLedsState, ledsState)) {
updateLeds(newLedsState); updateLeds(newLedsState);
ledsState = newLedsState; ledsState = newLedsState;
@@ -245,16 +220,6 @@ export async function init(): Promise<void> {
for (const [drivePath, ledsNames] of Object.entries(ledsMapping)) { for (const [drivePath, ledsNames] of Object.entries(ledsMapping)) {
leds.set('/dev/disk/by-path/' + drivePath, new RGBLed(ledsNames)); leds.set('/dev/disk/by-path/' + drivePath, new RGBLed(ledsNames));
} }
ledColors = (await settings.get('ledColors')) || {};
ledAnimationFunctions = {
blinkGreen: createAnimationFunction(blink, ledColors['green']),
blinkPurple: createAnimationFunction(blink, ledColors['purple']),
staticRed: createAnimationFunction(one, ledColors['red']),
staticGreen: createAnimationFunction(one, ledColors['green']),
staticBlue: createAnimationFunction(one, ledColors['blue']),
staticWhite: createAnimationFunction(one, ledColors['white']),
staticBlack: createAnimationFunction(one, ledColors['black']),
};
observe(_.debounce(stateObserver, 1000, { maxWait: 1000 })); observe(_.debounce(stateObserver, 1000, { maxWait: 1000 }));
} }
} }

View File

@@ -72,6 +72,26 @@ export function getImage(): SourceMetadata | undefined {
return store.getState().toJS().selection.image; return store.getState().toJS().selection.image;
} }
export function getImagePath(): string | undefined {
return store.getState().toJS().selection.image?.path;
}
export function getImageSize(): number | undefined {
return store.getState().toJS().selection.image?.size;
}
export function getImageName(): string | undefined {
return store.getState().toJS().selection.image?.name;
}
export function getImageLogo(): string | undefined {
return store.getState().toJS().selection.image?.logo;
}
export function getImageSupportUrl(): string | undefined {
return store.getState().toJS().selection.image?.supportUrl;
}
/** /**
* @summary Check if there is a selected drive * @summary Check if there is a selected drive
*/ */

View File

@@ -38,20 +38,23 @@ export const DEFAULT_HEIGHT = 480;
* - `~/Library/Application Support/etcher` on macOS * - `~/Library/Application Support/etcher` on macOS
* See https://electronjs.org/docs/api/app#appgetpathname * See https://electronjs.org/docs/api/app#appgetpathname
* *
* NOTE: We use the remote property when this module * NOTE: The ternary is due to this module being loaded both,
* is loaded in the Electron's renderer process * Electron's main process and renderer process
*/ */
const app = electron.app || electron.remote.app; const app = electron.app || electron.remote.app;
const USER_DATA_DIR = app.getPath('userData'); const USER_DATA_DIR = app.getPath('userData');
const CONFIG_PATH = join(USER_DATA_DIR, 'config.json'); const CONFIG_PATH = join(USER_DATA_DIR, 'config.json');
const DOWNLOADS_DIR = app.getPath('downloads');
async function readConfigFile(filename: string): Promise<_.Dictionary<any>> { async function readConfigFile(filename: string): Promise<_.Dictionary<any>> {
let contents = '{}'; let contents = '{}';
try { try {
contents = await fs.readFile(filename, { encoding: 'utf8' }); contents = await fs.readFile(filename, { encoding: 'utf8' });
} catch (error: any) { } catch (error) {
// noop // noop
} }
try { try {
@@ -77,10 +80,14 @@ export async function writeConfigFile(
const DEFAULT_SETTINGS: _.Dictionary<any> = { const DEFAULT_SETTINGS: _.Dictionary<any> = {
errorReporting: true, errorReporting: true,
updatesEnabled: ['appimage', 'nsis', 'dmg'].includes(packageJSON.packageType), unmountOnSuccess: true,
validateWriteOnSuccess: true,
updatesEnabled: !_.includes(['rpm', 'deb'], packageJSON.packageType),
desktopNotifications: true, desktopNotifications: true,
autoBlockmapping: true, autoBlockmapping: true,
decompressFirst: true, decompressFirst: true,
saveUrlImage: false,
saveUrlImageTo: DOWNLOADS_DIR,
}; };
const settings = _.cloneDeep(DEFAULT_SETTINGS); const settings = _.cloneDeep(DEFAULT_SETTINGS);
@@ -104,7 +111,7 @@ export async function set(
settings[key] = value; settings[key] = value;
try { try {
await writeConfigFileFn(CONFIG_PATH, settings); await writeConfigFileFn(CONFIG_PATH, settings);
} catch (error: any) { } catch (error) {
// Revert to previous value if persisting settings failed // Revert to previous value if persisting settings failed
settings[key] = previousValue; settings[key] = previousValue;
throw error; throw error;

View File

@@ -16,7 +16,6 @@
import * as Immutable from 'immutable'; import * as Immutable from 'immutable';
import * as _ from 'lodash'; import * as _ from 'lodash';
import { basename } from 'path';
import * as redux from 'redux'; import * as redux from 'redux';
import { v4 as uuidV4 } from 'uuid'; import { v4 as uuidV4 } from 'uuid';
@@ -63,7 +62,7 @@ export const DEFAULT_STATE = Immutable.fromJS({
}, },
isFlashing: false, isFlashing: false,
devicePaths: [], devicePaths: [],
failedDeviceErrors: [], failedDevicePaths: [],
flashResults: {}, flashResults: {},
flashState: { flashState: {
active: 0, active: 0,
@@ -80,7 +79,7 @@ export const DEFAULT_STATE = Immutable.fromJS({
*/ */
export enum Actions { export enum Actions {
SET_DEVICE_PATHS, SET_DEVICE_PATHS,
SET_FAILED_DEVICE_ERRORS, SET_FAILED_DEVICE_PATHS,
SET_AVAILABLE_TARGETS, SET_AVAILABLE_TARGETS,
SET_FLASH_STATE, SET_FLASH_STATE,
RESET_FLASH_STATE, RESET_FLASH_STATE,
@@ -134,16 +133,11 @@ function storeReducer(
}); });
} }
// Drives order is a list of devicePaths
const drivesOrder = settings.getSync('drivesOrder') ?? [];
drives = _.sortBy(drives, [ drives = _.sortBy(drives, [
// System drives last // System drives last
(d) => !!d.isSystem, (d) => !!d.isSystem,
// Devices with no devicePath first (usbboot) // Devices with no devicePath first (usbboot)
(d) => !!d.devicePath, (d) => !!d.devicePath,
// Sort as defined in the drivesOrder setting if there is one (only for Linux with udev)
(d) => drivesOrder.indexOf(basename(d.devicePath || '')),
// Then sort by devicePath (only available on Linux with udev) or device // Then sort by devicePath (only available on Linux with udev) or device
(d) => d.devicePath || d.device, (d) => d.devicePath || d.device,
]); ]);
@@ -175,7 +169,7 @@ function storeReducer(
); );
const shouldAutoselectAll = Boolean( const shouldAutoselectAll = Boolean(
settings.getSync('autoSelectAllDrives'), settings.getSync('disableExplicitDriveSelection'),
); );
const AUTOSELECT_DRIVE_COUNT = 1; const AUTOSELECT_DRIVE_COUNT = 1;
const nonStaleSelectedDevices = nonStaleNewState const nonStaleSelectedDevices = nonStaleNewState
@@ -197,13 +191,18 @@ function storeReducer(
drives, drives,
(accState, drive) => { (accState, drive) => {
if ( if (
constraints.isDriveValid(drive, image) && _.every([
!drive.isReadOnly && constraints.isDriveValid(drive, image),
constraints.isDriveSizeRecommended(drive, image) && constraints.isDriveSizeRecommended(drive, image),
// We don't want to auto-select large drives execpt is autoSelectAllDrives is true
(!constraints.isDriveSizeLarge(drive) || shouldAutoselectAll) && // We don't want to auto-select large drives
// We don't want to auto-select system drives !constraints.isDriveSizeLarge(drive),
!constraints.isSystemDrive(drive)
// We don't want to auto-select system drives,
// even when "unsafe mode" is enabled
!constraints.isSystemDrive(drive),
]) ||
(shouldAutoselectAll && constraints.isDriveValid(drive, image))
) { ) {
// Auto-select this drive // Auto-select this drive
return storeReducer(accState, { return storeReducer(accState, {
@@ -270,7 +269,7 @@ function storeReducer(
.set('flashState', DEFAULT_STATE.get('flashState')) .set('flashState', DEFAULT_STATE.get('flashState'))
.set('flashResults', DEFAULT_STATE.get('flashResults')) .set('flashResults', DEFAULT_STATE.get('flashResults'))
.set('devicePaths', DEFAULT_STATE.get('devicePaths')) .set('devicePaths', DEFAULT_STATE.get('devicePaths'))
.set('failedDeviceErrors', DEFAULT_STATE.get('failedDeviceErrors')) .set('failedDevicePaths', DEFAULT_STATE.get('failedDevicePaths'))
.set( .set(
'lastAverageFlashingSpeed', 'lastAverageFlashingSpeed',
DEFAULT_STATE.get('lastAverageFlashingSpeed'), DEFAULT_STATE.get('lastAverageFlashingSpeed'),
@@ -517,8 +516,8 @@ function storeReducer(
return state.set('devicePaths', action.data); return state.set('devicePaths', action.data);
} }
case Actions.SET_FAILED_DEVICE_ERRORS: { case Actions.SET_FAILED_DEVICE_PATHS: {
return state.set('failedDeviceErrors', action.data); return state.set('failedDevicePaths', action.data);
} }
default: { default: {

View File

@@ -102,9 +102,10 @@ function validateMixpanelConfig(config: {
* This function sends the debug message to product analytics services. * This function sends the debug message to product analytics services.
*/ */
export function logEvent(message: string, data: _.Dictionary<any> = {}) { export function logEvent(message: string, data: _.Dictionary<any> = {}) {
const { applicationSessionUuid, flashingWorkflowUuid } = store const {
.getState() applicationSessionUuid,
.toJS(); flashingWorkflowUuid,
} = store.getState().toJS();
resinCorvus.logEvent(message, { resinCorvus.logEvent(message, {
...data, ...data,
sample: mixpanelSample, sample: mixpanelSample,

View File

@@ -15,15 +15,10 @@
*/ */
import * as sdk from 'etcher-sdk'; import * as sdk from 'etcher-sdk';
import {
Adapter,
BlockDeviceAdapter,
UsbbootDeviceAdapter,
} from 'etcher-sdk/build/scanner/adapters';
import { geteuid, platform } from 'process'; import { geteuid, platform } from 'process';
const adapters: Adapter[] = [ const adapters: sdk.scanner.adapters.Adapter[] = [
new BlockDeviceAdapter({ new sdk.scanner.adapters.BlockDeviceAdapter({
includeSystemDrives: () => true, includeSystemDrives: () => true,
}), }),
]; ];
@@ -31,15 +26,14 @@ const adapters: Adapter[] = [
// Can't use permissions.isElevated() here as it returns a promise and we need to set // Can't use permissions.isElevated() here as it returns a promise and we need to set
// module.exports = scanner right now. // module.exports = scanner right now.
if (platform !== 'linux' || geteuid() === 0) { if (platform !== 'linux' || geteuid() === 0) {
adapters.push(new UsbbootDeviceAdapter()); adapters.push(new sdk.scanner.adapters.UsbbootDeviceAdapter());
} }
if (platform === 'win32') { if (
const { platform === 'win32' &&
DriverlessDeviceAdapter: driverless, sdk.scanner.adapters.DriverlessDeviceAdapter !== undefined
// tslint:disable-next-line:no-var-requires ) {
} = require('etcher-sdk/build/scanner/adapters/driverless'); adapters.push(new sdk.scanner.adapters.DriverlessDeviceAdapter());
adapters.push(new driverless());
} }
export const scanner = new sdk.scanner.Scanner(adapters); export const scanner = new sdk.scanner.Scanner(adapters);

View File

@@ -15,8 +15,9 @@
*/ */
import { Drive as DrivelistDrive } from 'drivelist'; import { Drive as DrivelistDrive } from 'drivelist';
import * as electron from 'electron';
import * as sdk from 'etcher-sdk'; import * as sdk from 'etcher-sdk';
import { Dictionary } from 'lodash'; import * as _ from 'lodash';
import * as ipc from 'node-ipc'; import * as ipc from 'node-ipc';
import * as os from 'os'; import * as os from 'os';
import * as path from 'path'; import * as path from 'path';
@@ -24,7 +25,6 @@ import * as path from 'path';
import * as packageJSON from '../../../../package.json'; import * as packageJSON from '../../../../package.json';
import * as errors from '../../../shared/errors'; import * as errors from '../../../shared/errors';
import * as permissions from '../../../shared/permissions'; import * as permissions from '../../../shared/permissions';
import { getAppPath } from '../../../shared/utils';
import { SourceMetadata } from '../components/source-selector/source-selector'; import { SourceMetadata } from '../components/source-selector/source-selector';
import * as flashState from '../models/flash-state'; import * as flashState from '../models/flash-state';
import * as selectionState from '../models/selection-state'; import * as selectionState from '../models/selection-state';
@@ -93,7 +93,11 @@ function terminateServer() {
} }
function writerArgv(): string[] { function writerArgv(): string[] {
let entryPoint = path.join(getAppPath(), 'generated', 'child-writer.js'); let entryPoint = path.join(
electron.remote.app.getAppPath(),
'generated',
'child-writer.js',
);
// AppImages run over FUSE, so the files inside the mount point // AppImages run over FUSE, so the files inside the mount point
// can only be accessed by the user that mounted the AppImage. // can only be accessed by the user that mounted the AppImage.
// This means we can't re-spawn Etcher as root from the same // This means we can't re-spawn Etcher as root from the same
@@ -129,14 +133,6 @@ function writerEnv() {
interface FlashResults { interface FlashResults {
skip?: boolean; skip?: boolean;
cancelled?: boolean; cancelled?: boolean;
results?: {
bytesWritten: number;
devices: {
failed: number;
successful: number;
};
errors: Error[];
};
} }
async function performWrite( async function performWrite(
@@ -147,7 +143,14 @@ async function performWrite(
let cancelled = false; let cancelled = false;
let skip = false; let skip = false;
ipc.serve(); ipc.serve();
const { autoBlockmapping, decompressFirst } = await settings.getAll(); const {
unmountOnSuccess,
validateWriteOnSuccess,
autoBlockmapping,
decompressFirst,
saveUrlImage,
saveUrlImageTo,
} = await settings.getAll();
return await new Promise((resolve, reject) => { return await new Promise((resolve, reject) => {
ipc.server.on('error', (error) => { ipc.server.on('error', (error) => {
terminateServer(); terminateServer();
@@ -166,22 +169,22 @@ async function performWrite(
driveCount: drives.length, driveCount: drives.length,
uuid: flashState.getFlashUuid(), uuid: flashState.getFlashUuid(),
flashInstanceUuid: flashState.getFlashUuid(), flashInstanceUuid: flashState.getFlashUuid(),
unmountOnSuccess,
validateWriteOnSuccess,
}; };
ipc.server.on('fail', ({ device, error }) => { ipc.server.on('fail', ({ device, error }) => {
if (device.devicePath) { if (device.devicePath) {
flashState.addFailedDeviceError({ device, error }); flashState.addFailedDevicePath({ device, error });
} }
handleErrorLogging(error, analyticsData); handleErrorLogging(error, analyticsData);
}); });
ipc.server.on('done', (event) => { ipc.server.on('done', (event) => {
event.results.errors = event.results.errors.map( event.results.errors = _.map(event.results.errors, (data) => {
(data: Dictionary<any> & { message: string }) => {
return errors.fromJSON(data); return errors.fromJSON(data);
}, });
); _.merge(flashResults, event);
flashResults.results = event.results;
}); });
ipc.server.on('abort', () => { ipc.server.on('abort', () => {
@@ -201,15 +204,19 @@ async function performWrite(
image, image,
destinations: drives, destinations: drives,
SourceType: image.SourceType.name, SourceType: image.SourceType.name,
validateWriteOnSuccess,
autoBlockmapping, autoBlockmapping,
unmountOnSuccess,
decompressFirst, decompressFirst,
saveUrlImage,
saveUrlImageTo,
}); });
}); });
const argv = writerArgv(); const argv = writerArgv();
ipc.server.on('start', async () => { ipc.server.on('start', async () => {
console.log(`Elevating command: ${argv.join(' ')}`); console.log(`Elevating command: ${_.join(argv, ' ')}`);
const env = writerEnv(); const env = writerEnv();
try { try {
const results = await permissions.elevateCommand(argv, { const results = await permissions.elevateCommand(argv, {
@@ -218,7 +225,7 @@ async function performWrite(
}); });
flashResults.cancelled = cancelled || results.cancelled; flashResults.cancelled = cancelled || results.cancelled;
flashResults.skip = skip; flashResults.skip = skip;
} catch (error: any) { } catch (error) {
// This happens when the child is killed using SIGKILL // This happens when the child is killed using SIGKILL
const SIGKILL_EXIT_CODE = 137; const SIGKILL_EXIT_CODE = 137;
if (error.code === SIGKILL_EXIT_CODE) { if (error.code === SIGKILL_EXIT_CODE) {
@@ -231,11 +238,11 @@ async function performWrite(
} }
console.log('Flash results', flashResults); console.log('Flash results', flashResults);
// The flash wasn't cancelled and we didn't get a 'done' event // This likely means the child died halfway through
if ( if (
!flashResults.cancelled && !flashResults.cancelled &&
!flashResults.skip && !flashResults.skip &&
flashResults.results === undefined !_.get(flashResults, ['results', 'bytesWritten'])
) { ) {
reject( reject(
errors.createUserError({ errors.createUserError({
@@ -268,7 +275,7 @@ export async function flash(
throw new Error('There is already a flash in progress'); throw new Error('There is already a flash in progress');
} }
await flashState.setFlashingFlag(); flashState.setFlashingFlag();
flashState.setDevicePaths( flashState.setDevicePaths(
drives.map((d) => d.devicePath).filter((p) => p != null) as string[], drives.map((d) => d.devicePath).filter((p) => p != null) as string[],
); );
@@ -280,18 +287,17 @@ export async function flash(
uuid: flashState.getFlashUuid(), uuid: flashState.getFlashUuid(),
status: 'started', status: 'started',
flashInstanceUuid: flashState.getFlashUuid(), flashInstanceUuid: flashState.getFlashUuid(),
unmountOnSuccess: await settings.get('unmountOnSuccess'),
validateWriteOnSuccess: await settings.get('validateWriteOnSuccess'),
}; };
analytics.logEvent('Flash', analyticsData); analytics.logEvent('Flash', analyticsData);
try { try {
const result = await write(image, drives, flashState.setProgressState); const result = await write(image, drives, flashState.setProgressState);
await flashState.unsetFlashingFlag(result); flashState.unsetFlashingFlag(result);
} catch (error: any) { } catch (error) {
await flashState.unsetFlashingFlag({ flashState.unsetFlashingFlag({ cancelled: false, errorCode: error.code });
cancelled: false,
errorCode: error.code,
});
windowProgress.clear(); windowProgress.clear();
const { results = {} } = flashState.getFlashResults(); const { results = {} } = flashState.getFlashResults();
const eventData = { const eventData = {
@@ -332,11 +338,13 @@ export async function cancel(type: string) {
const status = type.toLowerCase(); const status = type.toLowerCase();
const drives = selectionState.getSelectedDevices(); const drives = selectionState.getSelectedDevices();
const analyticsData = { const analyticsData = {
image: selectionState.getImage()?.path, image: selectionState.getImagePath(),
drives, drives,
driveCount: drives.length, driveCount: drives.length,
uuid: flashState.getFlashUuid(), uuid: flashState.getFlashUuid(),
flashInstanceUuid: flashState.getFlashUuid(), flashInstanceUuid: flashState.getFlashUuid(),
unmountOnSuccess: await settings.get('unmountOnSuccess'),
validateWriteOnSuccess: await settings.get('validateWriteOnSuccess'),
status, status,
}; };
analytics.logEvent('Cancel', analyticsData); analytics.logEvent('Cancel', analyticsData);
@@ -349,7 +357,7 @@ export async function cancel(type: string) {
if (socket !== undefined) { if (socket !== undefined) {
ipc.server.emit(socket, status); ipc.server.emit(socket, status);
} }
} catch (error: any) { } catch (error) {
analytics.logException(error); analytics.logException(error);
} }
} }

View File

@@ -22,7 +22,7 @@ export interface FlashState {
percentage?: number; percentage?: number;
speed: number; speed: number;
position: number; position: number;
type?: 'decompressing' | 'flashing' | 'verifying'; type?: 'decompressing' | 'flashing' | 'verifying' | 'downloading';
} }
export function fromFlashState({ export function fromFlashState({
@@ -62,6 +62,12 @@ export function fromFlashState({
} else { } else {
return { status: 'Finishing...' }; return { status: 'Finishing...' };
} }
} else if (type === 'downloading') {
if (percentage == null) {
return { status: 'Downloading...' };
} else if (percentage < 100) {
return { position: `${percentage}%`, status: 'Downloading...' };
}
} }
return { status: 'Failed' }; return { status: 'Failed' };
} }

View File

@@ -27,7 +27,7 @@ async function mountSourceDrive() {
if (sourceDrivePath) { if (sourceDrivePath) {
try { try {
await electron.ipcRenderer.invoke('mount-drive', sourceDrivePath); await electron.ipcRenderer.invoke('mount-drive', sourceDrivePath);
} catch (error: any) { } catch (error) {
// noop // noop
} }
} }
@@ -40,6 +40,12 @@ async function mountSourceDrive() {
* Notice that by image, we mean *.img/*.iso/*.zip/etc files. * Notice that by image, we mean *.img/*.iso/*.zip/etc files.
*/ */
export async function selectImage(): Promise<string | undefined> { export async function selectImage(): Promise<string | undefined> {
return await openDialog();
}
export async function openDialog(
type: 'openFile' | 'openDirectory' = 'openFile',
) {
await mountSourceDrive(); await mountSourceDrive();
const options: electron.OpenDialogOptions = { const options: electron.OpenDialogOptions = {
// This variable is set when running in GNU/Linux from // This variable is set when running in GNU/Linux from
@@ -50,8 +56,10 @@ export async function selectImage(): Promise<string | undefined> {
// //
// See: https://github.com/probonopd/AppImageKit/commit/1569d6f8540aa6c2c618dbdb5d6fcbf0003952b7 // See: https://github.com/probonopd/AppImageKit/commit/1569d6f8540aa6c2c618dbdb5d6fcbf0003952b7
defaultPath: process.env.OWD, defaultPath: process.env.OWD,
properties: ['openFile', 'treatPackageAsDirectory'], properties: [type, 'treatPackageAsDirectory'],
filters: [ filters:
type === 'openFile'
? [
{ {
name: 'OS Images', name: 'OS Images',
extensions: SUPPORTED_EXTENSIONS, extensions: SUPPORTED_EXTENSIONS,
@@ -60,13 +68,14 @@ export async function selectImage(): Promise<string | undefined> {
name: 'All', name: 'All',
extensions: ['*'], extensions: ['*'],
}, },
], ]
: undefined,
}; };
const currentWindow = electron.remote.getCurrentWindow(); const currentWindow = electron.remote.getCurrentWindow();
const [file] = ( const [path] = (
await electron.remote.dialog.showOpenDialog(currentWindow, options) await electron.remote.dialog.showOpenDialog(currentWindow, options)
).filePaths; ).filePaths;
return file; return path;
} }
/** /**

View File

@@ -15,7 +15,6 @@
*/ */
import { exec } from 'child_process'; import { exec } from 'child_process';
import { withTmpFile } from 'etcher-sdk/build/tmp';
import { readFile } from 'fs'; import { readFile } from 'fs';
import { chain, trim } from 'lodash'; import { chain, trim } from 'lodash';
import { platform } from 'os'; import { platform } from 'os';
@@ -23,6 +22,8 @@ import { join } from 'path';
import { env } from 'process'; import { env } from 'process';
import { promisify } from 'util'; import { promisify } from 'util';
import { withTmpFile } from '../../../shared/tmp';
const readFileAsync = promisify(readFile); const readFileAsync = promisify(readFile);
const execAsync = promisify(exec); const execAsync = promisify(exec);
@@ -40,11 +41,11 @@ async function getWmicNetworkDrivesOutput(): Promise<string> {
// So we just redirect to a file and read it afterwards as we know it will be ucs2 encoded. // So we just redirect to a file and read it afterwards as we know it will be ucs2 encoded.
const options = { const options = {
// Close the file once it's created // Close the file once it's created
keepOpen: false, discardDescriptor: true,
// Wmic fails with "Invalid global switch" when the "/output:" switch filename contains a dash ("-") // Wmic fails with "Invalid global switch" when the "/output:" switch filename contains a dash ("-")
prefix: 'tmp', prefix: 'tmp',
}; };
return withTmpFile(options, async ({ path }) => { return withTmpFile(options, async (path) => {
const command = [ const command = [
join(env.SystemRoot as string, 'System32', 'Wbem', 'wmic'), join(env.SystemRoot as string, 'System32', 'Wbem', 'wmic'),
'path', 'path',

View File

@@ -59,27 +59,6 @@ const getErrorMessageFromCode = (errorCode: string) => {
return ''; return '';
}; };
function notifySuccess(
iconPath: string,
basename: string,
drives: any,
devices: { successful: number; failed: number },
) {
notification.send(
'Flash complete!',
messages.info.flashComplete(basename, drives, devices),
iconPath,
);
}
function notifyFailure(iconPath: string, basename: string, drives: any) {
notification.send(
'Oops! Looks like the flash failed.',
messages.error.flashFailure(basename, drives),
iconPath,
);
}
async function flashImageToDrive( async function flashImageToDrive(
isFlashing: boolean, isFlashing: boolean,
goToSuccess: () => void, goToSuccess: () => void,
@@ -105,20 +84,20 @@ async function flashImageToDrive(
if (!flashState.wasLastFlashCancelled()) { if (!flashState.wasLastFlashCancelled()) {
const { const {
results = { devices: { successful: 0, failed: 0 } }, results = { devices: { successful: 0, failed: 0 } },
skip,
cancelled,
} = flashState.getFlashResults(); } = flashState.getFlashResults();
if (!skip && !cancelled) { notification.send(
if (results.devices.successful > 0) { 'Flash complete!',
notifySuccess(iconPath, basename, drives, results.devices); messages.info.flashComplete(basename, drives as any, results.devices),
} else { iconPath,
notifyFailure(iconPath, basename, drives); );
}
}
goToSuccess(); goToSuccess();
} }
} catch (error: any) { } catch (error) {
notifyFailure(iconPath, basename, drives); notification.send(
'Oops! Looks like the flash failed.',
messages.error.flashFailure(path.basename(image.path), drives),
iconPath,
);
let errorMessage = getErrorMessageFromCode(error.code); let errorMessage = getErrorMessageFromCode(error.code);
if (!errorMessage) { if (!errorMessage) {
error.image = basename; error.image = basename;
@@ -156,7 +135,6 @@ interface FlashStepProps {
failed: number; failed: number;
speed?: number; speed?: number;
eta?: number; eta?: number;
width: string;
} }
export interface DriveWithWarnings extends constraints.DrivelistDrive { export interface DriveWithWarnings extends constraints.DrivelistDrive {
@@ -221,11 +199,7 @@ export class FlashStep extends React.PureComponent<
const drives = selection.getSelectedDrives().map((drive) => { const drives = selection.getSelectedDrives().map((drive) => {
return { return {
...drive, ...drive,
statuses: constraints.getDriveImageCompatibilityStatuses( statuses: constraints.getDriveImageCompatibilityStatuses(drive),
drive,
undefined,
true,
),
}; };
}); });
if (drives.length === 0 || this.props.isFlashing) { if (drives.length === 0 || this.props.isFlashing) {
@@ -263,7 +237,6 @@ export class FlashStep extends React.PureComponent<
<Flex <Flex
flexDirection="column" flexDirection="column"
alignItems="start" alignItems="start"
width={this.props.width}
style={this.props.style} style={this.props.style}
> >
<FlashSvg <FlashSvg
@@ -335,7 +308,6 @@ export class FlashStep extends React.PureComponent<
)} )}
{this.state.showDriveSelectorModal && ( {this.state.showDriveSelectorModal && (
<TargetSelectorModal <TargetSelectorModal
write={true}
cancel={() => this.setState({ showDriveSelectorModal: false })} cancel={() => this.setState({ showDriveSelectorModal: false })}
done={(modalTargets) => { done={(modalTargets) => {
selectAllTargets(modalTargets); selectAllTargets(modalTargets);

View File

@@ -48,6 +48,7 @@ import { FlashStep } from './Flash';
import EtcherSvg from '../../../assets/etcher.svg'; import EtcherSvg from '../../../assets/etcher.svg';
import { SafeWebview } from '../../components/safe-webview/safe-webview'; import { SafeWebview } from '../../components/safe-webview/safe-webview';
import { colors } from '../../theme';
const Icon = styled(BaseIcon)` const Icon = styled(BaseIcon)`
margin-right: 20px; margin-right: 20px;
@@ -87,9 +88,7 @@ const StepBorder = styled.div<{
position: relative; position: relative;
height: 2px; height: 2px;
background-color: ${(props) => background-color: ${(props) =>
props.disabled props.disabled ? colors.dark.disabled.foreground : colors.dark.foreground};
? props.theme.colors.dark.disabled.foreground
: props.theme.colors.dark.foreground};
width: 120px; width: 120px;
top: 19px; top: 19px;
@@ -132,13 +131,12 @@ export class MainPage extends React.Component<
} }
private stateHelper(): MainPageStateFromStore { private stateHelper(): MainPageStateFromStore {
const image = selectionState.getImage();
return { return {
isFlashing: flashState.isFlashing(), isFlashing: flashState.isFlashing(),
hasImage: selectionState.hasImage(), hasImage: selectionState.hasImage(),
hasDrive: selectionState.hasDrive(), hasDrive: selectionState.hasDrive(),
imageLogo: image?.logo, imageLogo: selectionState.getImageLogo(),
imageSize: image?.size, imageSize: selectionState.getImageSize(),
imageName: getImageBasename(selectionState.getImage()), imageName: getImageBasename(selectionState.getImage()),
driveTitle: getDrivesTitle(), driveTitle: getDrivesTitle(),
driveLabel: getDriveListLabel(), driveLabel: getDriveListLabel(),
@@ -239,7 +237,6 @@ export class MainPage extends React.Component<
)} )}
<FlashStep <FlashStep
width={this.state.isWebviewShowing ? '220px' : '200px'}
goToSuccess={() => this.setState({ current: 'success' })} goToSuccess={() => this.setState({ current: 'success' })}
shouldFlashStepBeDisabled={shouldFlashStepBeDisabled} shouldFlashStepBeDisabled={shouldFlashStepBeDisabled}
isFlashing={this.state.isFlashing} isFlashing={this.state.isFlashing}
@@ -276,9 +273,9 @@ export class MainPage extends React.Component<
style={{ style={{
// Allow window to be dragged from header // Allow window to be dragged from header
// @ts-ignore // @ts-ignore
WebkitAppRegion: 'drag', '-webkit-app-region': 'drag',
position: 'relative', position: 'relative',
zIndex: 2, zIndex: 1,
}} }}
> >
<Flex width="100%" /> <Flex width="100%" />
@@ -304,7 +301,7 @@ export class MainPage extends React.Component<
onClick={() => this.setState({ hideSettings: false })} onClick={() => this.setState({ hideSettings: false })}
style={{ style={{
// Make touch events click instead of dragging // Make touch events click instead of dragging
WebkitAppRegion: 'no-drag', '-webkit-app-region': 'no-drag',
}} }}
/> />
{!settings.getSync('disableExternalLinks') && ( {!settings.getSync('disableExternalLinks') && (
@@ -312,14 +309,14 @@ export class MainPage extends React.Component<
icon={<QuestionCircleSvg height="1em" fill="currentColor" />} icon={<QuestionCircleSvg height="1em" fill="currentColor" />}
onClick={() => onClick={() =>
openExternal( openExternal(
selectionState.getImage()?.supportUrl || selectionState.getImageSupportUrl() ||
'https://github.com/balena-io/etcher/blob/master/SUPPORT.md', 'https://github.com/balena-io/etcher/blob/master/SUPPORT.md',
) )
} }
tabIndex={6} tabIndex={6}
style={{ style={{
// Make touch events click instead of dragging // Make touch events click instead of dragging
WebkitAppRegion: 'no-drag', '-webkit-app-region': 'no-drag',
}} }}
/> />
)} )}

View File

@@ -1,10 +0,0 @@
// @ts-nocheck
import { main } from './app';
if (module.hot) {
module.hot.accept('./app', () => {
main();
});
}
main();

View File

@@ -126,6 +126,22 @@ const modalFooterShadowCss = css`
export const Modal = styled(({ style, children, ...props }) => { export const Modal = styled(({ style, children, ...props }) => {
return ( return (
<Provider
theme={_.merge({}, theme, {
header: {
height: '50px',
},
layer: {
extend: () => `
${theme.layer.extend()}
> div:last-child {
top: 0;
}
`,
},
})}
>
<ModalBase <ModalBase
position="top" position="top"
width="97vw" width="97vw"
@@ -145,11 +161,12 @@ export const Modal = styled(({ style, children, ...props }) => {
{...children} {...children}
</ScrollableFlex> </ScrollableFlex>
</ModalBase> </ModalBase>
</Provider>
); );
})` })`
> div { > div {
padding: 0; padding: 0;
height: 99%; height: 100%;
> div:first-child { > div:first-child {
height: 81%; height: 81%;

View File

@@ -71,11 +71,7 @@ export const colors = {
const font = 'SourceSansPro'; const font = 'SourceSansPro';
export const theme = _.merge({}, Theme, { export const theme = _.merge({}, Theme, {
colors,
font, font,
header: {
height: '40px',
},
global: { global: {
font: { font: {
family: font, family: font,
@@ -100,7 +96,6 @@ export const theme = _.merge({}, Theme, {
font-size: 16px; font-size: 16px;
&& { && {
width: 200px;
height: 48px; height: 48px;
} }

View File

@@ -0,0 +1,28 @@
/*
* Copyright 2020 balena.io
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* @summary Truncate text from the start with an ellipsis
*/
export function startEllipsis(input: string, limit: number): string {
// Do nothing, the string doesn't need truncation.
if (input.length <= limit) {
return input;
}
const lastPart = input.slice(input.length - limit, input.length);
return `${lastPart}`;
}

View File

@@ -1,18 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<svg version="1.1" viewBox="0 0 39 90" xmlns="http://www.w3.org/2000/svg">
<g fill="none" fill-rule="evenodd">
<g transform="translate(-380 -166)">
<g transform="translate(380 166)">
<path d="m30.88 39.87h-23.363v23.209c0 0.6909 0.56062 1.251 1.251 1.251h20.861c0.69114 0 1.251-0.55986 1.251-1.251v-23.209zm-22.363 0.9999h21.363l4e-4 22.209c0 0.13886-0.11214 0.251-0.251 0.251h-20.861l-0.057452-0.0066403c-0.11075-0.026055-0.19355-0.12572-0.19355-0.24436l-4e-4 -22.209z" fill="#2A506F" fill-rule="nonzero"/>
<path d="m16.558 48.924h-3.967c-0.58314 0-1.055 0.47186-1.055 1.055v2.732c0 0.58235 0.47206 1.055 1.055 1.055h3.967c0.58223 0 1.054-0.47295 1.054-1.055v-2.732c0-0.58285-0.47156-1.055-1.054-1.055zm-3.967 1h3.967c0.029872 0 0.054 0.024158 0.054 0.055v2.732c0 0.030327-0.024612 0.055-0.054 0.055h-3.967c-0.030373 0-0.055-0.024658-0.055-0.055v-2.732c0-0.030858 0.024142-0.055 0.055-0.055z" fill="#2A506F" fill-rule="nonzero"/>
<path d="m25.97 48.924h-3.967c-0.58314 0-1.055 0.47186-1.055 1.055v2.732c0 0.58235 0.47206 1.055 1.055 1.055h3.967c0.58223 0 1.054-0.47295 1.054-1.055v-2.732c0-0.58285-0.47156-1.055-1.054-1.055zm-3.967 1h3.967c0.029872 0 0.054 0.024158 0.054 0.055v2.732c0 0.030327-0.024612 0.055-0.054 0.055h-3.967c-0.030373 0-0.055-0.024658-0.055-0.055v-2.732c0-0.030858 0.024142-0.055 0.055-0.055z" fill="#2A506F" fill-rule="nonzero"/>
<path d="m37.398 5.418v30.534c0 2.43-1.988 4.418-4.418 4.418h-27.562c-2.43 0-4.418-1.988-4.418-4.418v-30.534c0-2.43 1.988-4.418 4.418-4.418h27.562c2.43 0 4.418 1.988 4.418 4.418" fill="#2A506F"/>
<path d="m32.98-5.6843e-14h-27.562c-2.9823 0-5.418 2.4357-5.418 5.418v30.534c0 2.9823 2.4357 5.418 5.418 5.418h27.562c2.9823 0 5.418-2.4357 5.418-5.418v-30.534c0-2.9823-2.4357-5.418-5.418-5.418zm-27.562 2h27.562c1.8777 0 3.418 1.5403 3.418 3.418v30.534c0 1.8777-1.5403 3.418-3.418 3.418h-27.562c-1.8777 0-3.418-1.5403-3.418-3.418v-30.534c0-1.8777 1.5403-3.418 3.418-3.418z" fill="#2A506F" fill-rule="nonzero"/>
<path d="m19.147 73.551c0.24546 0 0.44961 0.17688 0.49194 0.41012l0.0080557 0.089876v14.882c0 0.27614-0.22386 0.5-0.5 0.5-0.24546 0-0.44961-0.17688-0.49194-0.41012l-0.0080557-0.089876v-14.882c0-0.27614 0.22386-0.5 0.5-0.5z" fill="#2A506F" fill-rule="nonzero"/>
<line x1="19.147" x2="14.532" y1="88.933" y2="84.214" stroke="#2A506F" stroke-linecap="round"/>
<line x1="19.147" x2="23.866" y1="88.933" y2="84.318" stroke="#2A506F" stroke-linecap="round"/>
<path d="m14.007 26.177c0.51076 0 0.96749-0.071211 1.3702-0.21363s0.74649-0.33887 1.0313-0.58934 0.50339-0.54268 0.65564-0.87664 0.22837-0.69247 0.22837-1.0755c0-0.3536-0.051567-0.66546-0.1547-0.93557s-0.2431-0.50585-0.4199-0.7072-0.38798-0.37816-0.63354-0.5304-0.50585-0.2873-0.78087-0.40517l-1.3702-0.58934c-0.19645-0.078578-0.38798-0.16452-0.5746-0.25783s-0.35851-0.20136-0.51567-0.32413-0.28239-0.2652-0.3757-0.42727-0.13997-0.36097-0.13997-0.5967c0-0.442 0.16452-0.78824 0.49357-1.0387s0.76368-0.3757 1.3039-0.3757c0.45182 0 0.85699 0.081034 1.2155 0.2431s0.6851 0.38552 0.97977 0.67037l0.663-0.7956c-0.34378-0.3536-0.76123-0.6409-1.2523-0.8619s-1.0264-0.3315-1.6059-0.3315c-0.442 0-0.84717 0.063845-1.2155 0.19153s-0.68756 0.30695-0.95767 0.53777-0.48129 0.50339-0.63354 0.8177-0.22837 0.65318-0.22837 1.0166c0 0.3536 0.058934 0.66546 0.1768 0.93557s0.27011 0.50339 0.45674 0.69984 0.3978 0.36342 0.63354 0.50094 0.46656 0.25538 0.69247 0.3536l1.3849 0.60407c0.22591 0.10804 0.43709 0.21118 0.63354 0.3094s0.36588 0.20872 0.5083 0.3315 0.25538 0.27011 0.33887 0.442 0.12523 0.38061 0.12523 0.62617c0 0.47147-0.1768 0.85208-0.5304 1.1418s-0.84963 0.43464-1.4881 0.43464c-0.50094 0-0.98468-0.1105-1.4512-0.3315s-0.87173-0.51321-1.2155-0.87664l-0.73667 0.85454c0.42236 0.442 0.92329 0.79069 1.5028 1.0461s1.2081 0.38307 1.8859 0.38307zm6.2664-0.1768v-4.5968c0.24556-0.60898 0.53286-1.0362 0.8619-1.2818s0.64581-0.36834 0.9503-0.36834c0.14733 0 0.27011 0.0098223 0.36834 0.029467s0.20627 0.049111 0.32413 0.0884l0.23573-1.0608c-0.22591-0.098223-0.48129-0.14733-0.76614-0.14733-0.41254 0-0.79315 0.1326-1.1418 0.3978s-0.64581 0.62371-0.89137 1.0755h-0.0442l-0.10313-1.2965h-1.0019v7.1604h1.2081zm6.5758 0.1768c0.43218 0 0.84471-0.081034 1.2376-0.2431s0.7514-0.38552 1.0755-0.67037l-0.5304-0.81034c-0.22591 0.19645-0.47884 0.36588-0.75877 0.5083s-0.58688 0.21363-0.92084 0.21363c-0.32413 0-0.62371-0.0663-0.89874-0.1989s-0.5083-0.31922-0.69984-0.55987-0.34132-0.52795-0.44937-0.8619-0.16207-0.7072-0.16207-1.1197 0.056478-0.78824 0.16943-1.1271 0.26766-0.63108 0.4641-0.87664 0.43218-0.43464 0.7072-0.56724 0.5746-0.1989 0.89874-0.1989c0.28485 0 0.54268 0.058934 0.7735 0.1768s0.44937 0.27011 0.65564 0.45674l0.6188-0.7956c-0.25538-0.22591-0.55005-0.42236-0.884-0.58934s-0.73667-0.25047-1.2081-0.25047c-0.46165 0-0.90119 0.083489-1.3186 0.25047s-0.78333 0.41254-1.0976 0.73667-0.56478 0.71948-0.7514 1.186-0.27993 0.99942-0.27993 1.5986c0 0.58934 0.085945 1.1173 0.25783 1.5838s0.40762 0.85945 0.7072 1.1787 0.65564 0.56232 1.0682 0.7293 0.85454 0.25047 1.326 0.25047z" fill="#fff" fill-rule="nonzero"/>
</g>
</g>
</g>
</svg>

Before

Width:  |  Height:  |  Size: 5.0 KiB

View File

@@ -43,7 +43,7 @@ async function checkForUpdates(interval: number) {
const release = await autoUpdater.checkForUpdates(); const release = await autoUpdater.checkForUpdates();
const isOutdated = const isOutdated =
semver.compare(release.updateInfo.version, version) > 0; semver.compare(release.updateInfo.version, version) > 0;
const shouldUpdate = release.updateInfo.stagingPercentage !== 0; // undefinded (default) means 100% const shouldUpdate = release.updateInfo.stagingPercentage || 0 > 0;
if (shouldUpdate && isOutdated) { if (shouldUpdate && isOutdated) {
await autoUpdater.downloadUpdate(); await autoUpdater.downloadUpdate();
packageUpdated = true; packageUpdated = true;
@@ -97,7 +97,6 @@ const sourceSelectorReady = new Promise((resolve) => {
async function selectImageURL(url?: string) { async function selectImageURL(url?: string) {
// 'data:,' is the default chromedriver url that is passed as last argument when running spectron tests // 'data:,' is the default chromedriver url that is passed as last argument when running spectron tests
if (url !== undefined && url !== 'data:,') { if (url !== undefined && url !== 'data:,') {
url = url.replace(/\/$/, ''); // on windows the url ends with an extra slash
url = url.startsWith(scheme) ? url.slice(scheme.length) : url; url = url.startsWith(scheme) ? url.slice(scheme.length) : url;
await sourceSelectorReady; await sourceSelectorReady;
electron.BrowserWindow.getAllWindows().forEach((window) => { electron.BrowserWindow.getAllWindows().forEach((window) => {
@@ -134,7 +133,7 @@ async function createMainWindow() {
width, width,
height, height,
frame: !fullscreen, frame: !fullscreen,
useContentSize: true, useContentSize: false,
show: false, show: false,
resizable: false, resizable: false,
maximizable: false, maximizable: false,
@@ -148,7 +147,6 @@ async function createMainWindow() {
webPreferences: { webPreferences: {
backgroundThrottling: false, backgroundThrottling: false,
nodeIntegration: true, nodeIntegration: true,
contextIsolation: false,
webviewTag: true, webviewTag: true,
zoomFactor: width / defaultWidth, zoomFactor: width / defaultWidth,
enableRemoteModule: true, enableRemoteModule: true,

View File

@@ -15,30 +15,18 @@
*/ */
import { Drive as DrivelistDrive } from 'drivelist'; import { Drive as DrivelistDrive } from 'drivelist';
import { import * as sdk from 'etcher-sdk';
BlockDevice,
File,
Http,
Metadata,
SourceDestination,
} from 'etcher-sdk/build/source-destination';
import {
MultiDestinationProgress,
OnProgressFunction,
OnFailFunction,
decompressThenFlash,
DECOMPRESSED_IMAGE_PREFIX,
} from 'etcher-sdk/build/multi-write';
import { cleanupTmpFiles } from 'etcher-sdk/build/tmp'; import { cleanupTmpFiles } from 'etcher-sdk/build/tmp';
import { promises as fs } from 'fs';
import * as _ from 'lodash';
import * as ipc from 'node-ipc'; import * as ipc from 'node-ipc';
import { totalmem } from 'os'; import { totalmem } from 'os';
import { BlockDevice, File, Http } from 'etcher-sdk/build/source-destination';
import { toJSON } from '../../shared/errors'; import { toJSON } from '../../shared/errors';
import { GENERAL_ERROR, SUCCESS } from '../../shared/exit-codes'; import { GENERAL_ERROR, SUCCESS } from '../../shared/exit-codes';
import { delay, isJson } from '../../shared/utils'; import { delay } from '../../shared/utils';
import { SourceMetadata } from '../app/components/source-selector/source-selector'; import { SourceMetadata } from '../app/components/source-selector/source-selector';
import axios from 'axios';
import * as _ from 'lodash';
ipc.config.id = process.env.IPC_CLIENT_ID as string; ipc.config.id = process.env.IPC_CLIENT_ID as string;
ipc.config.socketRoot = process.env.IPC_SOCKET_ROOT as string; ipc.config.socketRoot = process.env.IPC_SOCKET_ROOT as string;
@@ -71,7 +59,7 @@ function log(message: string) {
*/ */
async function terminate(exitCode: number) { async function terminate(exitCode: number) {
ipc.disconnect(IPC_SERVER_ID); ipc.disconnect(IPC_SERVER_ID);
await cleanupTmpFiles(Date.now(), DECOMPRESSED_IMAGE_PREFIX); await cleanupTmpFiles(Date.now());
process.nextTick(() => { process.nextTick(() => {
process.exit(exitCode || SUCCESS); process.exit(exitCode || SUCCESS);
}); });
@@ -86,25 +74,14 @@ async function handleError(error: Error) {
await terminate(GENERAL_ERROR); await terminate(GENERAL_ERROR);
} }
export interface FlashError extends Error { interface WriteResult {
description: string; bytesWritten: number;
device: string; devices: {
code: string;
}
export interface WriteResult {
bytesWritten?: number;
devices?: {
failed: number; failed: number;
successful: number; successful: number;
}; };
errors: FlashError[]; errors: Array<Error & { device: string }>;
sourceMetadata?: Metadata; sourceMetadata: sdk.sourceDestination.Metadata;
}
export interface FlashResults extends WriteResult {
skip?: boolean;
cancelled?: boolean;
} }
/** /**
@@ -126,15 +103,19 @@ async function writeAndValidate({
onProgress, onProgress,
onFail, onFail,
}: { }: {
source: SourceDestination; source: sdk.sourceDestination.SourceDestination;
destinations: BlockDevice[]; destinations: sdk.sourceDestination.BlockDevice[];
verify: boolean; verify: boolean;
autoBlockmapping: boolean; autoBlockmapping: boolean;
decompressFirst: boolean; decompressFirst: boolean;
onProgress: OnProgressFunction; onProgress: sdk.multiWrite.OnProgressFunction;
onFail: OnFailFunction; onFail: sdk.multiWrite.OnFailFunction;
}): Promise<WriteResult> { }): Promise<WriteResult> {
const { sourceMetadata, failures, bytesWritten } = await decompressThenFlash({ const {
sourceMetadata,
failures,
bytesWritten,
} = await sdk.multiWrite.decompressThenFlash({
source, source,
destinations, destinations,
onFail, onFail,
@@ -158,8 +139,8 @@ async function writeAndValidate({
sourceMetadata, sourceMetadata,
}; };
for (const [destination, error] of failures) { for (const [destination, error] of failures) {
const err = error as FlashError; const err = error as Error & { device: string; description: string };
const drive = destination as BlockDevice; const drive = destination as sdk.sourceDestination.BlockDevice;
err.device = drive.device; err.device = drive.device;
err.description = drive.description; err.description = drive.description;
result.errors.push(err); result.errors.push(err);
@@ -170,10 +151,18 @@ async function writeAndValidate({
interface WriteOptions { interface WriteOptions {
image: SourceMetadata; image: SourceMetadata;
destinations: DrivelistDrive[]; destinations: DrivelistDrive[];
unmountOnSuccess: boolean;
validateWriteOnSuccess: boolean;
autoBlockmapping: boolean; autoBlockmapping: boolean;
decompressFirst: boolean; decompressFirst: boolean;
SourceType: string; SourceType: string;
httpRequest?: any; saveUrlImage: boolean;
saveUrlImageTo: string;
}
interface ProgressState
extends Omit<sdk.multiWrite.MultiDestinationProgress, 'type'> {
type: sdk.multiWrite.MultiDestinationProgress['type'] | 'downloading';
} }
ipc.connectTo(IPC_SERVER_ID, () => { ipc.connectTo(IPC_SERVER_ID, () => {
@@ -211,7 +200,7 @@ ipc.connectTo(IPC_SERVER_ID, () => {
* @example * @example
* writer.on('progress', onProgress) * writer.on('progress', onProgress)
*/ */
const onProgress = (state: MultiDestinationProgress) => { const onProgress = (state: ProgressState) => {
ipc.of[IPC_SERVER_ID].emit('state', state); ipc.of[IPC_SERVER_ID].emit('state', state);
}; };
@@ -247,7 +236,10 @@ ipc.connectTo(IPC_SERVER_ID, () => {
* @example * @example
* writer.on('fail', onFail) * writer.on('fail', onFail)
*/ */
const onFail = (destination: SourceDestination, error: Error) => { const onFail = (
destination: sdk.sourceDestination.SourceDestination,
error: Error,
) => {
ipc.of[IPC_SERVER_ID].emit('fail', { ipc.of[IPC_SERVER_ID].emit('fail', {
// TODO: device should be destination // TODO: device should be destination
// @ts-ignore (destination.drive is private) // @ts-ignore (destination.drive is private)
@@ -260,12 +252,14 @@ ipc.connectTo(IPC_SERVER_ID, () => {
const imagePath = options.image.path; const imagePath = options.image.path;
log(`Image: ${imagePath}`); log(`Image: ${imagePath}`);
log(`Devices: ${destinations.join(', ')}`); log(`Devices: ${destinations.join(', ')}`);
log(`Umount on success: ${options.unmountOnSuccess}`);
log(`Validate on success: ${options.validateWriteOnSuccess}`);
log(`Auto blockmapping: ${options.autoBlockmapping}`); log(`Auto blockmapping: ${options.autoBlockmapping}`);
log(`Decompress first: ${options.decompressFirst}`); log(`Decompress first: ${options.decompressFirst}`);
const dests = options.destinations.map((destination) => { const dests = options.destinations.map((destination) => {
return new BlockDevice({ return new sdk.sourceDestination.BlockDevice({
drive: destination, drive: destination,
unmountOnSuccess: true, unmountOnSuccess: options.unmountOnSuccess,
write: true, write: true,
direct: true, direct: true,
}); });
@@ -284,28 +278,22 @@ ipc.connectTo(IPC_SERVER_ID, () => {
path: imagePath, path: imagePath,
}); });
} else { } else {
const decodedImagePath = decodeURIComponent(imagePath); if (options.saveUrlImage) {
if (isJson(decodedImagePath)) { source = await saveFileBeforeFlash(
const imagePathObject = JSON.parse(decodedImagePath); imagePath,
source = new Http({ options.saveUrlImageTo,
url: imagePathObject.url, onProgress,
avoidRandomAccess: true, onFail,
axiosInstance: axios.create(_.omit(imagePathObject, ['url'])), );
auth: options.image.auth,
});
} else { } else {
source = new Http({ source = new Http({ url: imagePath, avoidRandomAccess: true });
url: imagePath,
avoidRandomAccess: true,
auth: options.image.auth,
});
} }
} }
} }
const results = await writeAndValidate({ const results = await writeAndValidate({
source, source,
destinations: dests, destinations: dests,
verify: true, verify: options.validateWriteOnSuccess,
autoBlockmapping: options.autoBlockmapping, autoBlockmapping: options.autoBlockmapping,
decompressFirst: options.decompressFirst, decompressFirst: options.decompressFirst,
onProgress, onProgress,
@@ -318,7 +306,8 @@ ipc.connectTo(IPC_SERVER_ID, () => {
ipc.of[IPC_SERVER_ID].emit('done', { results }); ipc.of[IPC_SERVER_ID].emit('done', { results });
await delay(DISCONNECT_DELAY); await delay(DISCONNECT_DELAY);
await terminate(exitCode); await terminate(exitCode);
} catch (error: any) { } catch (error) {
log(`Error: ${error.message}`);
exitCode = GENERAL_ERROR; exitCode = GENERAL_ERROR;
ipc.of[IPC_SERVER_ID].emit('error', toJSON(error)); ipc.of[IPC_SERVER_ID].emit('error', toJSON(error));
} }
@@ -331,3 +320,43 @@ ipc.connectTo(IPC_SERVER_ID, () => {
ipc.of[IPC_SERVER_ID].emit('ready', {}); ipc.of[IPC_SERVER_ID].emit('ready', {});
}); });
}); });
async function saveFileBeforeFlash(
imagePath: string,
saveUrlImageTo: string,
onProgress: (state: ProgressState) => void,
onFail: (
destination: sdk.sourceDestination.SourceDestination,
error: Error,
) => void,
) {
const urlImage = new Http({ url: imagePath, avoidRandomAccess: true });
const source = await urlImage.getInnerSource();
const metadata = await source.getMetadata();
const fileName = `${saveUrlImageTo}/${metadata.name}`;
let alreadyDownloaded = false;
try {
alreadyDownloaded = (await fs.stat(fileName)).isFile();
} catch (error) {
if (error.code !== 'ENOENT') {
throw error;
}
}
if (!alreadyDownloaded) {
await sdk.multiWrite.decompressThenFlash({
source,
destinations: [new File({ path: fileName, write: true })],
onProgress: (progress) => {
onProgress({
...progress,
type: 'downloading',
});
},
onFail: (...args) => {
onFail(...args);
},
verify: true,
});
}
return new File({ path: fileName });
}

View File

@@ -15,12 +15,11 @@
*/ */
import { execFile } from 'child_process'; import { execFile } from 'child_process';
import { app, remote } from 'electron';
import { join } from 'path'; import { join } from 'path';
import { env } from 'process'; import { env } from 'process';
import { promisify } from 'util'; import { promisify } from 'util';
import { getAppPath } from '../utils';
const execFileAsync = promisify(execFile); const execFileAsync = promisify(execFile);
const SUCCESSFUL_AUTH_MARKER = 'AUTHENTICATION SUCCEEDED'; const SUCCESSFUL_AUTH_MARKER = 'AUTHENTICATION SUCCEEDED';
@@ -38,7 +37,7 @@ export async function sudo(
env: { env: {
PATH: env.PATH, PATH: env.PATH,
SUDO_ASKPASS: join( SUDO_ASKPASS: join(
getAppPath(), (app || remote.app).getAppPath(),
__dirname, __dirname,
'sudo-askpass.osascript.js', 'sudo-askpass.osascript.js',
), ),
@@ -50,7 +49,7 @@ export async function sudo(
stdout: stdout.slice(EXPECTED_SUCCESSFUL_AUTH_MARKER.length), stdout: stdout.slice(EXPECTED_SUCCESSFUL_AUTH_MARKER.length),
stderr, stderr,
}; };
} catch (error: any) { } catch (error) {
if (error.code === 1) { if (error.code === 1) {
if (!error.stdout.startsWith(EXPECTED_SUCCESSFUL_AUTH_MARKER)) { if (!error.stdout.startsWith(EXPECTED_SUCCESSFUL_AUTH_MARKER)) {
return { cancelled: true }; return { cancelled: true };

View File

@@ -34,6 +34,16 @@ export type DrivelistDrive = Drive & {
displayName: string; displayName: string;
}; };
/**
* @summary Check if a drive is locked
*
* @description
* This usually points out a locked SD Card.
*/
export function isDriveLocked(drive: DrivelistDrive): boolean {
return Boolean(drive.isReadOnly);
}
/** /**
* @summary Check if a drive is a system drive * @summary Check if a drive is a system drive
*/ */
@@ -63,7 +73,9 @@ export function isSourceDrive(
): boolean { ): boolean {
if (selection) { if (selection) {
if (selection.drive) { if (selection.drive) {
return selection.drive.device === drive.device; const sourcePath = selection.drive.devicePath || selection.drive.device;
const drivePath = drive.devicePath || drive.device;
return pathIsInside(sourcePath, drivePath);
} }
if (selection.path) { if (selection.path) {
return sourceIsInsideDrive(selection.path, drive); return sourceIsInsideDrive(selection.path, drive);
@@ -105,18 +117,24 @@ export function isDriveLargeEnough(
} }
/** /**
* @summary Check if a drive is valid, i.e. large enough for an image * @summary Check if a drive is disabled (i.e. not ready for selection)
*/
export function isDriveDisabled(drive: DrivelistDrive): boolean {
return drive.disabled || false;
}
/**
* @summary Check if a drive is valid, i.e. not locked and large enough for an image
*/ */
export function isDriveValid( export function isDriveValid(
drive: DrivelistDrive, drive: DrivelistDrive,
image?: SourceMetadata, image?: SourceMetadata,
write: boolean = true,
): boolean { ): boolean {
return ( return (
!write || !isDriveLocked(drive) &&
(!drive.disabled &&
isDriveLargeEnough(drive, image) && isDriveLargeEnough(drive, image) &&
!isSourceDrive(drive, image as SourceMetadata)) !isSourceDrive(drive, image as SourceMetadata) &&
!isDriveDisabled(drive)
); );
} }
@@ -197,19 +215,17 @@ export const statuses = {
*/ */
export function getDriveImageCompatibilityStatuses( export function getDriveImageCompatibilityStatuses(
drive: DrivelistDrive, drive: DrivelistDrive,
image: SourceMetadata | undefined, image?: SourceMetadata,
write: boolean,
) { ) {
const statusList = []; const statusList = [];
// Mind the order of the if-statements if you modify. // Mind the order of the if-statements if you modify.
if (drive.isReadOnly && write) { if (isDriveLocked(drive)) {
statusList.push({ statusList.push({
type: COMPATIBILITY_STATUS_TYPES.ERROR, type: COMPATIBILITY_STATUS_TYPES.ERROR,
message: messages.compatibility.locked(), message: messages.compatibility.locked(),
}); });
} } else if (
if (
!_.isNil(drive) && !_.isNil(drive) &&
!_.isNil(drive.size) && !_.isNil(drive.size) &&
!isDriveLargeEnough(drive, image) !isDriveLargeEnough(drive, image)
@@ -248,11 +264,10 @@ export function getDriveImageCompatibilityStatuses(
*/ */
export function getListDriveImageCompatibilityStatuses( export function getListDriveImageCompatibilityStatuses(
drives: DrivelistDrive[], drives: DrivelistDrive[],
image: SourceMetadata | undefined, image: SourceMetadata,
write: boolean,
) { ) {
return drives.flatMap((drive) => { return drives.flatMap((drive) => {
return getDriveImageCompatibilityStatuses(drive, image, write); return getDriveImageCompatibilityStatuses(drive, image);
}); });
} }
@@ -264,12 +279,9 @@ export function getListDriveImageCompatibilityStatuses(
*/ */
export function hasDriveImageCompatibilityStatus( export function hasDriveImageCompatibilityStatus(
drive: DrivelistDrive, drive: DrivelistDrive,
image: SourceMetadata | undefined, image: SourceMetadata,
write: boolean,
) { ) {
return Boolean( return Boolean(getDriveImageCompatibilityStatuses(drive, image).length);
getDriveImageCompatibilityStatuses(drive, image, write).length,
);
} }
export interface DriveStatus { export interface DriveStatus {

View File

@@ -81,10 +81,13 @@ export const compatibility = {
} as const; } as const;
export const warning = { export const warning = {
tooSmall: (source: { size: number }, target: { size: number }) => { unrecommendedDriveSize: (
image: { recommendedDriveSize: number },
drive: { device: string; size: number },
) => {
return outdent({ newline: ' ' })` return outdent({ newline: ' ' })`
The selected source is ${prettyBytes(source.size - target.size)} This image recommends a ${prettyBytes(image.recommendedDriveSize)}
larger than this drive. drive, however ${drive.device} is only ${prettyBytes(drive.size)}.
`; `;
}, },
@@ -114,16 +117,8 @@ export const warning = {
].join(' '); ].join(' ');
}, },
driveMissingPartitionTable: () => {
return outdent({ newline: ' ' })`
It looks like this is not a bootable drive.
The drive does not appear to contain a partition table,
and might not be recognized or bootable by your device.
`;
},
largeDriveSize: () => { largeDriveSize: () => {
return "This is a large drive! Make sure it doesn't contain files that you want to keep."; return 'This is a large drive! Make sure it doesn\'t contain files that you want to keep.';
}, },
systemDrive: () => { systemDrive: () => {

View File

@@ -15,32 +15,30 @@
*/ */
import * as childProcess from 'child_process'; import * as childProcess from 'child_process';
import { withTmpFile } from 'etcher-sdk/build/tmp';
import { promises as fs } from 'fs'; import { promises as fs } from 'fs';
import * as _ from 'lodash'; import * as _ from 'lodash';
import * as os from 'os'; import * as os from 'os';
import * as semver from 'semver'; import * as semver from 'semver';
import * as sudoPrompt from '@balena/sudo-prompt'; import * as sudoPrompt from 'sudo-prompt';
import { promisify } from 'util'; import { promisify } from 'util';
import { sudo as catalinaSudo } from './catalina-sudo/sudo'; import { sudo as catalinaSudo } from './catalina-sudo/sudo';
import * as errors from './errors'; import * as errors from './errors';
import { withTmpFile } from './tmp';
const execAsync = promisify(childProcess.exec); const execAsync = promisify(childProcess.exec);
const execFileAsync = promisify(childProcess.execFile); const execFileAsync = promisify(childProcess.execFile);
type Std = string | Buffer | undefined;
function sudoExecAsync( function sudoExecAsync(
cmd: string, cmd: string,
options: { name: string }, options: { name: string },
): Promise<{ stdout: Std; stderr: Std }> { ): Promise<{ stdout: string; stderr: string }> {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
sudoPrompt.exec( sudoPrompt.exec(
cmd, cmd,
options, options,
(error: Error | undefined, stdout: Std, stderr: Std) => { (error: Error | null, stdout: string, stderr: string) => {
if (error) { if (error != null) {
reject(error); reject(error);
} else { } else {
resolve({ stdout, stderr }); resolve({ stdout, stderr });
@@ -62,7 +60,7 @@ export async function isElevated(): Promise<boolean> {
// See http://stackoverflow.com/a/28268802 // See http://stackoverflow.com/a/28268802
try { try {
await execAsync('fltmc'); await execAsync('fltmc');
} catch (error: any) { } catch (error) {
if (error.code === os.constants.errno.EPERM) { if (error.code === os.constants.errno.EPERM) {
return false; return false;
} }
@@ -148,7 +146,7 @@ async function elevateScriptCatalina(
try { try {
const { cancelled } = await catalinaSudo(cmd); const { cancelled } = await catalinaSudo(cmd);
return { cancelled }; return { cancelled };
} catch (error: any) { } catch (error) {
throw errors.createError({ title: error.stderr }); throw errors.createError({ title: error.stderr });
} }
} }
@@ -174,11 +172,10 @@ export async function elevateCommand(
); );
return await withTmpFile( return await withTmpFile(
{ {
keepOpen: false,
prefix: 'balena-etcher-electron-', prefix: 'balena-etcher-electron-',
postfix: '.cmd', postfix: '.cmd',
}, },
async ({ path }) => { async (path) => {
await fs.writeFile(path, launchScript); await fs.writeFile(path, launchScript);
if (isWindows) { if (isWindows) {
return elevateScriptWindows(path, options.applicationName); return elevateScriptWindows(path, options.applicationName);
@@ -192,7 +189,7 @@ export async function elevateCommand(
} }
try { try {
return await elevateScriptUnix(path, options.applicationName); return await elevateScriptUnix(path, options.applicationName);
} catch (error: any) { } catch (error) {
// We're hardcoding internal error messages declared by `sudo-prompt`. // We're hardcoding internal error messages declared by `sudo-prompt`.
// There doesn't seem to be a better way to handle these errors, so // There doesn't seem to be a better way to handle these errors, so
// for now, we should make sure we double check if the error messages // for now, we should make sure we double check if the error messages

27
lib/shared/tmp.ts Normal file
View File

@@ -0,0 +1,27 @@
import * as tmp from 'tmp';
function tmpFileAsync(
options: tmp.FileOptions,
): Promise<{ path: string; cleanup: () => void }> {
return new Promise((resolve, reject) => {
tmp.file(options, (error, path, _fd, cleanup) => {
if (error) {
reject(error);
} else {
resolve({ path, cleanup });
}
});
});
}
export async function withTmpFile<T>(
options: tmp.FileOptions,
fn: (path: string) => Promise<T>,
): Promise<T> {
const { path, cleanup } = await tmpFileAsync(options);
try {
return await fn(path);
} finally {
cleanup();
}
}

View File

@@ -15,7 +15,6 @@
*/ */
import axios from 'axios'; import axios from 'axios';
import { app, remote } from 'electron';
import { Dictionary } from 'lodash'; import { Dictionary } from 'lodash';
import * as errors from './errors'; import * as errors from './errors';
@@ -48,25 +47,3 @@ export async function delay(duration: number): Promise<void> {
setTimeout(resolve, duration); setTimeout(resolve, duration);
}); });
} }
export function getAppPath(): string {
return (
(app || remote.app)
.getAppPath()
// With macOS universal builds, getAppPath() returns the path to an app.asar file containing an index.js file which will
// include the app-x64 or app-arm64 folder depending on the arch.
// We don't care about the app.asar file, we want the actual folder.
.replace(/\.asar$/, () =>
process.platform === 'darwin' ? '-' + process.arch : '',
)
);
}
export function isJson(jsonString: string) {
try {
JSON.parse(jsonString);
} catch (e) {
return false;
}
return true;
}

File diff suppressed because it is too large Load Diff

View File

@@ -2,7 +2,7 @@
"name": "balena-etcher", "name": "balena-etcher",
"private": true, "private": true,
"displayName": "balenaEtcher", "displayName": "balenaEtcher",
"version": "1.7.9", "version": "1.5.109",
"packageType": "local", "packageType": "local",
"main": "generated/etcher.js", "main": "generated/etcher.js",
"description": "Flash OS images to SD cards and USB drives, safely and easily.", "description": "Flash OS images to SD cards and USB drives, safely and easily.",
@@ -15,19 +15,20 @@
"scripts": { "scripts": {
"lint-ts": "balena-lint --fix --typescript typings lib tests scripts/clean-shrinkwrap.ts webpack.config.ts", "lint-ts": "balena-lint --fix --typescript typings lib tests scripts/clean-shrinkwrap.ts webpack.config.ts",
"lint-css": "prettier --write lib/**/*.css", "lint-css": "prettier --write lib/**/*.css",
"lint": "npm run lint-ts && npm run lint-css", "lint-spell": "codespell --dictionary - --dictionary dictionary.txt --skip *.ttf *.svg *.gz,*.bz2,*.xz,*.zip,*.img,*.dmg,*.iso,*.rpi-sdcard,*.wic,.DS_Store,*.dtb,*.dtbo,*.dat,*.elf,*.bin,*.foo,xz-without-extension lib tests docs Makefile *.md LICENSE",
"test-spectron": "mocha --recursive --reporter spec --require ts-node/register/transpile-only --require-main tests/gui/allow-renderer-process-reuse.ts tests/spectron/runner.spec.ts", "lint": "npm run lint-ts && npm run lint-css && npm run lint-spell",
"test-gui": "electron-mocha --recursive --reporter spec --require ts-node/register/transpile-only --require-main tests/gui/allow-renderer-process-reuse.ts --full-trace --no-sandbox --renderer tests/gui/**/*.ts", "test-spectron": "mocha --recursive --reporter spec --require ts-node/register --require-main tests/gui/allow-renderer-process-reuse.ts tests/spectron/runner.spec.ts",
"test-shared": "electron-mocha --recursive --reporter spec --require ts-node/register/transpile-only --require-main tests/gui/allow-renderer-process-reuse.ts --full-trace --no-sandbox tests/shared/**/*.ts", "test-gui": "electron-mocha --recursive --reporter spec --require ts-node/register --require-main tests/gui/allow-renderer-process-reuse.ts --full-trace --no-sandbox --renderer tests/gui/**/*.ts",
"test-shared": "electron-mocha --recursive --reporter spec --require ts-node/register --require-main tests/gui/allow-renderer-process-reuse.ts --full-trace --no-sandbox tests/shared/**/*.ts",
"test": "npm run lint && npm run test-gui && npm run test-shared && npm run test-spectron && npm run sanity-checks", "test": "npm run lint && npm run test-gui && npm run test-shared && npm run test-spectron && npm run sanity-checks",
"sanity-checks": "bash scripts/ci/ensure-all-file-extensions-in-gitattributes.sh", "sanity-checks": "bash scripts/ci/ensure-all-file-extensions-in-gitattributes.sh",
"start": "./node_modules/.bin/electron .", "start": "./node_modules/.bin/electron .",
"postinstall": "electron-rebuild -t prod,dev,optional", "postshrinkwrap": "ts-node ./scripts/clean-shrinkwrap.ts",
"webpack": "webpack", "webpack": "webpack",
"watch": "webpack serve --no-optimization-minimize --config ./webpack.dev.config.ts", "watch": "webpack --watch",
"concourse-build-electron": "npm run webpack", "concourse-build-electron": "npm run webpack",
"concourse-test": "npx npm@6.14.8 test", "concourse-test": "npx npm@6.14.5 test",
"concourse-test-electron": "npx npm@6.14.8 test" "concourse-test-electron": "npx npm@6.14.5 test"
}, },
"husky": { "husky": {
"hooks": { "hooks": {
@@ -44,78 +45,72 @@
}, },
"author": "Balena Inc. <hello@etcher.io>", "author": "Balena Inc. <hello@etcher.io>",
"license": "Apache-2.0", "license": "Apache-2.0",
"platformSpecificDependencies": [
"fsevents",
"winusb-driver-generator"
],
"devDependencies": { "devDependencies": {
"@balena/lint": "5.3.0", "@balena/lint": "^5.0.4",
"@balena/sudo-prompt": "9.2.1-workaround-windows-amperstand-in-username-0849e215b947987a643fe5763902aea201255534", "@fortawesome/fontawesome-free": "^5.13.1",
"@fortawesome/fontawesome-free": "5.13.1", "@svgr/webpack": "^5.4.0",
"@svgr/webpack": "5.5.0", "@types/chai": "^4.2.7",
"@types/chai": "4.2.7", "@types/copy-webpack-plugin": "^6.0.0",
"@types/copy-webpack-plugin": "6.0.0", "@types/mime-types": "^2.1.0",
"@types/mime-types": "2.1.0", "@types/mini-css-extract-plugin": "^0.9.1",
"@types/mini-css-extract-plugin": "1.2.2", "@types/mocha": "^8.0.3",
"@types/mocha": "8.0.3", "@types/node": "^12.12.39",
"@types/node": "14.14.41", "@types/node-ipc": "^9.1.2",
"@types/node-ipc": "9.1.2", "@types/react-dom": "^16.8.4",
"@types/react": "16.8.5", "@types/semver": "^7.1.0",
"@types/react-dom": "16.8.4", "@types/sinon": "^9.0.0",
"@types/semver": "7.1.0", "@types/terser-webpack-plugin": "^4.1.0",
"@types/sinon": "9.0.0", "@types/tmp": "^0.2.0",
"@types/terser-webpack-plugin": "5.0.2", "@types/webpack-node-externals": "^2.5.0",
"@types/tmp": "0.2.0", "chai": "^4.2.0",
"@types/webpack-node-externals": "2.5.0", "copy-webpack-plugin": "^6.0.1",
"aws4-axios": "2.2.1", "css-loader": "^4.2.1",
"chai": "4.2.0", "d3": "^4.13.0",
"copy-webpack-plugin": "7.0.0", "debug": "^4.2.0",
"css-loader": "5.0.1", "electron": "9.2.1",
"d3": "4.13.0", "electron-builder": "^22.7.0",
"debug": "4.2.0", "electron-mocha": "^9.1.0",
"electron": "12.2.3", "electron-notarize": "^1.0.0",
"electron-builder": "22.10.5", "electron-rebuild": "^1.11.0",
"electron-mocha": "9.3.2", "electron-updater": "^4.3.2",
"electron-notarize": "1.0.0", "etcher-sdk": "^4.1.30",
"electron-rebuild": "3.2.5", "file-loader": "^6.0.0",
"electron-updater": "4.3.5", "husky": "^4.2.5",
"esbuild-loader": "2.16.0", "immutable": "^3.8.1",
"etcher-sdk": "6.3.0", "lint-staged": "^10.2.2",
"file-loader": "6.2.0", "lodash": "^4.17.10",
"husky": "4.2.5", "mini-css-extract-plugin": "^0.10.0",
"immutable": "3.8.1", "mocha": "^8.0.1",
"lint-staged": "10.2.2", "native-addon-loader": "^2.0.1",
"lodash": "4.17.10", "node-ipc": "^9.1.1",
"mini-css-extract-plugin": "1.3.3",
"mocha": "8.0.1",
"native-addon-loader": "2.0.1",
"node-ipc": "9.1.1",
"omit-deep-lodash": "1.1.4", "omit-deep-lodash": "1.1.4",
"outdent": "0.7.1", "outdent": "^0.7.1",
"path-is-inside": "1.0.2", "path-is-inside": "^1.0.2",
"pnp-webpack-plugin": "1.6.4", "pretty-bytes": "^5.3.0",
"pretty-bytes": "5.3.0", "react": "^16.8.5",
"react": "16.8.5", "react-dom": "^16.8.5",
"react-dom": "16.8.5", "redux": "^4.0.5",
"redux": "4.0.5", "rendition": "^18.8.3",
"rendition": "19.2.0", "resin-corvus": "^2.0.5",
"resin-corvus": "2.0.5", "semver": "^7.3.2",
"semver": "7.3.2", "simple-progress-webpack-plugin": "^1.1.2",
"simple-progress-webpack-plugin": "1.1.2", "sinon": "^9.0.2",
"sinon": "9.0.2", "spectron": "^11.0.0",
"spectron": "14.0.0", "string-replace-loader": "^2.3.0",
"string-replace-loader": "3.0.1", "styled-components": "^5.1.0",
"style-loader": "2.0.0", "sudo-prompt": "github:zvin/sudo-prompt#workaround-windows-amperstand-in-username",
"styled-components": "5.1.0", "sys-class-rgb-led": "^2.1.0",
"sys-class-rgb-led": "3.0.0", "tmp": "^0.2.1",
"terser-webpack-plugin": "5.2.5", "ts-loader": "^8.0.0",
"ts-loader": "8.0.12", "ts-node": "^9.0.0",
"ts-node": "9.1.1", "tslib": "^2.0.0",
"tslib": "2.0.0", "typescript": "^4.0.2",
"typescript": "4.4.4", "uuid": "^8.1.0",
"url-loader": "4.1.1", "webpack": "^4.40.2",
"uuid": "8.1.0", "webpack-cli": "^3.3.9"
"webpack": "5.11.0",
"webpack-cli": "4.2.0",
"webpack-dev-server": "4.5.0"
},
"versionist": {
"publishedAt": "2022-04-22T13:10:47.906Z"
} }
} }

View File

@@ -6,15 +6,6 @@ sentry:
team: resinio team: resinio
type: electron type: electron
triggerNotification: triggerNotification:
version: 1.7.9 version: 1.5.81
stagingPercentage: 100 stagingPercentage: 100
upstream:
- repo: etcher-sdk
url: https://github.com/balena-io-modules/etcher-sdk
module: 'etcher-sdk'
- repo: sys-class-rgb-led
url: https://github.com/balena-io-modules/sys-class-rgb-led
module: sys-class-rgb-led
- repo: rendition
url: https://github.com/balena-io-modules/rendition
module: rendition

View File

@@ -1,2 +1,3 @@
codespell==1.12.0
awscli==1.11.87 awscli==1.11.87
shyaml==0.5.0 shyaml==0.5.0

View File

@@ -29,15 +29,11 @@ const SHRINKWRAP_FILENAME = path.join(__dirname, '..', 'npm-shrinkwrap.json');
async function main() { async function main() {
try { try {
const cleaned = omit(shrinkwrap, packageInfo.platformSpecificDependencies); const cleaned = omit(shrinkwrap, packageInfo.platformSpecificDependencies);
for (const item of Object.values(cleaned.dependencies)) {
// @ts-ignore
item.dev = true;
}
await writeFileAsync( await writeFileAsync(
SHRINKWRAP_FILENAME, SHRINKWRAP_FILENAME,
JSON.stringify(cleaned, null, JSON_INDENT), JSON.stringify(cleaned, null, JSON_INDENT),
); );
} catch (error: any) { } catch (error) {
console.log(`[ERROR] Couldn't write shrinkwrap file: ${error.stack}`); console.log(`[ERROR] Couldn't write shrinkwrap file: ${error.stack}`);
process.exitCode = 1; process.exitCode = 1;
} }

View File

@@ -573,8 +573,7 @@ describe('Model: flashState', function () {
}); });
describe('.getFlashUuid()', function () { describe('.getFlashUuid()', function () {
const UUID_REGEX = const UUID_REGEX = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/;
/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/;
it('should be initially undefined', function () { it('should be initially undefined', function () {
expect(flashState.getFlashUuid()).to.be.undefined; expect(flashState.getFlashUuid()).to.be.undefined;

View File

@@ -33,6 +33,26 @@ describe('Model: selectionState', function () {
expect(selectionState.getImage()).to.be.undefined; expect(selectionState.getImage()).to.be.undefined;
}); });
it('getImagePath() should return undefined', function () {
expect(selectionState.getImagePath()).to.be.undefined;
});
it('getImageSize() should return undefined', function () {
expect(selectionState.getImageSize()).to.be.undefined;
});
it('getImageName() should return undefined', function () {
expect(selectionState.getImageName()).to.be.undefined;
});
it('getImageLogo() should return undefined', function () {
expect(selectionState.getImageLogo()).to.be.undefined;
});
it('getImageSupportUrl() should return undefined', function () {
expect(selectionState.getImageSupportUrl()).to.be.undefined;
});
it('hasDrive() should return false', function () { it('hasDrive() should return false', function () {
const hasDrive = selectionState.hasDrive(); const hasDrive = selectionState.hasDrive();
expect(hasDrive).to.be.false; expect(hasDrive).to.be.false;
@@ -359,6 +379,43 @@ describe('Model: selectionState', function () {
}); });
}); });
describe('.getImagePath()', function () {
it('should return the image path', function () {
const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('foo.img');
});
});
describe('.getImageSize()', function () {
it('should return the image size', function () {
const imageSize = selectionState.getImageSize();
expect(imageSize).to.equal(999999999);
});
});
describe('.getImageName()', function () {
it('should return the image name', function () {
const imageName = selectionState.getImageName();
expect(imageName).to.equal('Raspbian');
});
});
describe('.getImageLogo()', function () {
it('should return the image logo', function () {
const imageLogo = selectionState.getImageLogo();
expect(imageLogo).to.equal(
'<svg><text fill="red">Raspbian</text></svg>',
);
});
});
describe('.getImageSupportUrl()', function () {
it('should return the image support url', function () {
const imageSupportUrl = selectionState.getImageSupportUrl();
expect(imageSupportUrl).to.equal('https://www.raspbian.org/forums/');
});
});
describe('.hasImage()', function () { describe('.hasImage()', function () {
it('should return true', function () { it('should return true', function () {
const hasImage = selectionState.hasImage(); const hasImage = selectionState.hasImage();
@@ -378,9 +435,9 @@ describe('Model: selectionState', function () {
SourceType: File, SourceType: File,
}); });
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('bar.img'); expect(imagePath).to.equal('bar.img');
const imageSize = selectionState.getImage()?.size; const imageSize = selectionState.getImageSize();
expect(imageSize).to.equal(999999999); expect(imageSize).to.equal(999999999);
}); });
}); });
@@ -389,9 +446,9 @@ describe('Model: selectionState', function () {
it('should clear the image', function () { it('should clear the image', function () {
selectionState.deselectImage(); selectionState.deselectImage();
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.be.undefined; expect(imagePath).to.be.undefined;
const imageSize = selectionState.getImage()?.size; const imageSize = selectionState.getImageSize();
expect(imageSize).to.be.undefined; expect(imageSize).to.be.undefined;
}); });
}); });
@@ -415,9 +472,9 @@ describe('Model: selectionState', function () {
it('should be able to set an image', function () { it('should be able to set an image', function () {
selectionState.selectSource(image); selectionState.selectSource(image);
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('foo.img'); expect(imagePath).to.equal('foo.img');
const imageSize = selectionState.getImage()?.size; const imageSize = selectionState.getImageSize();
expect(imageSize).to.equal(999999999); expect(imageSize).to.equal(999999999);
}); });
@@ -428,7 +485,7 @@ describe('Model: selectionState', function () {
archiveExtension: 'zip', archiveExtension: 'zip',
}); });
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('foo.zip'); expect(imagePath).to.equal('foo.zip');
}); });
@@ -439,7 +496,7 @@ describe('Model: selectionState', function () {
archiveExtension: 'xz', archiveExtension: 'xz',
}); });
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('foo.xz'); expect(imagePath).to.equal('foo.xz');
}); });
@@ -450,7 +507,7 @@ describe('Model: selectionState', function () {
archiveExtension: 'gz', archiveExtension: 'gz',
}); });
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('something.linux-x86-64.gz'); expect(imagePath).to.equal('something.linux-x86-64.gz');
}); });
@@ -618,12 +675,12 @@ describe('Model: selectionState', function () {
}); });
it('getImagePath() should return undefined', function () { it('getImagePath() should return undefined', function () {
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.be.undefined; expect(imagePath).to.be.undefined;
}); });
it('getImageSize() should return undefined', function () { it('getImageSize() should return undefined', function () {
const imageSize = selectionState.getImage()?.size; const imageSize = selectionState.getImageSize();
expect(imageSize).to.be.undefined; expect(imageSize).to.be.undefined;
}); });
@@ -643,12 +700,12 @@ describe('Model: selectionState', function () {
}); });
it('getImagePath() should return the image path', function () { it('getImagePath() should return the image path', function () {
const imagePath = selectionState.getImage()?.path; const imagePath = selectionState.getImagePath();
expect(imagePath).to.equal('foo.img'); expect(imagePath).to.equal('foo.img');
}); });
it('getImageSize() should return the image size', function () { it('getImageSize() should return the image size', function () {
const imageSize = selectionState.getImage()?.size; const imageSize = selectionState.getImageSize();
expect(imageSize).to.equal(999999999); expect(imageSize).to.equal(999999999);
}); });

View File

@@ -23,7 +23,7 @@ import * as settings from '../../../lib/gui/app/models/settings';
async function checkError(promise: Promise<any>, fn: (err: Error) => any) { async function checkError(promise: Promise<any>, fn: (err: Error) => any) {
try { try {
await promise; await promise;
} catch (error: any) { } catch (error) {
await fn(error); await fn(error);
return; return;
} }

View File

@@ -83,7 +83,7 @@ describe('Browser: imageWriter', () => {
imageWriter.flash(image, [fakeDrive], performWriteStub), imageWriter.flash(image, [fakeDrive], performWriteStub),
]); ]);
assert.fail('Writing twice should fail'); assert.fail('Writing twice should fail');
} catch (error: any) { } catch (error) {
expect(error.message).to.equal( expect(error.message).to.equal(
'There is already a flash in progress', 'There is already a flash in progress',
); );
@@ -133,7 +133,7 @@ describe('Browser: imageWriter', () => {
}); });
try { try {
await imageWriter.flash(image, [fakeDrive], performWriteStub); await imageWriter.flash(image, [fakeDrive], performWriteStub);
} catch (error: any) { } catch (error) {
expect(error).to.be.an.instanceof(Error); expect(error).to.be.an.instanceof(Error);
expect(error.message).to.equal('write error'); expect(error.message).to.equal('write error');
} }

View File

@@ -16,6 +16,7 @@
import { expect } from 'chai'; import { expect } from 'chai';
import * as settings from '../../../lib/gui/app/models/settings';
import * as progressStatus from '../../../lib/gui/app/modules/progress-status'; import * as progressStatus from '../../../lib/gui/app/modules/progress-status';
describe('Browser: progressStatus', function () { describe('Browser: progressStatus', function () {
@@ -29,6 +30,9 @@ describe('Browser: progressStatus', function () {
eta: 15, eta: 15,
speed: 100000000000000, speed: 100000000000000,
}; };
settings.set('unmountOnSuccess', true);
settings.set('validateWriteOnSuccess', true);
}); });
it('should report 0% if percentage == 0 but speed != 0', function () { it('should report 0% if percentage == 0 but speed != 0', function () {
@@ -37,14 +41,22 @@ describe('Browser: progressStatus', function () {
); );
}); });
it('should handle percentage == 0, flashing', function () { it('should handle percentage == 0, flashing, unmountOnSuccess', function () {
this.state.speed = 0; this.state.speed = 0;
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'0% Flashing...', '0% Flashing...',
); );
}); });
it('should handle percentage == 0, verifying', function () { it('should handle percentage == 0, flashing, !unmountOnSuccess', function () {
this.state.speed = 0;
settings.set('unmountOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'0% Flashing...',
);
});
it('should handle percentage == 0, verifying, unmountOnSuccess', function () {
this.state.speed = 0; this.state.speed = 0;
this.state.type = 'verifying'; this.state.type = 'verifying';
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
@@ -52,14 +64,31 @@ describe('Browser: progressStatus', function () {
); );
}); });
it('should handle percentage == 50, flashing', function () { it('should handle percentage == 0, verifying, !unmountOnSuccess', function () {
this.state.speed = 0;
this.state.type = 'verifying';
settings.set('unmountOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'0% Validating...',
);
});
it('should handle percentage == 50, flashing, unmountOnSuccess', function () {
this.state.percentage = 50; this.state.percentage = 50;
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'50% Flashing...', '50% Flashing...',
); );
}); });
it('should handle percentage == 50, verifying', function () { it('should handle percentage == 50, flashing, !unmountOnSuccess', function () {
this.state.percentage = 50;
settings.set('unmountOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'50% Flashing...',
);
});
it('should handle percentage == 50, verifying, unmountOnSuccess', function () {
this.state.percentage = 50; this.state.percentage = 50;
this.state.type = 'verifying'; this.state.type = 'verifying';
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
@@ -67,14 +96,40 @@ describe('Browser: progressStatus', function () {
); );
}); });
it('should handle percentage == 100, flashing', function () { it('should handle percentage == 50, verifying, !unmountOnSuccess', function () {
this.state.percentage = 50;
this.state.type = 'verifying';
settings.set('unmountOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'50% Validating...',
);
});
it('should handle percentage == 100, flashing, unmountOnSuccess, validateWriteOnSuccess', function () {
this.state.percentage = 100; this.state.percentage = 100;
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'Finishing...', 'Finishing...',
); );
}); });
it('should handle percentage == 100, verifying', function () { it('should handle percentage == 100, flashing, unmountOnSuccess, !validateWriteOnSuccess', function () {
this.state.percentage = 100;
settings.set('validateWriteOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'Finishing...',
);
});
it('should handle percentage == 100, flashing, !unmountOnSuccess, !validateWriteOnSuccess', function () {
this.state.percentage = 100;
settings.set('unmountOnSuccess', false);
settings.set('validateWriteOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'Finishing...',
);
});
it('should handle percentage == 100, verifying, unmountOnSuccess', function () {
this.state.percentage = 100; this.state.percentage = 100;
this.state.type = 'verifying'; this.state.type = 'verifying';
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
@@ -82,8 +137,9 @@ describe('Browser: progressStatus', function () {
); );
}); });
it('should handle percentage == 100, validating', function () { it('should handle percentage == 100, validatinf, !unmountOnSuccess', function () {
this.state.percentage = 100; this.state.percentage = 100;
settings.set('unmountOnSuccess', false);
expect(progressStatus.titleFromFlashState(this.state)).to.equal( expect(progressStatus.titleFromFlashState(this.state)).to.equal(
'Finishing...', 'Finishing...',
); );

View File

@@ -23,6 +23,37 @@ import * as constraints from '../../lib/shared/drive-constraints';
import * as messages from '../../lib/shared/messages'; import * as messages from '../../lib/shared/messages';
describe('Shared: DriveConstraints', function () { describe('Shared: DriveConstraints', function () {
describe('.isDriveLocked()', function () {
it('should return true if the drive is read-only', function () {
const result = constraints.isDriveLocked({
device: '/dev/disk2',
size: 999999999,
isReadOnly: true,
} as constraints.DrivelistDrive);
expect(result).to.be.true;
});
it('should return false if the drive is not read-only', function () {
const result = constraints.isDriveLocked({
device: '/dev/disk2',
size: 999999999,
isReadOnly: false,
} as constraints.DrivelistDrive);
expect(result).to.be.false;
});
it("should return false if we don't know if the drive is read-only", function () {
const result = constraints.isDriveLocked({
device: '/dev/disk2',
size: 999999999,
} as constraints.DrivelistDrive);
expect(result).to.be.false;
});
});
describe('.isSystemDrive()', function () { describe('.isSystemDrive()', function () {
it('should return true if the drive is a system drive', function () { it('should return true if the drive is a system drive', function () {
const result = constraints.isSystemDrive({ const result = constraints.isSystemDrive({
@@ -514,6 +545,40 @@ describe('Shared: DriveConstraints', function () {
}); });
}); });
describe('.isDriveDisabled()', function () {
it('should return true if the drive is disabled', function () {
const result = constraints.isDriveDisabled(({
device: '/dev/disk1',
size: 1000000000,
isReadOnly: false,
disabled: true,
} as unknown) as constraints.DrivelistDrive);
expect(result).to.be.true;
});
it('should return false if the drive is not disabled', function () {
const result = constraints.isDriveDisabled(({
device: '/dev/disk1',
size: 1000000000,
isReadOnly: false,
disabled: false,
} as unknown) as constraints.DrivelistDrive);
expect(result).to.be.false;
});
it('should return false if "disabled" is undefined', function () {
const result = constraints.isDriveDisabled({
device: '/dev/disk1',
size: 1000000000,
isReadOnly: false,
} as constraints.DrivelistDrive);
expect(result).to.be.false;
});
});
describe('.isDriveSizeRecommended()', function () { describe('.isDriveSizeRecommended()', function () {
const image: SourceMetadata = { const image: SourceMetadata = {
description: 'rpi.img', description: 'rpi.img',
@@ -680,7 +745,7 @@ describe('Shared: DriveConstraints', function () {
this.drive.disabled = false; this.drive.disabled = false;
}); });
it('should return false if the drive is not large enough and is the source drive', function () { it('should return false if the drive is not large enough and is a source drive', function () {
expect( expect(
constraints.isDriveValid(this.drive, { constraints.isDriveValid(this.drive, {
...image, ...image,
@@ -690,7 +755,7 @@ describe('Shared: DriveConstraints', function () {
).to.be.false; ).to.be.false;
}); });
it('should return false if the drive is not large enough and is not the source drive', function () { it('should return false if the drive is not large enough and is not a source drive', function () {
expect( expect(
constraints.isDriveValid(this.drive, { constraints.isDriveValid(this.drive, {
...image, ...image,
@@ -700,17 +765,17 @@ describe('Shared: DriveConstraints', function () {
).to.be.false; ).to.be.false;
}); });
it('should return true if the drive is large enough and is the source drive', function () { it('should return false if the drive is large enough and is a source drive', function () {
expect(constraints.isDriveValid(this.drive, image)).to.be.true; expect(constraints.isDriveValid(this.drive, image)).to.be.false;
}); });
it('should return true if the drive is large enough and is not the source drive', function () { it('should return false if the drive is large enough and is not a source drive', function () {
expect( expect(
constraints.isDriveValid(this.drive, { constraints.isDriveValid(this.drive, {
...image, ...image,
path: path.resolve(this.mountpoint, '../bar/rpi.img'), path: path.resolve(this.mountpoint, '../bar/rpi.img'),
}), }),
).to.be.true; ).to.be.false;
}); });
}); });
}); });
@@ -918,7 +983,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
expect(result).to.deep.equal([]); expect(result).to.deep.equal([]);
@@ -931,7 +995,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
const expectedTuples: Array<['WARNING' | 'ERROR', string]> = []; const expectedTuples: Array<['WARNING' | 'ERROR', string]> = [];
@@ -946,7 +1009,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = [['ERROR', 'containsImage']]; const expectedTuples = [['ERROR', 'containsImage']];
@@ -963,7 +1025,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
const expectedTuples = [['WARNING', 'system']]; const expectedTuples = [['WARNING', 'system']];
@@ -979,7 +1040,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
const expected = [ const expected = [
{ {
@@ -1000,7 +1060,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = []; const expectedTuples = [];
@@ -1017,7 +1076,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = [['ERROR', 'locked']]; const expectedTuples = [['ERROR', 'locked']];
@@ -1034,7 +1092,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = [['WARNING', 'sizeNotRecommended']]; const expectedTuples = [['WARNING', 'sizeNotRecommended']];
@@ -1051,7 +1108,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
const expectedTuples = [['WARNING', 'largeDrive']]; const expectedTuples = [['WARNING', 'largeDrive']];
@@ -1072,13 +1128,9 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = [ const expectedTuples = [['ERROR', 'locked']];
['ERROR', 'locked'],
['ERROR', 'containsImage'],
];
// @ts-ignore // @ts-ignore
expectStatusTypesAndMessagesToBe(result, expectedTuples); expectStatusTypesAndMessagesToBe(result, expectedTuples);
@@ -1092,7 +1144,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = [['ERROR', 'locked']]; const expectedTuples = [['ERROR', 'locked']];
@@ -1110,7 +1161,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
const expected = [ const expected = [
{ {
@@ -1131,7 +1181,6 @@ describe('Shared: DriveConstraints', function () {
const result = constraints.getDriveImageCompatibilityStatuses( const result = constraints.getDriveImageCompatibilityStatuses(
this.drive, this.drive,
this.image, this.image,
true,
); );
// @ts-ignore // @ts-ignore
const expectedTuples = [ const expectedTuples = [
@@ -1158,7 +1207,7 @@ describe('Shared: DriveConstraints', function () {
'/dev/disk6', '/dev/disk6',
]; ];
const drives = [ const drives = [
{ ({
device: drivePaths[0], device: drivePaths[0],
description: 'My Drive', description: 'My Drive',
size: 123456789, size: 123456789,
@@ -1166,8 +1215,8 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [{ path: __dirname }], mountpoints: [{ path: __dirname }],
isSystem: false, isSystem: false,
isReadOnly: false, isReadOnly: false,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
{ ({
device: drivePaths[1], device: drivePaths[1],
description: 'My Other Drive', description: 'My Other Drive',
size: 123456789, size: 123456789,
@@ -1175,8 +1224,8 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [], mountpoints: [],
isSystem: false, isSystem: false,
isReadOnly: true, isReadOnly: true,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
{ ({
device: drivePaths[2], device: drivePaths[2],
description: 'My Drive', description: 'My Drive',
size: 1234567, size: 1234567,
@@ -1184,8 +1233,8 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [], mountpoints: [],
isSystem: false, isSystem: false,
isReadOnly: false, isReadOnly: false,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
{ ({
device: drivePaths[3], device: drivePaths[3],
description: 'My Drive', description: 'My Drive',
size: 123456789, size: 123456789,
@@ -1193,8 +1242,8 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [], mountpoints: [],
isSystem: true, isSystem: true,
isReadOnly: false, isReadOnly: false,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
{ ({
device: drivePaths[4], device: drivePaths[4],
description: 'My Drive', description: 'My Drive',
size: 128000000001, size: 128000000001,
@@ -1202,8 +1251,8 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [], mountpoints: [],
isSystem: false, isSystem: false,
isReadOnly: false, isReadOnly: false,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
{ ({
device: drivePaths[5], device: drivePaths[5],
description: 'My Drive', description: 'My Drive',
size: 12345678, size: 12345678,
@@ -1211,8 +1260,8 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [], mountpoints: [],
isSystem: false, isSystem: false,
isReadOnly: false, isReadOnly: false,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
{ ({
device: drivePaths[6], device: drivePaths[6],
description: 'My Drive', description: 'My Drive',
size: 123456789, size: 123456789,
@@ -1220,7 +1269,7 @@ describe('Shared: DriveConstraints', function () {
mountpoints: [], mountpoints: [],
isSystem: false, isSystem: false,
isReadOnly: false, isReadOnly: false,
} as unknown as constraints.DrivelistDrive, } as unknown) as constraints.DrivelistDrive,
]; ];
const image: SourceMetadata = { const image: SourceMetadata = {
@@ -1238,7 +1287,7 @@ describe('Shared: DriveConstraints', function () {
describe('given no drives', function () { describe('given no drives', function () {
it('should return no statuses', function () { it('should return no statuses', function () {
expect( expect(
constraints.getListDriveImageCompatibilityStatuses([], image, true), constraints.getListDriveImageCompatibilityStatuses([], image),
).to.deep.equal([]); ).to.deep.equal([]);
}); });
}); });
@@ -1249,7 +1298,6 @@ describe('Shared: DriveConstraints', function () {
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(
[drives[0]], [drives[0]],
image, image,
true,
), ),
).to.deep.equal([ ).to.deep.equal([
{ {
@@ -1264,7 +1312,6 @@ describe('Shared: DriveConstraints', function () {
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(
[drives[1]], [drives[1]],
image, image,
true,
), ),
).to.deep.equal([ ).to.deep.equal([
{ {
@@ -1279,7 +1326,6 @@ describe('Shared: DriveConstraints', function () {
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(
[drives[2]], [drives[2]],
image, image,
true,
), ),
).to.deep.equal([ ).to.deep.equal([
{ {
@@ -1294,7 +1340,6 @@ describe('Shared: DriveConstraints', function () {
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(
[drives[3]], [drives[3]],
image, image,
true,
), ),
).to.deep.equal([ ).to.deep.equal([
{ {
@@ -1309,7 +1354,6 @@ describe('Shared: DriveConstraints', function () {
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(
[drives[4]], [drives[4]],
image, image,
true,
), ),
).to.deep.equal([ ).to.deep.equal([
{ {
@@ -1324,7 +1368,6 @@ describe('Shared: DriveConstraints', function () {
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(
[drives[5]], [drives[5]],
image, image,
true,
), ),
).to.deep.equal([ ).to.deep.equal([
{ {
@@ -1338,11 +1381,7 @@ describe('Shared: DriveConstraints', function () {
describe('given multiple drives with all warnings/errors', function () { describe('given multiple drives with all warnings/errors', function () {
it('should return all statuses', function () { it('should return all statuses', function () {
expect( expect(
constraints.getListDriveImageCompatibilityStatuses( constraints.getListDriveImageCompatibilityStatuses(drives, image),
drives,
image,
true,
),
).to.deep.equal([ ).to.deep.equal([
{ {
message: 'Source drive', message: 'Source drive',

View File

@@ -30,8 +30,9 @@ describe('Shared: SupportedFormats', function () {
], ],
(imagePath) => { (imagePath) => {
it(`should return true if filename is ${imagePath}`, function () { it(`should return true if filename is ${imagePath}`, function () {
const looksLikeWindowsImage = const looksLikeWindowsImage = supportedFormats.looksLikeWindowsImage(
supportedFormats.looksLikeWindowsImage(imagePath); imagePath,
);
expect(looksLikeWindowsImage).to.be.true; expect(looksLikeWindowsImage).to.be.true;
}); });
}, },
@@ -44,8 +45,9 @@ describe('Shared: SupportedFormats', function () {
], ],
(imagePath) => { (imagePath) => {
it(`should return false if filename is ${imagePath}`, function () { it(`should return false if filename is ${imagePath}`, function () {
const looksLikeWindowsImage = const looksLikeWindowsImage = supportedFormats.looksLikeWindowsImage(
supportedFormats.looksLikeWindowsImage(imagePath); imagePath,
);
expect(looksLikeWindowsImage).to.be.false; expect(looksLikeWindowsImage).to.be.false;
}); });
}, },

View File

@@ -15,20 +15,15 @@
*/ */
import { expect } from 'chai'; import { expect } from 'chai';
import { platform } from 'os';
import { Application } from 'spectron'; import { Application } from 'spectron';
import * as electronPath from 'electron'; import * as electronPath from 'electron';
// TODO: spectron fails to start on the CI with: describe('Spectron', function () {
// Error: Failed to create session.
// unknown error: Chrome failed to start: exited abnormally
if (platform() !== 'darwin') {
describe('Spectron', function () {
// Mainly for CI jobs // Mainly for CI jobs
this.timeout(40000); this.timeout(40000);
const app = new Application({ const app = new Application({
path: electronPath as unknown as string, path: (electronPath as unknown) as string,
args: ['--no-sandbox', '.'], args: ['--no-sandbox', '.'],
}); });
@@ -51,16 +46,12 @@ if (platform() !== 'darwin') {
expect(bounds.height).to.be.above(0); expect(bounds.height).to.be.above(0);
expect(bounds.width).to.be.above(0); expect(bounds.width).to.be.above(0);
expect(await app.browserWindow.isMinimized()).to.be.false; expect(await app.browserWindow.isMinimized()).to.be.false;
expect( expect(await app.browserWindow.isVisible()).to.be.true;
(await app.browserWindow.isVisible()) ||
(await app.browserWindow.isFocused()),
).to.be.true;
}); });
it('should set a proper title', async () => { it('should set a proper title', async () => {
// @ts-ignore (SpectronClient.getTitle exists) // @ts-ignore (SpectronClient.getTitle exists)
return expect(await app.client.getTitle()).to.equal('balenaEtcher'); return expect(await app.client.getTitle()).to.equal('Etcher');
}); });
}); });
}); });
}

View File

@@ -1,21 +1,12 @@
{ {
"compilerOptions": { "compilerOptions": {
"strict": true, "strict": true,
"target": "es2019",
"typeRoots": ["./node_modules/@types", "./typings"],
"module": "commonjs",
"lib": ["dom", "esnext"],
"declaration": true,
"declarationMap": true,
"jsx": "react",
"pretty": true,
"sourceMap": true,
"noUnusedLocals": true, "noUnusedLocals": true,
"noUnusedParameters": true, "noUnusedParameters": true,
"noImplicitReturns": true,
"noFallthroughCasesInSwitch": true,
"moduleResolution": "node",
"allowSyntheticDefaultImports": true,
"resolveJsonModule": true, "resolveJsonModule": true,
"target": "es2019",
"moduleResolution": "node",
"jsx": "react",
"typeRoots": ["./node_modules/@types", "./typings"]
} }
} }

View File

@@ -10,16 +10,7 @@
"jsx": "react", "jsx": "react",
"typeRoots": ["./node_modules/@types", "./typings"], "typeRoots": ["./node_modules/@types", "./typings"],
"importHelpers": true, "importHelpers": true,
"allowSyntheticDefaultImports": true, "allowSyntheticDefaultImports": true
"lib": ["dom", "esnext"],
"declaration": true,
"declarationMap": true,
"pretty": true,
"sourceMap": true,
"baseUrl": "./src",
"noImplicitReturns": true,
"noFallthroughCasesInSwitch": true,
"allowJs": true
}, },
"include": [ "include": [
"lib/**/*.ts", "lib/**/*.ts",

View File

@@ -1 +0,0 @@
declare module 'pnp-webpack-plugin';

View File

@@ -1 +1 @@
declare module '@balena/sudo-prompt'; declare module 'sudo-prompt';

View File

@@ -17,6 +17,7 @@
import * as CopyPlugin from 'copy-webpack-plugin'; import * as CopyPlugin from 'copy-webpack-plugin';
import { readdirSync } from 'fs'; import { readdirSync } from 'fs';
import * as _ from 'lodash'; import * as _ from 'lodash';
import * as MiniCssExtractPlugin from 'mini-css-extract-plugin';
import * as os from 'os'; import * as os from 'os';
import outdent from 'outdent'; import outdent from 'outdent';
import * as path from 'path'; import * as path from 'path';
@@ -24,9 +25,6 @@ import { env } from 'process';
import * as SimpleProgressWebpackPlugin from 'simple-progress-webpack-plugin'; import * as SimpleProgressWebpackPlugin from 'simple-progress-webpack-plugin';
import * as TerserPlugin from 'terser-webpack-plugin'; import * as TerserPlugin from 'terser-webpack-plugin';
import { BannerPlugin, NormalModuleReplacementPlugin } from 'webpack'; import { BannerPlugin, NormalModuleReplacementPlugin } from 'webpack';
import * as PnpWebpackPlugin from 'pnp-webpack-plugin';
import * as tsconfigRaw from './tsconfig.webpack.json';
/** /**
* Don't webpack package.json as mixpanel & sentry tokens * Don't webpack package.json as mixpanel & sentry tokens
@@ -34,7 +32,8 @@ import * as tsconfigRaw from './tsconfig.webpack.json';
*/ */
function externalPackageJson(packageJsonPath: string) { function externalPackageJson(packageJsonPath: string) {
return ( return (
{ request }: { context: string; request: string }, _context: string,
request: string,
callback: (error?: Error | null, result?: string) => void, callback: (error?: Error | null, result?: string) => void,
) => { ) => {
if (_.endsWith(request, 'package.json')) { if (_.endsWith(request, 'package.json')) {
@@ -51,7 +50,8 @@ function platformSpecificModule(
) { ) {
// Resolves module on platform, otherwise resolves the replacement // Resolves module on platform, otherwise resolves the replacement
return ( return (
{ request }: { context: string; request: string }, _context: string,
request: string,
callback: (error?: Error, result?: string, type?: string) => void, callback: (error?: Error, result?: string, type?: string) => void,
) => { ) => {
if (request === module && os.platform() !== platform) { if (request === module && os.platform() !== platform) {
@@ -70,8 +70,6 @@ function renameNodeModules(resourcePath: string) {
path path
.relative(__dirname, resourcePath) .relative(__dirname, resourcePath)
.replace('node_modules', 'modules') .replace('node_modules', 'modules')
// use the same name on all architectures so electron-builder can build a universal dmg on mac
.replace(LZMA_BINDINGS_FOLDER, LZMA_BINDINGS_FOLDER_RENAMED)
// file-loader expects posix paths, even on Windows // file-loader expects posix paths, even on Windows
.replace(/\\/g, '/') .replace(/\\/g, '/')
); );
@@ -91,7 +89,6 @@ function findLzmaNativeBindingsFolder(): string {
} }
const LZMA_BINDINGS_FOLDER = findLzmaNativeBindingsFolder(); const LZMA_BINDINGS_FOLDER = findLzmaNativeBindingsFolder();
const LZMA_BINDINGS_FOLDER_RENAMED = 'binding';
interface ReplacementRule { interface ReplacementRule {
search: string; search: string;
@@ -111,45 +108,19 @@ function replace(test: RegExp, ...replacements: ReplacementRule[]) {
}; };
} }
function fetchWasm(...where: string[]) {
const whereStr = where.map((x) => `'${x}'`).join(', ');
return outdent`
const Path = require('path');
let electron;
try {
// This doesn't exist in the child-writer
electron = require('electron');
} catch {
}
function appPath() {
return Path.isAbsolute(__dirname) ?
__dirname :
Path.join(
// With macOS universal builds, getAppPath() returns the path to an app.asar file containing an index.js file which will
// include the app-x64 or app-arm64 folder depending on the arch.
// We don't care about the app.asar file, we want the actual folder.
electron.remote.app.getAppPath().replace(/\\.asar$/, () => process.platform === 'darwin' ? '-' + process.arch : ''),
'generated'
);
}
scriptDirectory = Path.join(appPath(), 'modules', ${whereStr}, '/');
`;
}
const commonConfig = { const commonConfig = {
mode: 'production', mode: 'production',
optimization: { optimization: {
moduleIds: 'natural',
minimize: true, minimize: true,
minimizer: [ minimizer: [
new TerserPlugin({ new TerserPlugin({
parallel: true,
terserOptions: { terserOptions: {
compress: false, compress: false,
mangle: false, mangle: false,
format: { output: {
beautify: true,
comments: false, comments: false,
ecma: 2020, ecma: 2018,
}, },
}, },
extractComments: false, extractComments: false,
@@ -160,12 +131,7 @@ const commonConfig = {
rules: [ rules: [
{ {
test: /\.css$/, test: /\.css$/,
use: ['style-loader', 'css-loader'], use: 'css-loader',
},
{
test: /\.(woff|woff2|eot|ttf|otf)$/,
loader: 'file-loader',
options: { name: renameNodeModules },
}, },
{ {
test: /\.svg$/, test: /\.svg$/,
@@ -175,11 +141,9 @@ const commonConfig = {
test: /\.tsx?$/, test: /\.tsx?$/,
use: [ use: [
{ {
loader: 'esbuild-loader', loader: 'ts-loader',
options: { options: {
loader: 'tsx', configFile: 'tsconfig.webpack.json',
target: 'es2021',
tsconfigRaw,
}, },
}, },
], ],
@@ -211,7 +175,12 @@ const commonConfig = {
// remove node-pre-gyp magic from lzma-native // remove node-pre-gyp magic from lzma-native
{ {
search: 'require(binding_path)', search: 'require(binding_path)',
replace: `require('./${LZMA_BINDINGS_FOLDER}/lzma_native.node')`, replace: () => {
return `require('./${path.posix.join(
LZMA_BINDINGS_FOLDER,
'lzma_native.node',
)}')`;
},
}, },
// use regular stream module instead of readable-stream // use regular stream module instead of readable-stream
{ {
@@ -224,6 +193,11 @@ const commonConfig = {
search: 'require(binding_path)', search: 'require(binding_path)',
replace: "require('./build/Release/usb_bindings.node')", replace: "require('./build/Release/usb_bindings.node')",
}), }),
// remove bindings magic from ext2fs
replace(/node_modules\/ext2fs\/lib\/(ext2fs|binding)\.js$/, {
search: "require('bindings')('bindings')",
replace: "require('../build/Release/bindings.node')",
}),
// remove bindings magic from mountutils // remove bindings magic from mountutils
replace(/node_modules\/mountutils\/index\.js$/, { replace(/node_modules\/mountutils\/index\.js$/, {
search: outdent` search: outdent`
@@ -255,33 +229,9 @@ const commonConfig = {
"return await readFile(Path.join(__dirname, '..', 'blobs', filename));", "return await readFile(Path.join(__dirname, '..', 'blobs', filename));",
replace: outdent` replace: outdent`
const { app, remote } = require('electron'); const { app, remote } = require('electron');
return await readFile( return await readFile(Path.join((app || remote.app).getAppPath(), 'generated', __dirname.replace('node_modules', 'modules'), '..', 'blobs', filename));
Path.join(
// With macOS universal builds, getAppPath() returns the path to an app.asar file containing an index.js file which will
// include the app-x64 or app-arm64 folder depending on the arch.
// We don't care about the app.asar file, we want the actual folder.
(app || remote.app).getAppPath().replace(/\\.asar$/, () => process.platform === 'darwin' ? '-' + process.arch : ''),
'generated',
__dirname.replace('node_modules', 'modules'),
'..',
'blobs',
filename
)
);
`, `,
}), }),
// Use the libext2fs.wasm file in the generated folder
// The way to find the app directory depends on whether we run in the renderer or in the child-writer
// We use __dirname in the child-writer and electron.remote.app.getAppPath() in the renderer
replace(/node_modules\/ext2fs\/lib\/libext2fs\.js$/, {
search: 'scriptDirectory=__dirname+"/"',
replace: fetchWasm('ext2fs', 'lib'),
}),
// Same for node-crc-utils
replace(/node_modules\/@balena\/node-crc-utils\/crc32\.js$/, {
search: 'scriptDirectory=__dirname+"/"',
replace: fetchWasm('@balena', 'node-crc-utils'),
}),
// Copy native modules to generated folder // Copy native modules to generated folder
{ {
test: /\.node$/, test: /\.node$/,
@@ -298,20 +248,16 @@ const commonConfig = {
extensions: ['.node', '.js', '.json', '.ts', '.tsx'], extensions: ['.node', '.js', '.json', '.ts', '.tsx'],
}, },
plugins: [ plugins: [
PnpWebpackPlugin,
new SimpleProgressWebpackPlugin({ new SimpleProgressWebpackPlugin({
format: process.env.WEBPACK_PROGRESS || 'verbose', format: process.env.WEBPACK_PROGRESS || 'verbose',
}), }),
// Force axios to use http.js, not xhr.js as we need stream support // Force axios to use http.js, not xhr.js as we need stream support
// (its package.json file replaces http with xhr for browser targets). // (it's package.json file replaces http with xhr for browser targets).
new NormalModuleReplacementPlugin( new NormalModuleReplacementPlugin(
slashOrAntislash(/node_modules\/axios\/lib\/adapters\/xhr\.js/), slashOrAntislash(/node_modules\/axios\/lib\/adapters\/xhr\.js/),
'./http.js', './http.js',
), ),
], ],
resolveLoader: {
plugins: [PnpWebpackPlugin.moduleLoader(module)],
},
output: { output: {
path: path.join(__dirname, 'generated'), path: path.join(__dirname, 'generated'),
filename: '[name].js', filename: '[name].js',
@@ -335,21 +281,13 @@ const guiConfigCopyPatterns = [
from: 'node_modules/node-raspberrypi-usbboot/blobs', from: 'node_modules/node-raspberrypi-usbboot/blobs',
to: 'modules/node-raspberrypi-usbboot/blobs', to: 'modules/node-raspberrypi-usbboot/blobs',
}, },
{
from: 'node_modules/ext2fs/lib/libext2fs.wasm',
to: 'modules/ext2fs/lib/libext2fs.wasm',
},
{
from: 'node_modules/@balena/node-crc-utils/crc32.wasm',
to: 'modules/@balena/node-crc-utils/crc32.wasm',
},
]; ];
if (os.platform() === 'win32') { if (os.platform() === 'win32') {
// liblzma.dll is required on Windows for lzma-native // liblzma.dll is required on Windows for lzma-native
guiConfigCopyPatterns.push({ guiConfigCopyPatterns.push({
from: `node_modules/lzma-native/${LZMA_BINDINGS_FOLDER}/liblzma.dll`, from: `node_modules/lzma-native/${LZMA_BINDINGS_FOLDER}/liblzma.dll`,
to: `modules/lzma-native/${LZMA_BINDINGS_FOLDER_RENAMED}/liblzma.dll`, to: `modules/lzma-native/${LZMA_BINDINGS_FOLDER}/liblzma.dll`,
}); });
} }
@@ -361,19 +299,10 @@ const guiConfig = {
__filename: true, __filename: true,
}, },
entry: { entry: {
gui: path.join(__dirname, 'lib', 'gui', 'app', 'renderer.ts'), gui: path.join(__dirname, 'lib', 'gui', 'app', 'app.ts'),
}, },
// entry: path.join(__dirname, 'lib', 'gui', 'app', 'renderer.ts'),
plugins: [ plugins: [
...commonConfig.plugins, ...commonConfig.plugins,
new CopyPlugin({
patterns: [
{ from: 'lib/gui/app/index.html', to: 'index.html' },
// electron-builder doesn't bundle folders named "assets"
// See https://github.com/electron-userland/electron-builder/issues/4545
{ from: 'assets/icon.png', to: 'media/icon.png' },
],
}),
// Remove "Download the React DevTools for a better development experience" message // Remove "Download the React DevTools for a better development experience" message
new BannerPlugin({ new BannerPlugin({
banner: '__REACT_DEVTOOLS_GLOBAL_HOOK__ = { isDisabled: true };', banner: '__REACT_DEVTOOLS_GLOBAL_HOOK__ = { isDisabled: true };',
@@ -412,4 +341,41 @@ const childWriterConfig = {
}, },
}; };
export default [guiConfig, etcherConfig, childWriterConfig]; const cssConfig = {
mode: 'production',
optimization: {
minimize: false,
},
module: {
rules: [
{
test: /\.css$/i,
use: [MiniCssExtractPlugin.loader, 'css-loader'],
},
{
test: /\.(woff|woff2|eot|ttf|otf|svg)$/,
loader: 'file-loader',
options: { name: renameNodeModules },
},
],
},
plugins: [
new MiniCssExtractPlugin({ filename: '[name].css' }),
new CopyPlugin({
patterns: [
{ from: 'lib/gui/app/index.html', to: 'index.html' },
// electron-builder doesn't bundle folders named "assets"
// See https://github.com/electron-userland/electron-builder/issues/4545
{ from: 'assets/icon.png', to: 'media/icon.png' },
],
}),
],
entry: {
index: path.join(__dirname, 'lib', 'gui', 'app', 'css', 'main.css'),
},
output: {
path: path.join(__dirname, 'generated'),
},
};
module.exports = [cssConfig, guiConfig, etcherConfig, childWriterConfig];

View File

@@ -1,24 +0,0 @@
import configs from './webpack.config';
import { WebpackOptionsNormalized } from 'webpack';
import * as fs from 'fs';
const [
guiConfig,
etcherConfig,
childWriterConfig,
] = (configs as unknown) as WebpackOptionsNormalized[];
configs.forEach((config) => {
config.mode = 'development';
// @ts-ignore
config.devtool = 'source-map';
});
guiConfig.devServer = {
hot: true,
port: 3030,
};
fs.copyFileSync('./lib/gui/app/index.dev.html', './generated/index.html');
export default [guiConfig, etcherConfig, childWriterConfig];