APE Price: $0.47 (-0.14%)

Contract

0xB1Ca0595fedDcED494823195d72F9300e7e3975B

Overview

APE Balance

Apechain LogoApechain LogoApechain Logo0 APE

APE Value

$0.00

Multichain Info

No addresses found
Transaction Hash
Method
Block
From
To

There are no matching entries

Please try again later

Parent Transaction Hash Block From To
View All Internal Transactions

Loading...
Loading

Contract Source Code Verified (Exact Match)

Contract Name:
DungeonGame

Compiler Version
v0.8.20+commit.a1b79de6

Optimization Enabled:
Yes with 200 runs

Other Settings:
shanghai EvmVersion
File 1 of 11 : DungeonGame.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/utils/ReentrancyGuard.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "./interfaces/IDungeonGame.sol";
import "./interfaces/IDungeonEntry.sol";
import "./interfaces/INFTStats.sol";
import "./interfaces/IPrizePool.sol";
import "./libraries/EncounterLibrary.sol";

/// @title DungeonGame
/// @notice Core game logic for dungeon runs and encounters
contract DungeonGame is IDungeonGame, Ownable(msg.sender), ReentrancyGuard {
    // Events
    event FeesWithdrawn(address indexed owner, uint256 amount);
    event DungeonEntered(
        address indexed collection,
        uint256 indexed tokenId,
        uint256 indexed entryIndex
    );

    event CharacterSurvived(
        address indexed collection,
        uint256 indexed tokenId,
        uint256 newHp,
        uint256 xpGained
    );

    // Allow contract to receive native currency (for Pyth refunds)
    receive() external payable {}

    // Constants
    uint8 public constant ROOM_COUNT = 16;

    uint256 private constant BASE_XP = 100;
    uint256 private constant COMPLETION_BONUS = 500;

    CharacterState private emptyRoom =
        CharacterState({collection: address(0), tokenId: 0, currentHp: 0});

    // Core contract references
    address public dungeonEntry;
    address public immutable nftStats;
    address public immutable prizePool;

    uint256 public currentIndex = 0;
    uint8 public startRoom = 0;

    CharacterState[ROOM_COUNT] public rooms;

    mapping(uint64 => CharacterState) private pendingEntries;
    mapping(address => mapping(uint256 => uint256)) private characterEntryIndex;

    constructor(address _dungeonEntry, address _nftStats, address _prizePool) {
        require(_dungeonEntry != address(0), "Invalid DungeonEntry address");
        require(_nftStats != address(0), "Invalid NFTStats address");
        require(_prizePool != address(0), "Invalid PrizePool address");

        dungeonEntry = _dungeonEntry;
        nftStats = _nftStats;
        prizePool = _prizePool;
    }

    /// @notice Get the state of a specific room
    function getRoomState(
        uint256 roomNumber
    ) external view returns (CharacterState memory) {
        require(
            roomNumber >= 0 && roomNumber <= ROOM_COUNT - 1,
            "Invalid room number"
        );
        return rooms[roomNumber];
    }

    function getCharacterRoomNumber(
        address collection,
        uint256 tokenId
    ) public view returns (uint8) {
        uint256 entryIndex = characterEntryIndex[collection][tokenId];
        uint8 roomNumber = uint8(
            (currentIndex - entryIndex + startRoom) % ROOM_COUNT
        );
        return roomNumber;
    }

    function getCharacterState(
        address collection,
        uint256 tokenId
    ) external view returns (CharacterState memory) {
        uint8 roomNumber = getCharacterRoomNumber(collection, tokenId);
        return rooms[roomNumber];
    }

    /// @notice Add a new character to the dungeon queue
    /// @dev Only DungeonEntry can add characters
    function enterDungeon(address collection, uint256 tokenId) external {
        require(
            msg.sender == address(dungeonEntry),
            "Only DungeonEntry can add characters"
        );

        bytes32 randomNumber = keccak256(
            abi.encodePacked(block.timestamp, block.prevrandao)
        );

        _processEntry(collection, tokenId, randomNumber);
    }

    /// @notice Process encounter
    function _processEntry(
        address collection,
        uint256 tokenId,
        bytes32 randomNumber
    ) internal {
        // Push all characters forward one room, starting from the last room
        for (uint8 i = 0; i < ROOM_COUNT; i++) {
            uint8 roomIndex = (ROOM_COUNT + startRoom - 1 - i) % ROOM_COUNT;

            CharacterState storage character = rooms[roomIndex];

            // Skip empty rooms
            if (character.collection == address(0)) {
                continue;
            }

            EncounterLibrary.Encounter memory encounter = EncounterLibrary
                .generateEncounter(
                    roomIndex + 1, // Add 1 since EncounterLibrary expects 1-based room numbers
                    uint256(randomNumber)
                );

            INFTStats.NFTStatsData memory characterStats = INFTStats(nftStats)
                .getStats(character.collection, character.tokenId);

            EncounterResult memory result = EncounterLibrary.processEncounter(
                encounter,
                characterStats
            );

            int256 newHp = int256(character.currentHp) + result.hpChange;
            bool survived = newHp > 0;

            // Calculate XP (partial XP if failed)
            uint256 xpGained = survived
                ? encounter.baseXp
                : encounter.baseXp / 2;

            // Update character stats
            if (survived) {
                emit CharacterSurvived(
                    character.collection,
                    character.tokenId,
                    uint256(newHp),
                    xpGained
                );
                character.currentHp = uint256(newHp);
                INFTStats(nftStats).awardXP(
                    character.collection,
                    character.tokenId,
                    xpGained,
                    1
                );
            }

            // Only move if they survived the encounter
            if (survived) {
                if (roomIndex == ROOM_COUNT - 1) {
                    // Character completed the dungeon
                    _handleDungeonCompletion(
                        character.collection,
                        character.tokenId
                    );
                }
            } else {
                // End the run in DungeonEntry
                IDungeonEntry(dungeonEntry).endDungeonRun(
                    character.collection,
                    character.tokenId,
                    false
                );

                // Remove dead character
                rooms[roomIndex] = CharacterState({
                    collection: address(0),
                    tokenId: 0,
                    currentHp: 0
                });
            }
        }

        INFTStats.NFTStatsData memory stats = INFTStats(nftStats).getStats(
            collection,
            tokenId
        );

        // Store the entry index for this character
        characterEntryIndex[collection][tokenId] = currentIndex;

        // Place the new character in the start room
        rooms[startRoom] = CharacterState({
            collection: collection,
            tokenId: tokenId,
            currentHp: stats.hp
        });
        emit DungeonEntered(collection, tokenId, currentIndex);

        // Increment indices for next entry
        startRoom = (startRoom + 1) % ROOM_COUNT;
        currentIndex++;
    }

    function _handleDungeonCompletion(
        address collection,
        uint256 tokenId
    ) internal {
        // Award completion bonus XP
        INFTStats(nftStats).awardXP(
            collection,
            tokenId,
            COMPLETION_BONUS,
            ROOM_COUNT
        );

        // Clear the last room
        rooms[ROOM_COUNT - 1] = emptyRoom;

        IPrizePool(prizePool).registerWinner(collection, tokenId);

        emit DungeonCompleted(collection, tokenId);
    }

    /// @notice Withdraw accumulated fees from Pyth refunds
    /// @dev Only callable by contract owner
    function withdrawFees() external onlyOwner {
        uint256 balance = address(this).balance;
        require(balance > 0, "No fees to withdraw");

        (bool success, ) = msg.sender.call{value: balance}("");
        require(success, "Withdrawal failed");

        emit FeesWithdrawn(msg.sender, balance);
    }

    function getPendingEntry(
        uint64 sequenceNumber
    ) external view returns (CharacterState memory) {
        return pendingEntries[sequenceNumber];
    }

    function setDungeonEntry(address _dungeonEntry) external onlyOwner {
        require(_dungeonEntry != address(0), "Invalid dungeon entry address");
        dungeonEntry = _dungeonEntry;
    }
}

