Building a voting dApp with Pact and React
One of the best ways to learn a new technology is to get your hands dirty and build applications with it. In this tutorial we'll learn to use Kadena's smart contract language, Pact, to build a minimal voting application that runs on the Chainweb blockchain.
Along the way, we'll learn how to write and test smart contracts in Pact, the basic architecture of dApps on Chainweb, and about major concepts like gas stations and deploying contracts. We'll also get acquainted with several useful tools in the Kadena ecosystem, including the Pact local test server and JavaScript libraries you can use to interact with nodes running Pact.
Voting on the Blockchain
The complete code of this tutorial can also be found in the github repo.
Elections are a necessary part of democracies and democratic organizations. The voting systems used to administer elections must ensure a fair process and trustworthy result — easier said than done! Election security is a deep, fascinating topic, especially when it comes to online voting.
Blockchain technologies are well-suited to help secure online elections. A public blockchain gives participants a single view of all transactions, which makes it easy to verify votes without trusting a central election authority to tally and report the results.
Blockchain technologies don't solve all election security issues, but they're a strong foundation and researchers have proposed fully-secure online voting systems based on them. They've also seen success in the real world: Thailand's Democrat party held an election in which more than 120,000 registered Democrats voted via blockchain.
What We're Building
We'll build a tiny voting dApp prototype that lets anyone with a Kadena wallet address vote for a candidate from a selection of candidates. Each voter (i.e. address) can vote once. Some Kadena accounts are chosen as "election officials", and the smart contract grants them special privileges to select the candidates. Election officials can add new candidates at any time (but they can't remove candidates or adjust their votes).
Once the app is deployed, the election has begun! The frontend for our dApp will help users submit their votes and will display the total votes received by each candidate.
Setup
Requirements
Create Project Structure
Let's start by creating a basic project structure. Open your terminal and run the commands below:
1mkdir election-dapp && cd election-dapp 2mkdir pact 3mkdir front-end 4
We've got the
election-dapp
directory and two additional sub-directories:pact
, which holds the smart contractsfront-end
, which holds the front-end part of our application
Implementing the Voting Smart Contract
A typical developer workflow looks like this:
- Write contract code in
.pact
files - Write tests in
.repl
files - Execute your tests in the REPL
- Deploy to local pact server
- Deploy to Testnet
In this section we will focus on steps 1 to 3. Later, we'll deploy our smart contract to a local Pact server and to Testnet (the test version of Chainweb).
In your project directory, let's create two files:
pact/election.pact
, which will hold the source code for our smart contractpact/election.repl
, which will hold our tests
What is Pact REPL? The Pact REPL is an environment where we can load our Pact source code and work with it interactively. It's a best practice to include a
.repl
file next to your source code which imports your contract, calls functions from it, and inspects its current state to ensure everything is correct.We also have to import some dependencies to our project but first let's provide some context to better understand why we need them. In introduction we explained that our voting smart contract allows anyone with a wallet address to vote for a candidate. Kadena uses an account model so creating a wallet means creating an account for the native coin, KDA, which is a smart contract deployed on Kadena blockchain. The name of this contract is intuitively
coin
.The
coin
contract itself has two additional dependencies:fungible-v2
, an interface that each fungible token deployed on Kadena should implementfungible-xchain-v1
, an interface that provides standard capability for cross-chain transfers.
To be able to properly test our voting contract we will need to invoke functions defined in the
coin
contract so we have to include it in our project together with its dependencies, the fungible-v2
and fungible-xchain-v1
interfaces.You can get the latest version of the
coin
module here, the fungible-v2
interface here and the fungible-xchain-v1
interface here. Make sure to add these files to your project in the pact/root/
directory. You should have 3 new files: coin-v4.pact
, fungible-v2.pact
, fungible-xchain-v1.pact
.Before we begin writing code, let's recap the features of our voting contract:
- Voters can record 1 vote for a candidate of their choice from the list of options. In response to voting, we'll return confirmation of their vote by returning their vote back to them.
- Election administrators can add candidates (so the "add candidate" functionality should be guarded to only allow access to keys belonging to the election administrator accounts).
Election module
We're going to start by creating a Pact module called
election
and define a keyset named election-admin-keyset
which is used to guard certain features of the module.Let's copy the following code in the
election.pact
file:1;; election.pact 2 3;; Define a keyset with name `election-admin-keyset`. 4;; Keysets cannot be created in code, thus we read them in from the load message data. 5(define-keyset "free.election-admin-keyset" (read-keyset 'election-admin-keyset)) 6 7;; Define `election` module 8(module election GOVERNANCE 9 "Election demo module" 10 11 (defcap GOVERNANCE () 12 "Module governance capability that only allows the admin to update this module" 13 ;; Check if the tx was signed with the provided keyset, fail if not 14 (enforce-keyset "free.election-admin-keyset")) 15) 16
The
GOVERNANCE
keyword on the module definition line is the module governance capability and it references the capability defined right below using the defcap
construct. It's purpose is to restrict access to the module upgrade and administration operations, for example later on we'll add an insert-candidate
function that only administrators should be able to call and we'll use the GOVERNANCE capability to guard it. The implementation can be as simple as in our example, enforcing a keyset or more complex like tallying a stakeholder vote on an upgrade hash.Module names and keyset definitions are required to be unique. We will mention this again when we get to deploy our contract to Testnet, but you should keep this in mind when you think about choosing a name for your modules and keysets.
Capabilities
Capabilities offer a system to manage user rights in an explicit way, i.e. allow a user to perform some sensitive task if the required capability has been successfully acquired. If not, the transaction will fail.
Our module already defines one capability, the module governance capability. In addition to that, we're going to define the
ACCOUNT-OWNER
and the VOTED
capabilities. The VOTED
capability is used to emit an event when a vote has been made. The ACCOUNT-OWNER
capability validates the ownership of the KDA account that's used to identify a user. This might not be clear at first but let's look at the code:1;; election.pact 2 3 ;; Import `coin` module while only making the `details` function available 4 ;; in the `election` module body 5 (use coin [ details ]) 6 7 (defcap ACCOUNT-OWNER (account:string) 8 "Make sure the requester owns the KDA account" 9 10 ;; Get the guard of the given KDA account using coin.details function 11 ;; and execute it using `enforce-guard` 12 (enforce-guard (at 'guard (coin.details account))) 13 ) 14 15 (defcap VOTED (candidateId:string) 16 "Emit an event that indicates a vote has been made for the provided candidate" 17 @event 18 true 19 ) 20
The user submitting the vote is identified by the
account
parameter that we will pass to the vote
function that we'll implement later on. This parameter can take any value which is why we need to make sure the value provided is the correct one, in our case it should be the KDA account controlled by the tx initiator. Every KDA account has a guard which controls access to it and we're using the coin.details
function that returns an object of type fungible-v2.account-details
to retrieve this guard for the provided account. Finally we execute the guard using the built-in Pact function enforce-guard
.Don't forget to add the snippet above in the
election
module body.To learn more about guards and capabilities, please visit the Guards, Capabilities and Events section of the Pact official documentation.
Tables and data storage
So far we've defined a module and implemented some capabilities. Now we're going to talk about storing data. We have to store who the candidates are so you can vote for them and as well as who voted already so we can prevent double-voting.
Pact smart contracts store data in tables and each table has its own schema. For our voting contract we need 2 tables:
candidates
and votes
.1 ;; election.pact 2 3 ;; Define the `candidates-schema` schema 4 (defschema candidates-schema 5 "Candidates table schema" 6 7 ;; Candidates table has 2 columns, `name` of type string 8 ;; and `votes` which is an `integer` 9 name:string 10 votes:integer) 11 12 ;; Define the `votes-schema` schema 13 (defschema votes-schema 14 "Votes table schema" 15 16 ;; Votes table has one column, `cid` - Candidate id of type string 17 cid:string 18 ) 19 20 ;; Define the `votes` table that's using the `votes-schema` 21 (deftable votes:{votes-schema}) 22 23 ;; Define the `candidates` table that's using the `candidates-schema` 24 (deftable candidates:{candidates-schema}) 25
To summarize, we created a table to store candidates and their associated vote counts and one for storing what accounts have already voted to prevent double-voting.
To find out about all Pact's supported types you can check the Data Types section in the Pact official documentation.
Pact implements a key-row model which means a row is accessed by a single key. The key is implicitly present in the schema but it is our responsibility as developers to design the schema in a way that we can retrieve the information that we need using a single row query. Multiple row queries are very expensive and should not be used.
The row key is always a simple string, to not be confused with the cryptographic keys used for signing transaction.
Functionality
We've defined our data storage so now we can add functions to read and write data, i.e. candidates and votes. One of the core features of our voting contract is to allow users to vote for a candidate while preventing double-voting so let's implement it:
1 ;; election.pact 2 3 (defun user-voted:bool (account:string) 4 "Check if a user already voted" 5 6 ;; Read from the votes table using `account` param value as key 7 ;; with-default-read allows us to set default values for the table columns 8 ;; that are returned if the row does not exist. 9 (with-default-read votes account 10 11 ;; In this case we're setting the `cid` column default value to an empty string 12 { "cid": "" } 13 { "cid":= cid } 14 15 ;; Check if `cid` is an empty string or not, return true if not, 16 ;; i.e. user already voted and false otherwise, 17 ;; meaning the user did not vote yet 18 (> (length cid) 0)) 19 ) 20 21 (defun candidate-exists:bool (cid:string) 22 "Check if a candidate exists" 23 24 ;; Using a similar approach as in `user-voted` function, 25 ;; in this case to check if a candidate exists 26 (with-default-read candidates cid 27 { "name": "" } 28 { "name" := name } 29 (> (length name) 0)) 30 ) 31 32 (defun vote-protected (account:string candidateId:string) 33 "Safe vote" 34 35 ;; Check that the ACCOUNT-OWNER capability has already been granted, fail if not 36 (require-capability (ACCOUNT-OWNER account)) 37 38 ;; Read the current number of votes the candidate has 39 (with-read candidates candidateId { "votes" := votesCount } 40 41 ;; Increment the number of votes by 1 42 (update candidates candidateId { "votes": (+ votesCount 1) }) 43 44 ;; Record the vote in the `votes` table (prevent double-voting) 45 (insert votes account { "cid": candidateId }) 46 47 ;; Emit an event that can be used by the front-end component to update the number of 48 ;; votes displayed for a candidate 49 (emit-event (VOTED candidateId)) 50 ) 51 ) 52 53 (defun vote (account:string cid:string) 54 "Vote for a candidate" 55 56 ;; Prevent double-voting by checking if the user already voted through `user-voted` function 57 ;; and `enforce` the returned value is `false` 58 (let ((double-vote (user-voted account))) 59 (enforce (= double-vote false) "Multiple voting not allowed")) 60 61 ;; Prevent voting for a candidate that doesn't exist through `candidate-exists` 62 ;; function and `enforce` the returned value is `true` 63 (let ((exists (candidate-exists cid))) 64 (enforce (= exists true) "Candidate doesn't exist")) 65 66 ;; Try to acquire the `ACCOUNT-OWNER` capability which checks 67 ;; that the transaction owner is also the owner of the KDA account provided as parameter to our `vote` function. 68 (with-capability (ACCOUNT-OWNER account) 69 70 ;; While the `ACCOUNT-OWNER` capability is in scope we are calling `vote-protected` which is the function that updates the database 71 (vote-protected account cid)) 72 73 (format "Voted for candidate {}!" [cid]) 74 ) 75
A quick recap: we implemented a
vote
function that allows to vote for a candidate while preventing double-voting, voting for a candidate that doesn't exist or voting with an account that the user doesn't own.Now that we can vote, we also need a function to read the number of votes a candidate received:
1(defun get-votes:integer (cid:string) 2 "Get the votes count by cid" 3 4 ;; Read the row using cid as key and select only the `votes` column 5 (at 'votes (read candidates cid ['votes])) 6) 7
Last thing on the list is adding candidates:
1(defun insert-candidate (candidate) 2 "Insert a new candidate, admin operation" 3 4 ;; Try to acquire the GOVERNANCE capability 5 (with-capability (GOVERNANCE) 6 ;; While GOVERNANCE capability is in scope, insert the candidate 7 (let ((name (at 'name candidate))) 8 ;; The key has to be unique, otherwise this operation will fail 9 (insert candidates (at 'key candidate) { "name": (at 'name candidate), "votes": 0 }))) 10) 11 12(defun insert-candidates (candidates:list) 13 "Insert a list of candidates" 14 ;; Using the above defined `insert-candidate` to bulk-insert a list of candidates 15 (map (insert-candidate) candidates) 16) 17
Inserting a new candidate is an "admin-only" operation and we reused the already defined
GOVERNANCE
capability to guard it.We have now essentially completed our module. All the required functionality is implemented.
When a module is deployed, the tables that it defines need to be created. This is done using the
create-table
function. Insert the snippet below after the module's closing parenthesis:1;; election.pact 2 3;; Read the `upgrade` key from transaction data 4(if (read-msg "upgrade") 5 ;; If its value is true, it means we're upgrading the module 6 ["upgrade"] 7 ;; Otherwise, the transaction is deploying the module and we need to create the tables 8 [ 9 (create-table candidates) 10 (create-table votes) 11 ] 12) 13
Code outside the module will be called when the module is loaded the first time, when its deployed or upgraded. In the snippet above we are checking if the
upgrade
key that comes from transaction data is true
and only execute the create-table
calls if it's not since we cannot recreate tables when upgrading a module.You can find the complete source code of the
election.pact
contract here.It's time to summarize what we've learned so far:
- we can use Pact capabilities to protect certain features of our smart contract
- dynamic data is stored in tables, it's accessed using a key and we should design our tables in such way that we can retrieve the information using a single row query.
- we can validate the owner of an account by executing its guard
These are general concepts that you should keep in mind when you develop Pact smart contracts.
Testing the contract
We wrote quite a bit of code but at this point we don't know if it's working correctly. A critical step in smart-contract development process is writing a proper set of tests which is what we're going to focus on now.
We separated writing functionality and writing tests to make it easier to follow this tutorial but in a real-world scenario you should work on these in parallel.
We're going to start by setting up the environment data that we need for our tests, load the required modules, i.e.
coin
module and of our election
module and create some KDA accounts that we will use to vote later on.Open the
election.repl
file and copy the snippet below:1;; election.repl 2 3;; begin-tx and commit-tx simulate a transaction 4(begin-tx "Load modules") 5 6;; set transaction JSON data 7(env-data { 8 ;; Here we set the required keysets. 9 ;; Note: 10 ;; - in a real transaction, `admin-key` would be a public key 11 ;; - "keys-all" is a built-in predicate that specifies all keys are needed to sign a tx, 12 ;; in this case we only set one key 13 'election-admin-keyset: { "keys": ["admin-key"], "pred": "keys-all" }, 14 'alice-keyset: { "keys": ["alice-key"], "pred": "keys-all" }, 15 'bob-keyset: { "keys": ["bob-key"], "pred": "keys-all" }, 16 'namespace-keyset: { "keys": [ ], "pred": "keys-all" }, 17 18 ;; Upgrade key is set to false because we are deploying the modules 19 'upgrade: false 20}) 21 22;; All Pact modules must exist within a namespace on Chainweb, except for basic contracts provided by Kadena. 23;; There are two namespaces available for anyone to use on Chainweb: the 'free' namespace and the 'user' 24;; namespace. Our contract uses the "free" namespace, so we need to make sure it exists in our REPL 25;; environment. 26 27;; Defining a namespace requires that we provide two keysets. The first keyset indicates the user that must 28;; have signed any transaction that deploys code to the given namespace. The second keyset is the namespace 29;; admin's keyset, and it indicates that the admin must sign the transaction that creates the new namespace. 30;; For testing purposes we will use the same mock namespace keyset for both. 31(define-namespace "free" (read-keyset "namespace-keyset") (read-keyset "namespace-keyset")) 32 33;; load fungible-v2 interface 34(load "root/fungible-v2.pact") 35 36;; load fungible-xchain-v1 interace 37(load "root/fungible-xchain-v1.pact") 38 39;; load coin module 40(load "root/coin-v4.pact") 41 42;; create coin module tables 43(create-table coin.coin-table) 44(create-table coin.allocation-table) 45 46;; load election module 47(load "election.pact") 48 49;; commit the transaction 50(commit-tx) 51 52(begin-tx "Create KDA accounts") 53 54;; create "alice" KDA account 55(coin.create-account "alice" (read-keyset "alice-keyset")) 56;; create "bob" KDA account 57(coin.create-account "bob" (read-keyset "bob-keyset")) 58 59(commit-tx) 60
Now that this initial setup is done, we can go on and write some tests. Notice that we did not add any candidates just yet so any attempt to vote at this point should fail. Let's try it:
1;; election.repl 2 3(begin-tx "Vote for non-existing candidate") 4 5;; set the key signing this transaction, `alice-key` 6;; setting `caps` as an empty array translates into `unrestricted mode`, meaning our keyset 7;; can be used to sign anything, it's not restricted to a specific set of capabilities 8(env-sigs [{ "key": "alice-key", "caps": []}]) 9;; this test passes because the election.vote call fails 10(expect-failure "Can't vote for a non-existing candidate" (election.vote "alice" "5")) 11 12(commit-tx) 13
In the snippet above we've learned that we can use
expect-failure
to test that an expression will fail and that we can configure the keys and capabilities signing a transaction using env-sigs
.REPL-Only Functions
expect-failure
and env-sigs
are two of the many REPL-only functions that we can use in .repl
files to test Pact smart-contracts by simulating blockchain environment. You can check the complete list of REPL-only functions in the Pact official documentation.Next we're going to add some candidates and check if their number of votes is correctly initialized.
1;; election.repl 2 3(begin-tx "Add candidates") 4(use free.election) 5 6;; Need to provide the key that is part of the election-admin-keyset 7(env-sigs [{ "key": "admin-key", "caps": []}]) 8 9;; Call `insert-candidates` to add 3 candidates 10(free.election.insert-candidates [{ "key": "1", "name": "Candidate A" } { "key": "2", "name": "Candidate B" } { "key": "3", "name": "Candidate B" }]) 11 12;; test if votes count for candidate "1" is initialized with 0 13(expect "votes for Candidate A initialized" (get-votes "1") 0) 14 15;; test if votes count for candidate "2" is initialized with 0 16(expect "votes for Candidate B initialized" (get-votes "2") 0) 17 18;; test if votes count for candidate "3" is initialized with 0 19(expect "votes for Candidate C initialized" (get-votes "3") 0) 20 21(commit-tx) 22
We can use
expect
function to test that any 2 expressions value is equal, in this case we checked if get-votes
returns 0 for each candidate.Moving on, we want to validate that votes are correctly recorded, the
VOTED
event is emitted and double-voting is not allowed.1;; election.repl 2 3(begin-tx) 4(use free.election) 5;; we set the key signing this tx and the capabilities that can be signed 6;; coin.GAS is a capability that allows gas payments, we will talk more about gas and gas stations in the 7;; next section 8;; election.ACCOUNT-OWNER is the capability we implemented that validates the owner of the KDA account 9(env-sigs [{ "key": "alice-key", "caps": [(coin.GAS), (free.election.ACCOUNT-OWNER "alice")]}]) 10 11;; test if votes count for candidate "1" is correctly increased by 1 12;; 1. Retrieve the number of votes 13(let ((count (get-votes "1"))) 14 ;; 2. Vote 15 (vote "alice" "1") 16 ;; 3. Check if the vote was correctly recorded 17 (expect "votes count is increased by 1" (get-votes "1") (+ count 1))) 18 19;; Test if the `VOTED` event with parameter "1" was emitted in this transaction 20(expect "voted event" 21 [ { "name": "free.election.VOTED", "params": ["1"], "module-hash": (at 'hash (describe-module "free.election"))}] 22 (env-events true)) 23 24;; execute the same test using a different account 25(env-sigs [{ "key": "bob-key", "caps": [(coin.GAS), (free.election.ACCOUNT-OWNER "bob")]}]) 26;; test if votes count for candidate "2" is correctly increased by 1 27(let ((count (get-votes "2"))) 28 (vote "bob" "2") 29 (expect "votes count is increased by 1" (get-votes "2") (+ count 1))) 30 31(expect "voted event" 32 [ { "name": "free.election.VOTED", 33 "params": ["2"], 34 "module-hash": (at 'hash (describe-module "free.election")) 35 } 36 ] 37 (env-events true)) 38 39;; test that bob's attempt to vote twice fails 40(expect-failure "Double voting not allowed" (vote "bob" "1")) 41 42(commit-tx) 43
Notice the
let
construct that we used above, it is helpful when you need to bind some variables to be in the same scope as other logic that uses them. In our case we first loaded the number of votes and binded the result to count
variable which we compared with the new count after submitting a vote. Feel free to read more about let
and let*
in Pact official documentation.Write a test Can you think of some cases that we didn't cover? Hint: ACCOUNT-OWNER.
Try to write a test that validates that only the correct owner of an account can vote.
The only thing left is to run these tests and confirm everything is working:
1$ pact 2pact> (load "election.repl") 3
The REPL preserves state between subsequent runs unless the optional parameter
reset
is set to true (load "election.repl" true)
.Let's recap what we've learned in this section:
- we can test Pact smart-contracts using
.repl
scripts that simulate blockchain environment through a set of REPL-only functions - before writing tests we need to make sure all required modules are loaded as well as KDA accounts are created if we need them
- we can test functions returned values, emitted events, failure scenarios (and much more that we couldn't cover) :::
Implementing the Gas Station
A unique feature of Kadena is the ability to allow gas to be paid by a different entity than the one who initiated the transaction. This entity is what we call a gas station.
Gas is the cost necessary to perform a transaction on the network. Gas is paid to miners and its price varies based on supply and demand. It's a critical piece of the puzzle, but at the same time it brings up a UX problem. Every user needs to be aware of what gas is as well as how much gas they need to pay for their transaction. This causes significant friction and a less than ideal experience.
To help mitigate this problem Kadena brings an innovation to the game. Hello gas stations!
Gas stations are a way for dApps to subsidize gas costs for their users. This means that your user doesn't need to know what gas is or how much the gas price is, which translates into a smooth experience when interacting with your dApp.
In our voting app this will allow users to submit votes without paying for gas, instead gas will be subsidized by the gas station. In short, this means that miners will still be paid, but our users can vote for free.
The standard for gas station implementation is defined by the
gas-payer-v1
interface. The gas-payer-v1
interface is deployed to all chains on testnet
and mainnet
so you can directly use it in your contract. We can specify that a module implements an interface using the (implements INTERFACE)
construct.Pact interfaces are similar to Java's interfaces, Scala's traits, Haskell's typeclasses or Solidity's interfaces. If you're not familiar with this concept you can read more about it in Pact reference.
1(interface gas-payer-v1 2 3 (defcap GAS_PAYER:bool 4 ( user:string 5 limit:integer 6 price:decimal 7 ) 8 @doc 9 " Provide a capability indicating that declaring module supports \ 10 \ gas payment for USER for gas LIMIT and PRICE. Functionality \ 11 \ should require capability (coin.FUND_TX), and should validate \ 12 \ the spend of (limit * price), possibly updating some database \ 13 \ entry. \ 14 \ Should compose capability required for 'create-gas-payer-guard'." 15 @model 16 [ (property (user != "")) 17 (property (limit > 0)) 18 (property (price > 0.0)) 19 ] 20 ) 21 22 (defun create-gas-payer-guard:guard () 23 @doc 24 " Provide a guard suitable for controlling a coin account that can \ 25 \ pay gas via GAS_PAYER mechanics. Generally this is accomplished \ 26 \ by having GAS_PAYER compose an unparameterized, unmanaged capability \ 27 \ that is required in this guard. Thus, if coin contract is able to \ 28 \ successfully acquire GAS_PAYER, the composed 'anonymous' cap required \ 29 \ here will be in scope, and gas buy will succeed." 30 ) 31) 32
@doc
is a metadata field used to provide documentation and @model
is used by Pact tooling to verify the correctness of the implementation. You can read more about docs and metadata in Pact reference.Our module needs to implement all the functions and capabilities defined by the
gas-payer-v1
interface:GAS_PAYER
capabilitycreate-gas-payer-guard
function
A gas station allows someone to debit from a coin account that they do not own, gas station account, to pay the gas fee for a transaction under certain conditions. How exactly that happens, let's see below.
Create a new file
election-gas-station.pact
and paste the following snippet:1;; election-gas-station.pact 2 3(module election-gas-station GOVERNANCE 4 (defcap GOVERNANCE () 5 "Only admin can update the smart contract" 6 (enforce-keyset "free.election-admin-keyset")) 7 8 ; Signal that the module implements the gas-payer-v1 interface 9 (implements gas-payer-v1) 10 11 ; Import the coin module, we need it to create a KDA account that will be controlled 12 ; by the gas station 13 (use coin) 14) 15
Next we will implement the
gas-payer-v1
interface. We don't want to let users abuse our gas station so we'll have to add a limit for the maximum gas price we're willing to pay or make sure it can only be used to pay for transactions that are calling the election
module. Let's get to it:1;; election-gas-station.pact 2 3(defun chain-gas-price () 4 "Return gas price from chain-data" 5 ; chain-data is a built-in function that returns tx public metadata 6 ; we are using it to retrieve the tx gas price 7 (at 'gas-price (chain-data))) 8 9(defun enforce-below-or-at-gas-price:bool (gasPrice:decimal) 10 (enforce (<= (chain-gas-price) gasPrice) 11 (format "Gas Price must be smaller than or equal to {}" [gasPrice]))) 12 13(defcap GAS_PAYER:bool 14 ( user:string 15 limit:integer 16 price:decimal 17 ) 18 19 ; There are 2 types of Pact transactions: exec and cont 20 ; `cont` is used for multi-step pacts, `exec` is for regular transactions. 21 ; In our case transaction has to be of type `exec`. 22 (enforce (= "exec" (at "tx-type" (read-msg))) "Inside an exec") 23 24 ; A Pact transaction can have multiple function calls, but we only want to allow one 25 (enforce (= 1 (length (at "exec-code" (read-msg)))) "Tx of only one pact function") 26 27 ; Gas station can only be used to pay for gas consumed by functions defined in `free-election` module 28 (enforce 29 ; We take the first 15 characters and compare it with `(free.election` 30 ; to make sure a function from our module is called. 31 ; `free` is the namespace where our module will be deployed. 32 (= "(free.election." (take 15 (at 0 (at "exec-code" (read-msg))))) 33 "Only election module calls allowed") 34 35 ;; Limit the gas price that the gas station can pay 36 (enforce-below-or-at-gas-price 0.000001) 37 38 ; Import the `ALLOW_GAS` capability 39 (compose-capability (ALLOW_GAS)) 40) 41
To recap, the
GAS_PAYER
capability implementation performs a few checks and composes the ALLOW_GAS
capability that we will define next. chain-gas-price
and enforce-below-or-at-gas-price
are helper functions to limit the gas price that our gas station is willing to pay.1;; election-gas-station.pact 2 (defcap ALLOW_GAS () true) 3 4 (defun create-gas-payer-guard:guard () 5 (create-user-guard (gas-payer-guard)) 6 ) 7 8 (defun gas-payer-guard () 9 (require-capability (GAS)) 10 (require-capability (ALLOW_GAS)) 11 ) 12 13 (defconst GAS_STATION "election-gas-station") 14 15 (defun init () 16 (coin.create-account GAS_STATION (create-gas-payer-guard)) 17 ) 18) 19 20(if (read-msg 'upgrade) 21 ["upgrade"] 22 [ 23 (init) 24 ] 25) 26
First we define the
ALLOW_GAS
capability which is brought in scope by the GAS_PAYER
capability through compose-capability
function.Composing capabilities allows for modular factoring of guard code, e.g. an "outer" capability could be composed out of multiple "inner" capabilities. Also composed capabilities are only in scope when their parent capability is granted.
Then we implement the
gas-payer-guard
function which tests if GAS
(magic capability defined in coin contract) and ALLOW_GAS
capabilities have been granted which are needed to be able to pay for gas fees. By composing ALLOW_GAS
in GAS_PAYER
we hide the implementation details of GAS_PAYER
that gas-payer-guard
function does not need to know about. This is then used in create-gas-payer-guard
to create a special guard for the coin contract account from where the gas fees are paid.Last thing we need is to create an account where the funds will be stored which is what happens in the
init
function. As you can see, the guard of that account is the guard returned by create-gas-payer-guard
, essentially allowing access to the account as long as GAS
and ALLOW_GAS
capabilities have already been granted.To summarize, a gas station is a coin account with a special guard that's valid if both
GAS
and ALLOW_GAS
capabilities are granted. If you're wondering how GAS_PAYER
is granted, the answer is signature capabilities. We will see how this works in the frontend section of this tutorial where we interact with the smart contracts.Guards and capabilities are an entire topic that we cannot cover in detail in this tutorial. To learn more check the Guards, Capabilities and Events section of the Pact documentation.
Deploying to Chainweb
In order to deploy our contracts to the real blockchain network, whether it's Testnet or Mainnet we need to pay for the transaction using gas fees.
In this tutorial we are using Chainweaver wallet to create accounts and sign transactions. Head over to Chainweaver and create an account on
testnet
.Next step is to fund your
testnet
account using this faucet. You will receive 20 Testnet KDA.:::note Namespaces & Modules Names
Each module or interface needs to be part of a namespace. The
free
namespace is available to use on both mainnet
and testnet
.To set the namespace of a module we have to use the
namespace
function. Insert the following line at the beginning of election.pact
and election-gas-station.pact
files:1(namespace 'free) 2
Within the same namespace, each module name needs to be unique, similar requirement for defined keysets.
Also when accessing a module's function we have to use the fully qualified name {namespace}.{module-name}.{function-name}, e.g.
free.election.vote
. You can [read more about namespaces] here.
:::Here's a snippet that you can use to list all deployed modules by using the top-level
list-modules
built-in function:1const { PactCommand } = require('@kadena/client'); 2const { createExp } = require('@kadena/pactjs'); 3 4const NETWORK_ID = 'testnet04'; 5const CHAIN_ID = '0'; 6const API_HOST = `https://api.testnet.chainweb.com/chainweb/0.0/${NETWORK_ID}/chain/${CHAIN_ID}/pact`; 7 8listModules(); 9 10async function listModules() { 11 const pactCommand = new PactCommand(); 12 const publicMeta = { 13 chainId: CHAIN_ID, 14 gasLimit: 6000, 15 gasPrice: 0.001, 16 ttl: 600, 17 }; 18 pactCommand.code = createExp('list-modules'); 19 pactCommand.setMeta(publicMeta, NETWORK_ID); 20 21 const response = await pactCommand.local(API_HOST); 22} 23
You can use the snippets below to deploy your contract to chain 0 on
testnet
and mainnet
:1npm install @kadena/client 2npm install @kadena/chainweb-node-client 3
1const { PactCommand, signWithChainweaver } = require('@kadena/client'); 2const fs = require('fs'); 3 4const NETWORK_ID = 'testnet04'; 5const CHAIN_ID = '0'; 6const API_HOST = `https://api.testnet.chainweb.com/chainweb/0.0/${NETWORK_ID}/chain/${CHAIN_ID}/pact`; 7const CONTRACT_PATH = '../pact/election.pact'; 8const ACCOUNT_NAME = 'some-account-name'; 9const PUBLIC_KEY = 'some-public-key'; 10 11const pactCode = fs.readFileSync(CONTRACT_PATH, 'utf8'); 12 13deployContract(pactCode); 14 15async function deployContract(pactCode) { 16 const publicMeta = { 17 ttl: 28000, 18 gasLimit: 100000, 19 chainId: CHAIN_ID, 20 gasPrice: 0.000001, 21 sender: ACCOUNT_NAME, // the account paying for gas 22 }; 23 const pactCommand = new PactCommand() 24 .setMeta(publicMeta, NETWORK_ID) 25 .addCap('coin.GAS', PUBLIC_KEY) 26 .addData({ 27 'election-admin-keyset': [PUBLIC_KEY], 28 upgrade: false, 29 }); 30 pactCommand.code = pactCode; 31 32 const signedTransaction = await signWithChainweaver(pactCommand); 33 34 const response = await signedTransaction[0].send(API_HOST); 35 console.log(response); 36} 37
1const { PactCommand, signWithChainweaver } = require('@kadena/client'); 2const fs = require('fs'); 3 4const NETWORK_ID = 'mainnet01'; 5const CHAIN_ID = '1'; 6const API_HOST = `https://api.chainweb.com/chainweb/0.0/${NETWORK_ID}/chain/${CHAIN_ID}/pact`; 7const CONTRACT_PATH = '../pact/election.pact'; 8const ACCOUNT_NAME = 'some-account-name'; 9const PUBLIC_KEY = 'some-public-key'; 10 11const pactCode = fs.readFileSync(CONTRACT_PATH, 'utf8'); 12 13deployContract(pactCode); 14 15async function deployContract(pactCode) { 16 const publicMeta = { 17 ttl: 28000, 18 gasLimit: 65000, 19 chainId: CHAIN_ID, 20 gasPrice: 0.000001, 21 sender: ACCOUNT_NAME, // the account paying for gas 22 }; 23 const pactCommand = new PactCommand() 24 .setMeta(publicMeta, NETWORK_ID) 25 .addCap('coin.GAS', PUBLIC_KEY) 26 .addData({ 27 'election-admin-keyset': [PUBLIC_KEY], 28 upgrade: false, 29 }); 30 pactCommand.code = pactCode; 31 32 const signedTransaction = await signWithChainweaver(pactCommand); 33 34 const response = await signedTransaction[0].send(API_HOST); 35 console.log(response); 36} 37
In order to pay transaction fees on
mainnet
you will have to fund your account with real KDA. :::The above snippets can also be found in the tutorial repo.
Frontend
If you made it until here, congrats! We wrote, tested and deployed our smart contract but we're still missing a key component, a UI for users to interact with our dApp, so let's get this done.
Start by adding the required libraries from Kadena.js as a dependency to your project either via a package manager or add it to your asset pipeline similar to any other JavaScript library.
1npm init -y 2npm install @kadena/client @kadena/chainweb-node-client --save 3
Typescript
The Kadena.js team has created libraries that allow Javascript/Typescript users to easily interact with the Kadena Blockchain. Also there's a commandline tool
pactjs-cli
that allows generation of types from pact contracts, which we're going to make use of in this tutorial. Let's first add the required libraries to your project.1npm install typescript @kadena/types --save-dev 2npm install @kadena/pactjs-cli -g 3
create a file in the root of the front-end folder called 'tsconfig.json' and paste in the following JSON
1{ 2 "compilerOptions": { 3 "types": [".kadena/pactjs-generated"], 4 "module": "commonjs", 5 "esModuleInterop": true, 6 "target": "es6", 7 "moduleResolution": "node", 8 "sourceMap": true, 9 "outDir": "dist" 10 }, 11 "lib": ["es2015"] 12} 13
From the root of the front-end folder, use the following command to generate type for our
election
, election-gas-station
and coin
contract. Generating types for the coin
contract is necessary because when paying for gas we use the capability coin.GAS
from the coin contract so we also need those types generated.1pactjs contract-generate --file ../pact/election.pact; pactjs contract-generate --file ../pact/election-gas-station.pact; pactjs contract-generate --file ../pact/root/coin-v4.pact 2
The log shows what has happened. Inside the
node_modules
directory, a new package has been created: .kadena/pactjs-generated
. This package is extending the @kadena/client types to give you type information. Make sure to add "types": [".kadena/pactjs-generated"]
to your tsconfig.json.Our implementation
In this tutorial we're using ReactJS but you are free to use any framework that you are comfortable with. The main focus will be on blockchain and wallet interaction.
There are a few key aspects concerning a frontend implementation of a blockchain application:
- reading data from smart contracts
- allowing users to sign and submit transactions
- notify users when various actions take place like a transaction being mined or a smart contract event was emitted
The complete code of this tutorial can also be found in front-end folder in the tutorial repo. For demonstration purposes the election smart contracts have been deployed to testnet chain 0
Read Data
For this demo application we would like to display the number of votes that each candidate received. To do that we have to call the
get-votes
function from our election
module. Here's what that looks like:1// ./api.ts 2 3import { Pact, signWithChainweaver } from '@kadena/client' 4import { pollTransactions } from './utils' 5 6const NETWORK_ID = 'testnet04' 7const CHAIN_ID = '0' 8const API_HOST = `https://api.testnet.chainweb.com/chainweb/0.0/${NETWORK_ID}/chain/${CHAIN_ID}/pact` 9 10const accountKey = (account: string): string => account.split(':')[1] 11 12/** 13 * Return the amount of votes a candidate has received 14 * 15 * @param candidateId - The candidate's id 16 * @return the number of votes 17 */ 18export const getVotes = async (candidateId: string): Promise<number> => { 19 const transactionBuilder = Pact.modules['free.election']['get-votes'](candidateId) 20 const { result } = await transactionBuilder.local(API_HOST) 21 22 if (result.status === 'success') { 23 return result.data.valueOf() as number 24 } else { 25 console.log(result.error) 26 return 0 27 } 28} 29
We're sending a command to the
/local
endpoint where the pactCode
attribute is a call to our module function which returns the number of votes for the given candidate.Remember to always use the fully qualified name, namespace.module.function.
Here's a screenshot from our demo app where we display the candidates and the number of votes received by each candidate:
Sign & Send Transaction
The next step is to allow users to vote for a candidate. When it comes to updating on-chain data, each dApp has to implement the following flow:
- Create transaction
- Sign transaction
- Send transaction
- Notify when transaction is mined
In this tutorial we are using Chainweaver wallet to sign transactions, other wallets might have a different API but the steps mentioned above are similar. There might be the case where a wallet takes care of more than signing a transaction (e.g. it also sends it to the network) and you will have to adapt your implementation accordingly.
@kadena/client provides a couple of useful methods here:
signWithChainweaver
to interact with the Chainweaver signing API and send
on the ICommandBuilder
to submit the signed transaction to the network.In the snippet below we are constructing a transaction that calls the
free.election.vote
contract function to vote for a candidate.1// ./api.ts 2 3/** 4 * Vote for a candidate and poll the transaction status afterwards 5 * 6 * @param account - The account that is voting 7 * @param candidateId - The candidateId that is being voted for 8 * @return 9 */ 10export const vote = async ( 11 account: string, 12 candidateId: string, 13): Promise<void> => { 14 const transactionBuilder = Pact.modules['free.election'] 15 .vote(account, candidateId) 16 .addCap('coin.GAS', accountKey(account)) 17 .addCap('free.election.ACCOUNT-OWNER', accountKey(account), account) 18 .setMeta( 19 { 20 ttl: 28000, 21 gasLimit: 100000, 22 chainId: CHAIN_ID, 23 gasPrice: 0.000001, 24 sender: account, 25 }, 26 NETWORK_ID, 27 ); 28 29 const signedTransaction = await signWithChainweaver(transactionBuilder); 30 31 console.log(`Sending transaction: ${signedTransaction[0].code}`); 32 const response = await signedTransaction[0].send(API_HOST); 33 34 console.log('Send response: ', response); 35 const requestKey = response.requestKeys[0]; 36 await pollTransactions([requestKey], API_HOST); 37}; 38
Notice the
addCap
function where we define the capabilities that the user's keyset will have to sign. In this case we have two:coin.GAS
-> enables the payment of gas feesfree.election.ACCOUNT-OWNER
-> checks if the user is the owner of the KDA account
Scoping signatures Keep in mind, for security reasons a keyset should only sign specific capabilities and using a keyset in "unrestricted mode" is not recommended. Scoping the signature allows the signer to safely call untrusted code which is an important security feature of Pact and Kadena.
"Unrestricted mode" means that we do not define any capabilities when creating a transaction.
Since this is a transaction that requires gas fees, we now set
sender
(account paying for gas) to the name of the KDA account of the user. If we would want to utilize the gas station we deployed we would set the sender to the account owned by our gas station election-gas-station
and use the free.election-gas-station.GAS_PAYER
capability instead of coin.GAS
.Lastly, to get the result of a transaction we are using the
pollTransactions
helper method which can be found in the project repository.Going back to the UI, we implemented this signing flow using a modal window where users have to enter their KDA account. Once the account is entered and the user hasn't voted yet the Vote Now button will become available. Clicking on the Vote Now button will automatically open the Chainweaver signing wizard.
Below is the first step of the Chainweaver request signing wizard:
Once the transaction is signed, our dApp modal will automatically submit it to the network.
The request key together with the transaction result are displayed in the browsers console output.
Note: Since mining is an external process, while waiting for our transaction to be included in the blockchain, the user should be able to keep using the application freely.
As an extra excercise; modify the code to utilize the gasstation instead of having the user pay for gas fees.
Conclusion
It took a while but we are now at the end of this tutorial. Congratulations! You've managed to implement a complete dApp on Kadena blockchain and we hope you found this guide useful.
Stay tuned for more tutorials and we cannot wait to see what dApps YOU will build next!
Author:
Jermaine Jong
Software Engineer
Date:
Jan 20, 2023
Read:
38 mins read
Explore more tutorials
Tutorial
Pact Core Concepts Part 1 - Introduction to Blockchain Development with Kadena
Apr 20, 2023
Thomas Honeyman
Senior Engineer
Tutorial
Pact Core Concepts Part 2 - Learn Pact in 20 Minutes
Apr 27, 2023
Thomas Honeyman
Senior Engineer
Tutorial
Pact Core Concepts Part 3 - Testing and Formal Verification in the Pact REPL
May 04, 2023
Thomas Honeyman
Senior Engineer