2017-11-29 23:07:16 -05:00
|
|
|
import { showNotification } from 'actions/notifications';
|
2018-02-24 20:29:34 -05:00
|
|
|
import {
|
|
|
|
loadBityRatesSucceededSwap,
|
|
|
|
loadShapeshiftRatesSucceededSwap,
|
|
|
|
loadShapeshiftRatesFailedSwap,
|
|
|
|
loadBityRatesFailedSwap
|
|
|
|
} from 'actions/swap';
|
2017-11-29 23:07:16 -05:00
|
|
|
import { getAllRates } from 'api/bity';
|
|
|
|
import { delay } from 'redux-saga';
|
2018-02-24 20:29:34 -05:00
|
|
|
import { call, cancel, fork, put, race, take, select } from 'redux-saga/effects';
|
2017-11-29 23:07:16 -05:00
|
|
|
import { createMockTask } from 'redux-saga/utils';
|
2018-01-02 12:04:50 -06:00
|
|
|
import {
|
|
|
|
loadBityRates,
|
|
|
|
loadShapeshiftRates,
|
2018-02-24 20:29:34 -05:00
|
|
|
handleBityRates,
|
|
|
|
handleShapeshiftRates,
|
2018-01-02 12:04:50 -06:00
|
|
|
SHAPESHIFT_TIMEOUT,
|
2018-02-24 20:29:34 -05:00
|
|
|
POLLING_CYCLE
|
2018-01-02 12:04:50 -06:00
|
|
|
} from 'sagas/swap/rates';
|
|
|
|
import shapeshift from 'api/shapeshift';
|
|
|
|
import { TypeKeys } from 'actions/swap/constants';
|
2018-02-24 20:29:34 -05:00
|
|
|
import { getHasNotifiedRatesFailure } from 'selectors/swap';
|
2017-11-29 23:07:16 -05:00
|
|
|
|
|
|
|
describe('loadBityRates*', () => {
|
|
|
|
const gen1 = loadBityRates();
|
2017-12-11 12:44:53 -05:00
|
|
|
const apiResponse = {
|
|
|
|
BTCETH: {
|
|
|
|
id: 'BTCETH',
|
|
|
|
options: [{ id: 'BTC' }, { id: 'ETH' }],
|
|
|
|
rate: 23.27855114
|
|
|
|
},
|
|
|
|
ETHBTC: {
|
|
|
|
id: 'ETHBTC',
|
|
|
|
options: [{ id: 'ETH' }, { id: 'BTC' }],
|
|
|
|
rate: 0.042958
|
|
|
|
}
|
2017-11-29 23:07:16 -05:00
|
|
|
};
|
2018-02-24 20:29:34 -05:00
|
|
|
const err = { message: 'error' };
|
2018-03-07 18:36:05 -05:00
|
|
|
let random: () => number;
|
2017-11-29 23:07:16 -05:00
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
random = Math.random;
|
|
|
|
Math.random = () => 0.001;
|
|
|
|
});
|
|
|
|
|
|
|
|
afterAll(() => {
|
|
|
|
Math.random = random;
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should call getAllRates', () => {
|
|
|
|
expect(gen1.next().value).toEqual(call(getAllRates));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should put loadBityRatesSucceededSwap', () => {
|
2017-12-11 12:44:53 -05:00
|
|
|
expect(gen1.next(apiResponse).value).toEqual(put(loadBityRatesSucceededSwap(apiResponse)));
|
2017-11-29 23:07:16 -05:00
|
|
|
});
|
|
|
|
|
2018-02-24 20:29:34 -05:00
|
|
|
it(`should delay for ${POLLING_CYCLE}ms`, () => {
|
|
|
|
expect(gen1.next().value).toEqual(call(delay, POLLING_CYCLE));
|
2017-11-29 23:07:16 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle an exception', () => {
|
2018-02-24 20:29:34 -05:00
|
|
|
const errGen = loadBityRates();
|
|
|
|
errGen.next();
|
|
|
|
expect((errGen as any).throw(err).value).toEqual(select(getHasNotifiedRatesFailure));
|
|
|
|
expect(errGen.next(false).value).toEqual(put(showNotification('danger', err.message)));
|
|
|
|
expect(errGen.next().value).toEqual(put(loadBityRatesFailedSwap()));
|
|
|
|
expect(errGen.next().value).toEqual(call(delay, POLLING_CYCLE));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not notify on subsequent exceptions', () => {
|
|
|
|
const noNotifyErrGen = loadBityRates();
|
|
|
|
noNotifyErrGen.next();
|
|
|
|
expect((noNotifyErrGen as any).throw(err).value).toEqual(select(getHasNotifiedRatesFailure));
|
|
|
|
expect(noNotifyErrGen.next(true).value).toEqual(put(loadBityRatesFailedSwap()));
|
|
|
|
expect(noNotifyErrGen.next().value).toEqual(call(delay, POLLING_CYCLE));
|
2017-11-29 23:07:16 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-01-02 12:04:50 -06:00
|
|
|
describe('loadShapeshiftRates*', () => {
|
|
|
|
const gen1 = loadShapeshiftRates();
|
|
|
|
|
|
|
|
const apiResponse = {
|
|
|
|
['1SSTANT']: {
|
|
|
|
id: '1STANT',
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
id: '1ST',
|
|
|
|
status: 'available',
|
|
|
|
image: 'https://shapeshift.io/images/coins/firstblood.png',
|
|
|
|
name: 'FirstBlood'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'ANT',
|
|
|
|
status: 'available',
|
|
|
|
image: 'https://shapeshift.io/images/coins/aragon.png',
|
|
|
|
name: 'Aragon'
|
|
|
|
}
|
|
|
|
],
|
|
|
|
rate: '0.24707537',
|
|
|
|
limit: 5908.29166225,
|
|
|
|
min: 7.86382979
|
|
|
|
}
|
|
|
|
};
|
2018-02-24 20:29:34 -05:00
|
|
|
const err = 'error';
|
2018-03-07 18:36:05 -05:00
|
|
|
let random: () => number;
|
2018-01-02 12:04:50 -06:00
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
random = Math.random;
|
|
|
|
Math.random = () => 0.001;
|
|
|
|
});
|
|
|
|
|
|
|
|
afterAll(() => {
|
|
|
|
Math.random = random;
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should race shapeshift.getAllRates', () => {
|
|
|
|
expect(gen1.next().value).toEqual(
|
|
|
|
race({
|
|
|
|
tokens: call(shapeshift.getAllRates),
|
|
|
|
timeout: call(delay, SHAPESHIFT_TIMEOUT)
|
|
|
|
})
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should put loadShapeshiftRatesSucceededSwap', () => {
|
|
|
|
expect(gen1.next({ tokens: apiResponse }).value).toEqual(
|
2018-03-07 18:36:05 -05:00
|
|
|
put(loadShapeshiftRatesSucceededSwap(apiResponse as any))
|
2018-01-02 12:04:50 -06:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2018-02-24 20:29:34 -05:00
|
|
|
it(`should delay for ${POLLING_CYCLE}ms`, () => {
|
|
|
|
expect(gen1.next().value).toEqual(call(delay, POLLING_CYCLE));
|
2018-01-02 12:04:50 -06:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle an exception', () => {
|
2018-02-24 20:29:34 -05:00
|
|
|
const errGen = loadShapeshiftRates();
|
|
|
|
errGen.next();
|
|
|
|
expect((errGen as any).throw(err).value).toEqual(select(getHasNotifiedRatesFailure));
|
|
|
|
expect(errGen.next(false).value).toEqual(
|
|
|
|
put(
|
|
|
|
showNotification(
|
|
|
|
'danger',
|
|
|
|
'Failed to load swap rates from ShapeShift, please try again later'
|
|
|
|
)
|
|
|
|
)
|
2018-01-02 12:04:50 -06:00
|
|
|
);
|
2018-02-24 20:29:34 -05:00
|
|
|
expect(errGen.next().value).toEqual(put(loadShapeshiftRatesFailedSwap()));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not notify on subsequent exceptions', () => {
|
|
|
|
const noNotifyErrGen = loadShapeshiftRates();
|
|
|
|
noNotifyErrGen.next();
|
|
|
|
expect((noNotifyErrGen as any).throw(err).value).toEqual(select(getHasNotifiedRatesFailure));
|
|
|
|
expect(noNotifyErrGen.next(true).value).toEqual(put(loadShapeshiftRatesFailedSwap()));
|
2018-01-02 12:04:50 -06:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2017-11-29 23:07:16 -05:00
|
|
|
describe('handleBityRates*', () => {
|
|
|
|
const gen = handleBityRates();
|
|
|
|
const mockTask = createMockTask();
|
|
|
|
|
|
|
|
it('should fork loadBityRates', () => {
|
|
|
|
expect(gen.next().value).toEqual(fork(loadBityRates));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should take SWAP_STOP_LOAD_BITY_RATES', () => {
|
2018-01-02 12:04:50 -06:00
|
|
|
expect(gen.next(mockTask).value).toEqual(take(TypeKeys.SWAP_STOP_LOAD_BITY_RATES));
|
2017-11-29 23:07:16 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should cancel loadBityRatesTask', () => {
|
|
|
|
expect(gen.next().value).toEqual(cancel(mockTask));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be done', () => {
|
|
|
|
expect(gen.next().done).toEqual(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-01-02 12:04:50 -06:00
|
|
|
describe('handleShapeshiftRates*', () => {
|
2018-02-24 20:29:34 -05:00
|
|
|
const gen = handleShapeshiftRates();
|
2018-01-02 12:04:50 -06:00
|
|
|
const mockTask = createMockTask();
|
|
|
|
|
|
|
|
it('should fork loadShapeshiftRates', () => {
|
|
|
|
expect(gen.next().value).toEqual(fork(loadShapeshiftRates));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should take SWAP_STOP_LOAD_BITY_RATES', () => {
|
|
|
|
expect(gen.next(mockTask).value).toEqual(take(TypeKeys.SWAP_STOP_LOAD_SHAPESHIFT_RATES));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should cancel loadShapeShiftRatesTask', () => {
|
|
|
|
expect(gen.next().value).toEqual(cancel(mockTask));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be done', () => {
|
|
|
|
expect(gen.next().done).toEqual(true);
|
|
|
|
});
|
|
|
|
});
|