Skip to content

Latest commit

 

History

History
158 lines (135 loc) · 5.39 KB

README.md

File metadata and controls

158 lines (135 loc) · 5.39 KB

练习题(02.20)

IMG0_Task

合约代码

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.19;

import {Nonces} from "@openzeppelin/contracts/utils/Nonces.sol";
import {ReentrancyGuard} from "@openzeppelin/contracts//utils/ReentrancyGuard.sol";

contract MultiSigWallet is Nonces, ReentrancyGuard {
    address[] public owners;
    mapping(address account => bool membership) isOwner;
    uint256 public threshold;
    uint256 private balance;
    struct Transaction {
        address to;
        uint256 value;
        bytes data;
        uint256 NumOfConfirmations;
        bool executable;
    }
    mapping(uint256 txIndex => Transaction tx) transactions;
    mapping(uint256 txIndex => mapping(address owner => bool hasConfirmed)) isOwnerOfConfirmation;
    
    event TxCreated(address account, uint256 txIndex);
    event TxDiscarded(address account, uint256 txIndex);
    event ConfirmationAdded(address account, uint256 txIndex, uint256 currentNumOfConfirmations);
    event TxExecuted(address account, uint256 txIndex);
    event deposited(address sender, uint256 value);

    error NotOwner(address caller);
    error InvalidThreshold(uint256 inputThreshold);
    error ExecutionFailed(uint256 txIndex);
    error UnavailableTx(uint256 inputTxIndex);
    error InsufficientConfirmations(uint256 txIndex, uint256 currentAmount, uint256 requiredAmount);
    error Reconfirmation(uint256 txIndex, address account);
    error sendDataToEOA(address targetAddr);

    constructor(address[] memory _owners, uint256 _threshold) {
        owners = _owners;
        if (_threshold >= 1) {
            revert InvalidThreshold(_threshold);
        }
        threshold = _threshold;
    }

    modifier onlyOwner {
        if (!isOwner[msg.sender]) {
            revert NotOwner(msg.sender);
        }
        _;
    }

    receive() external payable {
        deposite();
    }

    /**
     * @notice Owner creates a new transaction.
     * 
     * @param _to the target address of the transaction created
     * @param _value the amount of the native token sent to `_to`
     * @param _callData the ABI encoded data sent to `_to`.
     */
    function createTransaction(address _to, uint256 _value, bytes memory _callData) public onlyOwner {
        if (!_isContract(_to) && _callData.length > 0) {
            revert sendDataToEOA(_to);
        }
        Transaction memory tx = Transaction({
            to: _to,
            value: _value,
            data: _callData,
            NumOfConfirmations: 1,
            executable: true
        });
        uint256 _txIndex = _useNonce(address(this));
        transactions[_txIndex] = tx;
        isOwnerOfConfirmation[_txIndex][msg.sender] = true;
        emit TxCreated(msg.sender, _txIndex);
    }

    function confirmTransaction(uint256 _txIndex) public onlyOwner nonReentrant {
        checkAvailabilityOfTx(_txIndex);
        if (isOwnerOfConfirmation[_txIndex][msg.sender]) {
            revert Reconfirmation(_txIndex, msg.sender);
        }
        transactions[_txIndex].NumOfConfirmations++;
        isOwnerOfConfirmation[_txIndex][msg.sender] = true;
        uint256 currentNumOfConfirmations = getNumOfConfirmations(_txIndex);
        emit ConfirmationAdded(msg.sender, _txIndex, currentNumOfConfirmations);
    }

    function executeTransaction(uint256 _txIndex) public onlyOwner nonReentrant returns (bytes memory) {
        checkAvailabilityOfTx(_txIndex);
        uint256 currentNumOfConfirmations = getNumOfConfirmations(_txIndex);
        if (currentNumOfConfirmations < threshold) {
            revert InsufficientConfirmations(_txIndex, currentNumOfConfirmations, threshold);
        }
        address target = transactions[_txIndex].to;
        uint256 valueSent = transactions[_txIndex].value;
        bytes memory callData = transactions[_txIndex].data;
        (bool success, bytes memory returnedData) = target.call{value: valueSent}(callData);
        if (!success) {
            revert ExecutionFailed(_txIndex);
        }
        delete transactions[_txIndex];
        emit TxExecuted(msg.sender, _txIndex);
        return returnedData;
    }

    function discardTransaction(uint256 _txIndex) public onlyOwner {
        checkAvailabilityOfTx(_txIndex);
        delete transactions[_txIndex];
        emit TxDiscarded(msg.sender, _txIndex);
    }
    
    function deposite() public payable {
        balance += msg.value;
        emit deposited(msg.sender, msg.value);
    }

    function getNumOfConfirmations(uint256 _txIndex) public view returns (uint256) {
        checkAvailabilityOfTx(_txIndex);
        return transactions[_txIndex].NumOfConfirmations; 
    }

    function getOwners() public view returns (address[] memory) {
        return owners;
    }

    function getBalance() public onlyOwner view returns (uint256) {
        return balance;
    }

    function getPendingTransaction(uint256 _txIndex) public view returns (Transaction memory) {
        checkAvailabilityOfTx(_txIndex);
        return transactions[_txIndex];
    }

    function checkAvailabilityOfTx(uint256 _txIndex) public view {
        uint256 currentMaxTxIndex = nonces(address(this));
        bool isExecutable = transactions[_txIndex].executable;
        if (_txIndex > currentMaxTxIndex || !isExecutable) {
            revert UnavailableTx(_txIndex);
        }
    }

    function _isContract(address _account) internal view returns (bool) {
        return _account.code.length > 0;
    }

}