-
Notifications
You must be signed in to change notification settings - Fork 526
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Seaport bulk sig support for smart accounts: AccountSeaportBulkSigSup…
…port (#633) * Add Seaport bulk sig support for smart accounts: AccountSeaportBulkSigSupport * fix build * fix build * fix build * run prettier * update gitmodules * fix lint errors
- Loading branch information
Showing
10 changed files
with
710 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
// SPDX-License-Identifier: Apache-2.0 | ||
pragma solidity ^0.8.11; | ||
|
||
import { ECDSA } from "solady/utils/ECDSA.sol"; | ||
import { SeaportOrderParser } from "./SeaportOrderParser.sol"; | ||
import { OrderParameters } from "seaport-types/src/lib/ConsiderationStructs.sol"; | ||
|
||
abstract contract SeaportEIP1271 is SeaportOrderParser { | ||
using ECDSA for bytes32; | ||
|
||
bytes32 private constant ACCOUNT_MESSAGE_TYPEHASH = keccak256("AccountMessage(bytes message)"); | ||
bytes32 private constant EIP712_TYPEHASH = | ||
keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"); | ||
bytes32 private immutable HASHED_NAME; | ||
bytes32 private immutable HASHED_VERSION; | ||
|
||
/// @notice The function selector of EIP1271.isValidSignature to be returned on sucessful signature verification. | ||
bytes4 public constant MAGICVALUE = 0x1626ba7e; | ||
|
||
constructor(string memory _name, string memory _version) { | ||
HASHED_NAME = keccak256(bytes(_name)); | ||
HASHED_VERSION = keccak256(bytes(_version)); | ||
} | ||
|
||
/// @notice See EIP-1271: https://eips.ethereum.org/EIPS/eip-1271 | ||
function isValidSignature( | ||
bytes32 _message, | ||
bytes memory _signature | ||
) public view virtual returns (bytes4 magicValue) { | ||
bytes32 targetDigest; | ||
bytes memory targetSig; | ||
|
||
// Handle OpenSea bulk order signatures that are >65 bytes in length. | ||
if (_signature.length > 65) { | ||
// Decode packed signature and order parameters. | ||
(bytes memory extractedPackedSig, OrderParameters memory orderParameters, uint256 counter) = abi.decode( | ||
_signature, | ||
(bytes, OrderParameters, uint256) | ||
); | ||
|
||
// Verify that the original digest matches the digest built with order parameters. | ||
bytes32 domainSeparator = _buildSeaportDomainSeparator(msg.sender); | ||
bytes32 orderHash = _deriveOrderHash(orderParameters, counter); | ||
|
||
require( | ||
_deriveEIP712Digest(domainSeparator, orderHash) == _message, | ||
"Seaport: order hash does not match the provided message." | ||
); | ||
|
||
// Build bulk signature digest | ||
targetDigest = _deriveEIP712Digest(domainSeparator, _computeBulkOrderProof(extractedPackedSig, orderHash)); | ||
|
||
// Extract the signature, which is the first 65 bytes | ||
targetSig = new bytes(65); | ||
for (uint256 i = 0; i < 65; i++) { | ||
targetSig[i] = extractedPackedSig[i]; | ||
} | ||
} else { | ||
targetDigest = getMessageHash(_message); | ||
targetSig = _signature; | ||
} | ||
|
||
address signer = targetDigest.recover(targetSig); | ||
|
||
if (_isAuthorizedSigner(signer)) { | ||
magicValue = MAGICVALUE; | ||
} | ||
} | ||
|
||
/** | ||
* @notice Returns the hash of message that should be signed for EIP1271 verification. | ||
* @param _hash The message hash pre replay protection. | ||
* @return messageHash The digest (with replay protection) to sign for EIP-1271 verification. | ||
*/ | ||
function getMessageHash(bytes32 _hash) public view returns (bytes32) { | ||
bytes32 messageHash = keccak256(abi.encode(_hash)); | ||
bytes32 typedDataHash = keccak256(abi.encode(ACCOUNT_MESSAGE_TYPEHASH, messageHash)); | ||
return keccak256(abi.encodePacked("\x19\x01", _accountDomainSeparator(), typedDataHash)); | ||
} | ||
|
||
/// @notice Returns the EIP712 domain separator for the contract. | ||
function _accountDomainSeparator() private view returns (bytes32) { | ||
return keccak256(abi.encode(EIP712_TYPEHASH, HASHED_NAME, HASHED_VERSION, block.chainid, address(this))); | ||
} | ||
|
||
/// @notice Returns whether a given signer is an authorized signer for the contract. | ||
function _isAuthorizedSigner(address _signer) internal view virtual returns (bool); | ||
} |
Oops, something went wrong.