enabling and disabling tokens tests wip

This commit is contained in:
Mikhail Mikheev 2019-06-05 17:01:45 +04:00
parent 4625ed1aa5
commit ca81ff73fa
7 changed files with 51 additions and 119 deletions

View File

@ -37,7 +37,7 @@ class TextField extends React.PureComponent<TextFieldProps> {
const underline = meta.active || (meta.visited && !meta.valid) const underline = meta.active || (meta.visited && !meta.valid)
const inputRoot = helperText ? classes.root : undefined const inputRoot = helperText ? classes.root : undefined
const inputProps = { ...restInput, autoComplete: 'off', 'data-testid': testId } const inputProps = { ...restInput, autoComplete: 'off', 'data-testid': testId }
const inputRootProps = { ...inputAdornment, disableUnderline: !underline, className: inputRoot } const inputRootProps = { ...inputAdornment, disableUnderline: !underline, className: inputRoot }
return ( return (

View File

@ -20,7 +20,9 @@ const calculateStyleBased = (minWidth, minHeight) => ({
minHeight: minHeight && `${minHeight}px`, minHeight: minHeight && `${minHeight}px`,
}) })
const GnoButton = ({ minWidth, minHeight, testId = '', ...props }: Props) => { const GnoButton = ({
minWidth, minHeight, testId = '', ...props
}: Props) => {
const style = calculateStyleBased(minWidth, minHeight) const style = calculateStyleBased(minWidth, minHeight)
return <Button style={style} data-testid={testId} {...props} /> return <Button style={style} data-testid={testId} {...props} />

View File

@ -25,6 +25,7 @@ import { setImageToPlaceholder } from '~/routes/safe/components/Balances/utils'
import { styles } from './style' import { styles } from './style'
export const ADD_CUSTOM_TOKEN_BUTTON_TEST_ID = 'add-custom-token-btn' export const ADD_CUSTOM_TOKEN_BUTTON_TEST_ID = 'add-custom-token-btn'
export const TOGGLE_TOKEN_TEST_ID = 'toggle-token-btn'
type Props = { type Props = {
classes: Object, classes: Object,
@ -167,7 +168,11 @@ class Tokens extends React.Component<Props, State> {
<ListItemText primary={token.symbol} secondary={token.name} /> <ListItemText primary={token.symbol} secondary={token.name} />
{token.address !== ETH_ADDRESS && ( {token.address !== ETH_ADDRESS && (
<ListItemSecondaryAction> <ListItemSecondaryAction>
<Switch onChange={this.onSwitch(token)} checked={isActive} /> <Switch
onChange={this.onSwitch(token)}
checked={isActive}
inputProps={{ 'data-testid': `${token.symbol}_${TOGGLE_TOKEN_TEST_ID}` }}
/>
</ListItemSecondaryAction> </ListItemSecondaryAction>
)} )}
</ListItem> </ListItem>

View File

@ -119,15 +119,6 @@ export const renderSafeView = (store: Store<GlobalState>, address: string) => {
return app return app
} }
export const travelToTokens = (store: Store<GlobalState>, address: string) => {
const app = renderApp(store)
const url = `${SAFELIST_ADDRESS}/${address}${SETTINS_ADDRESS}`
history.push(url)
return app
}
const INTERVAL = 500 const INTERVAL = 500
const MAX_TIMES_EXECUTED = 30 const MAX_TIMES_EXECUTED = 30
export const whenSafeDeployed = (): Promise<string> => new Promise((resolve, reject) => { export const whenSafeDeployed = (): Promise<string> => new Promise((resolve, reject) => {

View File

@ -1,23 +1,6 @@
// @flow // @flow
import * as TestUtils from 'react-dom/test-utils'
import { travelToTokens } from '~/test/builder/safe.dom.utils'
import { sleep } from '~/utils/timer'
import { type Token } from '~/logic/tokens/store/model/token' import { type Token } from '~/logic/tokens/store/model/token'
export const enableFirstToken = async (store: Store, safeAddress: string) => {
const TokensDom = await travelToTokens(store, safeAddress)
await sleep(400)
// WHEN
const inputs = TestUtils.scryRenderedDOMComponentsWithTag(TokensDom, 'input')
const ethTokenInput = inputs[2]
expect(ethTokenInput.hasAttribute('disabled')).toBe(true)
const firstTokenInput = inputs[0]
expect(firstTokenInput.hasAttribute('disabled')).toBe(false)
TestUtils.Simulate.change(firstTokenInput, { target: { checked: 'true' } })
}
export const testToken = (token: Token | typeof undefined, symbol: string) => { export const testToken = (token: Token | typeof undefined, symbol: string) => {
if (!token) throw new Error() if (!token) throw new Error()
expect(token.get('symbol')).toBe(symbol) expect(token.get('symbol')).toBe(symbol)

View File

@ -1,65 +1,68 @@
// @flow // @flow
import * as TestUtils from 'react-dom/test-utils'
import { List } from 'immutable' import { List } from 'immutable'
import { getWeb3 } from '~/logic/wallets/getWeb3' import { getWeb3 } from '~/logic/wallets/getWeb3'
import { type Match } from 'react-router-dom'
import Checkbox from '@material-ui/core/Checkbox' import Checkbox from '@material-ui/core/Checkbox'
import { getFirstTokenContract, getSecondTokenContract, sendTokenTo } from '~/test/utils/tokenMovements' import { getFirstTokenContract, getSecondTokenContract } from '~/test/utils/tokenMovements'
import { aNewStore } from '~/store' import { aNewStore } from '~/store'
import { aMinedSafe } from '~/test/builder/safe.redux.builder' import { aMinedSafe } from '~/test/builder/safe.redux.builder'
import { travelToTokens } from '~/test/builder/safe.dom.utils' import { renderSafeView } from '~/test/builder/safe.dom.utils'
import { sleep } from '~/utils/timer' import { sleep } from '~/utils/timer'
import { buildMathPropsFrom } from '~/test/utils/buildReactRouterProps' import { testToken } from '~/test/builder/tokens.dom.utils'
import { tokenListSelector } from '~/logic/tokens/store/selectors' import saveTokens from '~/logic/tokens/store/actions/saveTokens'
import { getActiveTokenAddresses } from '~/logic/tokens/utils/tokensStorage' import { clickOnManageTokens, toggleToken } from './utils/DOMNavigation'
import { enableFirstToken, testToken } from '~/test/builder/tokens.dom.utils' import { BALANCE_ROW_TEST_ID } from '~/routes/safe/components/Balances'
import * as fetchTokensModule from '~/logic/tokens/store/actions/fetchTokens' import { makeToken } from '~/logic/tokens/store/model/token'
import * as enhancedFetchModule from '~/utils/fetch'
describe('DOM > Feature > Enable and disable default tokens', () => { describe('DOM > Feature > Enable and disable default tokens', () => {
let web3 let web3
let accounts let accounts
let firstErc20Token let firstErc20Token
let secondErc20Token let secondErc20Token
let tokens
beforeAll(async () => { beforeAll(async () => {
web3 = getWeb3() web3 = getWeb3()
accounts = await web3.eth.getAccounts() accounts = await web3.eth.getAccounts()
firstErc20Token = await getFirstTokenContract(web3, accounts[0]) firstErc20Token = await getFirstTokenContract(web3, accounts[0])
secondErc20Token = await getSecondTokenContract(web3, accounts[0]) secondErc20Token = await getSecondTokenContract(web3, accounts[0])
// $FlowFixMe tokens = List([
enhancedFetchModule.enhancedFetch = jest.fn() makeToken({
enhancedFetchModule.enhancedFetch.mockImplementation(() => Promise.resolve({ address: firstErc20Token.address,
results: [ name: 'First Token Example',
{ symbol: 'FTE',
address: firstErc20Token.address, decimals: 18,
name: 'First Token Example', logoUri: 'https://upload.wikimedia.org/wikipedia/commons/c/c0/Earth_simple_icon.png',
symbol: 'FTE', }),
decimals: 18, makeToken({
logoUri: 'https://upload.wikimedia.org/wikipedia/commons/c/c0/Earth_simple_icon.png', address: secondErc20Token.address,
}, name: 'Second Token Example',
{ symbol: 'STE',
address: secondErc20Token.address, decimals: 18,
name: 'Second Token Example', logoUri: 'https://upload.wikimedia.org/wikipedia/commons/c/c0/Earth_simple_icon.png',
symbol: 'STE', }),
decimals: 18, ])
logoUri: 'https://upload.wikimedia.org/wikipedia/commons/c/c0/Earth_simple_icon.png',
},
],
}))
}) })
it('retrieves only ether as active token in first moment', async () => { it('allows to enable and disable tokens', async () => {
// GIVEN // GIVEN
const store = aNewStore() const store = aNewStore()
const safeAddress = await aMinedSafe(store) const safeAddress = await aMinedSafe(store)
await store.dispatch(fetchTokensModule.fetchTokens(safeAddress)) await store.dispatch(saveTokens(tokens))
// WHEN // WHEN
const TokensDom = await travelToTokens(store, safeAddress) const TokensDom = await renderSafeView(store, safeAddress)
await sleep(400) await sleep(400)
// Check if only ETH is enabled
TokensDom.getAllByTestId(BALANCE_ROW_TEST_ID)
console.log(store.getState().tokens.toJS())
// THEN // THEN
clickOnManageTokens(TokensDom)
toggleToken(TokensDom, 'FTE', firstErc20Token.address)
toggleToken(TokensDom, 'STE', secondErc20Token.address)
const tokens = TestUtils.scryRenderedComponentsWithType(TokensDom, TokenComponent) const tokens = TestUtils.scryRenderedComponentsWithType(TokensDom, TokenComponent)
expect(tokens.length).toBe(3) expect(tokens.length).toBe(3)
@ -71,60 +74,4 @@ describe('DOM > Feature > Enable and disable default tokens', () => {
if (!ethCheckbox) throw new Error() if (!ethCheckbox) throw new Error()
expect(ethCheckbox.props.disabled).toBe(true) expect(ethCheckbox.props.disabled).toBe(true)
}) })
it('fetch balances of only enabled tokens', async () => {
// GIVEN
const store = aNewStore()
const safeAddress = await aMinedSafe(store)
await sendTokenTo(safeAddress, '50', firstErc20Token)
await sendTokenTo(safeAddress, '50', secondErc20Token)
await store.dispatch(fetchTokensModule.fetchTokens(safeAddress))
const match: Match = buildMathPropsFrom(safeAddress)
let tokenList = tokenListSelector(store.getState(), { match })
expect(tokenList.count()).toBe(3)
await enableFirstToken(store, safeAddress)
tokenList = tokenListSelector(store.getState(), { match })
expect(tokenList.count()).toBe(3) // assuring the enableToken do not add extra info
// THEN
testToken(tokenList.get(0), 'FTE', true)
testToken(tokenList.get(1), 'STE', false)
testToken(tokenList.get(2), 'ETH', true)
const activeTokenList = activeTokensSelector(store.getState(), { match })
expect(activeTokenList.count()).toBe(2)
testToken(activeTokenList.get(0), 'FTE', true)
testToken(activeTokenList.get(1), 'ETH', true)
await store.dispatch(fetchTokensModule.fetchTokens(safeAddress))
const fundedTokenList = tokenListSelector(store.getState(), { match })
expect(fundedTokenList.count()).toBe(3)
testToken(fundedTokenList.get(0), 'FTE', true, '50')
testToken(fundedTokenList.get(1), 'STE', false, '0')
testToken(fundedTokenList.get(2), 'ETH', true, '0')
})
it('localStorage always returns a list', async () => {
const store = aNewStore()
const safeAddress = await aMinedSafe(store)
let tokens: List<string> = getActiveTokenAddresses(safeAddress)
expect(tokens).toEqual(List([]))
await store.dispatch(fetchTokensModule.fetchTokens(safeAddress))
tokens = getActiveTokenAddresses(safeAddress)
expect(tokens.count()).toBe(0)
await enableFirstToken(store, safeAddress)
tokens = getActiveTokenAddresses(safeAddress)
expect(tokens.count()).toBe(1)
await store.dispatch(fetchTokensModule.fetchTokens(safeAddress))
tokens = getActiveTokenAddresses(safeAddress)
expect(tokens.count()).toBe(1)
})
}) })

View File

@ -1,7 +1,7 @@
// @flow // @flow
import { fireEvent } from '@testing-library/react' import { fireEvent } from '@testing-library/react'
import { MANAGE_TOKENS_BUTTON_TEST_ID } from '~/routes/safe/components/Balances' import { MANAGE_TOKENS_BUTTON_TEST_ID } from '~/routes/safe/components/Balances'
import { ADD_CUSTOM_TOKEN_BUTTON_TEST_ID } from '~/routes/safe/components/Balances/Tokens/screens/TokenList' import { ADD_CUSTOM_TOKEN_BUTTON_TEST_ID, TOGGLE_TOKEN_TEST_ID } from '~/routes/safe/components/Balances/Tokens/screens/TokenList'
export const clickOnManageTokens = (dom: any): void => { export const clickOnManageTokens = (dom: any): void => {
const btn = dom.getByTestId(MANAGE_TOKENS_BUTTON_TEST_ID) const btn = dom.getByTestId(MANAGE_TOKENS_BUTTON_TEST_ID)
@ -14,3 +14,7 @@ export const clickOnAddCustomToken = (dom: any): void => {
fireEvent.click(btn) fireEvent.click(btn)
} }
export const toggleToken = (dom: any, symbol: string): void => {
const btn = dom.getByTestId(`${symbol}_${TOGGLE_TOKEN_TEST_ID}`)
}