-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathMetronome.sol
86 lines (76 loc) · 2.89 KB
/
Metronome.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.24;
import { ReentrancyGuard } from "@openzeppelin/contracts/utils/ReentrancyGuard.sol";
struct Schedule {
uint256 remainder;
uint256 divisor;
uint256 bounty;
}
contract Metronome is ReentrancyGuard {
mapping(uint256 => Schedule) public Schedules;
mapping(uint256 => uint256) public ScheduleBalances;
// scheduleID => blockNumber => true if claimed already and false if not
mapping(uint256 => mapping(uint256 => bool)) public ClaimedBounties;
uint256 public NumSchedules;
event ScheduleCreated(
uint256 indexed scheduleID,
uint256 indexed remainder,
uint256 indexed divisor,
uint256 bounty
);
event BalanceIncreased(uint256 indexed scheduleID, uint256 amount);
event BountyClaimed(uint256 indexed scheduleID, address indexed forAddress, uint256 payment);
error InvalidSchedule();
function createSchedule(
uint256 remainder,
uint256 divisor,
uint256 bounty
) external payable returns (uint256 scheduleID) {
if (divisor == 0) {
revert InvalidSchedule();
}
if (bounty == 0) {
revert InvalidSchedule();
}
if (remainder >= divisor) {
revert InvalidSchedule();
}
scheduleID = NumSchedules;
NumSchedules++;
Schedules[scheduleID] = Schedule(remainder, divisor, bounty);
emit ScheduleCreated(scheduleID, remainder, divisor, bounty);
if (msg.value > 0) {
ScheduleBalances[scheduleID] = msg.value;
emit BalanceIncreased(scheduleID, msg.value);
}
}
function increaseBalance(uint256 scheduleID) external payable {
ScheduleBalances[scheduleID] += msg.value;
emit BalanceIncreased(scheduleID, msg.value);
}
function _claim(uint256 scheduleID, address forAddress) internal {
if (
!ClaimedBounties[scheduleID][block.number] &&
(block.number % Schedules[scheduleID].divisor == Schedules[scheduleID].remainder)
) {
uint256 payment = ScheduleBalances[scheduleID];
if (payment > Schedules[scheduleID].bounty) {
payment = Schedules[scheduleID].bounty;
}
if (payment > 0) {
ClaimedBounties[scheduleID][block.number] = true;
ScheduleBalances[scheduleID] -= payment;
payable(forAddress).transfer(payment);
emit BountyClaimed(scheduleID, forAddress, payment);
}
}
}
function claim(uint256 scheduleID, address forAddress) external nonReentrant {
_claim(scheduleID, forAddress);
}
function claimBatch(uint256[] memory scheduleIDs, address forAddress) external nonReentrant {
for (uint256 i = 0; i < scheduleIDs.length; i++) {
_claim(scheduleIDs[i], forAddress);
}
}
}