From 1e9199db3238cdae215838a26d082b46b85f9eb3 Mon Sep 17 00:00:00 2001 From: apanizo Date: Tue, 5 Jun 2018 13:27:14 +0200 Subject: [PATCH 01/25] WA-238 Adding logic (including tests) for handling change threshold --- config/jest/jest.setup.js | 2 +- .../AddTransaction/createTransactions.js | 16 ++- .../AddTransaction/test/transactions.test.js | 34 ++--- .../AddTransaction/test/transactionsHelper.js | 3 +- .../Transactions/processTransactions.js | 14 ++- src/routes/safe/store/model/transaction.js | 2 + src/routes/safe/test/Safe.threshold.test.js | 117 ++++++++++++++++++ 7 files changed, 157 insertions(+), 31 deletions(-) create mode 100644 src/routes/safe/test/Safe.threshold.test.js diff --git a/config/jest/jest.setup.js b/config/jest/jest.setup.js index 8d9161a7..c97ef811 100644 --- a/config/jest/jest.setup.js +++ b/config/jest/jest.setup.js @@ -1,2 +1,2 @@ // @flow -jest.setTimeout(30000) +jest.setTimeout(45000) diff --git a/src/routes/safe/component/AddTransaction/createTransactions.js b/src/routes/safe/component/AddTransaction/createTransactions.js index 7ac3956b..d0d83569 100644 --- a/src/routes/safe/component/AddTransaction/createTransactions.js +++ b/src/routes/safe/component/AddTransaction/createTransactions.js @@ -47,6 +47,7 @@ export const storeTransaction = ( tx: string, safeAddress: string, safeThreshold: number, + data: string, ) => { const notMinedWhenOneOwnerSafe = confirmations.count() === 1 && !tx if (notMinedWhenOneOwnerSafe) { @@ -54,7 +55,7 @@ export const storeTransaction = ( } const transaction: Transaction = makeTransaction({ - name, nonce, value, confirmations, destination, threshold: safeThreshold, tx, + name, nonce, value, confirmations, destination, threshold: safeThreshold, tx, data, }) const safeTransactions = load(TX_KEY) || {} @@ -82,10 +83,11 @@ const hasOneOwner = (safe: Safe) => { export const createTransaction = async ( safe: Safe, txName: string, - txDestination: string, + txDest: string, txValue: number, nonce: number, user: string, + data: string = '0x', ) => { const web3 = getWeb3() const GnosisSafe = await getGnosisSafeContract(web3) @@ -97,19 +99,21 @@ export const createTransaction = async ( const thresholdIsOne = safe.get('confirmations') === 1 if (hasOneOwner(safe) || thresholdIsOne) { - const txConfirmationData = gnosisSafe.contract.execTransactionIfApproved.getData(txDestination, valueInWei, '0x', CALL, nonce) + const txConfirmationData = + gnosisSafe.contract.execTransactionIfApproved.getData(txDest, valueInWei, data, CALL, nonce) const txHash = await executeTransaction(txConfirmationData, user, safeAddress) checkReceiptStatus(txHash) const executedConfirmations: List = buildExecutedConfirmationFrom(safe.get('owners'), user) - return storeTransaction(txName, nonce, txDestination, txValue, user, executedConfirmations, txHash, safeAddress, safe.get('confirmations')) + return storeTransaction(txName, nonce, txDest, txValue, user, executedConfirmations, txHash, safeAddress, safe.get('confirmations'), data) } - const txConfirmationData = gnosisSafe.contract.approveTransactionWithParameters.getData(txDestination, valueInWei, '0x', CALL, nonce) + const txConfirmationData = + gnosisSafe.contract.approveTransactionWithParameters.getData(txDest, valueInWei, data, CALL, nonce) const txConfirmationHash = await executeTransaction(txConfirmationData, user, safeAddress) checkReceiptStatus(txConfirmationHash) const confirmations: List = buildConfirmationsFrom(safe.get('owners'), user, txConfirmationHash) - return storeTransaction(txName, nonce, txDestination, txValue, user, confirmations, '', safeAddress, safe.get('confirmations')) + return storeTransaction(txName, nonce, txDest, txValue, user, confirmations, '', safeAddress, safe.get('confirmations'), data) } diff --git a/src/routes/safe/component/AddTransaction/test/transactions.test.js b/src/routes/safe/component/AddTransaction/test/transactions.test.js index 367403bf..19c1fdeb 100644 --- a/src/routes/safe/component/AddTransaction/test/transactions.test.js +++ b/src/routes/safe/component/AddTransaction/test/transactions.test.js @@ -33,7 +33,7 @@ describe('Transactions Suite', () => { const txName = 'Buy butteries for project' const nonce: number = 10 const confirmations: List = buildConfirmationsFrom(owners, 'foo', 'confirmationHash') - storeTransaction(txName, nonce, destination, value, 'foo', confirmations, '', safe.get('address'), safe.get('confirmations')) + storeTransaction(txName, nonce, destination, value, 'foo', confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') // WHEN const transactions: Map> = loadSafeTransactions() @@ -45,7 +45,7 @@ describe('Transactions Suite', () => { if (!safeTransactions) { throw new Error() } testSizeOfTransactions(safeTransactions, 1) - testTransactionFrom(safeTransactions, 0, txName, nonce, value, 2, destination, 'foo', 'confirmationHash', owners.get(0), owners.get(1)) + testTransactionFrom(safeTransactions, 0, txName, nonce, value, 2, destination, '0x', 'foo', 'confirmationHash', owners.get(0), owners.get(1)) }) it('adds second confirmation to stored safe with one confirmation', async () => { @@ -55,12 +55,12 @@ describe('Transactions Suite', () => { const safeAddress = safe.get('address') const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(firstTxName, firstNonce, destination, value, creator, confirmations, '', safeAddress, safe.get('confirmations')) + storeTransaction(firstTxName, firstNonce, destination, value, creator, confirmations, '', safeAddress, safe.get('confirmations'), '0x') const secondTxName = 'Buy printers for project' const secondNonce: number = firstNonce + 100 const secondConfirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(secondTxName, secondNonce, destination, value, creator, secondConfirmations, '', safeAddress, safe.get('confirmations')) + storeTransaction(secondTxName, secondNonce, destination, value, creator, secondConfirmations, '', safeAddress, safe.get('confirmations'), '0x') // WHEN const transactions: Map> = loadSafeTransactions() @@ -72,8 +72,8 @@ describe('Transactions Suite', () => { if (!safeTxs) { throw new Error() } testSizeOfTransactions(safeTxs, 2) - testTransactionFrom(safeTxs, 0, firstTxName, firstNonce, value, 2, destination, 'foo', 'confirmationHash', owners.get(0), owners.get(1)) - testTransactionFrom(safeTxs, 1, secondTxName, secondNonce, value, 2, destination, 'foo', 'confirmationHash', owners.get(0), owners.get(1)) + testTransactionFrom(safeTxs, 0, firstTxName, firstNonce, value, 2, destination, '0x', 'foo', 'confirmationHash', owners.get(0), owners.get(1)) + testTransactionFrom(safeTxs, 1, secondTxName, secondNonce, value, 2, destination, '0x', 'foo', 'confirmationHash', owners.get(0), owners.get(1)) }) it('adds second confirmation to stored safe having two safes with one confirmation each', async () => { @@ -82,7 +82,7 @@ describe('Transactions Suite', () => { const safeAddress = safe.address const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safeAddress, safe.get('confirmations')) + storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safeAddress, safe.get('confirmations'), '0x') const secondSafe = SafeFactory.dailyLimitSafe(10, 2) const txSecondName = 'Buy batteris for Beta project' @@ -92,7 +92,7 @@ describe('Transactions Suite', () => { const secondConfirmations: List = buildConfirmationsFrom(secondSafe.get('owners'), secondCreator, 'confirmationHash') storeTransaction( txSecondName, txSecondNonce, destination, value, secondCreator, - secondConfirmations, '', secondSafeAddress, secondSafe.get('confirmations'), + secondConfirmations, '', secondSafeAddress, secondSafe.get('confirmations'), '0x', ) let transactions: Map> = loadSafeTransactions() @@ -112,7 +112,7 @@ describe('Transactions Suite', () => { const txConfirmations: List = buildConfirmationsFrom(owners, creator, 'secondConfirmationHash') storeTransaction( txFirstName, txFirstNonce, destination, value, creator, - txConfirmations, '', safe.get('address'), safe.get('confirmations'), + txConfirmations, '', safe.get('address'), safe.get('confirmations'), '0x', ) transactions = loadSafeTransactions() @@ -125,19 +125,19 @@ describe('Transactions Suite', () => { // Test 2 transactions of first safe testTransactionFrom( transactions.get(safe.address), 0, - txName, nonce, value, 2, destination, + txName, nonce, value, 2, destination, '0x', 'foo', 'confirmationHash', owners.get(0), owners.get(1), ) testTransactionFrom( transactions.get(safe.address), 1, - txFirstName, txFirstNonce, value, 2, destination, + txFirstName, txFirstNonce, value, 2, destination, '0x', 'foo', 'secondConfirmationHash', owners.get(0), owners.get(1), ) // Test one transaction of second safe testTransactionFrom( transactions.get(secondSafe.address), 0, - txSecondName, txSecondNonce, value, 2, destination, + txSecondName, txSecondNonce, value, 2, destination, '0x', '0x03db1a8b26d08df23337e9276a36b474510f0023', 'confirmationHash', secondSafe.get('owners').get(0), secondSafe.get('owners').get(1), ) }) @@ -148,10 +148,10 @@ describe('Transactions Suite', () => { const nonce: number = 10 const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations')) + storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') // WHEN - const createTxFnc = () => storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations')) + const createTxFnc = () => storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') expect(createTxFnc).toThrow(/Transaction with same nonce/) }) @@ -161,7 +161,7 @@ describe('Transactions Suite', () => { const nonce: number = 10 const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations')) + storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') // WHEN const transactions: Map> = loadSafeTransactions() @@ -185,7 +185,7 @@ describe('Transactions Suite', () => { const nonce: number = 10 const tx = '' const confirmations: List = buildExecutedConfirmationFrom(oneOwnerSafe.get('owners'), ownerName) - const createTxFnc = () => storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('confirmations')) + const createTxFnc = () => storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('confirmations'), '0x') expect(createTxFnc).toThrow(/The tx should be mined before storing it in safes with one owner/) }) @@ -197,7 +197,7 @@ describe('Transactions Suite', () => { const nonce: number = 10 const tx = 'validTxHash' const confirmations: List = buildExecutedConfirmationFrom(oneOwnerSafe.get('owners'), ownerName) - storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('confirmations')) + storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('confirmations'), '0x') // WHEN const safeTransactions: Map> = loadSafeTransactions() diff --git a/src/routes/safe/component/AddTransaction/test/transactionsHelper.js b/src/routes/safe/component/AddTransaction/test/transactionsHelper.js index 9b92df35..8ad22983 100644 --- a/src/routes/safe/component/AddTransaction/test/transactionsHelper.js +++ b/src/routes/safe/component/AddTransaction/test/transactionsHelper.js @@ -20,7 +20,7 @@ export const testSizeOfTransactions = (safeTxs: List | typeof undef export const testTransactionFrom = ( safeTxs: List | typeof undefined, pos: number, name: string, nonce: number, value: number, threshold: number, destination: string, - creator: string, txHash: string, + data: string, creator: string, txHash: string, firstOwner: Owner | typeof undefined, secondOwner: Owner | typeof undefined, ) => { if (!safeTxs) { throw new Error() } @@ -33,6 +33,7 @@ export const testTransactionFrom = ( expect(tx.get('destination')).toBe(destination) expect(tx.get('confirmations').count()).toBe(2) expect(tx.get('nonce')).toBe(nonce) + expect(tx.get('data')).toBe(data) const confirmations: List = tx.get('confirmations') const firstConfirmation: Confirmation | typeof undefined = confirmations.get(0) diff --git a/src/routes/safe/component/Transactions/processTransactions.js b/src/routes/safe/component/Transactions/processTransactions.js index dfbc64d2..32f0d123 100644 --- a/src/routes/safe/component/Transactions/processTransactions.js +++ b/src/routes/safe/component/Transactions/processTransactions.js @@ -20,9 +20,10 @@ export const updateTransaction = ( tx: string, safeAddress: string, safeThreshold: number, + data: string, ) => { const transaction: Transaction = makeTransaction({ - name, nonce, value, confirmations, destination, threshold: safeThreshold, tx, + name, nonce, value, confirmations, destination, threshold: safeThreshold, tx, data, }) const safeTransactions = load(TX_KEY) || {} @@ -36,7 +37,6 @@ export const updateTransaction = ( localStorage.setItem(TX_KEY, JSON.stringify(safeTransactions)) } -const getData = () => '0x' const getOperation = () => 0 const execTransaction = async ( @@ -45,8 +45,8 @@ const execTransaction = async ( txValue: number, nonce: number, executor: string, + data: string, ) => { - const data = getData() const CALL = getOperation() const web3 = getWeb3() const valueInWei = web3.toWei(txValue, 'ether') @@ -61,8 +61,8 @@ const execConfirmation = async ( txValue: number, nonce: number, executor: string, + data: string, ) => { - const data = getData() const CALL = getOperation() const web3 = getWeb3() const valueInWei = web3.toWei(txValue, 'ether') @@ -110,10 +110,11 @@ export const processTransaction = async ( const txName = tx.get('name') const txValue = tx.get('value') const txDestination = tx.get('destination') + const data = tx.get('data') const txHash = thresholdReached - ? await execTransaction(gnosisSafe, txDestination, txValue, nonce, userAddress) - : await execConfirmation(gnosisSafe, txDestination, txValue, nonce, userAddress) + ? await execTransaction(gnosisSafe, txDestination, txValue, nonce, userAddress, data) + : await execConfirmation(gnosisSafe, txDestination, txValue, nonce, userAddress, data) checkReceiptStatus(txHash) @@ -130,5 +131,6 @@ export const processTransaction = async ( thresholdReached ? txHash : '', safeAddress, threshold, + data, ) } diff --git a/src/routes/safe/store/model/transaction.js b/src/routes/safe/store/model/transaction.js index 83a664d0..e810ed00 100644 --- a/src/routes/safe/store/model/transaction.js +++ b/src/routes/safe/store/model/transaction.js @@ -11,6 +11,7 @@ export type TransactionProps = { confirmations: List, destination: string, tx: string, + data: string, } export const makeTransaction: RecordFactory = Record({ @@ -21,6 +22,7 @@ export const makeTransaction: RecordFactory = Record({ destination: '', tx: '', threshold: 0, + data: '', }) export type Transaction = RecordOf diff --git a/src/routes/safe/test/Safe.threshold.test.js b/src/routes/safe/test/Safe.threshold.test.js new file mode 100644 index 00000000..1ca38066 --- /dev/null +++ b/src/routes/safe/test/Safe.threshold.test.js @@ -0,0 +1,117 @@ +// @flow +import { aNewStore } from '~/store' +import { aDeployedSafe } from '~/routes/safe/store/test/builder/deployedSafe.builder' +import { getWeb3 } from '~/wallets/getWeb3' +import { sleep } from '~/utils/timer' +import { promisify } from '~/utils/promisify' +import { processTransaction } from '~/routes/safe/component/Transactions/processTransactions' +import { confirmationsTransactionSelector, safeSelector, safeTransactionsSelector } from '~/routes/safe/store/selectors/index' +import { getTransactionFromReduxStore } from '~/routes/safe/test/testMultisig' +import { buildMathPropsFrom } from '~/test/buildReactRouterProps' +import { createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' +import { getGnosisSafeContract } from '~/wallets/safeContracts' +import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' + +describe('React DOM TESTS > Change threshold', () => { + it('should update the threshold directly if safe has 1 threshold', async () => { + // GIVEN + const numOwners = 2 + const threshold = 1 + const store = aNewStore() + const address = await aDeployedSafe(store, 10, threshold, numOwners) + const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) + const match: Match = buildMathPropsFrom(address) + const safe = safeSelector(store.getState(), { match }) + const web3 = getWeb3() + const GnosisSafe = await getGnosisSafeContract(web3) + const gnosisSafe = GnosisSafe.at(address) + + // WHEN + const nonce = Date.now() + const data = gnosisSafe.contract.changeThreshold.getData(2) + await createTransaction(safe, "Change Safe's threshold", address, 0, nonce, accounts[0], data) + await sleep(1500) + await store.dispatch(fetchTransactions()) + + // THEN + const transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) + expect(transactions.count()).toBe(1) + + const thresholdTx: Transaction = transactions.get(0) + expect(thresholdTx.get('tx')).not.toBe(null) + expect(thresholdTx.get('tx')).not.toBe(undefined) + expect(thresholdTx.get('tx')).not.toBe('') + + const safeThreshold = await gnosisSafe.getThreshold() + expect(Number(safeThreshold)).toEqual(2) + }) + + const changeThreshold = async (store, safeAddress, executor) => { + const tx = getTransactionFromReduxStore(store, safeAddress) + const confirmed = confirmationsTransactionSelector(store.getState(), { transaction: tx }) + const data = tx.get('data') + expect(data).not.toBe(null) + expect(data).not.toBe(undefined) + expect(data).not.toBe('') + await processTransaction(safeAddress, tx, confirmed, executor) + await sleep(1800) + } + + it('should wait for confirmation to update threshold when safe has 1+ threshold', async () => { + // GIVEN + const numOwners = 3 + const threshold = 2 + const store = aNewStore() + const address = await aDeployedSafe(store, 10, threshold, numOwners) + const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) + const match: Match = buildMathPropsFrom(address) + const safe = safeSelector(store.getState(), { match }) + const web3 = getWeb3() + const GnosisSafe = await getGnosisSafeContract(web3) + const gnosisSafe = GnosisSafe.at(address) + + // WHEN + const nonce = Date.now() + const data = gnosisSafe.contract.changeThreshold.getData(3) + await createTransaction(safe, "Change Safe's threshold", address, 0, nonce, accounts[0], data) + await sleep(1500) + await store.dispatch(fetchTransactions()) + + let transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) + expect(transactions.count()).toBe(1) + + let thresholdTx: Transaction = transactions.get(0) + expect(thresholdTx.get('tx')).toBe('') + let firstOwnerConfirmation = thresholdTx.get('confirmations').get(0) + if (!firstOwnerConfirmation) throw new Error() + expect(firstOwnerConfirmation.get('status')).toBe(true) + let secondOwnerConfirmation = thresholdTx.get('confirmations').get(1) + if (!secondOwnerConfirmation) throw new Error() + expect(secondOwnerConfirmation.get('status')).toBe(false) + + let safeThreshold = await gnosisSafe.getThreshold() + expect(Number(safeThreshold)).toEqual(2) + + // THEN + await changeThreshold(store, address, accounts[1]) + safeThreshold = await gnosisSafe.getThreshold() + expect(Number(safeThreshold)).toEqual(3) + + await store.dispatch(fetchTransactions()) + sleep(1200) + transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) + expect(transactions.count()).toBe(1) + + thresholdTx = transactions.get(0) + expect(thresholdTx.get('tx')).not.toBe(undefined) + expect(thresholdTx.get('tx')).not.toBe(null) + expect(thresholdTx.get('tx')).not.toBe('') + + firstOwnerConfirmation = thresholdTx.get('confirmations').get(0) + if (!firstOwnerConfirmation) throw new Error() + expect(firstOwnerConfirmation.get('status')).toBe(true) + secondOwnerConfirmation = thresholdTx.get('confirmations').get(1) + if (!secondOwnerConfirmation) throw new Error() + expect(secondOwnerConfirmation.get('status')).toBe(true) + }) +}) From 45a5d1c4091e6db95d3b0058db5aab66b1c1653d Mon Sep 17 00:00:00 2001 From: apanizo Date: Tue, 5 Jun 2018 13:36:26 +0200 Subject: [PATCH 02/25] WA-235 Adding button Edit on confirmations section in Safe's view --- src/routes/safe/component/Safe/Confirmations.jsx | 13 ++++++++++++- src/routes/safe/component/Safe/index.jsx | 2 +- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/src/routes/safe/component/Safe/Confirmations.jsx b/src/routes/safe/component/Safe/Confirmations.jsx index 91c192d9..faf23d45 100644 --- a/src/routes/safe/component/Safe/Confirmations.jsx +++ b/src/routes/safe/component/Safe/Confirmations.jsx @@ -4,12 +4,16 @@ import { ListItem } from 'material-ui/List' import Avatar from 'material-ui/Avatar' import DoneAll from 'material-ui-icons/DoneAll' import ListItemText from '~/components/List/ListItemText' +import Button from '~/components/layout/Button' type Props = { confirmations: number, + onEditThreshold: () => void, } -const Confirmations = ({ confirmations }: Props) => ( +const EDIT_THRESHOLD_BUTTON_TEXT = 'EDIT' + +const Confirmations = ({ confirmations, onEditThreshold }: Props) => ( @@ -19,6 +23,13 @@ const Confirmations = ({ confirmations }: Props) => ( secondary={`${confirmations} required confirmations per transaction`} cut /> + ) diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index 8e88fff2..4cb26bd7 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -69,7 +69,7 @@ class GnoSafe extends React.PureComponent { - + {}} />
From d3af885074fc4bb0dd0db732db4ef399dfb9e736 Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 09:10:45 +0200 Subject: [PATCH 03/25] WA-235 Adapting buttton after threshold change --- src/components/forms/GnoForm/index.jsx | 2 +- .../AddTransaction/createTransactions.js | 10 +- src/routes/safe/component/Safe/index.jsx | 9 +- src/routes/safe/component/Threshold/index.jsx | 98 +++++++++++++++++++ .../safe/component/Threshold/selector.js | 11 +++ 5 files changed, 125 insertions(+), 5 deletions(-) create mode 100644 src/routes/safe/component/Threshold/index.jsx create mode 100644 src/routes/safe/component/Threshold/selector.js diff --git a/src/components/forms/GnoForm/index.jsx b/src/components/forms/GnoForm/index.jsx index d51637f1..7dc29a35 100644 --- a/src/components/forms/GnoForm/index.jsx +++ b/src/components/forms/GnoForm/index.jsx @@ -29,7 +29,7 @@ const GnoForm = ({ render={({ handleSubmit, ...rest }) => (
{render(rest)} - {children(rest.submitting)} + {children(rest.submitting, rest.submitSucceeded)}
)} /> diff --git a/src/routes/safe/component/AddTransaction/createTransactions.js b/src/routes/safe/component/AddTransaction/createTransactions.js index d0d83569..2bc545cc 100644 --- a/src/routes/safe/component/AddTransaction/createTransactions.js +++ b/src/routes/safe/component/AddTransaction/createTransactions.js @@ -80,6 +80,12 @@ const hasOneOwner = (safe: Safe) => { return owners.count() === 1 } +export const getSafeEthereumInstance = async (safeAddress) => { + const web3 = getWeb3() + const GnosisSafe = await getGnosisSafeContract(web3) + return GnosisSafe.at(safeAddress) +} + export const createTransaction = async ( safe: Safe, txName: string, @@ -90,10 +96,8 @@ export const createTransaction = async ( data: string = '0x', ) => { const web3 = getWeb3() - const GnosisSafe = await getGnosisSafeContract(web3) const safeAddress = safe.get('address') - const gnosisSafe = GnosisSafe.at(safeAddress) - + const gnosisSafe = await getSafeEthereumInstance(safeAddress) const valueInWei = web3.toWei(txValue, 'ether') const CALL = 0 diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index 4cb26bd7..cbb9f894 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -12,6 +12,7 @@ import List from 'material-ui/List' import Withdrawn from '~/routes/safe/component/Withdrawn' import Transactions from '~/routes/safe/component/Transactions' import AddTransaction from '~/routes/safe/component/AddTransaction' +import Threshold from '~/routes/safe/component/Threshold' import Address from './Address' import Balance from './Balance' @@ -59,6 +60,12 @@ class GnoSafe extends React.PureComponent { this.setState({ component: }) } + onEditThreshold = () => { + const { safe } = this.props + + this.setState({ component: }) + } + render() { const { safe, balance } = this.props const { component } = this.state @@ -69,7 +76,7 @@ class GnoSafe extends React.PureComponent { - {}} /> +
diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx new file mode 100644 index 00000000..86cca73c --- /dev/null +++ b/src/routes/safe/component/Threshold/index.jsx @@ -0,0 +1,98 @@ +// @flow +import * as React from 'react' +import Block from '~/components/layout/Block' +import Heading from '~/components/layout/Heading' +import Field from '~/components/forms/Field' +import TextField from '~/components/forms/TextField' +import GnoForm from '~/components/forms/GnoForm' +import { connect } from 'react-redux' +import Button from '~/components/layout/Button' +import Col from '~/components/layout/Col' +import Row from '~/components/layout/Row' +import { composeValidators, minValue, maxValue, mustBeInteger, required } from '~/components/forms/validator' +import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' +import { sleep } from '~/utils/timer' +import selector, { type SelectorProps } from './selector' + +type Props = SelectorProps & { + numOwners: number, + safe: Safe, +} + +const THRESHOLD_PARAM = 'threshold' + +const ThresholdComponent = ({ numOwners }: Props) => () => ( + + + {'Change safe\'s threshold'} + + + {`Actual number of owners: ${numOwners}`} + + + + + +) + +type State = { + initialValues: Object, +} + +class Threshold extends React.PureComponent { + state = { + initialValues: {}, + } + + onThreshold = async (values: Object) => { + const { safe, userAddress } = this.props // , fetchThreshold } = this.props + const newThreshold = values[THRESHOLD_PARAM] + const gnosisSafe = await getSafeEthereumInstance(safe.get('address')) + const nonce = Date.now() + const data = gnosisSafe.contract.changeThreshold.getData(newThreshold) + await createTransaction(safe, "Change Safe's threshold", safe.get('address'), 0, nonce, userAddress, data) + await sleep(1500) + // this.props.fetchThreshold(safe.get('address')) + } + + render() { + const { numOwners } = this.props + + return ( + + {(submitting: boolean, submitSucceeded: boolean) => ( + + + + + + )} + + ) + } +} + +export default connect(selector)(Threshold) diff --git a/src/routes/safe/component/Threshold/selector.js b/src/routes/safe/component/Threshold/selector.js new file mode 100644 index 00000000..9e7bfef1 --- /dev/null +++ b/src/routes/safe/component/Threshold/selector.js @@ -0,0 +1,11 @@ +// @flow +import { createStructuredSelector } from 'reselect' +import { userAccountSelector } from '~/wallets/store/selectors/index' + +export type SelectorProps = { + userAddress: userAccountSelector, +} + +export default createStructuredSelector({ + userAddress: userAccountSelector, +}) From ee0165ad28e0912358f59e029213e307744a655f Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 09:15:42 +0200 Subject: [PATCH 04/25] WA-238 Adding callback after changing threshold --- src/routes/safe/component/Safe/index.jsx | 2 +- src/routes/safe/component/Threshold/index.jsx | 8 +++++--- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index cbb9f894..788b2561 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -63,7 +63,7 @@ class GnoSafe extends React.PureComponent { onEditThreshold = () => { const { safe } = this.props - this.setState({ component: }) + this.setState({ component: }) } render() { diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx index 86cca73c..0ed1da98 100644 --- a/src/routes/safe/component/Threshold/index.jsx +++ b/src/routes/safe/component/Threshold/index.jsx @@ -17,6 +17,7 @@ import selector, { type SelectorProps } from './selector' type Props = SelectorProps & { numOwners: number, safe: Safe, + onReset: () => void, } const THRESHOLD_PARAM = 'threshold' @@ -62,13 +63,13 @@ class Threshold extends React.PureComponent { const gnosisSafe = await getSafeEthereumInstance(safe.get('address')) const nonce = Date.now() const data = gnosisSafe.contract.changeThreshold.getData(newThreshold) - await createTransaction(safe, "Change Safe's threshold", safe.get('address'), 0, nonce, userAddress, data) + await createTransaction(safe, `Change Safe's threshold [${nonce}]`, safe.get('address'), 0, nonce, userAddress, data) await sleep(1500) // this.props.fetchThreshold(safe.get('address')) } render() { - const { numOwners } = this.props + const { numOwners, onReset } = this.props return ( { From fad5132a60d18368b7680b7cbd6619bce95f70ab Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 09:28:32 +0200 Subject: [PATCH 05/25] WA-238 Adding actual threshold info in the component --- src/routes/safe/component/Threshold/index.jsx | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx index 0ed1da98..61c4c565 100644 --- a/src/routes/safe/component/Threshold/index.jsx +++ b/src/routes/safe/component/Threshold/index.jsx @@ -22,13 +22,13 @@ type Props = SelectorProps & { const THRESHOLD_PARAM = 'threshold' -const ThresholdComponent = ({ numOwners }: Props) => () => ( +const ThresholdComponent = ({ numOwners, safe }: Props) => () => ( {'Change safe\'s threshold'} - {`Actual number of owners: ${numOwners}`} + {`Safe's owners: ${numOwners} and Safe's threshold: ${safe.get('confirmations')}`} { } render() { - const { numOwners, onReset } = this.props + const { numOwners, onReset, safe } = this.props return ( {(submitting: boolean, submitSucceeded: boolean) => ( @@ -84,7 +85,7 @@ class Threshold extends React.PureComponent { variant="raised" color="primary" onClick={submitSucceeded ? onReset : undefined} - type="submit" + type={submitSucceeded ? 'button' : 'submit'} disabled={submitting} > { submitSucceeded ? 'VISIT TXs' : 'FINISH' } From 5b4e0256f272f90b973f84136dc5c096e9875fac Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 10:10:01 +0200 Subject: [PATCH 06/25] WA-235 adding fetchThreshold action --- src/routes/safe/store/actions/fetchThreshold.js | 12 ++++++++++++ 1 file changed, 12 insertions(+) create mode 100644 src/routes/safe/store/actions/fetchThreshold.js diff --git a/src/routes/safe/store/actions/fetchThreshold.js b/src/routes/safe/store/actions/fetchThreshold.js new file mode 100644 index 00000000..2ae28e69 --- /dev/null +++ b/src/routes/safe/store/actions/fetchThreshold.js @@ -0,0 +1,12 @@ +// @flow +import type { Dispatch as ReduxDispatch } from 'redux' +import { type GlobalState } from '~/store/index' +import { getSafeEthereumInstance } from '~/routes/safe/component/AddTransaction/createTransactions' +import updateThreshold from './updateThreshold' + +export default (safeAddress: string) => async (dispatch: ReduxDispatch) => { + const gnosisSafe = await getSafeEthereumInstance(safeAddress) + const actualThreshold = await gnosisSafe.getThreshold() + + return dispatch(updateThreshold(safeAddress, actualThreshold)) +} From b70b70c5df5a5d8b89bc5da4a9e045e3083c68ae Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 10:11:52 +0200 Subject: [PATCH 07/25] WA-235 adding updateThreshold action --- .../safe/store/actions/updateThreshold.js | 19 ++++++++ .../safe/store/test/threshold.reducer.js | 48 +++++++++++++++++++ 2 files changed, 67 insertions(+) create mode 100644 src/routes/safe/store/actions/updateThreshold.js create mode 100644 src/routes/safe/store/test/threshold.reducer.js diff --git a/src/routes/safe/store/actions/updateThreshold.js b/src/routes/safe/store/actions/updateThreshold.js new file mode 100644 index 00000000..b24f41fd --- /dev/null +++ b/src/routes/safe/store/actions/updateThreshold.js @@ -0,0 +1,19 @@ +// @flow +import { createAction } from 'redux-actions' + +export const UPDATE_THRESHOLD = 'UPDATE_THRESHOLD' + +type ThresholdProps = { + safeAddress: string, + threshold: number, +} + +const updateDailyLimit = createAction( + UPDATE_THRESHOLD, + (safeAddress: string, threshold: number): ThresholdProps => ({ + safeAddress, + threshold: Number(threshold), + }), +) + +export default updateDailyLimit diff --git a/src/routes/safe/store/test/threshold.reducer.js b/src/routes/safe/store/test/threshold.reducer.js new file mode 100644 index 00000000..b1d483c3 --- /dev/null +++ b/src/routes/safe/store/test/threshold.reducer.js @@ -0,0 +1,48 @@ +// @flow +import { SAFE_REDUCER_ID } from '~/routes/safe/store/reducer/safe' +import { aNewStore } from '~/store' +import updateThreshold from '~/routes/safe/store/actions/updateThreshold' +import { aDeployedSafe } from './builder/deployedSafe.builder' + +const thresholdReducerTests = () => { + describe('Safe Actions[updateThreshold]', () => { + let store + beforeEach(async () => { + store = aNewStore() + }) + + it('reducer should return 3 when a safe of 3 threshold has just been created', async () => { + // GIVEN + const safeThreshold = 3 + const numOwners = 3 + + // WHEN + const safeAddress = await aDeployedSafe(store, 0.5, safeThreshold, numOwners) + + // THEN + const safes = store.getState()[SAFE_REDUCER_ID] + const threshold = safes.get(safeAddress).get('confirmations') + expect(threshold).not.toBe(undefined) + expect(threshold).toBe(safeThreshold) + }) + + it('reducer should change correctly', async () => { + // GIVEN + const safeThreshold = 3 + const numOwners = 3 + const safeAddress = await aDeployedSafe(store, 0.5, safeThreshold, numOwners) + + // WHEN + const newThreshold = 1 + await store.dispatch(updateThreshold(safeAddress, newThreshold)) + + // THEN + const safes = store.getState()[SAFE_REDUCER_ID] + const threshold = safes.get(safeAddress).get('confirmations') + expect(threshold).not.toBe(undefined) + expect(threshold).toBe(newThreshold) + }) + }) +} + +export default thresholdReducerTests From 13c1152c5b0cd3d9fce9ad0d24d162dededd2ab1 Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 10:13:59 +0200 Subject: [PATCH 08/25] WA-235 Adding threshold test for checking store is updated correctly --- src/routes/safe/store/reducer/safe.js | 3 +++ src/routes/safe/store/test/safe.spec.js | 2 ++ 2 files changed, 5 insertions(+) diff --git a/src/routes/safe/store/reducer/safe.js b/src/routes/safe/store/reducer/safe.js index d430c49d..f8f64485 100644 --- a/src/routes/safe/store/reducer/safe.js +++ b/src/routes/safe/store/reducer/safe.js @@ -7,6 +7,7 @@ import { makeOwner } from '~/routes/safe/store/model/owner' import { type Safe, makeSafe } from '~/routes/safe/store/model/safe' import { load, saveSafes, SAFES_KEY } from '~/utils/localStorage' import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' +import updateThreshold, { UPDATE_THRESHOLD } from '~/routes/safe/store/actions/updateThreshold' export const SAFE_REDUCER_ID = 'safes' @@ -50,4 +51,6 @@ export default handleActions({ }, [UPDATE_DAILY_LIMIT]: (state: State, action: ActionType): State => state.updateIn([action.payload.safeAddress, 'dailyLimit'], () => makeDailyLimit(action.payload.dailyLimit)), + [UPDATE_THRESHOLD]: (state: State, action: ActionType): State => + state.updateIn([action.payload.safeAddress, 'confirmations'], () => action.payload.threshold), }, Map()) diff --git a/src/routes/safe/store/test/safe.spec.js b/src/routes/safe/store/test/safe.spec.js index 3c431b4b..8a0ebd4d 100644 --- a/src/routes/safe/store/test/safe.spec.js +++ b/src/routes/safe/store/test/safe.spec.js @@ -2,6 +2,7 @@ import balanceReducerTests from './balance.reducer' import safeReducerTests from './safe.reducer' import dailyLimitReducerTests from './dailyLimit.reducer' +import thresholdReducerTests from './threshold.reducer' import balanceSelectorTests from './balance.selector' import safeSelectorTests from './safe.selector' import grantedSelectorTests from './granted.selector' @@ -13,6 +14,7 @@ describe('Safe Test suite', () => { safeReducerTests() balanceReducerTests() dailyLimitReducerTests() + thresholdReducerTests() // SAFE SELECTOR safeSelectorTests() From 31e5833b8eadeba2d00c06dba9dc237b3d871626 Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 10:43:56 +0200 Subject: [PATCH 09/25] WA-238 Updating UI after processing TX for changing threshold --- src/routes/safe/component/Threshold/actions.js | 16 ++++++++++++++++ src/routes/safe/component/Threshold/index.jsx | 8 +++++--- .../safe/component/Transactions/actions.js | 8 +++++++- src/routes/safe/component/Transactions/index.jsx | 6 +++++- 4 files changed, 33 insertions(+), 5 deletions(-) create mode 100644 src/routes/safe/component/Threshold/actions.js diff --git a/src/routes/safe/component/Threshold/actions.js b/src/routes/safe/component/Threshold/actions.js new file mode 100644 index 00000000..e0b10a5a --- /dev/null +++ b/src/routes/safe/component/Threshold/actions.js @@ -0,0 +1,16 @@ +// @flow +import fetchThreshold from '~/routes/safe/store/actions/fetchThreshold' +import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' + +type FetchThreshold = typeof fetchThreshold +type FetchTransactions = typeof fetchTransactions + +export type Actions = { + fetchThreshold: FetchThreshold, + fetchTransactions: FetchTransactions, +} + +export default { + fetchThreshold, + fetchTransactions, +} diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx index 61c4c565..b2b66b66 100644 --- a/src/routes/safe/component/Threshold/index.jsx +++ b/src/routes/safe/component/Threshold/index.jsx @@ -13,8 +13,9 @@ import { composeValidators, minValue, maxValue, mustBeInteger, required } from ' import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' import { sleep } from '~/utils/timer' import selector, { type SelectorProps } from './selector' +import actions, { type Actions } from './actions' -type Props = SelectorProps & { +type Props = SelectorProps & Actions & { numOwners: number, safe: Safe, onReset: () => void, @@ -65,7 +66,8 @@ class Threshold extends React.PureComponent { const data = gnosisSafe.contract.changeThreshold.getData(newThreshold) await createTransaction(safe, `Change Safe's threshold [${nonce}]`, safe.get('address'), 0, nonce, userAddress, data) await sleep(1500) - // this.props.fetchThreshold(safe.get('address')) + this.props.fetchTransactions() + this.props.fetchThreshold(safe.get('address')) } render() { @@ -98,4 +100,4 @@ class Threshold extends React.PureComponent { } } -export default connect(selector)(Threshold) +export default connect(selector, actions)(Threshold) diff --git a/src/routes/safe/component/Transactions/actions.js b/src/routes/safe/component/Transactions/actions.js index 681ad469..e0b10a5a 100644 --- a/src/routes/safe/component/Transactions/actions.js +++ b/src/routes/safe/component/Transactions/actions.js @@ -1,10 +1,16 @@ // @flow +import fetchThreshold from '~/routes/safe/store/actions/fetchThreshold' import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' +type FetchThreshold = typeof fetchThreshold +type FetchTransactions = typeof fetchTransactions + export type Actions = { - fetchTransactions: typeof fetchTransactions, + fetchThreshold: FetchThreshold, + fetchTransactions: FetchTransactions, } export default { + fetchThreshold, fetchTransactions, } diff --git a/src/routes/safe/component/Transactions/index.jsx b/src/routes/safe/component/Transactions/index.jsx index 5dac8846..e8a22e87 100644 --- a/src/routes/safe/component/Transactions/index.jsx +++ b/src/routes/safe/component/Transactions/index.jsx @@ -17,10 +17,14 @@ type Props = SelectorProps & Actions & { } class Transactions extends React.Component { onProcessTx = async (tx: Transaction, alreadyConfirmed: number) => { - const { fetchTransactions, safeAddress, userAddress } = this.props + const { + fetchTransactions, safeAddress, userAddress, fetchThreshold, + } = this.props + await processTransaction(safeAddress, tx, alreadyConfirmed, userAddress) await sleep(1200) fetchTransactions() + fetchThreshold(safeAddress) } render() { From 058c6e04fd48b059d082789ab36b42116247e99c Mon Sep 17 00:00:00 2001 From: apanizo Date: Wed, 6 Jun 2018 11:26:45 +0200 Subject: [PATCH 10/25] WA-235 Fix tests - Adapting buttons' index based on the new layout --- .../safe/test/Safe.multisig.1owners1threshold.test.js | 2 +- src/routes/safe/test/Safe.withdrawn.test.js | 6 +++--- src/routes/safe/test/testMultisig.js | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js b/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js index 1567f58c..80d8b697 100644 --- a/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js +++ b/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js @@ -45,7 +45,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[1] + const addTxButton = buttons[2] expect(addTxButton.props.children).toEqual(ADD_MULTISIG_BUTTON_TEXT) await sleep(1800) // Give time to enable Add button TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(addTxButton, 'button')[0]) diff --git a/src/routes/safe/test/Safe.withdrawn.test.js b/src/routes/safe/test/Safe.withdrawn.test.js index a0803c57..a985b3cc 100644 --- a/src/routes/safe/test/Safe.withdrawn.test.js +++ b/src/routes/safe/test/Safe.withdrawn.test.js @@ -46,7 +46,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const withdrawnButton = buttons[0] + const withdrawnButton = buttons[1] expect(withdrawnButton.props.children).toEqual(WITHDRAWN_BUTTON_TEXT) TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(withdrawnButton, 'button')[0]) await sleep(4000) @@ -96,7 +96,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { const Safe = TestUtils.findRenderedComponentWithType(SafeDom, SafeView) // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[1] + const addTxButton = buttons[2] expect(addTxButton.props.children).toEqual(ADD_MULTISIG_BUTTON_TEXT) expect(addTxButton.props.disabled).toBe(true) @@ -110,7 +110,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { const Safe = TestUtils.findRenderedComponentWithType(SafeDom, SafeView) // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[0] + const addTxButton = buttons[1] expect(addTxButton.props.children).toEqual(WITHDRAWN_BUTTON_TEXT) expect(addTxButton.props.disabled).toBe(true) diff --git a/src/routes/safe/test/testMultisig.js b/src/routes/safe/test/testMultisig.js index c4264cd5..272ce34d 100644 --- a/src/routes/safe/test/testMultisig.js +++ b/src/routes/safe/test/testMultisig.js @@ -43,7 +43,7 @@ export const addFundsTo = async (SafeDom, destination: string) => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[1] + const addTxButton = buttons[2] expect(addTxButton.props.children).toEqual(ADD_MULTISIG_BUTTON_TEXT) await sleep(1800) // Give time to enable Add button TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(addTxButton, 'button')[0]) @@ -54,7 +54,7 @@ export const listTxsOf = (SafeDom) => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const seeTx = buttons[2] + const seeTx = buttons[3] expect(seeTx.props.children).toEqual(SEE_MULTISIG_BUTTON_TEXT) TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(seeTx, 'button')[0]) } From b4a293c4ffa1e2d8cf1ce64b099f0da27d564a57 Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 08:18:40 +0200 Subject: [PATCH 11/25] WA-235 Updating contracts --- safe-contracts/build/contracts/CreateAndAddModules.json | 8 +++++++- safe-contracts/build/contracts/DailyLimitModule.json | 8 +++++++- .../build/contracts/GnosisSafePersonalEdition.json | 8 +++++++- safe-contracts/build/contracts/GnosisSafeTeamEdition.json | 8 +++++++- safe-contracts/build/contracts/Migrations.json | 8 +++++++- safe-contracts/build/contracts/MultiSend.json | 8 +++++++- safe-contracts/build/contracts/ProxyFactory.json | 8 +++++++- safe-contracts/build/contracts/SocialRecoveryModule.json | 8 +++++++- safe-contracts/build/contracts/StateChannelModule.json | 8 +++++++- safe-contracts/build/contracts/WhitelistModule.json | 8 +++++++- 10 files changed, 70 insertions(+), 10 deletions(-) diff --git a/safe-contracts/build/contracts/CreateAndAddModules.json b/safe-contracts/build/contracts/CreateAndAddModules.json index 1e7d5e36..62751b26 100644 --- a/safe-contracts/build/contracts/CreateAndAddModules.json +++ b/safe-contracts/build/contracts/CreateAndAddModules.json @@ -1276,8 +1276,14 @@ "links": {}, "address": "0x5fd674bc2873513f8e5a19d69637d0211e476380", "transactionHash": "0x288775644c087eed5e41b96fecebdb23be4e6d40bef5b6fb9a2876c2a3145157" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0x194371bd036c34314ded31a7ffe7e66f5461a62d", + "transactionHash": "0x288775644c087eed5e41b96fecebdb23be4e6d40bef5b6fb9a2876c2a3145157" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.138Z" + "updatedAt": "2018-06-06T14:51:43.695Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/DailyLimitModule.json b/safe-contracts/build/contracts/DailyLimitModule.json index e9bcb94c..5b0be962 100644 --- a/safe-contracts/build/contracts/DailyLimitModule.json +++ b/safe-contracts/build/contracts/DailyLimitModule.json @@ -6699,8 +6699,14 @@ "links": {}, "address": "0x3bdceb07fddd50d259a059ca9a75ecda561d4afc", "transactionHash": "0xf501438a4ec967e2928d922e4af568a2a5365002f8b3f9e32117bbacfaa49331" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0x0fb244fb862c95c66250c6c72e7e91a3d41a47d5", + "transactionHash": "0xf501438a4ec967e2928d922e4af568a2a5365002f8b3f9e32117bbacfaa49331" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.129Z" + "updatedAt": "2018-06-06T14:51:43.688Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/GnosisSafePersonalEdition.json b/safe-contracts/build/contracts/GnosisSafePersonalEdition.json index 03ba515f..e3ea4668 100644 --- a/safe-contracts/build/contracts/GnosisSafePersonalEdition.json +++ b/safe-contracts/build/contracts/GnosisSafePersonalEdition.json @@ -9864,8 +9864,14 @@ "links": {}, "address": "0x8c55b458a53e8c6e9efa7f54e7be9ca76b43dd9b", "transactionHash": "0x67117c1452ee2f4b904621b6f30790ff998d1f1a72f11c6b71ef47e3dd254724" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0xc3d42147f9b51c9749d3362e8b9ee6cb94861778", + "transactionHash": "0x67117c1452ee2f4b904621b6f30790ff998d1f1a72f11c6b71ef47e3dd254724" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.124Z" + "updatedAt": "2018-06-06T14:51:43.672Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/GnosisSafeTeamEdition.json b/safe-contracts/build/contracts/GnosisSafeTeamEdition.json index 14a19442..9cb27b58 100644 --- a/safe-contracts/build/contracts/GnosisSafeTeamEdition.json +++ b/safe-contracts/build/contracts/GnosisSafeTeamEdition.json @@ -6876,8 +6876,14 @@ "links": {}, "address": "0xd4edae2f2d5718d1798deb48c062b939d6e9d4f4", "transactionHash": "0xa71d3b0b3752acc18733fa881f70c256d63562f28ccca9af910fad3beee9181a" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0xf2bc99498b610a01d76358d8e2fe251c9783a216", + "transactionHash": "0xa71d3b0b3752acc18733fa881f70c256d63562f28ccca9af910fad3beee9181a" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.114Z" + "updatedAt": "2018-06-06T14:51:43.677Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/Migrations.json b/safe-contracts/build/contracts/Migrations.json index d9d7c21a..5d1fcd56 100644 --- a/safe-contracts/build/contracts/Migrations.json +++ b/safe-contracts/build/contracts/Migrations.json @@ -1398,8 +1398,14 @@ "links": {}, "address": "0x2ebea54cbbd4f5491deba7a37605f8f0be3e3c9b", "transactionHash": "0xb6a19a7a679a1474c09c651e4151421f210afa3f47effed019d4c0206144ee5f" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0x4aa39923aa66871debec9b8aa9008a3b220eb1df", + "transactionHash": "0xb6a19a7a679a1474c09c651e4151421f210afa3f47effed019d4c0206144ee5f" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.139Z" + "updatedAt": "2018-06-06T14:51:43.697Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/MultiSend.json b/safe-contracts/build/contracts/MultiSend.json index 79da2b5d..41362d48 100644 --- a/safe-contracts/build/contracts/MultiSend.json +++ b/safe-contracts/build/contracts/MultiSend.json @@ -360,8 +360,14 @@ "links": {}, "address": "0x3946fcaaa0ba21aaffc5e06a3cc45debc9e07f7f", "transactionHash": "0xd044f1662e339061a8cabf2b06ac94a9f86fcccf3f5d80ebd1bea2a7542d4021" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0xef9e7829f057e4d640bf66e17e06b3ab5cae508d", + "transactionHash": "0xd044f1662e339061a8cabf2b06ac94a9f86fcccf3f5d80ebd1bea2a7542d4021" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.139Z" + "updatedAt": "2018-06-06T14:51:43.696Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/ProxyFactory.json b/safe-contracts/build/contracts/ProxyFactory.json index 5d2b5bed..a218d2eb 100644 --- a/safe-contracts/build/contracts/ProxyFactory.json +++ b/safe-contracts/build/contracts/ProxyFactory.json @@ -1011,8 +1011,14 @@ "links": {}, "address": "0xbefd9f4a40b1bec8ec730969a3508d1739fb2742", "transactionHash": "0x75ad1066b44cd801ac66a316dbe4c09e72636d72b70fd62eb647295a0fc5e285" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0xaa973df8ec251cf67ec387d5627d42dbb738605f", + "transactionHash": "0x75ad1066b44cd801ac66a316dbe4c09e72636d72b70fd62eb647295a0fc5e285" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.112Z" + "updatedAt": "2018-06-06T14:51:43.668Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/SocialRecoveryModule.json b/safe-contracts/build/contracts/SocialRecoveryModule.json index 8f32e0de..5e7d519b 100644 --- a/safe-contracts/build/contracts/SocialRecoveryModule.json +++ b/safe-contracts/build/contracts/SocialRecoveryModule.json @@ -7310,8 +7310,14 @@ "links": {}, "address": "0xfb1771240bb7edf209c70bd520a5d5424d23b084", "transactionHash": "0xf0cd95843453bdac02ad8018ef507479ea62989e56d69ad0ac1aad9d3a8515d2" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0x9be4b89520d1dd6f2d115192587689c6c9bd1a99", + "transactionHash": "0xf0cd95843453bdac02ad8018ef507479ea62989e56d69ad0ac1aad9d3a8515d2" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.135Z" + "updatedAt": "2018-06-06T14:51:43.692Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/StateChannelModule.json b/safe-contracts/build/contracts/StateChannelModule.json index a7ce755f..2ffa0349 100644 --- a/safe-contracts/build/contracts/StateChannelModule.json +++ b/safe-contracts/build/contracts/StateChannelModule.json @@ -5881,8 +5881,14 @@ "links": {}, "address": "0x589fd9eea7cca488a80e17a5105befff9616f11d", "transactionHash": "0x0396e1c9da4fa7bd313286e6033446dbb6e491f267956f8cf13202ce534fd0e6" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0xe8abcdb37db8e7c563de32c5d207433ce44d1445", + "transactionHash": "0x0396e1c9da4fa7bd313286e6033446dbb6e491f267956f8cf13202ce534fd0e6" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.118Z" + "updatedAt": "2018-06-06T14:51:43.684Z" } \ No newline at end of file diff --git a/safe-contracts/build/contracts/WhitelistModule.json b/safe-contracts/build/contracts/WhitelistModule.json index 7eff5a28..c0307ad7 100644 --- a/safe-contracts/build/contracts/WhitelistModule.json +++ b/safe-contracts/build/contracts/WhitelistModule.json @@ -4354,8 +4354,14 @@ "links": {}, "address": "0xca574a31a4cf1eeabeecaffc555bdc7f91c5caf9", "transactionHash": "0x463374c2fbc7eaff5b87e65c6a8fdc1177ef82c66084df6e7b88b506f99b193c" + }, + "1528296677763": { + "events": {}, + "links": {}, + "address": "0x66c535e20f0c90530431ebab626da0ebdd55ec2d", + "transactionHash": "0x463374c2fbc7eaff5b87e65c6a8fdc1177ef82c66084df6e7b88b506f99b193c" } }, "schemaVersion": "2.0.0", - "updatedAt": "2018-06-04T10:56:37.132Z" + "updatedAt": "2018-06-06T14:51:43.700Z" } \ No newline at end of file From e86f7827f05f17804f5fc82d69d0d56aa780d74c Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 09:41:47 +0200 Subject: [PATCH 12/25] WA-234 Fixing flow errors --- .../AddTransaction/createTransactions.js | 2 +- src/routes/safe/component/Threshold/index.jsx | 8 ++++-- .../Safe.multisig.3owners1threshold.test.js | 1 + .../Safe.multisig.3owners3threshold.test.js | 3 +++ src/routes/safe/test/Safe.threshold.test.js | 12 +++++++-- src/routes/safe/test/testMultisig.js | 25 +++++++++++++------ 6 files changed, 38 insertions(+), 13 deletions(-) diff --git a/src/routes/safe/component/AddTransaction/createTransactions.js b/src/routes/safe/component/AddTransaction/createTransactions.js index 2bc545cc..67b26891 100644 --- a/src/routes/safe/component/AddTransaction/createTransactions.js +++ b/src/routes/safe/component/AddTransaction/createTransactions.js @@ -80,7 +80,7 @@ const hasOneOwner = (safe: Safe) => { return owners.count() === 1 } -export const getSafeEthereumInstance = async (safeAddress) => { +export const getSafeEthereumInstance = async (safeAddress: string) => { const web3 = getWeb3() const GnosisSafe = await getGnosisSafeContract(web3) return GnosisSafe.at(safeAddress) diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx index b2b66b66..f7fffec8 100644 --- a/src/routes/safe/component/Threshold/index.jsx +++ b/src/routes/safe/component/Threshold/index.jsx @@ -12,18 +12,22 @@ import Row from '~/components/layout/Row' import { composeValidators, minValue, maxValue, mustBeInteger, required } from '~/components/forms/validator' import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' import { sleep } from '~/utils/timer' +import { type Safe } from '~/routes/safe/store/model/safe' import selector, { type SelectorProps } from './selector' import actions, { type Actions } from './actions' -type Props = SelectorProps & Actions & { +type ThresholdProps = { numOwners: number, safe: Safe, +} + +type Props = SelectorProps & Actions & ThresholdProps & { onReset: () => void, } const THRESHOLD_PARAM = 'threshold' -const ThresholdComponent = ({ numOwners, safe }: Props) => () => ( +const ThresholdComponent = ({ numOwners, safe }: ThresholdProps) => () => ( {'Change safe\'s threshold'} diff --git a/src/routes/safe/test/Safe.multisig.3owners1threshold.test.js b/src/routes/safe/test/Safe.multisig.3owners1threshold.test.js index 2227b94e..0a566f46 100644 --- a/src/routes/safe/test/Safe.multisig.3owners1threshold.test.js +++ b/src/routes/safe/test/Safe.multisig.3owners1threshold.test.js @@ -53,6 +53,7 @@ describe('React DOM TESTS > Multisig transactions from safe [3 owners & 1 thresh const confirmed = paragraphs[3].innerHTML const tx = getTransactionFromReduxStore(store, address) + if (!tx) throw new Error() expect(confirmed).toBe(tx.get('tx')) const ownerTx = paragraphs[6].innerHTML diff --git a/src/routes/safe/test/Safe.multisig.3owners3threshold.test.js b/src/routes/safe/test/Safe.multisig.3owners3threshold.test.js index f57014f8..d5ca92ac 100644 --- a/src/routes/safe/test/Safe.multisig.3owners3threshold.test.js +++ b/src/routes/safe/test/Safe.multisig.3owners3threshold.test.js @@ -45,6 +45,7 @@ describe('React DOM TESTS > Multisig transactions from safe [3 owners & 3 thresh const getAlreadyConfirmed = () => { const tx = getTransactionFromReduxStore(store, address) + if (!tx) throw new Error() const confirmed = confirmationsTransactionSelector(store.getState(), { transaction: tx }) return confirmed @@ -53,6 +54,7 @@ describe('React DOM TESTS > Multisig transactions from safe [3 owners & 3 thresh const makeConfirmation = async (executor) => { const alreadyConfirmed = getAlreadyConfirmed() const tx = getTransactionFromReduxStore(store, address) + if (!tx) throw new Error() await processTransaction(address, tx, alreadyConfirmed, executor) await sleep(800) store.dispatch(fetchTransactions()) @@ -96,6 +98,7 @@ describe('React DOM TESTS > Multisig transactions from safe [3 owners & 3 thresh const confirmedExecuted = paragraphsExecuted[3].innerHTML const tx = getTransactionFromReduxStore(store, address) + if (!tx) throw new Error() expect(confirmedExecuted).toBe(tx.get('tx')) }) }) diff --git a/src/routes/safe/test/Safe.threshold.test.js b/src/routes/safe/test/Safe.threshold.test.js index 1ca38066..edd5837c 100644 --- a/src/routes/safe/test/Safe.threshold.test.js +++ b/src/routes/safe/test/Safe.threshold.test.js @@ -3,6 +3,7 @@ import { aNewStore } from '~/store' import { aDeployedSafe } from '~/routes/safe/store/test/builder/deployedSafe.builder' import { getWeb3 } from '~/wallets/getWeb3' import { sleep } from '~/utils/timer' +import { type Match } from 'react-router-dom' import { promisify } from '~/utils/promisify' import { processTransaction } from '~/routes/safe/component/Transactions/processTransactions' import { confirmationsTransactionSelector, safeSelector, safeTransactionsSelector } from '~/routes/safe/store/selectors/index' @@ -22,6 +23,7 @@ describe('React DOM TESTS > Change threshold', () => { const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) const match: Match = buildMathPropsFrom(address) const safe = safeSelector(store.getState(), { match }) + if (!safe) throw new Error() const web3 = getWeb3() const GnosisSafe = await getGnosisSafeContract(web3) const gnosisSafe = GnosisSafe.at(address) @@ -37,7 +39,8 @@ describe('React DOM TESTS > Change threshold', () => { const transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) expect(transactions.count()).toBe(1) - const thresholdTx: Transaction = transactions.get(0) + const thresholdTx = transactions.get(0) + if (!thresholdTx) throw new Error() expect(thresholdTx.get('tx')).not.toBe(null) expect(thresholdTx.get('tx')).not.toBe(undefined) expect(thresholdTx.get('tx')).not.toBe('') @@ -48,6 +51,7 @@ describe('React DOM TESTS > Change threshold', () => { const changeThreshold = async (store, safeAddress, executor) => { const tx = getTransactionFromReduxStore(store, safeAddress) + if (!tx) throw new Error() const confirmed = confirmationsTransactionSelector(store.getState(), { transaction: tx }) const data = tx.get('data') expect(data).not.toBe(null) @@ -66,6 +70,7 @@ describe('React DOM TESTS > Change threshold', () => { const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) const match: Match = buildMathPropsFrom(address) const safe = safeSelector(store.getState(), { match }) + if (!safe) throw new Error() const web3 = getWeb3() const GnosisSafe = await getGnosisSafeContract(web3) const gnosisSafe = GnosisSafe.at(address) @@ -78,9 +83,11 @@ describe('React DOM TESTS > Change threshold', () => { await store.dispatch(fetchTransactions()) let transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) + if (!transactions) throw new Error() expect(transactions.count()).toBe(1) - let thresholdTx: Transaction = transactions.get(0) + let thresholdTx = transactions.get(0) + if (!thresholdTx) throw new Error() expect(thresholdTx.get('tx')).toBe('') let firstOwnerConfirmation = thresholdTx.get('confirmations').get(0) if (!firstOwnerConfirmation) throw new Error() @@ -103,6 +110,7 @@ describe('React DOM TESTS > Change threshold', () => { expect(transactions.count()).toBe(1) thresholdTx = transactions.get(0) + if (!thresholdTx) throw new Error() expect(thresholdTx.get('tx')).not.toBe(undefined) expect(thresholdTx.get('tx')).not.toBe(null) expect(thresholdTx.get('tx')).not.toBe('') diff --git a/src/routes/safe/test/testMultisig.js b/src/routes/safe/test/testMultisig.js index 272ce34d..9c10e3c2 100644 --- a/src/routes/safe/test/testMultisig.js +++ b/src/routes/safe/test/testMultisig.js @@ -9,8 +9,13 @@ import SafeView from '~/routes/safe/component/Safe' import TransactionsComponent from '~/routes/safe/component/Transactions' import TransactionComponent from '~/routes/safe/component/Transactions/Transaction' import { safeTransactionsSelector } from '~/routes/safe/store/selectors/index' +import { type GlobalState } from '~/store/index' -export const createMultisigTxFilling = async (SafeDom, AddTransactionComponent, store) => { +export const createMultisigTxFilling = async ( + SafeDom: React$Component, + AddTransactionComponent: React$ElementType, + store: Store, +) => { // Get AddTransaction form component const AddTransaction = TestUtils.findRenderedComponentWithType(SafeDom, AddTransactionComponent) @@ -36,7 +41,7 @@ export const checkBalanceOf = async (addressToTest: string, value: string) => { expect(safeBalance).toBe(value) } -export const addFundsTo = async (SafeDom, destination: string) => { +export const addFundsTo = async (SafeDom: React$Component, destination: string) => { // add funds to safe await addEtherTo(destination, '0.1') const Safe = TestUtils.findRenderedComponentWithType(SafeDom, SafeView) @@ -49,7 +54,7 @@ export const addFundsTo = async (SafeDom, destination: string) => { TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(addTxButton, 'button')[0]) } -export const listTxsOf = (SafeDom) => { +export const listTxsOf = (SafeDom: React$Component) => { const Safe = TestUtils.findRenderedComponentWithType(SafeDom, SafeView) // $FlowFixMe @@ -59,7 +64,7 @@ export const listTxsOf = (SafeDom) => { TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(seeTx, 'button')[0]) } -export const getTagFromTransaction = (SafeDom, tag: string) => { +export const getTagFromTransaction = (SafeDom: React$Component, tag: string) => { const Transactions = TestUtils.findRenderedComponentWithType(SafeDom, TransactionsComponent) if (!Transactions) throw new Error() const Transaction = TestUtils.findRenderedComponentWithType(Transactions, TransactionComponent) @@ -68,7 +73,11 @@ export const getTagFromTransaction = (SafeDom, tag: string) => { return TestUtils.scryRenderedDOMComponentsWithTag(Transaction, tag) } -export const expandTransactionOf = async (SafeDom, numOwners, safeThreshold) => { +export const expandTransactionOf = async ( + SafeDom: React$Component, + numOwners: number, + safeThreshold: number, +) => { const paragraphs = getTagFromTransaction(SafeDom, 'p') TestUtils.Simulate.click(paragraphs[2]) // expanded await sleep(1000) // Time to expand @@ -80,13 +89,13 @@ export const expandTransactionOf = async (SafeDom, numOwners, safeThreshold) => expect(paragraphsExpanded.length).toBe(paragraphs.length + numOwners) } -export const getTransactionFromReduxStore = (store, address) => { +export const getTransactionFromReduxStore = (store: Store, address: string, index: number = 0) => { const transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) - return transactions.get(0) + return transactions.get(index) } -export const confirmOwners = async (SafeDom, ...statusses: string[]) => { +export const confirmOwners = async (SafeDom: React$Component, ...statusses: string[]) => { const paragraphsWithOwners = getTagFromTransaction(SafeDom, 'h3') for (let i = 0; i < statusses.length; i += 1) { const ownerIndex = i + 6 From d9c05e3efc4792db73beb99545871f8c4844c4da Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 10:26:52 +0200 Subject: [PATCH 13/25] WA-234 Added test change owners test --- src/routes/safe/test/Safe.owners.test.js | 195 +++++++++++++++++++++++ 1 file changed, 195 insertions(+) create mode 100644 src/routes/safe/test/Safe.owners.test.js diff --git a/src/routes/safe/test/Safe.owners.test.js b/src/routes/safe/test/Safe.owners.test.js new file mode 100644 index 00000000..73bae217 --- /dev/null +++ b/src/routes/safe/test/Safe.owners.test.js @@ -0,0 +1,195 @@ +// @flow +import { aNewStore } from '~/store' +import { aDeployedSafe } from '~/routes/safe/store/test/builder/deployedSafe.builder' +import { getWeb3 } from '~/wallets/getWeb3' +import { sleep } from '~/utils/timer' +import { type Match } from 'react-router-dom' +import { promisify } from '~/utils/promisify' +import { processTransaction } from '~/routes/safe/component/Transactions/processTransactions' +import { confirmationsTransactionSelector, safeSelector, safeTransactionsSelector } from '~/routes/safe/store/selectors/index' +import { getTransactionFromReduxStore } from '~/routes/safe/test/testMultisig' +import { buildMathPropsFrom } from '~/test/buildReactRouterProps' +import { createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' +import { getGnosisSafeContract } from '~/wallets/safeContracts' +import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' +import { type GlobalState } from '~/store/index' +import { type Safe } from '~/routes/safe/store/model/safe' +import { type Transaction } from '~/routes/safe/store/model/transaction' + +const getSafeFrom = (state: GlobalState, safeAddress: string): Safe => { + const match: Match = buildMathPropsFrom(safeAddress) + const safe = safeSelector(state, { match }) + if (!safe) throw new Error() + + return safe +} + +const getGnosisSafeInstanceAt = async (safeAddress: string) => { + const web3 = getWeb3() + const GnosisSafe = await getGnosisSafeContract(web3) + const gnosisSafe = GnosisSafe.at(safeAddress) + + return gnosisSafe +} + +describe('React DOM TESTS > Add and remove owners', () => { + const assureExecuted = (transaction: Transaction) => { + expect(transaction.get('tx')).not.toBe(null) + expect(transaction.get('tx')).not.toBe(undefined) + expect(transaction.get('tx')).not.toBe('') + } + + const assureThresholdIs = async (gnosisSafe, threshold: number) => { + const safeThreshold = await gnosisSafe.getThreshold() + expect(Number(safeThreshold)).toEqual(threshold) + } + + const assureOwnersAre = async (gnosisSafe, ...owners) => { + const safeOwners = await gnosisSafe.getOwners() + expect(safeOwners.length).toEqual(owners.length) + for (let i = 0; i < owners.length; i += 1) { + expect(safeOwners[i]).toBe(owners[i]) + } + } + + it('adds owner without increasing the threshold', async () => { + // GIVEN + const numOwners = 2 + const threshold = 1 + const store = aNewStore() + const address = await aDeployedSafe(store, 10, threshold, numOwners) + const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) + const safe = getSafeFrom(store.getState(), address) + const gnosisSafe = await getGnosisSafeInstanceAt(address) + + // WHEN + await assureThresholdIs(gnosisSafe, 1) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1]) + const nonce = Date.now() + const accountIndex = 5 + const data = gnosisSafe.contract.addOwnerWithThreshold.getData(accounts[accountIndex], 1) + await createTransaction(safe, `Add Owner with index ${accountIndex}`, address, 0, nonce, accounts[0], data) + await sleep(1500) + await store.dispatch(fetchTransactions()) + + // THEN + const transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) + expect(transactions.count()).toBe(1) + const tx = transactions.get(0) + if (!tx) throw new Error() + assureExecuted(tx) + await assureOwnersAre(gnosisSafe, accounts[5], accounts[0], accounts[1]) + await assureThresholdIs(gnosisSafe, 1) + }) + + it('adds owner increasing the threshold', async () => { + // GIVEN + const numOwners = 2 + const threshold = 1 + const store = aNewStore() + const address = await aDeployedSafe(store, 10, threshold, numOwners) + const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) + const safe = getSafeFrom(store.getState(), address) + const gnosisSafe = await getGnosisSafeInstanceAt(address) + + // WHEN + await assureThresholdIs(gnosisSafe, 1) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1]) + const nonce = Date.now() + const accountIndex = 5 + const data = gnosisSafe.contract.addOwnerWithThreshold.getData(accounts[accountIndex], 2) + await createTransaction(safe, `Add Owner with index ${accountIndex}`, address, 0, nonce, accounts[0], data) + await sleep(1500) + await store.dispatch(fetchTransactions()) + + // THEN + const transactions = safeTransactionsSelector(store.getState(), { safeAddress: address }) + expect(transactions.count()).toBe(1) + const tx = transactions.get(0) + if (!tx) throw new Error() + assureExecuted(tx) + await assureOwnersAre(gnosisSafe, accounts[accountIndex], accounts[0], accounts[1]) + await assureThresholdIs(gnosisSafe, 2) + }) + + const processOwnerModification = async (store, safeAddress, executor) => { + const tx = getTransactionFromReduxStore(store, safeAddress) + if (!tx) throw new Error() + const confirmed = confirmationsTransactionSelector(store.getState(), { transaction: tx }) + const data = tx.get('data') + expect(data).not.toBe(null) + expect(data).not.toBe(undefined) + expect(data).not.toBe('') + + await processTransaction(safeAddress, tx, confirmed, executor) + await sleep(1800) + } + + it('remove owner without decreasing the threshold', async () => { + // GIVEN + const numOwners = 3 + const threshold = 2 + const store = aNewStore() + const address = await aDeployedSafe(store, 10, threshold, numOwners) + const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) + const safe = getSafeFrom(store.getState(), address) + const gnosisSafe = await getGnosisSafeInstanceAt(address) + + // WHEN + await assureThresholdIs(gnosisSafe, 2) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1], accounts[2]) + const nonce = Date.now() + const accountIndex = 2 + const data = gnosisSafe.contract.removeOwner.getData(accounts[accountIndex - 1], accounts[accountIndex], 2) + await createTransaction(safe, `Remove owner Address 3 ${nonce}`, address, 0, nonce, accounts[0], data) + await sleep(1500) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1], accounts[2]) + await store.dispatch(fetchTransactions()) + + + processOwnerModification(store, address, accounts[1]) + await sleep(3000) + await store.dispatch(fetchTransactions()) + await sleep(3000) + const tx = getTransactionFromReduxStore(store, address) + if (!tx) throw new Error() + const txHash = tx.get('tx') + expect(txHash).not.toBe('') + await assureThresholdIs(gnosisSafe, 2) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1]) + }) + + it('remove owner decreasing the threshold', async () => { + // GIVEN + const numOwners = 2 + const threshold = 2 + const store = aNewStore() + const address = await aDeployedSafe(store, 10, threshold, numOwners) + const accounts = await promisify(cb => getWeb3().eth.getAccounts(cb)) + const safe = getSafeFrom(store.getState(), address) + const gnosisSafe = await getGnosisSafeInstanceAt(address) + + // WHEN + await assureThresholdIs(gnosisSafe, 2) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1]) + const nonce = Date.now() + const accountIndex = 1 + const data = gnosisSafe.contract.removeOwner.getData(accounts[accountIndex - 1], accounts[accountIndex], 1) + await createTransaction(safe, `Remove owner Address 2 ${nonce}`, address, 0, nonce, accounts[0], data) + await sleep(1500) + await assureOwnersAre(gnosisSafe, accounts[0], accounts[1]) + await store.dispatch(fetchTransactions()) + + + processOwnerModification(store, address, accounts[1]) + await sleep(3000) + await store.dispatch(fetchTransactions()) + await sleep(3000) + const tx = getTransactionFromReduxStore(store, address) + if (!tx) throw new Error() + const txHash = tx.get('tx') + expect(txHash).not.toBe('') + await assureThresholdIs(gnosisSafe, 1) + await assureOwnersAre(gnosisSafe, accounts[0]) + }) +}) From 2609d3005800292a009436a859dba5ddf4645e95 Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 13:28:53 +0200 Subject: [PATCH 14/25] WA-238 Add Checkbox form component --- src/components/forms/Checkbox/index.jsx | 27 +++++++++++++++++++++++++ 1 file changed, 27 insertions(+) create mode 100644 src/components/forms/Checkbox/index.jsx diff --git a/src/components/forms/Checkbox/index.jsx b/src/components/forms/Checkbox/index.jsx new file mode 100644 index 00000000..b20e101e --- /dev/null +++ b/src/components/forms/Checkbox/index.jsx @@ -0,0 +1,27 @@ +// @flow +import React from 'react' +import Checkbox, { type CheckoxProps } from 'material-ui/Checkbox' + +class GnoCheckbox extends React.PureComponent { + render() { + const { + input: { + checked, name, onChange, ...restInput + }, + meta, + ...rest + } = this.props + + return ( + + ) + } +} + +export default GnoCheckbox From 4b2e0d3bebc78f7dd094c47fc05db3aa9494c83f Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 13:30:43 +0200 Subject: [PATCH 15/25] WA-238 Add Owner logic phase not updated the UI --- .../component/AddOwner/AddOwnerForm/index.jsx | 71 +++++++++++++ .../safe/component/AddOwner/Review/index.jsx | 43 ++++++++ src/routes/safe/component/AddOwner/actions.js | 16 +++ src/routes/safe/component/AddOwner/index.jsx | 99 +++++++++++++++++++ .../safe/component/AddOwner/selector.js | 11 +++ src/routes/safe/component/Safe/Owners.jsx | 13 ++- src/routes/safe/component/Safe/index.jsx | 9 +- .../safe/component/Transactions/index.jsx | 1 + 8 files changed, 261 insertions(+), 2 deletions(-) create mode 100644 src/routes/safe/component/AddOwner/AddOwnerForm/index.jsx create mode 100644 src/routes/safe/component/AddOwner/Review/index.jsx create mode 100644 src/routes/safe/component/AddOwner/actions.js create mode 100644 src/routes/safe/component/AddOwner/index.jsx create mode 100644 src/routes/safe/component/AddOwner/selector.js diff --git a/src/routes/safe/component/AddOwner/AddOwnerForm/index.jsx b/src/routes/safe/component/AddOwner/AddOwnerForm/index.jsx new file mode 100644 index 00000000..721869f2 --- /dev/null +++ b/src/routes/safe/component/AddOwner/AddOwnerForm/index.jsx @@ -0,0 +1,71 @@ +// @flow +import * as React from 'react' +import Field from '~/components/forms/Field' +import TextField from '~/components/forms/TextField' +import Checkbox from '~/components/forms/Checkbox' +import { composeValidators, required, mustBeEthereumAddress, uniqueAddress } from '~/components/forms/validator' +import Block from '~/components/layout/Block' +import Heading from '~/components/layout/Heading' + +export const CONFIRMATIONS_ERROR = 'Number of confirmations can not be higher than the number of owners' + +export const NAME_PARAM = 'name' +export const OWNER_ADDRESS_PARAM = 'ownerAddress' +export const INCREASE_PARAM = 'increase' + +export const safeFieldsValidation = (values: Object) => { + const errors = {} + + if (Number.parseInt(values.owners, 10) < Number.parseInt(values.confirmations, 10)) { + errors.confirmations = CONFIRMATIONS_ERROR + } + + return errors +} + +type Props = { + numOwners: number, + threshold: number, + addresses: string[] +} + +const AddOwnerForm = ({ addresses, numOwners, threshold }: Props) => () => ( + + + Add Owner + + + {`Actual number of owners: ${numOwners}, with threshold: ${threshold}`} + + + + + + + + + + Increase owner? + + +) + +export default AddOwnerForm diff --git a/src/routes/safe/component/AddOwner/Review/index.jsx b/src/routes/safe/component/AddOwner/Review/index.jsx new file mode 100644 index 00000000..6bc31b77 --- /dev/null +++ b/src/routes/safe/component/AddOwner/Review/index.jsx @@ -0,0 +1,43 @@ +// @flow +import * as React from 'react' +import { CircularProgress } from 'material-ui/Progress' +import Block from '~/components/layout/Block' +import Bold from '~/components/layout/Bold' +import Heading from '~/components/layout/Heading' +import Paragraph from '~/components/layout/Paragraph' +import { NAME_PARAM, OWNER_ADDRESS_PARAM, INCREASE_PARAM } from '~/routes/safe/component/AddOwner/AddOwnerForm' + +type FormProps = { + values: Object, + submitting: boolean, +} + +const spinnerStyle = { + minHeight: '50px', +} + +const Review = () => ({ values, submitting }: FormProps) => { + const text = values[INCREASE_PARAM] + ? 'This operation will increase the threshold of the safe' + : 'This operation will not modify the threshold of the safe' + + return ( + + Review the Add Owner operation + + Owner Name: {values[NAME_PARAM]} + + + Owner Address: {values[OWNER_ADDRESS_PARAM]} + + + {text} + + + { submitting && } + + + ) +} + +export default Review diff --git a/src/routes/safe/component/AddOwner/actions.js b/src/routes/safe/component/AddOwner/actions.js new file mode 100644 index 00000000..e0b10a5a --- /dev/null +++ b/src/routes/safe/component/AddOwner/actions.js @@ -0,0 +1,16 @@ +// @flow +import fetchThreshold from '~/routes/safe/store/actions/fetchThreshold' +import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' + +type FetchThreshold = typeof fetchThreshold +type FetchTransactions = typeof fetchTransactions + +export type Actions = { + fetchThreshold: FetchThreshold, + fetchTransactions: FetchTransactions, +} + +export default { + fetchThreshold, + fetchTransactions, +} diff --git a/src/routes/safe/component/AddOwner/index.jsx b/src/routes/safe/component/AddOwner/index.jsx new file mode 100644 index 00000000..4869314f --- /dev/null +++ b/src/routes/safe/component/AddOwner/index.jsx @@ -0,0 +1,99 @@ +// @flow +import * as React from 'react' +import { List } from 'immutable' +import Stepper from '~/components/Stepper' +import { sleep } from '~/utils/timer' +import { connect } from 'react-redux' +import { type Safe } from '~/routes/safe/store/model/safe' +import { type Owner } from '~/routes/safe/store/model/owner' +import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' +import AddOwnerForm, { NAME_PARAM, OWNER_ADDRESS_PARAM, INCREASE_PARAM } from './AddOwnerForm' +import Review from './Review' +import selector, { type SelectorProps } from './selector' +import actions, { type Actions } from './actions' + +const getSteps = () => [ + 'Fill Owner Form', 'Review Withdrawn', +] + +type Props = SelectorProps & Actions & { + safe: Safe, + threshold: number, +} + +type State = { + done: boolean, +} + +export const ADD_OWNER_RESET_BUTTON_TEXT = 'RESET' + +const getOwnerAddressesFrom = (owners: List) => { + if (!owners) { + return [] + } + + return owners.map((owner: Owner) => owner.get('address')) +} + +class AddOwner extends React.Component { + state = { + done: false, + } + + onAddOwner = async (values: Object) => { + try { + const { + safe, threshold, userAddress, fetchTransactions, // fetchOwners + } = this.props + const nonce = Date.now() + const newThreshold = values[INCREASE_PARAM] ? threshold + 1 : threshold + const newOwnerAddress = values[OWNER_ADDRESS_PARAM] + const newOwnerName = values[NAME_PARAM] + const safeAddress = safe.get('address') + const gnosisSafe = await getSafeEthereumInstance(safeAddress) + const data = gnosisSafe.contract.addOwnerWithThreshold.getData(newOwnerAddress, newThreshold) + await createTransaction(safe, `Add Owner ${newOwnerName}`, safeAddress, 0, nonce, userAddress, data) + await sleep(1500) + fetchTransactions() + // fetchOwners(safeAddress) + this.setState({ done: true }) + } catch (error) { + this.setState({ done: false }) + // eslint-disable-next-line + console.log('Error while adding owner ' + error) + } + } + + onReset = () => { + this.setState({ done: false }) + } + + render() { + const { safe } = this.props + const { done } = this.state + const steps = getSteps() + const finishedButton = + const addresses = getOwnerAddressesFrom(safe.get('owners')) + + return ( + + + + { AddOwnerForm } + + + { Review } + + + + ) + } +} + +export default connect(selector, actions)(AddOwner) diff --git a/src/routes/safe/component/AddOwner/selector.js b/src/routes/safe/component/AddOwner/selector.js new file mode 100644 index 00000000..9e7bfef1 --- /dev/null +++ b/src/routes/safe/component/AddOwner/selector.js @@ -0,0 +1,11 @@ +// @flow +import { createStructuredSelector } from 'reselect' +import { userAccountSelector } from '~/wallets/store/selectors/index' + +export type SelectorProps = { + userAddress: userAccountSelector, +} + +export default createStructuredSelector({ + userAddress: userAccountSelector, +}) diff --git a/src/routes/safe/component/Safe/Owners.jsx b/src/routes/safe/component/Safe/Owners.jsx index e4801fb0..4386d7ab 100644 --- a/src/routes/safe/component/Safe/Owners.jsx +++ b/src/routes/safe/component/Safe/Owners.jsx @@ -6,6 +6,7 @@ import Collapse from 'material-ui/transitions/Collapse' import ListItemText from '~/components/List/ListItemText' import List, { ListItem, ListItemIcon } from 'material-ui/List' import Avatar from 'material-ui/Avatar' +import Button from '~/components/layout/Button' import Group from 'material-ui-icons/Group' import Person from 'material-ui-icons/Person' import ExpandLess from 'material-ui-icons/ExpandLess' @@ -21,10 +22,13 @@ const styles = { type Props = Open & WithStyles & { owners: List, + onAddOwner: () => void, } +export const ADD_OWNER_BUTTON_TEXT = 'Add' + const Owners = openHoc(({ - open, toggle, owners, classes, + open, toggle, owners, classes, onAddOwner, }: Props) => ( @@ -35,6 +39,13 @@ const Owners = openHoc(({ {open ? : } + diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index 788b2561..648d315e 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -13,6 +13,7 @@ import Withdrawn from '~/routes/safe/component/Withdrawn' import Transactions from '~/routes/safe/component/Transactions' import AddTransaction from '~/routes/safe/component/AddTransaction' import Threshold from '~/routes/safe/component/Threshold' +import AddOwner from '~/routes/safe/component/AddOwner' import Address from './Address' import Balance from './Balance' @@ -66,6 +67,12 @@ class GnoSafe extends React.PureComponent { this.setState({ component: }) } + onAddOwner = () => { + const { safe } = this.props + + this.setState({ component: }) + } + render() { const { safe, balance } = this.props const { component } = this.state @@ -75,7 +82,7 @@ class GnoSafe extends React.PureComponent { - +
diff --git a/src/routes/safe/component/Transactions/index.jsx b/src/routes/safe/component/Transactions/index.jsx index e8a22e87..1518ed62 100644 --- a/src/routes/safe/component/Transactions/index.jsx +++ b/src/routes/safe/component/Transactions/index.jsx @@ -25,6 +25,7 @@ class Transactions extends React.Component { await sleep(1200) fetchTransactions() fetchThreshold(safeAddress) + // fetchOwners(safeAddress) } render() { From 260e7e128017e5621524777343d7737387c61b5d Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 15:29:06 +0200 Subject: [PATCH 16/25] WA-234 Adding Review process when changing threshold --- src/routes/safe/component/AddOwner/index.jsx | 2 +- src/routes/safe/component/Safe/index.jsx | 2 +- .../safe/component/Threshold/Review/index.jsx | 31 +++++ .../Threshold/ThresholdForm/index.jsx | 43 ++++++ src/routes/safe/component/Threshold/index.jsx | 125 +++++++----------- 5 files changed, 127 insertions(+), 76 deletions(-) create mode 100644 src/routes/safe/component/Threshold/Review/index.jsx create mode 100644 src/routes/safe/component/Threshold/ThresholdForm/index.jsx diff --git a/src/routes/safe/component/AddOwner/index.jsx b/src/routes/safe/component/AddOwner/index.jsx index 4869314f..61f95978 100644 --- a/src/routes/safe/component/AddOwner/index.jsx +++ b/src/routes/safe/component/AddOwner/index.jsx @@ -13,7 +13,7 @@ import selector, { type SelectorProps } from './selector' import actions, { type Actions } from './actions' const getSteps = () => [ - 'Fill Owner Form', 'Review Withdrawn', + 'Fill Owner Form', 'Review Add order operation', ] type Props = SelectorProps & Actions & { diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index 648d315e..06062d50 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -64,7 +64,7 @@ class GnoSafe extends React.PureComponent { onEditThreshold = () => { const { safe } = this.props - this.setState({ component: }) + this.setState({ component: }) } onAddOwner = () => { diff --git a/src/routes/safe/component/Threshold/Review/index.jsx b/src/routes/safe/component/Threshold/Review/index.jsx new file mode 100644 index 00000000..0483e876 --- /dev/null +++ b/src/routes/safe/component/Threshold/Review/index.jsx @@ -0,0 +1,31 @@ +// @flow +import * as React from 'react' +import { CircularProgress } from 'material-ui/Progress' +import Block from '~/components/layout/Block' +import Bold from '~/components/layout/Bold' +import Heading from '~/components/layout/Heading' +import Paragraph from '~/components/layout/Paragraph' +import { THRESHOLD_PARAM } from '~/routes/safe/component/Threshold/ThresholdForm' + +type FormProps = { + values: Object, + submitting: boolean, +} + +const spinnerStyle = { + minHeight: '50px', +} + +const Review = () => ({ values, submitting }: FormProps) => ( + + Review the Threshold operation + + The new threshold will be: {values[THRESHOLD_PARAM]} + + + { submitting && } + + +) + +export default Review diff --git a/src/routes/safe/component/Threshold/ThresholdForm/index.jsx b/src/routes/safe/component/Threshold/ThresholdForm/index.jsx new file mode 100644 index 00000000..b155238a --- /dev/null +++ b/src/routes/safe/component/Threshold/ThresholdForm/index.jsx @@ -0,0 +1,43 @@ +// @flow +import * as React from 'react' +import Block from '~/components/layout/Block' +import Heading from '~/components/layout/Heading' +import Field from '~/components/forms/Field' +import TextField from '~/components/forms/TextField' +import { composeValidators, minValue, maxValue, mustBeInteger, required } from '~/components/forms/validator' +import { type Safe } from '~/routes/safe/store/model/safe' + +export const THRESHOLD_PARAM = 'threshold' + +type ThresholdProps = { + numOwners: number, + safe: Safe, +} + +const ThresholdForm = ({ numOwners, safe }: ThresholdProps) => () => ( + + + {'Change safe\'s threshold'} + + + {`Safe's owners: ${numOwners} and Safe's threshold: ${safe.get('confirmations')}`} + + + + + +) + +export default ThresholdForm diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx index f7fffec8..5d144efe 100644 --- a/src/routes/safe/component/Threshold/index.jsx +++ b/src/routes/safe/component/Threshold/index.jsx @@ -1,105 +1,82 @@ // @flow import * as React from 'react' -import Block from '~/components/layout/Block' -import Heading from '~/components/layout/Heading' -import Field from '~/components/forms/Field' -import TextField from '~/components/forms/TextField' -import GnoForm from '~/components/forms/GnoForm' +import Stepper from '~/components/Stepper' import { connect } from 'react-redux' -import Button from '~/components/layout/Button' -import Col from '~/components/layout/Col' -import Row from '~/components/layout/Row' -import { composeValidators, minValue, maxValue, mustBeInteger, required } from '~/components/forms/validator' import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' import { sleep } from '~/utils/timer' import { type Safe } from '~/routes/safe/store/model/safe' +import ThresholdForm, { THRESHOLD_PARAM } from './ThresholdForm' import selector, { type SelectorProps } from './selector' import actions, { type Actions } from './actions' +import Review from './Review' -type ThresholdProps = { +type Props = SelectorProps & Actions & { numOwners: number, safe: Safe, -} - -type Props = SelectorProps & Actions & ThresholdProps & { onReset: () => void, } -const THRESHOLD_PARAM = 'threshold' - -const ThresholdComponent = ({ numOwners, safe }: ThresholdProps) => () => ( - - - {'Change safe\'s threshold'} - - - {`Safe's owners: ${numOwners} and Safe's threshold: ${safe.get('confirmations')}`} - - - - - -) +const getSteps = () => [ + 'Fill Change threshold Form', 'Review change threshold operation', +] type State = { - initialValues: Object, + done: boolean, } +export const CHANGE_THRESHOLD_RESET_BUTTON_TEXT = 'RESET' + class Threshold extends React.PureComponent { state = { - initialValues: {}, + done: false, } onThreshold = async (values: Object) => { - const { safe, userAddress } = this.props // , fetchThreshold } = this.props - const newThreshold = values[THRESHOLD_PARAM] - const gnosisSafe = await getSafeEthereumInstance(safe.get('address')) - const nonce = Date.now() - const data = gnosisSafe.contract.changeThreshold.getData(newThreshold) - await createTransaction(safe, `Change Safe's threshold [${nonce}]`, safe.get('address'), 0, nonce, userAddress, data) - await sleep(1500) - this.props.fetchTransactions() - this.props.fetchThreshold(safe.get('address')) + try { + const { safe, userAddress } = this.props // , fetchThreshold } = this.props + const newThreshold = values[THRESHOLD_PARAM] + const gnosisSafe = await getSafeEthereumInstance(safe.get('address')) + const nonce = Date.now() + const data = gnosisSafe.contract.changeThreshold.getData(newThreshold) + await createTransaction(safe, `Change Safe's threshold [${nonce}]`, safe.get('address'), 0, nonce, userAddress, data) + await sleep(1500) + await this.props.fetchTransactions() + await this.props.fetchThreshold(safe.get('address')) + this.setState({ done: true }) + } catch (error) { + this.setState({ done: false }) + // eslint-disable-next-line + console.log('Error while changing threshold ' + error) + } + } + + onReset = () => { + this.setState({ done: false }) } render() { - const { numOwners, onReset, safe } = this.props + const { numOwners, safe } = this.props + const { done } = this.state + const steps = getSteps() + const finishedButton = return ( - - {(submitting: boolean, submitSucceeded: boolean) => ( - - - - - - )} - + + + + { ThresholdForm } + + + { Review } + + + ) } } From 6cf958c1b51ae6b6246f6a4d0be76c36b564d1e5 Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 21:45:23 +0200 Subject: [PATCH 17/25] WA-234 Changing confirmations to threshold --- src/routes/safe/component/AddOwner/index.jsx | 2 +- .../AddTransaction/createTransactions.js | 6 ++--- .../AddTransaction/test/transactions.test.js | 22 +++++++++---------- src/routes/safe/component/Safe/index.jsx | 4 ++-- .../Threshold/ThresholdForm/index.jsx | 2 +- src/routes/safe/store/actions/addSafe.js | 4 ++-- src/routes/safe/store/model/safe.js | 4 ++-- src/routes/safe/store/reducer/safe.js | 2 +- .../safe/store/test/builder/safe.builder.js | 2 +- src/routes/safeList/components/SafeTable.jsx | 2 +- 10 files changed, 25 insertions(+), 25 deletions(-) diff --git a/src/routes/safe/component/AddOwner/index.jsx b/src/routes/safe/component/AddOwner/index.jsx index 61f95978..245947d1 100644 --- a/src/routes/safe/component/AddOwner/index.jsx +++ b/src/routes/safe/component/AddOwner/index.jsx @@ -84,7 +84,7 @@ class AddOwner extends React.Component { steps={steps} onReset={this.onReset} > - + { AddOwnerForm } diff --git a/src/routes/safe/component/AddTransaction/createTransactions.js b/src/routes/safe/component/AddTransaction/createTransactions.js index 67b26891..f7c25503 100644 --- a/src/routes/safe/component/AddTransaction/createTransactions.js +++ b/src/routes/safe/component/AddTransaction/createTransactions.js @@ -101,7 +101,7 @@ export const createTransaction = async ( const valueInWei = web3.toWei(txValue, 'ether') const CALL = 0 - const thresholdIsOne = safe.get('confirmations') === 1 + const thresholdIsOne = safe.get('threshold') === 1 if (hasOneOwner(safe) || thresholdIsOne) { const txConfirmationData = gnosisSafe.contract.execTransactionIfApproved.getData(txDest, valueInWei, data, CALL, nonce) @@ -109,7 +109,7 @@ export const createTransaction = async ( checkReceiptStatus(txHash) const executedConfirmations: List = buildExecutedConfirmationFrom(safe.get('owners'), user) - return storeTransaction(txName, nonce, txDest, txValue, user, executedConfirmations, txHash, safeAddress, safe.get('confirmations'), data) + return storeTransaction(txName, nonce, txDest, txValue, user, executedConfirmations, txHash, safeAddress, safe.get('threshold'), data) } const txConfirmationData = @@ -119,5 +119,5 @@ export const createTransaction = async ( const confirmations: List = buildConfirmationsFrom(safe.get('owners'), user, txConfirmationHash) - return storeTransaction(txName, nonce, txDest, txValue, user, confirmations, '', safeAddress, safe.get('confirmations'), data) + return storeTransaction(txName, nonce, txDest, txValue, user, confirmations, '', safeAddress, safe.get('threshold'), data) } diff --git a/src/routes/safe/component/AddTransaction/test/transactions.test.js b/src/routes/safe/component/AddTransaction/test/transactions.test.js index 19c1fdeb..39270c0f 100644 --- a/src/routes/safe/component/AddTransaction/test/transactions.test.js +++ b/src/routes/safe/component/AddTransaction/test/transactions.test.js @@ -33,7 +33,7 @@ describe('Transactions Suite', () => { const txName = 'Buy butteries for project' const nonce: number = 10 const confirmations: List = buildConfirmationsFrom(owners, 'foo', 'confirmationHash') - storeTransaction(txName, nonce, destination, value, 'foo', confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') + storeTransaction(txName, nonce, destination, value, 'foo', confirmations, '', safe.get('address'), safe.get('threshold'), '0x') // WHEN const transactions: Map> = loadSafeTransactions() @@ -55,12 +55,12 @@ describe('Transactions Suite', () => { const safeAddress = safe.get('address') const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(firstTxName, firstNonce, destination, value, creator, confirmations, '', safeAddress, safe.get('confirmations'), '0x') + storeTransaction(firstTxName, firstNonce, destination, value, creator, confirmations, '', safeAddress, safe.get('threshold'), '0x') const secondTxName = 'Buy printers for project' const secondNonce: number = firstNonce + 100 const secondConfirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(secondTxName, secondNonce, destination, value, creator, secondConfirmations, '', safeAddress, safe.get('confirmations'), '0x') + storeTransaction(secondTxName, secondNonce, destination, value, creator, secondConfirmations, '', safeAddress, safe.get('threshold'), '0x') // WHEN const transactions: Map> = loadSafeTransactions() @@ -82,7 +82,7 @@ describe('Transactions Suite', () => { const safeAddress = safe.address const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safeAddress, safe.get('confirmations'), '0x') + storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safeAddress, safe.get('threshold'), '0x') const secondSafe = SafeFactory.dailyLimitSafe(10, 2) const txSecondName = 'Buy batteris for Beta project' @@ -92,7 +92,7 @@ describe('Transactions Suite', () => { const secondConfirmations: List = buildConfirmationsFrom(secondSafe.get('owners'), secondCreator, 'confirmationHash') storeTransaction( txSecondName, txSecondNonce, destination, value, secondCreator, - secondConfirmations, '', secondSafeAddress, secondSafe.get('confirmations'), '0x', + secondConfirmations, '', secondSafeAddress, secondSafe.get('threshold'), '0x', ) let transactions: Map> = loadSafeTransactions() @@ -112,7 +112,7 @@ describe('Transactions Suite', () => { const txConfirmations: List = buildConfirmationsFrom(owners, creator, 'secondConfirmationHash') storeTransaction( txFirstName, txFirstNonce, destination, value, creator, - txConfirmations, '', safe.get('address'), safe.get('confirmations'), '0x', + txConfirmations, '', safe.get('address'), safe.get('threshold'), '0x', ) transactions = loadSafeTransactions() @@ -148,10 +148,10 @@ describe('Transactions Suite', () => { const nonce: number = 10 const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') + storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('threshold'), '0x') // WHEN - const createTxFnc = () => storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') + const createTxFnc = () => storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('threshold'), '0x') expect(createTxFnc).toThrow(/Transaction with same nonce/) }) @@ -161,7 +161,7 @@ describe('Transactions Suite', () => { const nonce: number = 10 const creator = 'foo' const confirmations: List = buildConfirmationsFrom(owners, creator, 'confirmationHash') - storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('confirmations'), '0x') + storeTransaction(txName, nonce, destination, value, creator, confirmations, '', safe.get('address'), safe.get('threshold'), '0x') // WHEN const transactions: Map> = loadSafeTransactions() @@ -185,7 +185,7 @@ describe('Transactions Suite', () => { const nonce: number = 10 const tx = '' const confirmations: List = buildExecutedConfirmationFrom(oneOwnerSafe.get('owners'), ownerName) - const createTxFnc = () => storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('confirmations'), '0x') + const createTxFnc = () => storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('threshold'), '0x') expect(createTxFnc).toThrow(/The tx should be mined before storing it in safes with one owner/) }) @@ -197,7 +197,7 @@ describe('Transactions Suite', () => { const nonce: number = 10 const tx = 'validTxHash' const confirmations: List = buildExecutedConfirmationFrom(oneOwnerSafe.get('owners'), ownerName) - storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('confirmations'), '0x') + storeTransaction(txName, nonce, destination, value, ownerName, confirmations, tx, oneOwnerSafe.get('address'), oneOwnerSafe.get('threshold'), '0x') // WHEN const safeTransactions: Map> = loadSafeTransactions() diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index 06062d50..4a4554aa 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -70,7 +70,7 @@ class GnoSafe extends React.PureComponent { onAddOwner = () => { const { safe } = this.props - this.setState({ component: }) + this.setState({ component: }) } render() { @@ -83,7 +83,7 @@ class GnoSafe extends React.PureComponent { - +
diff --git a/src/routes/safe/component/Threshold/ThresholdForm/index.jsx b/src/routes/safe/component/Threshold/ThresholdForm/index.jsx index b155238a..66a1e89d 100644 --- a/src/routes/safe/component/Threshold/ThresholdForm/index.jsx +++ b/src/routes/safe/component/Threshold/ThresholdForm/index.jsx @@ -20,7 +20,7 @@ const ThresholdForm = ({ numOwners, safe }: ThresholdProps) => () => ( {'Change safe\'s threshold'} - {`Safe's owners: ${numOwners} and Safe's threshold: ${safe.get('confirmations')}`} + {`Safe's owners: ${numOwners} and Safe's threshold: ${safe.get('threshold')}`} { const owners: List = buildOwnersFrom(ownersName, ownersAddress) const dailyLimit: DailyLimit = buildDailyLimitFrom(limit) return ({ - address, name, confirmations, owners, dailyLimit, + address, name, threshold, owners, dailyLimit, }) }, ) diff --git a/src/routes/safe/store/model/safe.js b/src/routes/safe/store/model/safe.js index 22e10581..94c068b7 100644 --- a/src/routes/safe/store/model/safe.js +++ b/src/routes/safe/store/model/safe.js @@ -7,7 +7,7 @@ import type { Owner } from '~/routes/safe/store/model/owner' export type SafeProps = { name: string, address: string, - confirmations: number, + threshold: number, owners: List, dailyLimit: DailyLimit, } @@ -15,7 +15,7 @@ export type SafeProps = { export const makeSafe: RecordFactory = Record({ name: '', address: '', - confirmations: 0, + threshold: 0, owners: List([]), dailyLimit: makeDailyLimit(), }) diff --git a/src/routes/safe/store/reducer/safe.js b/src/routes/safe/store/reducer/safe.js index f8f64485..785fc17d 100644 --- a/src/routes/safe/store/reducer/safe.js +++ b/src/routes/safe/store/reducer/safe.js @@ -52,5 +52,5 @@ export default handleActions({ [UPDATE_DAILY_LIMIT]: (state: State, action: ActionType): State => state.updateIn([action.payload.safeAddress, 'dailyLimit'], () => makeDailyLimit(action.payload.dailyLimit)), [UPDATE_THRESHOLD]: (state: State, action: ActionType): State => - state.updateIn([action.payload.safeAddress, 'confirmations'], () => action.payload.threshold), + state.updateIn([action.payload.safeAddress, 'threshold'], () => action.payload.threshold), }, Map()) diff --git a/src/routes/safe/store/test/builder/safe.builder.js b/src/routes/safe/store/test/builder/safe.builder.js index a1dcc2a7..55c442cd 100644 --- a/src/routes/safe/store/test/builder/safe.builder.js +++ b/src/routes/safe/store/test/builder/safe.builder.js @@ -20,7 +20,7 @@ class SafeBuilder { } withConfirmations(confirmations: number) { - this.safe = this.safe.set('confirmations', confirmations) + this.safe = this.safe.set('threshold', confirmations) return this } diff --git a/src/routes/safeList/components/SafeTable.jsx b/src/routes/safeList/components/SafeTable.jsx index 9706f66c..c13a0eeb 100644 --- a/src/routes/safeList/components/SafeTable.jsx +++ b/src/routes/safeList/components/SafeTable.jsx @@ -32,7 +32,7 @@ const SafeTable = ({ safes }: Props) => ( {safe.get('name')} {safe.get('address')} - {safe.get('confirmations')} + {safe.get('threshold')} {safe.get('owners').count()} {`${safe.get('dailyLimit').get('value')} ETH`} From 3428329398f87a3eec1a70054b8b6bfcc054b383 Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 21:47:36 +0200 Subject: [PATCH 18/25] WA-234 Refactor get safe contract instance --- src/routes/safe/test/Safe.owners.test.js | 10 +--------- src/wallets/safeContracts.js | 8 ++++++++ 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/src/routes/safe/test/Safe.owners.test.js b/src/routes/safe/test/Safe.owners.test.js index 73bae217..59a464c7 100644 --- a/src/routes/safe/test/Safe.owners.test.js +++ b/src/routes/safe/test/Safe.owners.test.js @@ -10,11 +10,11 @@ import { confirmationsTransactionSelector, safeSelector, safeTransactionsSelecto import { getTransactionFromReduxStore } from '~/routes/safe/test/testMultisig' import { buildMathPropsFrom } from '~/test/buildReactRouterProps' import { createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' -import { getGnosisSafeContract } from '~/wallets/safeContracts' import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' import { type GlobalState } from '~/store/index' import { type Safe } from '~/routes/safe/store/model/safe' import { type Transaction } from '~/routes/safe/store/model/transaction' +import { getGnosisSafeInstanceAt } from '~/wallets/safeContracts' const getSafeFrom = (state: GlobalState, safeAddress: string): Safe => { const match: Match = buildMathPropsFrom(safeAddress) @@ -24,14 +24,6 @@ const getSafeFrom = (state: GlobalState, safeAddress: string): Safe => { return safe } -const getGnosisSafeInstanceAt = async (safeAddress: string) => { - const web3 = getWeb3() - const GnosisSafe = await getGnosisSafeContract(web3) - const gnosisSafe = GnosisSafe.at(safeAddress) - - return gnosisSafe -} - describe('React DOM TESTS > Add and remove owners', () => { const assureExecuted = (transaction: Transaction) => { expect(transaction.get('tx')).not.toBe(null) diff --git a/src/wallets/safeContracts.js b/src/wallets/safeContracts.js index 81399db8..60a589ce 100644 --- a/src/wallets/safeContracts.js +++ b/src/wallets/safeContracts.js @@ -136,3 +136,11 @@ export const deploySafeContract = async ( return proxyFactoryMaster.createProxy(safeMaster.address, gnosisSafeData, { from: userAccount, gas, gasPrice }) } + +export const getGnosisSafeInstanceAt = async (safeAddress: string) => { + const web3 = getWeb3() + const GnosisSafe = await getGnosisSafeContract(web3) + const gnosisSafe = GnosisSafe.at(safeAddress) + + return gnosisSafe +} From aec0b5020042d903d7764bd3f6a5d6d018b84daf Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 22:10:10 +0200 Subject: [PATCH 19/25] WA-234 Loading safes info from Ethereum Node --- src/index.js | 4 ++ src/routes/safe/store/actions/fetchSafes.js | 55 ++++++++++++++++++++ src/routes/safe/store/actions/updateSafes.js | 8 +++ src/routes/safe/store/reducer/safe.js | 27 ++-------- src/routes/safe/store/test/safe.reducer.js | 22 +------- src/store/index.js | 3 +- 6 files changed, 74 insertions(+), 45 deletions(-) create mode 100644 src/routes/safe/store/actions/fetchSafes.js create mode 100644 src/routes/safe/store/actions/updateSafes.js diff --git a/src/index.js b/src/index.js index ec531904..e57c20bb 100644 --- a/src/index.js +++ b/src/index.js @@ -10,8 +10,12 @@ import PageFrame from '~/components/layout/PageFrame' import { history, store } from '~/store' import theme from '~/theme/mui' import AppRoutes from '~/routes' +import fetchSafes from '~/routes/safe/store/actions/fetchSafes' + import './index.scss' +store.dispatch(fetchSafes()) + const Root = () => ( diff --git a/src/routes/safe/store/actions/fetchSafes.js b/src/routes/safe/store/actions/fetchSafes.js new file mode 100644 index 00000000..b982ba92 --- /dev/null +++ b/src/routes/safe/store/actions/fetchSafes.js @@ -0,0 +1,55 @@ +// @flow +import type { Dispatch as ReduxDispatch } from 'redux' +import { List, Map } from 'immutable' +import { type GlobalState } from '~/store/index' +import { makeOwner } from '~/routes/safe/store/model/owner' +import { type SafeProps, type Safe, makeSafe } from '~/routes/safe/store/model/safe' +import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' +import { getDailyLimitFrom } from '~/routes/safe/component/Withdrawn/withdrawn' +import { getGnosisSafeInstanceAt } from '~/wallets/safeContracts' +import { load, SAFES_KEY } from '~/utils/localStorage' +import updateSafes from '~/routes/safe/store/actions/updateSafes' + +const buildOwnersFrom = (safeOwners: string[], storedOwners: Object[]) => ( + safeOwners.map((ownerAddress: string) => { + const foundOwner = storedOwners.find(owner => owner.address === ownerAddress) + return makeOwner(foundOwner || { name: 'UNKNOWN', address: ownerAddress }) + }) +) + +const buildSafe = async (storedSafe: Object) => { + const safeAddress = storedSafe.address + const gnosisSafe = await getGnosisSafeInstanceAt(safeAddress) + + const dailyLimit = makeDailyLimit(await getDailyLimitFrom(safeAddress, 0)) + const threshold = Number(await gnosisSafe.getThreshold()) + const owners = List(buildOwnersFrom(await gnosisSafe.getOwners(), storedSafe.owners)) + + const safe: SafeProps = { + address: safeAddress, + dailyLimit, + name: storedSafe.name, + threshold, + owners, + } + + return makeSafe(safe) +} + +const buildSafesFrom = async (loadedSafes: Object): Promise> => { + const safes = Map() + + const keys = Object.keys(loadedSafes) + const safeRecords = await Promise.all(keys.map((address: string) => buildSafe(loadedSafes[address]))) + + return safes.withMutations(async (map) => { + safeRecords.forEach((safe: Safe) => map.set(safe.get('address'), safe)) + }) +} + +export default () => async (dispatch: ReduxDispatch) => { + const storedSafes = load(SAFES_KEY) + const safes = storedSafes ? await buildSafesFrom(storedSafes) : Map() + + return dispatch(updateSafes(safes)) +} diff --git a/src/routes/safe/store/actions/updateSafes.js b/src/routes/safe/store/actions/updateSafes.js new file mode 100644 index 00000000..bc0daf24 --- /dev/null +++ b/src/routes/safe/store/actions/updateSafes.js @@ -0,0 +1,8 @@ +// @flow +import { createAction } from 'redux-actions' + +export const UPDATE_SAFES = 'UPDATE_SAFES' + +const updateSafesInBatch = createAction(UPDATE_SAFES) + +export default updateSafesInBatch diff --git a/src/routes/safe/store/reducer/safe.js b/src/routes/safe/store/reducer/safe.js index 785fc17d..d2a3dee0 100644 --- a/src/routes/safe/store/reducer/safe.js +++ b/src/routes/safe/store/reducer/safe.js @@ -1,37 +1,18 @@ // @flow -import { Map, List } from 'immutable' +import { Map } from 'immutable' import { handleActions, type ActionType } from 'redux-actions' import addSafe, { ADD_SAFE } from '~/routes/safe/store/actions/addSafe' import updateDailyLimit, { UPDATE_DAILY_LIMIT } from '~/routes/safe/store/actions/updateDailyLimit' -import { makeOwner } from '~/routes/safe/store/model/owner' import { type Safe, makeSafe } from '~/routes/safe/store/model/safe' -import { load, saveSafes, SAFES_KEY } from '~/utils/localStorage' +import { saveSafes } from '~/utils/localStorage' import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' import updateThreshold, { UPDATE_THRESHOLD } from '~/routes/safe/store/actions/updateThreshold' +import updateSafes, { UPDATE_SAFES } from '~/routes/safe/store/actions/updateSafes' export const SAFE_REDUCER_ID = 'safes' export type State = Map -const buildSafesFrom = (loadedSafes: Object): State => { - const safes: State = Map() - - return safes.withMutations((map: State) => { - Object.keys(loadedSafes).forEach((address) => { - const safe = loadedSafes[address] - safe.owners = List(safe.owners.map((owner => makeOwner(owner)))) - safe.dailyLimit = makeDailyLimit({ value: safe.dailyLimit.value, spentToday: safe.dailyLimit.spentToday }) - return map.set(address, makeSafe(safe)) - }) - }) -} - -export const safeInitialState = (): State => { - const storedSafes = load(SAFES_KEY) - - return storedSafes ? buildSafesFrom(storedSafes) : Map() -} - /* type Action = { key: string, @@ -44,6 +25,8 @@ action: AddSafeType */ export default handleActions({ + [UPDATE_SAFES]: (state: State, action: ActionType): State => + action.payload, [ADD_SAFE]: (state: State, action: ActionType): State => { const safes = state.set(action.payload.address, makeSafe(action.payload)) saveSafes(safes.toJSON()) diff --git a/src/routes/safe/store/test/safe.reducer.js b/src/routes/safe/store/test/safe.reducer.js index 57edaed7..e3e8af81 100644 --- a/src/routes/safe/store/test/safe.reducer.js +++ b/src/routes/safe/store/test/safe.reducer.js @@ -1,7 +1,7 @@ // @flow import { combineReducers, createStore, applyMiddleware, compose } from 'redux' import thunk from 'redux-thunk' -import safeReducer, { safeInitialState, SAFE_REDUCER_ID } from '~/routes/safe/store/reducer/safe' +import safeReducer, { SAFE_REDUCER_ID } from '~/routes/safe/store/reducer/safe' import addSafe from '~/routes/safe/store/actions/addSafe' import * as SafeFields from '~/routes/open/components/fields' import { getAccountsFrom, getNamesFrom } from '~/routes/open/utils/safeDataExtractor' @@ -56,26 +56,6 @@ const providerReducerTests = () => { // THEN expect(safes.get(address)).toEqual(SafeFactory.oneOwnerSafe()) }) - - it('reducer loads information from localStorage', async () => { - // GIVEN in beforeEach method - - // WHEN - store.dispatch(addSafe( - formValues[SafeFields.FIELD_NAME], - formValues.address, - formValues[SafeFields.FIELD_CONFIRMATIONS], - formValues[SafeFields.FIELD_DAILY_LIMIT], - getNamesFrom(formValues), - getAccountsFrom(formValues), - )) - - const anotherStore = aStore({ [SAFE_REDUCER_ID]: safeInitialState() }) - const safes = anotherStore.getState()[SAFE_REDUCER_ID] - - // THEN - expect(safeInitialState()).toEqual(safes) - }) }) } diff --git a/src/store/index.js b/src/store/index.js index 61528341..c968b44d 100644 --- a/src/store/index.js +++ b/src/store/index.js @@ -4,7 +4,7 @@ import { routerMiddleware, routerReducer } from 'react-router-redux' import { combineReducers, createStore, applyMiddleware, compose, type Reducer, type Store } from 'redux' import thunk from 'redux-thunk' import provider, { PROVIDER_REDUCER_ID, type State as ProviderState } from '~/wallets/store/reducer/provider' -import safe, { SAFE_REDUCER_ID, safeInitialState, type State as SafeState } from '~/routes/safe/store/reducer/safe' +import safe, { SAFE_REDUCER_ID, type State as SafeState } from '~/routes/safe/store/reducer/safe' import balances, { BALANCE_REDUCER_ID, type State as BalancesState } from '~/routes/safe/store/reducer/balances' import transactions, { type State as TransactionsState, transactionsInitialState, TRANSACTIONS_REDUCER_ID } from '~/routes/safe/store/reducer/transactions' @@ -33,7 +33,6 @@ const reducers: Reducer = combineReducers({ }) const initialState = { - [SAFE_REDUCER_ID]: safeInitialState(), [TRANSACTIONS_REDUCER_ID]: transactionsInitialState(), } From 2da7eab781abbef02be2fb7fdd209aca0f773de8 Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 22:25:32 +0200 Subject: [PATCH 20/25] WA-234 fetch safe info regularly in safe view --- src/routes/safe/container/actions.js | 6 +-- src/routes/safe/container/index.jsx | 9 +---- src/routes/safe/store/actions/fetchSafe.js | 42 +++++++++++++++++++++ src/routes/safe/store/actions/fetchSafes.js | 35 ++--------------- src/routes/safe/store/actions/updateSafe.js | 8 ++++ src/routes/safe/store/reducer/safe.js | 3 ++ 6 files changed, 61 insertions(+), 42 deletions(-) create mode 100644 src/routes/safe/store/actions/fetchSafe.js create mode 100644 src/routes/safe/store/actions/updateSafe.js diff --git a/src/routes/safe/container/actions.js b/src/routes/safe/container/actions.js index 4c0e607f..6ffe7123 100644 --- a/src/routes/safe/container/actions.js +++ b/src/routes/safe/container/actions.js @@ -1,13 +1,13 @@ // @flow +import fetchSafe from '~/routes/safe/store/actions/fetchSafe' import fetchBalance from '~/routes/safe/store/actions/fetchBalance' -import fetchDailyLimit from '~/routes/safe/store/actions/fetchDailyLimit' export type Actions = { + fetchSafe: typeof fetchSafe, fetchBalance: typeof fetchBalance, - fetchDailyLimit: typeof fetchDailyLimit, } export default { + fetchSafe, fetchBalance, - fetchDailyLimit, } diff --git a/src/routes/safe/container/index.jsx b/src/routes/safe/container/index.jsx index ea291ffc..d6579ead 100644 --- a/src/routes/safe/container/index.jsx +++ b/src/routes/safe/container/index.jsx @@ -14,17 +14,12 @@ type Props = Actions & SelectorProps & { class SafeView extends React.PureComponent { componentDidMount() { this.intervalId = setInterval(() => { - const { safe, fetchBalance } = this.props + const { safe, fetchSafe, fetchBalance } = this.props if (!safe) { return } - const safeAddress: string = safe.get('address') fetchBalance(safeAddress) + fetchSafe(safe) }, 1500) - - const { fetchDailyLimit, safe } = this.props - if (safe) { - fetchDailyLimit(safe.get('address')) - } } componentWillUnmount() { diff --git a/src/routes/safe/store/actions/fetchSafe.js b/src/routes/safe/store/actions/fetchSafe.js new file mode 100644 index 00000000..60b0cacf --- /dev/null +++ b/src/routes/safe/store/actions/fetchSafe.js @@ -0,0 +1,42 @@ +// @flow +import type { Dispatch as ReduxDispatch } from 'redux' +import { List } from 'immutable' +import { type GlobalState } from '~/store/index' +import { makeOwner } from '~/routes/safe/store/model/owner' +import { type SafeProps, type Safe, makeSafe } from '~/routes/safe/store/model/safe' +import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' +import { getDailyLimitFrom } from '~/routes/safe/component/Withdrawn/withdrawn' +import { getGnosisSafeInstanceAt } from '~/wallets/safeContracts' +import updateSafe from '~/routes/safe/store/actions/updateSafe' + +const buildOwnersFrom = (safeOwners: string[], storedOwners: Object[]) => ( + safeOwners.map((ownerAddress: string) => { + const foundOwner = storedOwners.find(owner => owner.address === ownerAddress) + return makeOwner(foundOwner || { name: 'UNKNOWN', address: ownerAddress }) + }) +) + +export const buildSafe = async (storedSafe: Object) => { + const safeAddress = storedSafe.address + const gnosisSafe = await getGnosisSafeInstanceAt(safeAddress) + + const dailyLimit = makeDailyLimit(await getDailyLimitFrom(safeAddress, 0)) + const threshold = Number(await gnosisSafe.getThreshold()) + const owners = List(buildOwnersFrom(await gnosisSafe.getOwners(), storedSafe.owners)) + + const safe: SafeProps = { + address: safeAddress, + dailyLimit, + name: storedSafe.name, + threshold, + owners, + } + + return makeSafe(safe) +} + +export default (safe: Safe) => async (dispatch: ReduxDispatch) => { + const safeRecord = await buildSafe(safe.toJSON()) + + return dispatch(updateSafe(safeRecord)) +} diff --git a/src/routes/safe/store/actions/fetchSafes.js b/src/routes/safe/store/actions/fetchSafes.js index b982ba92..196c1b40 100644 --- a/src/routes/safe/store/actions/fetchSafes.js +++ b/src/routes/safe/store/actions/fetchSafes.js @@ -1,40 +1,11 @@ // @flow import type { Dispatch as ReduxDispatch } from 'redux' -import { List, Map } from 'immutable' +import { Map } from 'immutable' import { type GlobalState } from '~/store/index' -import { makeOwner } from '~/routes/safe/store/model/owner' -import { type SafeProps, type Safe, makeSafe } from '~/routes/safe/store/model/safe' -import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' -import { getDailyLimitFrom } from '~/routes/safe/component/Withdrawn/withdrawn' -import { getGnosisSafeInstanceAt } from '~/wallets/safeContracts' import { load, SAFES_KEY } from '~/utils/localStorage' import updateSafes from '~/routes/safe/store/actions/updateSafes' - -const buildOwnersFrom = (safeOwners: string[], storedOwners: Object[]) => ( - safeOwners.map((ownerAddress: string) => { - const foundOwner = storedOwners.find(owner => owner.address === ownerAddress) - return makeOwner(foundOwner || { name: 'UNKNOWN', address: ownerAddress }) - }) -) - -const buildSafe = async (storedSafe: Object) => { - const safeAddress = storedSafe.address - const gnosisSafe = await getGnosisSafeInstanceAt(safeAddress) - - const dailyLimit = makeDailyLimit(await getDailyLimitFrom(safeAddress, 0)) - const threshold = Number(await gnosisSafe.getThreshold()) - const owners = List(buildOwnersFrom(await gnosisSafe.getOwners(), storedSafe.owners)) - - const safe: SafeProps = { - address: safeAddress, - dailyLimit, - name: storedSafe.name, - threshold, - owners, - } - - return makeSafe(safe) -} +import { buildSafe } from '~/routes/safe/store/actions/fetchSafe' +import { type Safe } from '~/routes/safe/store/model/safe' const buildSafesFrom = async (loadedSafes: Object): Promise> => { const safes = Map() diff --git a/src/routes/safe/store/actions/updateSafe.js b/src/routes/safe/store/actions/updateSafe.js new file mode 100644 index 00000000..89e30306 --- /dev/null +++ b/src/routes/safe/store/actions/updateSafe.js @@ -0,0 +1,8 @@ +// @flow +import { createAction } from 'redux-actions' + +export const UPDATE_SAFE = 'UPDATE_SAFE' + +const updateSafe = createAction(UPDATE_SAFE) + +export default updateSafe diff --git a/src/routes/safe/store/reducer/safe.js b/src/routes/safe/store/reducer/safe.js index d2a3dee0..bd585812 100644 --- a/src/routes/safe/store/reducer/safe.js +++ b/src/routes/safe/store/reducer/safe.js @@ -8,6 +8,7 @@ import { saveSafes } from '~/utils/localStorage' import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' import updateThreshold, { UPDATE_THRESHOLD } from '~/routes/safe/store/actions/updateThreshold' import updateSafes, { UPDATE_SAFES } from '~/routes/safe/store/actions/updateSafes' +import updateSafe, { UPDATE_SAFE } from '~/routes/safe/store/actions/updateSafe' export const SAFE_REDUCER_ID = 'safes' @@ -25,6 +26,8 @@ action: AddSafeType */ export default handleActions({ + [UPDATE_SAFE]: (state: State, action: ActionType): State => + state.set(action.payload.get('address'), action.payload), [UPDATE_SAFES]: (state: State, action: ActionType): State => action.payload, [ADD_SAFE]: (state: State, action: ActionType): State => { From f098d60e2803e3a6009212b3504b5b1ac85f7d7f Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 22:41:18 +0200 Subject: [PATCH 21/25] WA-234 Stop propagating event on add owner --- src/routes/safe/component/Safe/index.jsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/routes/safe/component/Safe/index.jsx b/src/routes/safe/component/Safe/index.jsx index 4a4554aa..23588ec9 100644 --- a/src/routes/safe/component/Safe/index.jsx +++ b/src/routes/safe/component/Safe/index.jsx @@ -67,9 +67,9 @@ class GnoSafe extends React.PureComponent { this.setState({ component: }) } - onAddOwner = () => { + onAddOwner = (e: SyntheticEvent) => { const { safe } = this.props - + e.stopPropagation() this.setState({ component: }) } From 860fb300eafc07248f05094f4afed2df2923b45a Mon Sep 17 00:00:00 2001 From: apanizo Date: Thu, 7 Jun 2018 22:43:52 +0200 Subject: [PATCH 22/25] WA-234 Provisional metamask safety check --- src/wallets/getWeb3.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/wallets/getWeb3.js b/src/wallets/getWeb3.js index 5e9602c5..8b5e43d3 100644 --- a/src/wallets/getWeb3.js +++ b/src/wallets/getWeb3.js @@ -5,7 +5,7 @@ import type { ProviderProps } from '~/wallets/store/model/provider' import { promisify } from '~/utils/promisify' let web3 -export const getWeb3 = () => web3 +export const getWeb3 = () => web3 || new Web3(window.web3.currentProvider) const isMetamask: Function = (web3Provider): boolean => { const isMetamaskConstructor = web3Provider.currentProvider.constructor.name === 'MetamaskInpageProvider' From 801dde497ef9df0b8e3567234ff6685330402c1c Mon Sep 17 00:00:00 2001 From: apanizo Date: Fri, 8 Jun 2018 09:15:57 +0200 Subject: [PATCH 23/25] WA-234 removing threshold and withdrawn actions in favor of fetchSafe --- src/routes/safe/component/AddOwner/actions.js | 4 -- src/routes/safe/component/AddOwner/index.jsx | 5 +- .../safe/component/Threshold/actions.js | 4 -- src/routes/safe/component/Threshold/index.jsx | 3 -- .../safe/component/Transactions/actions.js | 4 -- .../safe/component/Transactions/index.jsx | 6 +-- .../safe/component/Withdrawn/actions.js | 10 ---- src/routes/safe/component/Withdrawn/index.jsx | 8 +-- .../safe/store/actions/fetchDailyLimit.js | 13 ----- .../safe/store/actions/fetchThreshold.js | 12 ----- .../safe/store/actions/updateDailyLimit.js | 20 -------- .../safe/store/actions/updateThreshold.js | 19 ------- src/routes/safe/store/reducer/safe.js | 7 --- .../safe/store/test/dailyLimit.reducer.js | 51 ------------------- src/routes/safe/store/test/safe.spec.js | 4 -- .../safe/store/test/threshold.reducer.js | 48 ----------------- 16 files changed, 4 insertions(+), 214 deletions(-) delete mode 100644 src/routes/safe/component/Withdrawn/actions.js delete mode 100644 src/routes/safe/store/actions/fetchDailyLimit.js delete mode 100644 src/routes/safe/store/actions/fetchThreshold.js delete mode 100644 src/routes/safe/store/actions/updateDailyLimit.js delete mode 100644 src/routes/safe/store/actions/updateThreshold.js delete mode 100644 src/routes/safe/store/test/dailyLimit.reducer.js delete mode 100644 src/routes/safe/store/test/threshold.reducer.js diff --git a/src/routes/safe/component/AddOwner/actions.js b/src/routes/safe/component/AddOwner/actions.js index e0b10a5a..32f51f38 100644 --- a/src/routes/safe/component/AddOwner/actions.js +++ b/src/routes/safe/component/AddOwner/actions.js @@ -1,16 +1,12 @@ // @flow -import fetchThreshold from '~/routes/safe/store/actions/fetchThreshold' import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' -type FetchThreshold = typeof fetchThreshold type FetchTransactions = typeof fetchTransactions export type Actions = { - fetchThreshold: FetchThreshold, fetchTransactions: FetchTransactions, } export default { - fetchThreshold, fetchTransactions, } diff --git a/src/routes/safe/component/AddOwner/index.jsx b/src/routes/safe/component/AddOwner/index.jsx index 245947d1..62b65ce8 100644 --- a/src/routes/safe/component/AddOwner/index.jsx +++ b/src/routes/safe/component/AddOwner/index.jsx @@ -2,7 +2,6 @@ import * as React from 'react' import { List } from 'immutable' import Stepper from '~/components/Stepper' -import { sleep } from '~/utils/timer' import { connect } from 'react-redux' import { type Safe } from '~/routes/safe/store/model/safe' import { type Owner } from '~/routes/safe/store/model/owner' @@ -43,7 +42,7 @@ class AddOwner extends React.Component { onAddOwner = async (values: Object) => { try { const { - safe, threshold, userAddress, fetchTransactions, // fetchOwners + safe, threshold, userAddress, fetchTransactions, } = this.props const nonce = Date.now() const newThreshold = values[INCREASE_PARAM] ? threshold + 1 : threshold @@ -53,9 +52,7 @@ class AddOwner extends React.Component { const gnosisSafe = await getSafeEthereumInstance(safeAddress) const data = gnosisSafe.contract.addOwnerWithThreshold.getData(newOwnerAddress, newThreshold) await createTransaction(safe, `Add Owner ${newOwnerName}`, safeAddress, 0, nonce, userAddress, data) - await sleep(1500) fetchTransactions() - // fetchOwners(safeAddress) this.setState({ done: true }) } catch (error) { this.setState({ done: false }) diff --git a/src/routes/safe/component/Threshold/actions.js b/src/routes/safe/component/Threshold/actions.js index e0b10a5a..32f51f38 100644 --- a/src/routes/safe/component/Threshold/actions.js +++ b/src/routes/safe/component/Threshold/actions.js @@ -1,16 +1,12 @@ // @flow -import fetchThreshold from '~/routes/safe/store/actions/fetchThreshold' import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' -type FetchThreshold = typeof fetchThreshold type FetchTransactions = typeof fetchTransactions export type Actions = { - fetchThreshold: FetchThreshold, fetchTransactions: FetchTransactions, } export default { - fetchThreshold, fetchTransactions, } diff --git a/src/routes/safe/component/Threshold/index.jsx b/src/routes/safe/component/Threshold/index.jsx index 5d144efe..297b8d50 100644 --- a/src/routes/safe/component/Threshold/index.jsx +++ b/src/routes/safe/component/Threshold/index.jsx @@ -3,7 +3,6 @@ import * as React from 'react' import Stepper from '~/components/Stepper' import { connect } from 'react-redux' import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' -import { sleep } from '~/utils/timer' import { type Safe } from '~/routes/safe/store/model/safe' import ThresholdForm, { THRESHOLD_PARAM } from './ThresholdForm' import selector, { type SelectorProps } from './selector' @@ -39,9 +38,7 @@ class Threshold extends React.PureComponent { const nonce = Date.now() const data = gnosisSafe.contract.changeThreshold.getData(newThreshold) await createTransaction(safe, `Change Safe's threshold [${nonce}]`, safe.get('address'), 0, nonce, userAddress, data) - await sleep(1500) await this.props.fetchTransactions() - await this.props.fetchThreshold(safe.get('address')) this.setState({ done: true }) } catch (error) { this.setState({ done: false }) diff --git a/src/routes/safe/component/Transactions/actions.js b/src/routes/safe/component/Transactions/actions.js index e0b10a5a..32f51f38 100644 --- a/src/routes/safe/component/Transactions/actions.js +++ b/src/routes/safe/component/Transactions/actions.js @@ -1,16 +1,12 @@ // @flow -import fetchThreshold from '~/routes/safe/store/actions/fetchThreshold' import fetchTransactions from '~/routes/safe/store/actions/fetchTransactions' -type FetchThreshold = typeof fetchThreshold type FetchTransactions = typeof fetchTransactions export type Actions = { - fetchThreshold: FetchThreshold, fetchTransactions: FetchTransactions, } export default { - fetchThreshold, fetchTransactions, } diff --git a/src/routes/safe/component/Transactions/index.jsx b/src/routes/safe/component/Transactions/index.jsx index 1518ed62..10c3be1f 100644 --- a/src/routes/safe/component/Transactions/index.jsx +++ b/src/routes/safe/component/Transactions/index.jsx @@ -4,7 +4,6 @@ import { connect } from 'react-redux' import { type Transaction } from '~/routes/safe/store/model/transaction' import NoTransactions from '~/routes/safe/component/Transactions/NoTransactions' import GnoTransaction from '~/routes/safe/component/Transactions/Transaction' -import { sleep } from '~/utils/timer' import { processTransaction } from './processTransactions' import selector, { type SelectorProps } from './selector' import actions, { type Actions } from './actions' @@ -18,14 +17,11 @@ type Props = SelectorProps & Actions & { class Transactions extends React.Component { onProcessTx = async (tx: Transaction, alreadyConfirmed: number) => { const { - fetchTransactions, safeAddress, userAddress, fetchThreshold, + fetchTransactions, safeAddress, userAddress, } = this.props await processTransaction(safeAddress, tx, alreadyConfirmed, userAddress) - await sleep(1200) fetchTransactions() - fetchThreshold(safeAddress) - // fetchOwners(safeAddress) } render() { diff --git a/src/routes/safe/component/Withdrawn/actions.js b/src/routes/safe/component/Withdrawn/actions.js deleted file mode 100644 index 8caf6dff..00000000 --- a/src/routes/safe/component/Withdrawn/actions.js +++ /dev/null @@ -1,10 +0,0 @@ -// @flow -import fetchDailyLimit from '~/routes/safe/store/actions/fetchDailyLimit' - -export type Actions = { - fetchDailyLimit: typeof fetchDailyLimit, -} - -export default { - fetchDailyLimit, -} diff --git a/src/routes/safe/component/Withdrawn/index.jsx b/src/routes/safe/component/Withdrawn/index.jsx index c6ddba18..608792be 100644 --- a/src/routes/safe/component/Withdrawn/index.jsx +++ b/src/routes/safe/component/Withdrawn/index.jsx @@ -2,9 +2,7 @@ import * as React from 'react' import { connect } from 'react-redux' import Stepper from '~/components/Stepper' -import { sleep } from '~/utils/timer' import { type DailyLimit } from '~/routes/safe/store/model/dailyLimit' -import actions, { type Actions } from './actions' import selector, { type SelectorProps } from './selector' import withdrawn from './withdrawn' import WithdrawnForm from './WithdrawnForm' @@ -14,7 +12,7 @@ const getSteps = () => [ 'Fill Withdrawn Form', 'Review Withdrawn', ] -type Props = SelectorProps & Actions & { +type Props = SelectorProps & { safeAddress: string, dailyLimit: DailyLimit, } @@ -34,8 +32,6 @@ class Withdrawn extends React.Component { try { const { safeAddress, userAddress } = this.props await withdrawn(values, safeAddress, userAddress) - await sleep(3500) - this.props.fetchDailyLimit(safeAddress) this.setState({ done: true }) } catch (error) { this.setState({ done: false }) @@ -75,5 +71,5 @@ class Withdrawn extends React.Component { } } -export default connect(selector, actions)(Withdrawn) +export default connect(selector)(Withdrawn) diff --git a/src/routes/safe/store/actions/fetchDailyLimit.js b/src/routes/safe/store/actions/fetchDailyLimit.js deleted file mode 100644 index 87cb86b8..00000000 --- a/src/routes/safe/store/actions/fetchDailyLimit.js +++ /dev/null @@ -1,13 +0,0 @@ -// @flow -import type { Dispatch as ReduxDispatch } from 'redux' -import { type GlobalState } from '~/store/index' -import { getDailyLimitFrom } from '~/routes/safe/component/Withdrawn/withdrawn' -import { type DailyLimitProps } from '~/routes/safe/store/model/dailyLimit' -import updateDailyLimit from './updateDailyLimit' - -export default (safeAddress: string) => async (dispatch: ReduxDispatch) => { - const ethAddress = 0 - const dailyLimit: DailyLimitProps = await getDailyLimitFrom(safeAddress, ethAddress) - - return dispatch(updateDailyLimit(safeAddress, dailyLimit)) -} diff --git a/src/routes/safe/store/actions/fetchThreshold.js b/src/routes/safe/store/actions/fetchThreshold.js deleted file mode 100644 index 2ae28e69..00000000 --- a/src/routes/safe/store/actions/fetchThreshold.js +++ /dev/null @@ -1,12 +0,0 @@ -// @flow -import type { Dispatch as ReduxDispatch } from 'redux' -import { type GlobalState } from '~/store/index' -import { getSafeEthereumInstance } from '~/routes/safe/component/AddTransaction/createTransactions' -import updateThreshold from './updateThreshold' - -export default (safeAddress: string) => async (dispatch: ReduxDispatch) => { - const gnosisSafe = await getSafeEthereumInstance(safeAddress) - const actualThreshold = await gnosisSafe.getThreshold() - - return dispatch(updateThreshold(safeAddress, actualThreshold)) -} diff --git a/src/routes/safe/store/actions/updateDailyLimit.js b/src/routes/safe/store/actions/updateDailyLimit.js deleted file mode 100644 index 7e6ff7aa..00000000 --- a/src/routes/safe/store/actions/updateDailyLimit.js +++ /dev/null @@ -1,20 +0,0 @@ -// @flow -import { createAction } from 'redux-actions' -import { type DailyLimitProps } from '~/routes/safe/store/model/dailyLimit' - -export const UPDATE_DAILY_LIMIT = 'UPDATE_DAILY_LIMIT' - -type SpentTodayProps = { - safeAddress: string, - dailyLimit: DailyLimitProps, -} - -const updateDailyLimit = createAction( - UPDATE_DAILY_LIMIT, - (safeAddress: string, dailyLimit: DailyLimitProps): SpentTodayProps => ({ - safeAddress, - dailyLimit, - }), -) - -export default updateDailyLimit diff --git a/src/routes/safe/store/actions/updateThreshold.js b/src/routes/safe/store/actions/updateThreshold.js deleted file mode 100644 index b24f41fd..00000000 --- a/src/routes/safe/store/actions/updateThreshold.js +++ /dev/null @@ -1,19 +0,0 @@ -// @flow -import { createAction } from 'redux-actions' - -export const UPDATE_THRESHOLD = 'UPDATE_THRESHOLD' - -type ThresholdProps = { - safeAddress: string, - threshold: number, -} - -const updateDailyLimit = createAction( - UPDATE_THRESHOLD, - (safeAddress: string, threshold: number): ThresholdProps => ({ - safeAddress, - threshold: Number(threshold), - }), -) - -export default updateDailyLimit diff --git a/src/routes/safe/store/reducer/safe.js b/src/routes/safe/store/reducer/safe.js index bd585812..2f8a072c 100644 --- a/src/routes/safe/store/reducer/safe.js +++ b/src/routes/safe/store/reducer/safe.js @@ -2,11 +2,8 @@ import { Map } from 'immutable' import { handleActions, type ActionType } from 'redux-actions' import addSafe, { ADD_SAFE } from '~/routes/safe/store/actions/addSafe' -import updateDailyLimit, { UPDATE_DAILY_LIMIT } from '~/routes/safe/store/actions/updateDailyLimit' import { type Safe, makeSafe } from '~/routes/safe/store/model/safe' import { saveSafes } from '~/utils/localStorage' -import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' -import updateThreshold, { UPDATE_THRESHOLD } from '~/routes/safe/store/actions/updateThreshold' import updateSafes, { UPDATE_SAFES } from '~/routes/safe/store/actions/updateSafes' import updateSafe, { UPDATE_SAFE } from '~/routes/safe/store/actions/updateSafe' @@ -35,8 +32,4 @@ export default handleActions({ saveSafes(safes.toJSON()) return safes }, - [UPDATE_DAILY_LIMIT]: (state: State, action: ActionType): State => - state.updateIn([action.payload.safeAddress, 'dailyLimit'], () => makeDailyLimit(action.payload.dailyLimit)), - [UPDATE_THRESHOLD]: (state: State, action: ActionType): State => - state.updateIn([action.payload.safeAddress, 'threshold'], () => action.payload.threshold), }, Map()) diff --git a/src/routes/safe/store/test/dailyLimit.reducer.js b/src/routes/safe/store/test/dailyLimit.reducer.js deleted file mode 100644 index 9ee1eb5a..00000000 --- a/src/routes/safe/store/test/dailyLimit.reducer.js +++ /dev/null @@ -1,51 +0,0 @@ -// @flow -import { SAFE_REDUCER_ID } from '~/routes/safe/store/reducer/safe' -import fetchDailyLimit from '~/routes/safe/store/actions/fetchDailyLimit' -import { aNewStore } from '~/store' -import { addEtherTo } from '~/test/addEtherTo' -import { aDeployedSafe, executeWithdrawnOn } from './builder/deployedSafe.builder' - -const updateDailyLimitReducerTests = () => { - describe('Safe Actions[updateDailyLimit]', () => { - let store - beforeEach(async () => { - store = aNewStore() - }) - - it('reducer should return 0 as spentToday value from just deployed safe', async () => { - // GIVEN - const dailyLimitValue = 0.5 - const safeAddress = await aDeployedSafe(store, 0.5) - // WHEN - await store.dispatch(fetchDailyLimit(safeAddress)) - - // THEN - const safes = store.getState()[SAFE_REDUCER_ID] - const dailyLimit = safes.get(safeAddress).get('dailyLimit') - expect(dailyLimit).not.toBe(undefined) - expect(dailyLimit.value).toBe(dailyLimitValue) - expect(dailyLimit.spentToday).toBe(0) - }) - - it('reducer should return 0.1456 ETH as spentToday if the user has withdrawn 0.1456 from MAX of 0.3 ETH', async () => { - // GIVEN - const dailyLimitValue = 0.3 - const safeAddress = await aDeployedSafe(store, dailyLimitValue) - await addEtherTo(safeAddress, '0.5') - const value = 0.1456 - - // WHEN - await executeWithdrawnOn(safeAddress, value) - await store.dispatch(fetchDailyLimit(safeAddress)) - - // THEN - const safes = store.getState()[SAFE_REDUCER_ID] - const dailyLimit = safes.get(safeAddress).get('dailyLimit') - expect(dailyLimit).not.toBe(undefined) - expect(dailyLimit.value).toBe(dailyLimitValue) - expect(dailyLimit.spentToday).toBe(value) - }) - }) -} - -export default updateDailyLimitReducerTests diff --git a/src/routes/safe/store/test/safe.spec.js b/src/routes/safe/store/test/safe.spec.js index 8a0ebd4d..0db061e1 100644 --- a/src/routes/safe/store/test/safe.spec.js +++ b/src/routes/safe/store/test/safe.spec.js @@ -1,8 +1,6 @@ // @flow import balanceReducerTests from './balance.reducer' import safeReducerTests from './safe.reducer' -import dailyLimitReducerTests from './dailyLimit.reducer' -import thresholdReducerTests from './threshold.reducer' import balanceSelectorTests from './balance.selector' import safeSelectorTests from './safe.selector' import grantedSelectorTests from './granted.selector' @@ -13,8 +11,6 @@ describe('Safe Test suite', () => { // ACTIONS AND REDUCERS safeReducerTests() balanceReducerTests() - dailyLimitReducerTests() - thresholdReducerTests() // SAFE SELECTOR safeSelectorTests() diff --git a/src/routes/safe/store/test/threshold.reducer.js b/src/routes/safe/store/test/threshold.reducer.js deleted file mode 100644 index b1d483c3..00000000 --- a/src/routes/safe/store/test/threshold.reducer.js +++ /dev/null @@ -1,48 +0,0 @@ -// @flow -import { SAFE_REDUCER_ID } from '~/routes/safe/store/reducer/safe' -import { aNewStore } from '~/store' -import updateThreshold from '~/routes/safe/store/actions/updateThreshold' -import { aDeployedSafe } from './builder/deployedSafe.builder' - -const thresholdReducerTests = () => { - describe('Safe Actions[updateThreshold]', () => { - let store - beforeEach(async () => { - store = aNewStore() - }) - - it('reducer should return 3 when a safe of 3 threshold has just been created', async () => { - // GIVEN - const safeThreshold = 3 - const numOwners = 3 - - // WHEN - const safeAddress = await aDeployedSafe(store, 0.5, safeThreshold, numOwners) - - // THEN - const safes = store.getState()[SAFE_REDUCER_ID] - const threshold = safes.get(safeAddress).get('confirmations') - expect(threshold).not.toBe(undefined) - expect(threshold).toBe(safeThreshold) - }) - - it('reducer should change correctly', async () => { - // GIVEN - const safeThreshold = 3 - const numOwners = 3 - const safeAddress = await aDeployedSafe(store, 0.5, safeThreshold, numOwners) - - // WHEN - const newThreshold = 1 - await store.dispatch(updateThreshold(safeAddress, newThreshold)) - - // THEN - const safes = store.getState()[SAFE_REDUCER_ID] - const threshold = safes.get(safeAddress).get('confirmations') - expect(threshold).not.toBe(undefined) - expect(threshold).toBe(newThreshold) - }) - }) -} - -export default thresholdReducerTests From 9928fa2a6e3f176c8f66731a29d153a93162baa6 Mon Sep 17 00:00:00 2001 From: apanizo Date: Fri, 8 Jun 2018 09:31:19 +0200 Subject: [PATCH 24/25] WA-234 Fixing tests --- .../safe/test/Safe.multisig.1owners1threshold.test.js | 2 +- src/routes/safe/test/Safe.withdrawn.test.js | 6 +++--- src/routes/safe/test/testMultisig.js | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js b/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js index 80d8b697..68ef3ed3 100644 --- a/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js +++ b/src/routes/safe/test/Safe.multisig.1owners1threshold.test.js @@ -45,7 +45,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[2] + const addTxButton = buttons[3] expect(addTxButton.props.children).toEqual(ADD_MULTISIG_BUTTON_TEXT) await sleep(1800) // Give time to enable Add button TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(addTxButton, 'button')[0]) diff --git a/src/routes/safe/test/Safe.withdrawn.test.js b/src/routes/safe/test/Safe.withdrawn.test.js index a985b3cc..d3df01bc 100644 --- a/src/routes/safe/test/Safe.withdrawn.test.js +++ b/src/routes/safe/test/Safe.withdrawn.test.js @@ -46,7 +46,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const withdrawnButton = buttons[1] + const withdrawnButton = buttons[2] expect(withdrawnButton.props.children).toEqual(WITHDRAWN_BUTTON_TEXT) TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(withdrawnButton, 'button')[0]) await sleep(4000) @@ -96,7 +96,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { const Safe = TestUtils.findRenderedComponentWithType(SafeDom, SafeView) // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[2] + const addTxButton = buttons[3] expect(addTxButton.props.children).toEqual(ADD_MULTISIG_BUTTON_TEXT) expect(addTxButton.props.disabled).toBe(true) @@ -110,7 +110,7 @@ describe('React DOM TESTS > Withdrawn funds from safe', () => { const Safe = TestUtils.findRenderedComponentWithType(SafeDom, SafeView) // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[1] + const addTxButton = buttons[2] expect(addTxButton.props.children).toEqual(WITHDRAWN_BUTTON_TEXT) expect(addTxButton.props.disabled).toBe(true) diff --git a/src/routes/safe/test/testMultisig.js b/src/routes/safe/test/testMultisig.js index 9c10e3c2..8e469410 100644 --- a/src/routes/safe/test/testMultisig.js +++ b/src/routes/safe/test/testMultisig.js @@ -48,7 +48,7 @@ export const addFundsTo = async (SafeDom: React$Component, destination // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const addTxButton = buttons[2] + const addTxButton = buttons[3] expect(addTxButton.props.children).toEqual(ADD_MULTISIG_BUTTON_TEXT) await sleep(1800) // Give time to enable Add button TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(addTxButton, 'button')[0]) @@ -59,7 +59,7 @@ export const listTxsOf = (SafeDom: React$Component) => { // $FlowFixMe const buttons = TestUtils.scryRenderedComponentsWithType(Safe, Button) - const seeTx = buttons[3] + const seeTx = buttons[4] expect(seeTx.props.children).toEqual(SEE_MULTISIG_BUTTON_TEXT) TestUtils.Simulate.click(TestUtils.scryRenderedDOMComponentsWithTag(seeTx, 'button')[0]) } From 577d7e8749b7f5446d0b6e268ec2b1c5b4fdf0ec Mon Sep 17 00:00:00 2001 From: apanizo Date: Fri, 8 Jun 2018 10:26:01 +0200 Subject: [PATCH 25/25] WA-238 Decoupling storage of names in a separate step in local host --- src/routes/safe/component/AddOwner/index.jsx | 4 +++- src/routes/safe/store/actions/fetchSafe.js | 11 ++++++----- src/routes/safe/store/reducer/safe.js | 7 +++++-- src/utils/localStorage.js | 20 ++++++++++++++++++++ 4 files changed, 34 insertions(+), 8 deletions(-) diff --git a/src/routes/safe/component/AddOwner/index.jsx b/src/routes/safe/component/AddOwner/index.jsx index 62b65ce8..ee31bcdd 100644 --- a/src/routes/safe/component/AddOwner/index.jsx +++ b/src/routes/safe/component/AddOwner/index.jsx @@ -4,8 +4,9 @@ import { List } from 'immutable' import Stepper from '~/components/Stepper' import { connect } from 'react-redux' import { type Safe } from '~/routes/safe/store/model/safe' -import { type Owner } from '~/routes/safe/store/model/owner' +import { type Owner, makeOwner } from '~/routes/safe/store/model/owner' import { getSafeEthereumInstance, createTransaction } from '~/routes/safe/component/AddTransaction/createTransactions' +import { setOwners } from '~/utils/localStorage' import AddOwnerForm, { NAME_PARAM, OWNER_ADDRESS_PARAM, INCREASE_PARAM } from './AddOwnerForm' import Review from './Review' import selector, { type SelectorProps } from './selector' @@ -52,6 +53,7 @@ class AddOwner extends React.Component { const gnosisSafe = await getSafeEthereumInstance(safeAddress) const data = gnosisSafe.contract.addOwnerWithThreshold.getData(newOwnerAddress, newThreshold) await createTransaction(safe, `Add Owner ${newOwnerName}`, safeAddress, 0, nonce, userAddress, data) + setOwners(safeAddress, safe.get('owners').push(makeOwner({ name: newOwnerName, address: newOwnerAddress }))) fetchTransactions() this.setState({ done: true }) } catch (error) { diff --git a/src/routes/safe/store/actions/fetchSafe.js b/src/routes/safe/store/actions/fetchSafe.js index 60b0cacf..9102489a 100644 --- a/src/routes/safe/store/actions/fetchSafe.js +++ b/src/routes/safe/store/actions/fetchSafe.js @@ -1,6 +1,6 @@ // @flow import type { Dispatch as ReduxDispatch } from 'redux' -import { List } from 'immutable' +import { List, Map } from 'immutable' import { type GlobalState } from '~/store/index' import { makeOwner } from '~/routes/safe/store/model/owner' import { type SafeProps, type Safe, makeSafe } from '~/routes/safe/store/model/safe' @@ -8,11 +8,12 @@ import { makeDailyLimit } from '~/routes/safe/store/model/dailyLimit' import { getDailyLimitFrom } from '~/routes/safe/component/Withdrawn/withdrawn' import { getGnosisSafeInstanceAt } from '~/wallets/safeContracts' import updateSafe from '~/routes/safe/store/actions/updateSafe' +import { getOwners } from '~/utils/localStorage' -const buildOwnersFrom = (safeOwners: string[], storedOwners: Object[]) => ( +const buildOwnersFrom = (safeOwners: string[], storedOwners: Map) => ( safeOwners.map((ownerAddress: string) => { - const foundOwner = storedOwners.find(owner => owner.address === ownerAddress) - return makeOwner(foundOwner || { name: 'UNKNOWN', address: ownerAddress }) + const ownerName = storedOwners.get(ownerAddress.toLowerCase()) || 'UNKNOWN' + return makeOwner({ name: ownerName, address: ownerAddress }) }) ) @@ -22,7 +23,7 @@ export const buildSafe = async (storedSafe: Object) => { const dailyLimit = makeDailyLimit(await getDailyLimitFrom(safeAddress, 0)) const threshold = Number(await gnosisSafe.getThreshold()) - const owners = List(buildOwnersFrom(await gnosisSafe.getOwners(), storedSafe.owners)) + const owners = List(buildOwnersFrom(await gnosisSafe.getOwners(), getOwners(safeAddress))) const safe: SafeProps = { address: safeAddress, diff --git a/src/routes/safe/store/reducer/safe.js b/src/routes/safe/store/reducer/safe.js index 2f8a072c..dd941407 100644 --- a/src/routes/safe/store/reducer/safe.js +++ b/src/routes/safe/store/reducer/safe.js @@ -3,7 +3,7 @@ import { Map } from 'immutable' import { handleActions, type ActionType } from 'redux-actions' import addSafe, { ADD_SAFE } from '~/routes/safe/store/actions/addSafe' import { type Safe, makeSafe } from '~/routes/safe/store/model/safe' -import { saveSafes } from '~/utils/localStorage' +import { saveSafes, setOwners } from '~/utils/localStorage' import updateSafes, { UPDATE_SAFES } from '~/routes/safe/store/actions/updateSafes' import updateSafe, { UPDATE_SAFE } from '~/routes/safe/store/actions/updateSafe' @@ -28,7 +28,10 @@ export default handleActions({ [UPDATE_SAFES]: (state: State, action: ActionType): State => action.payload, [ADD_SAFE]: (state: State, action: ActionType): State => { - const safes = state.set(action.payload.address, makeSafe(action.payload)) + const safe: Safe = makeSafe(action.payload) + setOwners(safe.get('address'), safe.get('owners')) + + const safes = state.set(action.payload.address, safe) saveSafes(safes.toJSON()) return safes }, diff --git a/src/utils/localStorage.js b/src/utils/localStorage.js index e537d38a..072ed733 100644 --- a/src/utils/localStorage.js +++ b/src/utils/localStorage.js @@ -1,6 +1,10 @@ // @flow +import { List, Map } from 'immutable' +import { type Owner } from '~/routes/safe/store/model/owner' + export const SAFES_KEY = 'SAFES' export const TX_KEY = 'TX' +export const OWNERS_KEY = 'OWNERS' export const load = (key: string) => { try { @@ -27,3 +31,19 @@ export const saveSafes = (safes: Object) => { // Ignore write errors } } + +export const setOwners = (safeAddress: string, owners: List) => { + try { + const ownersAsMap = Map(owners.map((owner: Owner) => [owner.get('address').toLowerCase(), owner.get('name')])) + const serializedState = JSON.stringify(ownersAsMap) + localStorage.setItem(`${OWNERS_KEY}-${safeAddress}`, serializedState) + } catch (err) { + // Ignore write errors + } +} + +export const getOwners = (safeAddress: string): Map => { + const data = load(`${OWNERS_KEY}-${safeAddress}`) + + return data ? Map(data) : Map() +}