Browse Source

Merge branch 'develop' into qrstream

gre-patch-1
Gaëtan Renaudeau 6 years ago
parent
commit
c331f0ece2
No known key found for this signature in database GPG Key ID: 7B66B85F042E5451
  1. 13
      .circleci/config.yml
  2. 6
      package.json
  3. 6
      src/components/ExchangePage/index.js
  4. 1
      src/config/urls.js
  5. 3
      src/helpers/anonymizer.js
  6. 25
      src/main/app.js
  7. 3
      static/i18n/en/app.json
  8. 13
      static/images/logos/exchanges/kyber.svg
  9. 93
      test-e2e/password-lock-check.spec.js

13
.circleci/config.yml

@ -10,13 +10,22 @@ jobs:
<<: *defaults <<: *defaults
steps: steps:
- run: sudo apt-get install -y libudev-dev - run: sudo apt-get install -y libudev-dev
- run:
name: Install latest yarn
command: |
curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz" ;
sudo mkdir -p /opt ;
sudo tar -xzf yarn-v$YARN_VERSION.tar.gz -C /opt/ ;
sudo ln -sf /opt/yarn-v$YARN_VERSION/bin/yarn /usr/local/bin/yarn ;
sudo ln -sf /opt/yarn-v$YARN_VERSION/bin/yarnpkg /usr/local/bin/yarnpkg ;
rm yarn-v$YARN_VERSION.tar.gz
- checkout - checkout
- restore_cache: - restore_cache:
keys: keys:
- v10-yarn-packages-{{ checksum "yarn.lock" }} - v11-yarn-packages-{{ checksum "yarn.lock" }}
- run: yarn install - run: yarn install
- save_cache: - save_cache:
key: v10-yarn-packages-{{ checksum "yarn.lock" }} key: v11-yarn-packages-{{ checksum "yarn.lock" }}
paths: paths:
- node_modules - node_modules
- run: yarn lint - run: yarn lint

6
package.json

@ -3,7 +3,7 @@
"productName": "Ledger Live", "productName": "Ledger Live",
"description": "Ledger Live - Desktop", "description": "Ledger Live - Desktop",
"repository": "https://github.com/LedgerHQ/ledger-live-desktop", "repository": "https://github.com/LedgerHQ/ledger-live-desktop",
"version": "1.1.11", "version": "1.2.0",
"author": "Ledger", "author": "Ledger",
"license": "MIT", "license": "MIT",
"scripts": { "scripts": {
@ -167,5 +167,9 @@
"webpack-cli": "^2.0.14", "webpack-cli": "^2.0.14",
"yaml-loader": "^0.5.0" "yaml-loader": "^0.5.0"
}, },
"engines": {
"node": ">=8.9.0 <=8.12.0",
"yarn": "^1.10.1"
},
"private": true "private": true
} }

6
src/components/ExchangePage/index.js

