Browse Source

Merge pull request #247 from meriadec/master

Fetch installable apps from api instead of currency list
master
Meriadec Pillet 7 years ago
committed by GitHub
parent
commit
e42d8e6853
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 27
      src/components/ManagerPage/ManagerApp.js
  2. 65
      src/components/ManagerPage/index.js
  3. 11
      src/internals/usb/manager/constants.js
  4. 49
      src/internals/usb/manager/helpers.js
  5. 10
      src/internals/usb/manager/index.js
  6. 2
      src/renderer/events.js

27
src/components/ManagerPage/ManagerApp.js

@ -2,9 +2,6 @@
import React from 'react'
import styled from 'styled-components'
import { getIconByCoinType } from '@ledgerhq/currencies/react'
import type { Currency } from '@ledgerhq/currencies'
import Box, { Tabbable } from 'components/base/Box'
import Text from 'components/base/Text'
@ -12,30 +9,40 @@ import Text from 'components/base/Text'
const Container = styled(Box).attrs({
align: 'center',
justify: 'center',
m: 1,
m: 2,
})`
width: 150px;
height: 150px;
background: rgba(0, 0, 0, 0.05);
background: white;
`
// https://api.ledgerwallet.com/update/assets/icons/bitcoin
const ActionBtn = styled(Tabbable).attrs({
fontSize: 3,
})``
const AppIcon = styled.img`
display: block;
width: 50px;
height: 50px;
`
type Props = {
currency: Currency,
name: string,
icon: string,
onInstall: Function,
onUninstall: Function,
}
export default function ManagerApp(props: Props) {
const { currency, onInstall, onUninstall } = props
const Icon = getIconByCoinType(currency.coinType)
const { name, icon, onInstall, onUninstall } = props
const iconUrl = `https://api.ledgerwallet.com/update/assets/icons/${icon}`
return (
<Container flow={3}>
{Icon && <Icon size={24} />}
<Text>{currency.name}</Text>
<AppIcon src={iconUrl} />
<Text ff="Museo Sans|Bold">{name}</Text>
<Box horizontal flow={2}>
<ActionBtn onClick={onInstall}>{'Install'}</ActionBtn>
<ActionBtn onClick={onUninstall}>{'Remove'}</ActionBtn>

65
src/components/ManagerPage/index.js

@ -5,9 +5,7 @@ import { connect } from 'react-redux'
import styled from 'styled-components'
import { translate } from 'react-i18next'
import { compose } from 'redux'
import { listCurrencies } from '@ledgerhq/currencies'
import type { Currency } from '@ledgerhq/currencies'
import type { Device } from 'types/common'
import { runJob } from 'renderer/events'
@ -19,11 +17,13 @@ import Modal, { ModalBody } from 'components/base/Modal'
import ManagerApp from './ManagerApp'
const CURRENCIES = listCurrencies()
const ICONS_FALLBACK = {
bitcoin_testnet: 'bitcoin',
}
const List = styled(Box).attrs({
horizontal: true,
m: -1,
m: -2,
})`
flex-wrap: wrap;
`
@ -36,25 +36,55 @@ type Props = {
device: Device,
}
type Status = 'idle' | 'busy' | 'success' | 'error'
type Status = 'loading' | 'idle' | 'busy' | 'success' | 'error'
type LedgerApp = {
name: string,
icon: string,
app: Object,
}
type State = {
status: Status,
error: string | null,
appsList: LedgerApp[],
}
class ManagerPage extends PureComponent<Props, State> {
state = {
status: 'idle',
status: 'loading',
error: null,
appsList: [],
}
componentDidMount() {
this.fetchList()
}
componentWillUnmount() {
this._unmounted = true
}
_unmounted = false
async fetchList() {
const appsList = await runJob({
channel: 'usb',
job: 'manager.listApps',
successResponse: 'manager.listAppsSuccess',
errorResponse: 'manager.listAppsError',
})
if (!this._unmounted) {
this.setState({ appsList, status: 'idle' })
}
}
createDeviceJobHandler = options => (currency: Currency) => async () => {
createDeviceJobHandler = options => ({ app: appParams }) => async () => {
this.setState({ status: 'busy' })
try {
const { job, successResponse, errorResponse } = options
const { device: { path: devicePath } } = this.props
const data = { appName: currency.name.toLowerCase(), devicePath }
const data = { appParams, devicePath }
await runJob({ channel: 'usb', job, successResponse, errorResponse, data })
this.setState({ status: 'success' })
} catch (err) {
@ -78,10 +108,11 @@ class ManagerPage extends PureComponent<Props, State> {
renderList = () => (
<List>
{CURRENCIES.map(c => (
{this.state.appsList.map(c => (
<ManagerApp
key={c.coinType}
currency={c}
key={c.name}
name={c.name}
icon={ICONS_FALLBACK[c.icon] || c.icon}
onInstall={this.handleInstall(c)}
onUninstall={this.handleUninstall(c)}
/>
@ -100,9 +131,17 @@ class ManagerPage extends PureComponent<Props, State> {
<Box fontSize={8}>{'Connect your device'}</Box>
</Box>
)}
{deviceStatus === 'connected' && this.renderList()}
{deviceStatus === 'connected' && (
<Box>
{status === 'loading' ? (
<Box ff="Museo Sans|Bold">{'Loading app list...'}</Box>
) : (
this.renderList()
)}
</Box>
)}
<Modal
isOpened={status !== 'idle'}
isOpened={status !== 'idle' && status !== 'loading'}
render={() => (
<ModalBody p={6} align="center" justify="center" style={{ height: 300 }}>
{status === 'busy' ? (

11
src/internals/usb/manager/constants.js

@ -1,17 +1,6 @@
// Socket endpoint
export const BASE_SOCKET_URL = 'ws://api.ledgerwallet.com/update/install'
// Apparently params we need to add to websocket requests
//
// see https://github.com/LedgerHQ/ledger-manager-chrome
// > controllers/manager/ApplyUpdateController.scala
//
// @TODO: Get rid of them.
export const DEFAULT_SOCKET_PARAMS = {
perso: 'perso_11',
hash: '0000000000000000000000000000000000000000000000000000000000000000',
}
// List of APDUS
export const APDUS = {
GET_FIRMWARE: [0xe0, 0x01, 0x00, 0x00],

49
src/internals/usb/manager/helpers.js

@ -8,7 +8,7 @@ import noop from 'lodash/noop'
import type Transport from '@ledgerhq/hw-transport'
import type { IPCSend } from 'types/electron'
import { BASE_SOCKET_URL, DEFAULT_SOCKET_PARAMS, APDUS } from './constants'
import { BASE_SOCKET_URL, APDUS } from './constants'
type WebsocketType = {
send: (string, any) => void,
@ -22,6 +22,13 @@ type Message = {
data: any,
}
type LedgerAppParams = {
firmware: string,
firmwareKey: string,
delete: string,
deleteKey: string,
}
/**
* Generate handler which create transport with given
* `devicePath` then call action with it
@ -63,15 +70,9 @@ export function createTransportHandler(
*/
export async function installApp(
transport: Transport<*>,
{ appName }: { appName: string },
{ appParams }: { appParams: LedgerAppParams },
): Promise<void> {
log('INSTALL', `Request to install ${appName} app`)
return createSocketDialog(transport, ({ version }) => ({
firmware: `nanos/${version}/${appName}/app_latest`,
firmwareKey: `nanos/${version}/${appName}/app_latest_key`,
delete: `nanos/${version}/${appName}/app_del`,
deleteKey: `nanos/${version}/${appName}/app_del_key`,
}))
return createSocketDialog(transport, appParams)
}
/**
@ -79,15 +80,13 @@ export async function installApp(
*/
export async function uninstallApp(
transport: Transport<*>,
{ appName }: { appName: string },
{ appParams }: { appParams: LedgerAppParams },
): Promise<void> {
log('INSTALL', `Request to uninstall ${appName} app`)
return createSocketDialog(transport, ({ version }) => ({
firmware: `nanos/${version}/${appName}/app_del`,
firmwareKey: `nanos/${version}/${appName}/app_del_key`,
delete: `nanos/${version}/${appName}/app_del`,
deleteKey: `nanos/${version}/${appName}/app_del_key`,
}))
return createSocketDialog(transport, {
...appParams,
firmware: appParams.delete,
firmwareKey: appParams.deleteKey,
})
}
/**
@ -143,16 +142,10 @@ async function bulk(ws: WebsocketType, transport: Transport<*>, msg: Message) {
* Open socket connection with firmware api, and init a dialog
* with the device
*/
function createSocketDialog(transport: Transport<*>, buildParams: Function) {
function createSocketDialog(transport: Transport<*>, appParams: LedgerAppParams) {
return new Promise(async (resolve, reject) => {
try {
const { targetId, version } = await getFirmwareInfo(transport)
const fullParams = qs.stringify({
targetId,
...DEFAULT_SOCKET_PARAMS,
...buildParams({ targetId, version }),
})
const url = `${BASE_SOCKET_URL}?${fullParams}`
const url = `${BASE_SOCKET_URL}?${qs.stringify(appParams)}`
log('WS CONNECTING', url)
const ws: WebsocketType = new Websocket(url)
@ -195,7 +188,7 @@ function createSocketDialog(transport: Transport<*>, buildParams: Function) {
/**
* Retrieve targetId and firmware version from device
*/
async function getFirmwareInfo(transport: Transport<*>) {
export async function getFirmwareInfo(transport: Transport<*>) {
const res = await transport.send(...APDUS.GET_FIRMWARE)
const byteArray = [...res]
const data = byteArray.slice(0, byteArray.length - 2)
@ -212,7 +205,9 @@ async function getFirmwareInfo(transport: Transport<*>) {
function log(namespace: string, str: string = '', color?: string) {
namespace = namespace.padEnd(15)
const coloredNamespace = color ? chalk[color](namespace) : namespace
console.log(`${chalk.bold(`> ${coloredNamespace}`)} ${str}`) // eslint-disable-line no-console
if (__DEV__) {
console.log(`${chalk.bold(`> ${coloredNamespace}`)} ${str}`) // eslint-disable-line no-console
}
}
/**

10
src/internals/usb/manager/index.js

@ -17,6 +17,7 @@
*/
import type { IPCSend } from 'types/electron'
import axios from 'axios'
import { createTransportHandler, installApp, uninstallApp } from './helpers'
export default (send: IPCSend) => ({
@ -31,4 +32,13 @@ export default (send: IPCSend) => ({
successResponse: 'device.appUninstalled',
errorResponse: 'device.appUninstallError',
}),
listApps: async () => {
try {
const { data } = await axios.get('https://api.ledgerwallet.com/update/applications')
send('manager.listAppsSuccess', data['nanos-1.4'])
} catch (err) {
send('manager.listAppsError', { message: err.message, stack: err.stack })
}
},
})

2
src/renderer/events.js

@ -56,7 +56,7 @@ export function runJob({
job: string,
successResponse: string,
errorResponse: string,
data: any,
data?: any,
}): Promise<void> {
return new Promise((resolve, reject) => {
ipcRenderer.send(channel, { type: job, data })

Loading…
Cancel
Save