The PWNSignatureChecker library implements the isValidSignatureNow view function. This library is a modification of the Open-Zeppelin SignatureChecker library extended by support for EIP-2098 compact signatures.
PWNSignatureChecker.sol is written in Solidity version 0.8.16
isValidSignatureNow
Overview
This function will try to recover a signer of a given signature and check if is the same as the given signer address. For a contract account signer address, the function will check signature validity by calling isValidSignature function defined by EIP-1271.
This function takes three arguments supplied by the caller:
addresssigner - Address that should be a hash signer or a signature validator, in case of a contract account.
bytes32hash - Hash of a signed message that should be validated.
bytes memorysignature - Signature of a signed hash. Can be empty for a contract account signature validation. The signature can be standard (65 bytes) or compact (64 bytes) defined by EIP-2098.
Implementation
functionisValidSignatureNow(address signer,bytes32hash,bytesmemory signature) internalviewreturns (bool) {// Check that signature is valid for contract accountif (signer.code.length >0) { (bool success,bytesmemory result) = signer.staticcall( abi.encodeWithSelector(IERC1271.isValidSignature.selector,hash, signature) );return success && result.length ==32&& abi.decode(result, (bytes32)) ==bytes32(IERC1271.isValidSignature.selector); }// Check that signature is valid for EOAelse {bytes32 r;bytes32 s;uint8 v;// Standard signature data (65 bytes)if (signature.length ==65) {assembly { r :=mload(add(signature,0x20)) s :=mload(add(signature,0x40)) v :=byte(0,mload(add(signature,0x60))) } }// Compact signature data (64 bytes) - see EIP-2098elseif (signature.length ==64) {bytes32 vs;assembly { r :=mload(add(signature,0x20)) vs :=mload(add(signature,0x40)) } s = vs &bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff); v =uint8((uint256(vs) >>255) +27); } else {revertInvalidSignatureLength({ length: signature.length }); }return signer == ECDSA.recover(hash, v, r, s); }}