Skip to content

Commit

Permalink
feat: As a user, I want to have an example ecrecover module
Browse files Browse the repository at this point in the history
  • Loading branch information
zhanymkanov committed Oct 21, 2023
1 parent c877819 commit 2876ed7
Show file tree
Hide file tree
Showing 2 changed files with 128 additions and 0 deletions.
52 changes: 52 additions & 0 deletions contracts/src/examples/modules/ECDSAModule.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
// SPDX-License-Identifier: MIT
pragma solidity 0.8.21;

import { ECDSA } from "openzeppelin-contracts/contracts/utils/cryptography/ECDSA.sol";
import { Ownable } from "openzeppelin-contracts/contracts/access/Ownable.sol";

import { AbstractModule } from "../../interface/AbstractModule.sol";
import { AttestationPayload } from "../../types/Structs.sol";

/**
* @title ECDSA Module
* @author Consensys x DappSheriff
* @notice This contract illustrates a valid Module that is used to verify ECDSA signatures of payload
*/
contract ECDSAModule is AbstractModule, Ownable {
address public signer;
mapping(uint256 => bool) public usedNonces; // nonce => used

constructor(address initialOwner, address _signer) {
signer = _signer;
}

/// @dev This empty method prevents Foundry from counting this contract in code coverage
function test() public {}

/**
* @notice This method is used to run the module's validation logic
* @param attestationPayload - AttestationPayload containing the user address as `subject` and nonce as `attestationData`
* @param validationPayload - Payload encoded with abi.encode(uint256).toEthSignedMessageHash().sign(signer)
*/
function run(
AttestationPayload memory attestationPayload,
bytes memory validationPayload,
address /*txSender*/,
uint256 /*value*/
) public override {
address signee = abi.decode(attestationPayload.subject, (address));
uint256 nonce = abi.decode(attestationPayload.attestationData, (uint256));
if (usedNonces[nonce]) {
revert("Nonce already used");
}

bytes32 hash = ECDSA.toEthSignedMessageHash(abi.encodePacked(signee, nonce));
require(ECDSA.recover(hash, validationPayload) == signer, "Wrong signature");

usedNonces[nonce] = true;
}

function setSigner(address _signer) external onlyOwner {
signer = _signer;
}
}
76 changes: 76 additions & 0 deletions contracts/test/example/ECDSAModule.t.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
// SPDX-License-Identifier: MIT
pragma solidity 0.8.21;

import { Test } from "forge-std/Test.sol";
import { ECDSA } from "openzeppelin-contracts/contracts/utils/cryptography/ECDSA.sol";

import { ECDSAModule } from "../../src/examples/modules/ECDSAModule.sol";
import { AttestationPayload } from "../../src/types/Structs.sol";

contract ECDSAModuleTest is Test {
ECDSAModule private ecdsaModule;
address private signer;
uint256 private signerPk;

function setUp() public {
(signer, signerPk) = makeAddrAndKey("veraxUser");
ecdsaModule = new ECDSAModule(signer, signer);

vm.deal(signer, 1 ether);
}

function test_ECDSAModule_validSignature() public {
address user = makeAddr("user");
uint256 nonce = 1234567;
AttestationPayload memory attestationPayload = AttestationPayload(
bytes32(uint256(1234)),
0,
abi.encode(user),
abi.encode(nonce)
);

bytes32 hash = ECDSA.toEthSignedMessageHash(abi.encodePacked(user, nonce));

(uint8 v, bytes32 r, bytes32 s) = vm.sign(signerPk, hash);
bytes memory signature = abi.encodePacked(r, s, v);

ecdsaModule.run(attestationPayload, signature, signer, 0);
}

function test_ECDSAModule_revertInvalidSignature() public {
(address user, uint256 fakeKey) = makeAddrAndKey("user");
uint256 nonce = 1234567;
AttestationPayload memory attestationPayload = AttestationPayload(
bytes32(uint256(1234)),
0,
abi.encode(user),
abi.encode(nonce)
);

bytes32 hash = ECDSA.toEthSignedMessageHash(abi.encodePacked(user, nonce));
(uint8 v, bytes32 r, bytes32 s) = vm.sign(fakeKey, hash);
bytes memory signature = abi.encodePacked(r, s, v);

vm.expectRevert("Wrong signature");
ecdsaModule.run(attestationPayload, signature, user, 0);
}

function test_ECDSAModule_revertNonceUsed() public {
address user = makeAddr("user");
uint256 nonce = 1234567;
AttestationPayload memory attestationPayload = AttestationPayload(
bytes32(uint256(1234)),
0,
abi.encode(user),
abi.encode(nonce)
);

bytes32 hash = ECDSA.toEthSignedMessageHash(abi.encodePacked(user, nonce));
(uint8 v, bytes32 r, bytes32 s) = vm.sign(signerPk, hash);
bytes memory signature = abi.encodePacked(r, s, v);

ecdsaModule.run(attestationPayload, signature, user, 0);
vm.expectRevert("Nonce already used");
ecdsaModule.run(attestationPayload, signature, user, 0);
}
}

0 comments on commit 2876ed7

Please sign in to comment.