@ -71,6 +71,12 @@ const cards = shuffle([
url: urls.genesis, url: urls.genesis,
logo: <img src={i('logos/exchanges/genesis.svg')} alt="Genesis" width={150} />, logo: <img src={i('logos/exchanges/genesis.svg')} alt="Genesis" width={150} />,
}, },
{
key: 'kyber',
id: 'kyber',
url: urls.kyber,
logo: <img src={i('logos/exchanges/kyber.svg')} alt="KYBER" width={150} />,
},
]) ])
class ExchangePage extends PureComponent<Props> { class ExchangePage extends PureComponent<Props> {

1
src/config/urls.js

@ -30,6 +30,7 @@ export const urls = {
luno: 'http://luno.go2cloud.org/aff_c?offer_id=4&aff_id=1001&source=ledger', luno: 'http://luno.go2cloud.org/aff_c?offer_id=4&aff_id=1001&source=ledger',
shapeshift: 'https://shapeshift.io/#/coins?affiliate=ledger', shapeshift: 'https://shapeshift.io/#/coins?affiliate=ledger',
genesis: 'https://genesistrading.com/ledger-live/', genesis: 'https://genesistrading.com/ledger-live/',
kyber: 'http://kyber.network/swap?ref=0xE2D8481eeF31CDA994833974FFfEccd576f8D71E',
// Errors // Errors
errors: { errors: {

3
src/helpers/anonymizer.js

@ -46,6 +46,9 @@ function filepathRecursiveReplacer(obj: mixed, seen: Array<*>) {
} }
} }
} else { } else {
if (obj instanceof Error) {
obj.message = filepathReplace(obj.message)
}
for (const k in obj) { for (const k in obj) {
if (typeof obj.hasOwnProperty === 'function' && obj.hasOwnProperty(k)) { if (typeof obj.hasOwnProperty === 'function' && obj.hasOwnProperty(k)) {
const value = obj[k] const value = obj[k]

25
src/main/app.js

@ -65,21 +65,15 @@ const getDefaultUrl = () =>
__DEV__ ? `http://localhost:${ELECTRON_WEBPACK_WDS_PORT || ''}` : `file://${__dirname}/index.html` __DEV__ ? `http://localhost:${ELECTRON_WEBPACK_WDS_PORT || ''}` : `file://${__dirname}/index.html`
const saveWindowSettings = window => { const saveWindowSettings = window => {
window.on( const windowParamsHandler = () => {
'resize',
debounce(() => {
const [width, height] = window.getSize() const [width, height] = window.getSize()
db.setKey('windowParams', `${window.name}.dimensions`, { width, height })
}, 100),
)
window.on(
'move',
debounce(() => {
const [x, y] = window.getPosition() const [x, y] = window.getPosition()
db.setKey('windowParams', `${window.name}.dimensions`, { width, height })
db.setKey('windowParams', `${window.name}.positions`, { x, y }) db.setKey('windowParams', `${window.name}.positions`, { x, y })
}, 100), }
)
window.on('resize', debounce(windowParamsHandler, 100))
window.on('move', debounce(windowParamsHandler, 100))
} }
const defaultWindowOptions = { const defaultWindowOptions = {
@ -95,8 +89,8 @@ const defaultWindowOptions = {
} }
async function createMainWindow() { async function createMainWindow() {
const savedDimensions = await db.getKey('app', 'MainWindow.dimensions', {}) const savedDimensions = await db.getKey('windowParams', 'MainWindow.dimensions', {})
const savedPositions = await db.getKey('app', 'MainWindow.positions', null) const savedPositions = await db.getKey('windowParams', 'MainWindow.positions', null)
const width = savedDimensions.width || DEFAULT_WINDOW_WIDTH const width = savedDimensions.width || DEFAULT_WINDOW_WIDTH
const height = savedDimensions.height || DEFAULT_WINDOW_HEIGHT const height = savedDimensions.height || DEFAULT_WINDOW_HEIGHT
@ -121,7 +115,6 @@ async function createMainWindow() {
const window = new BrowserWindow(windowOptions) const window = new BrowserWindow(windowOptions)
window.name = 'MainWindow' window.name = 'MainWindow'
const url = getDefaultUrl() const url = getDefaultUrl()
if (devTools) { if (devTools) {
@ -172,8 +165,6 @@ const installExtensions = async () => {
).catch(console.log) // eslint-disable-line ).catch(console.log) // eslint-disable-line
} }
app.setAsDefaultProtocolClient('ledgerhq')
app.on('ready', async () => { app.on('ready', async () => {
if (__DEV__) { if (__DEV__) {
await installExtensions() await installExtensions()

3
static/i18n/en/app.json

@ -169,7 +169,8 @@
"paybis": "it is safe and easy to Buy Bitcoin with credit card from PayBis. Service operates in US, Canada, Germany, Russia and Saudi Arabia.", "paybis": "it is safe and easy to Buy Bitcoin with credit card from PayBis. Service operates in US, Canada, Germany, Russia and Saudi Arabia.",
"luno": "Luno makes it safe and easy to buy, store and learn about cryptocurrencies like Bitcoin and Ethereum", "luno": "Luno makes it safe and easy to buy, store and learn about cryptocurrencies like Bitcoin and Ethereum",
"shapeshift": "ShapeShift is an online marketplace where users can buy and sell digital assets. It is a fast and secure way for the world to buy and sell digital assets, with no lengthy signup process, no counterparty risk, and no friction.", "shapeshift": "ShapeShift is an online marketplace where users can buy and sell digital assets. It is a fast and secure way for the world to buy and sell digital assets, with no lengthy signup process, no counterparty risk, and no friction.",
"genesis": "Genesis is an institutional trading firm offering liquidity and borrow for digital currencies, including bitcoin, bitcoin cash, ethereum, ethereum classic, litecoin, and XRP." "genesis": "Genesis is an institutional trading firm offering liquidity and borrow for digital currencies, including bitcoin, bitcoin cash, ethereum, ethereum classic, litecoin, and XRP.",
"kyber": "KYBER, his a trading platform for exchange and conversion of ERC-20 tokens"
}, },
"genuinecheck": { "genuinecheck": {
"modal": { "modal": {

13
static/images/logos/exchanges/kyber.svg

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 9.9 KiB

93
test-e2e/password-lock-check.spec.js

@ -0,0 +1,93 @@
import { Application } from 'spectron'
import { waitForDisappear, waitForExpectedText } from './helpers'
const os = require('os')
const path = require('path')
const fs = require('fs')
const appVersion = require('../package.json')
let app
const TIMEOUT = 50 * 1000
let appPath
let configPath
const platform = os.platform()
if (platform === 'darwin') {
appPath = `./dist/mac/Ledger Live.app/Contents/MacOS/Ledger Live`
configPath = `${os.homedir()}/Library/Application Support/Ledger Live/`
} else if (platform === 'win32') {
appPath = `.\\dist\\win-unpacked\\Ledger Live.exe`
configPath = '%AppData\\Roaming\\Ledger Live'
} else {
appPath = `./dist/ledger-live-desktop-${appVersion.version}-linux-x86_64.AppImage`
configPath = '$HOME/apps/ledger-live-desktop-$ledgerLiveVersion-linux-x86_64.AppImage'
}
describe('Application launch', () => {
beforeEach(async () => {
app = new Application({
path: appPath,
env: {
SKIP_ONBOARDING: '1',
},
})
await app.start()
}, TIMEOUT)
afterEach(async () => {
if (app && app.isRunning()) {
await app.stop()
}
}, TIMEOUT)
test(
'Start app, activate password lock, check app.json, deactivate password lock',
async () => {
const title = await app.client.getTitle()
expect(title).toEqual('Ledger Live')
await app.client.waitUntilWindowLoaded()
await waitForDisappear(app, '#preload')
// Verify Account summary text
// Count user's accounts
const userAccountsList = await app.client.elements('[data-e2e=dashboard_AccountCardWrapper]')
const userAccountsCount = Object.keys(userAccountsList.value).length
// Check account number
const accountSummary = await app.client.getText('[data-e2e=dashboard_accountsSummaryDesc]')
const accountSummaryMessage = `Here's the summary of your ${userAccountsCount} accounts`
expect(accountSummary).toEqual(accountSummaryMessage)
// Go to settings
await app.client.click('[data-e2e=setting_button]')
await waitForExpectedText(app, '[data-e2e=settings_title]', 'Settings')
// Enable lock password
await app.client.click('[data-e2e=passwordLock_button]')
await waitForExpectedText(app, '[data-e2e=setPassword_modalTitle]', 'Set a password')
await app.client.setValue('[data-e2e=setPassword_NewPassword]', 5)
await app.client.setValue('[data-e2e=setPassword_ConfirmPassword]', 5)
await app.client.keys('Enter')
await waitForExpectedText(app, '[data-e2e=settings_title]', 'Settings')
await app.client.pause(2000)
// Verify in app.json that accounts data are encrypted
const tmpAppJSONPath = path.resolve(configPath, 'app.json')
const LockedfileContent = fs.readFileSync(tmpAppJSONPath, 'utf-8')
const accountsOperations = '"operations":[{'
await expect(LockedfileContent).not.toContain(accountsOperations)
// Disable password lock
await app.client.click('[data-e2e=passwordLock_button]')
await waitForExpectedText(app, '[data-e2e=modal_title]', 'Disable password lock')
await app.client.setValue('#password', 5)
await app.client.pause(500)
await app.client.keys('Enter')
await waitForExpectedText(app, '[data-e2e=settings_title]', 'Settings')
await app.client.pause(3000)
const UnlockedfileContent = fs.readFileSync(tmpAppJSONPath, 'utf-8')
// Verify in app.json that accounts data are not encrypted
await expect(UnlockedfileContent).toContain(accountsOperations)
await app.client.pause(1000)
},
TIMEOUT,
)
})
Loading…
Cancel
Save