Generate Bindings
Let's turn our attention to how we'll interact with the deployed smart contract. This is where the TypeScript bindings come in! But, I hear you ask: What are TypeScript bindings?
These bindings are a feature of the Stellar CLI that will generate and produce a fully typed NPM package ready for integration into your frontend. This means you can import and invoke a smart contract as if it were any other nodejs package! You get typed functions for each of your contract's functions, and those will result in a built, simulated, signable, and submittable assembled transaction!
This can be done with ANY contract that's live on the network! Or, you can use it on contracts you've only compiled locally, too.
We'll be generating our contract bindings, and keeping them in the same repository as our frontend code. However, you could do this in a lot of different ways:
- Your frontend can instantiate a
contract.Client
instance using thefromWasmHash
function of the JavaScript SDK. This can generate bindings on-the-fly as your users browse your application. - Your deploy process might include a step that builds/deploys/binds a contract package at deploy-time.
- You could even generate and publish a bindings package all by itself. Then
pnpm install <bindings_package_name>
can be done in any dapp that you (or somebody else) might need to interact with that contract.
The manual methodβ
Before you skip ahead! Take a look at this (brief) section. It's really useful to have a full understanding of what steps we're going through in the automated section. This will help you adapt and/or troubleshoot this tutorial for your specific purposes.
Install the compiled contractβ
The smart contract code needs to be installed to the network first. This uploads the compiled, binary Wasm file to the blockchain to be instantiated into a contract later on. From inside your project directory:
stellar contract install \
--source <identity or secret key> \
--network testnet \
--wasm ./target/wasm32-unknown-unknown/release/ye_olde_guestbook.wasm
Deploy a contract instanceβ
This will return a hexadecimal hash corresponding to the uploaded Wasm executable. This hash can then be used in the deploy command to create a new contract instance:
stellar contract deploy \
--source <identity or secret key> \
--network testnet \
--wasm-hash <wasm_hash_from_install_step>
Generate bindings for the deployed contractβ
Now we can (again) use the Stellar CLI to generate bindings from the contract we've just deployed. You can also generate these bindings from your local Wasm file using the --wasm-hash
parameter. The --overwrite
parameter is used to tell the CLI that it should output the generated bindings package, even if it finds the directory is not empty (i.e., we're re-binding a contract because we've modified the code and redeployed it).
stellar contract bindings typescript \
--network testnet \
--id <contract_address_from_deploy_step> \
--output-dir ./packages/ye_olde_guestbook \
--overwrite
We'll need to build the bindings package, since (in its initial state) the package is mostly TypeScript types and stubs for the various contract functions.
cd packages/ye_olde_guestbook
pnpm install
pnpm run build
cd ../..
Import the bindings package as a project dependencyβ
With our bindings generated, we can add it to our frontend project. Run this from the root of your project:
pnpm add file:./packages/ye_olde_guestbook
Import the bindings client into the SvelteKit projectβ
We're straying just a bit into the Svelte-ish side of things here. The main goal of this step is to get the contract client (which is the "bindings package" we've just generated) into our frontend in a way that makes it usable anywhere we need it. In SvelteKit, we put it into src/lib/contracts
because that means we can easily access the client by importing from $lib/contracts/ye_olde_guestbook
whenever and wherever we need it.
Now, we'll define the contract client in a way we can easily access it through the rest of our app.
import * as Client from "ye_olde_guestbook"; // import the package we just added as a dependency
import { PUBLIC_STELLAR_RPC_URL } from "$env/static/public"; // import the RPC url from the .env file
// instantiate and export the Client class from the bindings package
export default new Client.Client({
...Client.networks.testnet, // this includes the contract address and network passphrase
rpcUrl: PUBLIC_STELLAR_RPC_URL, // this is required to invoke the contract through RPC calls
});
The automated wayβ
That was a lot of steps and a lot of work wasn't it!? The good news is that our starter template (remember that?) comes with an initialize.js
script that will perform all of those actions for you! This script will go through all the following steps for you:
- Create and fund a keypair in the CLI
- Install and deploy all contracts in the
/contracts
directory - Generate bindings from the deployed contracts
- Create a
$lib/contracts/<contract_alias>.ts
file for easy import into your frontend code
You can always customize this script to suit your needs. Check out the source code here (which has been documented with comments). Or, you can see the officially maintained script in the soroban-template-astro
repository, as well.
Run the initialization script like so:
node initialize.js
For a more comprehensive overview of the process of creating, customizing, and using initialization scripts like this, check out the frontend template guide.
We've also added a command to the package.json
scripts, so you can run this initialize script simply by running (from your project's root directory):
pnpm run setup
Right, so we've now created a starter project, written a guestbook smart contract, and generated an NPM package that will help us interact with that contract on the network. Amazing!
Next up, let's take a look at how our users will connect with and interact with our dapp. It's time for passkeys! (insert air horn noises)π’