This Starter-Kit is a small tutorial of how to use the PyCardano library with opshin for interacting with a simple vesting contract on Cardano.
PyCardano is a Cardano library written in Python. It allows users to create and sign transactions without depending on third-party Cardano serialization tools, such as cardano-cli and cardano-serialization-lib, making it a lightweight library, which is simple and fast to set up in all types of environments.
opshin is a Smart Contract language based on Python. It allows users to define and compile Smart Contracts directly within a python environment. It also interacts seemlessly with PyCardano.
For executing the scripts in this starter kit you'll need access to a running Ogmios instance.
In case you don't want to install the required components yourself, you can use Demeter.run platform to create a cloud environment with access to common Cardano infrastructure. The following command will open this repo in a private, web-based VSCode IDE with access to a running Ogmios instance in the preview network.
We have included a number of python scripts for executing specific actions.
You can find scripts to initialize addresses and interact with the cardano-node in scripts.
src contains two folders, on_chain which hosts the actual opshin contract and off-chain which
hosts tooling to interact with the contract.
- Install uv.
On demeter.run or Linux/Ubuntu run
curl -LsSf https://astral.sh/uv/install.sh | shFollow the instructions diplayed to add uv to your local shell.
Otherwise, follow the official documentation here.
- Set up a virtual environment and install the required packages.
uv python install
uv venv
source .venv/bin/activate- Alternatively set up Ogmios or Blockfrost
Follow the instructions on Blockfrost to create a free account and get an API key for the preview network. Make sure the following environment variables are set (defaults are displayed):
export BLOCKFROST_API_KEY="previewyourkey"On demeter.run, simply add the Ogmios Extension for the Preview network through the project console website (the page that shows you demeter.run project -> Connected Extensions -> Browse Extensions -> Cardano Ogmios) If you want to add kupo, use the Kupo Extension as well.
Make sure the following environment variables are set (defaults are displayed):
export OGMIOS_API_HOST=localhost
export OGMIOS_API_PORT=1337
export OGMIOS_API_PROTOCOL=ws
export KUPO_API_HOST=None
export KUPO_API_PORT=80
export KUPO_API_PROTOCOL=httpOnce you have set up uv, you can start interacting with the contract through the prepared scripts.
First, we have to build the vesting contract and generate two key pairs, one for the owner of funds and one for the intended beneficiary.
uv run scripts/build.py
uv run scripts/create_key_pair.py owner
uv run scripts/create_key_pair.py beneficiaryMake sure that the owner is loaded up with some testnet ada before proceeding, by using the testnet faucet. You can find the address of the owner key by running this command
cat keys/owner.addrAfter requesting ada for the owner, send some ada to the beneficiary. The receiver address needs a small amount of ada in order to provide it as collateral when unlocking the funds later.
python3 src/off_chain/distribute.py owner beneficiary Then you can place a vested amount of ada at the contract. If you just requested funds for the owner address, you might need to wait a few minutes or the script will display an error that funds are missing.
python3 src/off_chain/make_vest.py owner beneficiary By default the deadline is 0 seconds after the creation of the vesting, so you can directly proceed and unlock the vested amount with the beneficiary!
python3 src/off_chain/collect_vest.py beneficiaryThat's it! You successfully compiled a Smart Contract on cardano and interacted with it through off-chain tooling. Feel free to dive into the provided scripts and start customizing them for your needs.
If you see the following errors when running the scripts, don't panic, they are temporary issues when the blockchain has not yet included your previous transactions.
pycardano.exception.TransactionFailedException: Failed to submit transaction. Error code: 400. Error message: {"contents":{"contents":{"contents":{"era":"ShelleyBasedEraConway","error":["ConwayUtxowFailure (UtxoFailure (BadInputsUTxO (fromList [TxIn (TxId {unTxId = SafeHash \"5033df02e3cb82488d8cf46fbe18c783c586320cdcb43b77e8d88df5e50ccbc8\"}) (TxIx {unTxIx = 1})])))","ConwayUtxowFailure (UtxoFailure (ValueNotConservedUTxO (Mismatch {mismatchSupplied = MaryValue (Coin 0) (MultiAsset (fromList [])), mismatchExpected = MaryValue (Coin 9936335092) (MultiAsset (fromList []))})))"],"kind":"ShelleyTxValidationError"},"tag":"TxValidationErrorInCardanoMode"},"tag":"TxCmdTxSubmitValidationError"},"tag":"TxSubmitFail"}
This means that the transaction you are trying to submit is invalid, because the previous transaction has not yet been included in a block. Just wait a few minutes and try again, it should work after a short while.
AssertionError: No script UTxOs found!
This means that the script UTxO you are trying to spend from does not exist. This can happen if the transaction creating the script UTxO has not yet been included in a block on the blockchain. Retry in a few minutes.
pycardano.exception.TransactionFailedException: Namespace(EvaluationFailure=Namespace(ScriptFailures=Namespace()))
This means there was an error when evaluating the script. The contract raised an error, maybe due to invalid inputs or due to a program bug. Unfortunately, Blockfrost does not provide detailed error messages for script failures. If you are running your own ogmios and kupo instances, you can get more detailed error messages that help you figuring out what went wrong.