William O'Beirne 33f5ede22a Minor swap rates refactor (#1162)
* Refactor swaps:
* Remove references to bity fallback, it aint happening
* Consolidate sagas to single orchestrator function like other sagas
* Grab rates once, dont continuously poll

* tscheck

* Re-instate the auto-fetching behavior. This time, only notify the first time.

* Make CurrentRates responsible for redux actions. Fix up some typings.

* Remove commented code.

* Update snapshot.
2018-02-24 19:29:34 -06:00

94 lines
3.2 KiB
TypeScript

import { showNotification } from 'actions/notifications';
import {
loadBityRatesSucceededSwap,
loadShapeshiftRatesSucceededSwap,
loadBityRatesFailedSwap,
loadShapeshiftRatesFailedSwap,
changeSwapProvider,
ChangeProviderSwapAcion
} from 'actions/swap';
import { TypeKeys } from 'actions/swap/constants';
import { getAllRates } from 'api/bity';
import { delay, SagaIterator } from 'redux-saga';
import { call, select, put, takeLatest, race, take, cancel, fork } from 'redux-saga/effects';
import shapeshift from 'api/shapeshift';
import { getSwap } from 'sagas/swap/orders';
import { getHasNotifiedRatesFailure } from 'selectors/swap';
export const SHAPESHIFT_TIMEOUT = 10000;
export const POLLING_CYCLE = 30000;
export function* loadBityRates(): SagaIterator {
while (true) {
try {
const data = yield call(getAllRates);
yield put(loadBityRatesSucceededSwap(data));
} catch (error) {
const hasNotified = yield select(getHasNotifiedRatesFailure);
if (!hasNotified) {
console.error('Failed to load rates from Bity:', error);
yield put(showNotification('danger', error.message));
}
yield put(loadBityRatesFailedSwap());
}
yield call(delay, POLLING_CYCLE);
}
}
export function* handleBityRates(): SagaIterator {
const loadBityRatesTask = yield fork(loadBityRates);
yield take(TypeKeys.SWAP_STOP_LOAD_BITY_RATES);
yield cancel(loadBityRatesTask);
}
export function* loadShapeshiftRates(): SagaIterator {
while (true) {
try {
// Race b/w api call and timeout
// getShapeShiftRates should be an api call that accepts a whitelisted arr of symbols
const { tokens } = yield race({
tokens: call(shapeshift.getAllRates),
timeout: call(delay, SHAPESHIFT_TIMEOUT)
});
// If tokens exist, put it into the redux state, otherwise switch to bity.
if (tokens) {
yield put(loadShapeshiftRatesSucceededSwap(tokens));
} else {
throw new Error('ShapeShift rates request timed out.');
}
} catch (error) {
const hasNotified = yield select(getHasNotifiedRatesFailure);
if (!hasNotified) {
console.error('Failed to fetch rates from shapeshift:', error);
yield put(
showNotification(
'danger',
'Failed to load swap rates from ShapeShift, please try again later'
)
);
}
yield put(loadShapeshiftRatesFailedSwap());
}
yield call(delay, POLLING_CYCLE);
}
}
export function* handleShapeshiftRates(): SagaIterator {
const loadShapeshiftRatesTask = yield fork(loadShapeshiftRates);
yield take(TypeKeys.SWAP_STOP_LOAD_SHAPESHIFT_RATES);
yield cancel(loadShapeshiftRatesTask);
}
export function* swapProvider(action: ChangeProviderSwapAcion): SagaIterator {
const swap = yield select(getSwap);
if (swap.provider !== action.payload) {
yield put(changeSwapProvider(action.payload));
}
}
export default function* swapRates(): SagaIterator {
yield takeLatest(TypeKeys.SWAP_LOAD_BITY_RATES_REQUESTED, handleBityRates);
yield takeLatest(TypeKeys.SWAP_LOAD_SHAPESHIFT_RATES_REQUESTED, handleShapeshiftRates);
yield takeLatest(TypeKeys.SWAP_CHANGE_PROVIDER, swapProvider);
}