2021-11-01 12:30:35 +01:00
|
|
|
// SPDX-License-Identifier: MIT
|
2022-09-29 20:18:02 +10:00
|
|
|
pragma solidity ^0.8.8;
|
2021-11-01 12:30:35 +01:00
|
|
|
|
2023-01-17 13:55:58 +01:00
|
|
|
import "./Configuration.sol";
|
2023-01-09 14:20:59 +01:00
|
|
|
import "./Requests.sol";
|
2023-01-10 14:16:32 +01:00
|
|
|
import "./Periods.sol";
|
2022-09-29 20:18:02 +10:00
|
|
|
|
2023-01-10 14:16:32 +01:00
|
|
|
abstract contract Proofs is Periods {
|
2023-01-23 11:57:10 +01:00
|
|
|
ProofConfig private _config;
|
2023-01-17 13:55:58 +01:00
|
|
|
|
2023-01-23 11:57:10 +01:00
|
|
|
constructor(ProofConfig memory config) Periods(config.period) {
|
2022-03-09 11:21:19 +01:00
|
|
|
require(block.number > 256, "Insufficient block height");
|
2023-01-23 11:57:10 +01:00
|
|
|
_config = config;
|
2022-03-02 15:44:58 +01:00
|
|
|
}
|
|
|
|
|
2023-01-23 11:57:10 +01:00
|
|
|
mapping(SlotId => uint256) private _slotStarts;
|
|
|
|
mapping(SlotId => uint256) private _probabilities;
|
|
|
|
mapping(SlotId => uint256) private _missed;
|
|
|
|
mapping(SlotId => mapping(Period => bool)) private _received;
|
|
|
|
mapping(SlotId => mapping(Period => bool)) private _missing;
|
2021-11-01 12:30:35 +01:00
|
|
|
|
2023-01-18 15:26:21 +01:00
|
|
|
function slotState(SlotId id) public view virtual returns (SlotState);
|
2021-11-03 13:24:50 +01:00
|
|
|
|
2023-01-09 14:41:28 +01:00
|
|
|
function missingProofs(SlotId slotId) public view returns (uint256) {
|
2023-01-23 11:57:10 +01:00
|
|
|
return _missed[slotId];
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
|
|
|
|
2023-03-08 17:19:49 +01:00
|
|
|
function resetMissingProofs(SlotId slotId) internal {
|
|
|
|
_missed[slotId] = 0;
|
|
|
|
}
|
|
|
|
|
2023-01-10 12:51:26 +01:00
|
|
|
function _startRequiringProofs(SlotId id, uint256 probability) internal {
|
2023-01-23 11:57:10 +01:00
|
|
|
_slotStarts[id] = block.timestamp;
|
|
|
|
_probabilities[id] = probability;
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
|
|
|
|
2023-01-19 16:47:29 +01:00
|
|
|
function _getPointer(SlotId id, Period period) internal view returns (uint8) {
|
2022-03-10 10:12:03 +01:00
|
|
|
uint256 blockNumber = block.number % 256;
|
2023-01-17 09:12:15 +01:00
|
|
|
uint256 periodNumber = Period.unwrap(period) % 256;
|
2023-01-09 14:20:59 +01:00
|
|
|
uint256 idOffset = uint256(SlotId.unwrap(id)) % 256;
|
2022-03-10 10:12:03 +01:00
|
|
|
uint256 pointer = (blockNumber + periodNumber + idOffset) % 256;
|
|
|
|
return uint8(pointer);
|
|
|
|
}
|
2022-03-08 15:58:08 +01:00
|
|
|
|
2023-01-17 08:37:59 +01:00
|
|
|
function getPointer(SlotId id) public view returns (uint8) {
|
2023-01-19 16:47:29 +01:00
|
|
|
return _getPointer(id, _blockPeriod());
|
2022-03-10 13:35:41 +01:00
|
|
|
}
|
|
|
|
|
2023-01-19 16:47:29 +01:00
|
|
|
function _getChallenge(uint8 pointer) internal view returns (bytes32) {
|
2022-03-10 10:12:03 +01:00
|
|
|
bytes32 hash = blockhash(block.number - 1 - pointer);
|
|
|
|
assert(uint256(hash) != 0);
|
|
|
|
return keccak256(abi.encode(hash));
|
|
|
|
}
|
2022-03-08 15:58:08 +01:00
|
|
|
|
2023-01-19 16:47:29 +01:00
|
|
|
function _getChallenge(
|
2023-01-10 11:04:04 +01:00
|
|
|
SlotId id,
|
2023-01-17 09:12:15 +01:00
|
|
|
Period period
|
2023-01-10 11:04:04 +01:00
|
|
|
) internal view returns (bytes32) {
|
2023-01-19 16:47:29 +01:00
|
|
|
return _getChallenge(_getPointer(id, period));
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
|
|
|
|
2023-01-17 08:37:59 +01:00
|
|
|
function getChallenge(SlotId id) public view returns (bytes32) {
|
2023-01-19 16:47:29 +01:00
|
|
|
return _getChallenge(id, _blockPeriod());
|
2022-03-10 13:04:46 +01:00
|
|
|
}
|
|
|
|
|
2023-01-10 11:04:04 +01:00
|
|
|
function _getProofRequirement(
|
|
|
|
SlotId id,
|
2023-01-17 09:12:15 +01:00
|
|
|
Period period
|
2023-01-10 11:04:04 +01:00
|
|
|
) internal view returns (bool isRequired, uint8 pointer) {
|
2023-01-16 16:31:04 +01:00
|
|
|
SlotState state = slotState(id);
|
2023-01-23 11:57:10 +01:00
|
|
|
Period start = _periodOf(_slotStarts[id]);
|
2023-01-19 16:47:29 +01:00
|
|
|
if (state != SlotState.Filled || !_isAfter(period, start)) {
|
2022-04-05 11:27:02 +02:00
|
|
|
return (false, 0);
|
2022-03-10 10:12:03 +01:00
|
|
|
}
|
2023-01-19 16:47:29 +01:00
|
|
|
pointer = _getPointer(id, period);
|
|
|
|
bytes32 challenge = _getChallenge(pointer);
|
2023-01-23 11:57:10 +01:00
|
|
|
uint256 probability = (_probabilities[id] * (256 - _config.downtime)) / 256;
|
2023-01-18 16:05:32 +01:00
|
|
|
isRequired = probability == 0 || uint256(challenge) % probability == 0;
|
2022-04-05 11:27:02 +02:00
|
|
|
}
|
|
|
|
|
2023-01-19 16:47:29 +01:00
|
|
|
function _isProofRequired(
|
2023-01-10 11:04:04 +01:00
|
|
|
SlotId id,
|
2023-01-17 09:12:15 +01:00
|
|
|
Period period
|
2023-01-10 11:04:04 +01:00
|
|
|
) internal view returns (bool) {
|
2022-04-05 11:27:02 +02:00
|
|
|
bool isRequired;
|
|
|
|
uint8 pointer;
|
2023-01-17 09:12:15 +01:00
|
|
|
(isRequired, pointer) = _getProofRequirement(id, period);
|
2023-01-23 11:57:10 +01:00
|
|
|
return isRequired && pointer >= _config.downtime;
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
|
|
|
|
2023-01-17 08:37:59 +01:00
|
|
|
function isProofRequired(SlotId id) public view returns (bool) {
|
2023-01-19 16:47:29 +01:00
|
|
|
return _isProofRequired(id, _blockPeriod());
|
2022-03-08 15:58:08 +01:00
|
|
|
}
|
|
|
|
|
2023-01-17 08:37:59 +01:00
|
|
|
function willProofBeRequired(SlotId id) public view returns (bool) {
|
2022-04-05 11:27:02 +02:00
|
|
|
bool isRequired;
|
|
|
|
uint8 pointer;
|
2023-01-19 16:47:29 +01:00
|
|
|
(isRequired, pointer) = _getProofRequirement(id, _blockPeriod());
|
2023-01-23 11:57:10 +01:00
|
|
|
return isRequired && pointer < _config.downtime;
|
2022-04-05 11:27:02 +02:00
|
|
|
}
|
|
|
|
|
2023-01-09 15:19:00 +01:00
|
|
|
function submitProof(SlotId id, bytes calldata proof) public {
|
2022-04-12 08:43:47 +02:00
|
|
|
require(proof.length > 0, "Invalid proof"); // TODO: replace by actual check
|
2023-01-23 11:57:10 +01:00
|
|
|
require(!_received[id][_blockPeriod()], "Proof already submitted");
|
|
|
|
_received[id][_blockPeriod()] = true;
|
2022-04-12 08:43:47 +02:00
|
|
|
emit ProofSubmitted(id, proof);
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
|
|
|
|
2023-01-10 14:16:32 +01:00
|
|
|
function _markProofAsMissing(SlotId id, Period missedPeriod) internal {
|
2023-01-19 16:47:29 +01:00
|
|
|
uint256 end = _periodEnd(missedPeriod);
|
2023-01-17 13:55:58 +01:00
|
|
|
require(end < block.timestamp, "Period has not ended yet");
|
2023-01-23 11:57:10 +01:00
|
|
|
require(block.timestamp < end + _config.timeout, "Validation timed out");
|
|
|
|
require(!_received[id][missedPeriod], "Proof was submitted, not missing");
|
2023-01-19 16:47:29 +01:00
|
|
|
require(_isProofRequired(id, missedPeriod), "Proof was not required");
|
2023-01-23 11:57:10 +01:00
|
|
|
require(!_missing[id][missedPeriod], "Proof already marked as missing");
|
|
|
|
_missing[id][missedPeriod] = true;
|
|
|
|
_missed[id] += 1;
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|
2022-04-12 08:43:47 +02:00
|
|
|
|
2023-01-09 14:20:59 +01:00
|
|
|
event ProofSubmitted(SlotId id, bytes proof);
|
2021-11-01 12:30:35 +01:00
|
|
|
}
|