Overview
ETH Balance
0 ETH
ETH Value
$0.00More Info
Private Name Tags
ContractCreator
Loading...
Loading
Contract Name:
ScrollBadgeTokenOwner
Compiler Version
v0.8.19+commit.7dd6d404
Optimization Enabled:
Yes with 200 runs
Other Settings:
paris EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; import {IERC721Metadata} from "@openzeppelin/contracts/token/ERC721/extensions/IERC721Metadata.sol"; import {IERC721} from "@openzeppelin/contracts/token/ERC721/IERC721.sol"; import {ScrollBadgeCustomPayload} from "../extensions/ScrollBadgeCustomPayload.sol"; import {ScrollBadgeSelfAttest} from "../extensions/ScrollBadgeSelfAttest.sol"; import {ScrollBadgeSingleton} from "../extensions/ScrollBadgeSingleton.sol"; import {ScrollBadge} from "../ScrollBadge.sol"; import {Unauthorized} from "../../Errors.sol"; string constant SCROLL_BADGE_NFT_OWNER_SCHEMA = "address tokenAddress, uint256 tokenId"; function decodePayloadData(bytes memory data) pure returns (address, uint256) { return abi.decode(data, (address, uint256)); } /// @title ScrollBadgeTokenOwner /// @notice A simple badge that attests that the user owns a specific NFT. contract ScrollBadgeTokenOwner is ScrollBadgeCustomPayload, ScrollBadgeSelfAttest, ScrollBadgeSingleton { error IncorrectBadgeOwner(); mapping(address => bool) public isTokenAllowed; constructor(address resolver_, address[] memory tokens_) ScrollBadge(resolver_) { for (uint256 i = 0; i < tokens_.length; ++i) { isTokenAllowed[tokens_[i]] = true; } } /// @inheritdoc ScrollBadge function onIssueBadge(Attestation calldata attestation) internal override (ScrollBadgeCustomPayload, ScrollBadgeSelfAttest, ScrollBadgeSingleton) returns (bool) { if (!super.onIssueBadge(attestation)) { return false; } // check that badge payload attestation is correct bytes memory payload = getPayload(attestation); (address tokenAddress, uint256 tokenId) = decodePayloadData(payload); if (!isTokenAllowed[tokenAddress]) { revert Unauthorized(); } if (IERC721(tokenAddress).ownerOf(tokenId) != attestation.recipient) { revert IncorrectBadgeOwner(); } return true; } /// @inheritdoc ScrollBadge function onRevokeBadge(Attestation calldata attestation) internal override (ScrollBadgeCustomPayload, ScrollBadgeSelfAttest, ScrollBadgeSingleton) returns (bool) { return super.onRevokeBadge(attestation); } /// @inheritdoc ScrollBadge function badgeTokenURI(bytes32 uid) public view override returns (string memory) { Attestation memory attestation = getAndValidateBadge(uid); bytes memory payload = getPayload(attestation); (address tokenAddress, uint256 tokenId) = decodePayloadData(payload); return IERC721Metadata(tokenAddress).tokenURI(tokenId); } /// @inheritdoc ScrollBadgeCustomPayload function getSchema() public pure override returns (string memory) { return SCROLL_BADGE_NFT_OWNER_SCHEMA; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { ISchemaRegistry } from "./ISchemaRegistry.sol"; import { ISemver } from "./ISemver.sol"; import { Attestation, Signature } from "./Common.sol"; /// @notice A struct representing the arguments of the attestation request. struct AttestationRequestData { address recipient; // The recipient of the attestation. uint64 expirationTime; // The time when the attestation expires (Unix timestamp). bool revocable; // Whether the attestation is revocable. bytes32 refUID; // The UID of the related attestation. bytes data; // Custom attestation data. uint256 value; // An explicit ETH amount to send to the resolver. This is important to prevent accidental user errors. } /// @notice A struct representing the full arguments of the attestation request. struct AttestationRequest { bytes32 schema; // The unique identifier of the schema. AttestationRequestData data; // The arguments of the attestation request. } /// @notice A struct representing the full arguments of the full delegated attestation request. struct DelegatedAttestationRequest { bytes32 schema; // The unique identifier of the schema. AttestationRequestData data; // The arguments of the attestation request. Signature signature; // The ECDSA signature data. address attester; // The attesting account. uint64 deadline; // The deadline of the signature/request. } /// @notice A struct representing the full arguments of the multi attestation request. struct MultiAttestationRequest { bytes32 schema; // The unique identifier of the schema. AttestationRequestData[] data; // The arguments of the attestation request. } /// @notice A struct representing the full arguments of the delegated multi attestation request. struct MultiDelegatedAttestationRequest { bytes32 schema; // The unique identifier of the schema. AttestationRequestData[] data; // The arguments of the attestation requests. Signature[] signatures; // The ECDSA signatures data. Please note that the signatures are assumed to be signed with increasing nonces. address attester; // The attesting account. uint64 deadline; // The deadline of the signature/request. } /// @notice A struct representing the arguments of the revocation request. struct RevocationRequestData { bytes32 uid; // The UID of the attestation to revoke. uint256 value; // An explicit ETH amount to send to the resolver. This is important to prevent accidental user errors. } /// @notice A struct representing the full arguments of the revocation request. struct RevocationRequest { bytes32 schema; // The unique identifier of the schema. RevocationRequestData data; // The arguments of the revocation request. } /// @notice A struct representing the arguments of the full delegated revocation request. struct DelegatedRevocationRequest { bytes32 schema; // The unique identifier of the schema. RevocationRequestData data; // The arguments of the revocation request. Signature signature; // The ECDSA signature data. address revoker; // The revoking account. uint64 deadline; // The deadline of the signature/request. } /// @notice A struct representing the full arguments of the multi revocation request. struct MultiRevocationRequest { bytes32 schema; // The unique identifier of the schema. RevocationRequestData[] data; // The arguments of the revocation request. } /// @notice A struct representing the full arguments of the delegated multi revocation request. struct MultiDelegatedRevocationRequest { bytes32 schema; // The unique identifier of the schema. RevocationRequestData[] data; // The arguments of the revocation requests. Signature[] signatures; // The ECDSA signatures data. Please note that the signatures are assumed to be signed with increasing nonces. address revoker; // The revoking account. uint64 deadline; // The deadline of the signature/request. } /// @title IEAS /// @notice EAS - Ethereum Attestation Service interface. interface IEAS is ISemver { /// @notice Emitted when an attestation has been made. /// @param recipient The recipient of the attestation. /// @param attester The attesting account. /// @param uid The UID the revoked attestation. /// @param schemaUID The UID of the schema. event Attested(address indexed recipient, address indexed attester, bytes32 uid, bytes32 indexed schemaUID); /// @notice Emitted when an attestation has been revoked. /// @param recipient The recipient of the attestation. /// @param attester The attesting account. /// @param schemaUID The UID of the schema. /// @param uid The UID the revoked attestation. event Revoked(address indexed recipient, address indexed attester, bytes32 uid, bytes32 indexed schemaUID); /// @notice Emitted when a data has been timestamped. /// @param data The data. /// @param timestamp The timestamp. event Timestamped(bytes32 indexed data, uint64 indexed timestamp); /// @notice Emitted when a data has been revoked. /// @param revoker The address of the revoker. /// @param data The data. /// @param timestamp The timestamp. event RevokedOffchain(address indexed revoker, bytes32 indexed data, uint64 indexed timestamp); /// @notice Returns the address of the global schema registry. /// @return The address of the global schema registry. function getSchemaRegistry() external view returns (ISchemaRegistry); /// @notice Attests to a specific schema. /// @param request The arguments of the attestation request. /// @return The UID of the new attestation. /// /// Example: /// attest({ /// schema: "0facc36681cbe2456019c1b0d1e7bedd6d1d40f6f324bf3dd3a4cef2999200a0", /// data: { /// recipient: "0xdEADBeAFdeAdbEafdeadbeafDeAdbEAFdeadbeaf", /// expirationTime: 0, /// revocable: true, /// refUID: "0x0000000000000000000000000000000000000000000000000000000000000000", /// data: "0xF00D", /// value: 0 /// } /// }) function attest(AttestationRequest calldata request) external payable returns (bytes32); /// @notice Attests to a specific schema via the provided ECDSA signature. /// @param delegatedRequest The arguments of the delegated attestation request. /// @return The UID of the new attestation. /// /// Example: /// attestByDelegation({ /// schema: '0x8e72f5bc0a8d4be6aa98360baa889040c50a0e51f32dbf0baa5199bd93472ebc', /// data: { /// recipient: '0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266', /// expirationTime: 1673891048, /// revocable: true, /// refUID: '0x0000000000000000000000000000000000000000000000000000000000000000', /// data: '0x1234', /// value: 0 /// }, /// signature: { /// v: 28, /// r: '0x148c...b25b', /// s: '0x5a72...be22' /// }, /// attester: '0xc5E8740aD971409492b1A63Db8d83025e0Fc427e', /// deadline: 1673891048 /// }) function attestByDelegation( DelegatedAttestationRequest calldata delegatedRequest ) external payable returns (bytes32); /// @notice Attests to multiple schemas. /// @param multiRequests The arguments of the multi attestation requests. The requests should be grouped by distinct /// schema ids to benefit from the best batching optimization. /// @return The UIDs of the new attestations. /// /// Example: /// multiAttest([{ /// schema: '0x33e9094830a5cba5554d1954310e4fbed2ef5f859ec1404619adea4207f391fd', /// data: [{ /// recipient: '0xdEADBeAFdeAdbEafdeadbeafDeAdbEAFdeadbeaf', /// expirationTime: 1673891048, /// revocable: true, /// refUID: '0x0000000000000000000000000000000000000000000000000000000000000000', /// data: '0x1234', /// value: 1000 /// }, /// { /// recipient: '0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266', /// expirationTime: 0, /// revocable: false, /// refUID: '0x480df4a039efc31b11bfdf491b383ca138b6bde160988222a2a3509c02cee174', /// data: '0x00', /// value: 0 /// }], /// }, /// { /// schema: '0x5ac273ce41e3c8bfa383efe7c03e54c5f0bff29c9f11ef6ffa930fc84ca32425', /// data: [{ /// recipient: '0xdEADBeAFdeAdbEafdeadbeafDeAdbEAFdeadbeaf', /// expirationTime: 0, /// revocable: true, /// refUID: '0x75bf2ed8dca25a8190c50c52db136664de25b2449535839008ccfdab469b214f', /// data: '0x12345678', /// value: 0 /// }, /// }]) function multiAttest(MultiAttestationRequest[] calldata multiRequests) external payable returns (bytes32[] memory); /// @notice Attests to multiple schemas using via provided ECDSA signatures. /// @param multiDelegatedRequests The arguments of the delegated multi attestation requests. The requests should be /// grouped by distinct schema ids to benefit from the best batching optimization. /// @return The UIDs of the new attestations. /// /// Example: /// multiAttestByDelegation([{ /// schema: '0x8e72f5bc0a8d4be6aa98360baa889040c50a0e51f32dbf0baa5199bd93472ebc', /// data: [{ /// recipient: '0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266', /// expirationTime: 1673891048, /// revocable: true, /// refUID: '0x0000000000000000000000000000000000000000000000000000000000000000', /// data: '0x1234', /// value: 0 /// }, /// { /// recipient: '0xdEADBeAFdeAdbEafdeadbeafDeAdbEAFdeadbeaf', /// expirationTime: 0, /// revocable: false, /// refUID: '0x0000000000000000000000000000000000000000000000000000000000000000', /// data: '0x00', /// value: 0 /// }], /// signatures: [{ /// v: 28, /// r: '0x148c...b25b', /// s: '0x5a72...be22' /// }, /// { /// v: 28, /// r: '0x487s...67bb', /// s: '0x12ad...2366' /// }], /// attester: '0x1D86495b2A7B524D747d2839b3C645Bed32e8CF4', /// deadline: 1673891048 /// }]) function multiAttestByDelegation( MultiDelegatedAttestationRequest[] calldata multiDelegatedRequests ) external payable returns (bytes32[] memory); /// @notice Revokes an existing attestation to a specific schema. /// @param request The arguments of the revocation request. /// /// Example: /// revoke({ /// schema: '0x8e72f5bc0a8d4be6aa98360baa889040c50a0e51f32dbf0baa5199bd93472ebc', /// data: { /// uid: '0x101032e487642ee04ee17049f99a70590c735b8614079fc9275f9dd57c00966d', /// value: 0 /// } /// }) function revoke(RevocationRequest calldata request) external payable; /// @notice Revokes an existing attestation to a specific schema via the provided ECDSA signature. /// @param delegatedRequest The arguments of the delegated revocation request. /// /// Example: /// revokeByDelegation({ /// schema: '0x8e72f5bc0a8d4be6aa98360baa889040c50a0e51f32dbf0baa5199bd93472ebc', /// data: { /// uid: '0xcbbc12102578c642a0f7b34fe7111e41afa25683b6cd7b5a14caf90fa14d24ba', /// value: 0 /// }, /// signature: { /// v: 27, /// r: '0xb593...7142', /// s: '0x0f5b...2cce' /// }, /// revoker: '0x244934dd3e31bE2c81f84ECf0b3E6329F5381992', /// deadline: 1673891048 /// }) function revokeByDelegation(DelegatedRevocationRequest calldata delegatedRequest) external payable; /// @notice Revokes existing attestations to multiple schemas. /// @param multiRequests The arguments of the multi revocation requests. The requests should be grouped by distinct /// schema ids to benefit from the best batching optimization. /// /// Example: /// multiRevoke([{ /// schema: '0x8e72f5bc0a8d4be6aa98360baa889040c50a0e51f32dbf0baa5199bd93472ebc', /// data: [{ /// uid: '0x211296a1ca0d7f9f2cfebf0daaa575bea9b20e968d81aef4e743d699c6ac4b25', /// value: 1000 /// }, /// { /// uid: '0xe160ac1bd3606a287b4d53d5d1d6da5895f65b4b4bab6d93aaf5046e48167ade', /// value: 0 /// }], /// }, /// { /// schema: '0x5ac273ce41e3c8bfa383efe7c03e54c5f0bff29c9f11ef6ffa930fc84ca32425', /// data: [{ /// uid: '0x053d42abce1fd7c8fcddfae21845ad34dae287b2c326220b03ba241bc5a8f019', /// value: 0 /// }, /// }]) function multiRevoke(MultiRevocationRequest[] calldata multiRequests) external payable; /// @notice Revokes existing attestations to multiple schemas via provided ECDSA signatures. /// @param multiDelegatedRequests The arguments of the delegated multi revocation attestation requests. The requests /// should be grouped by distinct schema ids to benefit from the best batching optimization. /// /// Example: /// multiRevokeByDelegation([{ /// schema: '0x8e72f5bc0a8d4be6aa98360baa889040c50a0e51f32dbf0baa5199bd93472ebc', /// data: [{ /// uid: '0x211296a1ca0d7f9f2cfebf0daaa575bea9b20e968d81aef4e743d699c6ac4b25', /// value: 1000 /// }, /// { /// uid: '0xe160ac1bd3606a287b4d53d5d1d6da5895f65b4b4bab6d93aaf5046e48167ade', /// value: 0 /// }], /// signatures: [{ /// v: 28, /// r: '0x148c...b25b', /// s: '0x5a72...be22' /// }, /// { /// v: 28, /// r: '0x487s...67bb', /// s: '0x12ad...2366' /// }], /// revoker: '0x244934dd3e31bE2c81f84ECf0b3E6329F5381992', /// deadline: 1673891048 /// }]) function multiRevokeByDelegation( MultiDelegatedRevocationRequest[] calldata multiDelegatedRequests ) external payable; /// @notice Timestamps the specified bytes32 data. /// @param data The data to timestamp. /// @return The timestamp the data was timestamped with. function timestamp(bytes32 data) external returns (uint64); /// @notice Timestamps the specified multiple bytes32 data. /// @param data The data to timestamp. /// @return The timestamp the data was timestamped with. function multiTimestamp(bytes32[] calldata data) external returns (uint64); /// @notice Revokes the specified bytes32 data. /// @param data The data to timestamp. /// @return The timestamp the data was revoked with. function revokeOffchain(bytes32 data) external returns (uint64); /// @notice Revokes the specified multiple bytes32 data. /// @param data The data to timestamp. /// @return The timestamp the data was revoked with. function multiRevokeOffchain(bytes32[] calldata data) external returns (uint64); /// @notice Returns an existing attestation by UID. /// @param uid The UID of the attestation to retrieve. /// @return The attestation data members. function getAttestation(bytes32 uid) external view returns (Attestation memory); /// @notice Checks whether an attestation exists. /// @param uid The UID of the attestation to retrieve. /// @return Whether an attestation exists. function isAttestationValid(bytes32 uid) external view returns (bool); /// @notice Returns the timestamp that the specified data was timestamped with. /// @param data The data to query. /// @return The timestamp the data was timestamped with. function getTimestamp(bytes32 data) external view returns (uint64); /// @notice Returns the timestamp that the specified data was timestamped with. /// @param data The data to query. /// @return The timestamp the data was timestamped with. function getRevokeOffchain(address revoker, bytes32 data) external view returns (uint64); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol) pragma solidity ^0.8.0; import "../IERC721.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional metadata extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Metadata is IERC721 { /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC721/IERC721.sol) pragma solidity ^0.8.0; import "../../utils/introspection/IERC165.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721 is IERC165 { /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in ``owner``'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom(address from, address to, uint256 tokenId, bytes calldata data) external; /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom(address from, address to, uint256 tokenId) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: Note that the caller is responsible to confirm that the recipient is capable of receiving ERC721 * or else they may be permanently lost. Usage of {safeTransferFrom} prevents loss, though the caller must * understand this adds an external call which potentially creates a reentrancy vulnerability. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 tokenId) external; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll} */ function isApprovedForAll(address owner, address operator) external view returns (bool); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; import {ScrollBadge} from "../ScrollBadge.sol"; import {decodeBadgeData} from "../../Common.sol"; import {MissingPayload} from "../../Errors.sol"; /// @title ScrollBadgeCustomPayload /// @notice This contract adds custom payload to ScrollBadge. abstract contract ScrollBadgeCustomPayload is ScrollBadge { /// @inheritdoc ScrollBadge function onIssueBadge(Attestation calldata attestation) internal virtual override returns (bool) { if (!super.onIssueBadge(attestation)) { return false; } bytes memory payload = getPayload(attestation); if (payload.length == 0) { revert MissingPayload(); } return true; } /// @inheritdoc ScrollBadge function onRevokeBadge(Attestation calldata attestation) internal virtual override returns (bool) { return super.onRevokeBadge(attestation); } /// @notice Return the badge payload. /// @param badge The Scroll badge attestation. /// @return The abi encoded badge payload. function getPayload(Attestation memory badge) public pure returns (bytes memory) { (, bytes memory payload) = decodeBadgeData(badge.data); return payload; } /// @notice Return the badge custom payload schema. /// @return The custom abi encoding schema used for the payload. /// @dev This schema serves as a decoding hint for clients. function getSchema() public virtual returns (string memory); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; import {ScrollBadge} from "../ScrollBadge.sol"; import {Unauthorized} from "../../Errors.sol"; /// @title ScrollBadgeSelfAttest /// @notice This contract ensures that only the badge recipient can attest. abstract contract ScrollBadgeSelfAttest is ScrollBadge { /// @inheritdoc ScrollBadge function onIssueBadge(Attestation calldata attestation) internal virtual override returns (bool) { if (!super.onIssueBadge(attestation)) { return false; } if (attestation.recipient != attestation.attester) { revert Unauthorized(); } return true; } /// @inheritdoc ScrollBadge function onRevokeBadge(Attestation calldata attestation) internal virtual override returns (bool) { return super.onRevokeBadge(attestation); } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; import {ScrollBadge} from "../ScrollBadge.sol"; import {SingletonBadge} from "../../Errors.sol"; /// @title ScrollBadgeSingleton /// @notice This contract only allows one active badge per wallet. abstract contract ScrollBadgeSingleton is ScrollBadge { /// @inheritdoc ScrollBadge function onIssueBadge(Attestation calldata attestation) internal virtual override returns (bool) { if (!super.onIssueBadge(attestation)) { return false; } if (hasBadge(attestation.recipient)) { revert SingletonBadge(); } return true; } /// @inheritdoc ScrollBadge function onRevokeBadge(Attestation calldata attestation) internal virtual override returns (bool) { return super.onRevokeBadge(attestation); } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; import {decodeBadgeData} from "../Common.sol"; import {IScrollBadge} from "../interfaces/IScrollBadge.sol"; import {IScrollBadgeResolver} from "../interfaces/IScrollBadgeResolver.sol"; import {AttestationBadgeMismatch, Unauthorized} from "../Errors.sol"; /// @title ScrollBadge /// @notice This contract implements the basic functionalities of a Scroll badge. /// It serves as the base contract for more complex badge functionalities. abstract contract ScrollBadge is IScrollBadge { // The global Scroll badge resolver contract. address public immutable resolver; // wallet address => badge count mapping(address => uint256) private _userBadgeCount; /// @dev Creates a new ScrollBadge instance. /// @param resolver_ The address of the global Scroll badge resolver contract. constructor(address resolver_) { resolver = resolver_; } /// @inheritdoc IScrollBadge function issueBadge(Attestation calldata attestation) public returns (bool) { // only callable from resolver if (msg.sender != address(resolver)) { revert Unauthorized(); } // delegate logic to subcontract if (!onIssueBadge(attestation)) { return false; } _userBadgeCount[attestation.recipient] += 1; emit IssueBadge(attestation.uid); return true; } /// @inheritdoc IScrollBadge function revokeBadge(Attestation calldata attestation) public returns (bool) { // only callable from resolver if (msg.sender != address(resolver)) { revert Unauthorized(); } // delegate logic to subcontract if (!onRevokeBadge(attestation)) { return false; } _userBadgeCount[attestation.recipient] -= 1; emit RevokeBadge(attestation.uid); return true; } /// @notice A resolver callback that should be implemented by child contracts. /// @param {attestation} The new attestation. /// @return Whether the attestation is valid. function onIssueBadge(Attestation calldata /*attestation*/ ) internal virtual returns (bool) { return true; } /// @notice A resolver callback that should be implemented by child contracts. /// @param {attestation} The existing attestation to be revoked. /// @return Whether the attestation can be revoked. function onRevokeBadge(Attestation calldata /*attestation*/ ) internal virtual returns (bool) { return true; } /// @inheritdoc IScrollBadge function getAndValidateBadge(bytes32 uid) public view returns (Attestation memory) { Attestation memory attestation = IScrollBadgeResolver(resolver).getAndValidateBadge(uid); (address badge,) = decodeBadgeData(attestation.data); if (badge != address(this)) { revert AttestationBadgeMismatch(uid); } return attestation; } /// @inheritdoc IScrollBadge function badgeTokenURI(bytes32 uid) public view virtual returns (string memory); /// @inheritdoc IScrollBadge function hasBadge(address user) public view returns (bool) { return _userBadgeCount[user] > 0; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; error Unauthorized(); // attestation errors // note: these don't include the uid since it is not known prior to the attestation. error BadgeNotAllowed(address badge); error BadgeNotFound(address badge); error ExpirationDisabled(); error MissingPayload(); error ResolverPaymentsDisabled(); error RevocationDisabled(); error SingletonBadge(); error UnknownSchema(); // query errors error AttestationBadgeMismatch(bytes32 uid); error AttestationExpired(bytes32 uid); error AttestationNotFound(bytes32 uid); error AttestationOwnerMismatch(bytes32 uid); error AttestationRevoked(bytes32 uid); error AttestationSchemaMismatch(bytes32 uid); // profile errors error BadgeCountReached(); error LengthMismatch(); error TokenNotOwnedByUser(address token, uint256 tokenId); // profile registry errors error CallerIsNotUserProfile(); error DuplicatedUsername(); error ExpiredSignature(); error ImplementationNotContract(); error InvalidReferrer(); error InvalidSignature(); error InvalidUsername(); error MsgValueMismatchWithMintFee(); error ProfileAlreadyMinted();
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { ISemver } from "./ISemver.sol"; import { ISchemaResolver } from "./resolver/ISchemaResolver.sol"; /// @notice A struct representing a record for a submitted schema. struct SchemaRecord { bytes32 uid; // The unique identifier of the schema. ISchemaResolver resolver; // Optional schema resolver. bool revocable; // Whether the schema allows revocations explicitly. string schema; // Custom specification of the schema (e.g., an ABI). } /// @title ISchemaRegistry /// @notice The interface of global attestation schemas for the Ethereum Attestation Service protocol. interface ISchemaRegistry is ISemver { /// @notice Emitted when a new schema has been registered /// @param uid The schema UID. /// @param registerer The address of the account used to register the schema. /// @param schema The schema data. event Registered(bytes32 indexed uid, address indexed registerer, SchemaRecord schema); /// @notice Submits and reserves a new schema /// @param schema The schema data schema. /// @param resolver An optional schema resolver. /// @param revocable Whether the schema allows revocations explicitly. /// @return The UID of the new schema. function register(string calldata schema, ISchemaResolver resolver, bool revocable) external returns (bytes32); /// @notice Returns an existing schema by UID /// @param uid The UID of the schema to retrieve. /// @return The schema data members. function getSchema(bytes32 uid) external view returns (SchemaRecord memory); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /// @title ISemver /// @notice A semver interface. interface ISemver { /// @notice Returns the full semver contract version. /// @return Semver contract version as a string. function version() external view returns (string memory); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; // A representation of an empty/uninitialized UID. bytes32 constant EMPTY_UID = 0; // A zero expiration represents an non-expiring attestation. uint64 constant NO_EXPIRATION_TIME = 0; error AccessDenied(); error DeadlineExpired(); error InvalidEAS(); error InvalidLength(); error InvalidSignature(); error NotFound(); /// @notice A struct representing ECDSA signature data. struct Signature { uint8 v; // The recovery ID. bytes32 r; // The x-coordinate of the nonce R. bytes32 s; // The signature data. } /// @notice A struct representing a single attestation. struct Attestation { bytes32 uid; // A unique identifier of the attestation. bytes32 schema; // The unique identifier of the schema. uint64 time; // The time when the attestation was created (Unix timestamp). uint64 expirationTime; // The time when the attestation expires (Unix timestamp). uint64 revocationTime; // The time when the attestation was revoked (Unix timestamp). bytes32 refUID; // The UID of the related attestation. address recipient; // The recipient of the attestation. address attester; // The attester/sender of the attestation. bool revocable; // Whether the attestation is revocable. bytes data; // Custom attestation data. } /// @notice A helper function to work with unchecked iterators in loops. function uncheckedInc(uint256 i) pure returns (uint256 j) { unchecked { j = i + 1; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; uint256 constant MAX_ATTACHED_BADGE_NUM = 48; string constant SCROLL_BADGE_SCHEMA = "address badge, bytes payload"; function decodeBadgeData(bytes memory data) pure returns (address, bytes memory) { return abi.decode(data, (address, bytes)); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; interface IScrollBadge { event IssueBadge(bytes32 indexed uid); event RevokeBadge(bytes32 indexed uid); /// @notice A resolver callback invoked in the `issueBadge` function in the parent contract. /// @param attestation The new attestation. /// @return Whether the attestation is valid. function issueBadge(Attestation calldata attestation) external returns (bool); /// @notice A resolver callback invoked in the `revokeBadge` function in the parent contract. /// @param attestation The new attestation. /// @return Whether the attestation can be revoked. function revokeBadge(Attestation calldata attestation) external returns (bool); /// @notice Validate and return a Scroll badge attestation. /// @param uid The attestation UID. /// @return The attestation. function getAndValidateBadge(bytes32 uid) external view returns (Attestation memory); /// @notice Returns the token URI corresponding to a certain badge UID, or the default /// badge token URI if the pass UID is 0x0. /// @param uid The badge UID, or 0x0. /// @return The badge token URI (same format as ERC721). function badgeTokenURI(bytes32 uid) external view returns (string memory); /// @notice Returns true if the user has one or more of this badge. /// @param user The user's wallet address. /// @return True if the user has one or more of this badge. function hasBadge(address user) external view returns (bool); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.19; import {Attestation} from "@eas/contracts/IEAS.sol"; interface IScrollBadgeResolver { /** * * Events * * */ /// @dev Emitted when a new badge is issued. /// @param uid The UID of the new badge attestation. event IssueBadge(bytes32 indexed uid); /// @dev Emitted when a badge is revoked. /// @param uid The UID of the revoked badge attestation. event RevokeBadge(bytes32 indexed uid); /// @dev Emitted when the auto-attach status of a badge is updated. /// @param badge The address of the badge contract. /// @param enable Auto-attach was enabled if true, disabled if false. event UpdateAutoAttachWhitelist(address indexed badge, bool indexed enable); /** * * Public View Functions * * */ /// @notice Return the Scroll badge attestation schema. /// @return The GUID of the Scroll badge attestation schema. function schema() external returns (bytes32); /// @notice The profile registry contract. /// @return The address of the profile registry. function registry() external returns (address); /// @notice The global EAS contract. /// @return The address of the global EAS contract. function eas() external returns (address); /// @notice Validate and return a Scroll badge attestation. /// @param uid The attestation UID. /// @return The attestation. function getAndValidateBadge(bytes32 uid) external view returns (Attestation memory); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import { ISemver } from "../ISemver.sol"; import { Attestation } from "../Common.sol"; /// @title ISchemaResolver /// @notice The interface of an optional schema resolver. interface ISchemaResolver is ISemver { /// @notice Checks if the resolver can be sent ETH. /// @return Whether the resolver supports ETH transfers. function isPayable() external pure returns (bool); /// @notice Processes an attestation and verifies whether it's valid. /// @param attestation The new attestation. /// @return Whether the attestation is valid. function attest(Attestation calldata attestation) external payable returns (bool); /// @notice Processes multiple attestations and verifies whether they are valid. /// @param attestations The new attestations. /// @param values Explicit ETH amounts which were sent with each attestation. /// @return Whether all the attestations are valid. function multiAttest( Attestation[] calldata attestations, uint256[] calldata values ) external payable returns (bool); /// @notice Processes an attestation revocation and verifies if it can be revoked. /// @param attestation The existing attestation to be revoked. /// @return Whether the attestation can be revoked. function revoke(Attestation calldata attestation) external payable returns (bool); /// @notice Processes revocation of multiple attestation and verifies they can be revoked. /// @param attestations The existing attestations to be revoked. /// @param values Explicit ETH amounts which were sent with each revocation. /// @return Whether the attestations can be revoked. function multiRevoke( Attestation[] calldata attestations, uint256[] calldata values ) external payable returns (bool); }
{ "remappings": [ "@eas/=node_modules/@ethereum-attestation-service/eas-contracts/", "@openzeppelin/=node_modules/@openzeppelin/", "solmate/=node_modules/solmate/src/", "ds-test/=lib/forge-std/lib/ds-test/src/", "forge-std/=lib/forge-std/src/" ], "optimizer": { "enabled": true, "runs": 200 }, "metadata": { "useLiteralContent": false, "bytecodeHash": "ipfs", "appendCBOR": true }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "evmVersion": "paris", "viaIR": false, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"resolver_","type":"address"},{"internalType":"address[]","name":"tokens_","type":"address[]"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"bytes32","name":"uid","type":"bytes32"}],"name":"AttestationBadgeMismatch","type":"error"},{"inputs":[],"name":"IncorrectBadgeOwner","type":"error"},{"inputs":[],"name":"MissingPayload","type":"error"},{"inputs":[],"name":"SingletonBadge","type":"error"},{"inputs":[],"name":"Unauthorized","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"uid","type":"bytes32"}],"name":"IssueBadge","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"uid","type":"bytes32"}],"name":"RevokeBadge","type":"event"},{"inputs":[{"internalType":"bytes32","name":"uid","type":"bytes32"}],"name":"badgeTokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"uid","type":"bytes32"}],"name":"getAndValidateBadge","outputs":[{"components":[{"internalType":"bytes32","name":"uid","type":"bytes32"},{"internalType":"bytes32","name":"schema","type":"bytes32"},{"internalType":"uint64","name":"time","type":"uint64"},{"internalType":"uint64","name":"expirationTime","type":"uint64"},{"internalType":"uint64","name":"revocationTime","type":"uint64"},{"internalType":"bytes32","name":"refUID","type":"bytes32"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"address","name":"attester","type":"address"},{"internalType":"bool","name":"revocable","type":"bool"},{"internalType":"bytes","name":"data","type":"bytes"}],"internalType":"struct Attestation","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"bytes32","name":"uid","type":"bytes32"},{"internalType":"bytes32","name":"schema","type":"bytes32"},{"internalType":"uint64","name":"time","type":"uint64"},{"internalType":"uint64","name":"expirationTime","type":"uint64"},{"internalType":"uint64","name":"revocationTime","type":"uint64"},{"internalType":"bytes32","name":"refUID","type":"bytes32"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"address","name":"attester","type":"address"},{"internalType":"bool","name":"revocable","type":"bool"},{"internalType":"bytes","name":"data","type":"bytes"}],"internalType":"struct Attestation","name":"badge","type":"tuple"}],"name":"getPayload","outputs":[{"internalType":"bytes","name":"","type":"bytes"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"getSchema","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"hasBadge","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"isTokenAllowed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"bytes32","name":"uid","type":"bytes32"},{"internalType":"bytes32","name":"schema","type":"bytes32"},{"internalType":"uint64","name":"time","type":"uint64"},{"internalType":"uint64","name":"expirationTime","type":"uint64"},{"internalType":"uint64","name":"revocationTime","type":"uint64"},{"internalType":"bytes32","name":"refUID","type":"bytes32"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"address","name":"attester","type":"address"},{"internalType":"bool","name":"revocable","type":"bool"},{"internalType":"bytes","name":"data","type":"bytes"}],"internalType":"struct Attestation","name":"attestation","type":"tuple"}],"name":"issueBadge","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"resolver","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"bytes32","name":"uid","type":"bytes32"},{"internalType":"bytes32","name":"schema","type":"bytes32"},{"internalType":"uint64","name":"time","type":"uint64"},{"internalType":"uint64","name":"expirationTime","type":"uint64"},{"internalType":"uint64","name":"revocationTime","type":"uint64"},{"internalType":"bytes32","name":"refUID","type":"bytes32"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"address","name":"attester","type":"address"},{"internalType":"bool","name":"revocable","type":"bool"},{"internalType":"bytes","name":"data","type":"bytes"}],"internalType":"struct Attestation","name":"attestation","type":"tuple"}],"name":"revokeBadge","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60a06040523480156200001157600080fd5b5060405162001119380380620011198339810160408190526200003491620000e9565b6001600160a01b03821660805260005b8151811015620000ad576001806000848481518110620000685762000068620001d2565b6020908102919091018101516001600160a01b03168252810191909152604001600020805460ff1916911515919091179055620000a581620001e8565b905062000044565b50505062000210565b80516001600160a01b0381168114620000ce57600080fd5b919050565b634e487b7160e01b600052604160045260246000fd5b60008060408385031215620000fd57600080fd5b6200010883620000b6565b602084810151919350906001600160401b03808211156200012857600080fd5b818601915086601f8301126200013d57600080fd5b815181811115620001525762000152620000d3565b8060051b604051601f19603f830116810181811085821117156200017a576200017a620000d3565b6040529182528482019250838101850191898311156200019957600080fd5b938501935b82851015620001c257620001b285620000b6565b845293850193928501926200019e565b8096505050505050509250929050565b634e487b7160e01b600052603260045260246000fd5b6000600182016200020957634e487b7160e01b600052601160045260246000fd5b5060010190565b608051610ed96200024060003960008181609d0152818161028c015281816103ca01526104920152610ed96000f3fe608060405234801561001057600080fd5b50600436106100935760003560e01c80638298b030116100665780638298b0301461013f5780638c6f12f014610152578063d753a63d14610172578063f15b3bad14610185578063f9eaee0d1461019857600080fd5b806304f3bcec1461009857806324830563146100dc5780635e50864f146100fc5780636b122fe014610137575b600080fd5b6100bf7f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b0390911681526020015b60405180910390f35b6100ef6100ea3660046107f5565b6101bb565b6040516100d3919061085e565b61012761010a3660046108a0565b6001600160a01b0316600090815260208190526040902054151590565b60405190151581526020016100d3565b6100ef61025f565b61012761014d3660046108bd565b61027f565b6101656101603660046107f5565b61035b565b6040516100d391906108f8565b6101276101803660046108bd565b610485565b6100ef610193366004610bb8565b610561565b6101276101a63660046108a0565b60016020526000908152604090205460ff1681565b606060006101c88361035b565b905060006101d582610561565b90506000806101e38361057b565b60405163c87b56dd60e01b81526004810182905291935091506001600160a01b0383169063c87b56dd90602401600060405180830381865afa15801561022d573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526102559190810190610c1c565b9695505050505050565b6060604051806060016040528060258152602001610e7f60259139905090565b6000336001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016146102c9576040516282b42960e81b815260040160405180910390fd5b6102d28261059b565b6102de57506000919050565b60016000806102f360e0860160c087016108a0565b6001600160a01b03166001600160a01b0316815260200190815260200160002060008282546103229190610c7a565b90915550506040518235907f504e4727721de18c6bf7f66448a6ff6da00aa4b1f00b6034e71723ae7ce6373a90600090a2506001919050565b604080516101408101825260008082526020820181905281830181905260608083018290526080830182905260a0830182905260c0830182905260e08301829052610100830182905261012083015291516308c6f12f60e41b8152600481018490529091906001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001690638c6f12f090602401600060405180830381865afa158015610411573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526104399190810190610cce565b9050600061044b8261012001516105ac565b5090506001600160a01b038116301461047e5760405163b923d26160e01b81526004810185905260240160405180910390fd5b5092915050565b6000336001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016146104cf576040516282b42960e81b815260040160405180910390fd5b6104d8826105c4565b6104e457506000919050565b60016000806104f960e0860160c087016108a0565b6001600160a01b03166001600160a01b0316815260200190815260200160002060008282546105289190610dc4565b90915550506040518235907fa0785ec0b9bf31a5475d33c716fb9f500f0ea0bb9e4bc10ec39d5db763c1da1590600090a2506001919050565b606060006105738361012001516105ac565b949350505050565b600080828060200190518101906105929190610dd7565b91509150915091565b60006105a6826106e3565b92915050565b60006060828060200190518101906105929190610e05565b60006105cf826106ee565b6105db57506000919050565b60006105e961019384610e55565b90506000806105f78361057b565b6001600160a01b038216600090815260016020526040902054919350915060ff16610634576040516282b42960e81b815260040160405180910390fd5b61064460e0860160c087016108a0565b6040516331a9108f60e11b8152600481018390526001600160a01b0391821691841690636352211e90602401602060405180830381865afa15801561068d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106b19190610e61565b6001600160a01b0316146106d857604051635d64474f60e11b815260040160405180910390fd5b506001949350505050565b60006105a68261073e565b60006106f982610749565b61070557506000919050565b61071861010a60e0840160c085016108a0565b1561073657604051630c59311960e11b815260040160405180910390fd5b506001919050565b60006105a6826107b0565b6000610754826107b9565b61076057506000919050565b610771610100830160e084016108a0565b6001600160a01b031661078a60e0840160c085016108a0565b6001600160a01b031614610736576040516282b42960e81b815260040160405180910390fd5b600060016105a6565b6000806107c861019384610e55565b905080516000036107ec57604051636de4b34760e01b815260040160405180910390fd5b50600192915050565b60006020828403121561080757600080fd5b5035919050565b60005b83811015610829578181015183820152602001610811565b50506000910152565b6000815180845261084a81602086016020860161080e565b601f01601f19169290920160200192915050565b6020815260006108716020830184610832565b9392505050565b6001600160a01b038116811461088d57600080fd5b50565b803561089b81610878565b919050565b6000602082840312156108b257600080fd5b813561087181610878565b6000602082840312156108cf57600080fd5b81356001600160401b038111156108e557600080fd5b8201610140818503121561087157600080fd5b6020815281516020820152602082015160408201526000604083015161092960608401826001600160401b03169052565b5060608301516001600160401b03811660808401525060808301516001600160401b03811660a08401525060a083015160c083015260c083015161097860e08401826001600160a01b03169052565b5060e0830151610100610995818501836001600160a01b03169052565b84015190506101206109aa8482018315159052565b840151610140848101529050610573610160840182610832565b634e487b7160e01b600052604160045260246000fd5b60405161014081016001600160401b03811182821017156109fd576109fd6109c4565b60405290565b604051601f8201601f191681016001600160401b0381118282101715610a2b57610a2b6109c4565b604052919050565b6001600160401b038116811461088d57600080fd5b803561089b81610a33565b801515811461088d57600080fd5b803561089b81610a53565b60006001600160401b03821115610a8557610a856109c4565b50601f01601f191660200190565b600082601f830112610aa457600080fd5b8135610ab7610ab282610a6c565b610a03565b818152846020838601011115610acc57600080fd5b816020850160208301376000918101602001919091529392505050565b60006101408284031215610afc57600080fd5b610b046109da565b90508135815260208201356020820152610b2060408301610a48565b6040820152610b3160608301610a48565b6060820152610b4260808301610a48565b608082015260a082013560a0820152610b5d60c08301610890565b60c0820152610b6e60e08301610890565b60e0820152610100610b81818401610a61565b90820152610120828101356001600160401b03811115610ba057600080fd5b610bac85828601610a93565b82840152505092915050565b600060208284031215610bca57600080fd5b81356001600160401b03811115610be057600080fd5b61057384828501610ae9565b6000610bfa610ab284610a6c565b9050828152838383011115610c0e57600080fd5b61087183602083018461080e565b600060208284031215610c2e57600080fd5b81516001600160401b03811115610c4457600080fd5b8201601f81018413610c5557600080fd5b61057384825160208401610bec565b634e487b7160e01b600052601160045260246000fd5b818103818111156105a6576105a6610c64565b805161089b81610a33565b805161089b81610878565b805161089b81610a53565b600082601f830112610cbf57600080fd5b61087183835160208501610bec565b600060208284031215610ce057600080fd5b81516001600160401b0380821115610cf757600080fd5b908301906101408286031215610d0c57600080fd5b610d146109da565b8251815260208301516020820152610d2e60408401610c8d565b6040820152610d3f60608401610c8d565b6060820152610d5060808401610c8d565b608082015260a083015160a0820152610d6b60c08401610c98565b60c0820152610d7c60e08401610c98565b60e0820152610100610d8f818501610ca3565b908201526101208381015183811115610da757600080fd5b610db388828701610cae565b918301919091525095945050505050565b808201808211156105a6576105a6610c64565b60008060408385031215610dea57600080fd5b8251610df581610878565b6020939093015192949293505050565b60008060408385031215610e1857600080fd5b8251610e2381610878565b60208401519092506001600160401b03811115610e3f57600080fd5b610e4b85828601610cae565b9150509250929050565b60006105a63683610ae9565b600060208284031215610e7357600080fd5b81516108718161087856fe6164647265737320746f6b656e416464726573732c2075696e7432353620746f6b656e4964a2646970667358221220a91d2b96687a80ee1733c28740a424b6ec3f68a3f21186344720862c5f1b2ba664736f6c634300081300330000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec01130000000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000000000000000000000000000000000000200000000000000000000000074670a3998d9d6622e32d0847ff5977c37e0ec9100000000000000000000000042bcacb8d24ba588cab8db0bb737dd2efca408ec
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106100935760003560e01c80638298b030116100665780638298b0301461013f5780638c6f12f014610152578063d753a63d14610172578063f15b3bad14610185578063f9eaee0d1461019857600080fd5b806304f3bcec1461009857806324830563146100dc5780635e50864f146100fc5780636b122fe014610137575b600080fd5b6100bf7f0000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec011381565b6040516001600160a01b0390911681526020015b60405180910390f35b6100ef6100ea3660046107f5565b6101bb565b6040516100d3919061085e565b61012761010a3660046108a0565b6001600160a01b0316600090815260208190526040902054151590565b60405190151581526020016100d3565b6100ef61025f565b61012761014d3660046108bd565b61027f565b6101656101603660046107f5565b61035b565b6040516100d391906108f8565b6101276101803660046108bd565b610485565b6100ef610193366004610bb8565b610561565b6101276101a63660046108a0565b60016020526000908152604090205460ff1681565b606060006101c88361035b565b905060006101d582610561565b90506000806101e38361057b565b60405163c87b56dd60e01b81526004810182905291935091506001600160a01b0383169063c87b56dd90602401600060405180830381865afa15801561022d573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526102559190810190610c1c565b9695505050505050565b6060604051806060016040528060258152602001610e7f60259139905090565b6000336001600160a01b037f0000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec011316146102c9576040516282b42960e81b815260040160405180910390fd5b6102d28261059b565b6102de57506000919050565b60016000806102f360e0860160c087016108a0565b6001600160a01b03166001600160a01b0316815260200190815260200160002060008282546103229190610c7a565b90915550506040518235907f504e4727721de18c6bf7f66448a6ff6da00aa4b1f00b6034e71723ae7ce6373a90600090a2506001919050565b604080516101408101825260008082526020820181905281830181905260608083018290526080830182905260a0830182905260c0830182905260e08301829052610100830182905261012083015291516308c6f12f60e41b8152600481018490529091906001600160a01b037f0000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec01131690638c6f12f090602401600060405180830381865afa158015610411573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526104399190810190610cce565b9050600061044b8261012001516105ac565b5090506001600160a01b038116301461047e5760405163b923d26160e01b81526004810185905260240160405180910390fd5b5092915050565b6000336001600160a01b037f0000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec011316146104cf576040516282b42960e81b815260040160405180910390fd5b6104d8826105c4565b6104e457506000919050565b60016000806104f960e0860160c087016108a0565b6001600160a01b03166001600160a01b0316815260200190815260200160002060008282546105289190610dc4565b90915550506040518235907fa0785ec0b9bf31a5475d33c716fb9f500f0ea0bb9e4bc10ec39d5db763c1da1590600090a2506001919050565b606060006105738361012001516105ac565b949350505050565b600080828060200190518101906105929190610dd7565b91509150915091565b60006105a6826106e3565b92915050565b60006060828060200190518101906105929190610e05565b60006105cf826106ee565b6105db57506000919050565b60006105e961019384610e55565b90506000806105f78361057b565b6001600160a01b038216600090815260016020526040902054919350915060ff16610634576040516282b42960e81b815260040160405180910390fd5b61064460e0860160c087016108a0565b6040516331a9108f60e11b8152600481018390526001600160a01b0391821691841690636352211e90602401602060405180830381865afa15801561068d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106b19190610e61565b6001600160a01b0316146106d857604051635d64474f60e11b815260040160405180910390fd5b506001949350505050565b60006105a68261073e565b60006106f982610749565b61070557506000919050565b61071861010a60e0840160c085016108a0565b1561073657604051630c59311960e11b815260040160405180910390fd5b506001919050565b60006105a6826107b0565b6000610754826107b9565b61076057506000919050565b610771610100830160e084016108a0565b6001600160a01b031661078a60e0840160c085016108a0565b6001600160a01b031614610736576040516282b42960e81b815260040160405180910390fd5b600060016105a6565b6000806107c861019384610e55565b905080516000036107ec57604051636de4b34760e01b815260040160405180910390fd5b50600192915050565b60006020828403121561080757600080fd5b5035919050565b60005b83811015610829578181015183820152602001610811565b50506000910152565b6000815180845261084a81602086016020860161080e565b601f01601f19169290920160200192915050565b6020815260006108716020830184610832565b9392505050565b6001600160a01b038116811461088d57600080fd5b50565b803561089b81610878565b919050565b6000602082840312156108b257600080fd5b813561087181610878565b6000602082840312156108cf57600080fd5b81356001600160401b038111156108e557600080fd5b8201610140818503121561087157600080fd5b6020815281516020820152602082015160408201526000604083015161092960608401826001600160401b03169052565b5060608301516001600160401b03811660808401525060808301516001600160401b03811660a08401525060a083015160c083015260c083015161097860e08401826001600160a01b03169052565b5060e0830151610100610995818501836001600160a01b03169052565b84015190506101206109aa8482018315159052565b840151610140848101529050610573610160840182610832565b634e487b7160e01b600052604160045260246000fd5b60405161014081016001600160401b03811182821017156109fd576109fd6109c4565b60405290565b604051601f8201601f191681016001600160401b0381118282101715610a2b57610a2b6109c4565b604052919050565b6001600160401b038116811461088d57600080fd5b803561089b81610a33565b801515811461088d57600080fd5b803561089b81610a53565b60006001600160401b03821115610a8557610a856109c4565b50601f01601f191660200190565b600082601f830112610aa457600080fd5b8135610ab7610ab282610a6c565b610a03565b818152846020838601011115610acc57600080fd5b816020850160208301376000918101602001919091529392505050565b60006101408284031215610afc57600080fd5b610b046109da565b90508135815260208201356020820152610b2060408301610a48565b6040820152610b3160608301610a48565b6060820152610b4260808301610a48565b608082015260a082013560a0820152610b5d60c08301610890565b60c0820152610b6e60e08301610890565b60e0820152610100610b81818401610a61565b90820152610120828101356001600160401b03811115610ba057600080fd5b610bac85828601610a93565b82840152505092915050565b600060208284031215610bca57600080fd5b81356001600160401b03811115610be057600080fd5b61057384828501610ae9565b6000610bfa610ab284610a6c565b9050828152838383011115610c0e57600080fd5b61087183602083018461080e565b600060208284031215610c2e57600080fd5b81516001600160401b03811115610c4457600080fd5b8201601f81018413610c5557600080fd5b61057384825160208401610bec565b634e487b7160e01b600052601160045260246000fd5b818103818111156105a6576105a6610c64565b805161089b81610a33565b805161089b81610878565b805161089b81610a53565b600082601f830112610cbf57600080fd5b61087183835160208501610bec565b600060208284031215610ce057600080fd5b81516001600160401b0380821115610cf757600080fd5b908301906101408286031215610d0c57600080fd5b610d146109da565b8251815260208301516020820152610d2e60408401610c8d565b6040820152610d3f60608401610c8d565b6060820152610d5060808401610c8d565b608082015260a083015160a0820152610d6b60c08401610c98565b60c0820152610d7c60e08401610c98565b60e0820152610100610d8f818501610ca3565b908201526101208381015183811115610da757600080fd5b610db388828701610cae565b918301919091525095945050505050565b808201808211156105a6576105a6610c64565b60008060408385031215610dea57600080fd5b8251610df581610878565b6020939093015192949293505050565b60008060408385031215610e1857600080fd5b8251610e2381610878565b60208401519092506001600160401b03811115610e3f57600080fd5b610e4b85828601610cae565b9150509250929050565b60006105a63683610ae9565b600060208284031215610e7357600080fd5b81516108718161087856fe6164647265737320746f6b656e416464726573732c2075696e7432353620746f6b656e4964a2646970667358221220a91d2b96687a80ee1733c28740a424b6ec3f68a3f21186344720862c5f1b2ba664736f6c63430008130033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
0000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec01130000000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000000000000000000000000000000000000200000000000000000000000074670a3998d9d6622e32d0847ff5977c37e0ec9100000000000000000000000042bcacb8d24ba588cab8db0bb737dd2efca408ec
-----Decoded View---------------
Arg [0] : resolver_ (address): 0x4560FECd62B14A463bE44D40fE5Cfd595eEc0113
Arg [1] : tokens_ (address[]): 0x74670A3998d9d6622E32D0847fF5977c37E0eC91,0x42bCaCb8D24Ba588cab8Db0BB737DD2eFca408EC
-----Encoded View---------------
5 Constructor Arguments found :
Arg [0] : 0000000000000000000000004560fecd62b14a463be44d40fe5cfd595eec0113
Arg [1] : 0000000000000000000000000000000000000000000000000000000000000040
Arg [2] : 0000000000000000000000000000000000000000000000000000000000000002
Arg [3] : 00000000000000000000000074670a3998d9d6622e32d0847ff5977c37e0ec91
Arg [4] : 00000000000000000000000042bcacb8d24ba588cab8db0bb737dd2efca408ec
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.