Blackjack
The following documentation goes through an example implementation of the popular casino game Blackjack on Sui. This guide walks through its components, providing a detailed look at the module's functions, structures, constants, and their significance in the overall gameplay mechanism.
A deployed version of the Blackjack game is online at Mysten Blackjack.
Building an on-chain blackjack game shares a lot of similarities with a Coin Flip game. This example covers the smart contracts (Move modules), backend logic (using serverless functions), and frontend logic.
- Prerequisites
- 
Set up your Sui account and CLI environment. 
Create Sui account and setup CLI environment
$ sui client
If this is the first time running the sui client CLI tool, it asks you to provide a Sui full node server URL and a meaningful environment alias. It also generates an address with a random key pair in sui.keystore and a config client.yaml.
By default, the client.yaml and sui.keystore files are located in ~/.sui/sui_config. For more information, refer to the Sui client CLI tutorial.
If this is not your first time running sui client, then you already have a client.yaml file in your local environment. If you'd like to create a new address for this tutorial, use the command:
$ sui client new-address ed25519
- Obtain test tokens.
How to obtain tokens
If you are connected to Devnet or Testnet networks, use the Faucet UI to request tokens.
If you are connected to a local full node, learn how to get local network tokens.
For more details on building a backend and deploying to Sui, check out the Coin Flip app example.
You can also find the full repository for this example here.
Gameplay
In this single-player version of blackjack, the player competes against a dealer, which is automated by the system. The dealer is equipped with a public BLS key that plays a central role in the game's mechanics. The dealer's actions are triggered by HTTP requests to serverless functions. Players place their bet to start the game. Upon initiating the game, a request is made to the backend (dealer), which processes it by signing and subsequently dealing two cards to the player and one to themselves.
The player has the option to 'Hit' or 'Stand.' Selecting 'Stand' triggers the dealer to draw cards until the total reaches 17 or higher. After the dealer stops, the smart contract steps in to compare the totals and declare the winner. On the other hand, choosing 'Hit' prompts the dealer to draw an additional card for the player.
Note that more complex Blackjack rules, such as splitting, are considered out of scope for this example, and are therefore not implemented.
Smart contracts
single_player_blackjack module
The single_player_blackjack.move module includes several constants that define game statuses and help track the game's progress:
- IN_PROGRESS
- PLAYER_WON_STATUS
- HOUSE_WON_STATUS
- TIE_STATUS
There are also constants for error handling, such as EInvalidBlsSig, EInsufficientBalance, and others, ensuring robust game mechanics.
Structs like GameCreatedEvent, GameOutcomeEvent, and HitDoneEvent capture the various events and actions within a game. The HitRequest and StandRequest structs ensure that a move (hit/stand) can be performed by the house only if the player has already requested it. HouseAdminCap and HouseData are crucial for maintaining the house's data, including balance and public key, while the Game struct contains all the necessary information about each game, such as player data, cards, and the current status.
The module's functions can be broadly categorized into initialization, game management, and utility functions. The init function sets up the house admin capability, while initialize_house_data prepares the house for the game by setting up the balance and public key. place_bet_and_create_game is the entry point for players to start a new game, involving a bet and random input. The functions first_deal, hit, and stand govern the core gameplay, handling the dealing of cards and player choices.
Utility functions like get_next_random_card and get_card_sum are essential for the game's mechanics, generating random cards and calculating hand values. The module also includes accessors for retrieving various pieces of game and house data.
For testing purposes, the module provides special functions like get_house_admin_cap_for_testing, player_won_post_handling_for_test and house_won_post_handling_for_test, ensuring that developers can thoroughly test the game mechanics and house data handling.
Backend
The backend is used for all the transactions that are executed by the dealer. The backend can be completely stateless, and for that reason serverless functions are utilized. As a result, the corresponding code lies under the app/src/app/api/ directory.
Directories structure
The backend code is split in the following sub directories:
- games/: The main code of the backend endpoints. Each file that is named- route.tsis served as an endpoint, in the path defined by the project structure (see Route Segments on NextJS App Router for details).
- health/: A simple health check endpoint to check the API availability.
- helpers/: Multiple helper functions used in various endpoints.
- services/: The core logic of the backend, that signs and executes the transactions on the Sui blockchain.
- utils/: Reusable methods for signing the transactions as the dealer, and sponsoring them with Shinami, to avoid gas coins equivocation.
High-Level endpoints specification
| HTTP Method | Path | Description | Request Body | 
|---|---|---|---|
| GET | /api/health | A simple health check endpoint to check the API availability. | txDigest | 
| POST | /api/games/{id}/deal | Executes the initial deal transaction after the game creation. | txDigest | 
| POST | /api/games/{id}/hit | Executes a hit move. | txDigest, ID of correspondingHitRequestobject | 
| POST | /api/games/{id}/stand | Executes a stand move. | txDigest, ID of correspondingStandRequestobject | 
Need of usage of waitForTransaction
An interesting aspect of developing a dApp on Sui, that is coupled to using a full node with/without a load balancer, and requires attention by the developer, is the occurrence of read-after-write and write-after-write cases.
Initial deal transaction
As an example, in the Blackjack game, just after the create game transaction that the user executes, the dealer executes the initial deal transaction. This one accepts an argument and modifies the game object, meaning that you are using an object that was just created.
To ensure that the game object is available in the full node that the dealer is using, you need to call waitForTransaction after the create game transaction.
Hit and stand transactions
In the same way, every time you re-fetch the game object in the frontend, make sure that the previous transaction that modified the game object is already available in the full node.
This leads to the need of exchanging the txDigest between the frontend and the backend, and use waitForTransaction on each write-after-write or read-after-write case.
Frontend
The page component, a central element of the Blackjack game's frontend module, is structured to create an interactive and responsive gaming experience. Written in React, it integrates several features and functions to handle the game's logic and user interactions effectively.
Directories structure
The frontend is a NextJS project, that follows the NextJS App Router project structure.
The main code of the frontend lies under the app/src/ directory.
The main sub-directories are:
- app/: The main code of the pages and the API endpoints.
- components/: The reusable components of the app, organized in sub-directories.
- hooks/: The custom hooks used in the app.
- helpers/,- utils/,- lib/: Multiple helper functions and utilities.
- types/: The types/interfaces used in the app.
- styles/: The global CSS files to style the app.
Components and custom hooks for state management
- 
Custom hooks: To keep the code as structured as possible, multiple custom hooks are utilized to manage the complex state of the game board at each step. The useBlackjackGamecustom hook encapsulates the game state and logic, exposing all the required information (with fields such asgame,isInitialDealLoading), and the required functionality (with methods such ashandleCreateGameandhandleHit) to display and play the game. Multiple additional custom hooks, such asuseCreateBlackjackGame, anduseMakeMoveInBlackjackGameare encapsulating their own piece of state and logic to make the code readable and maintainable.
- 
Component for game initialization: The StartGamecomponent is implemented to facilitate the creation of a new game. It uses thehandleCreateGamefunction of theuseBlackjackGamehook to execute the create-game transaction.
- 
Card displaying and management: The DealerCardsand thePlayerCardscomponents are used to display the total points and the cards owned by the dealer and the player respectively.
- 
Game actions: The GameActionscomponent is used to display the Hit and Stand buttons, and trigger the corresponding actions, as they are exported by theuseBlackjackGamehook to execute the corresponding transactions.
- 
BlackjackBanner: TheBlackjackBannercomponent is used as a custom view to display when the player wins with a blackjack.
Comparison: Blackjack and Coin Flip
Similarities
- 
Blockchain-based logic: Both games are built on Sui, leveraging its capabilities for decentralized applications. The core game logic for each resides in Move modules, ensuring secure and verifiable gameplay. 
- 
State management: In both games, state management is crucial. For Blackjack, this involves managing the player and dealer's hands and scores using React state hooks. In Coin Flip, the state is managed through Move structs like HouseData, which track the house's balance and other game-related details.
- 
Randomness and fair play: Both games emphasize randomness for fairness. Both games use Sui's on-chain randomness module, ensuring verifiable and truly random outcomes for every game. 
- 
Smart contract interactions: Each game involves smart contract interactions for game actions like placing bets, dealing cards (Blackjack), or making guesses (Coin Flip). These interactions are crucial for executing the game's logic on the blockchain. 
Differences
- 
Game mechanics and complexity: Blackjack is a more complex game with multiple actions (hit, stand, deal) and state updates, requiring a more dynamic frontend. In contrast, Coin Flip has a simpler mechanic centered around a single bet and guess outcome. 
- 
User interface (UI) complexity: The Blackjack game involves a more intricate UI to display cards, manage game states, and player interactions. Coin Flip, being simpler in gameplay, requires a less complex UI. 
- 
Backend processing: In Blackjack, the dealer is automated (the machine), and the player's actions directly influence game outcomes. In the Coin Flip game, the house (smart contract) plays a more passive role, primarily in initializing and finalizing the game based on the player's guess. 
- 
Module structure and focus: The Blackjack game focuses more on frontend interactions and real-time updates. The Coin Flip game delves into backend logic with structures like HouseCapandhouse_datafor initializing and managing game data securely on the blockchain.
- 
Multi-version implementation: The Coin Flip game mentions two versions: one susceptible to MEV attacks and another that is resistant, indicating a focus on security and user experience variations. Such variations aren't implemented in Blackjack. 
The complete app example can be found in the blackjack-sui repo.