mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
Connect gas price chart to gas station api.
This commit is contained in:
parent
e3f015c88f
commit
79de7a45ae
@ -1,6 +1,9 @@
|
|||||||
import { mockGasEstimateData } from './mock-gas-estimate-data'
|
|
||||||
import { clone, uniqBy } from 'ramda'
|
import { clone, uniqBy } from 'ramda'
|
||||||
import BigNumber from 'bignumber.js'
|
import BigNumber from 'bignumber.js'
|
||||||
|
import {
|
||||||
|
loadLocalStorageData,
|
||||||
|
saveLocalStorageData,
|
||||||
|
} from '../../lib/local-storage-helpers'
|
||||||
|
|
||||||
// Actions
|
// Actions
|
||||||
const BASIC_GAS_ESTIMATE_LOADING_FINISHED = 'metamask/gas/BASIC_GAS_ESTIMATE_LOADING_FINISHED'
|
const BASIC_GAS_ESTIMATE_LOADING_FINISHED = 'metamask/gas/BASIC_GAS_ESTIMATE_LOADING_FINISHED'
|
||||||
@ -15,6 +18,7 @@ const SET_CUSTOM_GAS_LIMIT = 'metamask/gas/SET_CUSTOM_GAS_LIMIT'
|
|||||||
const SET_CUSTOM_GAS_PRICE = 'metamask/gas/SET_CUSTOM_GAS_PRICE'
|
const SET_CUSTOM_GAS_PRICE = 'metamask/gas/SET_CUSTOM_GAS_PRICE'
|
||||||
const SET_CUSTOM_GAS_TOTAL = 'metamask/gas/SET_CUSTOM_GAS_TOTAL'
|
const SET_CUSTOM_GAS_TOTAL = 'metamask/gas/SET_CUSTOM_GAS_TOTAL'
|
||||||
const SET_PRICE_AND_TIME_ESTIMATES = 'metamask/gas/SET_PRICE_AND_TIME_ESTIMATES'
|
const SET_PRICE_AND_TIME_ESTIMATES = 'metamask/gas/SET_PRICE_AND_TIME_ESTIMATES'
|
||||||
|
const SET_API_ESTIMATES_LAST_RETRIEVED = 'metamask/gas/SET_API_ESTIMATES_LAST_RETRIEVED'
|
||||||
|
|
||||||
// TODO: determine if this approach to initState is consistent with conventional ducks pattern
|
// TODO: determine if this approach to initState is consistent with conventional ducks pattern
|
||||||
const initState = {
|
const initState = {
|
||||||
@ -38,6 +42,7 @@ const initState = {
|
|||||||
basicEstimateIsLoading: true,
|
basicEstimateIsLoading: true,
|
||||||
gasEstimatesLoading: true,
|
gasEstimatesLoading: true,
|
||||||
priceAndTimeEstimates: [],
|
priceAndTimeEstimates: [],
|
||||||
|
priceAndTimeEstimatesLastRetrieved: 0,
|
||||||
errors: {},
|
errors: {},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -108,6 +113,11 @@ export default function reducer ({ gas: gasState = initState }, action = {}) {
|
|||||||
...action.value,
|
...action.value,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
case SET_API_ESTIMATES_LAST_RETRIEVED:
|
||||||
|
return {
|
||||||
|
...newState,
|
||||||
|
priceAndTimeEstimatesLastRetrieved: action.value,
|
||||||
|
}
|
||||||
case RESET_CUSTOM_DATA:
|
case RESET_CUSTOM_DATA:
|
||||||
return {
|
return {
|
||||||
...newState,
|
...newState,
|
||||||
@ -192,34 +202,51 @@ export function fetchBasicGasEstimates () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function fetchGasEstimates (blockTime) {
|
export function fetchGasEstimates (blockTime) {
|
||||||
return (dispatch) => {
|
return (dispatch, getState) => {
|
||||||
|
const {
|
||||||
|
priceAndTimeEstimatesLastRetrieved,
|
||||||
|
priceAndTimeEstimates,
|
||||||
|
} = getState().gas
|
||||||
|
const timeLastRetrieved = priceAndTimeEstimatesLastRetrieved || loadLocalStorageData('GAS_API_ESTIMATES_LAST_RETRIEVED')
|
||||||
|
|
||||||
dispatch(gasEstimatesLoadingStarted())
|
dispatch(gasEstimatesLoadingStarted())
|
||||||
|
|
||||||
// TODO: uncomment code when live api is ready
|
const promiseToFetch = Date.now() - timeLastRetrieved > 75000
|
||||||
// return fetch('https://ethgasstation.info/json/predictTable.json', {
|
? fetch('https://ethgasstation.info/json/predictTable.json', {
|
||||||
// 'headers': {},
|
'headers': {},
|
||||||
// 'referrer': 'http://ethgasstation.info/json/',
|
'referrer': 'http://ethgasstation.info/json/',
|
||||||
// 'referrerPolicy': 'no-referrer-when-downgrade',
|
'referrerPolicy': 'no-referrer-when-downgrade',
|
||||||
// 'body': null,
|
'body': null,
|
||||||
// 'method': 'GET',
|
'method': 'GET',
|
||||||
// 'mode': 'cors'}
|
'mode': 'cors'}
|
||||||
// )
|
)
|
||||||
return new Promise(resolve => {
|
.then(r => r.json())
|
||||||
resolve(mockGasEstimateData)
|
.then(r => {
|
||||||
})
|
const estimatedPricesAndTimes = r.map(({ expectedTime, expectedWait, gasprice }) => ({ expectedTime, expectedWait, gasprice }))
|
||||||
// .then(r => r.json())
|
const estimatedTimeWithUniquePrices = uniqBy(({ expectedTime }) => expectedTime, estimatedPricesAndTimes)
|
||||||
.then(r => {
|
const timeMappedToSeconds = estimatedTimeWithUniquePrices.map(({ expectedWait, gasprice }) => {
|
||||||
const estimatedPricesAndTimes = r.map(({ expectedTime, expectedWait, gasprice }) => ({ expectedTime, expectedWait, gasprice }))
|
const expectedTime = (new BigNumber(expectedWait)).times(Number(blockTime), 10).toString(10)
|
||||||
const estimatedTimeWithUniquePrices = uniqBy(({ expectedTime }) => expectedTime, estimatedPricesAndTimes)
|
return {
|
||||||
const timeMappedToSeconds = estimatedTimeWithUniquePrices.map(({ expectedWait, gasprice }) => {
|
expectedTime,
|
||||||
const expectedTime = (new BigNumber(expectedWait)).times(Number(blockTime), 10).toString(10)
|
expectedWait,
|
||||||
return {
|
gasprice,
|
||||||
expectedTime,
|
}
|
||||||
expectedWait,
|
})
|
||||||
gasprice,
|
|
||||||
}
|
const timeRetrieved = Date.now()
|
||||||
|
dispatch(setApiEstimatesLastRetrieved(timeRetrieved))
|
||||||
|
saveLocalStorageData(timeRetrieved, 'GAS_API_ESTIMATES_LAST_RETRIEVED')
|
||||||
|
saveLocalStorageData(timeMappedToSeconds.slice(1), 'GAS_API_ESTIMATES')
|
||||||
|
|
||||||
|
return timeMappedToSeconds.slice(1)
|
||||||
})
|
})
|
||||||
dispatch(setPricesAndTimeEstimates(timeMappedToSeconds.slice(1)))
|
: Promise.resolve(priceAndTimeEstimates.length
|
||||||
|
? priceAndTimeEstimates
|
||||||
|
: loadLocalStorageData('GAS_API_ESTIMATES')
|
||||||
|
)
|
||||||
|
|
||||||
|
return promiseToFetch.then(estimates => {
|
||||||
|
dispatch(setPricesAndTimeEstimates(estimates))
|
||||||
dispatch(gasEstimatesLoadingFinished())
|
dispatch(gasEstimatesLoadingFinished())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -267,6 +294,13 @@ export function setCustomGasErrors (newErrors) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function setApiEstimatesLastRetrieved (retrievalTime) {
|
||||||
|
return {
|
||||||
|
type: SET_API_ESTIMATES_LAST_RETRIEVED,
|
||||||
|
value: retrievalTime,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export function resetCustomGasState () {
|
export function resetCustomGasState () {
|
||||||
return { type: RESET_CUSTOM_GAS_STATE }
|
return { type: RESET_CUSTOM_GAS_STATE }
|
||||||
}
|
}
|
||||||
|
@ -14,33 +14,52 @@ import GasReducer, {
|
|||||||
gasEstimatesLoadingStarted,
|
gasEstimatesLoadingStarted,
|
||||||
gasEstimatesLoadingFinished,
|
gasEstimatesLoadingFinished,
|
||||||
setPricesAndTimeEstimates,
|
setPricesAndTimeEstimates,
|
||||||
|
fetchGasEstimates,
|
||||||
|
setApiEstimatesLastRetrieved,
|
||||||
} from '../gas.duck.js'
|
} from '../gas.duck.js'
|
||||||
|
|
||||||
describe('Gas Duck', () => {
|
describe('Gas Duck', () => {
|
||||||
let tempFetch
|
let tempFetch
|
||||||
const fetchStub = sinon.stub().returns(new Promise(resolve => resolve({
|
let tempDateNow
|
||||||
json: () => new Promise(resolve => resolve({
|
const mockEthGasApiResponse = {
|
||||||
average: 'mockAverage',
|
average: 'mockAverage',
|
||||||
avgWait: 'mockAvgWait',
|
avgWait: 'mockAvgWait',
|
||||||
block_time: 'mockBlock_time',
|
block_time: 'mockBlock_time',
|
||||||
blockNum: 'mockBlockNum',
|
blockNum: 'mockBlockNum',
|
||||||
fast: 'mockFast',
|
fast: 'mockFast',
|
||||||
fastest: 'mockFastest',
|
fastest: 'mockFastest',
|
||||||
fastestWait: 'mockFastestWait',
|
fastestWait: 'mockFastestWait',
|
||||||
fastWait: 'mockFastWait',
|
fastWait: 'mockFastWait',
|
||||||
safeLow: 'mockSafeLow',
|
safeLow: 'mockSafeLow',
|
||||||
safeLowWait: 'mockSafeLowWait',
|
safeLowWait: 'mockSafeLowWait',
|
||||||
speed: 'mockSpeed',
|
speed: 'mockSpeed',
|
||||||
})),
|
}
|
||||||
})))
|
const mockPredictTableResponse = [
|
||||||
|
{ expectedTime: 100, expectedWait: 10, gasprice: 1, somethingElse: 'foobar' },
|
||||||
|
{ expectedTime: 50, expectedWait: 5, gasprice: 2, somethingElse: 'foobar' },
|
||||||
|
{ expectedTime: 20, expectedWait: 4, gasprice: 4, somethingElse: 'foobar' },
|
||||||
|
{ expectedTime: 10, expectedWait: 2, gasprice: 10, somethingElse: 'foobar' },
|
||||||
|
{ expectedTime: 1, expectedWait: 0.5, gasprice: 20, somethingElse: 'foobar' },
|
||||||
|
]
|
||||||
|
const fetchStub = sinon.stub().callsFake((url) => new Promise(resolve => {
|
||||||
|
const dataToResolve = url.match(/ethgasAPI/)
|
||||||
|
? mockEthGasApiResponse
|
||||||
|
: mockPredictTableResponse
|
||||||
|
resolve({
|
||||||
|
json: () => new Promise(resolve => resolve(dataToResolve)),
|
||||||
|
})
|
||||||
|
}))
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
tempFetch = global.fetch
|
tempFetch = global.fetch
|
||||||
|
tempDateNow = global.Date.now
|
||||||
global.fetch = fetchStub
|
global.fetch = fetchStub
|
||||||
|
global.Date.now = () => 2000000
|
||||||
})
|
})
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
global.fetch = tempFetch
|
global.fetch = tempFetch
|
||||||
|
global.Date.now = tempDateNow
|
||||||
})
|
})
|
||||||
|
|
||||||
const mockState = {
|
const mockState = {
|
||||||
@ -70,6 +89,7 @@ describe('Gas Duck', () => {
|
|||||||
errors: {},
|
errors: {},
|
||||||
gasEstimatesLoading: true,
|
gasEstimatesLoading: true,
|
||||||
priceAndTimeEstimates: [],
|
priceAndTimeEstimates: [],
|
||||||
|
priceAndTimeEstimatesLastRetrieved: 0,
|
||||||
|
|
||||||
}
|
}
|
||||||
const BASIC_GAS_ESTIMATE_LOADING_FINISHED = 'metamask/gas/BASIC_GAS_ESTIMATE_LOADING_FINISHED'
|
const BASIC_GAS_ESTIMATE_LOADING_FINISHED = 'metamask/gas/BASIC_GAS_ESTIMATE_LOADING_FINISHED'
|
||||||
@ -83,6 +103,7 @@ describe('Gas Duck', () => {
|
|||||||
const SET_CUSTOM_GAS_PRICE = 'metamask/gas/SET_CUSTOM_GAS_PRICE'
|
const SET_CUSTOM_GAS_PRICE = 'metamask/gas/SET_CUSTOM_GAS_PRICE'
|
||||||
const SET_CUSTOM_GAS_TOTAL = 'metamask/gas/SET_CUSTOM_GAS_TOTAL'
|
const SET_CUSTOM_GAS_TOTAL = 'metamask/gas/SET_CUSTOM_GAS_TOTAL'
|
||||||
const SET_PRICE_AND_TIME_ESTIMATES = 'metamask/gas/SET_PRICE_AND_TIME_ESTIMATES'
|
const SET_PRICE_AND_TIME_ESTIMATES = 'metamask/gas/SET_PRICE_AND_TIME_ESTIMATES'
|
||||||
|
const SET_API_ESTIMATES_LAST_RETRIEVED = 'metamask/gas/SET_API_ESTIMATES_LAST_RETRIEVED'
|
||||||
|
|
||||||
describe('GasReducer()', () => {
|
describe('GasReducer()', () => {
|
||||||
it('should initialize state', () => {
|
it('should initialize state', () => {
|
||||||
@ -193,6 +214,16 @@ describe('Gas Duck', () => {
|
|||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('should set priceAndTimeEstimatesLastRetrieved when receivinga SET_API_ESTIMATES_LAST_RETRIEVED action', () => {
|
||||||
|
assert.deepEqual(
|
||||||
|
GasReducer(mockState, {
|
||||||
|
type: SET_API_ESTIMATES_LAST_RETRIEVED,
|
||||||
|
value: 1500000000000,
|
||||||
|
}),
|
||||||
|
Object.assign({ priceAndTimeEstimatesLastRetrieved: 1500000000000 }, mockState.gas)
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
it('should set errors when receiving a SET_CUSTOM_GAS_ERRORS action', () => {
|
it('should set errors when receiving a SET_CUSTOM_GAS_ERRORS action', () => {
|
||||||
assert.deepEqual(
|
assert.deepEqual(
|
||||||
GasReducer(mockState, {
|
GasReducer(mockState, {
|
||||||
@ -279,6 +310,75 @@ describe('Gas Duck', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('fetchGasEstimates', () => {
|
||||||
|
const mockDistpatch = sinon.spy()
|
||||||
|
it('should call fetch with the expected params', async () => {
|
||||||
|
global.fetch.resetHistory()
|
||||||
|
await fetchGasEstimates(5)(mockDistpatch, () => ({ gas: Object.assign(
|
||||||
|
{},
|
||||||
|
initState,
|
||||||
|
{ priceAndTimeEstimatesLastRetrieved: 1000000 }
|
||||||
|
) }))
|
||||||
|
assert.deepEqual(
|
||||||
|
mockDistpatch.getCall(0).args,
|
||||||
|
[{ type: GAS_ESTIMATE_LOADING_STARTED} ]
|
||||||
|
)
|
||||||
|
assert.deepEqual(
|
||||||
|
global.fetch.getCall(0).args,
|
||||||
|
[
|
||||||
|
'https://ethgasstation.info/json/predictTable.json',
|
||||||
|
{
|
||||||
|
'headers': {},
|
||||||
|
'referrer': 'http://ethgasstation.info/json/',
|
||||||
|
'referrerPolicy': 'no-referrer-when-downgrade',
|
||||||
|
'body': null,
|
||||||
|
'method': 'GET',
|
||||||
|
'mode': 'cors',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
mockDistpatch.getCall(1).args,
|
||||||
|
[{ type: SET_API_ESTIMATES_LAST_RETRIEVED, value: 2000000 }]
|
||||||
|
)
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
mockDistpatch.getCall(2).args,
|
||||||
|
[{
|
||||||
|
type: SET_PRICE_AND_TIME_ESTIMATES,
|
||||||
|
value: [
|
||||||
|
{
|
||||||
|
expectedTime: '25',
|
||||||
|
expectedWait: 5,
|
||||||
|
gasprice: 2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
expectedTime: '20',
|
||||||
|
expectedWait: 4,
|
||||||
|
gasprice: 4,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
expectedTime: '10',
|
||||||
|
expectedWait: 2,
|
||||||
|
gasprice: 10,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
expectedTime: '2.5',
|
||||||
|
expectedWait: 0.5,
|
||||||
|
gasprice: 20,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
|
||||||
|
}]
|
||||||
|
)
|
||||||
|
assert.deepEqual(
|
||||||
|
mockDistpatch.getCall(3).args,
|
||||||
|
[{ type: GAS_ESTIMATE_LOADING_FINISHED }]
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
describe('gasEstimatesLoadingStarted', () => {
|
describe('gasEstimatesLoadingStarted', () => {
|
||||||
it('should create the correct action', () => {
|
it('should create the correct action', () => {
|
||||||
assert.deepEqual(
|
assert.deepEqual(
|
||||||
@ -351,6 +451,15 @@ describe('Gas Duck', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('setApiEstimatesLastRetrieved', () => {
|
||||||
|
it('should create the correct action', () => {
|
||||||
|
assert.deepEqual(
|
||||||
|
setApiEstimatesLastRetrieved(1234),
|
||||||
|
{ type: SET_API_ESTIMATES_LAST_RETRIEVED, value: 1234 }
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
describe('resetCustomGasState', () => {
|
describe('resetCustomGasState', () => {
|
||||||
it('should create the correct action', () => {
|
it('should create the correct action', () => {
|
||||||
assert.deepEqual(
|
assert.deepEqual(
|
||||||
|
20
ui/lib/local-storage-helpers.js
Normal file
20
ui/lib/local-storage-helpers.js
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
export function loadLocalStorageData (itemKey) {
|
||||||
|
try {
|
||||||
|
const serializedData = localStorage.getItem(itemKey)
|
||||||
|
if (serializedData === null) {
|
||||||
|
return undefined
|
||||||
|
}
|
||||||
|
return JSON.parse(serializedData)
|
||||||
|
} catch (err) {
|
||||||
|
return undefined
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function saveLocalStorageData (data, itemKey) {
|
||||||
|
try {
|
||||||
|
const serializedData = JSON.stringify(data)
|
||||||
|
localStorage.setItem(itemKey, serializedData)
|
||||||
|
} catch (err) {
|
||||||
|
console.warn(err)
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user