|
| 1 | +// SPDX-License-Identifier: MIT |
| 2 | + |
| 3 | +pragma solidity ^0.8; |
| 4 | + |
| 5 | +import "./IArbitrable.sol"; |
| 6 | + |
| 7 | +/** |
| 8 | + * @title ArbitrableExample |
| 9 | + * An example of the arbitrable contract which connects to the arbitator that implements IArbitrator interface. |
| 10 | + */ |
| 11 | +contract ArbitrableExample is IArbitrable { |
| 12 | + struct DisputeStruct { |
| 13 | + bool isRuled; // Whether the dispute has been ruled or not. |
| 14 | + uint256 ruling; // Ruling given by the arbitrator. |
| 15 | + uint256 numberOfRulingOptions; // The number of choices the arbitrator can give. |
| 16 | + } |
| 17 | + |
| 18 | + IArbitrator public immutable arbitrator; // Arbitrator is set in constructor and never changed. |
| 19 | + |
| 20 | + mapping(uint256 => uint256) public externalIDtoLocalID; // Maps external (arbitrator side) dispute IDs to local dispute IDs. |
| 21 | + |
| 22 | + DisputeStruct[] public disputes; // Stores the disputes' info. disputes[disputeID]. |
| 23 | + |
| 24 | + /** @dev Constructor |
| 25 | + * @param _arbitrator The arbitrator to rule on created disputes. |
| 26 | + */ |
| 27 | + constructor(IArbitrator _arbitrator) { |
| 28 | + arbitrator = _arbitrator; |
| 29 | + } |
| 30 | + |
| 31 | + /** @dev TRUSTED. Calls createDispute function of the specified arbitrator to create a dispute. |
| 32 | + Note that we don’t need to check that msg.value is enough to pay arbitration fees as it’s the responsibility of the arbitrator contract. |
| 33 | + * @param _numberOfRulingOptions Number of ruling options. Must be greater than 1, otherwise there is nothing to choose from. |
| 34 | + * @param _arbitratorExtraData Extra data for the arbitrator. |
| 35 | + * @return disputeID Dispute id (on arbitrator side) of the dispute created. |
| 36 | + */ |
| 37 | + function createDispute(uint256 _numberOfRulingOptions, bytes calldata _arbitratorExtraData) |
| 38 | + external |
| 39 | + payable |
| 40 | + returns (uint256 disputeID) |
| 41 | + { |
| 42 | + require(_numberOfRulingOptions > 1, "Incorrect number of choices"); |
| 43 | + |
| 44 | + uint256 localDisputeID = disputes.length; |
| 45 | + disputes.push(DisputeStruct({isRuled: false, ruling: 0, numberOfRulingOptions: _numberOfRulingOptions})); |
| 46 | + |
| 47 | + disputeID = arbitrator.createDispute{value: msg.value}(_numberOfRulingOptions, _arbitratorExtraData); |
| 48 | + |
| 49 | + externalIDtoLocalID[disputeID] = localDisputeID; |
| 50 | + } |
| 51 | + |
| 52 | + /** @dev To be called by the arbitrator of the dispute, to declare the winning ruling. |
| 53 | + * @param _externalDisputeID ID of the dispute in arbitrator contract. |
| 54 | + * @param _ruling The ruling choice of the arbitration. |
| 55 | + */ |
| 56 | + function rule(uint256 _externalDisputeID, uint256 _ruling) external override { |
| 57 | + uint256 localDisputeID = externalIDtoLocalID[_externalDisputeID]; |
| 58 | + DisputeStruct storage dispute = disputes[localDisputeID]; |
| 59 | + require(msg.sender == address(arbitrator), "Only the arbitrator can execute this."); |
| 60 | + require(_ruling <= dispute.numberOfRulingOptions, "Invalid ruling."); |
| 61 | + require(dispute.isRuled == false, "This dispute has been ruled already."); |
| 62 | + |
| 63 | + dispute.isRuled = true; |
| 64 | + dispute.ruling = _ruling; |
| 65 | + |
| 66 | + emit Ruling(IArbitrator(msg.sender), _externalDisputeID, dispute.ruling); |
| 67 | + } |
| 68 | +} |
0 commit comments