218 lines
5.7 KiB
TypeScript
Raw Normal View History

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-15 03:21:33 +05:00
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
);