File 2 of 11 : Ownable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol)

pragma solidity ^0.8.20;

import {Context} from "../utils/Context.sol";

/**
 * @dev Contract module which provides a basic access control mechanism, where
 * there is an account (an owner) that can be granted exclusive access to
 * specific functions.
 *
 * The initial owner is set to the address provided by the deployer. This can
 * later be changed with {transferOwnership}.
 *
 * This module is used through inheritance. It will make available the modifier
 * `onlyOwner`, which can be applied to your functions to restrict their use to
 * the owner.
 */
abstract contract Ownable is Context {
    address private _owner;

    /**
     * @dev The caller account is not authorized to perform an operation.
     */
    error OwnableUnauthorizedAccount(address account);

    /**
     * @dev The owner is not a valid owner account. (eg. `address(0)`)
     */
    error OwnableInvalidOwner(address owner);

    event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);

    /**
     * @dev Initializes the contract setting the address provided by the deployer as the initial owner.
     */
    constructor(address initialOwner) {
        if (initialOwner == address(0)) {
            revert OwnableInvalidOwner(address(0));
        }
        _transferOwnership(initialOwner);
    }

    /**
     * @dev Throws if called by any account other than the owner.
     */
    modifier onlyOwner() {
        _checkOwner();
        _;
    }

    /**
     * @dev Returns the address of the current owner.
     */
    function owner() public view virtual returns (address) {
        return _owner;
    }

    /**
     * @dev Throws if the sender is not the owner.
     */
    function _checkOwner() internal view virtual {
        if (owner() != _msgSender()) {
            revert OwnableUnauthorizedAccount(_msgSender());
        }
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby disabling any functionality that is only available to the owner.
     */
    function renounceOwnership() public virtual onlyOwner {
        _transferOwnership(address(0));
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Can only be called by the current owner.
     */
    function transferOwnership(address newOwner) public virtual onlyOwner {
        if (newOwner == address(0)) {
            revert OwnableInvalidOwner(address(0));
        }
        _transferOwnership(newOwner);
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Internal function without access restriction.
     */
    function _transferOwnership(address newOwner) internal virtual {
        address oldOwner = _owner;
        _owner = newOwner;
        emit OwnershipTransferred(oldOwner, newOwner);
    }
}

File 3 of 11 : ReentrancyGuard.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (utils/ReentrancyGuard.sol)

pragma solidity ^0.8.20;

/**
 * @dev Contract module that helps prevent reentrant calls to a function.
 *
 * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
 * available, which can be applied to functions to make sure there are no nested
 * (reentrant) calls to them.
 *
 * Note that because there is a single `nonReentrant` guard, functions marked as
 * `nonReentrant` may not call one another. This can be worked around by making
 * those functions `private`, and then adding `external` `nonReentrant` entry
 * points to them.
 *
 * TIP: If EIP-1153 (transient storage) is available on the chain you're deploying at,
 * consider using {ReentrancyGuardTransient} instead.
 *
 * TIP: If you would like to learn more about reentrancy and alternative ways
 * to protect against it, check out our blog post
 * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
 */
abstract contract ReentrancyGuard {
    // Booleans are more expensive than uint256 or any type that takes up a full
    // word because each write operation emits an extra SLOAD to first read the
    // slot's contents, replace the bits taken up by the boolean, and then write
    // back. This is the compiler's defense against contract upgrades and
    // pointer aliasing, and it cannot be disabled.

    // The values being non-zero value makes deployment a bit more expensive,
    // but in exchange the refund on every call to nonReentrant will be lower in
    // amount. Since refunds are capped to a percentage of the total
    // transaction's gas, it is best to keep them low in cases like this one, to
    // increase the likelihood of the full refund coming into effect.
    uint256 private constant NOT_ENTERED = 1;
    uint256 private constant ENTERED = 2;

    uint256 private _status;

    /**
     * @dev Unauthorized reentrant call.
     */
    error ReentrancyGuardReentrantCall();

    constructor() {
        _status = NOT_ENTERED;
    }

    /**
     * @dev Prevents a contract from calling itself, directly or indirectly.
     * Calling a `nonReentrant` function from another `nonReentrant`
     * function is not supported. It is possible to prevent this from happening
     * by making the `nonReentrant` function external, and making it call a
     * `private` function that does the actual work.
     */
    modifier nonReentrant() {
        _nonReentrantBefore();
        _;
        _nonReentrantAfter();
    }

    function _nonReentrantBefore() private {
        // On the first call to nonReentrant, _status will be NOT_ENTERED
        if (_status == ENTERED) {
            revert ReentrancyGuardReentrantCall();
        }

        // Any calls to nonReentrant after this point will fail
        _status = ENTERED;
    }

    function _nonReentrantAfter() private {
        // By storing the original value once again, a refund is triggered (see
        // https://eips.ethereum.org/EIPS/eip-2200)
        _status = NOT_ENTERED;
    }

    /**
     * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a
     * `nonReentrant` function in the call stack.
     */
    function _reentrancyGuardEntered() internal view returns (bool) {
        return _status == ENTERED;
    }
}

File 4 of 11 : IERC721.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (token/ERC721/IERC721.sol)

pragma solidity ^0.8.20;

import {IERC165} from "../../utils/introspection/IERC165.sol";

/**
 * @dev Required interface of an ERC-721 compliant contract.
 */
interface IERC721 is IERC165 {
    /**
     * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
     */
    event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
     */
    event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
     */
    event ApprovalForAll(address indexed owner, address indexed operator, bool approved);

    /**
     * @dev Returns the number of tokens in ``owner``'s account.
     */
    function balanceOf(address owner) external view returns (uint256 balance);

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) external view returns (address owner);

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon
     *   a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(address from, address to, uint256 tokenId, bytes calldata data) external;

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
     * are aware of the ERC-721 protocol to prevent tokens from being forever locked.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or
     *   {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon
     *   a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(address from, address to, uint256 tokenId) external;

    /**
     * @dev Transfers `tokenId` token from `from` to `to`.
     *
     * WARNING: Note that the caller is responsible to confirm that the recipient is capable of receiving ERC-721
     * or else they may be permanently lost. Usage of {safeTransferFrom} prevents loss, though the caller must
     * understand this adds an external call which potentially creates a reentrancy vulnerability.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(address from, address to, uint256 tokenId) external;

    /**
     * @dev Gives permission to `to` to transfer `tokenId` token to another account.
     * The approval is cleared when the token is transferred.
     *
     * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
     *
     * Requirements:
     *
     * - The caller must own the token or be an approved operator.
     * - `tokenId` must exist.
     *
     * Emits an {Approval} event.
     */
    function approve(address to, uint256 tokenId) external;

    /**
     * @dev Approve or remove `operator` as an operator for the caller.
     * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
     *
     * Requirements:
     *
     * - The `operator` cannot be the address zero.
     *
     * Emits an {ApprovalForAll} event.
     */
    function setApprovalForAll(address operator, bool approved) external;

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) external view returns (address operator);

    /**
     * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
     *
     * See {setApprovalForAll}
     */
    function isApprovedForAll(address owner, address operator) external view returns (bool);
}

File 5 of 11 : IDungeonGame.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

/// @title IDungeonGame
/// @notice Interface for core dungeon game mechanics and progression
interface IDungeonGame {
    /// @notice Structure for dungeon room state
    struct RoomState {
        address collection;
        uint256 tokenId;
        uint256 entryIndex;      // Used for ordering characters
        uint256 currentHp;
        uint256 currentAttack;
        uint256 currentSpeed;
        bool isOccupied;
    }

    /// @notice Structure for encounter results
    struct EncounterResult {
        int256 hpChange;
        string encounterDescription;
    }

    /// @notice Structure for character state
    struct CharacterState {
        address collection;
        uint256 tokenId;
        uint256 currentHp;
    }

    /// @notice Event emitted when an encounter is completed
    event EncounterCompleted(
        address indexed collection,
        uint256 indexed tokenId,
        uint256 roomNumber,
        uint256 xpGained,
        bool survived,
        string encounterDescription
    );
    
    /// @notice Event emitted when a dungeon run is completed
    event DungeonCompleted(
        address indexed collection,
        uint256 indexed tokenId
    );

    /// @notice Add a new character to the dungeon queue
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    function enterDungeon(address collection, uint256 tokenId) external;

    /// @notice Get the current room number for a character
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @return Current room number
    function getCharacterRoomNumber(address collection, uint256 tokenId) external view returns (uint8);

    /// @notice Get the current room state for a character
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @return Current room state
    function getCharacterState(address collection, uint256 tokenId) external view returns (CharacterState memory);


    /// @notice Get the state of a specific room
    /// @param roomNumber Room number to get state for
    /// @return CharacterState Memory of the room state
    function getRoomState(uint256 roomNumber) external view returns (CharacterState memory);
}

File 6 of 11 : IDungeonEntry.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

/// @title IDungeonEntry
/// @notice Interface for managing dungeon entry and run initialization
interface IDungeonEntry {
    /// @notice Structure for active dungeon run state
    struct DungeonRun {
        uint256 currentHp;
        uint256 currentRoom;
        bool isActive;
    }

    /// @notice Event emitted when a dungeon run is started
    event DungeonRunStarted(
        address indexed collection,
        uint256 indexed tokenId,
        address indexed player,
        uint256 entryFee
    );
    
    /// @notice Event emitted when a dungeon run is ended
    event DungeonRunEnded(
        address indexed collection,
        uint256 indexed tokenId,
        bool success
    );

    /// @notice Start a new dungeon run for an NFT
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    function startDungeonRun(address collection, uint256 tokenId) external payable;

    /// @notice End an active dungeon run (called by DungeonGame)
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @param success Whether the run was successful
    function endDungeonRun(address collection, uint256 tokenId, bool success) external;


    /// @notice Check if an NFT has an active dungeon run
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @return bool True if NFT has an active run
    function hasActiveRun(address collection, uint256 tokenId) external view returns (bool);

    /// @notice Get the current entry fee for dungeon runs
    /// @return uint256 Current entry fee in wei
    function getEntryFee() external view returns (uint256);
}

File 7 of 11 : INFTStats.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

/// @title INFTStats
/// @notice Interface for managing individual NFT stats and progression
interface INFTStats {
    /// @notice Structure for NFT permanent stats
    struct NFTStatsData {
        uint256 hp;
        uint256 attack;
        uint256 speed;
        uint256 level;
        uint256 currentXP;
        uint256 xpToNextLevel;
        uint256 dungeonRuns;
        uint256 successfulRuns;
        uint256 roomsCleared;
        bool initialized;
    }

    /// @notice Event emitted when an NFT's stats are initialized
    event StatsInitialized(address indexed collection, uint256 indexed tokenId, uint256 hp, uint256 attack, uint256 speed);
    
    /// @notice Event emitted when an NFT's stats are boosted
    event StatsBoosted(address indexed collection, uint256 indexed tokenId, uint256 newHp, uint256 newAttack, uint256 newSpeed);
    
    /// @notice Event emitted when XP is gained
    event XPGained(address indexed collection, uint256 indexed tokenId, uint256 xpGained, uint256 newTotalXP);
    
    /// @notice Event emitted when a level up occurs
    event LevelUp(address indexed collection, uint256 indexed tokenId, uint256 newLevel);
    
    /// @notice Event emitted when a run is recorded
    event RunRecorded(address indexed collection, uint256 indexed tokenId, bool success, uint256 roomsCleared, uint256 xpGained);

    /// @notice Initialize stats for an NFT based on its collection's base stats
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    function initializeStats(address collection, uint256 tokenId) external;

    /// @notice Award XP for dungeon progress
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @param xpAmount Amount of XP to award
    /// @param roomsCleared Number of rooms cleared in this run
    function awardXP(
        address collection,
        uint256 tokenId,
        uint256 xpAmount,
        uint256 roomsCleared
    ) external;

    /// @notice Calculate XP required for next level
    /// @param currentLevel Current level of the NFT
    /// @return uint256 XP required for next level
    function getXPForNextLevel(uint256 currentLevel) external pure returns (uint256);

    /// @notice Get the stat increases for a level up
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @return hpIncrease Amount HP increases
    /// @return attackIncrease Amount Attack increases
    /// @return speedIncrease Amount Speed increases
    function getLevelUpStats(
        address collection,
        uint256 tokenId
    ) external view returns (
        uint256 hpIncrease,
        uint256 attackIncrease,
        uint256 speedIncrease
    );

    /// @notice Record a dungeon run attempt
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @param success Whether the run was successful
    function recordRun(address collection, uint256 tokenId, bool success) external;

    /// @notice Get current stats for an NFT
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @return NFTStatsData struct containing current stats
    function getStats(address collection, uint256 tokenId) external view returns (NFTStatsData memory);

    /// @notice Check if an NFT has been initialized
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    /// @return bool True if NFT has been initialized
    function isInitialized(address collection, uint256 tokenId) external view returns (bool);
}

File 8 of 11 : IPrizePool.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

/// @title IPrizePool
/// @notice Interface for managing dungeon rewards and prize distribution
interface IPrizePool {
    /// @notice Event emitted when entry fee is deposited
    event EntryFeeDeposited(address indexed collection, uint256 indexed tokenId, uint256 amount);
    
    /// @notice Event emitted when reward is claimed
    event RewardClaimed(address indexed collection, uint256 indexed tokenId, address indexed recipient, uint256 amount);
    
    /// @notice Event emitted when prize pool parameters are updated
    event PrizePoolParametersUpdated(uint256 entryFee, uint256 winnerShare);

    /// @notice Deposit entry fee for a dungeon run
    function depositEntryFee() external payable;

    /// @notice Register a winner and immediately transfer their prize
    /// @param collection Address of the NFT collection
    /// @param tokenId Token ID of the NFT
    function registerWinner(address collection, uint256 tokenId) external;

    /// @notice Get current prize pool amount
    /// @return uint256 Current prize pool amount
    function getCurrentPrizePool() external view returns (uint256);

    /// @notice Gets the current entry fee
    /// @return uint256 Current entry fee
    function getEntryFee() external view returns (uint256);

    /// @notice Gets total entry fees collected
    /// @return uint256 Total entry fees
    function getTotalEntryFees() external view returns (uint256);

    /// @notice Gets total prizes paid out
    /// @return uint256 Total prizes paid
    function getTotalPrizesPaid() external view returns (uint256);

    /// @notice Gets current treasury balance
    /// @return uint256 Current treasury balance
    function getTreasuryBalance() external view returns (uint256);
}

File 9 of 11 : EncounterLibrary.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

import "../interfaces/IDungeonGame.sol";
import "../interfaces/INFTStats.sol";

/// @title EncounterLibrary
/// @notice Library for generating and processing dungeon encounters
library EncounterLibrary {
    // Encounter types
    enum EncounterType {
        Combat,      // Standard combat encounter
        Trap,       // Environmental hazard
        Blessing,   // Positive encounter
        Elite,      // Stronger combat encounter
        Boss        // Room 16 boss encounter
    }

    // Encounter definition
    struct Encounter {
        EncounterType encounterType;
        uint256 difficulty;       // 1-100 scale
        int256 baseHpChange;     // Base HP modification
        int256 baseAttackMod;    // Temporary attack modification
        int256 baseSpeedMod;     // Temporary speed modification
        uint256 baseXp;          // Base XP reward
        string description;       // Encounter description
    }

    // Constants for encounter generation
    uint256 private constant BASE_DIFFICULTY_PER_ROOM = 6; // ~100 difficulty by room 16
    uint256 private constant ELITE_CHANCE = 15;           // 15% chance for elite encounter
    uint256 private constant BLESSING_CHANCE = 10;        // 10% chance for blessing
    uint256 private constant TRAP_CHANCE = 20;           // 20% chance for trap
    
    // Constants for encounter effects
    uint256 private constant BASE_DAMAGE = 20;
    uint256 private constant ELITE_DAMAGE_MULTIPLIER = 2;
    uint256 private constant BOSS_DAMAGE_MULTIPLIER = 3;
    int256 private constant MAX_STAT_MODIFICATION = 50;   // Maximum temporary stat change

    /// @notice Generate an encounter for a specific room
    /// @param roomNumber Current room number (1-16)
    /// @param randomness Random number for encounter generation
    /// @return Encounter struct with encounter details
    function generateEncounter(
        uint256 roomNumber,
        uint256 randomness
    ) internal pure returns (Encounter memory) {
        require(roomNumber > 0 && roomNumber <= 16, "Invalid room number");

        // Room 16 is always a boss encounter
        if (roomNumber == 16) {
            return _generateBossEncounter();
        }

        // Use randomness to determine encounter type
        uint256 encounterRoll = randomness % 100;
        
        // Scale difficulty with room number
        uint256 difficulty = BASE_DIFFICULTY_PER_ROOM * roomNumber;
        
        // Select encounter type based on roll
        if (encounterRoll < BLESSING_CHANCE) {
            return _generateBlessing(roomNumber, difficulty);
        } else if (encounterRoll < BLESSING_CHANCE + TRAP_CHANCE) {
            return _generateTrap(roomNumber, difficulty);
        } else if (encounterRoll < BLESSING_CHANCE + TRAP_CHANCE + ELITE_CHANCE) {
            return _generateEliteEncounter(roomNumber, difficulty);
        } else {
            return _generateCombatEncounter(roomNumber, difficulty);
        }
    }

    /// @notice Process an encounter for a character
    /// @param encounter The encounter to process
    /// @param characterStats Current character stats
    /// @return IDungeonGame.EncounterResult Result of the encounter
    function processEncounter(
        Encounter memory encounter,
        INFTStats.NFTStatsData memory characterStats
    ) internal pure returns (IDungeonGame.EncounterResult memory) {
        // Calculate final HP change based on character stats
        int256 hpChange = _calculateHpChange(encounter, characterStats);

        
        return IDungeonGame.EncounterResult({
            hpChange: hpChange,
            encounterDescription: encounter.description
        });
    }

    // Internal encounter generation functions

    function _generateCombatEncounter(
        uint256 roomNumber,
        uint256 difficulty
    ) private pure returns (Encounter memory) {
        int256 damage = -int256(BASE_DAMAGE + (difficulty / 2));
        return Encounter({
            encounterType: EncounterType.Combat,
            difficulty: difficulty,
            baseHpChange: damage,
            baseAttackMod: 0,
            baseSpeedMod: 0,
            baseXp: 100 + (roomNumber * 10),
            description: "A hostile enemy appears!"
        });
    }

    function _generateEliteEncounter(
        uint256 roomNumber,
        uint256 difficulty
    ) private pure returns (Encounter memory) {
        int256 damage = -int256((BASE_DAMAGE + (difficulty / 2)) * ELITE_DAMAGE_MULTIPLIER);
        return Encounter({
            encounterType: EncounterType.Elite,
            difficulty: difficulty,
            baseHpChange: damage,
            baseAttackMod: int256(MAX_STAT_MODIFICATION / 2),
            baseSpeedMod: -int256(MAX_STAT_MODIFICATION / 4),
            baseXp: (150 + (roomNumber * 15)),
            description: "An elite enemy blocks your path!"
        });
    }

    function _generateBossEncounter() private pure returns (Encounter memory) {
        return Encounter({
            encounterType: EncounterType.Boss,
            difficulty: 100,
            baseHpChange: -int256(BASE_DAMAGE * BOSS_DAMAGE_MULTIPLIER),
            baseAttackMod: -int256(MAX_STAT_MODIFICATION),
            baseSpeedMod: -int256(MAX_STAT_MODIFICATION / 2),
            baseXp: 1000,
            description: "The dungeon boss emerges!"
        });
    }

    function _generateBlessing(
        uint256 roomNumber,
        uint256 difficulty
    ) private pure returns (Encounter memory) {
        return Encounter({
            encounterType: EncounterType.Blessing,
            difficulty: difficulty,
            baseHpChange: int256(BASE_DAMAGE),
            baseAttackMod: int256(MAX_STAT_MODIFICATION / 2),
            baseSpeedMod: int256(MAX_STAT_MODIFICATION / 2),
            baseXp: 50 + (roomNumber * 5),
            description: "You discover a magical blessing!"
        });
    }

    function _generateTrap(
        uint256 roomNumber,
        uint256 difficulty
    ) private pure returns (Encounter memory) {
        return Encounter({
            encounterType: EncounterType.Trap,
            difficulty: difficulty,
            baseHpChange: -int256(BASE_DAMAGE / 2),
            baseAttackMod: -int256(MAX_STAT_MODIFICATION / 4),
            baseSpeedMod: -int256(MAX_STAT_MODIFICATION / 4),
            baseXp: 75 + (roomNumber * 7),
            description: "You triggered a trap!"
        });
    }

    // Internal helper functions
    function _calculateHpChange(
        Encounter memory encounter,
        INFTStats.NFTStatsData memory characterStats
    ) private pure returns (int256) {
        // Base damage
        int256 hpChange = encounter.baseHpChange;

        // Modify based on character stats
        if (encounter.encounterType == EncounterType.Combat || 
            encounter.encounterType == EncounterType.Elite ||
            encounter.encounterType == EncounterType.Boss) {
            // Higher attack reduces damage taken
            uint256 attackMitigation = characterStats.attack / 10;
            // Higher speed increases chance to dodge
            uint256 speedMitigation = characterStats.speed / 20;
            
            hpChange += int256(attackMitigation + speedMitigation);
        }
        
        return hpChange;
    }
}

File 10 of 11 : Context.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol)

