Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Tasks: Implement new time lock modes #104

Merged
merged 9 commits into from
Oct 19, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions packages/authorizer/contracts/AuthorizedHelpers.sol
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,14 @@ contract AuthorizedHelpers {
r[2] = p3;
}

function authParams(uint256 p1, uint256 p2, uint256 p3, uint256 p4) internal pure returns (uint256[] memory r) {
r = new uint256[](4);
r[0] = p1;
r[1] = p2;
r[2] = p3;
r[3] = p4;
}

function authParams(address p1, address p2, uint256 p3, uint256 p4) internal pure returns (uint256[] memory r) {
r = new uint256[](4);
r[0] = uint256(uint160(p1));
Expand Down
2 changes: 1 addition & 1 deletion packages/authorizer/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@mimic-fi/v3-authorizer",
"version": "0.1.0",
"version": "0.1.1",
"license": "GPL-3.0",
"main": "dist/index.js",
"types": "dist/index.d.ts",
Expand Down
271 changes: 202 additions & 69 deletions packages/tasks/contracts/base/TimeLockedTask.sol
Original file line number Diff line number Diff line change
Expand Up @@ -14,32 +14,60 @@

pragma solidity ^0.8.3;

import '@quant-finance/solidity-datetime/contracts/DateTime.sol';
import '@mimic-fi/v3-authorizer/contracts/Authorized.sol';

import '../interfaces/base/ITimeLockedTask.sol';

/**
* @dev Time lock config for tasks. It allows limiting the frequency of a task.
*/
abstract contract TimeLockedTask is ITimeLockedTask, Authorized {
// Period in seconds that must pass after a task has been executed
uint256 public override timeLockDelay;
using DateTime for uint256;

uint256 private constant DAYS_28 = 60 * 60 * 24 * 28;

/**
* @dev Time-locks supports different frequency modes
* @param Seconds To indicate the execution must occur every certain number of seconds
* @param OnDay To indicate the execution must occur on day number from 1 to 28
* @param OnLastMonthDay To indicate the execution must occur on the last day of the month
* @param EverySomeMonths To indicate the execution must occur every certain number of months
*/
enum Mode {
Seconds,
OnDay,
OnLastMonthDay,
EverySomeMonths
}

// Time lock mode
Mode internal _mode;

// Time lock frequency
uint256 internal _frequency;

// Future timestamp in which the task can be executed
uint256 public override timeLockExpiration;
// Future timestamp since when the task can be executed
uint256 internal _allowedAt;

// Period in seconds during when a time-locked task can be executed right after it becomes executable
uint256 public override timeLockExecutionPeriod;
// Next future timestamp since when the task can be executed to be set, only used internally
uint256 internal _nextAllowedAt;

// Period in seconds during when a time-locked task can be executed since the allowed timestamp
uint256 internal _window;

/**
* @dev Time lock config params. Only used in the initializer.
* @param delay Period in seconds that must pass after a task has been executed
* @param nextExecutionTimestamp Next time when the task can be executed
* @param executionPeriod Period in seconds during when a time-locked task can be executed
* @param mode Time lock mode
* @param frequency Time lock frequency value
* @param allowedAt Time lock allowed date
* @param window Time lock execution window
*/
struct TimeLockConfig {
uint256 delay;
uint256 nextExecutionTimestamp;
uint256 executionPeriod;
uint8 mode;
uint256 frequency;
uint256 allowedAt;
uint256 window;
}

/**
Expand All @@ -55,100 +83,205 @@ abstract contract TimeLockedTask is ITimeLockedTask, Authorized {
* @param config Time locked task config
*/
function __TimeLockedTask_init_unchained(TimeLockConfig memory config) internal onlyInitializing {
_setTimeLockDelay(config.delay);
_setTimeLockExpiration(config.nextExecutionTimestamp);
_setTimeLockExecutionPeriod(config.executionPeriod);
_setTimeLock(config.mode, config.frequency, config.allowedAt, config.window);
}

/**
* @dev Sets the time-lock delay
* @param delay New delay to be set
* @dev Tells the time-lock related information
*/
function setTimeLockDelay(uint256 delay) external override authP(authParams(delay)) {
_setTimeLockDelay(delay);
function getTimeLock() external view returns (uint8 mode, uint256 frequency, uint256 allowedAt, uint256 window) {
return (uint8(_mode), _frequency, _allowedAt, _window);
}

/**
* @dev Sets the time-lock expiration timestamp
* @param expiration New expiration timestamp to be set
* @dev Sets a new time lock
*/
function setTimeLockExpiration(uint256 expiration) external override authP(authParams(expiration)) {
_setTimeLockExpiration(expiration);
function setTimeLock(uint8 mode, uint256 frequency, uint256 allowedAt, uint256 window)
external
override
authP(authParams(mode, frequency, allowedAt, window))
{
_setTimeLock(mode, frequency, allowedAt, window);
}

/**
* @dev Sets the time-lock execution period
* @param period New execution period to be set
* @dev Before time locked task hook
*/
function setTimeLockExecutionPeriod(uint256 period) external override authP(authParams(period)) {
_setTimeLockExecutionPeriod(period);
function _beforeTimeLockedTask(address, uint256) internal virtual {
// Load storage variables
Mode mode = _mode;
uint256 frequency = _frequency;
uint256 allowedAt = _allowedAt;
uint256 window = _window;

// First we check the current timestamp is not in the past
if (block.timestamp < allowedAt) revert TaskTimeLockActive(block.timestamp, allowedAt);

if (mode == Mode.Seconds) {
if (frequency == 0) return;

// If no window is set, the next allowed date is simply moved the number of seconds set as frequency.
// Otherwise, the offset must be validated and the next allowed date is set to the next period.
if (window == 0) _nextAllowedAt = block.timestamp + frequency;
else {
uint256 diff = block.timestamp - allowedAt;
uint256 periods = diff / frequency;
uint256 offset = diff - (periods * frequency);
if (offset > window) revert TaskTimeLockActive(block.timestamp, allowedAt);
_nextAllowedAt = allowedAt + ((periods + 1) * frequency);
}
} else {
uint256 day;
uint256 monthsToAdd;

if (mode == Mode.EverySomeMonths) {
// Check the difference in months matches the frequency value
uint256 diff = allowedAt.diffMonths(block.timestamp);
if (diff % frequency != 0) revert TaskTimeLockActive(block.timestamp, allowedAt);
day = allowedAt.getDay();
monthsToAdd = frequency;
} else {
if (mode == Mode.OnDay) {
day = allowedAt.getDay();
} else if (mode == Mode.OnLastMonthDay) {
day = block.timestamp.getDaysInMonth();
} else {
revert TaskInvalidFrequencyMode(uint8(mode));
}

// Check the current day matches the one in the configuration
if (block.timestamp.getDay() != day) revert TaskTimeLockActive(block.timestamp, allowedAt);
monthsToAdd = 1;
}

// Construct when would be the current allowed timestamp only considering the current month and year
uint256 currentAllowedAt = _getCurrentAllowedDateForMonthlyRelativeFrequency(allowedAt, day);

// Since we already checked the current timestamp is not before the allowed timestamp set,
// we simply need to check we are withing the allowed execution window
facuspagnuolo marked this conversation as resolved.
Show resolved Hide resolved
if (block.timestamp - currentAllowedAt > window) revert TaskTimeLockActive(block.timestamp, allowedAt);

// Finally set the next allowed date to the corresponding number of months from the current date
_nextAllowedAt = _getNextAllowedDateForMonthlyRelativeFrequency(currentAllowedAt, monthsToAdd);
}
}

/**
* @dev Tells the number of delay periods passed between the last expiration timestamp and the current timestamp
* @dev After time locked task hook
*/
function _getDelayPeriods() internal view returns (uint256) {
uint256 diff = block.timestamp - timeLockExpiration;
return diff / timeLockDelay;
function _afterTimeLockedTask(address, uint256) internal virtual {
if (_nextAllowedAt == 0) return;
_setTimeLockAllowedAt(_nextAllowedAt);
_nextAllowedAt = 0;
}

/**
* @dev Before time locked task hook
* @dev Sets a new time lock
*/
function _beforeTimeLockedTask(address, uint256) internal virtual {
if (block.timestamp < timeLockExpiration) revert TaskTimeLockNotExpired(timeLockExpiration, block.timestamp);
function _setTimeLock(uint8 mode, uint256 frequency, uint256 allowedAt, uint256 window) internal {
if (mode == uint8(Mode.Seconds)) {
// The execution window and timestamp are optional, but both must be given or none
// If given the execution window cannot be larger than the number of seconds
// Also, if these are given the frequency must be checked as well, otherwise it could be unsetting the lock
if (window > 0 || allowedAt > 0) {
if (frequency == 0) revert TaskInvalidFrequency(mode, frequency);
if (window == 0 || window > frequency) revert TaskInvalidAllowedWindow(mode, window);
if (allowedAt == 0) revert TaskInvalidAllowedDate(mode, allowedAt);
}
} else if (mode == uint8(Mode.OnDay)) {
// It must be valid for every month, then the frequency value cannot be larger than 28 days
if (frequency == 0 || frequency > 28) revert TaskInvalidFrequency(mode, frequency);

// The execution window that cannot be larger than 28 days
if (window == 0 || window > DAYS_28) revert TaskInvalidAllowedWindow(mode, window);

// The allowed date must match the specified frequency value
if (allowedAt == 0 || allowedAt.getDay() != frequency) revert TaskInvalidAllowedDate(mode, allowedAt);
} else if (mode == uint8(Mode.OnLastMonthDay)) {
// There must be no frequency value in this case
if (frequency != 0) revert TaskInvalidFrequency(mode, frequency);

if (timeLockExecutionPeriod > 0) {
uint256 diff = block.timestamp - timeLockExpiration;
uint256 periods = diff / timeLockDelay;
uint256 offset = diff - (periods * timeLockDelay);
if (offset > timeLockExecutionPeriod) revert TaskTimeLockWaitNextPeriod(offset, timeLockExecutionPeriod);
// The execution window that cannot be larger than 28 days
if (window == 0 || window > DAYS_28) revert TaskInvalidAllowedWindow(mode, window);

// The allowed date timestamp must be the last day of the month
if (allowedAt == 0) revert TaskInvalidAllowedDate(mode, allowedAt);
if (allowedAt.getDay() != allowedAt.getDaysInMonth()) revert TaskInvalidAllowedDate(mode, allowedAt);
} else if (mode == uint8(Mode.EverySomeMonths)) {
// There is no limit on the number of months
if (frequency == 0) revert TaskInvalidFrequency(mode, frequency);

// The execution window cannot be larger than the number of months considering months of 28 days
if (window == 0 || window > frequency * DAYS_28) revert TaskInvalidAllowedWindow(mode, window);

// The execution allowed at timestamp and the day cannot be greater than the 28th
if (allowedAt == 0 || allowedAt.getDay() > 28) revert TaskInvalidAllowedDate(mode, allowedAt);
} else {
revert TaskInvalidFrequencyMode(mode);
}

_mode = Mode(mode);
_frequency = frequency;
_allowedAt = allowedAt;
_window = window;

emit TimeLockSet(mode, frequency, allowedAt, window);
}

/**
* @dev After time locked task hook
* @dev Sets the time-lock execution allowed timestamp
* @param allowedAt New execution allowed timestamp to be set
*/
function _afterTimeLockedTask(address, uint256) internal virtual {
if (timeLockDelay > 0) {
uint256 nextExpirationTimestamp;
if (timeLockExpiration == 0) {
nextExpirationTimestamp = block.timestamp + timeLockDelay;
} else {
uint256 diff = block.timestamp - timeLockExpiration;
uint256 nextPeriod = (diff / timeLockDelay) + 1;
nextExpirationTimestamp = timeLockExpiration + (nextPeriod * timeLockDelay);
}
_setTimeLockExpiration(nextExpirationTimestamp);
}
function _setTimeLockAllowedAt(uint256 allowedAt) internal {
_allowedAt = allowedAt;
emit TimeLockAllowedAtSet(allowedAt);
}

/**
* @dev Sets the time-lock delay
* @param delay New delay to be set
* @dev Tells the allowed date based on a current allowed date considering the current timestamp and a specific day.
* It builds a new date using the current timestamp's month and year, following by the specified day, and using
* the current allowed date hours, minutes, and seconds.
*/
function _setTimeLockDelay(uint256 delay) internal {
if (delay < timeLockExecutionPeriod) revert TaskExecutionPeriodGtDelay(timeLockExecutionPeriod, delay);
timeLockDelay = delay;
emit TimeLockDelaySet(delay);
function _getCurrentAllowedDateForMonthlyRelativeFrequency(uint256 allowedAt, uint256 day)
private
view
returns (uint256)
{
(uint256 year, uint256 month, ) = block.timestamp.timestampToDate();
return _getAllowedDateFor(allowedAt, year, month, day);
}

/**
* @dev Sets the time-lock expiration timestamp
* @param expiration New expiration timestamp to be set
* @dev Tells the next allowed date based on a current allowed date considering a number of months to increase
*/
function _setTimeLockExpiration(uint256 expiration) internal {
timeLockExpiration = expiration;
emit TimeLockExpirationSet(expiration);
function _getNextAllowedDateForMonthlyRelativeFrequency(uint256 allowedAt, uint256 monthsToIncrease)
private
view
returns (uint256)
{
(uint256 year, uint256 month, uint256 day) = allowedAt.timestampToDate();
uint256 nextMonth = month + (monthsToIncrease % 12);
facuspagnuolo marked this conversation as resolved.
Show resolved Hide resolved
uint256 nextYear = nextMonth > month ? year : (year + 1);
facuspagnuolo marked this conversation as resolved.
Show resolved Hide resolved
uint256 nextDay = _mode == Mode.OnLastMonthDay ? DateTime._getDaysInMonth(nextYear, nextMonth) : day;
return _getAllowedDateFor(allowedAt, nextYear, nextMonth, nextDay);
}

/**
* @dev Sets the time-lock execution period
* @param period New execution period to be set
* @dev Builds an allowed date using a specific year, month, and day
*/
function _setTimeLockExecutionPeriod(uint256 period) internal {
if (period > timeLockDelay) revert TaskExecutionPeriodGtDelay(period, timeLockDelay);
timeLockExecutionPeriod = period;
emit TimeLockExecutionPeriodSet(period);
function _getAllowedDateFor(uint256 allowedAt, uint256 year, uint256 month, uint256 day)
private
pure
returns (uint256)
{
return
DateTime.timestampFromDateTime(
year,
month,
day,
allowedAt.getHour(),
allowedAt.getMinute(),
allowedAt.getSecond()
);
}
}
Loading
Loading