2017-09-14 04:57:33 +02:00
|
|
|
const abi = require('human-standard-token-abi')
|
2016-04-14 00:28:44 +02:00
|
|
|
const ethUtil = require('ethereumjs-util')
|
2017-08-29 01:06:10 +02:00
|
|
|
const hexToBn = require('../../app/scripts/lib/hex-to-bn')
|
2017-08-23 01:16:56 +02:00
|
|
|
const vreme = new (require('vreme'))()
|
|
|
|
|
2017-08-29 16:50:48 +02:00
|
|
|
const MIN_GAS_PRICE_GWEI_BN = new ethUtil.BN(1)
|
|
|
|
const GWEI_FACTOR = new ethUtil.BN(1e9)
|
|
|
|
const MIN_GAS_PRICE_BN = MIN_GAS_PRICE_GWEI_BN.mul(GWEI_FACTOR)
|
|
|
|
|
2017-08-23 01:16:56 +02:00
|
|
|
// formatData :: ( date: <Unix Timestamp> ) -> String
|
|
|
|
function formatDate (date) {
|
2018-08-03 05:20:15 +02:00
|
|
|
return vreme.format(new Date(date), '3/16/2014 at 14:30')
|
2017-08-23 01:16:56 +02:00
|
|
|
}
|
2016-04-14 00:28:44 +02:00
|
|
|
|
|
|
|
var valueTable = {
|
2016-06-21 22:18:32 +02:00
|
|
|
wei: '1000000000000000000',
|
|
|
|
kwei: '1000000000000000',
|
|
|
|
mwei: '1000000000000',
|
|
|
|
gwei: '1000000000',
|
2016-04-14 00:28:44 +02:00
|
|
|
szabo: '1000000',
|
2016-06-21 22:18:32 +02:00
|
|
|
finney: '1000',
|
2016-04-14 00:28:44 +02:00
|
|
|
ether: '1',
|
2016-06-21 22:18:32 +02:00
|
|
|
kether: '0.001',
|
|
|
|
mether: '0.000001',
|
|
|
|
gether: '0.000000001',
|
|
|
|
tether: '0.000000000001',
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
var bnTable = {}
|
|
|
|
for (var currency in valueTable) {
|
|
|
|
bnTable[currency] = new ethUtil.BN(valueTable[currency], 10)
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
valuesFor: valuesFor,
|
|
|
|
addressSummary: addressSummary,
|
2016-07-07 02:58:46 +02:00
|
|
|
miniAddressSummary: miniAddressSummary,
|
2016-05-19 21:06:45 +02:00
|
|
|
isAllOneCase: isAllOneCase,
|
|
|
|
isValidAddress: isValidAddress,
|
2018-06-14 15:55:55 +02:00
|
|
|
isValidENSAddress,
|
2016-04-14 00:28:44 +02:00
|
|
|
numericBalance: numericBalance,
|
2016-05-18 22:41:08 +02:00
|
|
|
parseBalance: parseBalance,
|
2016-04-14 00:28:44 +02:00
|
|
|
formatBalance: formatBalance,
|
2016-07-07 21:39:07 +02:00
|
|
|
generateBalanceObject: generateBalanceObject,
|
2016-04-14 00:28:44 +02:00
|
|
|
dataSize: dataSize,
|
|
|
|
readableDate: readableDate,
|
|
|
|
normalizeToWei: normalizeToWei,
|
2016-05-19 23:46:50 +02:00
|
|
|
normalizeEthStringToWei: normalizeEthStringToWei,
|
2016-04-20 03:56:22 +02:00
|
|
|
normalizeNumberToWei: normalizeNumberToWei,
|
2016-04-14 00:28:44 +02:00
|
|
|
valueTable: valueTable,
|
|
|
|
bnTable: bnTable,
|
2016-10-19 23:33:30 +02:00
|
|
|
isHex: isHex,
|
2017-08-23 01:16:56 +02:00
|
|
|
formatDate,
|
2017-08-29 01:06:10 +02:00
|
|
|
bnMultiplyByFraction,
|
|
|
|
getTxFeeBn,
|
2017-09-13 10:25:39 +02:00
|
|
|
shortenBalance,
|
2017-09-14 04:57:33 +02:00
|
|
|
getContractAtAddress,
|
2017-09-12 01:22:20 +02:00
|
|
|
exportAsFile: exportAsFile,
|
2017-09-12 21:23:21 +02:00
|
|
|
isInvalidChecksumAddress,
|
2017-09-20 19:37:12 +02:00
|
|
|
allNull,
|
2018-01-16 21:14:03 +01:00
|
|
|
getTokenAddressFromTokenObject,
|
2018-04-17 21:57:55 +02:00
|
|
|
checksumAddress,
|
2018-07-20 23:18:50 +02:00
|
|
|
addressSlicer,
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function valuesFor (obj) {
|
2016-04-14 00:28:44 +02:00
|
|
|
if (!obj) return []
|
|
|
|
return Object.keys(obj)
|
2016-06-21 22:18:32 +02:00
|
|
|
.map(function (key) { return obj[key] })
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
|
2016-07-07 02:58:46 +02:00
|
|
|
function addressSummary (address, firstSegLength = 10, lastSegLength = 4, includeHex = true) {
|
|
|
|
if (!address) return ''
|
2018-04-17 21:57:55 +02:00
|
|
|
let checked = checksumAddress(address)
|
2016-07-07 02:58:46 +02:00
|
|
|
if (!includeHex) {
|
|
|
|
checked = ethUtil.stripHexPrefix(checked)
|
|
|
|
}
|
|
|
|
return checked ? checked.slice(0, firstSegLength) + '...' + checked.slice(checked.length - lastSegLength) : '...'
|
|
|
|
}
|
|
|
|
|
|
|
|
function miniAddressSummary (address) {
|
2016-05-20 01:14:16 +02:00
|
|
|
if (!address) return ''
|
2018-04-17 21:57:55 +02:00
|
|
|
var checked = checksumAddress(address)
|
2016-07-07 02:58:46 +02:00
|
|
|
return checked ? checked.slice(0, 4) + '...' + checked.slice(-4) : '...'
|
2016-05-19 21:06:45 +02:00
|
|
|
}
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function isValidAddress (address) {
|
2016-05-19 21:06:45 +02:00
|
|
|
var prefixed = ethUtil.addHexPrefix(address)
|
2017-06-05 21:09:19 +02:00
|
|
|
if (address === '0x0000000000000000000000000000000000000000') return false
|
2016-06-03 03:42:09 +02:00
|
|
|
return (isAllOneCase(prefixed) && ethUtil.isValidAddress(prefixed)) || ethUtil.isValidChecksumAddress(prefixed)
|
2016-05-19 21:06:45 +02:00
|
|
|
}
|
|
|
|
|
2018-06-14 15:55:55 +02:00
|
|
|
function isValidENSAddress (address) {
|
|
|
|
return address.match(/^.{7,}\.(eth|test)$/)
|
|
|
|
}
|
|
|
|
|
2017-09-12 21:23:21 +02:00
|
|
|
function isInvalidChecksumAddress (address) {
|
|
|
|
var prefixed = ethUtil.addHexPrefix(address)
|
|
|
|
if (address === '0x0000000000000000000000000000000000000000') return false
|
2017-09-12 21:52:13 +02:00
|
|
|
return !isAllOneCase(prefixed) && !ethUtil.isValidChecksumAddress(prefixed) && ethUtil.isValidAddress(prefixed)
|
2017-09-12 21:23:21 +02:00
|
|
|
}
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function isAllOneCase (address) {
|
2016-05-20 01:14:16 +02:00
|
|
|
if (!address) return true
|
2016-05-19 21:06:45 +02:00
|
|
|
var lower = address.toLowerCase()
|
|
|
|
var upper = address.toUpperCase()
|
|
|
|
return address === lower || address === upper
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Takes wei Hex, returns wei BN, even if input is null
|
2016-06-21 22:18:32 +02:00
|
|
|
function numericBalance (balance) {
|
2016-04-14 00:28:44 +02:00
|
|
|
if (!balance) return new ethUtil.BN(0, 16)
|
|
|
|
var stripped = ethUtil.stripHexPrefix(balance)
|
|
|
|
return new ethUtil.BN(stripped, 16)
|
|
|
|
}
|
|
|
|
|
2016-05-18 22:41:08 +02:00
|
|
|
// Takes hex, returns [beforeDecimal, afterDecimal]
|
2016-06-21 22:18:32 +02:00
|
|
|
function parseBalance (balance) {
|
2016-06-21 22:56:04 +02:00
|
|
|
var beforeDecimal, afterDecimal
|
2016-07-12 01:01:52 +02:00
|
|
|
const wei = numericBalance(balance)
|
|
|
|
var weiString = wei.toString()
|
2016-06-21 22:56:04 +02:00
|
|
|
const trailingZeros = /0+$/
|
2016-06-21 00:46:29 +02:00
|
|
|
|
2016-07-12 01:01:52 +02:00
|
|
|
beforeDecimal = weiString.length > 18 ? weiString.slice(0, weiString.length - 18) : '0'
|
2016-06-21 22:56:04 +02:00
|
|
|
afterDecimal = ('000000000000000000' + wei).slice(-18).replace(trailingZeros, '')
|
|
|
|
if (afterDecimal === '') { afterDecimal = '0' }
|
2016-05-18 22:41:08 +02:00
|
|
|
return [beforeDecimal, afterDecimal]
|
|
|
|
}
|
|
|
|
|
2016-07-07 06:32:36 +02:00
|
|
|
// Takes wei hex, returns an object with three properties.
|
|
|
|
// Its "formatted" property is what we generally use to render values.
|
2018-10-26 10:26:43 +02:00
|
|
|
function formatBalance (balance, decimalsToKeep, needsParse = true, ticker = 'ETH') {
|
2016-08-20 02:18:11 +02:00
|
|
|
var parsed = needsParse ? parseBalance(balance) : balance.split('.')
|
2016-05-18 22:41:08 +02:00
|
|
|
var beforeDecimal = parsed[0]
|
|
|
|
var afterDecimal = parsed[1]
|
2016-07-18 21:06:27 +02:00
|
|
|
var formatted = 'None'
|
2016-07-07 20:20:02 +02:00
|
|
|
if (decimalsToKeep === undefined) {
|
|
|
|
if (beforeDecimal === '0') {
|
|
|
|
if (afterDecimal !== '0') {
|
|
|
|
var sigFigs = afterDecimal.match(/^0*(.{2})/) // default: grabs 2 most significant digits
|
|
|
|
if (sigFigs) { afterDecimal = sigFigs[0] }
|
2018-10-26 10:26:43 +02:00
|
|
|
formatted = '0.' + afterDecimal + ` ${ticker}`
|
2016-07-07 20:20:02 +02:00
|
|
|
}
|
|
|
|
} else {
|
2018-10-26 10:26:43 +02:00
|
|
|
formatted = beforeDecimal + '.' + afterDecimal.slice(0, 3) + ` ${ticker}`
|
2016-05-26 03:29:03 +02:00
|
|
|
}
|
2016-06-21 22:18:32 +02:00
|
|
|
} else {
|
2016-07-07 20:20:02 +02:00
|
|
|
afterDecimal += Array(decimalsToKeep).join('0')
|
2018-10-26 10:26:43 +02:00
|
|
|
formatted = beforeDecimal + '.' + afterDecimal.slice(0, decimalsToKeep) + ` ${ticker}`
|
2016-05-26 03:29:03 +02:00
|
|
|
}
|
2016-07-07 20:20:02 +02:00
|
|
|
return formatted
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
|
2016-07-12 01:31:12 +02:00
|
|
|
|
2016-07-12 02:44:00 +02:00
|
|
|
function generateBalanceObject (formattedBalance, decimalsToKeep = 1) {
|
2016-07-07 21:07:47 +02:00
|
|
|
var balance = formattedBalance.split(' ')[0]
|
|
|
|
var label = formattedBalance.split(' ')[1]
|
|
|
|
var beforeDecimal = balance.split('.')[0]
|
|
|
|
var afterDecimal = balance.split('.')[1]
|
2016-07-12 02:44:00 +02:00
|
|
|
var shortBalance = shortenBalance(balance, decimalsToKeep)
|
2016-07-07 21:07:47 +02:00
|
|
|
|
2016-07-14 00:46:36 +02:00
|
|
|
if (beforeDecimal === '0' && afterDecimal.substr(0, 5) === '00000') {
|
2016-07-18 18:38:27 +02:00
|
|
|
// eslint-disable-next-line eqeqeq
|
2016-07-18 17:54:06 +02:00
|
|
|
if (afterDecimal == 0) {
|
|
|
|
balance = '0'
|
|
|
|
} else {
|
|
|
|
balance = '<1.0e-5'
|
|
|
|
}
|
2016-07-14 00:46:36 +02:00
|
|
|
} else if (beforeDecimal !== '0') {
|
|
|
|
balance = `${beforeDecimal}.${afterDecimal.slice(0, decimalsToKeep)}`
|
|
|
|
}
|
2016-07-07 21:07:47 +02:00
|
|
|
|
2016-07-12 01:31:12 +02:00
|
|
|
return { balance, label, shortBalance }
|
|
|
|
}
|
|
|
|
|
2016-07-12 02:44:00 +02:00
|
|
|
function shortenBalance (balance, decimalsToKeep = 1) {
|
2016-07-12 01:31:12 +02:00
|
|
|
var truncatedValue
|
|
|
|
var convertedBalance = parseFloat(balance)
|
|
|
|
if (convertedBalance > 1000000) {
|
2016-07-12 02:44:00 +02:00
|
|
|
truncatedValue = (balance / 1000000).toFixed(decimalsToKeep)
|
2016-08-19 00:59:37 +02:00
|
|
|
return `${truncatedValue}m`
|
2016-07-12 01:31:12 +02:00
|
|
|
} else if (convertedBalance > 1000) {
|
2016-07-12 02:44:00 +02:00
|
|
|
truncatedValue = (balance / 1000).toFixed(decimalsToKeep)
|
2016-08-19 00:59:37 +02:00
|
|
|
return `${truncatedValue}k`
|
2016-08-16 19:48:31 +02:00
|
|
|
} else if (convertedBalance === 0) {
|
2016-08-08 02:23:35 +02:00
|
|
|
return '0'
|
2016-08-19 00:59:37 +02:00
|
|
|
} else if (convertedBalance < 0.001) {
|
|
|
|
return '<0.001'
|
2016-08-19 01:01:40 +02:00
|
|
|
} else if (convertedBalance < 1) {
|
2016-08-19 00:59:37 +02:00
|
|
|
var stringBalance = convertedBalance.toString()
|
|
|
|
if (stringBalance.split('.')[1].length > 3) {
|
2016-08-19 01:01:40 +02:00
|
|
|
return convertedBalance.toFixed(3)
|
2016-08-19 00:59:37 +02:00
|
|
|
} else {
|
|
|
|
return stringBalance
|
|
|
|
}
|
2016-07-12 01:31:12 +02:00
|
|
|
} else {
|
2016-07-14 00:46:36 +02:00
|
|
|
return convertedBalance.toFixed(decimalsToKeep)
|
2016-07-12 01:31:12 +02:00
|
|
|
}
|
2016-07-07 21:07:47 +02:00
|
|
|
}
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function dataSize (data) {
|
2016-04-14 00:28:44 +02:00
|
|
|
var size = data ? ethUtil.stripHexPrefix(data).length : 0
|
2016-06-21 22:18:32 +02:00
|
|
|
return size + ' bytes'
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Takes a BN and an ethereum currency name,
|
|
|
|
// returns a BN in wei
|
2016-06-21 22:18:32 +02:00
|
|
|
function normalizeToWei (amount, currency) {
|
2016-04-14 00:28:44 +02:00
|
|
|
try {
|
2016-04-20 03:56:22 +02:00
|
|
|
return amount.mul(bnTable.wei).div(bnTable[currency])
|
2016-04-14 00:28:44 +02:00
|
|
|
} catch (e) {}
|
|
|
|
return amount
|
|
|
|
}
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function normalizeEthStringToWei (str) {
|
2016-05-19 23:46:50 +02:00
|
|
|
const parts = str.split('.')
|
|
|
|
let eth = new ethUtil.BN(parts[0], 10).mul(bnTable.wei)
|
|
|
|
if (parts[1]) {
|
|
|
|
var decimal = parts[1]
|
2016-06-21 22:18:32 +02:00
|
|
|
while (decimal.length < 18) {
|
2016-05-19 23:46:50 +02:00
|
|
|
decimal += '0'
|
|
|
|
}
|
2017-12-28 02:31:38 +01:00
|
|
|
if (decimal.length > 18) {
|
|
|
|
decimal = decimal.slice(0, 18)
|
|
|
|
}
|
2016-05-19 23:46:50 +02:00
|
|
|
const decimalBN = new ethUtil.BN(decimal, 10)
|
|
|
|
eth = eth.add(decimalBN)
|
|
|
|
}
|
|
|
|
return eth
|
|
|
|
}
|
|
|
|
|
2016-05-19 23:21:35 +02:00
|
|
|
var multiple = new ethUtil.BN('10000', 10)
|
2016-06-21 22:18:32 +02:00
|
|
|
function normalizeNumberToWei (n, currency) {
|
2016-05-19 23:21:35 +02:00
|
|
|
var enlarged = n * 10000
|
2016-04-20 03:56:22 +02:00
|
|
|
var amount = new ethUtil.BN(String(enlarged), 10)
|
|
|
|
return normalizeToWei(amount, currency).div(multiple)
|
|
|
|
}
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function readableDate (ms) {
|
2016-04-14 00:28:44 +02:00
|
|
|
var date = new Date(ms)
|
|
|
|
var month = date.getMonth()
|
|
|
|
var day = date.getDate()
|
|
|
|
var year = date.getFullYear()
|
|
|
|
var hours = date.getHours()
|
2016-06-21 22:18:32 +02:00
|
|
|
var minutes = '0' + date.getMinutes()
|
|
|
|
var seconds = '0' + date.getSeconds()
|
2016-04-14 00:28:44 +02:00
|
|
|
|
2016-06-21 22:56:04 +02:00
|
|
|
var dateStr = `${month}/${day}/${year}`
|
2016-04-14 00:28:44 +02:00
|
|
|
var time = `${hours}:${minutes.substr(-2)}:${seconds.substr(-2)}`
|
2016-06-21 22:56:04 +02:00
|
|
|
return `${dateStr} ${time}`
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
2016-10-19 23:33:30 +02:00
|
|
|
|
|
|
|
function isHex (str) {
|
2016-10-22 01:39:14 +02:00
|
|
|
return Boolean(str.match(/^(0x)?[0-9a-fA-F]+$/))
|
2016-10-19 23:33:30 +02:00
|
|
|
}
|
2017-09-12 01:22:20 +02:00
|
|
|
|
2017-08-29 01:06:10 +02:00
|
|
|
function bnMultiplyByFraction (targetBN, numerator, denominator) {
|
|
|
|
const numBN = new ethUtil.BN(numerator)
|
|
|
|
const denomBN = new ethUtil.BN(denominator)
|
|
|
|
return targetBN.mul(numBN).div(denomBN)
|
|
|
|
}
|
|
|
|
|
|
|
|
function getTxFeeBn (gas, gasPrice = MIN_GAS_PRICE_BN.toString(16), blockGasLimit) {
|
|
|
|
const gasBn = hexToBn(gas)
|
|
|
|
const gasPriceBn = hexToBn(gasPrice)
|
|
|
|
const txFeeBn = gasBn.mul(gasPriceBn)
|
|
|
|
|
2017-08-29 16:50:48 +02:00
|
|
|
return txFeeBn.toString(16)
|
2017-08-29 01:06:10 +02:00
|
|
|
}
|
2017-09-14 04:57:33 +02:00
|
|
|
|
|
|
|
function getContractAtAddress (tokenAddress) {
|
|
|
|
return global.eth.contract(abi).at(tokenAddress)
|
|
|
|
}
|
2017-09-18 20:28:10 +02:00
|
|
|
|
2018-08-07 06:57:32 +02:00
|
|
|
function exportAsFile (filename, data, type = 'text/csv') {
|
2017-09-12 01:22:20 +02:00
|
|
|
// source: https://stackoverflow.com/a/33542499 by Ludovic Feltz
|
2018-08-07 06:57:32 +02:00
|
|
|
const blob = new Blob([data], {type})
|
2017-09-12 01:22:20 +02:00
|
|
|
if (window.navigator.msSaveOrOpenBlob) {
|
|
|
|
window.navigator.msSaveBlob(blob, filename)
|
|
|
|
} else {
|
|
|
|
const elem = window.document.createElement('a')
|
2018-04-05 01:42:54 +02:00
|
|
|
elem.target = '_blank'
|
2017-09-12 01:22:20 +02:00
|
|
|
elem.href = window.URL.createObjectURL(blob)
|
|
|
|
elem.download = filename
|
|
|
|
document.body.appendChild(elem)
|
|
|
|
elem.click()
|
|
|
|
document.body.removeChild(elem)
|
|
|
|
}
|
|
|
|
}
|
2017-09-20 19:37:12 +02:00
|
|
|
|
|
|
|
function allNull (obj) {
|
|
|
|
return Object.entries(obj).every(([key, value]) => value === null)
|
|
|
|
}
|
2018-01-16 21:14:03 +01:00
|
|
|
|
|
|
|
function getTokenAddressFromTokenObject (token) {
|
|
|
|
return Object.values(token)[0].address.toLowerCase()
|
|
|
|
}
|
2018-04-17 21:57:55 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Safely checksumms a potentially-null address
|
2018-05-05 17:11:53 +02:00
|
|
|
*
|
2018-04-17 21:57:55 +02:00
|
|
|
* @param {String} [address] - address to checksum
|
|
|
|
* @returns {String} - checksummed address
|
|
|
|
*/
|
|
|
|
function checksumAddress (address) {
|
|
|
|
return address ? ethUtil.toChecksumAddress(address) : ''
|
|
|
|
}
|
2018-07-20 23:18:50 +02:00
|
|
|
|
|
|
|
function addressSlicer (address = '') {
|
|
|
|
if (address.length < 11) {
|
|
|
|
return address
|
|
|
|
}
|
|
|
|
|
|
|
|
return `${address.slice(0, 6)}...${address.slice(-4)}`
|
|
|
|
}
|