Browse Source

Merge pull request #504 from meriadec/feature/hard-reset

Feature/hard reset
master
Meriadec Pillet 7 years ago
committed by GitHub
parent
commit
226d9880d1
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 2
      src/commands/index.js
  2. 20
      src/commands/libcoreHardReset.js
  3. 15
      src/components/SettingsPage/sections/Profile.js
  4. 20
      src/components/base/Button/index.js
  5. 14
      src/components/base/Modal/ConfirmModal.js
  6. 14
      src/helpers/hardReset.js
  7. 10
      src/middlewares/db.js
  8. 50
      src/renderer/init.js

2
src/commands/index.js

@ -14,6 +14,7 @@ import installMcu from 'commands/installMcu'
import installOsuFirmware from 'commands/installOsuFirmware' import installOsuFirmware from 'commands/installOsuFirmware'
import isDashboardOpen from 'commands/isDashboardOpen' import isDashboardOpen from 'commands/isDashboardOpen'
import libcoreGetVersion from 'commands/libcoreGetVersion' import libcoreGetVersion from 'commands/libcoreGetVersion'
import libcoreHardReset from 'commands/libcoreHardReset'
import libcoreScanAccounts from 'commands/libcoreScanAccounts' import libcoreScanAccounts from 'commands/libcoreScanAccounts'
import libcoreSignAndBroadcast from 'commands/libcoreSignAndBroadcast' import libcoreSignAndBroadcast from 'commands/libcoreSignAndBroadcast'
import libcoreSyncAccount from 'commands/libcoreSyncAccount' import libcoreSyncAccount from 'commands/libcoreSyncAccount'
@ -38,6 +39,7 @@ const all: Array<Command<any, any>> = [
installOsuFirmware, installOsuFirmware,
isDashboardOpen, isDashboardOpen,
libcoreGetVersion, libcoreGetVersion,
libcoreHardReset,
libcoreScanAccounts, libcoreScanAccounts,
libcoreSignAndBroadcast, libcoreSignAndBroadcast,
libcoreSyncAccount, libcoreSyncAccount,

20
src/commands/libcoreHardReset.js

@ -0,0 +1,20 @@
// @flow
import { createCommand } from 'helpers/ipc'
import { Observable } from 'rxjs'
import withLibcore from 'helpers/withLibcore'
const cmd = createCommand('libcoreHardReset', () =>
Observable.create(o => {
withLibcore(async (core, njsWalletPool) => {
try {
njsWalletPool.eraseDataSince(new Date(0))
o.complete()
} catch (e) {
o.error(e)
}
})
}),
)
export default cmd

15
src/components/SettingsPage/sections/Profile.js

@ -9,6 +9,7 @@ import { cleanAccountsCache } from 'actions/accounts'
import { unlock } from 'reducers/application' // FIXME should be in actions import { unlock } from 'reducers/application' // FIXME should be in actions
import db, { setEncryptionKey } from 'helpers/db' import db, { setEncryptionKey } from 'helpers/db'
import { delay } from 'helpers/promise' import { delay } from 'helpers/promise'
import hardReset from 'helpers/hardReset'
import type { SettingsState } from 'reducers/settings' import type { SettingsState } from 'reducers/settings'
import type { T } from 'types/common' import type { T } from 'types/common'
@ -47,6 +48,7 @@ type State = {
isSoftResetModalOpened: boolean, isSoftResetModalOpened: boolean,
isPasswordModalOpened: boolean, isPasswordModalOpened: boolean,
isDisablePasswordModalOpened: boolean, isDisablePasswordModalOpened: boolean,
isHardResetting: boolean,
} }
class TabProfile extends PureComponent<Props, State> { class TabProfile extends PureComponent<Props, State> {
@ -55,6 +57,7 @@ class TabProfile extends PureComponent<Props, State> {
isSoftResetModalOpened: false, isSoftResetModalOpened: false,
isPasswordModalOpened: false, isPasswordModalOpened: false,
isDisablePasswordModalOpened: false, isDisablePasswordModalOpened: false,
isHardResetting: false,
} }
setPassword = password => { setPassword = password => {
@ -89,9 +92,13 @@ class TabProfile extends PureComponent<Props, State> {
} }
handleHardReset = async () => { handleHardReset = async () => {
db.resetAll() this.setState({ isHardResetting: true })
await delay(500) try {
remote.getCurrentWindow().webContents.reload() await hardReset()
remote.getCurrentWindow().webContents.reloadIgnoringCache()
} catch (err) {
this.setState({ isHardResetting: false })
}
} }
handleChangePasswordCheck = isChecked => { handleChangePasswordCheck = isChecked => {
@ -125,6 +132,7 @@ class TabProfile extends PureComponent<Props, State> {
isHardResetModalOpened, isHardResetModalOpened,
isPasswordModalOpened, isPasswordModalOpened,
isDisablePasswordModalOpened, isDisablePasswordModalOpened,
isHardResetting,
} = this.state } = this.state
const isPasswordEnabled = settings.password.isEnabled === true const isPasswordEnabled = settings.password.isEnabled === true
return ( return (
@ -200,6 +208,7 @@ class TabProfile extends PureComponent<Props, State> {
<ConfirmModal <ConfirmModal
isDanger isDanger
isLoading={isHardResetting}
isOpened={isHardResetModalOpened} isOpened={isHardResetModalOpened}
onClose={this.handleCloseHardResetModal} onClose={this.handleCloseHardResetModal}
onReject={this.handleCloseHardResetModal} onReject={this.handleCloseHardResetModal}

20
src/components/base/Button/index.js

@ -6,9 +6,10 @@ import { space, fontSize, fontWeight, color } from 'styled-system'
import noop from 'lodash/noop' import noop from 'lodash/noop'
import { darken, lighten } from 'styles/helpers' import { darken, lighten } from 'styles/helpers'
import fontFamily from 'styles/styled/fontFamily' import fontFamily from 'styles/styled/fontFamily'
import Spinner from 'components/base/Spinner'
const buttonStyles = { const buttonStyles = {
primary: { primary: {
default: p => ` default: p => `
@ -73,6 +74,14 @@ const buttonStyles = {
padding-right: ${space[1]}px; padding-right: ${space[1]}px;
`, `,
}, },
isLoading: {
default: () => `
padding-left: 40px;
padding-right: 40px;
pointer-events: none;
opacity: 0.7;
`,
},
} }
function getStyles(props, state) { function getStyles(props, state) {
@ -127,14 +136,15 @@ type Props = {
onClick?: Function, onClick?: Function,
small?: boolean, small?: boolean,
padded?: boolean, padded?: boolean,
isLoading?: boolean,
} }
const Button = (props: Props) => { const Button = (props: Props) => {
const { onClick, children, disabled } = props const { onClick, children, disabled, isLoading } = props
const isClickDisabled = disabled || isLoading
return ( return (
<Base {...props} onClick={disabled ? undefined : onClick}> <Base {...props} onClick={isClickDisabled ? undefined : onClick}>
{children} {isLoading ? <Spinner size={16} /> : children}
</Base> </Base>
) )
} }

14
src/components/base/Modal/ConfirmModal.js

@ -21,6 +21,7 @@ type Props = {
onReject: Function, onReject: Function,
onConfirm: Function, onConfirm: Function,
t: T, t: T,
isLoading?: boolean,
} }
class ConfirmModal extends PureComponent<Props> { class ConfirmModal extends PureComponent<Props> {
@ -35,6 +36,7 @@ class ConfirmModal extends PureComponent<Props> {
isDanger, isDanger,
onReject, onReject,
onConfirm, onConfirm,
isLoading,
t, t,
...props ...props
} = this.props } = this.props
@ -44,9 +46,10 @@ class ConfirmModal extends PureComponent<Props> {
return ( return (
<Modal <Modal
isOpened={isOpened} isOpened={isOpened}
preventBackdropClick={isLoading}
{...props} {...props}
render={({ onClose }) => ( render={({ onClose }) => (
<ModalBody onClose={onClose}> <ModalBody onClose={isLoading ? undefined : onClose}>
<ModalTitle>{title}</ModalTitle> <ModalTitle>{title}</ModalTitle>
<ModalContent> <ModalContent>
{subTitle && ( {subTitle && (
@ -59,8 +62,13 @@ class ConfirmModal extends PureComponent<Props> {
</Box> </Box>
</ModalContent> </ModalContent>
<ModalFooter horizontal align="center" justify="flex-end" flow={2}> <ModalFooter horizontal align="center" justify="flex-end" flow={2}>
<Button onClick={onReject}>{realCancelText}</Button> {!isLoading && <Button onClick={onReject}>{realCancelText}</Button>}
<Button onClick={onConfirm} primary={!isDanger} danger={isDanger}> <Button
onClick={onConfirm}
primary={!isDanger}
danger={isDanger}
isLoading={isLoading}
>
{realConfirmText} {realConfirmText}
</Button> </Button>
</ModalFooter> </ModalFooter>

14
src/helpers/hardReset.js

@ -0,0 +1,14 @@
import libcoreHardReset from 'commands/libcoreHardReset'
import { disable as disableDBMiddleware } from 'middlewares/db'
import db from 'helpers/db'
import { delay } from 'helpers/promise'
export default async function hardReset() {
// TODO: wait for the libcoreHardReset to be finished
// actually, libcore doesnt goes back to js thread
await Promise.race([libcoreHardReset.send().toPromise(), delay(500)])
disableDBMiddleware()
db.resetAll()
await delay(500)
}

10
src/middlewares/db.js

@ -6,8 +6,16 @@ import { accountsSelector } from 'reducers/accounts'
import { settingsExportSelector, areSettingsLoaded } from 'reducers/settings' import { settingsExportSelector, areSettingsLoaded } from 'reducers/settings'
import CounterValues from 'helpers/countervalues' import CounterValues from 'helpers/countervalues'
let DB_MIDDLEWARE_ENABLED = true
// ability to temporary disable the db middleware from outside
export const disable = (ms = 1000) => {
DB_MIDDLEWARE_ENABLED = false
setTimeout(() => (DB_MIDDLEWARE_ENABLED = true), ms)
}
export default store => next => action => { export default store => next => action => {
if (action.type.startsWith('DB:')) { if (DB_MIDDLEWARE_ENABLED && action.type.startsWith('DB:')) {
const [, type] = action.type.split(':') const [, type] = action.type.split(':')
store.dispatch({ type, payload: action.payload }) store.dispatch({ type, payload: action.payload })
const state = store.getState() const state = store.getState()

50
src/renderer/init.js

@ -20,43 +20,39 @@ import libcoreGetVersion from 'commands/libcoreGetVersion'
import db from 'helpers/db' import db from 'helpers/db'
import dbMiddleware from 'middlewares/db' import dbMiddleware from 'middlewares/db'
import CounterValues from 'helpers/countervalues' import CounterValues from 'helpers/countervalues'
import hardReset from 'helpers/hardReset'
import App from 'components/App' import App from 'components/App'
import 'styles/global' import 'styles/global'
if (process.env.LEDGER_RESET_ALL) {
db.resetAll()
}
// Init db with defaults if needed
db.init('settings', {})
const history = createHistory()
const store = createStore({ history, dbMiddleware })
const rootNode = document.getElementById('app') const rootNode = document.getElementById('app')
const settings = db.get('settings') async function init() {
store.dispatch(fetchSettings(settings)) if (process.env.LEDGER_RESET_ALL) {
await hardReset()
}
const countervaluesData = db.get('countervalues') // Init db with defaults if needed
if (countervaluesData) { db.init('settings', {})
store.dispatch(CounterValues.importAction(countervaluesData))
}
const state = store.getState() const history = createHistory()
const language = getLanguage(state) const store = createStore({ history, dbMiddleware })
const locked = isLocked(state)
moment.locale(language) const settings = db.get('settings')
store.dispatch(fetchSettings(settings))
function r(Comp) { const countervaluesData = db.get('countervalues')
if (rootNode) { if (countervaluesData) {
render(<AppContainer>{Comp}</AppContainer>, rootNode) store.dispatch(CounterValues.importAction(countervaluesData))
} }
}
async function init() { const state = store.getState()
const language = getLanguage(state)
const locked = isLocked(state)
moment.locale(language)
// FIXME IMO init() really should only be for window. any other case is a hack! // FIXME IMO init() really should only be for window. any other case is a hack!
const isMainWindow = remote.getCurrentWindow().name === 'MainWindow' const isMainWindow = remote.getCurrentWindow().name === 'MainWindow'
@ -92,6 +88,12 @@ async function init() {
} }
} }
function r(Comp) {
if (rootNode) {
render(<AppContainer>{Comp}</AppContainer>, rootNode)
}
}
init().catch(e => { init().catch(e => {
// for now we make the app crash instead of pending forever. later we can render the error OR try to recover, but probably this is unrecoverable cases. // for now we make the app crash instead of pending forever. later we can render the error OR try to recover, but probably this is unrecoverable cases.
logger.error(e) logger.error(e)

Loading…
Cancel
Save