In this tutorial series, we are going to build a simplistic but functional module using the (Cosmos SDK)[https://github.com/cosmos/cosmos-sdk/] and learn the basics so that you can get started building your own modules and decentralized applications. In this tutorial we will build a "nameservice", a mapping of strings to other strings (similar to (Namecoin)[https://namecoin.org/], (ENS)[https://ens.domains/], or (Handshake)[https://handshake.org/]), in which to buy the name, the buyer has to pay the current owner more than the current owner paid to buy it!
All of the final source code for this tutorial project is in this directory, however, it is highly recommended that you follow along manually and try building the project yourself!
The main core of a Cosmos SDK module is a piece called the Keeper. It is what handles interaction with the store, has references to other keepers, and often contains most of the core functionality of a module. To begin, let's create a file called keeper.go
and place it in a folder called ./x/nameservice
that will hold our module. It is general practice to keep the modules in the ./x/
folder.
In this file, let's start by placing the following code.
package nameservice
import (
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/x/bank"
)
// Keeper - handlers sets/gets of custom variables for your module
type Keeper struct {
coinKeeper bank.Keeper
namesStoreKey sdk.StoreKey // The (unexposed) key used to access the store from the Context.
ownersStoreKey sdk.StoreKey // The (unexposed) key used to access the store from the Context.
priceStoreKey sdk.StoreKey // The (unexposed) key used to access the store from the Context.
cdc *codec.Codec // The wire codec for binary encoding/decoding.
}
Let's break this down. The package name nameservice
is the name of the package that this file is part of. In Go, all code has to be part of a package.
Next we import the main SDK
package and the bank
module from the cosmos-sdk repository.
Next, we create the Keeper struct itself. In this keeper there are a couple of key pieces:
bank.Keeper
- This is a reference to the Keeper from the module. This allows code in this module to be able to call functions from the bank module.sdk.StoreKey
- The SDK uses an object capabilities approach to accessing parts of the sections of the application state. This is to allow developers to employ a least authority approach limiting the capabilities of a faulty or malicious module from affecting parts of state it doesn't need access to. In this module, we will use three stores:namesStoreKey
- This is the main store that stores the value string that the name points to (i.e. The mapping from domain name -> IP Address)ownersStoreKey
- This store contains the current owner of this namepriceStoreKey
- This store contains the price that the current owner paid. And buying of this name must spend more than the current owner.
*codec.Codec
- This is a pointer to the codec that is used by Amino to encode and decode binary structs.
First let's add a function to set the string the a name resolves to.
// SetName - sets the value string that a name resolves to
func (k Keeper) SetName(ctx sdk.Context, name string, value string) {
store := ctx.KVStore(k.namesStoreKey)
store.Set([]byte(name), []byte(value))
}
In this method on the Keeper, we first get the store object for the name resolutions using the the namesStoreKey
from the Keeper.
Next, we insert the <name, value>
pair into the store using its .Set([]byte, []byte)
method. As the store only takes []byte
while we have string
s, we first need to cast the string
s to []byte
and the use them as parameters into the Set
method.
Next, let's add a method to actually resolve the names.
// ResolveName - returns the string that the name resolves to
func (k Keeper) ResolveName(ctx sdk.Context, name string) string {
store := ctx.KVStore(k.namesStoreKey)
bz := store.Get([]byte(name))
return string(bz)
}
Here, like in the SetName method, we first get the store using the StoreKey
. Next, instead of using the Set
method on the store key, we use the .Get([]byte) []byte
method. As the parameter into the function, we pass the key, which is the name
string casted to []byte
, and get back the result in the form of []byte
. We cast this to a string
and return the result.
We now add similar functions for Getting and Setting Owners.
// GetOwner - get the current owner of a name
func (k Keeper) GetOwner(ctx sdk.Context, name string) sdk.AccAddress {
store := ctx.KVStore(k.ownersStoreKey)
bz := store.Get([]byte(name))
return bz
}
// SetOwner - sets the current owner of a name
func (k Keeper) SetOwner(ctx sdk.Context, name string, owner sdk.AccAddress) {
store := ctx.KVStore(k.ownersStoreKey)
store.Set([]byte(name), owner)
}
// HasOwner - returns whether or not the name already has an owner
func (k Keeper) HasOwner(ctx sdk.Context, name string) bool {
store := ctx.KVStore(k.ownersStoreKey)
bz := store.Get([]byte(name))
return bz != nil
}
Note that now, instead of accessing the the data from the namesStoreKey
store, we now get it from the ownersStoreKey
store. Because sdk.AccAddress is a type alias for []byte
, we can natively cast to it. We also added an extra function HasOwner
that tells us whether a name already has an owner or not.
Finally, we will add a getter and setter for the Price of a name.
// GetPrice - gets the current price of a name. If price doesn't exist yet, set to 1steak.
func (k Keeper) GetPrice(ctx sdk.Context, name string) sdk.Coins {
if !k.HasOwner(ctx, name) {
return sdk.Coins{sdk.NewInt64Coin("mycoin", 1)}
}
store := ctx.KVStore(k.priceStoreKey)
bz := store.Get([]byte(name))
var price sdk.Coins
k.cdc.MustUnmarshalBinary(bz, &price)
return price
}
// SetPrice - sets the current price of a name
func (k Keeper) SetPrice(ctx sdk.Context, name string, price sdk.Coins) {
store := ctx.KVStore(k.priceStoreKey)
store.Set([]byte(name), k.cdc.MustMarshalBinary(price))
}
We put this data in the priceStoreKey
store. Note that sdk.Coins
does not have it's own Bytes encoding, and so, to marshal and unmarshal the price for inserting and removing from store, we use Amino (read more about Amino here: https://github.com/tendermint/go-amino/)
When getting the price for a name that has no owner (and thus no price), we will return 1steak as the price.
Now that we have the keeper setup, it is time to built the msgs and handlers that actually allow users to buy and set names.
Let's first setup the different messages that a user can use to interact with this module. The Cosmos SDK define a standard interface that all Msgs must satisfy:
// Transactions messages must fulfill the Msg
type Msg interface {
// Return the message type.
// Must be alphanumeric or empty.
Type() string
// Returns a human-readable string for the message, intended for utilization
// within tags
Name() string
// ValidateBasic does a simple validation check that
// doesn't require access to any other information.
ValidateBasic() Error
// Get the canonical byte representation of the Msg.
GetSignBytes() []byte
// Signers returns the addrs of signers that must sign.
// CONTRACT: All signatures must be present to be valid.
// CONTRACT: Returns addrs in some deterministic order.
GetSigners() []AccAddress
}
We'll start by defining MsgSetName
in a new file called msgs.go
in the nameservice
package, a Msg that allows owner of an address to set the result of resolving a name.
type MsgSetName struct {
NameID string
Value string
Owner sdk.AccAddress
}
func NewMsgSetName(name string, value string, owner sdk.AccAddress) MsgSetName {
return MsgBuyName{
NameID: name,
Value: value,
Owner: owner,
}
}
The MsgSetName
has three attributes:
name
- The name trying to be setvalue
- What the name resolves toowner
- The owner of that name
Note that we use the field name NameID
rather than Name
as .Name()
is the name of a method on the Msg
interface. This will be resolved in a future update of the SDK. cosmos/cosmos-sdk#2456
// Implements Msg.
func (msg MsgSetName) Type() string { return "nameservice" }
func (msg MsgSetName) Name() string { return "set_name"}
These is used by the SDK to route msgs to the proper module for handling and for adding human readable names to tags.
// Implements Msg.
func (msg MsgSetName) ValidateBasic() sdk.Error {
if msg.Owner.Empty() {
return sdk.ErrInvalidAddress(msg.Owner.String())
}
if len(msg.NameID) == 0 || len(msg.Value) == 0 {
return sdk.ErrUnknownRequest("Name and Value cannot be empty")
}
return nil
}
This is used to provide some basic stateless checks on the validity of the msg. In this case, we check that none of the attributes are empty.
// Implements Msg.
func (msg MsgSetName) GetSignBytes() []byte {
b, err := json.Marshal(msg)
if err != nil {
panic(err)
}
return sdk.MustSortJSON(b)
}
This defines how the Msg gets encoded for signing. This should usually be in JSON and should not be modified in most cases.
// Implements Msg.
func (msg MsgSetName) GetSigners() []sdk.AccAddress {
return []sdk.AccAddress{msg.Owner}
}
This allows the Msg to define who's signature is required on a Tx in order for it to be valid. In this case, for example, the MsgSetName
requires that the Owner
sign the transaction trying to reset what the name points to.
Now that we have the MsgSetName
defined, we now have to define the handler that actually executes the Msg.
In a new file called handler.go
in the nameservice
package, we start off with:
package nameservice
import (
"fmt"
"reflect"
sdk "github.com/cosmos/cosmos-sdk/types"
)
// NewHandler returns a handler for "nameservice" type messages.
func NewHandler(keeper Keeper) sdk.Handler {
return func(ctx sdk.Context, msg sdk.Msg) sdk.Result {
switch msg := msg.(type) {
case MsgSetName:
return handleMsgSetName(ctx, keeper, msg)
default:
errMsg := fmt.Sprintf("Unrecognized nameservice Msg type: %v", reflect.TypeOf(msg).Name())
return sdk.ErrUnknownRequest(errMsg).Result()
}
}
}
This is essentially a subrouter that directs messages coming into this module to the proper handler for the message. At the moment, we only have one Msg/Handler.
In the same file, we define the function handleMsgSetName
.
// Handle MsgSetName
func handleMsgSetName(ctx sdk.Context, keeper Keeper, msg MsgSetName) sdk.Result {
if !msg.Owner.Equals(keeper.GetOwner(ctx, msg.NameID)) { // Checks if the the msg sender is the same as the current owner
return sdk.ErrUnauthorized("Incorrect Owner").Result() // If not, throw an error
}
keeper.SetName(ctx, msg.NameID, msg.Value) // If so, set the name to the value specified in the msg.
return sdk.Result{} // return
}
In this function we check to see if the Msg sender is actually the owner of the name (which we get using keeper.GetOwner
). If so, we let them set the name by calling the function on the keeper. If not, we throw an error.
Great, now owners can set names! But what if a name doesn't have an owner yet? We need a way for people to buy names!
We define the Msg for buying names and add it to the msgs.go
file:
type MsgBuyName struct {
NameID string
Bid sdk.Coins
Buyer sdk.AccAddress
}
func NewMsgBuyName(name string, bid sdk.Coins, buyer sdk.AccAddress) MsgBuyName {
return MsgBuyName{
NameID: name,
Bid: bid,
Buyer: buyer,
}
}
// Implements Msg.
func (msg MsgBuyName) Type() string { return "nameservice" }
func (msg MsgSetName) Name() string { return "buy_name"}
// Implements Msg.
func (msg MsgBuyName) ValidateBasic() sdk.Error {
if msg.Buyer.Empty() {
return sdk.ErrInvalidAddress(msg.Buyer.String())
}
if len(msg.NameID) == 0 {
return sdk.ErrUnknownRequest("Name and Value cannot be empty")
}
if !msg.Bid.IsPositive() {
return sdk.ErrInsufficientCoins("Bids must be positive")
}
return nil
}
// Implements Msg.
func (msg MsgBuyName) GetSignBytes() []byte {
b, err := json.Marshal(msg)
if err != nil {
panic(err)
}
return sdk.MustSortJSON(b)
}
// Implements Msg.
func (msg MsgBuyName) GetSigners() []sdk.AccAddress {
return []sdk.AccAddress{msg.Buyer}
}
In the handler.go
file, we add the MsgBuyName
handler to the module router, so it now looks like this:
// NewHandler returns a handler for "nameservice" type messages.
func NewHandler(keeper Keeper) sdk.Handler {
return func(ctx sdk.Context, msg sdk.Msg) sdk.Result {
switch msg := msg.(type) {
case MsgSetName:
return handleMsgSetName(ctx, keeper, msg)
case MsgBuyName:
return handleMsgBuyName(ctx, keeper, msg)
default:
errMsg := fmt.Sprintf("Unrecognized nameservice Msg type: %v", reflect.TypeOf(msg).Name())
return sdk.ErrUnknownRequest(errMsg).Result()
}
}
}
And we add the actual handle function to the handler.go
file:
// Handle MsgBuyName
func handleMsgBuyName(ctx sdk.Context, keeper Keeper, msg MsgBuyName) sdk.Result {
if keeper.GetPrice(ctx, msg.NameID).IsGTE(msg.Bid) { // Checks if the the bid price is greater than the price paid by the current owner
return sdk.ErrInsufficientCoins("Bid not high enough").Result() // If not, throw an error
}
if keeper.HasOwner(ctx, msg.NameID) {
_, err := keeper.coinKeeper.SendCoins(ctx, msg.Buyer, keeper.GetOwner(ctx, msg.NameID), msg.Bid)
if err != nil {
return sdk.ErrInsufficientCoins("Buyer does not have enough coins").Result()
}
} else {
_, _, err := keeper.coinKeeper.SubtractCoins(ctx, msg.Buyer, msg.Bid) // If so, deduct the Bid amount from the sender
if err != nil {
return sdk.ErrInsufficientCoins("Buyer does not have enough coins").Result()
}
}
keeper.SetOwner(ctx, msg.NameID, msg.Buyer)
keeper.SetPrice(ctx, msg.NameID, msg.Bid)
return sdk.Result{}
}
In this function, we check to make sure the bid is higher than the current price. If it is, we check to see whether the name already has an owner. If it does, they get transferred the money from the Buyer. If it doesn't, the money just gets burned from the buyer. If either SubtractCoins
or SendCoins
returns a non-nil error, the handler throws an error, reverting the transaction. Otherwise, we set the buyer to the new owner, set the new price to be the current bid, and return.
Now that the core logic of our nameservice is finished, let's actually build an app that uses the module. The main focus of this tutorial was the building of the core module, and the rest of the tutorial is just to get the app up and running, so the explanations will be less exhaustive from here on out. In most cases, you'll be using similar boilercode as well.
In your module's folder create a querier.go
file. This file allows for advanced queries into the application state.
package nameservice
import (
"github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types"
abci "github.com/tendermint/tendermint/abci/types"
)
// query endpoints supported by the governance Querier
const (
QueryResolve = "resolve"
QueryWhois = "whois"
)
func NewQuerier(keeper Keeper) sdk.Querier {
return func(ctx sdk.Context, path []string, req abci.RequestQuery) (res []byte, err sdk.Error) {
switch path[0] {
case QueryResolve:
return queryResolve(ctx, path[1:], req, keeper)
case QueryWhois:
return queryWhois(ctx, path[1:], req, keeper)
default:
return nil, sdk.ErrUnknownRequest("unknown nameservice query endpoint")
}
}
}
// nolint: unparam
func queryResolve(ctx sdk.Context, path []string, req abci.RequestQuery, keeper Keeper) (res []byte, err sdk.Error) {
name := path[0]
value := keeper.ResolveName(ctx, name)
if value == "" {
return []byte{}, sdk.ErrUnknownRequest("could not resolve name")
}
return []byte(value), nil
}
type Whois struct {
Value string `json:"value"`
Owner sdk.AccAddress `json:"owner"`
Price sdk.Coins `json:"price"`
}
// nolint: unparam
func queryWhois(ctx sdk.Context, path []string, req abci.RequestQuery, keeper Keeper) (res []byte, err sdk.Error) {
name := path[0]
whois := Whois{}
whois.Value = keeper.ResolveName(ctx, name)
whois.Owner = keeper.GetOwner(ctx, name)
whois.Price = keeper.GetPrice(ctx, name)
bz, err2 := codec.MarshalJSONIndent(keeper.cdc, whois)
if err2 != nil {
panic("could not marshal result to JSON")
}
return bz, nil
}
In your module's folder create a codec.go
file. This allows Amino to register the MsgSetName
and MsgBuyName
.
package nameservice
import (
"github.com/cosmos/cosmos-sdk/codec"
)
// Register concrete types on wire codec
func RegisterCodec(cdc *codec.Codec) {
cdc.RegisterConcrete(MsgSetName{}, "nameservice/SetName", nil)
cdc.RegisterConcrete(MsgBuyName{}, "nameservice/BuyName", nil)
}
Next, in the module's folder, create two files:
./client/cli/query.go
./client/cli/tx.go
These will enable our cli to understand our module.
query.go
package cli
import (
"fmt"
"github.com/spf13/cobra"
"github.com/cosmos/cosmos-sdk/client/context"
"github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types"
)
type QueryResult struct {
Value string `json:"value"`
Owner sdk.AccAddress `json:"owner"`
Price sdk.Coins `json:"price"`
}
// GetCmdResolveName queries information about a name
func GetCmdResolveName(queryRoute string, cdc *codec.Codec) *cobra.Command {
cmd := &cobra.Command{
Use: "resolve [name]",
Short: "resolve name",
Args: cobra.ExactArgs(1),
RunE: func(cmd *cobra.Command, args []string) error {
name := args[0]
cliCtx := context.NewCLIContext().WithCodec(cdc)
res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/%s/proposals/%s", queryRoute, name), nil)
if err != nil {
fmt.Printf("could not resolve name - %s \n", string(name))
return nil
}
fmt.Println(string(res))
return nil
},
}
return cmd
}
// GetCmdWhois queries information about a domain
func GetCmdWhois(queryRoute string, cdc *codec.Codec) *cobra.Command {
cmd := &cobra.Command{
Use: "whois [name]",
Short: "Query whois info of name",
Args: cobra.ExactArgs(1),
RunE: func(cmd *cobra.Command, args []string) error {
name := args[0]
cliCtx := context.NewCLIContext().WithCodec(cdc)
res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/%s/whois/%s", queryRoute, name), nil)
if err != nil {
fmt.Printf("could not resolve whois - %s \n", string(name))
return nil
}
fmt.Println(string(res))
return nil
},
}
return cmd
}
tx.go
package cli
import (
"github.com/spf13/cobra"
"github.com/cosmos/cosmos-sdk/client/context"
"github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/x/auth"
authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli"
"github.com/sunnya97/sdk-nameservice-example/x/nameservice"
)
const (
flagName = "name"
flagValue = "value"
flagAmount = "amount"
)
func GetCmdBuyName(cdc *codec.Codec) *cobra.Command {
cmd := &cobra.Command{
Use: "buy-name [name] [amount]",
Short: "bid for existing name or claim new name",
Args: cobra.ExactArgs(2),
RunE: func(cmd *cobra.Command, args []string) error {
cliCtx := context.NewCLIContext().
WithCodec(cdc).
WithAccountDecoder(authcmd.GetAccountDecoder(cdc))
if err := cliCtx.EnsureAccountExists(); err != nil {
return err
}
name := args[0]
amount := args[1]
coins, err := sdk.ParseCoins(amount)
if err != nil {
return err
}
account, err := cliCtx.GetFromAddress()
if err != nil {
return err
}
msg := nameservice.MsgBuyName{
NameID: name,
Bid: coins,
Buyer: account,
}
tx := auth.StdTx{
Msgs: []sdk.Msg{msg},
}
bz := cdc.MustMarshalBinary(tx)
_, err = cliCtx.BroadcastTx(bz)
return err
},
}
return cmd
}
func GetCmdSetName(cdc *codec.Codec) *cobra.Command {
cmd := &cobra.Command{
Use: "set-name [name] [value]",
Short: "set the value associated with a name that you own",
Args: cobra.ExactArgs(2),
RunE: func(cmd *cobra.Command, args []string) error {
cliCtx := context.NewCLIContext().
WithCodec(cdc).
WithAccountDecoder(authcmd.GetAccountDecoder(cdc))
if err := cliCtx.EnsureAccountExists(); err != nil {
return err
}
name := args[0]
value := args[1]
account, err := cliCtx.GetFromAddress()
if err != nil {
return err
}
msg := nameservice.MsgSetName{
NameID: name,
Value: value,
Owner: account,
}
tx := auth.StdTx{
Msgs: []sdk.Msg{msg},
}
bz := cdc.MustMarshalBinary(tx)
_, err = cliCtx.BroadcastTx(bz)
return err
},
}
return cmd
}
Next, in the root of our project directory, let's create a new file called app.go
. At the top of the file, let's declare the package and import our dependencies.
package app
import (
"os"
abci "github.com/tendermint/tendermint/abci/types"
cmn "github.com/tendermint/tendermint/libs/common"
dbm "github.com/tendermint/tendermint/libs/db"
"github.com/tendermint/tendermint/libs/log"
bam "github.com/cosmos/cosmos-sdk/baseapp"
"github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/x/auth"
"github.com/cosmos/cosmos-sdk/x/bank"
faucet "github.com/sunnya97/sdk-faucet-module"
"github.com/sunnya97/sdk-nameservice-example/x/nameservice"
)
Here we imported some dependencies from Tendermint, from the Cosmos SDK, and then the four modules we will use in our app: auth
, bank
, faucet
and nameservice
.
Next we'll declare the name and struct for our app. In this example, we'll call it Nameshake, a portmanteau of Handshake and Namecoin.
const (
appName = "Nameshake"
)
type NameshakeApp struct {
*bam.BaseApp
cdc *codec.Codec
keyMain *sdk.KVStoreKey
keyAccount *sdk.KVStoreKey
keyNSnames *sdk.KVStoreKey
keyNSowners *sdk.KVStoreKey
keyNSprices *sdk.KVStoreKey
accountMapper auth.AccountMapper
bankKeeper bank.Keeper
nsKeeper nameservice.Keeper
}
Now we will create a constructor for a new HandshakeApp. In this, we will generate all storeKeys and Keepers. We will register the routes, mount the stores, and set the initChainer (explained next).
func NewNameshakeApp(logger log.Logger, db dbm.DB) *NameshakeApp {
cdc := MakeCodec()
bApp := bam.NewBaseApp(appName, logger, db, auth.DefaultTxDecoder(cdc))
var app = &NameshakeApp{
BaseApp: bApp,
cdc: cdc,
keyMain: sdk.NewKVStoreKey("main"),
keyAccount: sdk.NewKVStoreKey("acc"),
keyNSnames: sdk.NewKVStoreKey("ns_names"),
keyNSowners: sdk.NewKVStoreKey("ns_owners"),
keyNSprices: sdk.NewKVStoreKey("ns_prices"),
}
app.accountMapper = auth.NewAccountMapper(
app.cdc,
app.keyAccount,
auth.ProtoBaseAccount,
)
app.bankKeeper = bank.NewBaseKeeper(app.accountMapper)
app.nsKeeper = nameservice.NewKeeper(
app.bankKeeper,
app.keyNSnames,
app.keyNSowners,
app.keyNSprices,
app.cdc,
)
app.Router().
AddRoute("nameservice", nameservice.NewHandler(app.nsKeeper)).
AddRoute("faucet", faucet.NewHandler(app.bankKeeper))
app.QueryRouter().
AddRoute("nameservice", nameservice.NewQuerier(app.nsKeeper))
app.SetInitChainer(app.initChainer)
app.MountStoresIAVL(
app.keyMain,
app.keyAccount,
app.keyNSnames,
app.keyNSowners,
app.keyNSprices,
)
err := app.LoadLatestVersion(app.keyMain)
if err != nil {
cmn.Exit(err.Error())
}
return app
}
Next, we'll add an initChainer function so we can generate accounts with initial balance from the genesis.json
.
type GenesisState struct {
Accounts []auth.BaseAccount `json:"accounts"`
}
func (app *NameshakeApp) initChainer(ctx sdk.Context, req abci.RequestInitChain) abci.ResponseInitChain {
stateJSON := req.AppStateBytes
genesisState := new(GenesisState)
err := app.cdc.UnmarshalJSON(stateJSON, genesisState)
if err != nil {
panic(err)
}
for _, acc := range genesisState.Accounts {
acc.AccountNumber = app.accountMapper.GetNextAccountNumber(ctx)
app.accountMapper.SetAccount(ctx, &acc)
}
return abci.ResponseInitChain{}
}
And finally, a helper function to generate an amino codec.
func MakeCodec() *codec.Codec {
var cdc = codec.New()
auth.RegisterCodec(cdc)
bank.RegisterCodec(cdc)
nameservice.RegisterCodec(cdc)
faucet.RegisterCodec(cdc)
sdk.RegisterCodec(cdc)
codec.RegisterCrypto(cdc)
return cdc
}
Next, we'll create two files in the root of the project directory that will instantiate the two main pieces of software, the blockchain node and the CLI for interacting with the chain.
./cmd/nameshaked/main.go
./cmd/nameshakecli/main.go
nameshaked/main.go
package main
import (
"encoding/json"
"io"
"os"
"github.com/spf13/cobra"
abci "github.com/tendermint/tendermint/abci/types"
"github.com/tendermint/tendermint/libs/cli"
dbm "github.com/tendermint/tendermint/libs/db"
"github.com/tendermint/tendermint/libs/log"
tmtypes "github.com/tendermint/tendermint/types"
"github.com/cosmos/cosmos-sdk/server"
app "github.com/sunnya97/sdk-nameservice-example"
)
var DefaultNodeHome = os.ExpandEnv("$HOME/.nameshaked")
var appInit = server.AppInit{
AppGenState: server.SimpleAppGenState,
AppGenTx: server.SimpleAppGenTx,
}
func main() {
cdc := app.MakeCodec()
ctx := server.NewDefaultContext()
cobra.EnableCommandSorting = false
rootCmd := &cobra.Command{
Use: "nameshaked",
Short: "Nameshake App Daemon (server)",
PersistentPreRunE: server.PersistentPreRunEFn(ctx),
}
server.AddCommands(ctx, cdc, rootCmd, appInit,
server.ConstructAppCreator(newApp, "nameshake"),
server.ConstructAppExporter(exportAppStateAndTMValidators, "nameshake"))
// prepare and add flags
executor := cli.PrepareBaseCmd(rootCmd, "NS", DefaultNodeHome)
err := executor.Execute()
if err != nil {
// handle with #870
panic(err)
}
}
func newApp(logger log.Logger, db dbm.DB, traceStore io.Writer) abci.Application {
return app.NewNameshakeApp(logger, db)
}
func exportAppStateAndTMValidators(
logger log.Logger, db dbm.DB, traceStore io.Writer,
) (json.RawMessage, []tmtypes.GenesisValidator, error) {
return nil, nil, nil
}
nameshakecli/main.go
package cli
import (
"fmt"
"github.com/spf13/cobra"
"github.com/cosmos/cosmos-sdk/client/context"
"github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types"
)
type QueryResult struct {
Value string `json:"value"`
Owner sdk.AccAddress `json:"owner"`
Price sdk.Coins `json:"price"`
}
// GetCmdResolveName queries information about a name
func GetCmdResolveName(queryRoute string, cdc *codec.Codec) *cobra.Command {
cmd := &cobra.Command{
Use: "resolve [name]",
Short: "resolve name",
Args: cobra.ExactArgs(1),
RunE: func(cmd *cobra.Command, args []string) error {
name := args[0]
cliCtx := context.NewCLIContext().WithCodec(cdc)
res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/%s/proposals/%s", queryRoute, name), nil)
if err != nil {
fmt.Printf("could not resolve name - %s \n", string(name))
return nil
}
fmt.Println(string(res))
return nil
},
}
return cmd
}
// GetCmdWhois queries information about a domain
func GetCmdWhois(queryRoute string, cdc *codec.Codec) *cobra.Command {
cmd := &cobra.Command{
Use: "whois [name]",
Short: "Query whois info of name",
Args: cobra.ExactArgs(1),
RunE: func(cmd *cobra.Command, args []string) error {
name := args[0]
cliCtx := context.NewCLIContext().WithCodec(cdc)
res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/%s/whois/%s", queryRoute, name), nil)
if err != nil {
fmt.Printf("could not resolve whois - %s \n", string(name))
return nil
}
fmt.Println(string(res))
return nil
},
}
return cmd
}
Finally add the following files to the root directory.
Gopkg.toml
# Gopkg.toml example
#
# Refer to https://github.com/golang/dep/blob/master/docs/Gopkg.toml.md
# for detailed Gopkg.toml documentation.
#
# required = ["github.com/user/thing/cmd/thing"]
# ignored = ["github.com/user/project/pkgX", "bitbucket.org/user/project/pkgA/pkgY"]
#
# [[constraint]]
# name = "github.com/user/project"
# version = "1.0.0"
#
# [[override]]
# name = "github.com/x/y"
# version = "2.4.0"
#
# [prune]
# non-go = false
# go-tests = true
# unused-packages = true
[[constraint]]
name = "github.com/cosmos/cosmos-sdk"
branch = "develop"
[[override]]
name = "github.com/golang/protobuf"
version = "=1.1.0"
[[constraint]]
name = "github.com/spf13/cobra"
version = "~0.0.1"
[[constraint]]
name = "github.com/spf13/viper"
version = "~1.0.0"
[[override]]
name = "github.com/tendermint/go-amino"
version = "=v0.12.0"
[[override]]
name = "github.com/tendermint/iavl"
version = "=v0.11.0"
[[override]]
name = "github.com/tendermint/tendermint"
version = "=0.25.0"
[prune]
go-tests = true
unused-packages = true
Makefile
# Gopkg.toml example
#
# Refer to https://github.com/golang/dep/blob/master/docs/Gopkg.toml.md
# for detailed Gopkg.toml documentation.
#
# required = ["github.com/user/thing/cmd/thing"]
# ignored = ["github.com/user/project/pkgX", "bitbucket.org/user/project/pkgA/pkgY"]
#
# [[constraint]]
# name = "github.com/user/project"
# version = "1.0.0"
#
# [[override]]
# name = "github.com/x/y"
# version = "2.4.0"
#
# [prune]
# non-go = false
# go-tests = true
# unused-packages = true
[[constraint]]
name = "github.com/cosmos/cosmos-sdk"
branch = "develop"
[[override]]
name = "github.com/golang/protobuf"
version = "=1.1.0"
[[constraint]]
name = "github.com/spf13/cobra"
version = "~0.0.1"
[[constraint]]
name = "github.com/spf13/viper"
version = "~1.0.0"
[[override]]
name = "github.com/tendermint/go-amino"
version = "=v0.12.0"
[[override]]
name = "github.com/tendermint/iavl"
version = "=v0.11.0"
[[override]]
name = "github.com/tendermint/tendermint"
version = "=0.25.0"
[prune]
go-tests = true
unused-packages = true
Start by installing Dep.
go get -v github.com/golang/dep/cmd/dep
Next, run
dep ensure
Finally run
make install