Introduction

We're going to build an example application that mirrors the golang cosmos-sdk example application called Nameservice. There is also a tutorial for that application which you can find here for comparison.

Application Specification

The Nameservice application is a simple marketplace for a name resolution service. Let us say that a Name resolves to type called Whois where

data Whois = Whois
  { whoisValue :: Text
  , whoisOwner :: Address
  , whoisPrice :: Amount
  }

This means that users can buy and sell entries in a shared mapping of type Name -> Whois where:

  1. An unclaimed Name can be bought by a user and set to an arbitrary value.
  2. Existing (Name, Whois) pairs can be updated by their owner or sold to a new owner for the price.
  3. Existing (Name, Whois) pairs can be deleted by their owner and the owner receives a refund for the purchase price.

The application consists of three modules:

  1. Auth - Manages accounts for users, things like nonces and token balances.
  2. Token - Allows users to manage their tokens, things like transfering or burning.
  3. Nameservice - Controls the shared Name -> Value mapping described above.

How to Read this Tutorial

First a warning. The hs-abci-sdk package is a sophisticated framework for building blockchain applications backed by tendermint consensus. As it is a framework, there are certain points when syntax is simplified at the expense of introducing indirection, type synonyms, and a few type families. There was a serious amount of effort to expose as little of this as possible, but alas sometimes things will be confusing and it's best to blindly follow the examples.

This tutorial is largely written as a literate haskell file to simulate developing the Nameservice app from scratch. The file structure is similar to the actual app. We will partially develop a haskell module corresponding to what you find in the app, but possibly not the whole thing. Thus whenever we depend on a haskell module in the tutorial, rather than importing from the tutorial itself we will import from the app.

The benefit of this is that we don't have to develop the entire application in this tutorial. Any breaking changes in the app will (hopefully) break the tutorial and so if you can read this, the tutorial is correct.

Tutorial Goals

The goal of this tutorial is to explain how the Nameservice app is constructed using the hs-abci-sdk package. Nameservice is a relatively simple but still non-trivial application. If you would like to start with something simpler, you can view the tutorial for the simple-storage example application.

This tutorial should teach you:

  1. How to construct application specific modules.
  2. How to enable a module to receive application specific transactions.
  3. How to compose modules and wire up an application.
  4. How to add event logging, console logging, and other effects to module.
  5. How to use the type system to control the capabilities of a module.

The SDK makes heavy use of the effects system brought to haskell by the polysemy library. We're not going to explain how this library works here, there are several existing tutorials that do this already. Suffice it to say that polysemy encourages the application developer to develop modules that have well defined roles and scopes, and to prohibit certain modules from interfering with the roles and scopes of other modules unless explicitly allowed by the type system.

It also allows the application developer to construct modules without much regard for how they will plug into the SDK, leaving that job to the SDK itself.