Multi-threaded minting smart contract for the Bitcoin Cash network.
The CashScript code for the smart contract is contract/mint.cash
and the corresponding artifact contract/mint.json
.
The repo also includes helper functions to set up a smart contract with minting-setup.js
and to claim payouts with invoke-payout.js
.
The contract design is multi-threaded meaning multiple contract UTXOs running in parallel. This set up enables multiple simultaneous mint interactions. Clients can randomly pick one of the threads or an application server can manage idle/active UTXOs.
This design also means the mint is not actually sequential, i.e. NFT number 16 can be minted before NFT number 12, depending on how much each of the different threads is used.
The minting contract is designed to be practical. For example the smart contract separates the NFT output to the user from their change output to ensure compatibility with different CashTokens wallets. The contract also has a separate payout function instead of having a payout output on each minting transaction, which would create thousands of small payout outputs. Lastly, it burns the minting NFTs when the contract ends, this way there are no inactive minting NFTs remaining at the end of the contract.
The minting contract is optimized to be minimal in size (the redeem script is only 163 bytes).
For example, the tokenId
is not included as a contract constructor argument which saves ~30 bytes in contract size.
The size for the different txs are as follows:
- Minting transaction, no change ~505 bytes
- Minting transaction, with change ~539 bytes
- Payout transaction, continue mint ~430 bytes
- Payout transaction, end mint ~350 bytes
The contract code itself is clean & commented as to easily be auditable and serve as an example for contract creators.
The contract is flexible, it allows for an optional change output in mintNFT
and does not impose fixed, hardcoded miner fees.
The contract combines burnMintingNft
functionality in a minimal way into the payout
function.
git clone git@github.com:cashninjas/minting-contract.git
npm install
You need to create a minting NFT with a CashTokens wallet in advance and you need to separate the authChain so you can manage your NFT collection's metadata. For more info about CashTokens metadata, see the bcmr specification.
Next, to set up a minting contract, configure the parameters of the mint in mintingParams.js
:
{
tokenId: "",
collectionSize: 10_000,
mintPriceSats: 5_000_000,
payoutAddres: "", // with bitcoincash: or bchtest: prefix
numberOfThreads: 5,
network : "chipnet"
}
To actually do the minting setup on-chain, configure the wallet holding the minting NFT and a little BCH in a .env
file.
Example .env file:
SEEDPHRASE_SETUP = ""
DERIVATIONPATH_SETUP = "m/44'/145'/0'/0/0"
SEEDPHRASE_PAYOUT = ""
DERIVATIONPATH_PAYOUT = "m/44'/145'/0'/0/0"
You need to provide the correct DERIVATIONPATH_SETUP
to an adrress so mainnet-js can initialize a single address wallet.
For example if your minting NFT is at address index 4 in your Electron Cash, change m/44'/145'/0'/0/0
to m/44'/145'/0'/0/4
.
Then create the minting set up with
node minting-setup.js
This will create the different threads for the minting contract with the CashScript advanced TransactionBuilder.
Each tread is a UTXO on the smart contract address with a minting NFT. The minting NFTs each have a different starting commitment, starting from the VMnumber zero (an empty commitment).
To invoke the payouts from the minting contract, configure the wallet authorized to claim the payout in the .env
file.
Then claim the contract payouts with
node invoke-payout.js