mirror of
https://github.com/status-im/MyCrypto.git
synced 2025-01-28 03:44:58 +00:00
985ea0fb89
* [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
218 lines
5.7 KiB
TypeScript
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
|
|
);
|