Daniel Kmak 985ea0fb89 Ethereum Alarm Clock Integration (#1343)
* [FEATURE] Initial EAC integration.

* Title and explanation

* [FEATURE] Move the Schedule Payment to the main tab.

* [FEATURE] TimeBounty slider.

* [FEATURE] Move to main menu.

* [FEATURE] Redirection to the DApp for details.

* [FEATURE] Timestamp scheduling

* Scheduling: Basic date and time widget

* Linting fixes

* Moved the datetime field to new tab

* Fixed push errors

* Added missing specs

* Undid unintentional UI change

* Fixed some failing tests

* Ignore datetime parameter when checking if a transaction is full

* Added a date selector widget and renamed ScheduleTimestamp to ScheduleDate

* Marked componentDidMount

* Initialized Pikaday

* Revert "Initialized Pikaday"

This reverts commit 4e5bf5b2b882f236f5977400abf9b7092cbd1592.

* Revert "Marked componentDidMount"

This reverts commit 85d52192ac58f4b6ca9219e702f7390cd27e582f.

* Revert "Added a date selector widget and renamed ScheduleTimestamp to ScheduleDate"

This reverts commit aaad0ac9b565a78d1bfc631754160919fd38a59b.

* Converted the date picker into a datetime picker

* Added decent styling to the datetimepicker

* Added validation to the datetime picker

* Fixed prepush errors for scheduling timestamp

* Adjusted validation logic scheduling timestamp

* [FEATURE] Move scheduling to main tab.

* [FEATURE] Timezone selector

* [FEATURE] Scheduling: Timezone selector

* Removed zombie files

* [FEATURE] Reimplement Time Bounty.

* [FEATURE] Time/block selector

* [FEATURE] Add Window Size field.

* [FEATURE] Time/block switch functionality

* Implemented time/block switcher fuctionality

* [FEATURE] Add Schedule Gas Price field.

* [FEATURE] Scheduling toggle

* [FEATURE] Add basic styling and network check.

* [FEATURE] Add Schedule Gas Limit field

* [FEATURE] "Scheduled" button styling

* Reordered, renamed and centered scheduling elements

* Added the toggle button styling

* Class -> ClassName

* [FEATURE] Add Deposit field

*  [FEATURE] Move scheduling code into one directory

* [FIX] Scheduling responsiveness

* [FIX] Datetime picker not working on md screens

* [FEATURE] Timestamp Scheduling basic functionality

* [FIX] Fix data serialization.

* [FEATURE] Timezone inclusion

* [FEATURE] Add ChronoLogic logo.

* [FEATURE] Add link to image.

* [FIX] Update CSS of logo.

* [FEATURE] Default Window Size

* [FEATURE] Modified Help component to enable acting as a tooltip

* [FEATURE] Call contract to validate scheduling params

* [FIX] Change moment import to fix tests

* [FEATURE] Gas estimation for scheduling

* [FEATURE] Additional validation

* [FEATURE] UI changes and descriptions

* [FEATURE] Add tooltip to window and fix fee display.

* [FIX] Fix ethereumjs-abi dependency.

* [FEATURE] Hide scheduling when sending tokens.

* [FIX] Improved styling datetime picker

* [FEATURE] Add Redux state for scheduling

* [FEATURE] Create Toggle component, Share code between components

* [FEATURE] Use Tooltip component for help.

* [FEATURE] Better datetime picker

* [FEATURE] Remove fee

* Trigger mycryptobuild

* [FIX] Timestamp scheduling - Validation match

* [FIX] EAC integration touchups

* [FIX] Code review fixes

* [FIX] Window Size type

* [FIX] Type fixes.

* [FIX] Make tooltips only show on icons + resposiveness fixes

* [FIX] Break tooltips into more lines

* [FIX] Remove unnecessary code.

* [FIX] Remove unnecessary code.

* [FIX] Remove unnecessary types declaration.

* [FIX] Fee class names
2018-04-14 17:21:33 -05:00

218 lines
5.7 KiB
TypeScript

import BN from 'bn.js';
import abi from 'ethereumjs-abi';
import { toWei, Units, gasPriceToBase, Address, Wei } from '../units';
import { toBuffer } from 'ethereumjs-util';
import RequestFactory from './contracts/RequestFactory';
import { ICurrentValue } from 'selectors/transaction';
const TIME_BOUNTY_MIN = Wei('1');
export const EAC_SCHEDULING_CONFIG = {
DAPP_ADDRESS: 'https://app.chronologic.network',
SCHEDULE_GAS_LIMIT_FALLBACK: Wei('21000'),
SCHEDULE_GAS_PRICE_FALLBACK: 20, // Gwei
FEE: Wei('0'),
FUTURE_EXECUTION_COST: Wei('180000'),
SCHEDULING_GAS_LIMIT: Wei('1500000'),
WINDOW_SIZE_DEFAULT_TIME: 10,
WINDOW_SIZE_DEFAULT_BLOCK: 90,
TIME_BOUNTY_MIN,
TIME_BOUNTY_DEFAULT: TIME_BOUNTY_MIN,
TIME_BOUNTY_MAX: toWei('900', Units.ether.length - 1), // 900 ETH
SCHEDULE_TIMESTAMP_FORMAT: 'YYYY-MM-DD HH:mm:ss',
DEFAULT_SCHEDULING_METHOD: 'time',
ALLOW_SCHEDULING_MIN_AFTER_NOW: 5
};
export const EAC_ADDRESSES = {
KOVAN: {
blockScheduler: '0x1afc19a7e642761ba2b55d2a45b32c7ef08269d1',
requestFactory: '0x496e2b6089bde77293a994469b08e9f266d87adb',
timestampScheduler: '0xc6370807f0164bdf10a66c08d0dab1028dbe80a3'
}
};
export const calcEACFutureExecutionCost = (
callGas: Wei,
callGasPrice: Wei,
timeBounty: Wei | null
) => {
const totalGas = callGas.add(EAC_SCHEDULING_CONFIG.FUTURE_EXECUTION_COST);
if (!timeBounty) {
timeBounty = EAC_SCHEDULING_CONFIG.TIME_BOUNTY_MIN;
}
return timeBounty.add(EAC_SCHEDULING_CONFIG.FEE).add(totalGas.mul(callGasPrice));
};
export const calcEACEndowment = (
callGas: Wei | null,
callValue: Wei | null,
callGasPrice: Wei | null,
timeBounty: Wei | null
) => {
callValue = callValue || Wei('0');
timeBounty = timeBounty || Wei('0');
return callValue.add(
calcEACFutureExecutionCost(
callGas || EAC_SCHEDULING_CONFIG.SCHEDULE_GAS_LIMIT_FALLBACK,
callGasPrice || gasPriceToBase(EAC_SCHEDULING_CONFIG.SCHEDULE_GAS_PRICE_FALLBACK),
timeBounty
)
);
};
export const calcEACTotalCost = (
callGas: Wei,
gasPrice: Wei,
callGasPrice: Wei | null,
timeBounty: Wei | null
) => {
if (!callGasPrice) {
callGasPrice = gasPriceToBase(EAC_SCHEDULING_CONFIG.SCHEDULE_GAS_PRICE_FALLBACK);
}
const deployCost = gasPrice.mul(EAC_SCHEDULING_CONFIG.SCHEDULING_GAS_LIMIT);
const futureExecutionCost = calcEACFutureExecutionCost(callGas, callGasPrice, timeBounty);
return deployCost.add(futureExecutionCost);
};
export const getScheduleData = (
toAddress: string,
callData: string | Buffer = '',
callGas: Wei | null,
callValue: Wei | null,
windowSize: BN | null,
windowStart: number,
callGasPrice: Wei | null,
timeBounty: Wei | null,
requiredDeposit: Wei | null
) => {
if (!requiredDeposit || requiredDeposit.lt(Wei('0'))) {
requiredDeposit = Wei('0');
}
if (typeof callData === 'string') {
callData = toBuffer(callData);
}
/*
* Checks if any of these values are null or invalid
* due to an user input.
*/
if (
!callValue ||
!callGas ||
!callGasPrice ||
!windowStart ||
!windowSize ||
!timeBounty ||
timeBounty.lt(Wei('0')) ||
callGasPrice.lt(Wei('0')) ||
windowSize.lt(new BN(0)) ||
windowSize.bitLength() > 256
) {
return;
}
return abi.simpleEncode('schedule(address,bytes,uint[8]):(address)', toAddress, callData, [
callGas,
callValue,
windowSize,
windowStart,
callGasPrice,
EAC_SCHEDULING_CONFIG.FEE,
timeBounty,
requiredDeposit
]);
};
enum SchedulingParamsError {
InsufficientEndowment,
ReservedWindowBiggerThanExecutionWindow,
InvalidTemporalUnit,
ExecutionWindowTooSoon,
CallGasTooHigh,
EmptyToAddress
}
export const parseSchedulingParametersValidity = (isValid: boolean[]) => {
const errorsIndexMapping = [
SchedulingParamsError.InsufficientEndowment,
SchedulingParamsError.ReservedWindowBiggerThanExecutionWindow,
SchedulingParamsError.InvalidTemporalUnit,
SchedulingParamsError.ExecutionWindowTooSoon,
SchedulingParamsError.CallGasTooHigh,
SchedulingParamsError.EmptyToAddress
];
const errors: SchedulingParamsError[] = [];
isValid.forEach((boolIsTrue, index) => {
if (!boolIsTrue) {
errors.push(errorsIndexMapping[index]);
}
});
return errors;
};
export const getValidateRequestParamsData = (
toAddress: string,
callData = '',
callGas: Wei,
callValue: ICurrentValue['value'],
windowSize: BN | null,
windowStart: number,
gasPrice: Wei,
timeBounty: Wei | null,
requiredDeposit: Wei,
isTimestamp: boolean,
endowment: Wei,
fromAddress: string
): string => {
windowSize = windowSize || new BN(0);
timeBounty = timeBounty || Wei('0');
const temporalUnit = isTimestamp ? 2 : 1;
const freezePeriod = isTimestamp ? 3 * 60 : 10; // 3 minutes or 10 blocks
const reservedWindowSize = isTimestamp ? 5 * 60 : 16; // 5 minutes or 16 blocks
const claimWindowSize = isTimestamp ? 60 * 60 : 255; // 60 minutes or 255 blocks
const feeRecipient = '0x0'; // stub
return RequestFactory.validateRequestParams.encodeInput({
_addressArgs: [fromAddress, feeRecipient, toAddress],
_uintArgs: [
EAC_SCHEDULING_CONFIG.FEE,
timeBounty,
claimWindowSize,
freezePeriod,
reservedWindowSize,
temporalUnit,
windowSize,
windowStart,
callGas,
callValue,
gasPrice,
requiredDeposit
],
_callData: callData,
_endowment: endowment
});
};
export const getTXDetailsCheckURL = (txHash: string) => {
return `${EAC_SCHEDULING_CONFIG.DAPP_ADDRESS}/awaiting/scheduler/${txHash}`;
};
export const getSchedulerAddress = (scheduleType: string | null): Address =>
Address(
scheduleType === 'time'
? EAC_ADDRESSES.KOVAN.timestampScheduler
: EAC_ADDRESSES.KOVAN.blockScheduler
);