ETH Price: $1,819.13 (+0.41%)

Transaction Decoder

Block:
16775190 at Mar-07-2023 08:18:35 AM +UTC
Transaction Fee:
0.002426164342832064 ETH $4.41
Gas Used:
109,443 Gas / 22.168291648 Gwei

Emitted Events:

248 Collection.Transfer( from=0x00000000...000000000, to=[Sender] 0x58b4c61b7a320607737e347a5c00dd0b26f28f36, id=1263 )
249 Collection.RecordCreated( tokenId=1263, sequenceId=1, data=1251 )
250 SplitWallet.ReceiveETH( split=SplitWallet, amount=50000000000000000 )

Account State Difference:

  Address   Before After State Difference Code
0x363C5dC3...48e10f567
0x58b4c61B...b26F28f36
0.098979181160227 Eth
Nonce: 2
0.046553016817394936 Eth
Nonce: 3
0.052426164342832064
(beaverbuild)
132.24590273729492266 Eth132.24593009804492266 Eth0.00002736075
0xBE36007b...5AE99494F 67.38183305555555555 Eth67.43183305555555555 Eth0.05

Execution Trace

ETH 0.05 DropEngineV2.mint( collection=0x363C5dC3ff5A93C9AB1ec54337d211148e10f567, sequenceId=1, count=1 ) => ( tokenId=1263 )
  • Collection.mintRecord( to=0x58b4c61B7a320607737E347a5C00DD0b26F28f36, sequenceId=1 ) => ( tokenId=1263 )
    • Collection.mintRecord( to=0x58b4c61B7a320607737E347a5C00DD0b26F28f36, sequenceId=1 ) => ( tokenId=1263 )
    • ETH 0.05 SplitWallet.CALL( )
      File 1 of 4: DropEngineV2
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Simple single owner authorization mixin.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/Owned.sol)
      abstract contract Owned {
          /*//////////////////////////////////////////////////////////////
                                       EVENTS
          //////////////////////////////////////////////////////////////*/
          event OwnershipTransferred(address indexed user, address indexed newOwner);
          /*//////////////////////////////////////////////////////////////
                                  OWNERSHIP STORAGE
          //////////////////////////////////////////////////////////////*/
          address public owner;
          modifier onlyOwner() virtual {
              require(msg.sender == owner, "UNAUTHORIZED");
              _;
          }
          /*//////////////////////////////////////////////////////////////
                                     CONSTRUCTOR
          //////////////////////////////////////////////////////////////*/
          constructor(address _owner) {
              owner = _owner;
              emit OwnershipTransferred(address(0), _owner);
          }
          /*//////////////////////////////////////////////////////////////
                                   OWNERSHIP LOGIC
          //////////////////////////////////////////////////////////////*/
          function transferOwnership(address newOwner) public virtual onlyOwner {
              owner = newOwner;
              emit OwnershipTransferred(msg.sender, newOwner);
          }
      }
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Read and write to persistent storage at a fraction of the cost.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SSTORE2.sol)
      /// @author Modified from 0xSequence (https://github.com/0xSequence/sstore2/blob/master/contracts/SSTORE2.sol)
      library SSTORE2 {
          uint256 internal constant DATA_OFFSET = 1; // We skip the first byte as it's a STOP opcode to ensure the contract can't be called.
          /*//////////////////////////////////////////////////////////////
                                     WRITE LOGIC
          //////////////////////////////////////////////////////////////*/
          function write(bytes memory data) internal returns (address pointer) {
              // Prefix the bytecode with a STOP opcode to ensure it cannot be called.
              bytes memory runtimeCode = abi.encodePacked(hex"00", data);
              bytes memory creationCode = abi.encodePacked(
                  //---------------------------------------------------------------------------------------------------------------//
                  // Opcode  | Opcode + Arguments  | Description  | Stack View                                                     //
                  //---------------------------------------------------------------------------------------------------------------//
                  // 0x60    |  0x600B             | PUSH1 11     | codeOffset                                                     //
                  // 0x59    |  0x59               | MSIZE        | 0 codeOffset                                                   //
                  // 0x81    |  0x81               | DUP2         | codeOffset 0 codeOffset                                        //
                  // 0x38    |  0x38               | CODESIZE     | codeSize codeOffset 0 codeOffset                               //
                  // 0x03    |  0x03               | SUB          | (codeSize - codeOffset) 0 codeOffset                           //
                  // 0x80    |  0x80               | DUP          | (codeSize - codeOffset) (codeSize - codeOffset) 0 codeOffset   //
                  // 0x92    |  0x92               | SWAP3        | codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset)   //
                  // 0x59    |  0x59               | MSIZE        | 0 codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset) //
                  // 0x39    |  0x39               | CODECOPY     | 0 (codeSize - codeOffset)                                      //
                  // 0xf3    |  0xf3               | RETURN       |                                                                //
                  //---------------------------------------------------------------------------------------------------------------//
                  hex"60_0B_59_81_38_03_80_92_59_39_F3", // Returns all code in the contract except for the first 11 (0B in hex) bytes.
                  runtimeCode // The bytecode we want the contract to have after deployment. Capped at 1 byte less than the code size limit.
              );
              assembly {
                  // Deploy a new contract with the generated creation code.
                  // We start 32 bytes into the code to avoid copying the byte length.
                  pointer := create(0, add(creationCode, 32), mload(creationCode))
              }
              require(pointer != address(0), "DEPLOYMENT_FAILED");
          }
          /*//////////////////////////////////////////////////////////////
                                     READ LOGIC
          //////////////////////////////////////////////////////////////*/
          function read(address pointer) internal view returns (bytes memory) {
              return readBytecode(pointer, DATA_OFFSET, pointer.code.length - DATA_OFFSET);
          }
          function read(address pointer, uint256 start) internal view returns (bytes memory) {
              start += DATA_OFFSET;
              return readBytecode(pointer, start, pointer.code.length - start);
          }
          function read(
              address pointer,
              uint256 start,
              uint256 end
          ) internal view returns (bytes memory) {
              start += DATA_OFFSET;
              end += DATA_OFFSET;
              require(pointer.code.length >= end, "OUT_OF_BOUNDS");
              return readBytecode(pointer, start, end - start);
          }
          /*//////////////////////////////////////////////////////////////
                                INTERNAL HELPER LOGIC
          //////////////////////////////////////////////////////////////*/
          function readBytecode(
              address pointer,
              uint256 start,
              uint256 size
          ) private view returns (bytes memory data) {
              assembly {
                  // Get a pointer to some free memory.
                  data := mload(0x40)
                  // Update the free memory pointer to prevent overriding our data.
                  // We use and(x, not(31)) as a cheaper equivalent to sub(x, mod(x, 32)).
                  // Adding 31 to size and running the result through the logic above ensures
                  // the memory pointer remains word-aligned, following the Solidity convention.
                  mstore(0x40, add(data, and(add(add(size, 32), 31), not(31))))
                  // Store the size of the data in the first 32 byte chunk of free memory.
                  mstore(data, size)
                  // Copy the code into memory right after the 32 bytes we used to store the size.
                  extcodecopy(pointer, add(data, 32), start, size)
              }
          }
      }
      // SPDX-License-Identifier: MIT
      // OpenZeppelin Contracts (last updated v4.7.0) (utils/Base64.sol)
      pragma solidity ^0.8.0;
      /**
       * @dev Provides a set of functions to operate with Base64 strings.
       *
       * _Available since v4.5._
       */
      library Base64 {
          /**
           * @dev Base64 Encoding/Decoding Table
           */
          string internal constant _TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
          /**
           * @dev Converts a `bytes` to its Bytes64 `string` representation.
           */
          function encode(bytes memory data) internal pure returns (string memory) {
              /**
               * Inspired by Brecht Devos (Brechtpd) implementation - MIT licence
               * https://github.com/Brechtpd/base64/blob/e78d9fd951e7b0977ddca77d92dc85183770daf4/base64.sol
               */
              if (data.length == 0) return "";
              // Loads the table into memory
              string memory table = _TABLE;
              // Encoding takes 3 bytes chunks of binary data from `bytes` data parameter
              // and split into 4 numbers of 6 bits.
              // The final Base64 length should be `bytes` data length multiplied by 4/3 rounded up
              // - `data.length + 2`  -> Round up
              // - `/ 3`              -> Number of 3-bytes chunks
              // - `4 *`              -> 4 characters for each chunk
              string memory result = new string(4 * ((data.length + 2) / 3));
              /// @solidity memory-safe-assembly
              assembly {
                  // Prepare the lookup table (skip the first "length" byte)
                  let tablePtr := add(table, 1)
                  // Prepare result pointer, jump over length
                  let resultPtr := add(result, 32)
                  // Run over the input, 3 bytes at a time
                  for {
                      let dataPtr := data
                      let endPtr := add(data, mload(data))
                  } lt(dataPtr, endPtr) {
                  } {
                      // Advance 3 bytes
                      dataPtr := add(dataPtr, 3)
                      let input := mload(dataPtr)
                      // To write each character, shift the 3 bytes (18 bits) chunk
                      // 4 times in blocks of 6 bits for each character (18, 12, 6, 0)
                      // and apply logical AND with 0x3F which is the number of
                      // the previous character in the ASCII table prior to the Base64 Table
                      // The result is then added to the table to get the character to write,
                      // and finally write it in the result pointer but with a left shift
                      // of 256 (1 byte) - 8 (1 ASCII char) = 248 bits
                      mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))
                      resultPtr := add(resultPtr, 1) // Advance
                      mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))
                      resultPtr := add(resultPtr, 1) // Advance
                      mstore8(resultPtr, mload(add(tablePtr, and(shr(6, input), 0x3F))))
                      resultPtr := add(resultPtr, 1) // Advance
                      mstore8(resultPtr, mload(add(tablePtr, and(input, 0x3F))))
                      resultPtr := add(resultPtr, 1) // Advance
                  }
                  // When data `bytes` is not exactly 3 bytes long
                  // it is padded with `=` characters at the end
                  switch mod(mload(data), 3)
                  case 1 {
                      mstore8(sub(resultPtr, 1), 0x3d)
                      mstore8(sub(resultPtr, 2), 0x3d)
                  }
                  case 2 {
                      mstore8(sub(resultPtr, 1), 0x3d)
                  }
              }
              return result;
          }
      }
      // SPDX-License-Identifier: MIT
      // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)
      pragma solidity ^0.8.0;
      /**
       * @dev String operations.
       */
      library Strings {
          bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
          uint8 private constant _ADDRESS_LENGTH = 20;
          /**
           * @dev Converts a `uint256` to its ASCII `string` decimal representation.
           */
          function toString(uint256 value) internal pure returns (string memory) {
              // Inspired by OraclizeAPI's implementation - MIT licence
              // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol
              if (value == 0) {
                  return "0";
              }
              uint256 temp = value;
              uint256 digits;
              while (temp != 0) {
                  digits++;
                  temp /= 10;
              }
              bytes memory buffer = new bytes(digits);
              while (value != 0) {
                  digits -= 1;
                  buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
                  value /= 10;
              }
              return string(buffer);
          }
          /**
           * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
           */
          function toHexString(uint256 value) internal pure returns (string memory) {
              if (value == 0) {
                  return "0x00";
              }
              uint256 temp = value;
              uint256 length = 0;
              while (temp != 0) {
                  length++;
                  temp >>= 8;
              }
              return toHexString(value, length);
          }
          /**
           * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
           */
          function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
              bytes memory buffer = new bytes(2 * length + 2);
              buffer[0] = "0";
              buffer[1] = "x";
              for (uint256 i = 2 * length + 1; i > 1; --i) {
                  buffer[i] = _HEX_SYMBOLS[value & 0xf];
                  value >>= 4;
              }
              require(value == 0, "Strings: hex length insufficient");
              return string(buffer);
          }
          /**
           * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.
           */
          function toHexString(address addr) internal pure returns (string memory) {
              return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
          }
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      /*
      ███╗   ███╗███████╗████████╗ █████╗ ██╗      █████╗ ██████╗ ███████╗██╗
      ████╗ ████║██╔════╝╚══██╔══╝██╔══██╗██║     ██╔══██╗██╔══██╗██╔════╝██║
      ██╔████╔██║█████╗     ██║   ███████║██║     ███████║██████╔╝█████╗  ██║
      ██║╚██╔╝██║██╔══╝     ██║   ██╔══██║██║     ██╔══██║██╔══██╗██╔══╝  ██║
      ██║ ╚═╝ ██║███████╗   ██║   ██║  ██║███████╗██║  ██║██████╔╝███████╗███████╗
      ╚═╝     ╚═╝╚══════╝   ╚═╝   ╚═╝  ╚═╝╚══════╝╚═╝  ╚═╝╚═════╝ ╚══════╝╚══════╝
      Deployed by Metalabel with 💖 as a permanent application on the Ethereum blockchain.
      Metalabel is a growing universe of tools, knowledge, and resources for
      metalabels and cultural collectives.
      Our purpose is to establish the metalabel as key infrastructure for creative
      collectives and to inspire a new culture of creative collaboration and mutual
      support.
      OUR SQUAD
      Anna Bulbrook (Curator)
      Austin Robey (Community)
      Brandon Valosek (Engineer)
      Ilya Yudanov (Designer)
      Lauren Dorman (Engineer)
      Rob Kalin (Board)
      Yancey Strickler (Director)
      https://metalabel.xyz
      */
      import {Owned} from "@metalabel/solmate/src/auth/Owned.sol";
      import {SSTORE2} from "@metalabel/solmate/src/utils/SSTORE2.sol";
      import {Base64} from "@openzeppelin/contracts/utils/Base64.sol";
      import {Strings} from "@openzeppelin/contracts/utils/Strings.sol";
      import {ICollection} from "../interfaces/ICollection.sol";
      import {IEngine, SequenceData} from "../interfaces/IEngine.sol";
      import {INodeRegistry} from "../interfaces/INodeRegistry.sol";
      /// @notice Data stored engine-side for each drop.
      /// - Royalty percentage is stored as basis points, eg 5% = 500
      /// - If maxRecordPerTransaction is 0, there is not limit
      /// - Protocol fee is written to drop data at configure-time to lock in protocol
      ///   fee and avoid an additional storage read at mint-time
      struct DropData {
          uint96 price;
          uint16 royaltyBps;
          bool allowContractMints;
          bool randomizeMetadataVariants;
          uint8 maxRecordsPerTransaction;
          address revenueRecipient;
          uint16 primarySaleFeeBps;
          uint96 priceDecayPerDay;
          uint64 decayStopTimestamp;
          // 59 bytes total / 5 remaining for a two-word slot
      }
      /// @notice A single attribute of an NFT's metadata
      struct NFTMetadataAttribute {
          string trait_type;
          string value;
      }
      /// @notice Metadata stored for a single record variant
      /// @dev Storage is written via SSTORE2
      struct NFTMetadata {
          string name;
          string description;
          string image;
          string external_url;
          string metalabel_record_variant_name;
          string metalabel_release_metadata_uri;
          uint16[] metalabel_record_contents;
          NFTMetadataAttribute[] attributes;
      }
      /// @notice Metalabel engine that implements a multi-NFT drop.
      /// - All metadata is stored onchain via SSTORE2.
      /// - Price can decay over time or be constant throughout the drop.
      /// - Metadata variants can be p-randomized or fixed.
      /// - Enabling or disabling smart contract mints is set per-sequence.
      /// - Multiple records can be minted in a single trx, configurable per-sequence.
      /// - The owner of this contract can set a primary sale fee that is taken from
      ///   all primary sales revenue and retained by this drop engine.
      contract DropEngineV2 is IEngine, Owned {
          // ---
          // Errors
          // ---
          /// @notice Invalid msg.value on purchase
          error IncorrectPaymentAmount();
          /// @notice If price or recipient is zero, they both have to be zero
          error InvalidPriceOrRecipient();
          /// @notice An invalid value was used for the royalty bps.
          error InvalidRoyaltyBps();
          /// @notice An invalid value was used for the primary sale fee.
          error InvalidPrimarySaleFee();
          /// @notice If smart contract mints are not allowed, msg.sender must be an
          /// EOA
          error MinterMustBeEOA();
          /// @notice If minting more than the max allowed per transaction
          error InvalidMintAmount();
          /// @notice An invalid price decay stop time or per day decay was used.
          error InvalidPriceDecayConfig();
          /// @notice Unable to forward ETH to the revenue recipient or unable to
          /// withdraw funds
          error CouldNotTransferEth();
          // ---
          // Events
          // ---
          /// @notice A new drop was created.
          /// @dev The collection already emits a SequenceCreated event, we're
          /// emitting the additional engine-specific data here.
          event DropCreated(address collection, uint16 sequenceId, DropData dropData);
          /// @notice The primary sale for this drop engine was set
          event PrimarySaleFeeSet(uint16 primarySaleFeeBps);
          // ---
          // Storage
          // ---
          /// @notice Drop data for a given collection + sequence ID.
          mapping(address => mapping(uint16 => DropData)) public drops;
          /// @notice The SSTORE2 contract storage address for a given sequence's list
          /// of metadata variants
          mapping(address => mapping(uint16 => address))
              public metadataStoragePointers;
          /// @notice A primary sales fee that is paid at mint time. Can be adjusted
          /// by contract owner. Fee is written into the drop's DropData structure, so
          /// fee at configure-time is locked. Fees are accumulated in the contract
          /// and can be withdrawn by the contract owner
          uint16 public primarySaleFeeBps;
          /// @notice A reference to the core protocol's node registry.
          /// @dev While this is not directly used by the engine, it is surfaced in
          /// the onchain generated JSON metadata for records as a way of creating a
          /// concrete link back to the cataloging protocol.
          INodeRegistry public immutable nodeRegistry;
          // ---
          // Constructor
          // ---
          constructor(address _contractOwner, INodeRegistry _nodeRegistry)
              Owned(_contractOwner)
          {
              nodeRegistry = _nodeRegistry;
          }
          // ---
          // Admin functionality
          // ---
          /// @notice Set the primary sale fee for all drops configured on this
          /// engine. Only callable by owner
          function setPrimarySaleFeeBps(uint16 fee) external onlyOwner {
              if (fee > 10000) revert InvalidPrimarySaleFee();
              primarySaleFeeBps = fee;
              emit PrimarySaleFeeSet(fee);
          }
          // ---
          // Permissionless functions
          // ---
          /// @notice Transfer ETH from the contract that has accumulated from fees to
          /// the owner's account. Can be called by any address.
          function transferFeesToOwner() external {
              (bool success, ) = owner.call{value: address(this).balance}("");
              if (!success) revert CouldNotTransferEth();
          }
          // ---
          // Mint functionality
          // ---
          /// @notice Mint records. Returns the first token ID minted
          function mint(
              ICollection collection,
              uint16 sequenceId,
              uint8 count
          ) external payable returns (uint256 tokenId) {
              DropData storage drop = drops[address(collection)][sequenceId];
              // block SC mints if flagged
              if (!drop.allowContractMints && msg.sender != tx.origin) {
                  revert MinterMustBeEOA();
              }
              // Ensure not minting too many
              if (
                  drop.maxRecordsPerTransaction > 0 &&
                  count > drop.maxRecordsPerTransaction
              ) {
                  revert InvalidMintAmount();
              }
              // Resolve current unit price (which may change over time if there's a
              // price decay configuration) and total order price
              uint256 unitPrice = currentPrice(collection, sequenceId);
              uint256 orderPrice = unitPrice * count;
              // Ensure correct payment was sent with the transaction. Checking less
              // than to allow sender to overpay (likely happens for all decaying
              // prices). We refund the difference below.
              if (msg.value < orderPrice) {
                  revert IncorrectPaymentAmount();
              }
              for (uint256 i = 0; i < count; i++) {
                  // If collection is a malicious contract, that does not impact any
                  // state in the engine.  If it's a valid protocol-deployed
                  // collection, then it will work as expected.
                  //
                  // Collection enforces max mint supply and mint window, so we're not
                  // checking that here
                  uint256 id = collection.mintRecord(msg.sender, sequenceId);
                  // return the first minted token ID, caller can infer subsequent
                  // sequential IDs
                  tokenId = tokenId != 0 ? tokenId : id;
              }
              // Amount to forward to the revenue recipient is the total order price
              // minus the locked-in primary sale fee that was recorded at
              // configure-time.  The remaining ETH (after refund) will stay in this
              // contract, withdrawable by the owner at a later date via
              // transferFeesToOwner
              uint256 amountToForward = orderPrice -
                  ((orderPrice * drop.primarySaleFeeBps) / 10000);
              // Amount to refund message sender is any difference in order price and
              // msg.value. This happens if the caller overpays, which will generally
              // always happen on decaying price mints
              uint256 amountToRefund = msg.value > orderPrice
                  ? msg.value - orderPrice
                  : 0;
              // Refund caller
              if (amountToRefund > 0) {
                  (bool success, ) = msg.sender.call{value: amountToRefund}("");
                  if (!success) revert CouldNotTransferEth();
              }
              // Forward ETH to the revenue recipient
              if (amountToForward > 0) {
                  (bool success, ) = drop.revenueRecipient.call{
                      value: amountToForward
                  }("");
                  if (!success) revert CouldNotTransferEth();
              }
          }
          /// @notice Get the current price of a record in a given sequence. This will
          /// return a price even if the sequence is not currently mintable (i.e. the
          /// mint window hasn't started yet or the minting window has closed).
          function currentPrice(ICollection collection, uint16 sequenceId)
              public
              view
              returns (uint256 unitPrice)
          {
              DropData storage drop = drops[address(collection)][sequenceId];
              // Compute unit price based on decay and timestamp.
              // First compute how many seconds until the decay cutoff time, after
              // which price will remain constant. Then compute the marginal increase
              // in unit price by multiplying the base price by
              //
              //   (decay per day * seconds until decay stop) / 1 day
              //
              uint64 secondsBeforeDecayStop = block.timestamp <
                  drop.decayStopTimestamp
                  ? drop.decayStopTimestamp - uint64(block.timestamp)
                  : 0;
              uint256 inflateUnitPriceBy = (uint256(drop.priceDecayPerDay) *
                  secondsBeforeDecayStop) / 1 days;
              unitPrice = drop.price + inflateUnitPriceBy;
          }
          // ---
          // IEngine setup
          // ---
          /// @inheritdoc IEngine
          /// @dev There is no access control on this function, we infer the
          /// collection from msg.sender, and use that to key all stored data. If
          /// somebody calls this function with bogus info (instead of it getting
          /// called via the collection), it just wastes storage but does not impact
          /// contract functionality
          function configureSequence(
              uint16 sequenceId,
              SequenceData calldata sequenceData,
              bytes calldata engineData
          ) external override {
              (DropData memory dropData, NFTMetadata[] memory metadatas) = abi.decode(
                  engineData,
                  (DropData, NFTMetadata[])
              );
              // This drop is a "free drop" if and only if the price is zero and decay
              // per day is zero
              bool isFreeDrop = dropData.price == 0 && dropData.priceDecayPerDay == 0;
              // Ensure that if this is a free drop, there's no revenue recipient, and
              // vice versa
              if ((isFreeDrop) != (dropData.revenueRecipient == address(0))) {
                  revert InvalidPriceOrRecipient();
              }
              // Don't allow setting a decay stop time in the past (or before the mint
              // window opens) unless it's zero.
              if (
                  dropData.decayStopTimestamp != 0 &&
                  (dropData.decayStopTimestamp < block.timestamp ||
                      dropData.decayStopTimestamp <
                      sequenceData.sealedBeforeTimestamp)
              ) {
                  revert InvalidPriceDecayConfig();
              }
              // Don't allow setting a decay stop time after the mint window closes
              if (
                  sequenceData.sealedAfterTimestamp > 0 && // sealed = 0 -> no end
                  dropData.decayStopTimestamp > sequenceData.sealedAfterTimestamp
              ) {
                  revert InvalidPriceDecayConfig();
              }
              // Ensure that if decay stop time is set, decay per day is set, and vice
              // versa
              if (
                  (dropData.decayStopTimestamp == 0) !=
                  (dropData.priceDecayPerDay == 0)
              ) {
                  revert InvalidPriceDecayConfig();
              }
              // Ensure royaltyBps is in range
              if (dropData.royaltyBps > 10000) revert InvalidRoyaltyBps();
              // To ensure that creators know the protocol fee they are effectively
              // agreeing to during sequence creation time, we require that they set
              // the primary sale fee correctly here. This also ensures the drop
              // engine owner cannot frontrun a fee change
              if (dropData.primarySaleFeeBps != primarySaleFeeBps) {
                  revert InvalidPrimarySaleFee();
              }
              // write metadata blob to chain
              metadataStoragePointers[msg.sender][sequenceId] = SSTORE2.write(
                  abi.encode(metadatas)
              );
              // Write engine data (passed through from the collection when the
              // collection admin calls `configureSequence`) to a struct in the engine
              // with all the needed info.
              drops[msg.sender][sequenceId] = dropData;
              emit DropCreated(msg.sender, sequenceId, dropData);
          }
          // ---
          // IEngine views
          // ---
          /// @inheritdoc IEngine
          /// @dev Token URI is constructed programmatically from stored metadata by
          /// creating the JSON string and base64ing it
          function getTokenURI(address collection, uint256 tokenId)
              external
              view
              override
              returns (string memory tokenURI)
          {
              uint16 sequenceId = ICollection(collection).tokenSequenceId(tokenId);
              uint80 editionNumber = ICollection(collection).tokenMintData(tokenId);
              SequenceData memory sequenceData = ICollection(collection)
                  .getSequenceData(sequenceId);
              NFTMetadata memory metadata = getStoredMetadataVariant(
                  collection,
                  tokenId
              );
              // Construct edition string as either "1" or "1/1000" depending on if
              // this was an open edition
              string memory sEdition = sequenceData.maxSupply == 0
                  ? Strings.toString(editionNumber)
                  : string.concat(
                      Strings.toString(editionNumber),
                      "/",
                      Strings.toString(sequenceData.maxSupply)
                  );
              // Edition number and variant name are always included
              string memory attributesInnerJson = string.concat(
                  '{"trait_type": "Record Edition", "value": "',
                  sEdition,
                  '"}, {"trait_type": "Record Variant", "value": "',
                  metadata.metalabel_record_variant_name,
                  '"}',
                  metadata.attributes.length > 0 ? ", " : ""
              );
              // Additional attributes from metadata blob
              for (uint256 i = 0; i < metadata.attributes.length; i++) {
                  attributesInnerJson = string.concat(
                      attributesInnerJson,
                      i > 0 ? ", " : "",
                      '{"trait_type": "',
                      metadata.attributes[i].trait_type,
                      '", "value": "',
                      metadata.attributes[i].value,
                      '"}'
                  );
              }
              // create the contents array
              string memory contentsInnerJson = "[";
              for (
                  uint256 i = 0;
                  i < metadata.metalabel_record_contents.length;
                  i++
              ) {
                  contentsInnerJson = string.concat(
                      contentsInnerJson,
                      Strings.toString(metadata.metalabel_record_contents[i]),
                      i == metadata.metalabel_record_contents.length - 1 ? "]" : ", "
                  );
              }
              // Compose the final JSON payload. Split across multiple string.concat
              // calls due to stack limitations
              string memory json = string.concat(
                  '{"name":"',
                  metadata.name,
                  " ",
                  sEdition,
                  '", "description":"',
                  metadata.description,
                  '", "image": "',
                  metadata.image,
                  '", "external_url": "',
                  metadata.external_url,
                  '", '
              );
              json = string.concat(
                  json,
                  '"metalabel": { "node_registry_address": "',
                  Strings.toHexString(uint256(uint160(address(nodeRegistry))), 20),
                  '", "record_variant_name": "',
                  metadata.metalabel_record_variant_name,
                  '", "release_metadata_uri": "',
                  metadata.metalabel_release_metadata_uri,
                  '", "record_contents": ',
                  contentsInnerJson,
                  '}, "attributes": [',
                  attributesInnerJson,
                  "]}"
              );
              // Prepend base64 prefix + encode JSON
              tokenURI = string.concat(
                  "data:application/json;base64,",
                  Base64.encode(bytes(json))
              );
          }
          /// @notice Get the onchain metadata variant for a specific record
          /// @dev This is a view function that reads from SSTORE2 storage and picks
          /// the random or sequential variant, the full onchain metadata is
          /// generated in tokenURI
          function getStoredMetadataVariant(address collection, uint256 tokenId)
              public
              view
              returns (NFTMetadata memory metadata)
          {
              uint16 sequenceId = ICollection(collection).tokenSequenceId(tokenId);
              uint80 editionNumber = ICollection(collection).tokenMintData(tokenId);
              // Load all metadata variants from SSTORE2 storage
              NFTMetadata[] memory metadatas = abi.decode(
                  SSTORE2.read(metadataStoragePointers[collection][sequenceId]),
                  (NFTMetadata[])
              );
              // Metadata variants are default sequential, but can be pseudo-random
              // if the randomizeMetadataVariants flag is set.
              // Using (edition - 1) for sequential since edition number starts at 1
              uint256 idx = (editionNumber - 1) % metadatas.length;
              if (drops[collection][sequenceId].randomizeMetadataVariants) {
                  idx =
                      uint256(
                          keccak256(
                              abi.encodePacked(
                                  collection,
                                  sequenceId,
                                  editionNumber,
                                  tokenId
                              )
                          )
                      ) %
                      metadatas.length;
              }
              metadata = metadatas[idx];
          }
          /// @inheritdoc IEngine
          /// @dev Royalty bps and recipient is per-sequence.
          function getRoyaltyInfo(
              address collection,
              uint256 tokenId,
              uint256 salePrice
          ) external view override returns (address, uint256) {
              uint16 sequenceId = ICollection(collection).tokenSequenceId(tokenId);
              DropData storage drop = drops[collection][sequenceId];
              return (drop.revenueRecipient, (salePrice * drop.royaltyBps) / 10000);
          }
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      import {SequenceData} from "./IEngine.sol";
      /// @notice Collections are ERC721 contracts that contain records.
      interface ICollection {
          /// @notice Mint a new record with custom immutable token data. Only
          /// callable by the sequence-specific engine.
          function mintRecord(
              address to,
              uint16 sequenceId,
              uint80 tokenData
          ) external returns (uint256 tokenId);
          /// @notice Mint a new record with the edition number of the sequence
          /// written to the immutable token data. Only callable by the
          /// sequence-specific engine.
          function mintRecord(address to, uint16 sequenceId)
              external
              returns (uint256 tokenId);
          /// @notice Get the sequence ID for a given token.
          function tokenSequenceId(uint256 tokenId)
              external
              view
              returns (uint16 sequenceId);
          /// @notice Get the immutable mint data for a given token.
          function tokenMintData(uint256 tokenId) external view returns (uint80 data);
          /// @notice Get the sequence data for a given sequence ID.
          function getSequenceData(uint16 sequenceId)
              external
              view
              returns (SequenceData memory);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      /// @notice Data stored in the collection for each sequence.
      /// @dev We could use smaller ints for timestamps and supply, but we'd still be
      /// stuck with a 2-word storage layout. engine + dropNodeId is 28 bytes, leaving
      /// us with only 4 bytes for the remaining parameters.
      struct SequenceData {
          uint64 sealedBeforeTimestamp;
          uint64 sealedAfterTimestamp;
          uint64 maxSupply;
          uint64 minted;
          // ^ 1 word
          IEngine engine;
          uint64 dropNodeId;
          // 4 bytes remaining
      }
      /// @notice An engine contract implements record minting mechanics, tokenURI
      /// computation, and royalty computation.
      interface IEngine {
          /// @notice Called by the collection when a new sequence is configured.
          /// @dev An arbitrary bytes buffer engineData is forwarded from the
          /// collection that can be used to pass setup and configuration data
          function configureSequence(
              uint16 sequenceId,
              SequenceData calldata sequence,
              bytes calldata engineData
          ) external;
          /// @notice Called by the collection to resolve tokenURI.
          function getTokenURI(address collection, uint256 tokenId)
              external
              view
              returns (string memory);
          /// @notice Called by the collection to resolve royalties.
          function getRoyaltyInfo(
              address collection,
              uint256 tokenId,
              uint256 salePrice
          ) external view returns (address receiver, uint256 royaltyAmount);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      enum NodeType {
          INVALID_NODE_TYPE,
          METALABEL,
          RELEASE
      }
      /// @notice Data stored per node.
      struct NodeData {
          NodeType nodeType;
          uint64 owner;
          uint64 parent;
          uint64 groupNode;
          // 7 bytes remaining
      }
      /// @notice The node registry maintains a tree of ownable nodes that are used to
      /// catalog logical entities and manage access control in the Metalabel
      /// universe.
      interface INodeRegistry {
          /// @notice Create a new node. Child nodes can specify an group node that
          /// will be used to determine ownership, and a separate logical parent that
          /// expresses the entity relationship.  Child nodes can only be created if
          /// msg.sender is an authorized manager of the parent node.
          function createNode(
              NodeType nodeType,
              uint64 owner,
              uint64 parent,
              uint64 groupNode,
              address[] memory initialControllers,
              string memory metadata
          ) external returns (uint64 id);
          /// @notice Determine if an address is authorized to manage a node.
          /// A node can be managed by an address if any of the following conditions
          /// are true:
          ///   - The address's account is the owner of the node
          ///   - The address's account is the owner of the node's group node
          ///   - The address is an authorized controller of the node
          ///   - The address is an authorized controller of the node's group node
          function isAuthorizedAddressForNode(uint64 node, address subject)
              external
              view
              returns (bool isAuthorized);
          /// @notice Resolve node owner account.
          function ownerOf(uint64 id) external view returns (uint64);
      }
      

      File 2 of 4: Collection
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Data stored per-token, fits into a single storage word
      struct TokenData {
          address owner;
          uint16 sequenceId;
          uint80 data;
      }
      /// @notice Modern, minimalist, and gas efficient ERC-721 implementation.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol)
      abstract contract ERC721 {
          /*//////////////////////////////////////////////////////////////
                                       EVENTS
          //////////////////////////////////////////////////////////////*/
          event Transfer(address indexed from, address indexed to, uint256 indexed id);
          event Approval(address indexed owner, address indexed spender, uint256 indexed id);
          event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
          /*//////////////////////////////////////////////////////////////
                               METADATA STORAGE/LOGIC
          //////////////////////////////////////////////////////////////*/
          function tokenURI(uint256 id) public view virtual returns (string memory);
          /*//////////////////////////////////////////////////////////////
                            ERC721 BALANCE/OWNER STORAGE
          //////////////////////////////////////////////////////////////*/
          mapping(uint256 => TokenData) internal _tokenData;
          mapping(address => uint256) internal _balanceOf;
          function ownerOf(uint256 id) public view virtual returns (address owner) {
              require((owner = _tokenData[id].owner) != address(0), "NOT_MINTED");
          }
          function balanceOf(address owner) public view virtual returns (uint256) {
              require(owner != address(0), "ZERO_ADDRESS");
              return _balanceOf[owner];
          }
          /*//////////////////////////////////////////////////////////////
                               ERC721 APPROVAL STORAGE
          //////////////////////////////////////////////////////////////*/
          mapping(uint256 => address) public getApproved;
          mapping(address => mapping(address => bool)) public isApprovedForAll;
          /*//////////////////////////////////////////////////////////////
                                    ERC721 LOGIC
          //////////////////////////////////////////////////////////////*/
          function approve(address spender, uint256 id) public virtual {
              address owner = _tokenData[id].owner;
              require(msg.sender == owner || isApprovedForAll[owner][msg.sender], "NOT_AUTHORIZED");
              getApproved[id] = spender;
              emit Approval(owner, spender, id);
          }
          function setApprovalForAll(address operator, bool approved) public virtual {
              isApprovedForAll[msg.sender][operator] = approved;
              emit ApprovalForAll(msg.sender, operator, approved);
          }
          function transferFrom(
              address from,
              address to,
              uint256 id
          ) public virtual {
              require(from == _tokenData[id].owner, "WRONG_FROM");
              require(to != address(0), "INVALID_RECIPIENT");
              require(
                  msg.sender == from || isApprovedForAll[from][msg.sender] || msg.sender == getApproved[id],
                  "NOT_AUTHORIZED"
              );
              // Underflow of the sender's balance is impossible because we check for
              // ownership above and the recipient's balance can't realistically overflow.
              unchecked {
                  _balanceOf[from]--;
                  _balanceOf[to]++;
              }
              _tokenData[id].owner = to;
              delete getApproved[id];
              emit Transfer(from, to, id);
          }
          function safeTransferFrom(
              address from,
              address to,
              uint256 id
          ) public virtual {
              transferFrom(from, to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, "") ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          function safeTransferFrom(
              address from,
              address to,
              uint256 id,
              bytes calldata data
          ) public virtual {
              transferFrom(from, to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, data) ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          /*//////////////////////////////////////////////////////////////
                                    ERC165 LOGIC
          //////////////////////////////////////////////////////////////*/
          function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
              return
                  interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165
                  interfaceId == 0x80ac58cd || // ERC165 Interface ID for ERC721
                  interfaceId == 0x5b5e139f; // ERC165 Interface ID for ERC721Metadata
          }
          /*//////////////////////////////////////////////////////////////
                              INTERNAL MINT/BURN LOGIC
          //////////////////////////////////////////////////////////////*/
          function _mint(address to, uint256 id) internal virtual {
              return _mint(to, id, 0, 0);
          }
          function _mint(address to, uint256 id, uint16 sequenceId, uint80 data) internal virtual {
              require(to != address(0), "INVALID_RECIPIENT");
              require(_tokenData[id].owner == address(0), "ALREADY_MINTED");
              // Counter overflow is incredibly unrealistic.
              unchecked {
                  _balanceOf[to]++;
              }
              _tokenData[id] = TokenData({
                  owner: to,
                  sequenceId: sequenceId,
                  data: data
              });
              emit Transfer(address(0), to, id);
          }
          function _burn(uint256 id) internal virtual {
              address owner = _tokenData[id].owner;
              require(owner != address(0), "NOT_MINTED");
              // Ownership check above ensures no underflow.
              unchecked {
                  _balanceOf[owner]--;
              }
              delete _tokenData[id];
              delete getApproved[id];
              emit Transfer(owner, address(0), id);
          }
          /*//////////////////////////////////////////////////////////////
                              INTERNAL SAFE MINT LOGIC
          //////////////////////////////////////////////////////////////*/
          function _safeMint(address to, uint256 id) internal virtual {
              _mint(to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, "") ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          function _safeMint(
              address to,
              uint256 id,
              bytes memory data
          ) internal virtual {
              _mint(to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, data) ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          /*//////////////////////////////////////////////////////////////
                          METALABEL ADDED FUNCTIONALITY
          //////////////////////////////////////////////////////////////*/
          function getTokenData(uint256 id) external view virtual returns (TokenData memory) {
              TokenData memory data = _tokenData[id];
              require(data.owner != address(0), "NOT_MINTED");
              return data;
          }
      }
      /// @notice A generic interface for a contract which properly accepts ERC721 tokens.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol)
      abstract contract ERC721TokenReceiver {
          function onERC721Received(
              address,
              address,
              uint256,
              bytes calldata
          ) external virtual returns (bytes4) {
              return ERC721TokenReceiver.onERC721Received.selector;
          }
      }
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Read and write to persistent storage at a fraction of the cost.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SSTORE2.sol)
      /// @author Modified from 0xSequence (https://github.com/0xSequence/sstore2/blob/master/contracts/SSTORE2.sol)
      library SSTORE2 {
          uint256 internal constant DATA_OFFSET = 1; // We skip the first byte as it's a STOP opcode to ensure the contract can't be called.
          /*//////////////////////////////////////////////////////////////
                                     WRITE LOGIC
          //////////////////////////////////////////////////////////////*/
          function write(bytes memory data) internal returns (address pointer) {
              // Prefix the bytecode with a STOP opcode to ensure it cannot be called.
              bytes memory runtimeCode = abi.encodePacked(hex"00", data);
              bytes memory creationCode = abi.encodePacked(
                  //---------------------------------------------------------------------------------------------------------------//
                  // Opcode  | Opcode + Arguments  | Description  | Stack View                                                     //
                  //---------------------------------------------------------------------------------------------------------------//
                  // 0x60    |  0x600B             | PUSH1 11     | codeOffset                                                     //
                  // 0x59    |  0x59               | MSIZE        | 0 codeOffset                                                   //
                  // 0x81    |  0x81               | DUP2         | codeOffset 0 codeOffset                                        //
                  // 0x38    |  0x38               | CODESIZE     | codeSize codeOffset 0 codeOffset                               //
                  // 0x03    |  0x03               | SUB          | (codeSize - codeOffset) 0 codeOffset                           //
                  // 0x80    |  0x80               | DUP          | (codeSize - codeOffset) (codeSize - codeOffset) 0 codeOffset   //
                  // 0x92    |  0x92               | SWAP3        | codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset)   //
                  // 0x59    |  0x59               | MSIZE        | 0 codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset) //
                  // 0x39    |  0x39               | CODECOPY     | 0 (codeSize - codeOffset)                                      //
                  // 0xf3    |  0xf3               | RETURN       |                                                                //
                  //---------------------------------------------------------------------------------------------------------------//
                  hex"60_0B_59_81_38_03_80_92_59_39_F3", // Returns all code in the contract except for the first 11 (0B in hex) bytes.
                  runtimeCode // The bytecode we want the contract to have after deployment. Capped at 1 byte less than the code size limit.
              );
              assembly {
                  // Deploy a new contract with the generated creation code.
                  // We start 32 bytes into the code to avoid copying the byte length.
                  pointer := create(0, add(creationCode, 32), mload(creationCode))
              }
              require(pointer != address(0), "DEPLOYMENT_FAILED");
          }
          /*//////////////////////////////////////////////////////////////
                                     READ LOGIC
          //////////////////////////////////////////////////////////////*/
          function read(address pointer) internal view returns (bytes memory) {
              return readBytecode(pointer, DATA_OFFSET, pointer.code.length - DATA_OFFSET);
          }
          function read(address pointer, uint256 start) internal view returns (bytes memory) {
              start += DATA_OFFSET;
              return readBytecode(pointer, start, pointer.code.length - start);
          }
          function read(
              address pointer,
              uint256 start,
              uint256 end
          ) internal view returns (bytes memory) {
              start += DATA_OFFSET;
              end += DATA_OFFSET;
              require(pointer.code.length >= end, "OUT_OF_BOUNDS");
              return readBytecode(pointer, start, end - start);
          }
          /*//////////////////////////////////////////////////////////////
                                INTERNAL HELPER LOGIC
          //////////////////////////////////////////////////////////////*/
          function readBytecode(
              address pointer,
              uint256 start,
              uint256 size
          ) private view returns (bytes memory data) {
              assembly {
                  // Get a pointer to some free memory.
                  data := mload(0x40)
                  // Update the free memory pointer to prevent overriding our data.
                  // We use and(x, not(31)) as a cheaper equivalent to sub(x, mod(x, 32)).
                  // Adding 31 to size and running the result through the logic above ensures
                  // the memory pointer remains word-aligned, following the Solidity convention.
                  mstore(0x40, add(data, and(add(add(size, 32), 31), not(31))))
                  // Store the size of the data in the first 32 byte chunk of free memory.
                  mstore(data, size)
                  // Copy the code into memory right after the 32 bytes we used to store the size.
                  extcodecopy(pointer, add(data, 32), start, size)
              }
          }
      }
      // SPDX-License-Identifier: MIT
      // OpenZeppelin Contracts v4.4.1 (interfaces/IERC165.sol)
      pragma solidity ^0.8.0;
      import "../utils/introspection/IERC165.sol";
      // SPDX-License-Identifier: MIT
      // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)
      pragma solidity ^0.8.0;
      import "../utils/introspection/IERC165.sol";
      /**
       * @dev Interface for the NFT Royalty Standard.
       *
       * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal
       * support for royalty payments across all NFT marketplaces and ecosystem participants.
       *
       * _Available since v4.5._
       */
      interface IERC2981 is IERC165 {
          /**
           * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of
           * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.
           */
          function royaltyInfo(uint256 tokenId, uint256 salePrice)
              external
              view
              returns (address receiver, uint256 royaltyAmount);
      }
      // 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.17;
      /*
      ███╗   ███╗███████╗████████╗ █████╗ ██╗      █████╗ ██████╗ ███████╗██╗
      ████╗ ████║██╔════╝╚══██╔══╝██╔══██╗██║     ██╔══██╗██╔══██╗██╔════╝██║
      ██╔████╔██║█████╗     ██║   ███████║██║     ███████║██████╔╝█████╗  ██║
      ██║╚██╔╝██║██╔══╝     ██║   ██╔══██║██║     ██╔══██║██╔══██╗██╔══╝  ██║
      ██║ ╚═╝ ██║███████╗   ██║   ██║  ██║███████╗██║  ██║██████╔╝███████╗███████╗
      ╚═╝     ╚═╝╚══════╝   ╚═╝   ╚═╝  ╚═╝╚══════╝╚═╝  ╚═╝╚═════╝ ╚══════╝╚══════╝
      Deployed by Metalabel with 💖 as a permanent application on the Ethereum blockchain.
      Metalabel is a growing universe of tools, knowledge, and resources for
      metalabels and cultural collectives.
      Our purpose is to establish the metalabel as key infrastructure for creative
      collectives and to inspire a new culture of creative collaboration and mutual
      support.
      OUR SQUAD
      Anna Bulbrook (Curator)
      Austin Robey (Community)
      Brandon Valosek (Engineer)
      Ilya Yudanov (Designer)
      Lauren Dorman (Engineer)
      Rob Kalin (Board)
      Yancey Strickler (Director)
      https://metalabel.xyz
      */
      import {IERC165} from "@openzeppelin/contracts/interfaces/IERC165.sol";
      import {IERC2981} from "@openzeppelin/contracts/interfaces/IERC2981.sol";
      import {ERC721} from "@metalabel/solmate/src/tokens/ERC721.sol";
      import {SSTORE2} from "@metalabel/solmate/src/utils/SSTORE2.sol";
      import {INodeRegistry} from "./interfaces/INodeRegistry.sol";
      import {IEngine, SequenceData} from "./interfaces/IEngine.sol";
      import {ICollection} from "./interfaces/ICollection.sol";
      import {Resource, AccessControlData} from "./Resource.sol";
      /// @notice Immutable data stored per-collection.
      /// @dev This is stored via SSTORE2 to save gas.
      struct ImmutableCollectionData {
          string name;
          string symbol;
          string contractURI;
      }
      /// @notice Collections are ERC721 contracts that contain records.
      /// - Minting logic, tokenURI, and royalties are delegated to an external engine
      ///     contract
      /// - Sequences are a mapping between an external engine contract and parameters
      ///     stored in the collection
      /// - Multiple sequences can be configured for a single collection, records may
      ///     be rendered and minted in a variety of different ways
      contract Collection is ERC721, Resource, ICollection, IERC2981 {
          // ---
          // Errors
          // ---
          /// @notice The init function was called more than once.
          error AlreadyInitialized();
          /// @notice A record mint attempt was made for a sequence that is currently
          /// sealed.
          error SequenceIsSealed();
          /// @notice A record mint attempt was made for a sequence that has no
          /// remaining supply.
          error SequenceSupplyExhausted();
          /// @notice An invalid sequence config was provided during configuration.
          error InvalidSequenceConfig();
          /// @notice msg.sender during a mint call did not match expected engine
          /// origin.
          error InvalidMintRequest();
          // ---
          // Events
          // ---
          /// @notice A new record was minted.
          /// @dev The underlying ERC721 implementation already emits a Transfer event
          /// on mint, this event announces the sequence the token is minted into and
          /// its immutable token data.
          event RecordCreated(
              uint256 indexed tokenId,
              uint16 indexed sequenceId,
              uint80 data
          );
          /// @notice A sequence has been set or updated.
          event SequenceConfigured(
              uint16 indexed sequenceId,
              SequenceData sequenceData,
              bytes engineData
          );
          /// @notice The owner address of this collection was updated.
          event OwnershipTransferred(
              address indexed previousOwner,
              address indexed newOwner
          );
          // ---
          // Storage
          // ---
          /// @notice Total number of records minted in this collection.
          uint256 public totalSupply;
          /// @notice Total number of sequences configured in this collection.
          uint16 public sequenceCount;
          /// @notice Only for marketplace interop, can be set by owner of the control
          /// node.
          address public owner;
          /// @notice The SSTORE2 storage pointer for immutable collection data.
          /// @dev This data is exposed through name, symbol, and contractURI views
          address internal immutableStoragePointer;
          /// @notice Information about each sequence.
          mapping(uint16 => SequenceData) internal sequences;
          // ---
          // Constructor
          // ---
          /// @dev Constructor only called during deployment of the implementation,
          /// all storage should be set up in init function which is called atomically
          /// after clone deployment
          constructor() {
              // Write dummy data to the immutable storage pointer to prevent
              // initialization of the implementation contract.
              immutableStoragePointer = SSTORE2.write(
                  abi.encode(
                      ImmutableCollectionData({name: "", symbol: "", contractURI: ""})
                  )
              );
          }
          // ---
          // Clone init
          // ---
          /// @notice Initialize contract state.
          /// @dev Should be called immediately after deploying the clone in the same
          /// transaction.
          function init(
              address _owner,
              AccessControlData calldata _accessControl,
              string calldata _metadata,
              ImmutableCollectionData calldata _data
          ) external {
              if (immutableStoragePointer != address(0)) revert AlreadyInitialized();
              immutableStoragePointer = SSTORE2.write(abi.encode(_data));
              // Set ERC721 market interop.
              owner = _owner;
              emit OwnershipTransferred(address(0), owner);
              // Assign access control data.
              accessControl = _accessControl;
              // This memberships collection is a resource that can be cataloged -
              // emit the initial metadata value
              emit Broadcast("metadata", _metadata);
          }
          // ---
          // Admin functionality
          // ---
          /// @notice Change the owner address of this collection.
          /// @dev This is only here for market interop, access control is handled via
          /// the control node.
          function setOwner(address _owner) external onlyAuthorized {
              address previousOwner = owner;
              owner = _owner;
              emit OwnershipTransferred(previousOwner, _owner);
          }
          /// @notice Create a new sequence configuration.
          /// @dev The _engineData bytes parameter is arbitrary data that is passed
          /// directly to the engine powering this new sequence.
          function configureSequence(
              SequenceData calldata _sequence,
              bytes calldata _engineData
          ) external onlyAuthorized {
              // The drop this sequence is associated with must be manageable by
              // msg.sender. This is in addition to the onlyAuthorized modifier which
              // asserts msg.sender can manage the control node of the whole
              // collection.
              // msg.sender is either a metalabel admin EOA, or a controller contract
              // that has been authorized to do drops on the drop node.
              if (
                  !accessControl.nodeRegistry.isAuthorizedAddressForNode(
                      _sequence.dropNodeId,
                      msg.sender
                  )
              ) {
                  revert NotAuthorized();
              }
              // If there is sealedAfter timestamp (i.e timebound sequence), ensure
              // that sealedBefore is strictly less than sealedAfter AND that
              // sealedAfter occurs strictly in the future. If sealedAfter is zero,
              // there is no time limit. We are allowing cases where sealedBefore
              // occurs in the past.
              if (
                  _sequence.sealedAfterTimestamp > 0 &&
                  (_sequence.sealedBeforeTimestamp >=
                      _sequence.sealedAfterTimestamp ||
                      _sequence.sealedAfterTimestamp <= block.timestamp)
              ) {
                  revert InvalidSequenceConfig();
              }
              // Prevent having a minted count before the sequence starts. This
              // wouldn't break anything, but would cause the indexed "minted" amount
              // from actual mints to differ from the sequence data tracking total
              // supply, which is non-ideal and worth the small gas to check.
              //
              // We're not using a separate struct here for inputs that omits the
              // minted count field, being able to copy from calldata to storage is
              // nice.
              if (_sequence.minted != 0) {
                  revert InvalidSequenceConfig();
              }
              // Write sequence data to storage
              uint16 sequenceId = ++sequenceCount;
              sequences[sequenceId] = _sequence;
              emit SequenceConfigured(sequenceId, _sequence, _engineData);
              // Invoke configureSequence on the engine to give it a chance to setup
              // and store any needed info. Doing this after event emitting so that
              // indexers see the sequence first before any engine-side events
              _sequence.engine.configureSequence(sequenceId, _sequence, _engineData);
          }
          // ---
          // Engine functionality
          // ---
          /// @inheritdoc ICollection
          function mintRecord(
              address to,
              uint16 sequenceId,
              uint80 tokenData
          ) external returns (uint256 tokenId) {
              SequenceData storage sequence = sequences[sequenceId];
              _validateSequence(sequence);
              // Mint the record.
              tokenId = ++totalSupply;
              ++sequence.minted;
              _mint(to, tokenId, sequenceId, tokenData);
              emit RecordCreated(tokenId, sequenceId, tokenData);
          }
          /// @inheritdoc ICollection
          function mintRecord(address to, uint16 sequenceId)
              external
              returns (uint256 tokenId)
          {
              SequenceData storage sequence = sequences[sequenceId];
              _validateSequence(sequence);
              // Mint the record.
              tokenId = ++totalSupply;
              uint64 editionNumber = ++sequence.minted;
              _mint(to, tokenId, sequenceId, editionNumber);
              emit RecordCreated(tokenId, sequenceId, editionNumber);
          }
          /// @dev Ensure a given sequence is valid to mint into by the current msg.sender
          function _validateSequence(SequenceData memory sequence) internal view {
              // Ensure that only the engine for this sequence can mint records. Mint
              // transactions termiante on the engine side - the engine then invokes
              // the mint functions on the Collection.
              if (sequence.engine != IEngine(msg.sender)) {
                  revert InvalidMintRequest();
              }
              // Ensure that mint is not happening before or after allowed window.
              if (
                  block.timestamp < sequence.sealedBeforeTimestamp ||
                  (sequence.sealedAfterTimestamp > 0 && // sealed after = 0 => no end
                      block.timestamp >= sequence.sealedAfterTimestamp)
              ) {
                  revert SequenceIsSealed();
              }
              // Ensure we have remaining supply to mint
              if (sequence.maxSupply > 0 && sequence.minted >= sequence.maxSupply) {
                  revert SequenceSupplyExhausted();
              }
          }
          // ---
          // ICollection views
          // ---
          /// @inheritdoc ICollection
          function getSequenceData(uint16 sequenceId)
              external
              view
              override
              returns (SequenceData memory sequence)
          {
              sequence = sequences[sequenceId];
          }
          // ---
          // ERC721 views
          // ---
          /// @notice The collection name.
          function name() public view virtual returns (string memory value) {
              value = _resolveImmutableStorage().name;
          }
          /// @notice The collection symbol.
          function symbol() public view virtual returns (string memory value) {
              value = _resolveImmutableStorage().symbol;
          }
          /// @inheritdoc ERC721
          /// @dev Resolve token URI from the engine powering the sequence.
          function tokenURI(uint256 tokenId)
              public
              view
              virtual
              override
              returns (string memory uri)
          {
              IEngine engine = sequences[_tokenData[tokenId].sequenceId].engine;
              uri = engine.getTokenURI(address(this), tokenId);
          }
          /// @notice Get the contract URI.
          function contractURI() public view virtual returns (string memory value) {
              value = _resolveImmutableStorage().contractURI;
          }
          // ---
          // Misc views
          // ---
          /// @inheritdoc ICollection
          function tokenSequenceId(uint256 tokenId)
              external
              view
              returns (uint16 sequenceId)
          {
              sequenceId = _tokenData[tokenId].sequenceId;
          }
          /// @inheritdoc ICollection
          /// @dev Token mint data is either edition number or arbitrary custom data
          /// passed in the by the engine at mint-time.
          function tokenMintData(uint256 tokenId)
              external
              view
              returns (uint80 data)
          {
              data = _tokenData[tokenId].data;
          }
          // ---
          // ERC2981 functionality
          // ---
          /// @inheritdoc IERC2981
          /// @dev Resolve royalty info from the engine powering the sequence.
          function royaltyInfo(uint256 tokenId, uint256 salePrice)
              external
              view
              returns (address receiver, uint256 royaltyAmount)
          {
              IEngine engine = sequences[_tokenData[tokenId].sequenceId].engine;
              return engine.getRoyaltyInfo(address(this), tokenId, salePrice);
          }
          // ---
          // Introspection
          // ---
          /// @inheritdoc IERC165
          /// @dev ERC165 checks return true for: ERC165, ERC721, and ERC2981.
          function supportsInterface(bytes4 interfaceId)
              public
              view
              virtual
              override(ERC721, IERC165)
              returns (bool)
          {
              return
                  interfaceId == type(IERC2981).interfaceId ||
                  super.supportsInterface(interfaceId);
          }
          // ---
          // Internal views
          // ---
          /// @dev Read name/symbol/contractURI strings via SSTORE2.
          function _resolveImmutableStorage()
              internal
              view
              returns (ImmutableCollectionData memory data)
          {
              data = abi.decode(
                  SSTORE2.read(immutableStoragePointer),
                  (ImmutableCollectionData)
              );
          }
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      import {SequenceData} from "./IEngine.sol";
      /// @notice Collections are ERC721 contracts that contain records.
      interface ICollection {
          /// @notice Mint a new record with custom immutable token data. Only
          /// callable by the sequence-specific engine.
          function mintRecord(
              address to,
              uint16 sequenceId,
              uint80 tokenData
          ) external returns (uint256 tokenId);
          /// @notice Mint a new record with the edition number of the sequence
          /// written to the immutable token data. Only callable by the
          /// sequence-specific engine.
          function mintRecord(address to, uint16 sequenceId)
              external
              returns (uint256 tokenId);
          /// @notice Get the sequence ID for a given token.
          function tokenSequenceId(uint256 tokenId)
              external
              view
              returns (uint16 sequenceId);
          /// @notice Get the immutable mint data for a given token.
          function tokenMintData(uint256 tokenId) external view returns (uint80 data);
          /// @notice Get the sequence data for a given sequence ID.
          function getSequenceData(uint16 sequenceId)
              external
              view
              returns (SequenceData memory);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      /// @notice Data stored in the collection for each sequence.
      /// @dev We could use smaller ints for timestamps and supply, but we'd still be
      /// stuck with a 2-word storage layout. engine + dropNodeId is 28 bytes, leaving
      /// us with only 4 bytes for the remaining parameters.
      struct SequenceData {
          uint64 sealedBeforeTimestamp;
          uint64 sealedAfterTimestamp;
          uint64 maxSupply;
          uint64 minted;
          // ^ 1 word
          IEngine engine;
          uint64 dropNodeId;
          // 4 bytes remaining
      }
      /// @notice An engine contract implements record minting mechanics, tokenURI
      /// computation, and royalty computation.
      interface IEngine {
          /// @notice Called by the collection when a new sequence is configured.
          /// @dev An arbitrary bytes buffer engineData is forwarded from the
          /// collection that can be used to pass setup and configuration data
          function configureSequence(
              uint16 sequenceId,
              SequenceData calldata sequence,
              bytes calldata engineData
          ) external;
          /// @notice Called by the collection to resolve tokenURI.
          function getTokenURI(address collection, uint256 tokenId)
              external
              view
              returns (string memory);
          /// @notice Called by the collection to resolve royalties.
          function getRoyaltyInfo(
              address collection,
              uint256 tokenId,
              uint256 salePrice
          ) external view returns (address receiver, uint256 royaltyAmount);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      enum NodeType {
          INVALID_NODE_TYPE,
          METALABEL,
          RELEASE
      }
      /// @notice Data stored per node.
      struct NodeData {
          NodeType nodeType;
          uint64 owner;
          uint64 parent;
          uint64 groupNode;
          // 7 bytes remaining
      }
      /// @notice The node registry maintains a tree of ownable nodes that are used to
      /// catalog logical entities and manage access control in the Metalabel
      /// universe.
      interface INodeRegistry {
          /// @notice Create a new node. Child nodes can specify an group node that
          /// will be used to determine ownership, and a separate logical parent that
          /// expresses the entity relationship.  Child nodes can only be created if
          /// msg.sender is an authorized manager of the parent node.
          function createNode(
              NodeType nodeType,
              uint64 owner,
              uint64 parent,
              uint64 groupNode,
              address[] memory initialControllers,
              string memory metadata
          ) external returns (uint64 id);
          /// @notice Determine if an address is authorized to manage a node.
          /// A node can be managed by an address if any of the following conditions
          /// are true:
          ///   - The address's account is the owner of the node
          ///   - The address's account is the owner of the node's group node
          ///   - The address is an authorized controller of the node
          ///   - The address is an authorized controller of the node's group node
          function isAuthorizedAddressForNode(uint64 node, address subject)
              external
              view
              returns (bool isAuthorized);
          /// @notice Resolve node owner account.
          function ownerOf(uint64 id) external view returns (uint64);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      import {INodeRegistry} from "./INodeRegistry.sol";
      /// @notice An on-chain resource that is intended to be cataloged within the
      /// Metalabel universe
      interface IResource {
          /// @notice Broadcast an arbitrary message.
          event Broadcast(string topic, string message);
          /// @notice Return the node registry contract address.
          function nodeRegistry() external view returns (INodeRegistry);
          /// @notice Return the control node ID for this resource.
          function controlNode() external view returns (uint64 nodeId);
          /// @notice Return true if the given address is authorized to manage this
          /// resource.
          function isAuthorized(address subject)
              external
              view
              returns (bool authorized);
          /// @notice Emit an on-chain message. msg.sender must be authorized to
          /// manage this resource's control node
          function broadcast(string calldata topic, string calldata message) external;
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      /*
      ███╗   ███╗███████╗████████╗ █████╗ ██╗      █████╗ ██████╗ ███████╗██╗
      ████╗ ████║██╔════╝╚══██╔══╝██╔══██╗██║     ██╔══██╗██╔══██╗██╔════╝██║
      ██╔████╔██║█████╗     ██║   ███████║██║     ███████║██████╔╝█████╗  ██║
      ██║╚██╔╝██║██╔══╝     ██║   ██╔══██║██║     ██╔══██║██╔══██╗██╔══╝  ██║
      ██║ ╚═╝ ██║███████╗   ██║   ██║  ██║███████╗██║  ██║██████╔╝███████╗███████╗
      ╚═╝     ╚═╝╚══════╝   ╚═╝   ╚═╝  ╚═╝╚══════╝╚═╝  ╚═╝╚═════╝ ╚══════╝╚══════╝
      Deployed by Metalabel with 💖 as a permanent application on the Ethereum blockchain.
      Metalabel is a growing universe of tools, knowledge, and resources for
      metalabels and cultural collectives.
      Our purpose is to establish the metalabel as key infrastructure for creative
      collectives and to inspire a new culture of creative collaboration and mutual
      support.
      OUR SQUAD
      Anna Bulbrook (Curator)
      Austin Robey (Community)
      Brandon Valosek (Engineer)
      Ilya Yudanov (Designer)
      Lauren Dorman (Engineer)
      Rob Kalin (Board)
      Yancey Strickler (Director)
      https://metalabel.xyz
      */
      import {IResource} from "./interfaces/IResource.sol";
      import {INodeRegistry} from "./interfaces/INodeRegistry.sol";
      /// @notice Data stored for handling access control resolution.
      struct AccessControlData {
          INodeRegistry nodeRegistry;
          uint64 controlNodeId;
          // 4 bytes remaining
      }
      /// @notice A resource that can be cataloged on the Metalabel protocol.
      contract Resource is IResource {
          // ---
          // Errors
          // ---
          /// @notice Unauthorized msg.sender attempted to interact with this resource
          error NotAuthorized();
          // ---
          // Storage
          // ---
          /// @notice Access control data for this resource.
          AccessControlData public accessControl;
          // ---
          // Modifiers
          // ---
          /// @dev Make a function only callable by a msg.sender that is authorized to
          /// manage the control node of this resource
          modifier onlyAuthorized() {
              if (
                  !accessControl.nodeRegistry.isAuthorizedAddressForNode(
                      accessControl.controlNodeId,
                      msg.sender
                  )
              ) {
                  revert NotAuthorized();
              }
              _;
          }
          // ---
          // Admin functionality
          // ---
          /// @inheritdoc IResource
          function broadcast(string calldata topic, string calldata message)
              external
              onlyAuthorized
          {
              emit Broadcast(topic, message);
          }
          // ---
          // Resource views
          // ---
          /// @inheritdoc IResource
          function nodeRegistry() external view virtual returns (INodeRegistry) {
              return accessControl.nodeRegistry;
          }
          /// @inheritdoc IResource
          function controlNode() external view virtual returns (uint64 nodeId) {
              return accessControl.controlNodeId;
          }
          /// @inheritdoc IResource
          function isAuthorized(address subject)
              public
              view
              virtual
              returns (bool authorized)
          {
              authorized = accessControl.nodeRegistry.isAuthorizedAddressForNode(
                  accessControl.controlNodeId,
                  subject
              );
          }
      }
      

      File 3 of 4: SplitWallet
      // SPDX-License-Identifier: GPL-3.0-or-later
      pragma solidity 0.8.4;
      import {ISplitMain} from './interfaces/ISplitMain.sol';
      import {ERC20} from '@rari-capital/solmate/src/tokens/ERC20.sol';
      import {SafeTransferLib} from '@rari-capital/solmate/src/utils/SafeTransferLib.sol';
      /**
       * ERRORS
       */
      /// @notice Unauthorized sender
      error Unauthorized();
      /**
       * @title SplitWallet
       * @author 0xSplits <[email protected]>
       * @notice The implementation logic for `SplitProxy`.
       * @dev `SplitProxy` handles `receive()` itself to avoid the gas cost with `DELEGATECALL`.
       */
      contract SplitWallet {
        using SafeTransferLib for address;
        using SafeTransferLib for ERC20;
        /**
         * EVENTS
         */
        /** @notice emitted after each successful ETH transfer to proxy
         *  @param split Address of the split that received ETH
         *  @param amount Amount of ETH received
         */
        event ReceiveETH(address indexed split, uint256 amount);
        /**
         * STORAGE
         */
        /**
         * STORAGE - CONSTANTS & IMMUTABLES
         */
        /// @notice address of SplitMain for split distributions & EOA/SC withdrawals
        ISplitMain public immutable splitMain;
        /**
         * MODIFIERS
         */
        /// @notice Reverts if the sender isn't SplitMain
        modifier onlySplitMain() {
          if (msg.sender != address(splitMain)) revert Unauthorized();
          _;
        }
        /**
         * CONSTRUCTOR
         */
        constructor() {
          splitMain = ISplitMain(msg.sender);
        }
        /**
         * FUNCTIONS - PUBLIC & EXTERNAL
         */
        /** @notice Sends amount `amount` of ETH in proxy to SplitMain
         *  @dev payable reduces gas cost; no vulnerability to accidentally lock
         *  ETH introduced since fn call is restricted to SplitMain
         *  @param amount Amount to send
         */
        function sendETHToMain(uint256 amount) external payable onlySplitMain() {
          address(splitMain).safeTransferETH(amount);
        }
        /** @notice Sends amount `amount` of ERC20 `token` in proxy to SplitMain
         *  @dev payable reduces gas cost; no vulnerability to accidentally lock
         *  ETH introduced since fn call is restricted to SplitMain
         *  @param token Token to send
         *  @param amount Amount to send
         */
        function sendERC20ToMain(ERC20 token, uint256 amount)
          external
          payable
          onlySplitMain()
        {
          token.safeTransfer(address(splitMain), amount);
        }
      }
      // SPDX-License-Identifier: GPL-3.0-or-later
      pragma solidity 0.8.4;
      import {ERC20} from '@rari-capital/solmate/src/tokens/ERC20.sol';
      /**
       * @title ISplitMain
       * @author 0xSplits <[email protected]>
       */
      interface ISplitMain {
        /**
         * FUNCTIONS
         */
        function walletImplementation() external returns (address);
        function createSplit(
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee,
          address controller
        ) external returns (address);
        function predictImmutableSplitAddress(
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee
        ) external view returns (address);
        function updateSplit(
          address split,
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee
        ) external;
        function transferControl(address split, address newController) external;
        function cancelControlTransfer(address split) external;
        function acceptControl(address split) external;
        function makeSplitImmutable(address split) external;
        function distributeETH(
          address split,
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee,
          address distributorAddress
        ) external;
        function updateAndDistributeETH(
          address split,
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee,
          address distributorAddress
        ) external;
        function distributeERC20(
          address split,
          ERC20 token,
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee,
          address distributorAddress
        ) external;
        function updateAndDistributeERC20(
          address split,
          ERC20 token,
          address[] calldata accounts,
          uint32[] calldata percentAllocations,
          uint32 distributorFee,
          address distributorAddress
        ) external;
        function withdraw(
          address account,
          uint256 withdrawETH,
          ERC20[] calldata tokens
        ) external;
        /**
         * EVENTS
         */
        /** @notice emitted after each successful split creation
         *  @param split Address of the created split
         */
        event CreateSplit(address indexed split);
        /** @notice emitted after each successful split update
         *  @param split Address of the updated split
         */
        event UpdateSplit(address indexed split);
        /** @notice emitted after each initiated split control transfer
         *  @param split Address of the split control transfer was initiated for
         *  @param newPotentialController Address of the split's new potential controller
         */
        event InitiateControlTransfer(
          address indexed split,
          address indexed newPotentialController
        );
        /** @notice emitted after each canceled split control transfer
         *  @param split Address of the split control transfer was canceled for
         */
        event CancelControlTransfer(address indexed split);
        /** @notice emitted after each successful split control transfer
         *  @param split Address of the split control was transferred for
         *  @param previousController Address of the split's previous controller
         *  @param newController Address of the split's new controller
         */
        event ControlTransfer(
          address indexed split,
          address indexed previousController,
          address indexed newController
        );
        /** @notice emitted after each successful ETH balance split
         *  @param split Address of the split that distributed its balance
         *  @param amount Amount of ETH distributed
         *  @param distributorAddress Address to credit distributor fee to
         */
        event DistributeETH(
          address indexed split,
          uint256 amount,
          address indexed distributorAddress
        );
        /** @notice emitted after each successful ERC20 balance split
         *  @param split Address of the split that distributed its balance
         *  @param token Address of ERC20 distributed
         *  @param amount Amount of ERC20 distributed
         *  @param distributorAddress Address to credit distributor fee to
         */
        event DistributeERC20(
          address indexed split,
          ERC20 indexed token,
          uint256 amount,
          address indexed distributorAddress
        );
        /** @notice emitted after each successful withdrawal
         *  @param account Address that funds were withdrawn to
         *  @param ethAmount Amount of ETH withdrawn
         *  @param tokens Addresses of ERC20s withdrawn
         *  @param tokenAmounts Amounts of corresponding ERC20s withdrawn
         */
        event Withdrawal(
          address indexed account,
          uint256 ethAmount,
          ERC20[] tokens,
          uint256[] tokenAmounts
        );
      }
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Modern and gas efficient ERC20 + EIP-2612 implementation.
      /// @author Solmate (https://github.com/Rari-Capital/solmate/blob/main/src/tokens/ERC20.sol)
      /// @author Modified from Uniswap (https://github.com/Uniswap/uniswap-v2-core/blob/master/contracts/UniswapV2ERC20.sol)
      /// @dev Do not manually set balances without updating totalSupply, as the sum of all user balances must not exceed it.
      abstract contract ERC20 {
          /*///////////////////////////////////////////////////////////////
                                        EVENTS
          //////////////////////////////////////////////////////////////*/
          event Transfer(address indexed from, address indexed to, uint256 amount);
          event Approval(address indexed owner, address indexed spender, uint256 amount);
          /*///////////////////////////////////////////////////////////////
                                   METADATA STORAGE
          //////////////////////////////////////////////////////////////*/
          string public name;
          string public symbol;
          uint8 public immutable decimals;
          /*///////////////////////////////////////////////////////////////
                                    ERC20 STORAGE
          //////////////////////////////////////////////////////////////*/
          uint256 public totalSupply;
          mapping(address => uint256) public balanceOf;
          mapping(address => mapping(address => uint256)) public allowance;
          /*///////////////////////////////////////////////////////////////
                                   EIP-2612 STORAGE
          //////////////////////////////////////////////////////////////*/
          bytes32 public constant PERMIT_TYPEHASH =
              keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
          uint256 internal immutable INITIAL_CHAIN_ID;
          bytes32 internal immutable INITIAL_DOMAIN_SEPARATOR;
          mapping(address => uint256) public nonces;
          /*///////////////////////////////////////////////////////////////
                                     CONSTRUCTOR
          //////////////////////////////////////////////////////////////*/
          constructor(
              string memory _name,
              string memory _symbol,
              uint8 _decimals
          ) {
              name = _name;
              symbol = _symbol;
              decimals = _decimals;
              INITIAL_CHAIN_ID = block.chainid;
              INITIAL_DOMAIN_SEPARATOR = computeDomainSeparator();
          }
          /*///////////////////////////////////////////////////////////////
                                    ERC20 LOGIC
          //////////////////////////////////////////////////////////////*/
          function approve(address spender, uint256 amount) public virtual returns (bool) {
              allowance[msg.sender][spender] = amount;
              emit Approval(msg.sender, spender, amount);
              return true;
          }
          function transfer(address to, uint256 amount) public virtual returns (bool) {
              balanceOf[msg.sender] -= amount;
              // Cannot overflow because the sum of all user
              // balances can't exceed the max uint256 value.
              unchecked {
                  balanceOf[to] += amount;
              }
              emit Transfer(msg.sender, to, amount);
              return true;
          }
          function transferFrom(
              address from,
              address to,
              uint256 amount
          ) public virtual returns (bool) {
              uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals.
              if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount;
              balanceOf[from] -= amount;
              // Cannot overflow because the sum of all user
              // balances can't exceed the max uint256 value.
              unchecked {
                  balanceOf[to] += amount;
              }
              emit Transfer(from, to, amount);
              return true;
          }
          /*///////////////////////////////////////////////////////////////
                                    EIP-2612 LOGIC
          //////////////////////////////////////////////////////////////*/
          function permit(
              address owner,
              address spender,
              uint256 value,
              uint256 deadline,
              uint8 v,
              bytes32 r,
              bytes32 s
          ) public virtual {
              require(deadline >= block.timestamp, "PERMIT_DEADLINE_EXPIRED");
              // Unchecked because the only math done is incrementing
              // the owner's nonce which cannot realistically overflow.
              unchecked {
                  bytes32 digest = keccak256(
                      abi.encodePacked(
                          "\\x19\\x01",
                          DOMAIN_SEPARATOR(),
                          keccak256(abi.encode(PERMIT_TYPEHASH, owner, spender, value, nonces[owner]++, deadline))
                      )
                  );
                  address recoveredAddress = ecrecover(digest, v, r, s);
                  require(recoveredAddress != address(0) && recoveredAddress == owner, "INVALID_SIGNER");
                  allowance[recoveredAddress][spender] = value;
              }
              emit Approval(owner, spender, value);
          }
          function DOMAIN_SEPARATOR() public view virtual returns (bytes32) {
              return block.chainid == INITIAL_CHAIN_ID ? INITIAL_DOMAIN_SEPARATOR : computeDomainSeparator();
          }
          function computeDomainSeparator() internal view virtual returns (bytes32) {
              return
                  keccak256(
                      abi.encode(
                          keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"),
                          keccak256(bytes(name)),
                          keccak256("1"),
                          block.chainid,
                          address(this)
                      )
                  );
          }
          /*///////////////////////////////////////////////////////////////
                             INTERNAL MINT/BURN LOGIC
          //////////////////////////////////////////////////////////////*/
          function _mint(address to, uint256 amount) internal virtual {
              totalSupply += amount;
              // Cannot overflow because the sum of all user
              // balances can't exceed the max uint256 value.
              unchecked {
                  balanceOf[to] += amount;
              }
              emit Transfer(address(0), to, amount);
          }
          function _burn(address from, uint256 amount) internal virtual {
              balanceOf[from] -= amount;
              // Cannot underflow because a user's balance
              // will never be larger than the total supply.
              unchecked {
                  totalSupply -= amount;
              }
              emit Transfer(from, address(0), amount);
          }
      }
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      import {ERC20} from "../tokens/ERC20.sol";
      /// @notice Safe ETH and ERC20 transfer library that gracefully handles missing return values.
      /// @author Solmate (https://github.com/Rari-Capital/solmate/blob/main/src/utils/SafeTransferLib.sol)
      /// @author Modified from Gnosis (https://github.com/gnosis/gp-v2-contracts/blob/main/src/contracts/libraries/GPv2SafeERC20.sol)
      /// @dev Use with caution! Some functions in this library knowingly create dirty bits at the destination of the free memory pointer.
      library SafeTransferLib {
          /*///////////////////////////////////////////////////////////////
                                  ETH OPERATIONS
          //////////////////////////////////////////////////////////////*/
          function safeTransferETH(address to, uint256 amount) internal {
              bool callStatus;
              assembly {
                  // Transfer the ETH and store if it succeeded or not.
                  callStatus := call(gas(), to, amount, 0, 0, 0, 0)
              }
              require(callStatus, "ETH_TRANSFER_FAILED");
          }
          /*///////////////////////////////////////////////////////////////
                                 ERC20 OPERATIONS
          //////////////////////////////////////////////////////////////*/
          function safeTransferFrom(
              ERC20 token,
              address from,
              address to,
              uint256 amount
          ) internal {
              bool callStatus;
              assembly {
                  // Get a pointer to some free memory.
                  let freeMemoryPointer := mload(0x40)
                  // Write the abi-encoded calldata to memory piece by piece:
                  mstore(freeMemoryPointer, 0x23b872dd00000000000000000000000000000000000000000000000000000000) // Begin with the function selector.
                  mstore(add(freeMemoryPointer, 4), and(from, 0xffffffffffffffffffffffffffffffffffffffff)) // Mask and append the "from" argument.
                  mstore(add(freeMemoryPointer, 36), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Mask and append the "to" argument.
                  mstore(add(freeMemoryPointer, 68), amount) // Finally append the "amount" argument. No mask as it's a full 32 byte value.
                  // Call the token and store if it succeeded or not.
                  // We use 100 because the calldata length is 4 + 32 * 3.
                  callStatus := call(gas(), token, 0, freeMemoryPointer, 100, 0, 0)
              }
              require(didLastOptionalReturnCallSucceed(callStatus), "TRANSFER_FROM_FAILED");
          }
          function safeTransfer(
              ERC20 token,
              address to,
              uint256 amount
          ) internal {
              bool callStatus;
              assembly {
                  // Get a pointer to some free memory.
                  let freeMemoryPointer := mload(0x40)
                  // Write the abi-encoded calldata to memory piece by piece:
                  mstore(freeMemoryPointer, 0xa9059cbb00000000000000000000000000000000000000000000000000000000) // Begin with the function selector.
                  mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Mask and append the "to" argument.
                  mstore(add(freeMemoryPointer, 36), amount) // Finally append the "amount" argument. No mask as it's a full 32 byte value.
                  // Call the token and store if it succeeded or not.
                  // We use 68 because the calldata length is 4 + 32 * 2.
                  callStatus := call(gas(), token, 0, freeMemoryPointer, 68, 0, 0)
              }
              require(didLastOptionalReturnCallSucceed(callStatus), "TRANSFER_FAILED");
          }
          function safeApprove(
              ERC20 token,
              address to,
              uint256 amount
          ) internal {
              bool callStatus;
              assembly {
                  // Get a pointer to some free memory.
                  let freeMemoryPointer := mload(0x40)
                  // Write the abi-encoded calldata to memory piece by piece:
                  mstore(freeMemoryPointer, 0x095ea7b300000000000000000000000000000000000000000000000000000000) // Begin with the function selector.
                  mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Mask and append the "to" argument.
                  mstore(add(freeMemoryPointer, 36), amount) // Finally append the "amount" argument. No mask as it's a full 32 byte value.
                  // Call the token and store if it succeeded or not.
                  // We use 68 because the calldata length is 4 + 32 * 2.
                  callStatus := call(gas(), token, 0, freeMemoryPointer, 68, 0, 0)
              }
              require(didLastOptionalReturnCallSucceed(callStatus), "APPROVE_FAILED");
          }
          /*///////////////////////////////////////////////////////////////
                               INTERNAL HELPER LOGIC
          //////////////////////////////////////////////////////////////*/
          function didLastOptionalReturnCallSucceed(bool callStatus) private pure returns (bool success) {
              assembly {
                  // Get how many bytes the call returned.
                  let returnDataSize := returndatasize()
                  // If the call reverted:
                  if iszero(callStatus) {
                      // Copy the revert message into memory.
                      returndatacopy(0, 0, returnDataSize)
                      // Revert with the same message.
                      revert(0, returnDataSize)
                  }
                  switch returnDataSize
                  case 32 {
                      // Copy the return data into memory.
                      returndatacopy(0, 0, returnDataSize)
                      // Set success to whether it returned true.
                      success := iszero(iszero(mload(0)))
                  }
                  case 0 {
                      // There was no return data.
                      success := 1
                  }
                  default {
                      // It returned some malformed input.
                      success := 0
                  }
              }
          }
      }
      

      File 4 of 4: Collection
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Data stored per-token, fits into a single storage word
      struct TokenData {
          address owner;
          uint16 sequenceId;
          uint80 data;
      }
      /// @notice Modern, minimalist, and gas efficient ERC-721 implementation.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol)
      abstract contract ERC721 {
          /*//////////////////////////////////////////////////////////////
                                       EVENTS
          //////////////////////////////////////////////////////////////*/
          event Transfer(address indexed from, address indexed to, uint256 indexed id);
          event Approval(address indexed owner, address indexed spender, uint256 indexed id);
          event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
          /*//////////////////////////////////////////////////////////////
                               METADATA STORAGE/LOGIC
          //////////////////////////////////////////////////////////////*/
          function tokenURI(uint256 id) public view virtual returns (string memory);
          /*//////////////////////////////////////////////////////////////
                            ERC721 BALANCE/OWNER STORAGE
          //////////////////////////////////////////////////////////////*/
          mapping(uint256 => TokenData) internal _tokenData;
          mapping(address => uint256) internal _balanceOf;
          function ownerOf(uint256 id) public view virtual returns (address owner) {
              require((owner = _tokenData[id].owner) != address(0), "NOT_MINTED");
          }
          function balanceOf(address owner) public view virtual returns (uint256) {
              require(owner != address(0), "ZERO_ADDRESS");
              return _balanceOf[owner];
          }
          /*//////////////////////////////////////////////////////////////
                               ERC721 APPROVAL STORAGE
          //////////////////////////////////////////////////////////////*/
          mapping(uint256 => address) public getApproved;
          mapping(address => mapping(address => bool)) public isApprovedForAll;
          /*//////////////////////////////////////////////////////////////
                                    ERC721 LOGIC
          //////////////////////////////////////////////////////////////*/
          function approve(address spender, uint256 id) public virtual {
              address owner = _tokenData[id].owner;
              require(msg.sender == owner || isApprovedForAll[owner][msg.sender], "NOT_AUTHORIZED");
              getApproved[id] = spender;
              emit Approval(owner, spender, id);
          }
          function setApprovalForAll(address operator, bool approved) public virtual {
              isApprovedForAll[msg.sender][operator] = approved;
              emit ApprovalForAll(msg.sender, operator, approved);
          }
          function transferFrom(
              address from,
              address to,
              uint256 id
          ) public virtual {
              require(from == _tokenData[id].owner, "WRONG_FROM");
              require(to != address(0), "INVALID_RECIPIENT");
              require(
                  msg.sender == from || isApprovedForAll[from][msg.sender] || msg.sender == getApproved[id],
                  "NOT_AUTHORIZED"
              );
              // Underflow of the sender's balance is impossible because we check for
              // ownership above and the recipient's balance can't realistically overflow.
              unchecked {
                  _balanceOf[from]--;
                  _balanceOf[to]++;
              }
              _tokenData[id].owner = to;
              delete getApproved[id];
              emit Transfer(from, to, id);
          }
          function safeTransferFrom(
              address from,
              address to,
              uint256 id
          ) public virtual {
              transferFrom(from, to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, "") ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          function safeTransferFrom(
              address from,
              address to,
              uint256 id,
              bytes calldata data
          ) public virtual {
              transferFrom(from, to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, data) ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          /*//////////////////////////////////////////////////////////////
                                    ERC165 LOGIC
          //////////////////////////////////////////////////////////////*/
          function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
              return
                  interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165
                  interfaceId == 0x80ac58cd || // ERC165 Interface ID for ERC721
                  interfaceId == 0x5b5e139f; // ERC165 Interface ID for ERC721Metadata
          }
          /*//////////////////////////////////////////////////////////////
                              INTERNAL MINT/BURN LOGIC
          //////////////////////////////////////////////////////////////*/
          function _mint(address to, uint256 id) internal virtual {
              return _mint(to, id, 0, 0);
          }
          function _mint(address to, uint256 id, uint16 sequenceId, uint80 data) internal virtual {
              require(to != address(0), "INVALID_RECIPIENT");
              require(_tokenData[id].owner == address(0), "ALREADY_MINTED");
              // Counter overflow is incredibly unrealistic.
              unchecked {
                  _balanceOf[to]++;
              }
              _tokenData[id] = TokenData({
                  owner: to,
                  sequenceId: sequenceId,
                  data: data
              });
              emit Transfer(address(0), to, id);
          }
          function _burn(uint256 id) internal virtual {
              address owner = _tokenData[id].owner;
              require(owner != address(0), "NOT_MINTED");
              // Ownership check above ensures no underflow.
              unchecked {
                  _balanceOf[owner]--;
              }
              delete _tokenData[id];
              delete getApproved[id];
              emit Transfer(owner, address(0), id);
          }
          /*//////////////////////////////////////////////////////////////
                              INTERNAL SAFE MINT LOGIC
          //////////////////////////////////////////////////////////////*/
          function _safeMint(address to, uint256 id) internal virtual {
              _mint(to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, "") ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          function _safeMint(
              address to,
              uint256 id,
              bytes memory data
          ) internal virtual {
              _mint(to, id);
              require(
                  to.code.length == 0 ||
                      ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, data) ==
                      ERC721TokenReceiver.onERC721Received.selector,
                  "UNSAFE_RECIPIENT"
              );
          }
          /*//////////////////////////////////////////////////////////////
                          METALABEL ADDED FUNCTIONALITY
          //////////////////////////////////////////////////////////////*/
          function getTokenData(uint256 id) external view virtual returns (TokenData memory) {
              TokenData memory data = _tokenData[id];
              require(data.owner != address(0), "NOT_MINTED");
              return data;
          }
      }
      /// @notice A generic interface for a contract which properly accepts ERC721 tokens.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol)
      abstract contract ERC721TokenReceiver {
          function onERC721Received(
              address,
              address,
              uint256,
              bytes calldata
          ) external virtual returns (bytes4) {
              return ERC721TokenReceiver.onERC721Received.selector;
          }
      }
      // SPDX-License-Identifier: AGPL-3.0-only
      pragma solidity >=0.8.0;
      /// @notice Read and write to persistent storage at a fraction of the cost.
      /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SSTORE2.sol)
      /// @author Modified from 0xSequence (https://github.com/0xSequence/sstore2/blob/master/contracts/SSTORE2.sol)
      library SSTORE2 {
          uint256 internal constant DATA_OFFSET = 1; // We skip the first byte as it's a STOP opcode to ensure the contract can't be called.
          /*//////////////////////////////////////////////////////////////
                                     WRITE LOGIC
          //////////////////////////////////////////////////////////////*/
          function write(bytes memory data) internal returns (address pointer) {
              // Prefix the bytecode with a STOP opcode to ensure it cannot be called.
              bytes memory runtimeCode = abi.encodePacked(hex"00", data);
              bytes memory creationCode = abi.encodePacked(
                  //---------------------------------------------------------------------------------------------------------------//
                  // Opcode  | Opcode + Arguments  | Description  | Stack View                                                     //
                  //---------------------------------------------------------------------------------------------------------------//
                  // 0x60    |  0x600B             | PUSH1 11     | codeOffset                                                     //
                  // 0x59    |  0x59               | MSIZE        | 0 codeOffset                                                   //
                  // 0x81    |  0x81               | DUP2         | codeOffset 0 codeOffset                                        //
                  // 0x38    |  0x38               | CODESIZE     | codeSize codeOffset 0 codeOffset                               //
                  // 0x03    |  0x03               | SUB          | (codeSize - codeOffset) 0 codeOffset                           //
                  // 0x80    |  0x80               | DUP          | (codeSize - codeOffset) (codeSize - codeOffset) 0 codeOffset   //
                  // 0x92    |  0x92               | SWAP3        | codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset)   //
                  // 0x59    |  0x59               | MSIZE        | 0 codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset) //
                  // 0x39    |  0x39               | CODECOPY     | 0 (codeSize - codeOffset)                                      //
                  // 0xf3    |  0xf3               | RETURN       |                                                                //
                  //---------------------------------------------------------------------------------------------------------------//
                  hex"60_0B_59_81_38_03_80_92_59_39_F3", // Returns all code in the contract except for the first 11 (0B in hex) bytes.
                  runtimeCode // The bytecode we want the contract to have after deployment. Capped at 1 byte less than the code size limit.
              );
              assembly {
                  // Deploy a new contract with the generated creation code.
                  // We start 32 bytes into the code to avoid copying the byte length.
                  pointer := create(0, add(creationCode, 32), mload(creationCode))
              }
              require(pointer != address(0), "DEPLOYMENT_FAILED");
          }
          /*//////////////////////////////////////////////////////////////
                                     READ LOGIC
          //////////////////////////////////////////////////////////////*/
          function read(address pointer) internal view returns (bytes memory) {
              return readBytecode(pointer, DATA_OFFSET, pointer.code.length - DATA_OFFSET);
          }
          function read(address pointer, uint256 start) internal view returns (bytes memory) {
              start += DATA_OFFSET;
              return readBytecode(pointer, start, pointer.code.length - start);
          }
          function read(
              address pointer,
              uint256 start,
              uint256 end
          ) internal view returns (bytes memory) {
              start += DATA_OFFSET;
              end += DATA_OFFSET;
              require(pointer.code.length >= end, "OUT_OF_BOUNDS");
              return readBytecode(pointer, start, end - start);
          }
          /*//////////////////////////////////////////////////////////////
                                INTERNAL HELPER LOGIC
          //////////////////////////////////////////////////////////////*/
          function readBytecode(
              address pointer,
              uint256 start,
              uint256 size
          ) private view returns (bytes memory data) {
              assembly {
                  // Get a pointer to some free memory.
                  data := mload(0x40)
                  // Update the free memory pointer to prevent overriding our data.
                  // We use and(x, not(31)) as a cheaper equivalent to sub(x, mod(x, 32)).
                  // Adding 31 to size and running the result through the logic above ensures
                  // the memory pointer remains word-aligned, following the Solidity convention.
                  mstore(0x40, add(data, and(add(add(size, 32), 31), not(31))))
                  // Store the size of the data in the first 32 byte chunk of free memory.
                  mstore(data, size)
                  // Copy the code into memory right after the 32 bytes we used to store the size.
                  extcodecopy(pointer, add(data, 32), start, size)
              }
          }
      }
      // SPDX-License-Identifier: MIT
      // OpenZeppelin Contracts v4.4.1 (interfaces/IERC165.sol)
      pragma solidity ^0.8.0;
      import "../utils/introspection/IERC165.sol";
      // SPDX-License-Identifier: MIT
      // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)
      pragma solidity ^0.8.0;
      import "../utils/introspection/IERC165.sol";
      /**
       * @dev Interface for the NFT Royalty Standard.
       *
       * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal
       * support for royalty payments across all NFT marketplaces and ecosystem participants.
       *
       * _Available since v4.5._
       */
      interface IERC2981 is IERC165 {
          /**
           * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of
           * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.
           */
          function royaltyInfo(uint256 tokenId, uint256 salePrice)
              external
              view
              returns (address receiver, uint256 royaltyAmount);
      }
      // 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.17;
      /*
      ███╗   ███╗███████╗████████╗ █████╗ ██╗      █████╗ ██████╗ ███████╗██╗
      ████╗ ████║██╔════╝╚══██╔══╝██╔══██╗██║     ██╔══██╗██╔══██╗██╔════╝██║
      ██╔████╔██║█████╗     ██║   ███████║██║     ███████║██████╔╝█████╗  ██║
      ██║╚██╔╝██║██╔══╝     ██║   ██╔══██║██║     ██╔══██║██╔══██╗██╔══╝  ██║
      ██║ ╚═╝ ██║███████╗   ██║   ██║  ██║███████╗██║  ██║██████╔╝███████╗███████╗
      ╚═╝     ╚═╝╚══════╝   ╚═╝   ╚═╝  ╚═╝╚══════╝╚═╝  ╚═╝╚═════╝ ╚══════╝╚══════╝
      Deployed by Metalabel with 💖 as a permanent application on the Ethereum blockchain.
      Metalabel is a growing universe of tools, knowledge, and resources for
      metalabels and cultural collectives.
      Our purpose is to establish the metalabel as key infrastructure for creative
      collectives and to inspire a new culture of creative collaboration and mutual
      support.
      OUR SQUAD
      Anna Bulbrook (Curator)
      Austin Robey (Community)
      Brandon Valosek (Engineer)
      Ilya Yudanov (Designer)
      Lauren Dorman (Engineer)
      Rob Kalin (Board)
      Yancey Strickler (Director)
      https://metalabel.xyz
      */
      import {IERC165} from "@openzeppelin/contracts/interfaces/IERC165.sol";
      import {IERC2981} from "@openzeppelin/contracts/interfaces/IERC2981.sol";
      import {ERC721} from "@metalabel/solmate/src/tokens/ERC721.sol";
      import {SSTORE2} from "@metalabel/solmate/src/utils/SSTORE2.sol";
      import {INodeRegistry} from "./interfaces/INodeRegistry.sol";
      import {IEngine, SequenceData} from "./interfaces/IEngine.sol";
      import {ICollection} from "./interfaces/ICollection.sol";
      import {Resource, AccessControlData} from "./Resource.sol";
      /// @notice Immutable data stored per-collection.
      /// @dev This is stored via SSTORE2 to save gas.
      struct ImmutableCollectionData {
          string name;
          string symbol;
          string contractURI;
      }
      /// @notice Collections are ERC721 contracts that contain records.
      /// - Minting logic, tokenURI, and royalties are delegated to an external engine
      ///     contract
      /// - Sequences are a mapping between an external engine contract and parameters
      ///     stored in the collection
      /// - Multiple sequences can be configured for a single collection, records may
      ///     be rendered and minted in a variety of different ways
      contract Collection is ERC721, Resource, ICollection, IERC2981 {
          // ---
          // Errors
          // ---
          /// @notice The init function was called more than once.
          error AlreadyInitialized();
          /// @notice A record mint attempt was made for a sequence that is currently
          /// sealed.
          error SequenceIsSealed();
          /// @notice A record mint attempt was made for a sequence that has no
          /// remaining supply.
          error SequenceSupplyExhausted();
          /// @notice An invalid sequence config was provided during configuration.
          error InvalidSequenceConfig();
          /// @notice msg.sender during a mint call did not match expected engine
          /// origin.
          error InvalidMintRequest();
          // ---
          // Events
          // ---
          /// @notice A new record was minted.
          /// @dev The underlying ERC721 implementation already emits a Transfer event
          /// on mint, this event announces the sequence the token is minted into and
          /// its immutable token data.
          event RecordCreated(
              uint256 indexed tokenId,
              uint16 indexed sequenceId,
              uint80 data
          );
          /// @notice A sequence has been set or updated.
          event SequenceConfigured(
              uint16 indexed sequenceId,
              SequenceData sequenceData,
              bytes engineData
          );
          /// @notice The owner address of this collection was updated.
          event OwnershipTransferred(
              address indexed previousOwner,
              address indexed newOwner
          );
          // ---
          // Storage
          // ---
          /// @notice Total number of records minted in this collection.
          uint256 public totalSupply;
          /// @notice Total number of sequences configured in this collection.
          uint16 public sequenceCount;
          /// @notice Only for marketplace interop, can be set by owner of the control
          /// node.
          address public owner;
          /// @notice The SSTORE2 storage pointer for immutable collection data.
          /// @dev This data is exposed through name, symbol, and contractURI views
          address internal immutableStoragePointer;
          /// @notice Information about each sequence.
          mapping(uint16 => SequenceData) internal sequences;
          // ---
          // Constructor
          // ---
          /// @dev Constructor only called during deployment of the implementation,
          /// all storage should be set up in init function which is called atomically
          /// after clone deployment
          constructor() {
              // Write dummy data to the immutable storage pointer to prevent
              // initialization of the implementation contract.
              immutableStoragePointer = SSTORE2.write(
                  abi.encode(
                      ImmutableCollectionData({name: "", symbol: "", contractURI: ""})
                  )
              );
          }
          // ---
          // Clone init
          // ---
          /// @notice Initialize contract state.
          /// @dev Should be called immediately after deploying the clone in the same
          /// transaction.
          function init(
              address _owner,
              AccessControlData calldata _accessControl,
              string calldata _metadata,
              ImmutableCollectionData calldata _data
          ) external {
              if (immutableStoragePointer != address(0)) revert AlreadyInitialized();
              immutableStoragePointer = SSTORE2.write(abi.encode(_data));
              // Set ERC721 market interop.
              owner = _owner;
              emit OwnershipTransferred(address(0), owner);
              // Assign access control data.
              accessControl = _accessControl;
              // This memberships collection is a resource that can be cataloged -
              // emit the initial metadata value
              emit Broadcast("metadata", _metadata);
          }
          // ---
          // Admin functionality
          // ---
          /// @notice Change the owner address of this collection.
          /// @dev This is only here for market interop, access control is handled via
          /// the control node.
          function setOwner(address _owner) external onlyAuthorized {
              address previousOwner = owner;
              owner = _owner;
              emit OwnershipTransferred(previousOwner, _owner);
          }
          /// @notice Create a new sequence configuration.
          /// @dev The _engineData bytes parameter is arbitrary data that is passed
          /// directly to the engine powering this new sequence.
          function configureSequence(
              SequenceData calldata _sequence,
              bytes calldata _engineData
          ) external onlyAuthorized {
              // The drop this sequence is associated with must be manageable by
              // msg.sender. This is in addition to the onlyAuthorized modifier which
              // asserts msg.sender can manage the control node of the whole
              // collection.
              // msg.sender is either a metalabel admin EOA, or a controller contract
              // that has been authorized to do drops on the drop node.
              if (
                  !accessControl.nodeRegistry.isAuthorizedAddressForNode(
                      _sequence.dropNodeId,
                      msg.sender
                  )
              ) {
                  revert NotAuthorized();
              }
              // If there is sealedAfter timestamp (i.e timebound sequence), ensure
              // that sealedBefore is strictly less than sealedAfter AND that
              // sealedAfter occurs strictly in the future. If sealedAfter is zero,
              // there is no time limit. We are allowing cases where sealedBefore
              // occurs in the past.
              if (
                  _sequence.sealedAfterTimestamp > 0 &&
                  (_sequence.sealedBeforeTimestamp >=
                      _sequence.sealedAfterTimestamp ||
                      _sequence.sealedAfterTimestamp <= block.timestamp)
              ) {
                  revert InvalidSequenceConfig();
              }
              // Prevent having a minted count before the sequence starts. This
              // wouldn't break anything, but would cause the indexed "minted" amount
              // from actual mints to differ from the sequence data tracking total
              // supply, which is non-ideal and worth the small gas to check.
              //
              // We're not using a separate struct here for inputs that omits the
              // minted count field, being able to copy from calldata to storage is
              // nice.
              if (_sequence.minted != 0) {
                  revert InvalidSequenceConfig();
              }
              // Write sequence data to storage
              uint16 sequenceId = ++sequenceCount;
              sequences[sequenceId] = _sequence;
              emit SequenceConfigured(sequenceId, _sequence, _engineData);
              // Invoke configureSequence on the engine to give it a chance to setup
              // and store any needed info. Doing this after event emitting so that
              // indexers see the sequence first before any engine-side events
              _sequence.engine.configureSequence(sequenceId, _sequence, _engineData);
          }
          // ---
          // Engine functionality
          // ---
          /// @inheritdoc ICollection
          function mintRecord(
              address to,
              uint16 sequenceId,
              uint80 tokenData
          ) external returns (uint256 tokenId) {
              SequenceData storage sequence = sequences[sequenceId];
              _validateSequence(sequence);
              // Mint the record.
              tokenId = ++totalSupply;
              ++sequence.minted;
              _mint(to, tokenId, sequenceId, tokenData);
              emit RecordCreated(tokenId, sequenceId, tokenData);
          }
          /// @inheritdoc ICollection
          function mintRecord(address to, uint16 sequenceId)
              external
              returns (uint256 tokenId)
          {
              SequenceData storage sequence = sequences[sequenceId];
              _validateSequence(sequence);
              // Mint the record.
              tokenId = ++totalSupply;
              uint64 editionNumber = ++sequence.minted;
              _mint(to, tokenId, sequenceId, editionNumber);
              emit RecordCreated(tokenId, sequenceId, editionNumber);
          }
          /// @dev Ensure a given sequence is valid to mint into by the current msg.sender
          function _validateSequence(SequenceData memory sequence) internal view {
              // Ensure that only the engine for this sequence can mint records. Mint
              // transactions termiante on the engine side - the engine then invokes
              // the mint functions on the Collection.
              if (sequence.engine != IEngine(msg.sender)) {
                  revert InvalidMintRequest();
              }
              // Ensure that mint is not happening before or after allowed window.
              if (
                  block.timestamp < sequence.sealedBeforeTimestamp ||
                  (sequence.sealedAfterTimestamp > 0 && // sealed after = 0 => no end
                      block.timestamp >= sequence.sealedAfterTimestamp)
              ) {
                  revert SequenceIsSealed();
              }
              // Ensure we have remaining supply to mint
              if (sequence.maxSupply > 0 && sequence.minted >= sequence.maxSupply) {
                  revert SequenceSupplyExhausted();
              }
          }
          // ---
          // ICollection views
          // ---
          /// @inheritdoc ICollection
          function getSequenceData(uint16 sequenceId)
              external
              view
              override
              returns (SequenceData memory sequence)
          {
              sequence = sequences[sequenceId];
          }
          // ---
          // ERC721 views
          // ---
          /// @notice The collection name.
          function name() public view virtual returns (string memory value) {
              value = _resolveImmutableStorage().name;
          }
          /// @notice The collection symbol.
          function symbol() public view virtual returns (string memory value) {
              value = _resolveImmutableStorage().symbol;
          }
          /// @inheritdoc ERC721
          /// @dev Resolve token URI from the engine powering the sequence.
          function tokenURI(uint256 tokenId)
              public
              view
              virtual
              override
              returns (string memory uri)
          {
              IEngine engine = sequences[_tokenData[tokenId].sequenceId].engine;
              uri = engine.getTokenURI(address(this), tokenId);
          }
          /// @notice Get the contract URI.
          function contractURI() public view virtual returns (string memory value) {
              value = _resolveImmutableStorage().contractURI;
          }
          // ---
          // Misc views
          // ---
          /// @inheritdoc ICollection
          function tokenSequenceId(uint256 tokenId)
              external
              view
              returns (uint16 sequenceId)
          {
              sequenceId = _tokenData[tokenId].sequenceId;
          }
          /// @inheritdoc ICollection
          /// @dev Token mint data is either edition number or arbitrary custom data
          /// passed in the by the engine at mint-time.
          function tokenMintData(uint256 tokenId)
              external
              view
              returns (uint80 data)
          {
              data = _tokenData[tokenId].data;
          }
          // ---
          // ERC2981 functionality
          // ---
          /// @inheritdoc IERC2981
          /// @dev Resolve royalty info from the engine powering the sequence.
          function royaltyInfo(uint256 tokenId, uint256 salePrice)
              external
              view
              returns (address receiver, uint256 royaltyAmount)
          {
              IEngine engine = sequences[_tokenData[tokenId].sequenceId].engine;
              return engine.getRoyaltyInfo(address(this), tokenId, salePrice);
          }
          // ---
          // Introspection
          // ---
          /// @inheritdoc IERC165
          /// @dev ERC165 checks return true for: ERC165, ERC721, and ERC2981.
          function supportsInterface(bytes4 interfaceId)
              public
              view
              virtual
              override(ERC721, IERC165)
              returns (bool)
          {
              return
                  interfaceId == type(IERC2981).interfaceId ||
                  super.supportsInterface(interfaceId);
          }
          // ---
          // Internal views
          // ---
          /// @dev Read name/symbol/contractURI strings via SSTORE2.
          function _resolveImmutableStorage()
              internal
              view
              returns (ImmutableCollectionData memory data)
          {
              data = abi.decode(
                  SSTORE2.read(immutableStoragePointer),
                  (ImmutableCollectionData)
              );
          }
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      import {SequenceData} from "./IEngine.sol";
      /// @notice Collections are ERC721 contracts that contain records.
      interface ICollection {
          /// @notice Mint a new record with custom immutable token data. Only
          /// callable by the sequence-specific engine.
          function mintRecord(
              address to,
              uint16 sequenceId,
              uint80 tokenData
          ) external returns (uint256 tokenId);
          /// @notice Mint a new record with the edition number of the sequence
          /// written to the immutable token data. Only callable by the
          /// sequence-specific engine.
          function mintRecord(address to, uint16 sequenceId)
              external
              returns (uint256 tokenId);
          /// @notice Get the sequence ID for a given token.
          function tokenSequenceId(uint256 tokenId)
              external
              view
              returns (uint16 sequenceId);
          /// @notice Get the immutable mint data for a given token.
          function tokenMintData(uint256 tokenId) external view returns (uint80 data);
          /// @notice Get the sequence data for a given sequence ID.
          function getSequenceData(uint16 sequenceId)
              external
              view
              returns (SequenceData memory);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      /// @notice Data stored in the collection for each sequence.
      /// @dev We could use smaller ints for timestamps and supply, but we'd still be
      /// stuck with a 2-word storage layout. engine + dropNodeId is 28 bytes, leaving
      /// us with only 4 bytes for the remaining parameters.
      struct SequenceData {
          uint64 sealedBeforeTimestamp;
          uint64 sealedAfterTimestamp;
          uint64 maxSupply;
          uint64 minted;
          // ^ 1 word
          IEngine engine;
          uint64 dropNodeId;
          // 4 bytes remaining
      }
      /// @notice An engine contract implements record minting mechanics, tokenURI
      /// computation, and royalty computation.
      interface IEngine {
          /// @notice Called by the collection when a new sequence is configured.
          /// @dev An arbitrary bytes buffer engineData is forwarded from the
          /// collection that can be used to pass setup and configuration data
          function configureSequence(
              uint16 sequenceId,
              SequenceData calldata sequence,
              bytes calldata engineData
          ) external;
          /// @notice Called by the collection to resolve tokenURI.
          function getTokenURI(address collection, uint256 tokenId)
              external
              view
              returns (string memory);
          /// @notice Called by the collection to resolve royalties.
          function getRoyaltyInfo(
              address collection,
              uint256 tokenId,
              uint256 salePrice
          ) external view returns (address receiver, uint256 royaltyAmount);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      enum NodeType {
          INVALID_NODE_TYPE,
          METALABEL,
          RELEASE
      }
      /// @notice Data stored per node.
      struct NodeData {
          NodeType nodeType;
          uint64 owner;
          uint64 parent;
          uint64 groupNode;
          // 7 bytes remaining
      }
      /// @notice The node registry maintains a tree of ownable nodes that are used to
      /// catalog logical entities and manage access control in the Metalabel
      /// universe.
      interface INodeRegistry {
          /// @notice Create a new node. Child nodes can specify an group node that
          /// will be used to determine ownership, and a separate logical parent that
          /// expresses the entity relationship.  Child nodes can only be created if
          /// msg.sender is an authorized manager of the parent node.
          function createNode(
              NodeType nodeType,
              uint64 owner,
              uint64 parent,
              uint64 groupNode,
              address[] memory initialControllers,
              string memory metadata
          ) external returns (uint64 id);
          /// @notice Determine if an address is authorized to manage a node.
          /// A node can be managed by an address if any of the following conditions
          /// are true:
          ///   - The address's account is the owner of the node
          ///   - The address's account is the owner of the node's group node
          ///   - The address is an authorized controller of the node
          ///   - The address is an authorized controller of the node's group node
          function isAuthorizedAddressForNode(uint64 node, address subject)
              external
              view
              returns (bool isAuthorized);
          /// @notice Resolve node owner account.
          function ownerOf(uint64 id) external view returns (uint64);
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      import {INodeRegistry} from "./INodeRegistry.sol";
      /// @notice An on-chain resource that is intended to be cataloged within the
      /// Metalabel universe
      interface IResource {
          /// @notice Broadcast an arbitrary message.
          event Broadcast(string topic, string message);
          /// @notice Return the node registry contract address.
          function nodeRegistry() external view returns (INodeRegistry);
          /// @notice Return the control node ID for this resource.
          function controlNode() external view returns (uint64 nodeId);
          /// @notice Return true if the given address is authorized to manage this
          /// resource.
          function isAuthorized(address subject)
              external
              view
              returns (bool authorized);
          /// @notice Emit an on-chain message. msg.sender must be authorized to
          /// manage this resource's control node
          function broadcast(string calldata topic, string calldata message) external;
      }
      // SPDX-License-Identifier: MIT
      pragma solidity 0.8.17;
      /*
      ███╗   ███╗███████╗████████╗ █████╗ ██╗      █████╗ ██████╗ ███████╗██╗
      ████╗ ████║██╔════╝╚══██╔══╝██╔══██╗██║     ██╔══██╗██╔══██╗██╔════╝██║
      ██╔████╔██║█████╗     ██║   ███████║██║     ███████║██████╔╝█████╗  ██║
      ██║╚██╔╝██║██╔══╝     ██║   ██╔══██║██║     ██╔══██║██╔══██╗██╔══╝  ██║
      ██║ ╚═╝ ██║███████╗   ██║   ██║  ██║███████╗██║  ██║██████╔╝███████╗███████╗
      ╚═╝     ╚═╝╚══════╝   ╚═╝   ╚═╝  ╚═╝╚══════╝╚═╝  ╚═╝╚═════╝ ╚══════╝╚══════╝
      Deployed by Metalabel with 💖 as a permanent application on the Ethereum blockchain.
      Metalabel is a growing universe of tools, knowledge, and resources for
      metalabels and cultural collectives.
      Our purpose is to establish the metalabel as key infrastructure for creative
      collectives and to inspire a new culture of creative collaboration and mutual
      support.
      OUR SQUAD
      Anna Bulbrook (Curator)
      Austin Robey (Community)
      Brandon Valosek (Engineer)
      Ilya Yudanov (Designer)
      Lauren Dorman (Engineer)
      Rob Kalin (Board)
      Yancey Strickler (Director)
      https://metalabel.xyz
      */
      import {IResource} from "./interfaces/IResource.sol";
      import {INodeRegistry} from "./interfaces/INodeRegistry.sol";
      /// @notice Data stored for handling access control resolution.
      struct AccessControlData {
          INodeRegistry nodeRegistry;
          uint64 controlNodeId;
          // 4 bytes remaining
      }
      /// @notice A resource that can be cataloged on the Metalabel protocol.
      contract Resource is IResource {
          // ---
          // Errors
          // ---
          /// @notice Unauthorized msg.sender attempted to interact with this resource
          error NotAuthorized();
          // ---
          // Storage
          // ---
          /// @notice Access control data for this resource.
          AccessControlData public accessControl;
          // ---
          // Modifiers
          // ---
          /// @dev Make a function only callable by a msg.sender that is authorized to
          /// manage the control node of this resource
          modifier onlyAuthorized() {
              if (
                  !accessControl.nodeRegistry.isAuthorizedAddressForNode(
                      accessControl.controlNodeId,
                      msg.sender
                  )
              ) {
                  revert NotAuthorized();
              }
              _;
          }
          // ---
          // Admin functionality
          // ---
          /// @inheritdoc IResource
          function broadcast(string calldata topic, string calldata message)
              external
              onlyAuthorized
          {
              emit Broadcast(topic, message);
          }
          // ---
          // Resource views
          // ---
          /// @inheritdoc IResource
          function nodeRegistry() external view virtual returns (INodeRegistry) {
              return accessControl.nodeRegistry;
          }
          /// @inheritdoc IResource
          function controlNode() external view virtual returns (uint64 nodeId) {
              return accessControl.controlNodeId;
          }
          /// @inheritdoc IResource
          function isAuthorized(address subject)
              public
              view
              virtual
              returns (bool authorized)
          {
              authorized = accessControl.nodeRegistry.isAuthorizedAddressForNode(
                  accessControl.controlNodeId,
                  subject
              );
          }
      }