pragma solidity ^0.8.20;

/**
 * @dev Provides information about the current execution context, including the
 * sender of the transaction and its data. While these are generally available
 * via msg.sender and msg.data, they should not be accessed in such a direct
 * manner, since when dealing with meta-transactions the account sending and
 * paying for execution may not be the actual sender (as far as an application
 * is concerned).
 *
 * This contract is only required for intermediate, library-like contracts.
 */
abstract contract Context {
    function _msgSender() internal view virtual returns (address) {
        return msg.sender;
    }

    function _msgData() internal view virtual returns (bytes calldata) {
        return msg.data;
    }

    function _contextSuffixLength() internal view virtual returns (uint256) {
        return 0;
    }
}

File 11 of 11 : IERC165.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v5.1.0) (utils/introspection/IERC165.sol)

pragma solidity ^0.8.20;

/**
 * @dev Interface of the ERC-165 standard, as defined in the
 * https://eips.ethereum.org/EIPS/eip-165[ERC].
 *
 * 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[ERC 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);
}

Settings
{
  "remappings": [
    "@openzeppelin/contracts/=lib/openzeppelin-contracts/contracts/",
    "@pythnetwork/entropy-sdk-solidity/=../node_modules/@pythnetwork/entropy-sdk-solidity/",
    "ds-test/=lib/forge-std/lib/ds-test/src/",
    "forge-std/=lib/forge-std/src/",
    "erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/",
    "halmos-cheatcodes/=lib/openzeppelin-contracts/lib/halmos-cheatcodes/src/",
    "openzeppelin-contracts/=lib/openzeppelin-contracts/",
    "pyth-sdk-solidity/=lib/pyth-sdk-solidity/"
  ],
  "optimizer": {
    "enabled": true,
    "runs": 200
  },
  "metadata": {
    "useLiteralContent": false,
    "bytecodeHash": "ipfs",
    "appendCBOR": true
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "devdoc",
        "userdoc",
        "metadata",
        "abi"
      ]
    }
  },
  "evmVersion": "shanghai",
  "libraries": {}
}

Contract Security Audit

Contract ABI

API
[{"inputs":[{"internalType":"address","name":"_dungeonEntry","type":"address"},{"internalType":"address","name":"_nftStats","type":"address"},{"internalType":"address","name":"_prizePool","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"inputs":[],"name":"ReentrancyGuardReentrantCall","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"collection","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newHp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"xpGained","type":"uint256"}],"name":"CharacterSurvived","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"collection","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"DungeonCompleted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"collection","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"entryIndex","type":"uint256"}],"name":"DungeonEntered","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"collection","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"roomNumber","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"xpGained","type":"uint256"},{"indexed":false,"internalType":"bool","name":"survived","type":"bool"},{"indexed":false,"internalType":"string","name":"encounterDescription","type":"string"}],"name":"EncounterCompleted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"FeesWithdrawn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[],"name":"ROOM_COUNT","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"dungeonEntry","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"enterDungeon","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getCharacterRoomNumber","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getCharacterState","outputs":[{"components":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"currentHp","type":"uint256"}],"internalType":"struct IDungeonGame.CharacterState","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint64","name":"sequenceNumber","type":"uint64"}],"name":"getPendingEntry","outputs":[{"components":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"currentHp","type":"uint256"}],"internalType":"struct IDungeonGame.CharacterState","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"roomNumber","type":"uint256"}],"name":"getRoomState","outputs":[{"components":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"currentHp","type":"uint256"}],"internalType":"struct IDungeonGame.CharacterState","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"nftStats","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"prizePool","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"rooms","outputs":[{"internalType":"address","name":"collection","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"currentHp","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_dungeonEntry","type":"address"}],"name":"setDungeonEntry","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"startRoom","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]

6101206040525f60c081905260e0819052610100819052600280546001600160a01b0319169055600381905560048190556006556007805460ff1916905534801562000049575f80fd5b5060405162001b7a38038062001b7a8339810160408190526200006c9162000242565b33806200009357604051631e4fbdf760e01b81525f60048201526024015b60405180910390fd5b6200009e81620001d7565b50600180556001600160a01b038316620000fb5760405162461bcd60e51b815260206004820152601c60248201527f496e76616c69642044756e67656f6e456e74727920616464726573730000000060448201526064016200008a565b6001600160a01b038216620001535760405162461bcd60e51b815260206004820152601860248201527f496e76616c6964204e465453746174732061646472657373000000000000000060448201526064016200008a565b6001600160a01b038116620001ab5760405162461bcd60e51b815260206004820152601960248201527f496e76616c6964205072697a65506f6f6c20616464726573730000000000000060448201526064016200008a565b600580546001600160a01b0319166001600160a01b039485161790559082166080521660a05262000289565b5f80546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b80516001600160a01b03811681146200023d575f80fd5b919050565b5f805f6060848603121562000255575f80fd5b620002608462000226565b9250620002706020850162000226565b9150620002806040850162000226565b90509250925092565b60805160a0516118ac620002ce5f395f8181610276015261104101525f818161033a0152818161095f01528181610aaf01528181610c720152610f6b01526118ac5ff3fe6080604052600436106100fd575f3560e01c8063715018a6116100925780639d157ea0116100625780639d157ea0146102eb578063b996e27d1461030a578063e602ef3614610329578063f0477d791461035c578063f2fde38b1461037b575f80fd5b8063715018a614610251578063719ce73e146102655780637d304847146102b05780638da5cb5b146102cf575f80fd5b80633c42dd9c116100cd5780633c42dd9c146101ef5780633dd5911114610203578063476343ee14610222578063607e9d7714610238575f80fd5b806305e72e42146101085780631911ce781461013e5780631bae0ac81461018857806326987b60146101cc575f80fd5b3661010457005b5f80fd5b348015610113575f80fd5b50610127610122366004611589565b61039a565b60405160ff90911681526020015b60405180910390f35b348015610149575f80fd5b5061015d6101583660046115b1565b6103f8565b6040805182516001600160a01b03168152602080840151908201529181015190820152606001610135565b348015610193575f80fd5b506101a76101a23660046115d8565b61044b565b604080516001600160a01b039094168452602084019290925290820152606001610135565b3480156101d7575f80fd5b506101e160065481565b604051908152602001610135565b3480156101fa575f80fd5b50610127601081565b34801561020e575f80fd5b5061015d61021d3660046115d8565b61047b565b34801561022d575f80fd5b5061023661052b565b005b348015610243575f80fd5b506007546101279060ff1681565b34801561025c575f80fd5b50610236610639565b348015610270575f80fd5b506102987f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b039091168152602001610135565b3480156102bb575f80fd5b506102366102ca3660046115ef565b61064c565b3480156102da575f80fd5b505f546001600160a01b0316610298565b3480156102f6575f80fd5b50610236610305366004611589565b6106cc565b348015610315575f80fd5b50600554610298906001600160a01b031681565b348015610334575f80fd5b506102987f000000000000000000000000000000000000000000000000000000000000000081565b348015610367575f80fd5b5061015d610376366004611589565b610777565b348015610386575f80fd5b506102366103953660046115ef565b6107e0565b6001600160a01b0382165f908152603960209081526040808320848452909152812054600754600654839160109160ff909116906103d990859061161c565b6103e3919061162f565b6103ed9190611656565b925050505b92915050565b61040061150f565b5067ffffffffffffffff165f90815260386020908152604091829020825160608101845281546001600160a01b03168152600182015492810192909252600201549181019190915290565b6008816010811061045a575f80fd5b60030201805460018201546002909201546001600160a01b03909116925083565b61048361150f565b61048f60016010611669565b60ff168211156104dc5760405162461bcd60e51b815260206004820152601360248201527224b73b30b634b2103937b7b690373ab6b132b960691b60448201526064015b60405180910390fd5b600882601081106104ef576104ef611682565b604080516060810182526003929092029290920180546001600160a01b0316825260018101546020830152600201549181019190915292915050565b61053361081d565b47806105775760405162461bcd60e51b81526020600482015260136024820152724e6f206665657320746f20776974686472617760681b60448201526064016104d3565b6040515f90339083908381818185875af1925050503d805f81146105b6576040519150601f19603f3d011682016040523d82523d5f602084013e6105bb565b606091505b50509050806106005760405162461bcd60e51b815260206004820152601160248201527015da5d1a191c985dd85b0819985a5b1959607a1b60448201526064016104d3565b60405182815233907fc0819c13be868895eb93e40eaceb96de976442fa1d404e5c55f14bb65a8c489a9060200160405180910390a25050565b61064161081d565b61064a5f610849565b565b61065461081d565b6001600160a01b0381166106aa5760405162461bcd60e51b815260206004820152601d60248201527f496e76616c69642064756e67656f6e20656e747279206164647265737300000060448201526064016104d3565b600580546001600160a01b0319166001600160a01b0392909216919091179055565b6005546001600160a01b031633146107325760405162461bcd60e51b8152602060048201526024808201527f4f6e6c792044756e67656f6e456e7472792063616e20616464206368617261636044820152637465727360e01b60648201526084016104d3565b5f424460405160200161074f929190918252602082015260400190565b604051602081830303815290604052805190602001209050610772838383610898565b505050565b61077f61150f565b5f61078a848461039a565b905060088160ff16601081106107a2576107a2611682565b604080516060810182526003929092029290920180546001600160a01b03168252600181015460208301526002015491810191909152949350505050565b6107e861081d565b6001600160a01b03811661081157604051631e4fbdf760e01b81525f60048201526024016104d3565b61081a81610849565b50565b5f546001600160a01b0316331461064a5760405163118cdaa760e01b81523360048201526024016104d3565b5f80546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b5f5b601060ff82161015610c49576007545f9060109083906001906108c09060ff1684611696565b6108ca9190611669565b6108d49190611669565b6108de91906116af565b90505f60088260ff16601081106108f7576108f7611682565b6003020180549091506001600160a01b0316610914575050610c37565b5f61092c610923846001611696565b60ff1686610df6565b82546001840154604051630368516960e01b81526001600160a01b03928316600482015260248101919091529192505f917f00000000000000000000000000000000000000000000000000000000000000009091169063036851699060440161014060405180830381865afa1580156109a7573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906109cb9190611715565b90505f6109d88383610ef7565b90505f815f015185600201546109ee919061179d565b90505f8082139081610a105760028660a00151610a0b91906117bc565b610a16565b8560a001515b90508115610b09576001870154875460408051868152602081018590526001600160a01b03909216917f6dfed4d35f3671480d53c1b0f0cea670d9b92b20280e2cd62e65f9c0a0de5c4d910160405180910390a360028701839055865460018089015460405163012a988f60e31b81526001600160a01b03938416600482015260248101919091526044810184905260648101919091527f000000000000000000000000000000000000000000000000000000000000000090911690630954c478906084015f604051808303815f87803b158015610af2575f80fd5b505af1158015610b04573d5f803e3d5ffd5b505050505b8115610b4657610b1b60016010611669565b60ff168860ff1603610b415786546001880154610b41916001600160a01b031690610f36565b610c2e565b6005548754600189015460405163ceef46bd60e01b81526001600160a01b03928316600482015260248101919091525f604482015291169063ceef46bd906064015f604051808303815f87803b158015610b9e575f80fd5b505af1158015610bb0573d5f803e3d5ffd5b5050505060405180606001604052805f6001600160a01b031681526020015f81526020015f81525060088960ff1660108110610bee57610bee611682565b82516003919091029190910180546001600160a01b0319166001600160a01b03909216919091178155602082015160018201556040909101516002909101555b50505050505050505b80610c41816117cf565b91505061089a565b50604051630368516960e01b81526001600160a01b038481166004830152602482018490525f917f00000000000000000000000000000000000000000000000000000000000000009091169063036851699060440161014060405180830381865afa158015610cba573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610cde9190611715565b6006546001600160a01b0386165f8181526039602090815260408083208984528252918290209390935580516060810182529182529181018690528251918101919091526007549192509060089060ff1660108110610d3f57610d3f611682565b82516003919091029190910180546001600160a01b0319166001600160a01b0392831617815560208301516001820155604092830151600290910155600654915185918716907faef57a2e740e7aa2c0ef80e2d976766d8b4b1cc896fae50d9a4c15cc5f0b9c56905f90a4600754601090610dbe9060ff166001611696565b610dc891906116af565b6007805460ff191660ff9290921691909117905560068054905f610deb836117ed565b919050555050505050565b610dfe611536565b5f83118015610e0e575060108311155b610e505760405162461bcd60e51b815260206004820152601360248201527224b73b30b634b2103937b7b690373ab6b132b960691b60448201526064016104d3565b82601003610e6757610e606110d3565b90506103f2565b5f610e73606484611656565b90505f610e81856006611805565b9050600a821015610e9f57610e96858261117c565b925050506103f2565b610eab6014600a61162f565b821015610ebc57610e96858261121e565b600f610eca6014600a61162f565b610ed4919061162f565b821015610ee557610e9685826112e0565b610e9685826113b8565b505092915050565b604080518082019091525f8152606060208201525f610f16848461146c565b6040805180820190915290815260c0850151602082015291505092915050565b60405163012a988f60e31b81526001600160a01b038381166004830152602482018390526101f46044830152601060648301527f00000000000000000000000000000000000000000000000000000000000000001690630954c478906084015f604051808303815f87803b158015610fac575f80fd5b505af1158015610fbe573d5f803e3d5ffd5b505050506002600860016010610fd49190611669565b60ff1660108110610fe757610fe7611682565b82546003919091029190910180546001600160a01b0319166001600160a01b039283161781556001808401549082015560029283015492019190915560405163f79edfcd60e01b81528382166004820152602481018390527f00000000000000000000000000000000000000000000000000000000000000009091169063f79edfcd906044015f604051808303815f87803b158015611084575f80fd5b505af1158015611096573d5f803e3d5ffd5b50506040518392506001600160a01b03851691507f83c7c1485e35802f043337a3bbc33306ed0d06395209bcfa732d95eeab9776bb905f90a35050565b6110db611536565b6040805160e08101825260048152606460208201529081016110ff60036014611805565b61110890611830565b81526020016111176032611830565b81526020016111286002603261184a565b61113190611830565b81526020016103e881526020016040518060400160405280601981526020017f5468652064756e67656f6e20626f737320656d65726765732100000000000000815250815250905090565b611184611536565b6040805160e08101909152806002815260208101849052601460408201526060016111b16002603261184a565b81526020016111c26002603261184a565b81526020016111d2856005611805565b6111dd90603261162f565b81526040805180820190915260208082527f596f7520646973636f7665722061206d61676963616c20626c657373696e672182820152909101529392505050565b611226611536565b6040805160e081018252600181526020810184905290810161124a600260146117bc565b61125390611830565b81526020016112646004603261184a565b61126d90611830565b815260200161127e6004603261184a565b61128790611830565b8152602001611297856007611805565b6112a290604b61162f565b815260200160405180604001604052806015815260200174596f7520747269676765726564206120747261702160581b815250815250905092915050565b6112e8611536565b5f60026112f581856117bc565b61130090601461162f565b61130a9190611805565b61131390611830565b6040805160e0810182526003815260208101869052908101829052909150606081016113416002603261184a565b81526020016113526004603261184a565b61135b90611830565b815260200161136b86600f611805565b61137690609661162f565b81526040805180820190915260208082527f416e20656c69746520656e656d7920626c6f636b7320796f75722070617468218282015290910152949350505050565b6113c0611536565b5f6113cc6002846117bc565b6113d790601461162f565b6113e090611830565b6040805160e08101909152909150805f81526020018481526020018281526020015f81526020015f815260200185600a61141a9190611805565b61142590606461162f565b81526020016040518060400160405280601881526020017f4120686f7374696c6520656e656d79206170706561727321000000000000000081525081525091505092915050565b60408201515f9081845160048111156114875761148761181c565b14806114a557506003845160048111156114a3576114a361181c565b145b806114c257506004845160048111156114c0576114c061181c565b145b15611508575f600a84602001516114d991906117bc565b90505f601485604001516114ed91906117bc565b90506114f9818361162f565b611503908461179d565b925050505b9392505050565b60405180606001604052805f6001600160a01b031681526020015f81526020015f81525090565b6040805160e08101909152805f81526020015f81526020015f81526020015f81526020015f81526020015f8152602001606081525090565b80356001600160a01b0381168114611584575f80fd5b919050565b5f806040838503121561159a575f80fd5b6115a38361156e565b946020939093013593505050565b5f602082840312156115c1575f80fd5b813567ffffffffffffffff81168114611508575f80fd5b5f602082840312156115e8575f80fd5b5035919050565b5f602082840312156115ff575f80fd5b6115088261156e565b634e487b7160e01b5f52601160045260245ffd5b818103818111156103f2576103f2611608565b808201808211156103f2576103f2611608565b634e487b7160e01b5f52601260045260245ffd5b5f8261166457611664611642565b500690565b60ff82811682821603908111156103f2576103f2611608565b634e487b7160e01b5f52603260045260245ffd5b60ff81811683821601908111156103f2576103f2611608565b5f60ff8316806116c1576116c1611642565b8060ff84160691505092915050565b604051610140810167ffffffffffffffff8111828210171561170057634e487b7160e01b5f52604160045260245ffd5b60405290565b80518015158114611584575f80fd5b5f6101408284031215611726575f80fd5b61172e6116d0565b825181526020830151602082015260408301516040820152606083015160608201526080830151608082015260a083015160a082015260c083015160c082015260e083015160e0820152610100808401518183015250610120611792818501611706565b908201529392505050565b8082018281125f831280158216821582161715610eef57610eef611608565b5f826117ca576117ca611642565b500490565b5f60ff821660ff81036117e4576117e4611608565b60010192915050565b5f600182016117fe576117fe611608565b5060010190565b80820281158282048414176103f2576103f2611608565b634e487b7160e01b5f52602160045260245ffd5b5f600160ff1b820161184457611844611608565b505f0390565b5f8261185857611858611642565b600160ff1b82145f198414161561187157611871611608565b50059056fea2646970667358221220866272219670dc4ec7011a6da0faff530e60bb004fdecc64b06cecb777c6eba064736f6c63430008140033000000000000000000000000d1cd2d2539a23ec350c61dd534129c5a75016fb4000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc00000000000000000000000088bbfab62d7ecbfa016fd20e7d337b3bf9452760

Deployed Bytecode

0x6080604052600436106100fd575f3560e01c8063715018a6116100925780639d157ea0116100625780639d157ea0146102eb578063b996e27d1461030a578063e602ef3614610329578063f0477d791461035c578063f2fde38b1461037b575f80fd5b8063715018a614610251578063719ce73e146102655780637d304847146102b05780638da5cb5b146102cf575f80fd5b80633c42dd9c116100cd5780633c42dd9c146101ef5780633dd5911114610203578063476343ee14610222578063607e9d7714610238575f80fd5b806305e72e42146101085780631911ce781461013e5780631bae0ac81461018857806326987b60146101cc575f80fd5b3661010457005b5f80fd5b348015610113575f80fd5b50610127610122366004611589565b61039a565b60405160ff90911681526020015b60405180910390f35b348015610149575f80fd5b5061015d6101583660046115b1565b6103f8565b6040805182516001600160a01b03168152602080840151908201529181015190820152606001610135565b348015610193575f80fd5b506101a76101a23660046115d8565b61044b565b604080516001600160a01b039094168452602084019290925290820152606001610135565b3480156101d7575f80fd5b506101e160065481565b604051908152602001610135565b3480156101fa575f80fd5b50610127601081565b34801561020e575f80fd5b5061015d61021d3660046115d8565b61047b565b34801561022d575f80fd5b5061023661052b565b005b348015610243575f80fd5b506007546101279060ff1681565b34801561025c575f80fd5b50610236610639565b348015610270575f80fd5b506102987f00000000000000000000000088bbfab62d7ecbfa016fd20e7d337b3bf945276081565b6040516001600160a01b039091168152602001610135565b3480156102bb575f80fd5b506102366102ca3660046115ef565b61064c565b3480156102da575f80fd5b505f546001600160a01b0316610298565b3480156102f6575f80fd5b50610236610305366004611589565b6106cc565b348015610315575f80fd5b50600554610298906001600160a01b031681565b348015610334575f80fd5b506102987f000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc81565b348015610367575f80fd5b5061015d610376366004611589565b610777565b348015610386575f80fd5b506102366103953660046115ef565b6107e0565b6001600160a01b0382165f908152603960209081526040808320848452909152812054600754600654839160109160ff909116906103d990859061161c565b6103e3919061162f565b6103ed9190611656565b925050505b92915050565b61040061150f565b5067ffffffffffffffff165f90815260386020908152604091829020825160608101845281546001600160a01b03168152600182015492810192909252600201549181019190915290565b6008816010811061045a575f80fd5b60030201805460018201546002909201546001600160a01b03909116925083565b61048361150f565b61048f60016010611669565b60ff168211156104dc5760405162461bcd60e51b815260206004820152601360248201527224b73b30b634b2103937b7b690373ab6b132b960691b60448201526064015b60405180910390fd5b600882601081106104ef576104ef611682565b604080516060810182526003929092029290920180546001600160a01b0316825260018101546020830152600201549181019190915292915050565b61053361081d565b47806105775760405162461bcd60e51b81526020600482015260136024820152724e6f206665657320746f20776974686472617760681b60448201526064016104d3565b6040515f90339083908381818185875af1925050503d805f81146105b6576040519150601f19603f3d011682016040523d82523d5f602084013e6105bb565b606091505b50509050806106005760405162461bcd60e51b815260206004820152601160248201527015da5d1a191c985dd85b0819985a5b1959607a1b60448201526064016104d3565b60405182815233907fc0819c13be868895eb93e40eaceb96de976442fa1d404e5c55f14bb65a8c489a9060200160405180910390a25050565b61064161081d565b61064a5f610849565b565b61065461081d565b6001600160a01b0381166106aa5760405162461bcd60e51b815260206004820152601d60248201527f496e76616c69642064756e67656f6e20656e747279206164647265737300000060448201526064016104d3565b600580546001600160a01b0319166001600160a01b0392909216919091179055565b6005546001600160a01b031633146107325760405162461bcd60e51b8152602060048201526024808201527f4f6e6c792044756e67656f6e456e7472792063616e20616464206368617261636044820152637465727360e01b60648201526084016104d3565b5f424460405160200161074f929190918252602082015260400190565b604051602081830303815290604052805190602001209050610772838383610898565b505050565b61077f61150f565b5f61078a848461039a565b905060088160ff16601081106107a2576107a2611682565b604080516060810182526003929092029290920180546001600160a01b03168252600181015460208301526002015491810191909152949350505050565b6107e861081d565b6001600160a01b03811661081157604051631e4fbdf760e01b81525f60048201526024016104d3565b61081a81610849565b50565b5f546001600160a01b0316331461064a5760405163118cdaa760e01b81523360048201526024016104d3565b5f80546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b5f5b601060ff82161015610c49576007545f9060109083906001906108c09060ff1684611696565b6108ca9190611669565b6108d49190611669565b6108de91906116af565b90505f60088260ff16601081106108f7576108f7611682565b6003020180549091506001600160a01b0316610914575050610c37565b5f61092c610923846001611696565b60ff1686610df6565b82546001840154604051630368516960e01b81526001600160a01b03928316600482015260248101919091529192505f917f000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc9091169063036851699060440161014060405180830381865afa1580156109a7573d5f803e3d5ffd5b505050506040513d601f19601f820116820180604052508101906109cb9190611715565b90505f6109d88383610ef7565b90505f815f015185600201546109ee919061179d565b90505f8082139081610a105760028660a00151610a0b91906117bc565b610a16565b8560a001515b90508115610b09576001870154875460408051868152602081018590526001600160a01b03909216917f6dfed4d35f3671480d53c1b0f0cea670d9b92b20280e2cd62e65f9c0a0de5c4d910160405180910390a360028701839055865460018089015460405163012a988f60e31b81526001600160a01b03938416600482015260248101919091526044810184905260648101919091527f000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc90911690630954c478906084015f604051808303815f87803b158015610af2575f80fd5b505af1158015610b04573d5f803e3d5ffd5b505050505b8115610b4657610b1b60016010611669565b60ff168860ff1603610b415786546001880154610b41916001600160a01b031690610f36565b610c2e565b6005548754600189015460405163ceef46bd60e01b81526001600160a01b03928316600482015260248101919091525f604482015291169063ceef46bd906064015f604051808303815f87803b158015610b9e575f80fd5b505af1158015610bb0573d5f803e3d5ffd5b5050505060405180606001604052805f6001600160a01b031681526020015f81526020015f81525060088960ff1660108110610bee57610bee611682565b82516003919091029190910180546001600160a01b0319166001600160a01b03909216919091178155602082015160018201556040909101516002909101555b50505050505050505b80610c41816117cf565b91505061089a565b50604051630368516960e01b81526001600160a01b038481166004830152602482018490525f917f000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc9091169063036851699060440161014060405180830381865afa158015610cba573d5f803e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610cde9190611715565b6006546001600160a01b0386165f8181526039602090815260408083208984528252918290209390935580516060810182529182529181018690528251918101919091526007549192509060089060ff1660108110610d3f57610d3f611682565b82516003919091029190910180546001600160a01b0319166001600160a01b0392831617815560208301516001820155604092830151600290910155600654915185918716907faef57a2e740e7aa2c0ef80e2d976766d8b4b1cc896fae50d9a4c15cc5f0b9c56905f90a4600754601090610dbe9060ff166001611696565b610dc891906116af565b6007805460ff191660ff9290921691909117905560068054905f610deb836117ed565b919050555050505050565b610dfe611536565b5f83118015610e0e575060108311155b610e505760405162461bcd60e51b815260206004820152601360248201527224b73b30b634b2103937b7b690373ab6b132b960691b60448201526064016104d3565b82601003610e6757610e606110d3565b90506103f2565b5f610e73606484611656565b90505f610e81856006611805565b9050600a821015610e9f57610e96858261117c565b925050506103f2565b610eab6014600a61162f565b821015610ebc57610e96858261121e565b600f610eca6014600a61162f565b610ed4919061162f565b821015610ee557610e9685826112e0565b610e9685826113b8565b505092915050565b604080518082019091525f8152606060208201525f610f16848461146c565b6040805180820190915290815260c0850151602082015291505092915050565b60405163012a988f60e31b81526001600160a01b038381166004830152602482018390526101f46044830152601060648301527f000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc1690630954c478906084015f604051808303815f87803b158015610fac575f80fd5b505af1158015610fbe573d5f803e3d5ffd5b505050506002600860016010610fd49190611669565b60ff1660108110610fe757610fe7611682565b82546003919091029190910180546001600160a01b0319166001600160a01b039283161781556001808401549082015560029283015492019190915560405163f79edfcd60e01b81528382166004820152602481018390527f00000000000000000000000088bbfab62d7ecbfa016fd20e7d337b3bf94527609091169063f79edfcd906044015f604051808303815f87803b158015611084575f80fd5b505af1158015611096573d5f803e3d5ffd5b50506040518392506001600160a01b03851691507f83c7c1485e35802f043337a3bbc33306ed0d06395209bcfa732d95eeab9776bb905f90a35050565b6110db611536565b6040805160e08101825260048152606460208201529081016110ff60036014611805565b61110890611830565b81526020016111176032611830565b81526020016111286002603261184a565b61113190611830565b81526020016103e881526020016040518060400160405280601981526020017f5468652064756e67656f6e20626f737320656d65726765732100000000000000815250815250905090565b611184611536565b6040805160e08101909152806002815260208101849052601460408201526060016111b16002603261184a565b81526020016111c26002603261184a565b81526020016111d2856005611805565b6111dd90603261162f565b81526040805180820190915260208082527f596f7520646973636f7665722061206d61676963616c20626c657373696e672182820152909101529392505050565b611226611536565b6040805160e081018252600181526020810184905290810161124a600260146117bc565b61125390611830565b81526020016112646004603261184a565b61126d90611830565b815260200161127e6004603261184a565b61128790611830565b8152602001611297856007611805565b6112a290604b61162f565b815260200160405180604001604052806015815260200174596f7520747269676765726564206120747261702160581b815250815250905092915050565b6112e8611536565b5f60026112f581856117bc565b61130090601461162f565b61130a9190611805565b61131390611830565b6040805160e0810182526003815260208101869052908101829052909150606081016113416002603261184a565b81526020016113526004603261184a565b61135b90611830565b815260200161136b86600f611805565b61137690609661162f565b81526040805180820190915260208082527f416e20656c69746520656e656d7920626c6f636b7320796f75722070617468218282015290910152949350505050565b6113c0611536565b5f6113cc6002846117bc565b6113d790601461162f565b6113e090611830565b6040805160e08101909152909150805f81526020018481526020018281526020015f81526020015f815260200185600a61141a9190611805565b61142590606461162f565b81526020016040518060400160405280601881526020017f4120686f7374696c6520656e656d79206170706561727321000000000000000081525081525091505092915050565b60408201515f9081845160048111156114875761148761181c565b14806114a557506003845160048111156114a3576114a361181c565b145b806114c257506004845160048111156114c0576114c061181c565b145b15611508575f600a84602001516114d991906117bc565b90505f601485604001516114ed91906117bc565b90506114f9818361162f565b611503908461179d565b925050505b9392505050565b60405180606001604052805f6001600160a01b031681526020015f81526020015f81525090565b6040805160e08101909152805f81526020015f81526020015f81526020015f81526020015f81526020015f8152602001606081525090565b80356001600160a01b0381168114611584575f80fd5b919050565b5f806040838503121561159a575f80fd5b6115a38361156e565b946020939093013593505050565b5f602082840312156115c1575f80fd5b813567ffffffffffffffff81168114611508575f80fd5b5f602082840312156115e8575f80fd5b5035919050565b5f602082840312156115ff575f80fd5b6115088261156e565b634e487b7160e01b5f52601160045260245ffd5b818103818111156103f2576103f2611608565b808201808211156103f2576103f2611608565b634e487b7160e01b5f52601260045260245ffd5b5f8261166457611664611642565b500690565b60ff82811682821603908111156103f2576103f2611608565b634e487b7160e01b5f52603260045260245ffd5b60ff81811683821601908111156103f2576103f2611608565b5f60ff8316806116c1576116c1611642565b8060ff84160691505092915050565b604051610140810167ffffffffffffffff8111828210171561170057634e487b7160e01b5f52604160045260245ffd5b60405290565b80518015158114611584575f80fd5b5f6101408284031215611726575f80fd5b61172e6116d0565b825181526020830151602082015260408301516040820152606083015160608201526080830151608082015260a083015160a082015260c083015160c082015260e083015160e0820152610100808401518183015250610120611792818501611706565b908201529392505050565b8082018281125f831280158216821582161715610eef57610eef611608565b5f826117ca576117ca611642565b500490565b5f60ff821660ff81036117e4576117e4611608565b60010192915050565b5f600182016117fe576117fe611608565b5060010190565b80820281158282048414176103f2576103f2611608565b634e487b7160e01b5f52602160045260245ffd5b5f600160ff1b820161184457611844611608565b505f0390565b5f8261185857611858611642565b600160ff1b82145f198414161561187157611871611608565b50059056fea2646970667358221220866272219670dc4ec7011a6da0faff530e60bb004fdecc64b06cecb777c6eba064736f6c63430008140033

Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)

000000000000000000000000d1cd2d2539a23ec350c61dd534129c5a75016fb4000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc00000000000000000000000088bbfab62d7ecbfa016fd20e7d337b3bf9452760

-----Decoded View---------------
Arg [0] : _dungeonEntry (address): 0xD1cD2D2539a23ec350c61dD534129C5a75016fb4
Arg [1] : _nftStats (address): 0x532279f09c84F669cA96acB6a6A78C7c81420fbc
Arg [2] : _prizePool (address): 0x88bBfAB62D7EcbFa016fD20e7D337b3bf9452760

-----Encoded View---------------
3 Constructor Arguments found :
Arg [0] : 000000000000000000000000d1cd2d2539a23ec350c61dd534129c5a75016fb4
Arg [1] : 000000000000000000000000532279f09c84f669ca96acb6a6a78c7c81420fbc
Arg [2] : 00000000000000000000000088bbfab62d7ecbfa016fd20e7d337b3bf9452760


Block Transaction Difficulty Gas Used Reward
View All Blocks Produced

Block Uncle Number Difficulty Gas Used Reward
View All Uncles
Loading...
Loading
Loading...
Loading

Validator Index Block Amount
View All Withdrawals

Transaction Hash Block Value Eth2 PubKey Valid
View All Deposits

A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.