all files / contracts/verifierHelpers/ EpochKeyVerifierHelper.sol

100% Statements 14/14
85.71% Branches 12/14
100% Functions 4/4
100% Lines 19/19
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                                          13× 13× 13× 13×   13×               13× 12×   12×                     10×                                                      
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
 
import {Unirep} from '../Unirep.sol';
import {IVerifier} from '../interfaces/IVerifier.sol';
import {BaseVerifierHelper} from './BaseVerifierHelper.sol';
 
/// @title EpochKeyVerifierHelper
/// @dev https://developer.unirep.io/docs/contracts-api/verifiers/epoch-key-verifier-helper
contract EpochKeyVerifierHelper is BaseVerifierHelper {
    constructor(
        Unirep _unirep,
        IVerifier _verifier
    ) BaseVerifierHelper(_unirep, _verifier) {}
 
    /// @dev https://developer.unirep.io/docs/contracts-api/verifiers/epoch-key-verifier-helper#decodeepochkeysignals
    /// @param publicSignals The public signals of the snark proof
    /// @return signals The EpochKeySignals
    function decodeEpochKeySignals(
        uint256[] calldata publicSignals
    ) public pure returns (EpochKeySignals memory) {
        EpochKeySignals memory signals;
        signals.epochKey = publicSignals[0];
        signals.stateTreeRoot = publicSignals[1];
        signals.data = publicSignals[3];
        // now decode the control values
        (
            signals.nonce,
            signals.epoch,
            signals.attesterId,
            signals.revealNonce,
            signals.chainId
        ) = super.decodeEpochKeyControl(publicSignals[2]);
 
        if (signals.epochKey >= SNARK_SCALAR_FIELD) revert InvalidEpochKey();
        Iif (signals.attesterId >= type(uint160).max) revert AttesterInvalid();
 
        return signals;
    }
 
    /// @dev https://developer.unirep.io/docs/contracts-api/verifiers/epoch-key-verifier-helper#verifyandcheck
    /// @param publicSignals The public signals of the snark proof
    /// @param proof The proof data of the snark proof
    /// @return signals The EpochKeySignals
    function verifyAndCheck(
        uint256[] calldata publicSignals,
        uint256[8] calldata proof
    ) public view returns (EpochKeySignals memory) {
        EpochKeySignals memory signals = decodeEpochKeySignals(publicSignals);
 
        if (!verifier.verifyProof(publicSignals, proof)) revert InvalidProof();
 
        uint48 epoch = unirep.attesterCurrentEpoch(signals.attesterId);
        if (signals.epoch > epoch) revert InvalidEpoch();
 
        if (
            !unirep.attesterStateTreeRootExists(
                signals.attesterId,
                signals.epoch,
                signals.stateTreeRoot
            )
        ) revert InvalidStateTreeRoot(signals.stateTreeRoot);
 
        Iif (signals.chainId != chainid) revert ChainIdNotMatch(signals.chainId);
 
        return signals;
    }
 
    /// @dev https://developer.unirep.io/docs/contracts-api/verifiers/epoch-key-verifier-helper#verifyandcheckcaller
    /// @param publicSignals The public signals of the snark proof
    /// @param proof The proof data of the snark proof
    /// @return signals The EpochKeySignals
    function verifyAndCheckCaller(
        uint256[] calldata publicSignals,
        uint256[8] calldata proof
    ) public view returns (EpochKeySignals memory) {
        EpochKeySignals memory signals = verifyAndCheck(publicSignals, proof);
 
        if (signals.attesterId != uint160(msg.sender)) {
            revert CallerInvalid();
        }
 
        return signals;
    }
}